vgapp 1.1.3 → 1.1.5

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 (54) hide show
  1. package/CHANGELOG.md +30 -1
  2. package/README.md +48 -48
  3. package/agents.md +7 -0
  4. package/app/langs/en/buttons.json +17 -17
  5. package/app/langs/en/messages.json +36 -36
  6. package/app/langs/ru/buttons.json +17 -17
  7. package/app/langs/ru/messages.json +36 -36
  8. package/app/modules/vgfilepreview/js/i18n.js +56 -56
  9. package/app/modules/vgfilepreview/js/renderers/image-modal.js +145 -145
  10. package/app/modules/vgfilepreview/js/renderers/image.js +92 -92
  11. package/app/modules/vgfilepreview/js/renderers/index.js +19 -19
  12. package/app/modules/vgfilepreview/js/renderers/office-modal.js +168 -168
  13. package/app/modules/vgfilepreview/js/renderers/office.js +79 -79
  14. package/app/modules/vgfilepreview/js/renderers/pdf-modal.js +260 -260
  15. package/app/modules/vgfilepreview/js/renderers/pdf.js +76 -76
  16. package/app/modules/vgfilepreview/js/renderers/playlist.js +71 -71
  17. package/app/modules/vgfilepreview/js/renderers/text-modal.js +343 -343
  18. package/app/modules/vgfilepreview/js/renderers/text.js +83 -83
  19. package/app/modules/vgfilepreview/js/renderers/video-modal.js +272 -272
  20. package/app/modules/vgfilepreview/js/renderers/video.js +80 -80
  21. package/app/modules/vgfilepreview/js/renderers/zip-modal.js +522 -522
  22. package/app/modules/vgfilepreview/js/renderers/zip.js +89 -89
  23. package/app/modules/vgfilepreview/js/vgfilepreview.js +965 -530
  24. package/app/modules/vgfilepreview/readme.md +68 -68
  25. package/app/modules/vgfilepreview/scss/_variables.scss +113 -113
  26. package/app/modules/vgfilepreview/scss/vgfilepreview.scss +464 -460
  27. package/app/modules/vgfiles/js/base.js +463 -463
  28. package/app/modules/vgfiles/js/droppable.js +260 -260
  29. package/app/modules/vgfiles/js/render.js +153 -153
  30. package/app/modules/vgfiles/js/vgfiles.js +41 -41
  31. package/app/modules/vgfiles/readme.md +123 -123
  32. package/app/modules/vgfiles/scss/_variables.scss +18 -18
  33. package/app/modules/vgfiles/scss/vgfiles.scss +148 -148
  34. package/app/modules/vgformsender/js/vgformsender.js +13 -13
  35. package/app/modules/vgmodal/js/vgmodal.drag.js +332 -0
  36. package/app/modules/vgmodal/js/vgmodal.js +116 -14
  37. package/app/modules/vgmodal/js/vgmodal.resize.js +435 -0
  38. package/app/modules/vgnav/js/vgnav.js +135 -135
  39. package/app/modules/vgnav/readme.md +67 -67
  40. package/app/modules/vgnestable/README.md +307 -307
  41. package/app/modules/vgnestable/scss/_variables.scss +60 -60
  42. package/app/modules/vgnestable/scss/vgnestable.scss +163 -163
  43. package/app/modules/vgselect/js/vgselect.js +39 -39
  44. package/app/modules/vgselect/scss/vgselect.scss +22 -22
  45. package/app/modules/vgspy/readme.md +28 -28
  46. package/app/utils/js/components/audio-metadata.js +240 -240
  47. package/app/utils/js/components/file-icon.js +109 -109
  48. package/app/utils/js/components/file-preview.js +304 -304
  49. package/app/utils/js/components/sanitize.js +150 -150
  50. package/app/utils/js/components/video-metadata.js +140 -0
  51. package/build/vgapp.css +1 -1
  52. package/build/vgapp.css.map +1 -1
  53. package/index.scss +9 -9
  54. package/package.json +1 -1
@@ -1,522 +1,522 @@
1
- import VGModal from "../../../vgmodal";
2
-
3
- class ZipModal {
4
- constructor() {
5
- this._modalId = 'vg-filepreview-zip-modal';
6
- this._abortController = null;
7
- this._labels = {};
8
- this._entries = [];
9
- this._arrayBuffer = null;
10
- }
11
-
12
- static getInstance() {
13
- if (!ZipModal._instance) {
14
- ZipModal._instance = new ZipModal();
15
- }
16
-
17
- return ZipModal._instance;
18
- }
19
-
20
- open(payload = {}) {
21
- const src = String(payload.src || '').trim();
22
- if (!src) {
23
- return;
24
- }
25
-
26
- this._ensureModal();
27
- if (!this._modal || !this._content) {
28
- return;
29
- }
30
-
31
- this._labels = payload?.labels && typeof payload.labels === 'object' ? payload.labels : {};
32
- const defaultTitle = String(payload.defaultTitle || '').trim();
33
- const title = String(payload.title || '').trim();
34
- this._title.textContent = title || defaultTitle;
35
- this._content.innerHTML = `<div class="vg-filepreview-zip-modal__loading">${this._escapeHtml(this._label('loadingArchive'))}</div>`;
36
- this._modal.show();
37
-
38
- this._loadZipEntries(src);
39
- }
40
-
41
- close() {
42
- if (!this._modal) {
43
- return;
44
- }
45
-
46
- this._modal.hide();
47
- }
48
-
49
- _ensureModal() {
50
- if (this._modal && this._root) {
51
- return;
52
- }
53
-
54
- this._initModal();
55
- }
56
-
57
- _initModal() {
58
- const params = {
59
- centered: true,
60
- dismiss: true,
61
- backdrop: true,
62
- keyboard: true,
63
- sizes: {
64
- width: '800px',
65
- },
66
- animation: {
67
- enable: false
68
- }
69
- };
70
-
71
- const existed = document.getElementById(this._modalId);
72
- if (existed) {
73
- this._root = existed;
74
- this._modal = VGModal.getOrCreateInstance(existed, params);
75
- this._bindElements(existed);
76
- this._bindLifecycle(existed);
77
- return;
78
- }
79
-
80
- this._modal = VGModal.build(this._modalId, params, (modalInstance) => {
81
- const element = modalInstance._element;
82
- this._root = element;
83
- element.classList.add('vg-filepreview-zip-modal');
84
-
85
- const body = element.querySelector('.vg-modal-body');
86
- const content = element.querySelector('.vg-modal-content');
87
- if (!body || !content) {
88
- return;
89
- }
90
-
91
- body.classList.add('vg-filepreview-image-modal__body');
92
- body.innerHTML = '';
93
-
94
- let header = element.querySelector('.vg-modal-header');
95
- if (!header) {
96
- header = document.createElement('div');
97
- header.className = 'vg-modal-header';
98
- content.prepend(header);
99
- }
100
-
101
- this._title = document.createElement('div');
102
- this._title.className = 'vg-modal-title';
103
- this._title.textContent = '';
104
-
105
- this._content = document.createElement('div');
106
- this._content.className = 'vg-filepreview-zip-modal__content';
107
-
108
- header.appendChild(this._title);
109
- body.appendChild(this._content);
110
-
111
- this._bindLifecycle(element);
112
- });
113
- }
114
-
115
- _bindElements(root) {
116
- this._title = root.querySelector('.vg-filepreview-zip-modal__title');
117
- this._content = root.querySelector('.vg-filepreview-zip-modal__content');
118
- }
119
-
120
- _bindLifecycle(root) {
121
- if (!root || root.hasAttribute('data-vg-filepreview-zip-lifecycle-bind')) {
122
- return;
123
- }
124
-
125
- root.setAttribute('data-vg-filepreview-zip-lifecycle-bind', 'true');
126
- root.addEventListener('vg.modal.hidden', () => {
127
- this._destroyModal();
128
- });
129
- }
130
-
131
- _loadZipEntries(src) {
132
- if (this._abortController) {
133
- this._abortController.abort();
134
- }
135
- this._abortController = new AbortController();
136
-
137
- const cached = ZipModal._cache.get(src);
138
- if (cached?.entries && cached?.arrayBuffer) {
139
- this._entries = cached.entries;
140
- this._arrayBuffer = cached.arrayBuffer;
141
- this._renderEntries(this._entries);
142
- return;
143
- }
144
-
145
- fetch(src, {
146
- method: 'GET',
147
- signal: this._abortController.signal
148
- })
149
- .then((response) => {
150
- if (!response.ok) {
151
- throw new Error(`HTTP ${response.status}`);
152
- }
153
- return response.arrayBuffer();
154
- })
155
- .then((buffer) => {
156
- const entries = this._parseZipEntries(buffer);
157
- this._entries = entries;
158
- this._arrayBuffer = buffer;
159
- ZipModal._cache.set(src, {
160
- entries,
161
- arrayBuffer: buffer
162
- });
163
- this._renderEntries(entries);
164
- })
165
- .catch((error) => {
166
- if (error?.name === 'AbortError') {
167
- return;
168
- }
169
-
170
- if (this._content) {
171
- const errorText = this._resolveErrorText(error);
172
- this._content.innerHTML = `<div class="vg-filepreview-zip-modal__error">${this._escapeHtml(this._label('cannotOpenArchive'))}: ${this._escapeHtml(errorText)}</div>`;
173
- }
174
- });
175
- }
176
-
177
- _destroyModal() {
178
- if (this._abortController) {
179
- this._abortController.abort();
180
- this._abortController = null;
181
- }
182
-
183
- if (this._modal && typeof this._modal.dispose === 'function') {
184
- this._modal.dispose();
185
- }
186
-
187
- if (this._root && this._root.parentNode) {
188
- this._root.parentNode.removeChild(this._root);
189
- }
190
-
191
- this._root = null;
192
- this._modal = null;
193
- this._title = null;
194
- this._content = null;
195
- this._labels = null;
196
- this._entries = [];
197
- this._arrayBuffer = null;
198
- }
199
-
200
- _parseZipEntries(arrayBuffer) {
201
- const view = new DataView(arrayBuffer);
202
- const length = view.byteLength;
203
-
204
- if (length < 22) {
205
- throw new Error('invalid_zip');
206
- }
207
-
208
- const eocdOffset = this._findEndOfCentralDirectory(view);
209
- if (eocdOffset < 0) {
210
- throw new Error('central_directory_not_found');
211
- }
212
-
213
- const totalEntries = view.getUint16(eocdOffset + 10, true);
214
- const centralDirOffset = view.getUint32(eocdOffset + 16, true);
215
-
216
- const decoder = new TextDecoder('utf-8', { fatal: false });
217
- const entries = [];
218
- let ptr = centralDirOffset;
219
-
220
- for (let i = 0; i < totalEntries; i += 1) {
221
- if (ptr + 46 > length) {
222
- break;
223
- }
224
-
225
- const signature = view.getUint32(ptr, true);
226
- if (signature !== 0x02014b50) {
227
- break;
228
- }
229
-
230
- const compressedSize = view.getUint32(ptr + 20, true);
231
- const uncompressedSize = view.getUint32(ptr + 24, true);
232
- const compressionMethod = view.getUint16(ptr + 10, true);
233
- const fileNameLen = view.getUint16(ptr + 28, true);
234
- const extraLen = view.getUint16(ptr + 30, true);
235
- const commentLen = view.getUint16(ptr + 32, true);
236
- const localHeaderOffset = view.getUint32(ptr + 42, true);
237
-
238
- const nameStart = ptr + 46;
239
- const nameEnd = nameStart + fileNameLen;
240
- if (nameEnd > length) {
241
- break;
242
- }
243
-
244
- const fileNameBytes = new Uint8Array(arrayBuffer, nameStart, fileNameLen);
245
- const fileName = decoder.decode(fileNameBytes);
246
-
247
- entries.push({
248
- name: fileName,
249
- compressedSize,
250
- uncompressedSize,
251
- compressionMethod,
252
- localHeaderOffset,
253
- isDirectory: fileName.endsWith('/')
254
- });
255
-
256
- ptr = nameEnd + extraLen + commentLen;
257
- }
258
-
259
- return entries;
260
- }
261
-
262
- _findEndOfCentralDirectory(view) {
263
- const minOffset = Math.max(0, view.byteLength - 65557);
264
- for (let i = view.byteLength - 22; i >= minOffset; i -= 1) {
265
- if (view.getUint32(i, true) === 0x06054b50) {
266
- return i;
267
- }
268
- }
269
- return -1;
270
- }
271
-
272
- _renderEntries(entries) {
273
- if (!this._content) {
274
- return;
275
- }
276
-
277
- if (!entries.length) {
278
- this._content.innerHTML = `<div class="vg-filepreview-zip-modal__empty">${this._escapeHtml(this._label('archiveEmptyUnsupported'))}</div>`;
279
- return;
280
- }
281
-
282
- const rows = entries.map((entry, index) => {
283
- const type = entry.isDirectory
284
- ? this._label('typeDir')
285
- : this._label('typeFile');
286
- const compressed = entry.isDirectory ? '-' : this._formatSize(entry.compressedSize);
287
- const original = entry.isDirectory ? '-' : this._formatSize(entry.uncompressedSize);
288
- const previewAction = (!entry.isDirectory && this._canPreviewEntry(entry))
289
- ? `<button type="button" class="vg-filepreview-zip-modal__entry-btn" data-entry-index="${index}">${this._escapeHtml(this._label('previewEntry'))}</button>`
290
- : '-';
291
-
292
- return `
293
- <tr>
294
- <td>${index + 1}</td>
295
- <td class="name">${this._escapeHtml(entry.name)}</td>
296
- <td>${type}</td>
297
- <td>${compressed}</td>
298
- <td>${original}</td>
299
- <td>${previewAction}</td>
300
- </tr>`;
301
- }).join('');
302
-
303
- this._content.innerHTML = `
304
- <table class="vg-filepreview-zip-modal__table">
305
- <thead>
306
- <tr>
307
- <th>#</th>
308
- <th>${this._escapeHtml(this._label('tableName'))}</th>
309
- <th>${this._escapeHtml(this._label('tableType'))}</th>
310
- <th>${this._escapeHtml(this._label('tablePacked'))}</th>
311
- <th>${this._escapeHtml(this._label('tableSize'))}</th>
312
- <th>${this._escapeHtml(this._label('tablePreview'))}</th>
313
- </tr>
314
- </thead>
315
- <tbody>
316
- ${rows}
317
- </tbody>
318
- </table>`;
319
- this._bindEntriesPreview();
320
- }
321
-
322
- _label(key, fallback = '') {
323
- const value = this._labels?.[key];
324
- return typeof value === 'string' && value.trim() ? value : fallback;
325
- }
326
-
327
- _resolveErrorText(error) {
328
- const message = String(error?.message || '').trim();
329
- if (!message) {
330
- return this._label('unknownError');
331
- }
332
-
333
- if (message === 'invalid_zip') {
334
- return this._label('invalidZip');
335
- }
336
-
337
- if (message === 'central_directory_not_found') {
338
- return this._label('centralDirectoryNotFound');
339
- }
340
-
341
- return message;
342
- }
343
-
344
- _formatSize(bytes) {
345
- if (!Number.isFinite(bytes) || bytes < 0) {
346
- return '-';
347
- }
348
-
349
- const units = ['B', 'KB', 'MB', 'GB'];
350
- let value = bytes;
351
- let unitIndex = 0;
352
- while (value >= 1024 && unitIndex < units.length - 1) {
353
- value /= 1024;
354
- unitIndex += 1;
355
- }
356
-
357
- const digits = value >= 100 || unitIndex === 0 ? 0 : 2;
358
- return `${value.toFixed(digits)} ${units[unitIndex]}`;
359
- }
360
-
361
- _escapeHtml(value) {
362
- return String(value)
363
- .replace(/&/g, '&amp;')
364
- .replace(/</g, '&lt;')
365
- .replace(/>/g, '&gt;')
366
- .replace(/"/g, '&quot;')
367
- .replace(/'/g, '&#39;');
368
- }
369
-
370
- _bindEntriesPreview() {
371
- if (!this._content || this._content.hasAttribute('data-vg-filepreview-zip-preview-bind')) {
372
- return;
373
- }
374
-
375
- this._content.setAttribute('data-vg-filepreview-zip-preview-bind', 'true');
376
- this._content.addEventListener('click', (event) => {
377
- const button = event.target.closest('[data-entry-index]');
378
- if (!button) {
379
- return;
380
- }
381
-
382
- event.preventDefault();
383
- const index = Number(button.getAttribute('data-entry-index'));
384
- if (!Number.isFinite(index) || index < 0 || index >= this._entries.length) {
385
- return;
386
- }
387
- this._previewEntry(index);
388
- });
389
- }
390
-
391
- _canPreviewEntry(entry) {
392
- if (!entry || entry.isDirectory || !entry.name) {
393
- return false;
394
- }
395
-
396
- const ext = this._extractExtension(entry.name);
397
- const allowed = new Set([
398
- '.txt', '.md', '.json', '.xml', '.yml', '.yaml', '.csv', '.log',
399
- '.png', '.jpg', '.jpeg', '.gif', '.webp', '.svg', '.bmp'
400
- ]);
401
- return allowed.has(ext);
402
- }
403
-
404
- async _previewEntry(index) {
405
- if (!this._content) {
406
- return;
407
- }
408
-
409
- const entry = this._entries[index];
410
- if (!entry || !this._canPreviewEntry(entry)) {
411
- return;
412
- }
413
-
414
- let panel = this._content.querySelector('.vg-filepreview-zip-modal__entry-preview');
415
- if (!panel) {
416
- panel = document.createElement('div');
417
- panel.className = 'vg-filepreview-zip-modal__entry-preview';
418
- this._content.appendChild(panel);
419
- }
420
-
421
- panel.innerHTML = `<div class="vg-filepreview-zip-modal__entry-preview-loading">${this._escapeHtml(this._label('loadingEntry'))}</div>`;
422
-
423
- try {
424
- const uint8 = await this._extractEntryBytes(entry);
425
- const ext = this._extractExtension(entry.name);
426
-
427
- if (this._isImageExtension(ext)) {
428
- const blob = new Blob([uint8.buffer], { type: this._mimeByExtension(ext) });
429
- const url = URL.createObjectURL(blob);
430
- panel.innerHTML = `
431
- <div class="vg-filepreview-zip-modal__entry-preview-title">${this._escapeHtml(entry.name)}</div>
432
- <img class="vg-filepreview-zip-modal__entry-preview-image" src="${url}" alt="${this._escapeHtml(entry.name)}" />`;
433
- setTimeout(() => URL.revokeObjectURL(url), 2000);
434
- return;
435
- }
436
-
437
- const decoder = new TextDecoder('utf-8', { fatal: false });
438
- const text = decoder.decode(uint8);
439
- const safeText = this._escapeHtml(text || this._label('emptyFileInArchive'));
440
- panel.innerHTML = `
441
- <div class="vg-filepreview-zip-modal__entry-preview-title">${this._escapeHtml(entry.name)}</div>
442
- <pre class="vg-filepreview-zip-modal__entry-preview-text">${safeText}</pre>`;
443
- } catch (error) {
444
- panel.innerHTML = `<div class="vg-filepreview-zip-modal__entry-preview-error">${this._escapeHtml(this._label('cannotPreviewEntry'))}: ${this._escapeHtml(error?.message || this._label('unknownError'))}</div>`;
445
- }
446
- }
447
-
448
- async _extractEntryBytes(entry) {
449
- if (!this._arrayBuffer) {
450
- throw new Error(this._label('archiveBufferMissing'));
451
- }
452
-
453
- const view = new DataView(this._arrayBuffer);
454
- const offset = Number(entry.localHeaderOffset || 0);
455
- if (offset + 30 > view.byteLength) {
456
- throw new Error(this._label('entryCorrupted'));
457
- }
458
-
459
- const signature = view.getUint32(offset, true);
460
- if (signature !== 0x04034b50) {
461
- throw new Error(this._label('entryCorrupted'));
462
- }
463
-
464
- const nameLen = view.getUint16(offset + 26, true);
465
- const extraLen = view.getUint16(offset + 28, true);
466
- const dataStart = offset + 30 + nameLen + extraLen;
467
- const dataEnd = dataStart + Number(entry.compressedSize || 0);
468
- if (dataEnd > view.byteLength) {
469
- throw new Error(this._label('entryCorrupted'));
470
- }
471
-
472
- const compressed = new Uint8Array(this._arrayBuffer.slice(dataStart, dataEnd));
473
- const method = Number(entry.compressionMethod || 0);
474
-
475
- if (method === 0) {
476
- return compressed;
477
- }
478
-
479
- if (method === 8) {
480
- if (typeof DecompressionStream === 'undefined') {
481
- throw new Error(this._label('deflateNotSupported'));
482
- }
483
-
484
- const stream = new Blob([compressed]).stream().pipeThrough(new DecompressionStream('deflate-raw'));
485
- const decompressed = await new Response(stream).arrayBuffer();
486
- return new Uint8Array(decompressed);
487
- }
488
-
489
- throw new Error(this._label('compressionNotSupported'));
490
- }
491
-
492
- _extractExtension(name) {
493
- const value = String(name || '').toLowerCase();
494
- const clean = value.split('/').pop() || '';
495
- if (!clean.includes('.')) {
496
- return '';
497
- }
498
- return `.${clean.split('.').pop() || ''}`;
499
- }
500
-
501
- _isImageExtension(ext) {
502
- return new Set(['.png', '.jpg', '.jpeg', '.gif', '.webp', '.svg', '.bmp']).has(ext);
503
- }
504
-
505
- _mimeByExtension(ext) {
506
- const map = {
507
- '.png': 'image/png',
508
- '.jpg': 'image/jpeg',
509
- '.jpeg': 'image/jpeg',
510
- '.gif': 'image/gif',
511
- '.webp': 'image/webp',
512
- '.svg': 'image/svg+xml',
513
- '.bmp': 'image/bmp'
514
- };
515
- return map[ext] || 'application/octet-stream';
516
- }
517
-
518
- }
519
-
520
- ZipModal._cache = new Map();
521
-
522
- export default ZipModal;
1
+ import VGModal from "../../../vgmodal";
2
+
3
+ class ZipModal {
4
+ constructor() {
5
+ this._modalId = 'vg-filepreview-zip-modal';
6
+ this._abortController = null;
7
+ this._labels = {};
8
+ this._entries = [];
9
+ this._arrayBuffer = null;
10
+ }
11
+
12
+ static getInstance() {
13
+ if (!ZipModal._instance) {
14
+ ZipModal._instance = new ZipModal();
15
+ }
16
+
17
+ return ZipModal._instance;
18
+ }
19
+
20
+ open(payload = {}) {
21
+ const src = String(payload.src || '').trim();
22
+ if (!src) {
23
+ return;
24
+ }
25
+
26
+ this._ensureModal();
27
+ if (!this._modal || !this._content) {
28
+ return;
29
+ }
30
+
31
+ this._labels = payload?.labels && typeof payload.labels === 'object' ? payload.labels : {};
32
+ const defaultTitle = String(payload.defaultTitle || '').trim();
33
+ const title = String(payload.title || '').trim();
34
+ this._title.textContent = title || defaultTitle;
35
+ this._content.innerHTML = `<div class="vg-filepreview-zip-modal__loading">${this._escapeHtml(this._label('loadingArchive'))}</div>`;
36
+ this._modal.show();
37
+
38
+ this._loadZipEntries(src);
39
+ }
40
+
41
+ close() {
42
+ if (!this._modal) {
43
+ return;
44
+ }
45
+
46
+ this._modal.hide();
47
+ }
48
+
49
+ _ensureModal() {
50
+ if (this._modal && this._root) {
51
+ return;
52
+ }
53
+
54
+ this._initModal();
55
+ }
56
+
57
+ _initModal() {
58
+ const params = {
59
+ centered: true,
60
+ dismiss: true,
61
+ backdrop: true,
62
+ keyboard: true,
63
+ sizes: {
64
+ width: '800px',
65
+ },
66
+ animation: {
67
+ enable: false
68
+ }
69
+ };
70
+
71
+ const existed = document.getElementById(this._modalId);
72
+ if (existed) {
73
+ this._root = existed;
74
+ this._modal = VGModal.getOrCreateInstance(existed, params);
75
+ this._bindElements(existed);
76
+ this._bindLifecycle(existed);
77
+ return;
78
+ }
79
+
80
+ this._modal = VGModal.build(this._modalId, params, (modalInstance) => {
81
+ const element = modalInstance._element;
82
+ this._root = element;
83
+ element.classList.add('vg-filepreview-zip-modal');
84
+
85
+ const body = element.querySelector('.vg-modal-body');
86
+ const content = element.querySelector('.vg-modal-content');
87
+ if (!body || !content) {
88
+ return;
89
+ }
90
+
91
+ body.classList.add('vg-filepreview-image-modal__body');
92
+ body.innerHTML = '';
93
+
94
+ let header = element.querySelector('.vg-modal-header');
95
+ if (!header) {
96
+ header = document.createElement('div');
97
+ header.className = 'vg-modal-header';
98
+ content.prepend(header);
99
+ }
100
+
101
+ this._title = document.createElement('div');
102
+ this._title.className = 'vg-modal-title';
103
+ this._title.textContent = '';
104
+
105
+ this._content = document.createElement('div');
106
+ this._content.className = 'vg-filepreview-zip-modal__content';
107
+
108
+ header.appendChild(this._title);
109
+ body.appendChild(this._content);
110
+
111
+ this._bindLifecycle(element);
112
+ });
113
+ }
114
+
115
+ _bindElements(root) {
116
+ this._title = root.querySelector('.vg-filepreview-zip-modal__title');
117
+ this._content = root.querySelector('.vg-filepreview-zip-modal__content');
118
+ }
119
+
120
+ _bindLifecycle(root) {
121
+ if (!root || root.hasAttribute('data-vg-filepreview-zip-lifecycle-bind')) {
122
+ return;
123
+ }
124
+
125
+ root.setAttribute('data-vg-filepreview-zip-lifecycle-bind', 'true');
126
+ root.addEventListener('vg.modal.hidden', () => {
127
+ this._destroyModal();
128
+ });
129
+ }
130
+
131
+ _loadZipEntries(src) {
132
+ if (this._abortController) {
133
+ this._abortController.abort();
134
+ }
135
+ this._abortController = new AbortController();
136
+
137
+ const cached = ZipModal._cache.get(src);
138
+ if (cached?.entries && cached?.arrayBuffer) {
139
+ this._entries = cached.entries;
140
+ this._arrayBuffer = cached.arrayBuffer;
141
+ this._renderEntries(this._entries);
142
+ return;
143
+ }
144
+
145
+ fetch(src, {
146
+ method: 'GET',
147
+ signal: this._abortController.signal
148
+ })
149
+ .then((response) => {
150
+ if (!response.ok) {
151
+ throw new Error(`HTTP ${response.status}`);
152
+ }
153
+ return response.arrayBuffer();
154
+ })
155
+ .then((buffer) => {
156
+ const entries = this._parseZipEntries(buffer);
157
+ this._entries = entries;
158
+ this._arrayBuffer = buffer;
159
+ ZipModal._cache.set(src, {
160
+ entries,
161
+ arrayBuffer: buffer
162
+ });
163
+ this._renderEntries(entries);
164
+ })
165
+ .catch((error) => {
166
+ if (error?.name === 'AbortError') {
167
+ return;
168
+ }
169
+
170
+ if (this._content) {
171
+ const errorText = this._resolveErrorText(error);
172
+ this._content.innerHTML = `<div class="vg-filepreview-zip-modal__error">${this._escapeHtml(this._label('cannotOpenArchive'))}: ${this._escapeHtml(errorText)}</div>`;
173
+ }
174
+ });
175
+ }
176
+
177
+ _destroyModal() {
178
+ if (this._abortController) {
179
+ this._abortController.abort();
180
+ this._abortController = null;
181
+ }
182
+
183
+ if (this._modal && typeof this._modal.dispose === 'function') {
184
+ this._modal.dispose();
185
+ }
186
+
187
+ if (this._root && this._root.parentNode) {
188
+ this._root.parentNode.removeChild(this._root);
189
+ }
190
+
191
+ this._root = null;
192
+ this._modal = null;
193
+ this._title = null;
194
+ this._content = null;
195
+ this._labels = null;
196
+ this._entries = [];
197
+ this._arrayBuffer = null;
198
+ }
199
+
200
+ _parseZipEntries(arrayBuffer) {
201
+ const view = new DataView(arrayBuffer);
202
+ const length = view.byteLength;
203
+
204
+ if (length < 22) {
205
+ throw new Error('invalid_zip');
206
+ }
207
+
208
+ const eocdOffset = this._findEndOfCentralDirectory(view);
209
+ if (eocdOffset < 0) {
210
+ throw new Error('central_directory_not_found');
211
+ }
212
+
213
+ const totalEntries = view.getUint16(eocdOffset + 10, true);
214
+ const centralDirOffset = view.getUint32(eocdOffset + 16, true);
215
+
216
+ const decoder = new TextDecoder('utf-8', { fatal: false });
217
+ const entries = [];
218
+ let ptr = centralDirOffset;
219
+
220
+ for (let i = 0; i < totalEntries; i += 1) {
221
+ if (ptr + 46 > length) {
222
+ break;
223
+ }
224
+
225
+ const signature = view.getUint32(ptr, true);
226
+ if (signature !== 0x02014b50) {
227
+ break;
228
+ }
229
+
230
+ const compressedSize = view.getUint32(ptr + 20, true);
231
+ const uncompressedSize = view.getUint32(ptr + 24, true);
232
+ const compressionMethod = view.getUint16(ptr + 10, true);
233
+ const fileNameLen = view.getUint16(ptr + 28, true);
234
+ const extraLen = view.getUint16(ptr + 30, true);
235
+ const commentLen = view.getUint16(ptr + 32, true);
236
+ const localHeaderOffset = view.getUint32(ptr + 42, true);
237
+
238
+ const nameStart = ptr + 46;
239
+ const nameEnd = nameStart + fileNameLen;
240
+ if (nameEnd > length) {
241
+ break;
242
+ }
243
+
244
+ const fileNameBytes = new Uint8Array(arrayBuffer, nameStart, fileNameLen);
245
+ const fileName = decoder.decode(fileNameBytes);
246
+
247
+ entries.push({
248
+ name: fileName,
249
+ compressedSize,
250
+ uncompressedSize,
251
+ compressionMethod,
252
+ localHeaderOffset,
253
+ isDirectory: fileName.endsWith('/')
254
+ });
255
+
256
+ ptr = nameEnd + extraLen + commentLen;
257
+ }
258
+
259
+ return entries;
260
+ }
261
+
262
+ _findEndOfCentralDirectory(view) {
263
+ const minOffset = Math.max(0, view.byteLength - 65557);
264
+ for (let i = view.byteLength - 22; i >= minOffset; i -= 1) {
265
+ if (view.getUint32(i, true) === 0x06054b50) {
266
+ return i;
267
+ }
268
+ }
269
+ return -1;
270
+ }
271
+
272
+ _renderEntries(entries) {
273
+ if (!this._content) {
274
+ return;
275
+ }
276
+
277
+ if (!entries.length) {
278
+ this._content.innerHTML = `<div class="vg-filepreview-zip-modal__empty">${this._escapeHtml(this._label('archiveEmptyUnsupported'))}</div>`;
279
+ return;
280
+ }
281
+
282
+ const rows = entries.map((entry, index) => {
283
+ const type = entry.isDirectory
284
+ ? this._label('typeDir')
285
+ : this._label('typeFile');
286
+ const compressed = entry.isDirectory ? '-' : this._formatSize(entry.compressedSize);
287
+ const original = entry.isDirectory ? '-' : this._formatSize(entry.uncompressedSize);
288
+ const previewAction = (!entry.isDirectory && this._canPreviewEntry(entry))
289
+ ? `<button type="button" class="vg-filepreview-zip-modal__entry-btn" data-entry-index="${index}">${this._escapeHtml(this._label('previewEntry'))}</button>`
290
+ : '-';
291
+
292
+ return `
293
+ <tr>
294
+ <td>${index + 1}</td>
295
+ <td class="name">${this._escapeHtml(entry.name)}</td>
296
+ <td>${type}</td>
297
+ <td>${compressed}</td>
298
+ <td>${original}</td>
299
+ <td>${previewAction}</td>
300
+ </tr>`;
301
+ }).join('');
302
+
303
+ this._content.innerHTML = `
304
+ <table class="vg-filepreview-zip-modal__table">
305
+ <thead>
306
+ <tr>
307
+ <th>#</th>
308
+ <th>${this._escapeHtml(this._label('tableName'))}</th>
309
+ <th>${this._escapeHtml(this._label('tableType'))}</th>
310
+ <th>${this._escapeHtml(this._label('tablePacked'))}</th>
311
+ <th>${this._escapeHtml(this._label('tableSize'))}</th>
312
+ <th>${this._escapeHtml(this._label('tablePreview'))}</th>
313
+ </tr>
314
+ </thead>
315
+ <tbody>
316
+ ${rows}
317
+ </tbody>
318
+ </table>`;
319
+ this._bindEntriesPreview();
320
+ }
321
+
322
+ _label(key, fallback = '') {
323
+ const value = this._labels?.[key];
324
+ return typeof value === 'string' && value.trim() ? value : fallback;
325
+ }
326
+
327
+ _resolveErrorText(error) {
328
+ const message = String(error?.message || '').trim();
329
+ if (!message) {
330
+ return this._label('unknownError');
331
+ }
332
+
333
+ if (message === 'invalid_zip') {
334
+ return this._label('invalidZip');
335
+ }
336
+
337
+ if (message === 'central_directory_not_found') {
338
+ return this._label('centralDirectoryNotFound');
339
+ }
340
+
341
+ return message;
342
+ }
343
+
344
+ _formatSize(bytes) {
345
+ if (!Number.isFinite(bytes) || bytes < 0) {
346
+ return '-';
347
+ }
348
+
349
+ const units = ['B', 'KB', 'MB', 'GB'];
350
+ let value = bytes;
351
+ let unitIndex = 0;
352
+ while (value >= 1024 && unitIndex < units.length - 1) {
353
+ value /= 1024;
354
+ unitIndex += 1;
355
+ }
356
+
357
+ const digits = value >= 100 || unitIndex === 0 ? 0 : 2;
358
+ return `${value.toFixed(digits)} ${units[unitIndex]}`;
359
+ }
360
+
361
+ _escapeHtml(value) {
362
+ return String(value)
363
+ .replace(/&/g, '&amp;')
364
+ .replace(/</g, '&lt;')
365
+ .replace(/>/g, '&gt;')
366
+ .replace(/"/g, '&quot;')
367
+ .replace(/'/g, '&#39;');
368
+ }
369
+
370
+ _bindEntriesPreview() {
371
+ if (!this._content || this._content.hasAttribute('data-vg-filepreview-zip-preview-bind')) {
372
+ return;
373
+ }
374
+
375
+ this._content.setAttribute('data-vg-filepreview-zip-preview-bind', 'true');
376
+ this._content.addEventListener('click', (event) => {
377
+ const button = event.target.closest('[data-entry-index]');
378
+ if (!button) {
379
+ return;
380
+ }
381
+
382
+ event.preventDefault();
383
+ const index = Number(button.getAttribute('data-entry-index'));
384
+ if (!Number.isFinite(index) || index < 0 || index >= this._entries.length) {
385
+ return;
386
+ }
387
+ this._previewEntry(index);
388
+ });
389
+ }
390
+
391
+ _canPreviewEntry(entry) {
392
+ if (!entry || entry.isDirectory || !entry.name) {
393
+ return false;
394
+ }
395
+
396
+ const ext = this._extractExtension(entry.name);
397
+ const allowed = new Set([
398
+ '.txt', '.md', '.json', '.xml', '.yml', '.yaml', '.csv', '.log',
399
+ '.png', '.jpg', '.jpeg', '.gif', '.webp', '.svg', '.bmp'
400
+ ]);
401
+ return allowed.has(ext);
402
+ }
403
+
404
+ async _previewEntry(index) {
405
+ if (!this._content) {
406
+ return;
407
+ }
408
+
409
+ const entry = this._entries[index];
410
+ if (!entry || !this._canPreviewEntry(entry)) {
411
+ return;
412
+ }
413
+
414
+ let panel = this._content.querySelector('.vg-filepreview-zip-modal__entry-preview');
415
+ if (!panel) {
416
+ panel = document.createElement('div');
417
+ panel.className = 'vg-filepreview-zip-modal__entry-preview';
418
+ this._content.appendChild(panel);
419
+ }
420
+
421
+ panel.innerHTML = `<div class="vg-filepreview-zip-modal__entry-preview-loading">${this._escapeHtml(this._label('loadingEntry'))}</div>`;
422
+
423
+ try {
424
+ const uint8 = await this._extractEntryBytes(entry);
425
+ const ext = this._extractExtension(entry.name);
426
+
427
+ if (this._isImageExtension(ext)) {
428
+ const blob = new Blob([uint8.buffer], { type: this._mimeByExtension(ext) });
429
+ const url = URL.createObjectURL(blob);
430
+ panel.innerHTML = `
431
+ <div class="vg-filepreview-zip-modal__entry-preview-title">${this._escapeHtml(entry.name)}</div>
432
+ <img class="vg-filepreview-zip-modal__entry-preview-image" src="${url}" alt="${this._escapeHtml(entry.name)}" />`;
433
+ setTimeout(() => URL.revokeObjectURL(url), 2000);
434
+ return;
435
+ }
436
+
437
+ const decoder = new TextDecoder('utf-8', { fatal: false });
438
+ const text = decoder.decode(uint8);
439
+ const safeText = this._escapeHtml(text || this._label('emptyFileInArchive'));
440
+ panel.innerHTML = `
441
+ <div class="vg-filepreview-zip-modal__entry-preview-title">${this._escapeHtml(entry.name)}</div>
442
+ <pre class="vg-filepreview-zip-modal__entry-preview-text">${safeText}</pre>`;
443
+ } catch (error) {
444
+ panel.innerHTML = `<div class="vg-filepreview-zip-modal__entry-preview-error">${this._escapeHtml(this._label('cannotPreviewEntry'))}: ${this._escapeHtml(error?.message || this._label('unknownError'))}</div>`;
445
+ }
446
+ }
447
+
448
+ async _extractEntryBytes(entry) {
449
+ if (!this._arrayBuffer) {
450
+ throw new Error(this._label('archiveBufferMissing'));
451
+ }
452
+
453
+ const view = new DataView(this._arrayBuffer);
454
+ const offset = Number(entry.localHeaderOffset || 0);
455
+ if (offset + 30 > view.byteLength) {
456
+ throw new Error(this._label('entryCorrupted'));
457
+ }
458
+
459
+ const signature = view.getUint32(offset, true);
460
+ if (signature !== 0x04034b50) {
461
+ throw new Error(this._label('entryCorrupted'));
462
+ }
463
+
464
+ const nameLen = view.getUint16(offset + 26, true);
465
+ const extraLen = view.getUint16(offset + 28, true);
466
+ const dataStart = offset + 30 + nameLen + extraLen;
467
+ const dataEnd = dataStart + Number(entry.compressedSize || 0);
468
+ if (dataEnd > view.byteLength) {
469
+ throw new Error(this._label('entryCorrupted'));
470
+ }
471
+
472
+ const compressed = new Uint8Array(this._arrayBuffer.slice(dataStart, dataEnd));
473
+ const method = Number(entry.compressionMethod || 0);
474
+
475
+ if (method === 0) {
476
+ return compressed;
477
+ }
478
+
479
+ if (method === 8) {
480
+ if (typeof DecompressionStream === 'undefined') {
481
+ throw new Error(this._label('deflateNotSupported'));
482
+ }
483
+
484
+ const stream = new Blob([compressed]).stream().pipeThrough(new DecompressionStream('deflate-raw'));
485
+ const decompressed = await new Response(stream).arrayBuffer();
486
+ return new Uint8Array(decompressed);
487
+ }
488
+
489
+ throw new Error(this._label('compressionNotSupported'));
490
+ }
491
+
492
+ _extractExtension(name) {
493
+ const value = String(name || '').toLowerCase();
494
+ const clean = value.split('/').pop() || '';
495
+ if (!clean.includes('.')) {
496
+ return '';
497
+ }
498
+ return `.${clean.split('.').pop() || ''}`;
499
+ }
500
+
501
+ _isImageExtension(ext) {
502
+ return new Set(['.png', '.jpg', '.jpeg', '.gif', '.webp', '.svg', '.bmp']).has(ext);
503
+ }
504
+
505
+ _mimeByExtension(ext) {
506
+ const map = {
507
+ '.png': 'image/png',
508
+ '.jpg': 'image/jpeg',
509
+ '.jpeg': 'image/jpeg',
510
+ '.gif': 'image/gif',
511
+ '.webp': 'image/webp',
512
+ '.svg': 'image/svg+xml',
513
+ '.bmp': 'image/bmp'
514
+ };
515
+ return map[ext] || 'application/octet-stream';
516
+ }
517
+
518
+ }
519
+
520
+ ZipModal._cache = new Map();
521
+
522
+ export default ZipModal;