vgapp 1.1.6 → 1.1.7
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.
- package/CHANGELOG.md +10 -1
- package/README.md +48 -48
- package/app/langs/en/buttons.json +17 -17
- package/app/langs/en/messages.json +36 -36
- package/app/langs/ru/buttons.json +17 -17
- package/app/langs/ru/messages.json +36 -36
- package/app/modules/vgfilepreview/js/i18n.js +56 -56
- package/app/modules/vgfilepreview/js/renderers/image-modal.js +145 -145
- package/app/modules/vgfilepreview/js/renderers/image.js +92 -92
- package/app/modules/vgfilepreview/js/renderers/index.js +19 -19
- package/app/modules/vgfilepreview/js/renderers/office-modal.js +168 -168
- package/app/modules/vgfilepreview/js/renderers/office.js +79 -79
- package/app/modules/vgfilepreview/js/renderers/pdf-modal.js +260 -260
- package/app/modules/vgfilepreview/js/renderers/pdf.js +76 -76
- package/app/modules/vgfilepreview/js/renderers/playlist.js +71 -71
- package/app/modules/vgfilepreview/js/renderers/text-modal.js +343 -343
- package/app/modules/vgfilepreview/js/renderers/text.js +83 -83
- package/app/modules/vgfilepreview/js/renderers/video-modal.js +272 -272
- package/app/modules/vgfilepreview/js/renderers/video.js +80 -80
- package/app/modules/vgfilepreview/js/renderers/zip-modal.js +522 -522
- package/app/modules/vgfilepreview/js/renderers/zip.js +89 -89
- package/app/modules/vgfilepreview/js/vgfilepreview.js +7 -7
- package/app/modules/vgfilepreview/readme.md +68 -68
- package/app/modules/vgfilepreview/scss/_variables.scss +113 -113
- package/app/modules/vgfilepreview/scss/vgfilepreview.scss +464 -464
- package/app/modules/vgfiles/js/base.js +26 -26
- package/app/modules/vgfiles/js/droppable.js +260 -260
- package/app/modules/vgfiles/js/render.js +153 -153
- package/app/modules/vgfiles/js/vgfiles.js +41 -41
- package/app/modules/vgfiles/readme.md +123 -123
- package/app/modules/vgfiles/scss/_variables.scss +18 -18
- package/app/modules/vgfiles/scss/vgfiles.scss +148 -148
- package/app/modules/vgformsender/js/vgformsender.js +1 -1
- package/app/modules/vgmodal/js/vgmodal.drag.js +332 -332
- package/app/modules/vgmodal/js/vgmodal.js +33 -33
- package/app/modules/vgmodal/js/vgmodal.resize.js +435 -435
- package/app/modules/vgnav/js/vgnav.js +135 -135
- package/app/modules/vgnav/readme.md +67 -67
- package/app/modules/vgnestable/README.md +307 -307
- package/app/modules/vgnestable/scss/_variables.scss +60 -60
- package/app/modules/vgnestable/scss/vgnestable.scss +163 -163
- package/app/modules/vgselect/js/vgselect.js +39 -39
- package/app/modules/vgselect/scss/vgselect.scss +22 -22
- package/app/modules/vgspy/readme.md +28 -28
- package/app/utils/js/components/audio-metadata.js +240 -240
- package/app/utils/js/components/file-icon.js +109 -109
- package/app/utils/js/components/file-preview.js +304 -304
- package/app/utils/js/components/sanitize.js +150 -150
- package/app/utils/js/components/video-metadata.js +140 -140
- package/build/vgapp.css +1 -1
- package/build/vgapp.css.map +1 -1
- package/build/vgapp.js.map +1 -1
- package/index.scss +9 -9
- package/package.json +1 -1
|
@@ -1,304 +1,304 @@
|
|
|
1
|
-
const DEFAULT_OPTIONS = {
|
|
2
|
-
pathAttribute: 'data-vg-filepreview',
|
|
3
|
-
fieldsAttribute: 'data-fields',
|
|
4
|
-
validAttribute: 'data-vg-filepreview-valid',
|
|
5
|
-
errorAttribute: 'data-vg-filepreview-error',
|
|
6
|
-
errorValue: 'invalid-file-path',
|
|
7
|
-
fieldAttribute: 'data-vg-filepreview-field',
|
|
8
|
-
editableAttribute: 'data-vg-filepreview-editable',
|
|
9
|
-
editableFieldsAttribute: 'data-vg-filepreview-editable-fields',
|
|
10
|
-
missingFieldsAttribute: 'data-vg-filepreview-missing-fields'
|
|
11
|
-
};
|
|
12
|
-
|
|
13
|
-
class FilePreviewHelper {
|
|
14
|
-
constructor(element, options = {}) {
|
|
15
|
-
this._element = element;
|
|
16
|
-
this._options = Object.assign({}, DEFAULT_OPTIONS, options);
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
getFilePath() {
|
|
20
|
-
if (!this._element) {
|
|
21
|
-
return '';
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
const filePath = this._element.getAttribute(this._options.pathAttribute);
|
|
25
|
-
return typeof filePath === 'string' ? filePath.trim() : '';
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
getFields() {
|
|
29
|
-
if (!this._element) {
|
|
30
|
-
return [];
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
const rawFields = this._element.getAttribute(this._options.fieldsAttribute);
|
|
34
|
-
if (typeof rawFields !== 'string' || !rawFields.trim()) {
|
|
35
|
-
return [];
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
return rawFields
|
|
39
|
-
.split(',')
|
|
40
|
-
.map(field => field.trim())
|
|
41
|
-
.filter(field => field && /^[-_a-zA-Z0-9]+$/.test(field));
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
validateFilePath(path) {
|
|
45
|
-
if (!path || path === '#' || /^javascript:/i.test(path)) {
|
|
46
|
-
return { isValid: false, fileUrl: null };
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
try {
|
|
50
|
-
const parsed = new URL(path, window.location.origin);
|
|
51
|
-
const protocol = parsed.protocol;
|
|
52
|
-
|
|
53
|
-
if (!['http:', 'https:', 'blob:', 'data:'].includes(protocol)) {
|
|
54
|
-
return { isValid: false, fileUrl: null };
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
if (protocol === 'data:' || protocol === 'blob:') {
|
|
58
|
-
return { isValid: true, fileUrl: parsed };
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
const pathname = parsed.pathname || '';
|
|
62
|
-
const fileName = pathname.split('/').filter(Boolean).pop() || '';
|
|
63
|
-
if (!fileName || !fileName.includes('.')) {
|
|
64
|
-
return { isValid: false, fileUrl: null };
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
return { isValid: true, fileUrl: parsed };
|
|
68
|
-
} catch {
|
|
69
|
-
return { isValid: false, fileUrl: null };
|
|
70
|
-
}
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
syncState(isValid) {
|
|
74
|
-
if (!this._element) {
|
|
75
|
-
return;
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
this._element.setAttribute(this._options.validAttribute, isValid ? 'true' : 'false');
|
|
79
|
-
|
|
80
|
-
if (isValid) {
|
|
81
|
-
this._element.removeAttribute(this._options.errorAttribute);
|
|
82
|
-
return;
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
this._element.setAttribute(this._options.errorAttribute, this._options.errorValue);
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
resolveEditableFields(fields = []) {
|
|
89
|
-
if (!this._element || !Array.isArray(fields) || !fields.length) {
|
|
90
|
-
return {};
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
return fields.reduce((acc, field) => {
|
|
94
|
-
acc[field] = this._element.querySelector(`.${this._escapeClassName(field)}`);
|
|
95
|
-
return acc;
|
|
96
|
-
}, {});
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
syncEditableFields(fieldsMap = {}) {
|
|
100
|
-
if (!this._element) {
|
|
101
|
-
return;
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
const previousMarkedFields = this._element.querySelectorAll(`[${this._options.fieldAttribute}]`);
|
|
105
|
-
previousMarkedFields.forEach((node) => {
|
|
106
|
-
node.removeAttribute(this._options.fieldAttribute);
|
|
107
|
-
node.setAttribute(this._options.editableAttribute, 'false');
|
|
108
|
-
});
|
|
109
|
-
|
|
110
|
-
const editableFields = [];
|
|
111
|
-
const missingFields = [];
|
|
112
|
-
|
|
113
|
-
Object.entries(fieldsMap).forEach(([field, node]) => {
|
|
114
|
-
if (!node) {
|
|
115
|
-
missingFields.push(field);
|
|
116
|
-
return;
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
node.setAttribute(this._options.fieldAttribute, field);
|
|
120
|
-
node.setAttribute(this._options.editableAttribute, 'true');
|
|
121
|
-
editableFields.push(field);
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
if (editableFields.length) {
|
|
125
|
-
this._element.setAttribute(this._options.editableFieldsAttribute, editableFields.join(','));
|
|
126
|
-
} else {
|
|
127
|
-
this._element.removeAttribute(this._options.editableFieldsAttribute);
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
if (missingFields.length) {
|
|
131
|
-
this._element.setAttribute(this._options.missingFieldsAttribute, missingFields.join(','));
|
|
132
|
-
} else {
|
|
133
|
-
this._element.removeAttribute(this._options.missingFieldsAttribute);
|
|
134
|
-
}
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
_escapeClassName(value) {
|
|
138
|
-
if (window.CSS && typeof window.CSS.escape === 'function') {
|
|
139
|
-
return window.CSS.escape(value);
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
return String(value).replace(/([ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~])/g, '\\$1');
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
getFileMeta(path) {
|
|
146
|
-
const normalizedPath = typeof path === 'string' ? path.trim() : '';
|
|
147
|
-
const fallback = {
|
|
148
|
-
name: '',
|
|
149
|
-
ext: '',
|
|
150
|
-
originalName: '',
|
|
151
|
-
isMedia: false,
|
|
152
|
-
sizeBytes: null,
|
|
153
|
-
sizeText: ''
|
|
154
|
-
};
|
|
155
|
-
|
|
156
|
-
if (!normalizedPath) {
|
|
157
|
-
return fallback;
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
try {
|
|
161
|
-
const parsedUrl = new URL(normalizedPath, window.location.origin);
|
|
162
|
-
const pathname = parsedUrl.pathname || '';
|
|
163
|
-
const params = parsedUrl.searchParams;
|
|
164
|
-
const originalName = this._readOriginalName(params);
|
|
165
|
-
let fileName = decodeURIComponent(pathname.split('/').pop() || '');
|
|
166
|
-
|
|
167
|
-
// blob/data urls often do not contain real filename; prefer explicit original name
|
|
168
|
-
if (!fileName || !fileName.includes('.')) {
|
|
169
|
-
fileName = originalName || fileName;
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
const ext = fileName.includes('.') ? `.${fileName.split('.').pop() || ''}` : '';
|
|
173
|
-
const sizeBytes = this._readSize(params);
|
|
174
|
-
const mediaByExt = ['.mp3', '.wav', '.ogg', '.flac', '.aac', '.m4a', '.mp4', '.webm', '.mov', '.mkv', '.avi'];
|
|
175
|
-
|
|
176
|
-
return {
|
|
177
|
-
name: fileName,
|
|
178
|
-
ext: ext.toLowerCase(),
|
|
179
|
-
originalName,
|
|
180
|
-
isMedia: mediaByExt.includes(ext.toLowerCase()),
|
|
181
|
-
sizeBytes,
|
|
182
|
-
sizeText: this._formatSize(sizeBytes)
|
|
183
|
-
};
|
|
184
|
-
} catch {
|
|
185
|
-
const cleanPath = normalizedPath.split('#')[0].split('?')[0];
|
|
186
|
-
let fileName = decodeURIComponent(cleanPath.split('/').pop() || '');
|
|
187
|
-
const originalName = this._readOriginalName(null);
|
|
188
|
-
if (!fileName || !fileName.includes('.')) {
|
|
189
|
-
fileName = originalName || fileName;
|
|
190
|
-
}
|
|
191
|
-
const ext = fileName.includes('.') ? `.${fileName.split('.').pop() || ''}` : '';
|
|
192
|
-
const sizeBytes = this._readSize(null);
|
|
193
|
-
|
|
194
|
-
return {
|
|
195
|
-
name: fileName,
|
|
196
|
-
ext: ext.toLowerCase(),
|
|
197
|
-
originalName: originalName || '',
|
|
198
|
-
isMedia: false,
|
|
199
|
-
sizeBytes,
|
|
200
|
-
sizeText: this._formatSize(sizeBytes)
|
|
201
|
-
};
|
|
202
|
-
}
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
_readOriginalName(searchParams) {
|
|
206
|
-
const fromData = this._element?.getAttribute('data-original-name')
|
|
207
|
-
|| this._element?.getAttribute('data-vg-filepreview-original-name')
|
|
208
|
-
|| '';
|
|
209
|
-
if (fromData) {
|
|
210
|
-
return String(fromData).trim();
|
|
211
|
-
}
|
|
212
|
-
|
|
213
|
-
const keys = ['original_name', 'originalName', 'filename', 'name'];
|
|
214
|
-
for (const key of keys) {
|
|
215
|
-
const value = searchParams.get(key);
|
|
216
|
-
if (value) {
|
|
217
|
-
return String(value).trim();
|
|
218
|
-
}
|
|
219
|
-
}
|
|
220
|
-
|
|
221
|
-
return '';
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
_readSize(searchParams) {
|
|
225
|
-
const dataCandidates = [
|
|
226
|
-
this._element?.getAttribute('data-size'),
|
|
227
|
-
this._element?.getAttribute('data-vg-filepreview-size')
|
|
228
|
-
];
|
|
229
|
-
|
|
230
|
-
for (const value of dataCandidates) {
|
|
231
|
-
const parsed = this._parseSizeValue(value);
|
|
232
|
-
if (parsed !== null) {
|
|
233
|
-
return parsed;
|
|
234
|
-
}
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
if (!searchParams) {
|
|
238
|
-
return null;
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
const keys = ['size', 'size_bytes', 'bytes'];
|
|
242
|
-
for (const key of keys) {
|
|
243
|
-
const value = searchParams.get(key);
|
|
244
|
-
const parsed = this._parseSizeValue(value);
|
|
245
|
-
if (parsed !== null) {
|
|
246
|
-
return parsed;
|
|
247
|
-
}
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
return null;
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
_parseSizeValue(value) {
|
|
254
|
-
if (value === null || value === undefined) {
|
|
255
|
-
return null;
|
|
256
|
-
}
|
|
257
|
-
|
|
258
|
-
const normalized = String(value).trim().toLowerCase();
|
|
259
|
-
if (!normalized) {
|
|
260
|
-
return null;
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
const match = normalized.match(/^(\d+(?:\.\d+)?)\s*(b|kb|mb|gb|tb)?$/i);
|
|
264
|
-
if (!match) {
|
|
265
|
-
return null;
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
const number = Number.parseFloat(match[1]);
|
|
269
|
-
if (!Number.isFinite(number) || number < 0) {
|
|
270
|
-
return null;
|
|
271
|
-
}
|
|
272
|
-
|
|
273
|
-
const unit = (match[2] || 'b').toLowerCase();
|
|
274
|
-
const multipliers = {
|
|
275
|
-
b: 1,
|
|
276
|
-
kb: 1024,
|
|
277
|
-
mb: 1024 * 1024,
|
|
278
|
-
gb: 1024 * 1024 * 1024,
|
|
279
|
-
tb: 1024 * 1024 * 1024 * 1024
|
|
280
|
-
};
|
|
281
|
-
|
|
282
|
-
return Math.round(number * (multipliers[unit] || 1));
|
|
283
|
-
}
|
|
284
|
-
|
|
285
|
-
_formatSize(bytes) {
|
|
286
|
-
if (!Number.isFinite(bytes) || bytes < 0) {
|
|
287
|
-
return '';
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
const units = ['B', 'KB', 'MB', 'GB', 'TB'];
|
|
291
|
-
let value = bytes;
|
|
292
|
-
let unitIndex = 0;
|
|
293
|
-
|
|
294
|
-
while (value >= 1024 && unitIndex < units.length - 1) {
|
|
295
|
-
value /= 1024;
|
|
296
|
-
unitIndex += 1;
|
|
297
|
-
}
|
|
298
|
-
|
|
299
|
-
const fractionDigits = value >= 100 || unitIndex === 0 ? 0 : 2;
|
|
300
|
-
return `${value.toFixed(fractionDigits)} ${units[unitIndex]}`;
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
|
|
304
|
-
export default FilePreviewHelper;
|
|
1
|
+
const DEFAULT_OPTIONS = {
|
|
2
|
+
pathAttribute: 'data-vg-filepreview',
|
|
3
|
+
fieldsAttribute: 'data-fields',
|
|
4
|
+
validAttribute: 'data-vg-filepreview-valid',
|
|
5
|
+
errorAttribute: 'data-vg-filepreview-error',
|
|
6
|
+
errorValue: 'invalid-file-path',
|
|
7
|
+
fieldAttribute: 'data-vg-filepreview-field',
|
|
8
|
+
editableAttribute: 'data-vg-filepreview-editable',
|
|
9
|
+
editableFieldsAttribute: 'data-vg-filepreview-editable-fields',
|
|
10
|
+
missingFieldsAttribute: 'data-vg-filepreview-missing-fields'
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
class FilePreviewHelper {
|
|
14
|
+
constructor(element, options = {}) {
|
|
15
|
+
this._element = element;
|
|
16
|
+
this._options = Object.assign({}, DEFAULT_OPTIONS, options);
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
getFilePath() {
|
|
20
|
+
if (!this._element) {
|
|
21
|
+
return '';
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const filePath = this._element.getAttribute(this._options.pathAttribute);
|
|
25
|
+
return typeof filePath === 'string' ? filePath.trim() : '';
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
getFields() {
|
|
29
|
+
if (!this._element) {
|
|
30
|
+
return [];
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const rawFields = this._element.getAttribute(this._options.fieldsAttribute);
|
|
34
|
+
if (typeof rawFields !== 'string' || !rawFields.trim()) {
|
|
35
|
+
return [];
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
return rawFields
|
|
39
|
+
.split(',')
|
|
40
|
+
.map(field => field.trim())
|
|
41
|
+
.filter(field => field && /^[-_a-zA-Z0-9]+$/.test(field));
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
validateFilePath(path) {
|
|
45
|
+
if (!path || path === '#' || /^javascript:/i.test(path)) {
|
|
46
|
+
return { isValid: false, fileUrl: null };
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
try {
|
|
50
|
+
const parsed = new URL(path, window.location.origin);
|
|
51
|
+
const protocol = parsed.protocol;
|
|
52
|
+
|
|
53
|
+
if (!['http:', 'https:', 'blob:', 'data:'].includes(protocol)) {
|
|
54
|
+
return { isValid: false, fileUrl: null };
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
if (protocol === 'data:' || protocol === 'blob:') {
|
|
58
|
+
return { isValid: true, fileUrl: parsed };
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
const pathname = parsed.pathname || '';
|
|
62
|
+
const fileName = pathname.split('/').filter(Boolean).pop() || '';
|
|
63
|
+
if (!fileName || !fileName.includes('.')) {
|
|
64
|
+
return { isValid: false, fileUrl: null };
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
return { isValid: true, fileUrl: parsed };
|
|
68
|
+
} catch {
|
|
69
|
+
return { isValid: false, fileUrl: null };
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
syncState(isValid) {
|
|
74
|
+
if (!this._element) {
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
this._element.setAttribute(this._options.validAttribute, isValid ? 'true' : 'false');
|
|
79
|
+
|
|
80
|
+
if (isValid) {
|
|
81
|
+
this._element.removeAttribute(this._options.errorAttribute);
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
this._element.setAttribute(this._options.errorAttribute, this._options.errorValue);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
resolveEditableFields(fields = []) {
|
|
89
|
+
if (!this._element || !Array.isArray(fields) || !fields.length) {
|
|
90
|
+
return {};
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
return fields.reduce((acc, field) => {
|
|
94
|
+
acc[field] = this._element.querySelector(`.${this._escapeClassName(field)}`);
|
|
95
|
+
return acc;
|
|
96
|
+
}, {});
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
syncEditableFields(fieldsMap = {}) {
|
|
100
|
+
if (!this._element) {
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
const previousMarkedFields = this._element.querySelectorAll(`[${this._options.fieldAttribute}]`);
|
|
105
|
+
previousMarkedFields.forEach((node) => {
|
|
106
|
+
node.removeAttribute(this._options.fieldAttribute);
|
|
107
|
+
node.setAttribute(this._options.editableAttribute, 'false');
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
const editableFields = [];
|
|
111
|
+
const missingFields = [];
|
|
112
|
+
|
|
113
|
+
Object.entries(fieldsMap).forEach(([field, node]) => {
|
|
114
|
+
if (!node) {
|
|
115
|
+
missingFields.push(field);
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
node.setAttribute(this._options.fieldAttribute, field);
|
|
120
|
+
node.setAttribute(this._options.editableAttribute, 'true');
|
|
121
|
+
editableFields.push(field);
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
if (editableFields.length) {
|
|
125
|
+
this._element.setAttribute(this._options.editableFieldsAttribute, editableFields.join(','));
|
|
126
|
+
} else {
|
|
127
|
+
this._element.removeAttribute(this._options.editableFieldsAttribute);
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
if (missingFields.length) {
|
|
131
|
+
this._element.setAttribute(this._options.missingFieldsAttribute, missingFields.join(','));
|
|
132
|
+
} else {
|
|
133
|
+
this._element.removeAttribute(this._options.missingFieldsAttribute);
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
_escapeClassName(value) {
|
|
138
|
+
if (window.CSS && typeof window.CSS.escape === 'function') {
|
|
139
|
+
return window.CSS.escape(value);
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
return String(value).replace(/([ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~])/g, '\\$1');
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
getFileMeta(path) {
|
|
146
|
+
const normalizedPath = typeof path === 'string' ? path.trim() : '';
|
|
147
|
+
const fallback = {
|
|
148
|
+
name: '',
|
|
149
|
+
ext: '',
|
|
150
|
+
originalName: '',
|
|
151
|
+
isMedia: false,
|
|
152
|
+
sizeBytes: null,
|
|
153
|
+
sizeText: ''
|
|
154
|
+
};
|
|
155
|
+
|
|
156
|
+
if (!normalizedPath) {
|
|
157
|
+
return fallback;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
try {
|
|
161
|
+
const parsedUrl = new URL(normalizedPath, window.location.origin);
|
|
162
|
+
const pathname = parsedUrl.pathname || '';
|
|
163
|
+
const params = parsedUrl.searchParams;
|
|
164
|
+
const originalName = this._readOriginalName(params);
|
|
165
|
+
let fileName = decodeURIComponent(pathname.split('/').pop() || '');
|
|
166
|
+
|
|
167
|
+
// blob/data urls often do not contain real filename; prefer explicit original name
|
|
168
|
+
if (!fileName || !fileName.includes('.')) {
|
|
169
|
+
fileName = originalName || fileName;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
const ext = fileName.includes('.') ? `.${fileName.split('.').pop() || ''}` : '';
|
|
173
|
+
const sizeBytes = this._readSize(params);
|
|
174
|
+
const mediaByExt = ['.mp3', '.wav', '.ogg', '.flac', '.aac', '.m4a', '.mp4', '.webm', '.mov', '.mkv', '.avi'];
|
|
175
|
+
|
|
176
|
+
return {
|
|
177
|
+
name: fileName,
|
|
178
|
+
ext: ext.toLowerCase(),
|
|
179
|
+
originalName,
|
|
180
|
+
isMedia: mediaByExt.includes(ext.toLowerCase()),
|
|
181
|
+
sizeBytes,
|
|
182
|
+
sizeText: this._formatSize(sizeBytes)
|
|
183
|
+
};
|
|
184
|
+
} catch {
|
|
185
|
+
const cleanPath = normalizedPath.split('#')[0].split('?')[0];
|
|
186
|
+
let fileName = decodeURIComponent(cleanPath.split('/').pop() || '');
|
|
187
|
+
const originalName = this._readOriginalName(null);
|
|
188
|
+
if (!fileName || !fileName.includes('.')) {
|
|
189
|
+
fileName = originalName || fileName;
|
|
190
|
+
}
|
|
191
|
+
const ext = fileName.includes('.') ? `.${fileName.split('.').pop() || ''}` : '';
|
|
192
|
+
const sizeBytes = this._readSize(null);
|
|
193
|
+
|
|
194
|
+
return {
|
|
195
|
+
name: fileName,
|
|
196
|
+
ext: ext.toLowerCase(),
|
|
197
|
+
originalName: originalName || '',
|
|
198
|
+
isMedia: false,
|
|
199
|
+
sizeBytes,
|
|
200
|
+
sizeText: this._formatSize(sizeBytes)
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
_readOriginalName(searchParams) {
|
|
206
|
+
const fromData = this._element?.getAttribute('data-original-name')
|
|
207
|
+
|| this._element?.getAttribute('data-vg-filepreview-original-name')
|
|
208
|
+
|| '';
|
|
209
|
+
if (fromData) {
|
|
210
|
+
return String(fromData).trim();
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
const keys = ['original_name', 'originalName', 'filename', 'name'];
|
|
214
|
+
for (const key of keys) {
|
|
215
|
+
const value = searchParams.get(key);
|
|
216
|
+
if (value) {
|
|
217
|
+
return String(value).trim();
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
return '';
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
_readSize(searchParams) {
|
|
225
|
+
const dataCandidates = [
|
|
226
|
+
this._element?.getAttribute('data-size'),
|
|
227
|
+
this._element?.getAttribute('data-vg-filepreview-size')
|
|
228
|
+
];
|
|
229
|
+
|
|
230
|
+
for (const value of dataCandidates) {
|
|
231
|
+
const parsed = this._parseSizeValue(value);
|
|
232
|
+
if (parsed !== null) {
|
|
233
|
+
return parsed;
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
if (!searchParams) {
|
|
238
|
+
return null;
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
const keys = ['size', 'size_bytes', 'bytes'];
|
|
242
|
+
for (const key of keys) {
|
|
243
|
+
const value = searchParams.get(key);
|
|
244
|
+
const parsed = this._parseSizeValue(value);
|
|
245
|
+
if (parsed !== null) {
|
|
246
|
+
return parsed;
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
return null;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
_parseSizeValue(value) {
|
|
254
|
+
if (value === null || value === undefined) {
|
|
255
|
+
return null;
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
const normalized = String(value).trim().toLowerCase();
|
|
259
|
+
if (!normalized) {
|
|
260
|
+
return null;
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
const match = normalized.match(/^(\d+(?:\.\d+)?)\s*(b|kb|mb|gb|tb)?$/i);
|
|
264
|
+
if (!match) {
|
|
265
|
+
return null;
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
const number = Number.parseFloat(match[1]);
|
|
269
|
+
if (!Number.isFinite(number) || number < 0) {
|
|
270
|
+
return null;
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
const unit = (match[2] || 'b').toLowerCase();
|
|
274
|
+
const multipliers = {
|
|
275
|
+
b: 1,
|
|
276
|
+
kb: 1024,
|
|
277
|
+
mb: 1024 * 1024,
|
|
278
|
+
gb: 1024 * 1024 * 1024,
|
|
279
|
+
tb: 1024 * 1024 * 1024 * 1024
|
|
280
|
+
};
|
|
281
|
+
|
|
282
|
+
return Math.round(number * (multipliers[unit] || 1));
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
_formatSize(bytes) {
|
|
286
|
+
if (!Number.isFinite(bytes) || bytes < 0) {
|
|
287
|
+
return '';
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
const units = ['B', 'KB', 'MB', 'GB', 'TB'];
|
|
291
|
+
let value = bytes;
|
|
292
|
+
let unitIndex = 0;
|
|
293
|
+
|
|
294
|
+
while (value >= 1024 && unitIndex < units.length - 1) {
|
|
295
|
+
value /= 1024;
|
|
296
|
+
unitIndex += 1;
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
const fractionDigits = value >= 100 || unitIndex === 0 ? 0 : 2;
|
|
300
|
+
return `${value.toFixed(fractionDigits)} ${units[unitIndex]}`;
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
export default FilePreviewHelper;
|