vgapp 1.1.2 → 1.1.4

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 (55) hide show
  1. package/CHANGELOG.md +23 -5
  2. package/README.md +19 -16
  3. package/agents.md +6 -0
  4. package/app/langs/en/buttons.json +17 -2
  5. package/app/langs/en/messages.json +36 -1
  6. package/app/langs/ru/buttons.json +17 -2
  7. package/app/langs/ru/messages.json +69 -34
  8. package/app/modules/module-fn.js +15 -9
  9. package/app/modules/vgfilepreview/index.js +3 -0
  10. package/app/modules/vgfilepreview/js/i18n.js +56 -0
  11. package/app/modules/vgfilepreview/js/renderers/image-modal.js +145 -0
  12. package/app/modules/vgfilepreview/js/renderers/image.js +92 -0
  13. package/app/modules/vgfilepreview/js/renderers/index.js +19 -0
  14. package/app/modules/vgfilepreview/js/renderers/office-modal.js +168 -0
  15. package/app/modules/vgfilepreview/js/renderers/office.js +79 -0
  16. package/app/modules/vgfilepreview/js/renderers/pdf-modal.js +260 -0
  17. package/app/modules/vgfilepreview/js/renderers/pdf.js +76 -0
  18. package/app/modules/vgfilepreview/js/renderers/playlist.js +71 -0
  19. package/app/modules/vgfilepreview/js/renderers/text-modal.js +343 -0
  20. package/app/modules/vgfilepreview/js/renderers/text.js +83 -0
  21. package/app/modules/vgfilepreview/js/renderers/video-modal.js +272 -0
  22. package/app/modules/vgfilepreview/js/renderers/video.js +80 -0
  23. package/app/modules/vgfilepreview/js/renderers/zip-modal.js +522 -0
  24. package/app/modules/vgfilepreview/js/renderers/zip.js +89 -0
  25. package/app/modules/vgfilepreview/js/vgfilepreview.js +594 -0
  26. package/app/modules/vgfilepreview/readme.md +68 -0
  27. package/app/modules/vgfilepreview/scss/_variables.scss +113 -0
  28. package/app/modules/vgfilepreview/scss/vgfilepreview.scss +460 -0
  29. package/app/modules/vgfiles/js/base.js +463 -175
  30. package/app/modules/vgfiles/js/droppable.js +260 -260
  31. package/app/modules/vgfiles/js/render.js +153 -153
  32. package/app/modules/vgfiles/js/vgfiles.js +41 -29
  33. package/app/modules/vgfiles/readme.md +116 -217
  34. package/app/modules/vgfiles/scss/_variables.scss +18 -10
  35. package/app/modules/vgfiles/scss/vgfiles.scss +153 -59
  36. package/app/modules/vgformsender/js/vgformsender.js +13 -13
  37. package/app/modules/vgmodal/js/vgmodal.js +12 -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 -0
  47. package/app/utils/js/components/file-icon.js +109 -0
  48. package/app/utils/js/components/file-preview.js +304 -0
  49. package/app/utils/js/components/sanitize.js +150 -150
  50. package/build/vgapp.css +1 -1
  51. package/build/vgapp.css.map +1 -1
  52. package/build/vgapp.js.map +1 -1
  53. package/index.js +1 -0
  54. package/index.scss +9 -6
  55. package/package.json +1 -1
@@ -0,0 +1,594 @@
1
+ import BaseModule from "../../base-module";
2
+ import {mergeDeepObject} from "../../../utils/js/functions";
3
+ import FilePreviewHelper from "../../../utils/js/components/file-preview";
4
+ import {getSVG} from "../../module-fn";
5
+ import createFilePreviewRenderers from "./renderers";
6
+ import {createFilePreviewI18n, resolveFilePreviewLang} from "./i18n";
7
+
8
+ const NAME = 'filepreview';
9
+ const NAME_KEY = 'vg.filepreview';
10
+
11
+ class VGFilePreview extends BaseModule {
12
+ constructor(el, params = {}) {
13
+ super(el, params);
14
+
15
+ this._params = this._getParams(el, mergeDeepObject({
16
+ validate: true,
17
+ lang: 'ru',
18
+ ui: {
19
+ nameOnly: false,
20
+ preview: false
21
+ }
22
+ }, params));
23
+
24
+ this._filePath = '';
25
+ this._fileUrl = null;
26
+ this._isValid = false;
27
+ this._fileMeta = {};
28
+ this._fields = [];
29
+ this._editableFields = {};
30
+ this._helper = new FilePreviewHelper(this._element);
31
+ this._renderers = createFilePreviewRenderers();
32
+ this._lang = resolveFilePreviewLang(this._params.lang, this._element);
33
+ this._i18n = createFilePreviewI18n(this._lang);
34
+ this._inlineAudio = null;
35
+ this._inlineAudioSrc = '';
36
+ this._inlineAudioButton = null;
37
+ this._inlineAudioIcon = null;
38
+ this._inlineAudioContainer = null;
39
+
40
+ this.init();
41
+ }
42
+
43
+ static get NAME() {
44
+ return NAME;
45
+ }
46
+
47
+ static get NAME_KEY() {
48
+ return NAME_KEY;
49
+ }
50
+
51
+ get filePath() {
52
+ return this._filePath;
53
+ }
54
+
55
+ get fileUrl() {
56
+ return this._fileUrl;
57
+ }
58
+
59
+ get isValid() {
60
+ return this._isValid;
61
+ }
62
+
63
+ get fileMeta() {
64
+ return this._fileMeta;
65
+ }
66
+
67
+ get fields() {
68
+ return this._fields;
69
+ }
70
+
71
+ get editableFields() {
72
+ return this._editableFields;
73
+ }
74
+
75
+ init() {
76
+ const filePath = this._helper.getFilePath();
77
+ this._filePath = filePath;
78
+
79
+ if (this._params.validate) {
80
+ const validation = this._helper.validateFilePath(filePath);
81
+ this._isValid = validation.isValid;
82
+ this._fileUrl = validation.fileUrl;
83
+ } else {
84
+ this._isValid = true;
85
+ this._fileUrl = null;
86
+ }
87
+
88
+ this._helper.syncState(this._isValid);
89
+
90
+ if (!this._isValid) {
91
+ return false;
92
+ }
93
+
94
+ this._fileMeta = this._helper.getFileMeta(this._filePath);
95
+ return this.preview();
96
+ }
97
+
98
+ preview() {
99
+ this._setState('loading');
100
+ this._fields = this._helper.getFields();
101
+ this._editableFields = this._helper.resolveEditableFields(this._fields);
102
+ this._helper.syncEditableFields(this._editableFields);
103
+ this._renderIcon();
104
+ this._renderTextFields();
105
+ this._renderDownloadField();
106
+
107
+ if (this._shouldRenderPreview()) {
108
+ this._renderPreview();
109
+ return this._editableFields;
110
+ }
111
+
112
+ const previewField = this._editableFields.preview;
113
+ if (previewField) {
114
+ previewField.innerHTML = '';
115
+ }
116
+
117
+ this._element.removeAttribute('data-vg-filepreview-renderer');
118
+ this._setState('ready');
119
+
120
+ return this._editableFields;
121
+ }
122
+
123
+ _renderIcon() {
124
+ const iconField = this._editableFields.icon;
125
+ if (!iconField) {
126
+ return;
127
+ }
128
+
129
+ const imageSrc = this._getImageIconSrc();
130
+ if (imageSrc) {
131
+ iconField.innerHTML = '';
132
+ const image = document.createElement('img');
133
+ image.src = imageSrc;
134
+ image.alt = this._fileMeta?.originalName || this._fileMeta?.name || '';
135
+ image.className = 'vg-filepreview-icon-image';
136
+ image.loading = 'lazy';
137
+ image.addEventListener('error', () => {
138
+ this._renderDefaultIcon(iconField);
139
+ });
140
+ iconField.appendChild(image);
141
+ return;
142
+ }
143
+
144
+ this._renderDefaultIcon(iconField);
145
+ }
146
+
147
+ _renderTextFields() {
148
+ const nameField = this._editableFields.name;
149
+ if (nameField) {
150
+ if (this._isAudioFile()) {
151
+ this._renderAudioNameField(nameField);
152
+ this._element.setAttribute('data-vg-filepreview-renderer', 'audio');
153
+ } else if (this._fileMeta.name) {
154
+ nameField.classList.remove('vg-filepreview-audio-inline');
155
+ nameField.textContent = this._fileMeta.name;
156
+ this._applyNameClampStyles(nameField);
157
+ }
158
+ }
159
+
160
+ const extField = this._editableFields.ext;
161
+ if (extField && this._fileMeta.ext) {
162
+ extField.textContent = this._fileMeta.ext;
163
+ }
164
+
165
+ const sizeField = this._editableFields.size;
166
+ if (sizeField && this._fileMeta.sizeText) {
167
+ sizeField.textContent = this._fileMeta.sizeText;
168
+ }
169
+
170
+ const originalNameField = this._editableFields.original_name;
171
+ if (!originalNameField) {
172
+ return;
173
+ }
174
+
175
+ if (this._fileMeta.originalName) {
176
+ originalNameField.textContent = this._fileMeta.originalName;
177
+ this._applyNameClampStyles(originalNameField);
178
+ return;
179
+ }
180
+
181
+ if (this._fileMeta.isMedia) {
182
+ originalNameField.textContent = '';
183
+ }
184
+ }
185
+
186
+ _renderAudioNameField(nameField) {
187
+ const displayName = String(
188
+ this._element?.getAttribute('data-vg-filepreview-display-name')
189
+ || this._fileMeta?.originalName
190
+ || this._fileMeta?.name
191
+ || ''
192
+ ).trim();
193
+
194
+ const fileName = displayName;
195
+ if (!fileName) {
196
+ return;
197
+ }
198
+
199
+ nameField.innerHTML = '';
200
+ nameField.classList.add('vg-filepreview-audio-inline');
201
+
202
+ const button = document.createElement('button');
203
+ button.type = 'button';
204
+ button.className = 'vg-filepreview-audio-inline__toggle';
205
+ button.setAttribute('aria-label', 'Play/Pause');
206
+
207
+ const icon = document.createElement('span');
208
+ icon.className = 'vg-filepreview-audio-inline__icon';
209
+ button.appendChild(icon);
210
+
211
+ const text = document.createElement('span');
212
+ text.className = 'vg-filepreview-audio-inline__name';
213
+ text.textContent = fileName;
214
+ this._applyNameClampStyles(text);
215
+
216
+ button.addEventListener('click', (event) => {
217
+ event.preventDefault();
218
+ event.stopPropagation();
219
+ this._toggleInlineAudio();
220
+ });
221
+
222
+ nameField.appendChild(button);
223
+ nameField.appendChild(text);
224
+
225
+ this._inlineAudioButton = button;
226
+ this._inlineAudioIcon = icon;
227
+ const rootFile = this._element?.classList?.contains('file')
228
+ ? this._element
229
+ : this._element?.closest?.('.file');
230
+ this._inlineAudioContainer = rootFile || this._element || nameField;
231
+ this._setInlineAudioProgress(0);
232
+ const isCurrentAudioPlaying = VGFilePreview._activeAudioOwner === this && this._inlineAudio && !this._inlineAudio.paused;
233
+ this._syncInlineAudioIcon(isCurrentAudioPlaying);
234
+ }
235
+
236
+ _toggleInlineAudio() {
237
+ const src = this._fileUrl?.href || this._filePath || '';
238
+ if (!src || !this._inlineAudioButton) {
239
+ return;
240
+ }
241
+
242
+ if (VGFilePreview._activeAudioOwner && VGFilePreview._activeAudioOwner !== this) {
243
+ VGFilePreview._activeAudioOwner._stopInlineAudio();
244
+ }
245
+
246
+ if (!this._inlineAudio || this._inlineAudioSrc !== src) {
247
+ this._stopInlineAudio();
248
+ this._inlineAudio = new Audio(src);
249
+ this._inlineAudioSrc = src;
250
+ this._inlineAudio.addEventListener('ended', () => {
251
+ this._syncInlineAudioIcon(false);
252
+ this._setInlineAudioProgress(0);
253
+ });
254
+ this._inlineAudio.addEventListener('timeupdate', () => this._syncInlineAudioProgress());
255
+ this._inlineAudio.addEventListener('loadedmetadata', () => this._syncInlineAudioProgress());
256
+ }
257
+
258
+ if (this._inlineAudio.paused) {
259
+ this._inlineAudio.play().then(() => {
260
+ VGFilePreview._activeAudioOwner = this;
261
+ this._syncInlineAudioIcon(true);
262
+ }).catch(() => {
263
+ this._syncInlineAudioIcon(false);
264
+ });
265
+ return;
266
+ }
267
+
268
+ this._inlineAudio.pause();
269
+ this._syncInlineAudioIcon(false);
270
+ if (VGFilePreview._activeAudioOwner === this) {
271
+ VGFilePreview._activeAudioOwner = null;
272
+ }
273
+ }
274
+
275
+ _stopInlineAudio() {
276
+ if (!this._inlineAudio) {
277
+ this._syncInlineAudioIcon(false);
278
+ this._setInlineAudioProgress(0);
279
+ return;
280
+ }
281
+
282
+ this._inlineAudio.pause();
283
+ this._inlineAudio.currentTime = 0;
284
+ this._syncInlineAudioIcon(false);
285
+ this._setInlineAudioProgress(0);
286
+ if (VGFilePreview._activeAudioOwner === this) {
287
+ VGFilePreview._activeAudioOwner = null;
288
+ }
289
+ }
290
+
291
+ _syncInlineAudioIcon(isPlaying) {
292
+ if (!this._inlineAudioIcon) {
293
+ return;
294
+ }
295
+
296
+ this._inlineAudioIcon.innerHTML = isPlaying ? (getSVG('pause') || '') : (getSVG('play') || '');
297
+ }
298
+
299
+ _syncInlineAudioProgress() {
300
+ if (!this._inlineAudio) {
301
+ this._setInlineAudioProgress(0);
302
+ return;
303
+ }
304
+
305
+ const duration = Number(this._inlineAudio.duration || 0);
306
+ const currentTime = Number(this._inlineAudio.currentTime || 0);
307
+ if (!duration || !Number.isFinite(duration) || duration <= 0) {
308
+ this._setInlineAudioProgress(0);
309
+ return;
310
+ }
311
+
312
+ const progress = Math.max(0, Math.min(100, (currentTime / duration) * 100));
313
+ this._setInlineAudioProgress(progress);
314
+ }
315
+
316
+ _setInlineAudioProgress(percent) {
317
+ if (!this._inlineAudioContainer) {
318
+ return;
319
+ }
320
+
321
+ const normalized = Math.max(0, Math.min(100, Number(percent) || 0));
322
+ this._inlineAudioContainer.style.setProperty('--vg-filepreview-audio-inline-progress', `${normalized}%`);
323
+ }
324
+
325
+ _renderDownloadField() {
326
+ const downloadField = this._editableFields.download;
327
+ if (!downloadField) {
328
+ return;
329
+ }
330
+
331
+ const downloadLabel = this._i18n?.button('download') || '';
332
+ const downloadIcon = getSVG('download') || '';
333
+ const fieldTag = String(downloadField.tagName || '').toUpperCase();
334
+ let control = null;
335
+
336
+ if (fieldTag === 'A' || fieldTag === 'BUTTON') {
337
+ control = downloadField;
338
+ } else {
339
+ control = downloadField.querySelector('[data-vg-filepreview-download-control]');
340
+ if (!control) {
341
+ control = document.createElement('button');
342
+ control.type = 'button';
343
+ downloadField.innerHTML = '';
344
+ downloadField.appendChild(control);
345
+ }
346
+ }
347
+
348
+ control.classList.add('vg-filepreview-download-trigger');
349
+ control.setAttribute('data-vg-filepreview-download-control', 'true');
350
+
351
+ if (fieldTag === 'A' && control === downloadField) {
352
+ control.setAttribute('href', this._fileUrl?.href || this._filePath || '#');
353
+ if (this._fileMeta?.name) {
354
+ control.setAttribute('download', this._fileMeta.name);
355
+ }
356
+ } else if (String(control.tagName || '').toUpperCase() === 'BUTTON') {
357
+ control.setAttribute('type', 'button');
358
+ }
359
+
360
+ if (!control.hasAttribute('data-vg-filepreview-download-content-init')) {
361
+ control.setAttribute('data-vg-filepreview-download-content-init', 'true');
362
+ control.innerHTML = '';
363
+
364
+ if (downloadIcon) {
365
+ const icon = document.createElement('span');
366
+ icon.className = 'vg-filepreview-download-trigger__icon';
367
+ icon.innerHTML = downloadIcon;
368
+ control.appendChild(icon);
369
+ }
370
+
371
+ if (downloadLabel) {
372
+ const text = document.createElement('span');
373
+ text.className = 'vg-filepreview-download-trigger__text';
374
+ text.textContent = downloadLabel;
375
+ control.appendChild(text);
376
+ }
377
+ }
378
+
379
+ if (!control.hasAttribute('data-vg-filepreview-download-bind')) {
380
+ control.setAttribute('data-vg-filepreview-download-bind', 'true');
381
+ control.addEventListener('click', (event) => {
382
+ event.preventDefault();
383
+ this._downloadFile();
384
+ });
385
+ }
386
+ }
387
+
388
+ _downloadFile() {
389
+ const src = this._fileUrl?.href || this._filePath || '';
390
+ if (!src) {
391
+ return;
392
+ }
393
+
394
+ const fileName = this._fileMeta?.originalName || this._fileMeta?.name || 'file';
395
+
396
+ fetch(src, {
397
+ method: 'GET',
398
+ credentials: 'same-origin'
399
+ })
400
+ .then((response) => {
401
+ if (!response.ok) {
402
+ throw new Error(`HTTP ${response.status}`);
403
+ }
404
+ return response.blob();
405
+ })
406
+ .then((blob) => {
407
+ const objectUrl = URL.createObjectURL(blob);
408
+ this._downloadByLink(objectUrl, fileName);
409
+ setTimeout(() => URL.revokeObjectURL(objectUrl), 1000);
410
+ })
411
+ .catch(() => {
412
+ this._downloadByLink(src, fileName);
413
+ });
414
+ }
415
+
416
+ _downloadByLink(href, fileName) {
417
+ const link = document.createElement('a');
418
+ link.href = href;
419
+ link.style.display = 'none';
420
+ link.rel = 'noopener noreferrer';
421
+ if (fileName) {
422
+ link.setAttribute('download', fileName);
423
+ }
424
+
425
+ document.body.appendChild(link);
426
+ link.click();
427
+ document.body.removeChild(link);
428
+ }
429
+
430
+ _renderPreview() {
431
+ const isNameOnly = Boolean(this._params?.ui?.nameOnly);
432
+ const previewContainer = this._resolvePreviewContainer({
433
+ autoCreate: !isNameOnly
434
+ });
435
+ if (!previewContainer && !isNameOnly) {
436
+ this._setState('error');
437
+ return;
438
+ }
439
+
440
+ if (previewContainer) {
441
+ previewContainer.innerHTML = '';
442
+ }
443
+
444
+ const context = {
445
+ element: this._element,
446
+ filePath: this._filePath,
447
+ fileUrl: this._fileUrl,
448
+ fileMeta: this._fileMeta,
449
+ previewContainer,
450
+ lang: this._lang,
451
+ i18n: this._i18n,
452
+ ui: this._params?.ui || {}
453
+ };
454
+
455
+ let rendered = false;
456
+ this._renderers.forEach((renderer) => {
457
+ if (rendered || typeof renderer?.canRender !== 'function' || typeof renderer?.render !== 'function') {
458
+ return;
459
+ }
460
+
461
+ if (!renderer.canRender(context)) {
462
+ return;
463
+ }
464
+
465
+ try {
466
+ rendered = renderer.render(context) === true;
467
+ } catch (error) {
468
+ rendered = false;
469
+ }
470
+ if (rendered) {
471
+ this._element.setAttribute('data-vg-filepreview-renderer', renderer.name || 'custom');
472
+ }
473
+ });
474
+
475
+ if (!rendered && !isNameOnly) {
476
+ this._element.removeAttribute('data-vg-filepreview-renderer');
477
+ this._setState('empty');
478
+ return;
479
+ }
480
+ this._setState('ready');
481
+ }
482
+
483
+ _resolvePreviewContainer(params = {}) {
484
+ const autoCreate = !Object.prototype.hasOwnProperty.call(params, 'autoCreate') || Boolean(params.autoCreate);
485
+ const editablePreview = this._editableFields.preview;
486
+ if (editablePreview) {
487
+ editablePreview.setAttribute('data-vg-filepreview-slot', 'preview');
488
+ return editablePreview;
489
+ }
490
+
491
+ const existedPreview = this._element.querySelector('[data-vg-filepreview-slot="preview"]');
492
+ if (existedPreview) {
493
+ this._editableFields.preview = existedPreview;
494
+ return existedPreview;
495
+ }
496
+
497
+ if (!autoCreate) {
498
+ return null;
499
+ }
500
+
501
+ const container = document.createElement('div');
502
+ container.className = 'preview';
503
+ container.setAttribute('data-vg-filepreview-slot', 'preview');
504
+ this._element.appendChild(container);
505
+ this._editableFields.preview = container;
506
+
507
+ return container;
508
+ }
509
+
510
+ _shouldRenderPreview() {
511
+ return Boolean(this._params?.ui?.preview);
512
+ }
513
+
514
+ _getImageIconSrc() {
515
+ const ext = String(this._fileMeta?.ext || '').toLowerCase();
516
+ const imageExts = ['.jpg', '.jpeg', '.png', '.gif', '.webp', '.bmp', '.svg', '.avif', '.ico'];
517
+ if (!imageExts.includes(ext)) {
518
+ return '';
519
+ }
520
+
521
+ return this._fileUrl?.href || this._filePath || '';
522
+ }
523
+
524
+ _renderDefaultIcon(iconField) {
525
+ const icon = getSVG(this._filePath);
526
+ if (!icon) {
527
+ iconField.innerHTML = '';
528
+ return;
529
+ }
530
+
531
+ iconField.innerHTML = icon;
532
+ }
533
+
534
+ _applyNameClampStyles(field) {
535
+ if (!field || !field.style) {
536
+ return;
537
+ }
538
+
539
+ field.style.minWidth = '60px';
540
+ field.style.maxWidth = '100%';
541
+ field.style.overflow = 'hidden';
542
+ field.style.textOverflow = 'ellipsis';
543
+ field.style.whiteSpace = 'nowrap';
544
+ }
545
+
546
+ static init(element, params = {}) {
547
+ return VGFilePreview.getOrCreateInstance(element, params);
548
+ }
549
+
550
+ static stopActiveInlineAudio() {
551
+ const owner = VGFilePreview._activeAudioOwner;
552
+ if (owner && typeof owner._stopInlineAudio === 'function') {
553
+ owner._stopInlineAudio();
554
+ }
555
+
556
+ VGFilePreview._activeAudioOwner = null;
557
+ }
558
+
559
+ static stopActiveInlineAudioIfDetached(nodes = []) {
560
+ const owner = VGFilePreview._activeAudioOwner;
561
+ if (!owner || !owner._element || !Array.isArray(nodes) || !nodes.length) {
562
+ return;
563
+ }
564
+
565
+ const shouldStop = nodes.some((node) => {
566
+ if (!node || typeof node.contains !== 'function') {
567
+ return false;
568
+ }
569
+
570
+ return node === owner._element || node.contains(owner._element);
571
+ });
572
+
573
+ if (shouldStop) {
574
+ owner._stopInlineAudio();
575
+ VGFilePreview._activeAudioOwner = null;
576
+ }
577
+ }
578
+
579
+ _setState(state = '') {
580
+ const value = String(state || '').trim();
581
+ if (!value) {
582
+ this._element.removeAttribute('data-vg-filepreview-state');
583
+ return;
584
+ }
585
+ this._element.setAttribute('data-vg-filepreview-state', value);
586
+ }
587
+
588
+ _isAudioFile() {
589
+ const ext = String(this._fileMeta?.ext || '').toLowerCase();
590
+ return ['.mp3', '.wav', '.ogg', '.flac', '.aac', '.m4a', '.opus', '.wma'].includes(ext);
591
+ }
592
+ }
593
+
594
+ export default VGFilePreview;
@@ -0,0 +1,68 @@
1
+ ## VGFilePreview
2
+
3
+ Модуль предпросмотра файлов для элементов с атрибутом `data-vg-filepreview`.
4
+
5
+ ### Новые фичи
6
+
7
+ - Inline-аудио в поле `.name`: play/pause, прогресс через CSS-переменную `--vg-filepreview-audio-inline-progress`, и контроль единственного активного аудио.
8
+ - Унифицированная кнопка скачивания: модуль сам создает/инициализирует control в поле `download`, скачивает через `fetch + blob` и имеет fallback на прямую ссылку.
9
+ - Режим `ui.nameOnly`: рендер только действий по имени файла (без кнопок/контейнера предпросмотра).
10
+ - Автоопределение языка (`ru`/`en`) с приоритетом: `params.lang` -> `element[lang]` -> ближайший `[lang]` -> `<html lang>` -> `navigator.language`.
11
+ - Видео-плейлист между соседними превью: `prev/next`, циклическая навигация и hotkeys `ArrowLeft/ArrowRight`.
12
+ - ZIP-предпросмотр с кэшем: таблица содержимого архива + предпросмотр поддерживаемых файлов внутри архива (текст/изображения), включая deflate-raw через `DecompressionStream`.
13
+ - Текстовый/Markdown modal с кэшем, безопасной обработкой ссылок и прерыванием прошлых запросов (`AbortController`).
14
+
15
+ ### Что умеет модуль
16
+
17
+ - Определяет тип файла и подставляет SVG-иконку.
18
+ - Заполняет поля карточки: `name`, `ext`, `size`, `original_name`, `icon`, `download`, `preview`.
19
+ - Запускает предпросмотр по типам файлов:
20
+ - изображения (`png/jpg/webp/svg/...`);
21
+ - видео (`mp4/webm/mov/mkv/avi/m4v`);
22
+ - текст (`txt/md/csv/json/xml/yml/yaml/log/ini/conf/env`);
23
+ - `pdf`;
24
+ - office (`doc/docx/xls/xlsx/ppt/pptx/odt/ods/odp`);
25
+ - архивы (`zip`).
26
+ - Для интерактивных рендеров автоматически вешает обработчики клика на `.name` (`is-preview-action`).
27
+
28
+ ### Состояния и валидация
29
+
30
+ - Валидация пути через helper (`data-vg-filepreview-valid`, `data-vg-filepreview-error`).
31
+ - Текущее состояние рендера в `data-vg-filepreview-state`:
32
+ - `loading`
33
+ - `ready`
34
+ - `empty`
35
+ - `error`
36
+ - Выбранный рендерер в `data-vg-filepreview-renderer` (`image`, `video`, `pdf`, `office`, `zip`, `text`, `audio`).
37
+
38
+ ### Атрибуты и слоты
39
+
40
+ - `data-vg-filepreview` — путь к файлу (обязательный).
41
+ - `data-fields` — список полей для синхронизации (например, `name,size,download`).
42
+ - Поддерживаемые поля: `icon`, `name`, `ext`, `size`, `original_name`, `preview`, `download`.
43
+ - Если поле `preview` не найдено, модуль создает `<div class="preview" data-vg-filepreview-slot="preview">` автоматически (кроме `ui.nameOnly`).
44
+
45
+ ### Параметры
46
+
47
+ ```js
48
+ new VGFilePreview(element, {
49
+ validate: true,
50
+ lang: 'ru',
51
+ ui: {
52
+ nameOnly: false
53
+ }
54
+ });
55
+ ```
56
+
57
+ - `validate` (`boolean`, default `true`) — проверка пути до файла.
58
+ - `lang` (`'ru' | 'en'`) — язык кнопок/сообщений.
59
+ - `ui.nameOnly` (`boolean`, default `false`) — не создавать UI-контейнер предпросмотра.
60
+
61
+ ### Расширение
62
+
63
+ Рендереры подключаются через `js/renderers/index.js`.
64
+
65
+ 1. Создайте рендерер в `js/renderers/`.
66
+ 2. Реализуйте `canRender(context)` и `render(context)`.
67
+ 3. Добавьте рендерер в `js/renderers/index.js` в нужном порядке приоритета.
68
+