@windwalker-io/unicorn-next 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (175) hide show
  1. package/.editorconfig +18 -0
  2. package/.gulp.json +7 -0
  3. package/bin/release.mjs +47 -0
  4. package/dist/chunks/_arrayPush.js +168 -0
  5. package/dist/chunks/_arrayPush.js.map +1 -0
  6. package/dist/chunks/_baseRest.js +73 -0
  7. package/dist/chunks/_baseRest.js.map +1 -0
  8. package/dist/chunks/_commonjsHelpers.js +7 -0
  9. package/dist/chunks/_commonjsHelpers.js.map +1 -0
  10. package/dist/chunks/_getPrototype.js +130 -0
  11. package/dist/chunks/_getPrototype.js.map +1 -0
  12. package/dist/chunks/button-radio.js +147 -0
  13. package/dist/chunks/button-radio.js.map +1 -0
  14. package/dist/chunks/checkboxes-multi-select.js +44 -0
  15. package/dist/chunks/checkboxes-multi-select.js.map +1 -0
  16. package/dist/chunks/cloneDeep.js +287 -0
  17. package/dist/chunks/cloneDeep.js.map +1 -0
  18. package/dist/chunks/cropper.min.js +5 -0
  19. package/dist/chunks/cropper.min.js.map +1 -0
  20. package/dist/chunks/field-cascade-select.js +256 -0
  21. package/dist/chunks/field-cascade-select.js.map +1 -0
  22. package/dist/chunks/field-file-drag.js +218 -0
  23. package/dist/chunks/field-file-drag.js.map +1 -0
  24. package/dist/chunks/field-flatpickr.js +893 -0
  25. package/dist/chunks/field-flatpickr.js.map +1 -0
  26. package/dist/chunks/field-modal-select.js +403 -0
  27. package/dist/chunks/field-modal-select.js.map +1 -0
  28. package/dist/chunks/field-modal-tree.js +790 -0
  29. package/dist/chunks/field-modal-tree.js.map +1 -0
  30. package/dist/chunks/field-multi-uploader.js +256 -0
  31. package/dist/chunks/field-multi-uploader.js.map +1 -0
  32. package/dist/chunks/field-repeatable.js +132 -0
  33. package/dist/chunks/field-repeatable.js.map +1 -0
  34. package/dist/chunks/field-single-image-drag.js +338 -0
  35. package/dist/chunks/field-single-image-drag.js.map +1 -0
  36. package/dist/chunks/form.js +154 -0
  37. package/dist/chunks/form.js.map +1 -0
  38. package/dist/chunks/grid.js +345 -0
  39. package/dist/chunks/grid.js.map +1 -0
  40. package/dist/chunks/http-client.js +229 -0
  41. package/dist/chunks/http-client.js.map +1 -0
  42. package/dist/chunks/iframe-modal.js +124 -0
  43. package/dist/chunks/iframe-modal.js.map +1 -0
  44. package/dist/chunks/index.js +309 -0
  45. package/dist/chunks/index.js.map +1 -0
  46. package/dist/chunks/isArguments.js +146 -0
  47. package/dist/chunks/isArguments.js.map +1 -0
  48. package/dist/chunks/keep-tab.js +101 -0
  49. package/dist/chunks/keep-tab.js.map +1 -0
  50. package/dist/chunks/legacy.js +210 -0
  51. package/dist/chunks/legacy.js.map +1 -0
  52. package/dist/chunks/list-dependent.js +231 -0
  53. package/dist/chunks/list-dependent.js.map +1 -0
  54. package/dist/chunks/s3-multipart-uploader.js +172 -0
  55. package/dist/chunks/s3-multipart-uploader.js.map +1 -0
  56. package/dist/chunks/s3-uploader.js +136 -0
  57. package/dist/chunks/s3-uploader.js.map +1 -0
  58. package/dist/chunks/show-on.js +237 -0
  59. package/dist/chunks/show-on.js.map +1 -0
  60. package/dist/chunks/tinymce.js +196 -0
  61. package/dist/chunks/tinymce.js.map +1 -0
  62. package/dist/chunks/ui-bootstrap5.js +71 -0
  63. package/dist/chunks/ui-bootstrap5.js.map +1 -0
  64. package/dist/chunks/unicorn.js +2202 -0
  65. package/dist/chunks/unicorn.js.map +1 -0
  66. package/dist/chunks/validation.js +854 -0
  67. package/dist/chunks/validation.js.map +1 -0
  68. package/dist/editor.css +1 -0
  69. package/dist/index.d.ts +1427 -0
  70. package/dist/multi-level-menu.css +1 -0
  71. package/dist/switcher.css +1 -0
  72. package/dist/unicorn-next.css +12 -0
  73. package/dist/unicorn.js +125 -0
  74. package/dist/unicorn.js.map +1 -0
  75. package/fusionfile.mjs +155 -0
  76. package/images/ajax-loader.gif +0 -0
  77. package/images/placeholder/avatar.png +0 -0
  78. package/images/placeholder/image-16x10.png +0 -0
  79. package/images/placeholder/image-16x9.png +0 -0
  80. package/images/placeholder/image-1x1.png +0 -0
  81. package/images/placeholder/image-4x3.png +0 -0
  82. package/package.json +102 -0
  83. package/scss/bootstrap/multi-level-menu.scss +121 -0
  84. package/scss/editor.scss +116 -0
  85. package/scss/field/file-drag.scss +102 -0
  86. package/scss/field/single-image-drag.scss +88 -0
  87. package/scss/field/vue-drag-uploader.scss +160 -0
  88. package/scss/switcher.scss +156 -0
  89. package/src/app.ts +128 -0
  90. package/src/bootstrap/button-radio.ts +208 -0
  91. package/src/bootstrap/keep-tab.ts +155 -0
  92. package/src/composable/index.ts +21 -0
  93. package/src/composable/useCheckboxesMultiSelect.ts +22 -0
  94. package/src/composable/useFieldCascadeSelect.ts +9 -0
  95. package/src/composable/useFieldFileDrag.ts +9 -0
  96. package/src/composable/useFieldFlatpickr.ts +3 -0
  97. package/src/composable/useFieldModalSelect.ts +6 -0
  98. package/src/composable/useFieldModalTree.ts +3 -0
  99. package/src/composable/useFieldMultiUploader.ts +3 -0
  100. package/src/composable/useFieldRepeatable.ts +9 -0
  101. package/src/composable/useFieldSingleImageDrag.ts +5 -0
  102. package/src/composable/useForm.ts +43 -0
  103. package/src/composable/useGrid.ts +57 -0
  104. package/src/composable/useHttp.ts +8 -0
  105. package/src/composable/useIframeModal.ts +9 -0
  106. package/src/composable/useListDependent.ts +26 -0
  107. package/src/composable/useQueue.ts +13 -0
  108. package/src/composable/useS3Uploader.ts +32 -0
  109. package/src/composable/useShowOn.ts +9 -0
  110. package/src/composable/useStack.ts +13 -0
  111. package/src/composable/useTinymce.ts +29 -0
  112. package/src/composable/useTomSelect.ts +72 -0
  113. package/src/composable/useUIBootstrap5.ts +48 -0
  114. package/src/composable/useUniDirective.ts +32 -0
  115. package/src/composable/useValidation.ts +39 -0
  116. package/src/data.ts +36 -0
  117. package/src/events.ts +73 -0
  118. package/src/legacy/legacy.ts +186 -0
  119. package/src/legacy/loader.ts +125 -0
  120. package/src/module/checkboxes-multi-select.ts +54 -0
  121. package/src/module/field-cascade-select.ts +292 -0
  122. package/src/module/field-file-drag.ts +292 -0
  123. package/src/module/field-flatpickr.ts +127 -0
  124. package/src/module/field-modal-select.ts +174 -0
  125. package/src/module/field-modal-tree.ts +27 -0
  126. package/src/module/field-multi-uploader.ts +361 -0
  127. package/src/module/field-repeatable.ts +202 -0
  128. package/src/module/field-single-image-drag.ts +468 -0
  129. package/src/module/form.ts +223 -0
  130. package/src/module/grid.ts +465 -0
  131. package/src/module/http-client.ts +243 -0
  132. package/src/module/iframe-modal.ts +167 -0
  133. package/src/module/list-dependent.ts +321 -0
  134. package/src/module/s3-multipart-uploader.ts +300 -0
  135. package/src/module/s3-uploader.ts +234 -0
  136. package/src/module/show-on.ts +173 -0
  137. package/src/module/tinymce.ts +263 -0
  138. package/src/module/ui-bootstrap5.ts +107 -0
  139. package/src/module/validation.ts +1019 -0
  140. package/src/plugin/index.ts +1 -0
  141. package/src/plugin/php-adapter.ts +65 -0
  142. package/src/polyfill/form-request-submit.ts +31 -0
  143. package/src/polyfill/index.ts +9 -0
  144. package/src/service/animate.ts +58 -0
  145. package/src/service/crypto.ts +27 -0
  146. package/src/service/dom-watcher.ts +62 -0
  147. package/src/service/dom.ts +265 -0
  148. package/src/service/helper.ts +48 -0
  149. package/src/service/index.ts +10 -0
  150. package/src/service/lang.ts +122 -0
  151. package/src/service/loader.ts +152 -0
  152. package/src/service/router.ts +118 -0
  153. package/src/service/ui.ts +497 -0
  154. package/src/service/uri.ts +106 -0
  155. package/src/types/base.ts +9 -0
  156. package/src/types/index.ts +4 -0
  157. package/src/types/modal-tree.ts +12 -0
  158. package/src/types/plugin.ts +6 -0
  159. package/src/types/shims.d.ts +18 -0
  160. package/src/types/ui.ts +6 -0
  161. package/src/unicorn.ts +63 -0
  162. package/src/utilities/arr.ts +25 -0
  163. package/src/utilities/base.ts +9 -0
  164. package/src/utilities/data.ts +48 -0
  165. package/src/utilities/index.ts +5 -0
  166. package/src/utilities/tree.ts +20 -0
  167. package/src/vue/components/ModalTree/ModalTreeApp.vue +175 -0
  168. package/src/vue/components/ModalTree/TreeItem.vue +262 -0
  169. package/src/vue/components/ModalTree/TreeModal.vue +225 -0
  170. package/tests/test.js +4 -0
  171. package/tsconfig.js.json +25 -0
  172. package/tsconfig.json +17 -0
  173. package/vite.assets.config.ts +61 -0
  174. package/vite.config.test.ts +36 -0
  175. package/vite.config.ts +112 -0
@@ -0,0 +1,256 @@
1
+ import { g as getDefaultExportFromCjs } from "./_commonjsHelpers.js";
2
+ import { a4 as prepareAlpineDefer, f as initAlpineComponent, a5 as mergeDeep, K as uid, z as module, u as useHttpClient } from "./unicorn.js";
3
+ var alpineComponent = {};
4
+ var hasRequiredAlpineComponent;
5
+ function requireAlpineComponent() {
6
+ if (hasRequiredAlpineComponent) return alpineComponent;
7
+ hasRequiredAlpineComponent = 1;
8
+ Object.defineProperty(alpineComponent, "__esModule", { value: true });
9
+ class AlpineComponent2 {
10
+ }
11
+ alpineComponent.default = AlpineComponent2;
12
+ return alpineComponent;
13
+ }
14
+ var alpineComponentExports = /* @__PURE__ */ requireAlpineComponent();
15
+ const AlpineComponent = /* @__PURE__ */ getDefaultExportFromCjs(alpineComponentExports);
16
+ var component = {};
17
+ var hasRequiredComponent;
18
+ function requireComponent() {
19
+ if (hasRequiredComponent) return component;
20
+ hasRequiredComponent = 1;
21
+ Object.defineProperty(component, "__esModule", { value: true });
22
+ function Component2(component2) {
23
+ return function(...args) {
24
+ let instance = new component2(...args);
25
+ const proto = component2.prototype;
26
+ let data = {};
27
+ Object.getOwnPropertyNames(proto).forEach(function(key) {
28
+ if (key === "constructor") {
29
+ return;
30
+ }
31
+ const descriptor = Object.getOwnPropertyDescriptor(proto, key);
32
+ if (descriptor.value !== void 0) {
33
+ if (typeof descriptor.value === "function") {
34
+ data[key] = descriptor.value;
35
+ }
36
+ } else if (descriptor.get || descriptor.set) {
37
+ Object.defineProperty(data, key, { get: descriptor.get, set: descriptor.set });
38
+ }
39
+ });
40
+ return Object.assign(data, instance);
41
+ };
42
+ }
43
+ component.default = Component2;
44
+ return component;
45
+ }
46
+ var componentExports = /* @__PURE__ */ requireComponent();
47
+ const Component = /* @__PURE__ */ getDefaultExportFromCjs(componentExports);
48
+ var __getOwnPropDesc = /* @__PURE__ */ (() => Object.getOwnPropertyDescriptor)();
49
+ var __decorateClass = (decorators, target, key, kind) => {
50
+ var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target;
51
+ for (var i = decorators.length - 1, decorator; i >= 0; i--)
52
+ if (decorator = decorators[i])
53
+ result = decorator(result) || result;
54
+ return result;
55
+ };
56
+ const defaultOptions = {
57
+ id: "",
58
+ selected: "",
59
+ path: [],
60
+ ignoreSelf: null,
61
+ placeholder: "- Select -",
62
+ placeholders: [],
63
+ ajaxUrl: "",
64
+ ajaxValueField: "value",
65
+ source: [],
66
+ labels: [],
67
+ labelWidth: "col-md-3",
68
+ fieldWidth: "col",
69
+ readonly: false,
70
+ disabled: false,
71
+ valueField: "id",
72
+ textField: "title",
73
+ horizontal: null,
74
+ horizontalColWidth: null,
75
+ defaultValue: "",
76
+ onSelectInit: () => {
77
+ },
78
+ onChange: () => {
79
+ },
80
+ onValueInit: () => {
81
+ }
82
+ };
83
+ let FieldCascadeSelect = class extends AlpineComponent {
84
+ options;
85
+ el;
86
+ canModify = false;
87
+ lists = [];
88
+ ajaxUrl = "";
89
+ values = [];
90
+ constructor(options = {}) {
91
+ super();
92
+ this.options = mergeDeep({}, defaultOptions, options);
93
+ this.options.id = this.options.id || "cascade-select-" + uid();
94
+ }
95
+ init() {
96
+ this.canModify = !this.options.readonly && !this.options.disabled;
97
+ this.ajaxUrl = this.options.ajaxUrl;
98
+ this.values = this.options.path.slice().map(String);
99
+ let values = this.options.path.slice();
100
+ if (values.length === 0) {
101
+ values = [null];
102
+ } else {
103
+ values.unshift(null);
104
+ }
105
+ let promise = Promise.resolve();
106
+ let lastValue = null;
107
+ values.forEach((v, i) => {
108
+ promise = promise.then(() => {
109
+ return this.loadItems(v, i).then((list) => {
110
+ if (list.length > 0) {
111
+ this.lists.push(list);
112
+ }
113
+ });
114
+ });
115
+ lastValue = v;
116
+ });
117
+ this.el = this.$el;
118
+ module(this.$el, "cascade.select", () => this);
119
+ this.valueInit(this.$el, lastValue, values);
120
+ }
121
+ getLabel(i) {
122
+ return this.options.labels[i] || `Level ${i + 1}`;
123
+ }
124
+ getId(i) {
125
+ return `${this.options.id}__level-${i}`;
126
+ }
127
+ getListValue(i) {
128
+ return this.values[i] || "";
129
+ }
130
+ isSelected(i, item) {
131
+ return String(this.getListValue(i)) === String(item[this.options.valueField]);
132
+ }
133
+ getFinalValue() {
134
+ const values = this.values.slice();
135
+ if (values.length === 0) {
136
+ return this.options.defaultValue;
137
+ }
138
+ const v = values.filter((v2) => v2 != null).filter((v2) => v2 !== "").pop();
139
+ if (v == void 0) {
140
+ return this.options.defaultValue;
141
+ }
142
+ return v;
143
+ }
144
+ getLevel() {
145
+ return this.values.length;
146
+ }
147
+ async onChange(i, event) {
148
+ const el = event.target;
149
+ this.values[i] = el.value;
150
+ this.options.onChange(event);
151
+ event.stopPropagation();
152
+ const changeEvent = new CustomEvent("change", {
153
+ detail: {
154
+ el,
155
+ component: this,
156
+ value: el.value,
157
+ path: this.values
158
+ }
159
+ });
160
+ this.el?.dispatchEvent(changeEvent);
161
+ if (el.value === "") {
162
+ this.lists.splice(i + 1);
163
+ this.values.splice(i + 1);
164
+ return;
165
+ }
166
+ let list = await this.loadItems(el.value, i);
167
+ this.lists.splice(i + 1);
168
+ this.values.splice(i + 1);
169
+ if (list.length > 0) {
170
+ this.lists.push(list);
171
+ }
172
+ }
173
+ async loadItems(parentId, i) {
174
+ if (this.ajaxUrl) {
175
+ const http = await useHttpClient();
176
+ let res = await http.get(
177
+ this.ajaxUrl,
178
+ {
179
+ params: {
180
+ [this.options.ajaxValueField]: parentId,
181
+ self: this.options.ignoreSelf || null
182
+ }
183
+ }
184
+ );
185
+ return await res.data.data;
186
+ }
187
+ if (parentId) {
188
+ return Promise.resolve(
189
+ this.handleSourceItems(
190
+ this.findFromList(this.lists[i - 1] || [], parentId)?.children || []
191
+ )
192
+ );
193
+ }
194
+ return Promise.resolve(this.handleSourceItems(this.options.source));
195
+ }
196
+ valueInit($select, value, path) {
197
+ const event = new CustomEvent("value.init", {
198
+ detail: {
199
+ el: $select,
200
+ component: this,
201
+ value,
202
+ path
203
+ }
204
+ });
205
+ this.options.onSelectInit(event);
206
+ this.$el.dispatchEvent(event);
207
+ }
208
+ selectInit($select) {
209
+ const event = new CustomEvent("select.init", {
210
+ detail: {
211
+ el: $select,
212
+ component: this
213
+ }
214
+ });
215
+ this.options.onSelectInit(event);
216
+ this.$el.dispatchEvent(event);
217
+ }
218
+ handleSourceItems(items) {
219
+ return items.map((item) => {
220
+ return {
221
+ [this.options.valueField]: item.value[this.options.valueField],
222
+ [this.options.textField]: item.value[this.options.textField],
223
+ children: item.children
224
+ };
225
+ }).filter((item) => {
226
+ if (this.options.ignoreSelf) {
227
+ return item[this.options.valueField] != this.options.ignoreSelf;
228
+ }
229
+ return item;
230
+ });
231
+ }
232
+ findFromList(items, value) {
233
+ const found = items.filter((item) => item[this.options.valueField] == value);
234
+ return found.shift();
235
+ }
236
+ getPlaceholder(i) {
237
+ if (this.options.placeholders[i]) {
238
+ return this.options.placeholders[i];
239
+ }
240
+ return this.options.placeholder;
241
+ }
242
+ };
243
+ FieldCascadeSelect = /* @__PURE__ */ __decorateClass([
244
+ Component
245
+ ], FieldCascadeSelect);
246
+ async function init() {
247
+ await prepareAlpineDefer(() => {
248
+ Alpine.data("CascadeSelect", (options) => new FieldCascadeSelect(options));
249
+ });
250
+ await initAlpineComponent("data-cascade-select");
251
+ }
252
+ const ready = /* @__PURE__ */ init();
253
+ export {
254
+ ready
255
+ };
256
+ //# sourceMappingURL=field-cascade-select.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"field-cascade-select.js","sources":["../../../../../../node_modules/@rubenbimmel/alpine-class-component/dist/alpineComponent.js","../../../../../../node_modules/@rubenbimmel/alpine-class-component/dist/decorators/component.js","../../src/module/field-cascade-select.ts"],"sourcesContent":["\"use strict\";\r\nObject.defineProperty(exports, \"__esModule\", { value: true });\r\nclass AlpineComponent {\r\n}\r\nexports.default = AlpineComponent;\r\n","\"use strict\";\r\nObject.defineProperty(exports, \"__esModule\", { value: true });\r\nfunction Component(component) {\r\n return function (...args) {\r\n let instance = new component(...args);\r\n const proto = component.prototype;\r\n let data = {};\r\n Object.getOwnPropertyNames(proto).forEach(function (key) {\r\n if (key === 'constructor') {\r\n return;\r\n }\r\n const descriptor = Object.getOwnPropertyDescriptor(proto, key);\r\n if (descriptor.value !== void 0) {\r\n if (typeof descriptor.value === 'function') {\r\n data[key] = descriptor.value;\r\n }\r\n }\r\n else if (descriptor.get || descriptor.set) {\r\n Object.defineProperty(data, key, { get: descriptor.get, set: descriptor.set });\r\n }\r\n });\r\n return Object.assign(data, instance);\r\n };\r\n}\r\nexports.default = Component;\r\n","import AlpineComponent from '@rubenbimmel/alpine-class-component/dist/alpineComponent';\r\nimport Component from '@rubenbimmel/alpine-class-component/dist/decorators/component';\r\nimport { useHttpClient } from '../composable';\r\nimport { initAlpineComponent, module, prepareAlpineDefer, uid } from '../service';\r\nimport { mergeDeep } from '../utilities';\r\n\r\ninterface CascadeSelectOptions {\r\n id: string;\r\n selected: string;\r\n path: string[];\r\n ignoreSelf: boolean | null;\r\n placeholder: string;\r\n placeholders: string[];\r\n ajaxUrl: string;\r\n ajaxValueField: string;\r\n source: string[];\r\n labels: string[];\r\n labelWidth: string;\r\n fieldWidth: string;\r\n readonly: boolean;\r\n disabled: boolean;\r\n valueField: string;\r\n textField: string;\r\n horizontal: boolean | null;\r\n horizontalColWidth: string | null;\r\n defaultValue: string;\r\n onSelectInit: Function,\r\n onChange: Function,\r\n onValueInit: Function,\r\n}\r\n\r\nconst defaultOptions = {\r\n id: '',\r\n selected: '',\r\n path: [],\r\n ignoreSelf: null,\r\n placeholder: '- Select -',\r\n placeholders: [],\r\n ajaxUrl: '',\r\n ajaxValueField: 'value',\r\n source: [],\r\n labels: [],\r\n labelWidth: 'col-md-3',\r\n fieldWidth: 'col',\r\n readonly: false,\r\n disabled: false,\r\n valueField: 'id',\r\n textField: 'title',\r\n horizontal: null,\r\n horizontalColWidth: null,\r\n defaultValue: '',\r\n onSelectInit: () => {\r\n },\r\n onChange: () => {\r\n },\r\n onValueInit: () => {\r\n },\r\n};\r\n\r\n@Component\r\nclass FieldCascadeSelect extends AlpineComponent {\r\n options: CascadeSelectOptions;\r\n el?: HTMLElement;\r\n canModify: boolean = false;\r\n lists: any[] = [];\r\n ajaxUrl: string = '';\r\n values: Array<string | null> = [];\r\n\r\n constructor(options: Partial<CascadeSelectOptions> = {}) {\r\n super();\r\n\r\n this.options = mergeDeep({}, defaultOptions, options);\r\n\r\n this.options.id = this.options.id || 'cascade-select-' + uid();\r\n }\r\n\r\n init() {\r\n this.canModify = !this.options.readonly && !this.options.disabled;\r\n this.ajaxUrl = this.options.ajaxUrl;\r\n this.values = this.options.path.slice().map(String);\r\n\r\n let values: Array<string | null> = this.options.path.slice();\r\n\r\n if (values.length === 0) {\r\n values = [null];\r\n } else {\r\n values.unshift(null);\r\n }\r\n\r\n let promise = Promise.resolve();\r\n let lastValue: string | null = null;\r\n\r\n values.forEach((v, i) => {\r\n promise = promise.then(() => {\r\n return this.loadItems(v, i).then((list) => {\r\n if (list.length > 0) {\r\n this.lists.push(list);\r\n }\r\n });\r\n });\r\n\r\n lastValue = v;\r\n });\r\n\r\n this.el = this.$el;\r\n\r\n module(this.$el, 'cascade.select', () => this);\r\n\r\n this.valueInit(this.$el, lastValue, values);\r\n }\r\n\r\n getLabel(i: number) {\r\n return this.options.labels[i] || `Level ${i + 1}`;\r\n }\r\n\r\n getId(i: number) {\r\n return `${this.options.id}__level-${i}`;\r\n }\r\n\r\n getListValue(i: number) {\r\n return this.values[i] || '';\r\n }\r\n\r\n isSelected(i: number, item: any) {\r\n return String(this.getListValue(i)) === String(item[this.options.valueField]);\r\n }\r\n\r\n getFinalValue() {\r\n const values = this.values.slice();\r\n\r\n if (values.length === 0) {\r\n return this.options.defaultValue;\r\n }\r\n\r\n const v = values\r\n .filter(v => v != null)\r\n .filter(v => v !== '')\r\n .pop();\r\n\r\n if (v == undefined) {\r\n return this.options.defaultValue;\r\n }\r\n\r\n return v;\r\n }\r\n\r\n getLevel() {\r\n return this.values.length;\r\n }\r\n\r\n async onChange(i: number, event: Event) {\r\n const el = event.target as HTMLSelectElement;\r\n\r\n this.values[i] = el.value;\r\n\r\n this.options.onChange(event);\r\n\r\n event.stopPropagation();\r\n\r\n const changeEvent = new CustomEvent('change', {\r\n detail: {\r\n el,\r\n component: this,\r\n value: el.value,\r\n path: this.values\r\n }\r\n });\r\n\r\n this.el?.dispatchEvent(changeEvent);\r\n\r\n if (el.value === '') {\r\n // Clear child\r\n this.lists.splice(i + 1);\r\n this.values.splice(i + 1);\r\n return;\r\n }\r\n\r\n // Get child list\r\n let list = await this.loadItems(el.value, i);\r\n this.lists.splice(i + 1);\r\n this.values.splice(i + 1);\r\n if (list.length > 0) {\r\n this.lists.push(list);\r\n }\r\n }\r\n\r\n async loadItems(parentId: string | null, i: number) {\r\n // Ajax\r\n if (this.ajaxUrl) {\r\n const http = await useHttpClient();\r\n\r\n let res = await http.get(\r\n this.ajaxUrl,\r\n {\r\n params: {\r\n [this.options.ajaxValueField]: parentId,\r\n self: this.options.ignoreSelf || null\r\n }\r\n }\r\n );\r\n return await res.data.data;\r\n }\r\n\r\n // Source\r\n if (parentId) {\r\n return Promise.resolve(\r\n this.handleSourceItems(\r\n this.findFromList(this.lists[i - 1] || [], parentId)?.children || []\r\n )\r\n );\r\n }\r\n\r\n return Promise.resolve(this.handleSourceItems(this.options.source));\r\n }\r\n\r\n valueInit($select: HTMLElement, value: string | null, path: Array<string | null>) {\r\n const event = new CustomEvent('value.init', {\r\n detail: {\r\n el: $select,\r\n component: this,\r\n value,\r\n path\r\n }\r\n });\r\n\r\n this.options.onSelectInit(event);\r\n\r\n this.$el.dispatchEvent(event);\r\n }\r\n\r\n selectInit($select: HTMLElement) {\r\n const event = new CustomEvent('select.init', {\r\n detail: {\r\n el: $select,\r\n component: this,\r\n }\r\n });\r\n\r\n this.options.onSelectInit(event);\r\n\r\n this.$el.dispatchEvent(event);\r\n }\r\n\r\n handleSourceItems(items: any[]) {\r\n return items.map(item => {\r\n return {\r\n [this.options.valueField]: item.value[this.options.valueField],\r\n [this.options.textField]: item.value[this.options.textField],\r\n children: item.children\r\n };\r\n })\r\n .filter(item => {\r\n if (this.options.ignoreSelf) {\r\n return item[this.options.valueField] != this.options.ignoreSelf;\r\n }\r\n\r\n return item;\r\n });\r\n }\r\n\r\n findFromList(items: any[], value: string) {\r\n const found = items.filter(item => item[this.options.valueField] == value);\r\n\r\n return found.shift();\r\n }\r\n\r\n getPlaceholder(i: number) {\r\n if (this.options.placeholders[i]) {\r\n return this.options.placeholders[i];\r\n }\r\n\r\n return this.options.placeholder;\r\n }\r\n}\r\n\r\ndeclare global {\r\n var S: any;\r\n}\r\n\r\nasync function init() {\r\n await prepareAlpineDefer(() => {\r\n Alpine.data('CascadeSelect', (options: CascadeSelectOptions) => new FieldCascadeSelect(options));\r\n });\r\n\r\n await initAlpineComponent('data-cascade-select');\r\n}\r\n\r\nexport const ready = init();\r\n\r\nexport interface CascadeSelectModule {\r\n ready: typeof ready;\r\n}\r\n"],"names":["AlpineComponent","Component","component","v"],"mappings":"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","x_google_ignoreList":[0,1]}
@@ -0,0 +1,218 @@
1
+ import { a as useUniDirective, a6 as watchAttributes, a5 as mergeDeep, _ as __, K as uid, B as html, G as simpleAlert, a7 as injectCssToDocument } from "./unicorn.js";
2
+ const css = ".c-file-drag {\n --bs-card-border-color: var(--bs-gray-400);\n --fd-delete-color: var(--bs-danger);\n overflow: hidden;\n border: 1px solid var(--bs-card-border-color, #ddd);\n}\n.c-file-drag label {\n border: none;\n}\n\n.c-file-drag-input {\n position: relative;\n display: inline-block;\n width: 100%;\n min-height: 100px;\n cursor: pointer;\n}\n.c-file-drag-input input {\n position: relative;\n z-index: 2;\n width: 100%;\n margin: 0;\n overflow: hidden;\n opacity: 0;\n height: 100%;\n cursor: pointer;\n}\n.c-file-drag-input input.hover + label {\n background-color: #efefef;\n}\n.c-file-drag-input input.is-invalid ~ .c-file-drag-input__label {\n border-color: var(--bs-danger);\n}\n.c-file-drag-input input:disabled {\n opacity: 0;\n cursor: no-drop;\n}\n.c-file-drag-input input:disabled + label {\n background-color: #eee;\n color: #999;\n}\n.c-file-drag-input input:disabled + label button {\n display: none;\n}\n.c-file-drag-input__label {\n position: absolute;\n top: 0;\n bottom: 0;\n left: 0;\n right: 0;\n z-index: 1;\n padding: 0.375rem 0.75rem;\n color: #495057;\n background-color: #fff;\n border: 1px solid var(--bs-gray-400);\n border-radius: 0.25rem;\n height: 100%;\n text-align: center;\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n gap: 0.5rem;\n transition: all 0.3s;\n cursor: pointer;\n}\n.c-file-drag-input__label > span {\n display: inline-block;\n width: 100%;\n}\n.c-file-drag-input label::after {\n content: none !important;\n}\n.c-file-drag-preview .c-file-drag-preview__delete {\n --bs-link-color-rgb: var(--bs-dark-rgb);\n}\n.c-file-drag-preview .c-file-drag-preview__delete.active {\n --bs-link-color-rgb: var(--bs-primary-rgb);\n}";
3
+ /* @__PURE__ */ injectCssToDocument(document, css);
4
+ const defaultOptions = {
5
+ maxFiles: void 0,
6
+ maxSize: void 0,
7
+ placeholder: "",
8
+ height: 125
9
+ };
10
+ class FileDragElement extends HTMLElement {
11
+ static is = "uni-file-drag";
12
+ element;
13
+ overlayLabel;
14
+ button;
15
+ options;
16
+ get inputSelector() {
17
+ return this.getAttribute("selector") || "input[type=file]";
18
+ }
19
+ get multiple() {
20
+ return this.element.multiple;
21
+ }
22
+ connectedCallback() {
23
+ this.element = this.querySelector(this.inputSelector);
24
+ this.prepareElements();
25
+ const options = JSON.parse(this.getAttribute("options") || "{}") || {};
26
+ const observer = watchAttributes(this.element);
27
+ observer.watch("readonly", (el) => {
28
+ el.disabled = el.readOnly;
29
+ });
30
+ if (this.element.readOnly) {
31
+ this.element.disabled = true;
32
+ }
33
+ this.options = mergeDeep({}, defaultOptions, options);
34
+ this.bindEvent();
35
+ this.style.visibility = "";
36
+ this.style.height = (this.options.height || 100) + "px";
37
+ }
38
+ bindEvent() {
39
+ this.element.addEventListener("dragover", () => {
40
+ this.element.classList.add("hover");
41
+ });
42
+ this.element.addEventListener("dragleave", () => {
43
+ this.element.classList.remove("hover");
44
+ });
45
+ this.element.addEventListener("drop", () => {
46
+ this.element.classList.remove("hover");
47
+ });
48
+ this.onChange();
49
+ this.element.addEventListener("change", (e) => {
50
+ this.onChange(e);
51
+ });
52
+ this.element.addEventListener("input", (e) => {
53
+ this.onChange(e);
54
+ });
55
+ }
56
+ prepareElements() {
57
+ if (this.children.length === 0) {
58
+ this.createElementsLayout();
59
+ }
60
+ this.overlayLabel = this.querySelector("[data-overlay-label]");
61
+ let button = this.overlayLabel.querySelector("button");
62
+ if (!button) {
63
+ button = document.createElement("button");
64
+ button.type = "button";
65
+ button.setAttribute("class", "c-file-drag-input__button btn btn-success btn-sm px-2 py-1");
66
+ button.innerText = __("unicorn.field.file.drag.button.text");
67
+ this.overlayLabel.appendChild(button);
68
+ }
69
+ this.button = button;
70
+ }
71
+ createElementsLayout() {
72
+ this.id ||= "c-file-drag-" + uid();
73
+ const name = this.getAttribute("name") || "file";
74
+ const inputId = this.id + "__input";
75
+ const btnText = __("unicorn.field.file.drag.button.text");
76
+ const input = html(`<input id="${inputId}" type="file" name="${name}" />`);
77
+ const label = html(`<label class="px-3 c-file-drag-input__label"
78
+ data-overlay-label
79
+ for="${inputId}">
80
+ <span class="label-text d-block">
81
+ <span class="fa fa-upload"></span>
82
+ </span>
83
+ <button type="button" class="c-file-drag-input__button btn btn-success btn-sm px-2 py-1">
84
+ ${btnText}
85
+ </button>
86
+ </label>`);
87
+ this.element = input;
88
+ this.overlayLabel = label;
89
+ this.appendChild(input);
90
+ this.appendChild(label);
91
+ }
92
+ onChange(evt) {
93
+ const files = this.element.files || [];
94
+ const limit = this.options.maxFiles;
95
+ const maxSize = this.options.maxSize;
96
+ let placeholder = this.options.placeholder;
97
+ const accepted = (this.element.getAttribute("accept") || this.element.getAttribute("data-accepted") || "").split(",").map((v) => v.trim()).filter((v) => v.length > 0).map((v) => {
98
+ if (v.indexOf("/") === -1 && v[0] === ".") {
99
+ return v.substring(1);
100
+ }
101
+ return v;
102
+ });
103
+ let text;
104
+ if (!placeholder) {
105
+ if (this.multiple) {
106
+ placeholder = __("unicorn.field.file.drag.placeholder.multiple");
107
+ } else {
108
+ placeholder = __("unicorn.field.file.drag.placeholder.single");
109
+ }
110
+ }
111
+ if (limit && files.length > limit) {
112
+ this.alert(__("unicorn.field.file.drag.message.max.files", limit), "", "warning");
113
+ evt?.preventDefault();
114
+ return;
115
+ }
116
+ let fileSize = 0;
117
+ Array.prototype.forEach.call(files, (file) => {
118
+ this.checkFileType(accepted, file);
119
+ fileSize += file.size;
120
+ });
121
+ if (maxSize && fileSize / 1024 / 1024 > maxSize) {
122
+ this.alert(
123
+ __(
124
+ "unicorn.field.file.drag.message.max.size",
125
+ maxSize < 1 ? maxSize * 1024 + "KB" : maxSize + "MB"
126
+ ),
127
+ "",
128
+ "warning"
129
+ );
130
+ evt?.preventDefault();
131
+ return;
132
+ }
133
+ if (files.length > 1) {
134
+ text = `<span class="fa fa-files fa-copy"></span> ${__("unicorn.field.file.drag.selected", files.length)}`;
135
+ } else if (files.length === 1) {
136
+ text = `<span class="fa fa-file"></span> ${files[0].name}`;
137
+ } else {
138
+ text = `<span class="fa fa-upload"></span> ${placeholder}`;
139
+ }
140
+ this.overlayLabel.querySelector("span").innerHTML = text;
141
+ }
142
+ checkFileType(accepted, file) {
143
+ const fileExt = file.name.split(".").pop() || "";
144
+ if (accepted.length) {
145
+ let allow = false;
146
+ accepted.forEach((type) => {
147
+ if (allow) {
148
+ return;
149
+ }
150
+ if (type.indexOf("/") !== -1) {
151
+ if (this.compareMimeType(type, file.type)) {
152
+ allow = true;
153
+ }
154
+ } else {
155
+ if (type.toLowerCase() === fileExt.toLowerCase()) {
156
+ allow = true;
157
+ }
158
+ }
159
+ });
160
+ if (!allow) {
161
+ this.alert(
162
+ __("unicorn.field.file.drag.message.unaccepted.files"),
163
+ __("unicorn.field.file.drag.message.unaccepted.files.desc", accepted.join(", ")),
164
+ "warning"
165
+ );
166
+ throw new Error("Not accepted file ext");
167
+ }
168
+ }
169
+ }
170
+ compareMimeType(accepted, mime) {
171
+ const accepted2 = accepted.split("/");
172
+ const mime2 = mime.split("/");
173
+ if (accepted2[1] === "*") {
174
+ return accepted2[0] === mime2[0];
175
+ }
176
+ return accepted === mime;
177
+ }
178
+ async alert(title, text = "", type = "info") {
179
+ await simpleAlert(title, text, type);
180
+ }
181
+ }
182
+ /* @__PURE__ */ customElements.define(/* @__PURE__ */ (() => FileDragElement.is)(), FileDragElement);
183
+ const ready = /* @__PURE__ */ useUniDirective("file-drag-field", {
184
+ mounted(el) {
185
+ const input = el.querySelector("input[type=file]");
186
+ const placeholderInput = el.querySelector("[data-role=placeholder]");
187
+ const preview = el.querySelector(".c-file-drag-preview");
188
+ if (preview) {
189
+ const previewLink = preview.querySelector(".c-file-drag-preview__link");
190
+ const delButton = preview.querySelector(".c-file-drag-preview__delete");
191
+ let inputValue = placeholderInput.value;
192
+ let required = input.required;
193
+ if (placeholderInput.value) {
194
+ input.required = false;
195
+ }
196
+ delButton.addEventListener("click", () => {
197
+ if (delButton.classList.contains("active")) {
198
+ previewLink.style.textDecoration = "";
199
+ previewLink.style.setProperty("color", "");
200
+ placeholderInput.value = inputValue;
201
+ delButton.classList.remove("active");
202
+ input.required = false;
203
+ } else {
204
+ previewLink.style.textDecoration = "line-through";
205
+ previewLink.style.color = "var(--fd-delete-color, var(--bs-danger))";
206
+ placeholderInput.value = "";
207
+ delButton.classList.add("active");
208
+ input.required = required;
209
+ }
210
+ });
211
+ }
212
+ }
213
+ });
214
+ export {
215
+ FileDragElement,
216
+ ready
217
+ };
218
+ //# sourceMappingURL=field-file-drag.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"field-file-drag.js","sources":["../../src/module/field-file-drag.ts"],"sourcesContent":["import css from '../../scss/field/file-drag.scss?inline';\r\nimport { useUniDirective } from '../composable';\r\nimport { __, html, injectCssToDocument, simpleAlert, uid, watchAttributes } from '../service';\r\nimport { mergeDeep } from '../utilities';\r\n\r\ninjectCssToDocument(document, css);\r\n\r\nexport interface FileDragOptions {\r\n maxFiles: number | undefined;\r\n maxSize: number | undefined;\r\n placeholder: string;\r\n height: number;\r\n}\r\n\r\nconst defaultOptions: FileDragOptions = {\r\n maxFiles: undefined,\r\n maxSize: undefined,\r\n placeholder: '',\r\n height: 125,\r\n}\r\n\r\nexport class FileDragElement extends HTMLElement {\r\n static is = 'uni-file-drag';\r\n\r\n element!: HTMLInputElement;\r\n overlayLabel!: HTMLLabelElement;\r\n button!: HTMLButtonElement;\r\n options!: FileDragOptions;\r\n\r\n get inputSelector() {\r\n return this.getAttribute('selector') || 'input[type=file]';\r\n }\r\n\r\n get multiple() {\r\n return this.element.multiple;\r\n }\r\n\r\n connectedCallback(): void {\r\n this.element = this.querySelector(this.inputSelector)!;\r\n\r\n this.prepareElements();\r\n\r\n const options = JSON.parse(this.getAttribute('options') || '{}') || {};\r\n\r\n const observer = watchAttributes(this.element);\r\n observer.watch('readonly', (el) => {\r\n el.disabled = el.readOnly;\r\n });\r\n\r\n if (this.element.readOnly) {\r\n this.element.disabled = true;\r\n }\r\n\r\n this.options = mergeDeep({}, defaultOptions, options);\r\n\r\n this.bindEvent();\r\n\r\n this.style.visibility = '';\r\n\r\n this.style.height = (this.options.height || 100) + 'px';\r\n }\r\n\r\n bindEvent() {\r\n this.element.addEventListener('dragover', () => {\r\n this.element.classList.add('hover');\r\n });\r\n\r\n this.element.addEventListener('dragleave', () => {\r\n this.element.classList.remove('hover');\r\n });\r\n\r\n this.element.addEventListener('drop', () => {\r\n this.element.classList.remove('hover');\r\n });\r\n\r\n this.onChange();\r\n\r\n this.element.addEventListener('change', (e) => {\r\n this.onChange(e);\r\n });\r\n this.element.addEventListener('input', (e) => {\r\n this.onChange(e);\r\n });\r\n }\r\n\r\n prepareElements() {\r\n if (this.children.length === 0) {\r\n this.createElementsLayout();\r\n }\r\n\r\n this.overlayLabel = this.querySelector<HTMLLabelElement>('[data-overlay-label]')!;\r\n\r\n let button = this.overlayLabel.querySelector('button');\r\n\r\n // B/C for new file drag style\r\n if (!button) {\r\n button = document.createElement('button');\r\n button.type = 'button';\r\n button.setAttribute('class', 'c-file-drag-input__button btn btn-success btn-sm px-2 py-1');\r\n button.innerText = __('unicorn.field.file.drag.button.text');\r\n this.overlayLabel.appendChild(button);\r\n }\r\n\r\n this.button = button;\r\n }\r\n\r\n createElementsLayout() {\r\n this.id ||= 'c-file-drag-' + uid();\r\n const name = this.getAttribute('name') || 'file';\r\n const inputId = this.id + '__input';\r\n const btnText = __('unicorn.field.file.drag.button.text');\r\n\r\n const input = html(`<input id=\"${inputId}\" type=\"file\" name=\"${name}\" />`);\r\n const label = html(`<label class=\"px-3 c-file-drag-input__label\"\r\n data-overlay-label\r\n for=\"${inputId}\">\r\n <span class=\"label-text d-block\">\r\n <span class=\"fa fa-upload\"></span>\r\n </span>\r\n <button type=\"button\" class=\"c-file-drag-input__button btn btn-success btn-sm px-2 py-1\">\r\n ${btnText}\r\n </button>\r\n </label>`);\r\n\r\n this.element = input as HTMLInputElement;\r\n this.overlayLabel = label as HTMLLabelElement;\r\n\r\n this.appendChild(input);\r\n this.appendChild(label);\r\n }\r\n\r\n onChange(evt?: Event) {\r\n const files = this.element.files || [];\r\n const limit = this.options.maxFiles;\r\n const maxSize = this.options.maxSize;\r\n let placeholder = this.options.placeholder;\r\n\r\n const accepted = (this.element.getAttribute('accept') || this.element.getAttribute('data-accepted') || '')\r\n .split(',')\r\n .map(v => v.trim())\r\n .filter(v => v.length > 0)\r\n .map(v => {\r\n if (v.indexOf('/') === -1 && v[0] === '.') {\r\n return v.substring(1);\r\n }\r\n\r\n return v;\r\n });\r\n\r\n let text: string;\r\n\r\n if (!placeholder) {\r\n if (this.multiple) {\r\n placeholder = __('unicorn.field.file.drag.placeholder.multiple');\r\n } else {\r\n placeholder = __('unicorn.field.file.drag.placeholder.single');\r\n }\r\n }\r\n\r\n // Files limit\r\n if (limit && files.length > limit) {\r\n this.alert(__('unicorn.field.file.drag.message.max.files', limit), '', 'warning');\r\n evt?.preventDefault();\r\n return;\r\n }\r\n\r\n // Files size\r\n let fileSize = 0;\r\n Array.prototype.forEach.call(files, file => {\r\n this.checkFileType(accepted, file);\r\n\r\n fileSize += file.size;\r\n });\r\n\r\n if (maxSize && (fileSize / 1024 / 1024) > maxSize) {\r\n this.alert(\r\n __(\r\n 'unicorn.field.file.drag.message.max.size',\r\n maxSize < 1 ? (maxSize * 1024) + 'KB' : maxSize + 'MB'\r\n ),\r\n '',\r\n 'warning'\r\n );\r\n evt?.preventDefault();\r\n return;\r\n }\r\n\r\n if (files.length > 1) {\r\n text = `<span class=\"fa fa-files fa-copy\"></span> ${__('unicorn.field.file.drag.selected', files.length)}`;\r\n } else if (files.length === 1) {\r\n text = `<span class=\"fa fa-file\"></span> ${files[0].name}`;\r\n } else {\r\n text = `<span class=\"fa fa-upload\"></span> ${placeholder}`;\r\n }\r\n\r\n //replace the \"Choose a file\" label\r\n this.overlayLabel.querySelector<HTMLSpanElement>('span')!.innerHTML = text;\r\n }\r\n\r\n checkFileType(accepted: string[], file: File) {\r\n const fileExt = file.name.split('.').pop() || '';\r\n\r\n if (accepted.length) {\r\n let allow = false;\r\n\r\n accepted.forEach((type) => {\r\n if (allow) {\r\n return;\r\n }\r\n\r\n if (type.indexOf('/') !== -1) {\r\n if (this.compareMimeType(type, file.type)) {\r\n allow = true;\r\n }\r\n } else {\r\n if (type.toLowerCase() === fileExt.toLowerCase()) {\r\n allow = true;\r\n }\r\n }\r\n });\r\n\r\n if (!allow) {\r\n this.alert(\r\n __('unicorn.field.file.drag.message.unaccepted.files'),\r\n __('unicorn.field.file.drag.message.unaccepted.files.desc', accepted.join(', ')),\r\n 'warning'\r\n );\r\n throw new Error('Not accepted file ext');\r\n }\r\n }\r\n }\r\n\r\n compareMimeType(accepted: string, mime: string) {\r\n const accepted2 = accepted.split('/');\r\n const mime2 = mime.split('/');\r\n\r\n if (accepted2[1] === '*') {\r\n return accepted2[0] === mime2[0];\r\n }\r\n\r\n return accepted === mime;\r\n }\r\n\r\n async alert(title: string, text: string = '', type: string = 'info') {\r\n await simpleAlert(title, text, type);\r\n }\r\n}\r\n\r\ncustomElements.define(FileDragElement.is, FileDragElement);\r\n\r\nexport const ready = useUniDirective('file-drag-field', {\r\n mounted(el) {\r\n const input = el.querySelector<HTMLInputElement>('input[type=file]')!;\r\n const placeholderInput = el.querySelector<HTMLInputElement>('[data-role=placeholder]')!;\r\n\r\n const preview = el.querySelector('.c-file-drag-preview');\r\n\r\n if (preview) {\r\n const previewLink = preview.querySelector<HTMLAnchorElement>('.c-file-drag-preview__link')!;\r\n const delButton = preview.querySelector<HTMLAnchorElement>('.c-file-drag-preview__delete')!;\r\n // let linkTitle = previewLink.textContent;\r\n let inputValue = placeholderInput.value;\r\n let required = input.required;\r\n\r\n if (placeholderInput.value) {\r\n input.required = false;\r\n }\r\n\r\n delButton.addEventListener('click', () => {\r\n if (delButton.classList.contains('active')) {\r\n // Restore\r\n previewLink.style.textDecoration = '';\r\n previewLink.style.setProperty('color', '');\r\n placeholderInput.value = inputValue;\r\n delButton.classList.remove('active');\r\n input.required = false;\r\n } else {\r\n // Delete\r\n previewLink.style.textDecoration = 'line-through';\r\n previewLink.style.color = 'var(--fd-delete-color, var(--bs-danger))';\r\n placeholderInput.value = '';\r\n delButton.classList.add('active');\r\n input.required = required;\r\n }\r\n });\r\n }\r\n }\r\n});\r\n\r\nexport interface FileDragModule {\r\n FileDragElement: typeof FileDragElement;\r\n}\r\n"],"names":[],"mappings":"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"}