@file-viewer/core 2.0.10 → 2.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 (114) hide show
  1. package/README.en.md +2 -2
  2. package/README.md +2 -2
  3. package/dist/config/options.d.ts +1 -1
  4. package/dist/config/options.js +1 -1
  5. package/dist/contracts/types.d.ts +71 -1
  6. package/dist/headless.d.ts +2 -2
  7. package/dist/headless.js +1 -1
  8. package/dist/index.d.ts +9 -6
  9. package/dist/index.js +105 -48
  10. package/dist/platform/assets.d.ts +3 -1
  11. package/dist/platform/assets.js +18 -2
  12. package/dist/registry/capabilities.d.ts +2 -2
  13. package/dist/registry/capabilities.js +2 -1
  14. package/dist/registry/formats.d.ts +20 -7
  15. package/dist/registry/formats.js +14 -5
  16. package/dist/registry/registry.d.ts +8 -1
  17. package/dist/registry/registry.js +29 -0
  18. package/dist/renderers/image.js +1 -10
  19. package/dist/renderers/index.d.ts +320 -2
  20. package/dist/renderers/index.js +27 -157
  21. package/dist/viewer/createViewer.js +86 -3
  22. package/package.json +17 -44
  23. package/dist/renderers/archive.d.ts +0 -2
  24. package/dist/renderers/archive.js +0 -547
  25. package/dist/renderers/archiveCache.d.ts +0 -10
  26. package/dist/renderers/archiveCache.js +0 -96
  27. package/dist/renderers/archiveFallback.d.ts +0 -7
  28. package/dist/renderers/archiveFallback.js +0 -166
  29. package/dist/renderers/archiveShared.d.ts +0 -23
  30. package/dist/renderers/archiveShared.js +0 -71
  31. package/dist/renderers/audio.d.ts +0 -8
  32. package/dist/renderers/audio.js +0 -219
  33. package/dist/renderers/cad.d.ts +0 -2
  34. package/dist/renderers/cad.js +0 -446
  35. package/dist/renderers/code.d.ts +0 -11
  36. package/dist/renderers/code.js +0 -233
  37. package/dist/renderers/data.d.ts +0 -7
  38. package/dist/renderers/data.js +0 -370
  39. package/dist/renderers/drawing.d.ts +0 -10
  40. package/dist/renderers/drawing.js +0 -882
  41. package/dist/renderers/eda.d.ts +0 -2
  42. package/dist/renderers/eda.js +0 -434
  43. package/dist/renderers/edaParser.d.ts +0 -77
  44. package/dist/renderers/edaParser.js +0 -569
  45. package/dist/renderers/email.d.ts +0 -2
  46. package/dist/renderers/email.js +0 -463
  47. package/dist/renderers/epub.d.ts +0 -2
  48. package/dist/renderers/epub.js +0 -331
  49. package/dist/renderers/geo.d.ts +0 -2
  50. package/dist/renderers/geo.js +0 -284
  51. package/dist/renderers/markdown.d.ts +0 -2
  52. package/dist/renderers/markdown.js +0 -83
  53. package/dist/renderers/model.d.ts +0 -2
  54. package/dist/renderers/model.js +0 -567
  55. package/dist/renderers/ofd.d.ts +0 -2
  56. package/dist/renderers/ofd.js +0 -256
  57. package/dist/renderers/openDocument.d.ts +0 -2
  58. package/dist/renderers/openDocument.js +0 -122
  59. package/dist/renderers/pdf.d.ts +0 -3
  60. package/dist/renderers/pdf.js +0 -1001
  61. package/dist/renderers/pdfStyles.d.ts +0 -1
  62. package/dist/renderers/pdfStyles.js +0 -1
  63. package/dist/renderers/pptx.d.ts +0 -2
  64. package/dist/renderers/pptx.js +0 -217
  65. package/dist/renderers/spreadsheet/state.d.ts +0 -80
  66. package/dist/renderers/spreadsheet/state.js +0 -96
  67. package/dist/renderers/spreadsheet/view.d.ts +0 -25
  68. package/dist/renderers/spreadsheet/view.js +0 -833
  69. package/dist/renderers/spreadsheet/worker/index.d.ts +0 -2
  70. package/dist/renderers/spreadsheet/worker/index.js +0 -1
  71. package/dist/renderers/spreadsheet/worker/sheetjs/SheetJsModel.d.ts +0 -73
  72. package/dist/renderers/spreadsheet/worker/sheetjs/SheetJsModel.js +0 -623
  73. package/dist/renderers/spreadsheet/worker/sheetjs/color.d.ts +0 -2
  74. package/dist/renderers/spreadsheet/worker/sheetjs/color.js +0 -73
  75. package/dist/renderers/spreadsheet/worker/sheetjs/index.d.ts +0 -1
  76. package/dist/renderers/spreadsheet/worker/sheetjs/index.js +0 -1
  77. package/dist/renderers/spreadsheet/worker/sheetjs/parser.d.ts +0 -18
  78. package/dist/renderers/spreadsheet/worker/sheetjs/parser.js +0 -106
  79. package/dist/renderers/spreadsheet/worker/sheetjs/sheet.worker.d.ts +0 -1
  80. package/dist/renderers/spreadsheet/worker/sheetjs/sheet.worker.js +0 -11
  81. package/dist/renderers/spreadsheet/worker/type.d.ts +0 -57
  82. package/dist/renderers/spreadsheet/worker/type.js +0 -1
  83. package/dist/renderers/spreadsheet.d.ts +0 -3
  84. package/dist/renderers/spreadsheet.js +0 -929
  85. package/dist/renderers/typst.d.ts +0 -8
  86. package/dist/renderers/typst.js +0 -547
  87. package/dist/renderers/umd/parser.d.ts +0 -30
  88. package/dist/renderers/umd/parser.js +0 -408
  89. package/dist/renderers/umd.d.ts +0 -2
  90. package/dist/renderers/umd.js +0 -297
  91. package/dist/renderers/video.d.ts +0 -8
  92. package/dist/renderers/video.js +0 -108
  93. package/dist/renderers/wordDoc.d.ts +0 -5
  94. package/dist/renderers/wordDoc.js +0 -284
  95. package/dist/renderers/wordDocx.d.ts +0 -5
  96. package/dist/renderers/wordDocx.js +0 -985
  97. package/dist/renderers/wordDocx.worker.d.ts +0 -1
  98. package/dist/renderers/wordDocx.worker.js +0 -96
  99. package/vendor/ofd/dltech/jbig2/arithmetic_decoder.js +0 -183
  100. package/vendor/ofd/dltech/jbig2/ccitt.js +0 -1070
  101. package/vendor/ofd/dltech/jbig2/compatibility.js +0 -12
  102. package/vendor/ofd/dltech/jbig2/core_utils.js +0 -180
  103. package/vendor/ofd/dltech/jbig2/is_node.js +0 -27
  104. package/vendor/ofd/dltech/jbig2/jbig2.js +0 -2589
  105. package/vendor/ofd/dltech/jbig2/jbig2_stream.js +0 -81
  106. package/vendor/ofd/dltech/jbig2/primitives.js +0 -387
  107. package/vendor/ofd/dltech/jbig2/stream.js +0 -1348
  108. package/vendor/ofd/dltech/jbig2/util.js +0 -972
  109. package/vendor/ofd/dltech/ofd/ofd.d.ts +0 -11
  110. package/vendor/ofd/dltech/ofd/ofd.js +0 -100
  111. package/vendor/ofd/dltech/ofd/ofd_parser.js +0 -395
  112. package/vendor/ofd/dltech/ofd/ofd_render.js +0 -473
  113. package/vendor/ofd/dltech/ofd/ofd_util.js +0 -350
  114. package/vendor/ofd/dltech/ofd/pipeline.js +0 -26
@@ -1,463 +0,0 @@
1
- import { disposeFileViewerRendered } from '../rendering/handler.js';
2
- const emailStyle = `
3
- .email-viewer{position:relative;height:100%;min-height:0;display:flex;flex-direction:column;background:#f3f6f8;color:#172033;box-sizing:border-box}
4
- .email-viewer *{box-sizing:border-box}
5
- .email-header{padding:18px 22px;border-bottom:1px solid rgba(23,32,51,.08);background:#fff}
6
- .email-header>span{color:#1f7a58;font-size:12px;font-weight:900}
7
- .email-header h2{margin:4px 0 12px;font-size:22px;line-height:1.25}
8
- .email-meta{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px 18px}
9
- .email-meta p{margin:0;color:#526275;font-size:13px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
10
- .email-meta strong{margin-right:8px;color:#172033}
11
- .email-body{flex:1;min-height:0;display:grid;grid-template-columns:minmax(240px,300px) minmax(0,1fr)}
12
- .email-sidebar{min-height:0;display:flex;flex-direction:column;gap:14px;padding:14px;border-right:1px solid rgba(23,32,51,.08);background:rgba(255,255,255,.7)}
13
- .body-tabs{display:grid;grid-template-columns:repeat(3,minmax(0,1fr));gap:6px;padding:4px;border-radius:12px;background:rgba(23,32,51,.06)}
14
- .body-tabs button,.attachment-item,.attachment-preview-head button{font:inherit;cursor:pointer}
15
- .body-tabs button{height:34px;border:0;border-radius:9px;background:transparent;color:#64748b;font-size:12px;font-weight:800}
16
- .body-tabs button.active{background:#fff;color:#172033}
17
- .body-tabs button:disabled{opacity:.4;cursor:not-allowed}
18
- .attachment-panel{min-height:0;overflow:auto}
19
- .attachment-title{display:flex;justify-content:space-between;color:#172033;font-size:14px;margin-bottom:8px}
20
- .attachment-title span{color:#64748b}
21
- .attachment-empty{margin:8px 0 0;color:#64748b;font-size:12px}
22
- .attachment-item{width:100%;min-height:62px;display:grid;grid-template-columns:42px minmax(0,1fr);gap:10px;align-items:center;margin-bottom:8px;padding:9px;border:1px solid rgba(23,32,51,.08);border-radius:12px;background:#fff;text-align:left}
23
- .attachment-item:hover,.attachment-item.active{border-color:rgba(31,122,88,.28);box-shadow:0 10px 22px rgba(23,32,51,.08)}
24
- .attachment-item span{grid-row:span 2;height:38px;display:inline-flex;align-items:center;justify-content:center;border-radius:10px;background:rgba(31,122,88,.12);color:#1f7a58;font-size:11px;font-weight:900}
25
- .attachment-item strong,.attachment-item em{min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
26
- .attachment-item em{color:#64748b;font-size:12px;font-style:normal}
27
- .message-panel{min-width:0;min-height:0;display:grid;grid-template-rows:minmax(240px,46%) minmax(0,1fr)}
28
- .email-message-content{min-height:0}
29
- .email-html,.email-text{width:100%;height:100%;border:0;background:#fff}
30
- .email-text{margin:0;overflow:auto;padding:20px;white-space:pre-wrap;word-break:break-word;line-height:1.65}
31
- .attachment-preview{min-height:0;display:flex;flex-direction:column;border-top:1px solid rgba(23,32,51,.08)}
32
- .attachment-preview[hidden]{display:none}
33
- .attachment-preview-head{min-height:48px;display:flex;align-items:center;justify-content:space-between;gap:12px;padding:8px 14px;background:rgba(255,255,255,.78)}
34
- .attachment-preview-head button{height:32px;padding:0 12px;border:0;border-radius:9px;background:#1f7a58;color:#fff;font-weight:800}
35
- .attachment-target{flex:1;min-height:0;overflow:auto}
36
- .email-attachment-render{width:100%;height:100%;min-height:320px}
37
- .email-state{position:absolute;inset:0;display:flex;align-items:center;justify-content:center;gap:14px;background:rgba(243,246,248,.86);z-index:2}
38
- .email-state span{width:34px;height:34px;border-radius:999px;border:3px solid rgba(31,122,88,.16);border-top-color:#1f7a58;animation:email-spin .9s linear infinite}
39
- .email-error{position:absolute;right:18px;bottom:18px;width:min(460px,calc(100% - 36px));padding:14px;border-radius:14px;background:#fff7e8;color:#8a4b00;box-shadow:0 16px 36px rgba(23,32,51,.14);z-index:3}
40
- .email-error p{margin:6px 0 0}
41
- @keyframes email-spin{to{transform:rotate(360deg)}}
42
- @media (prefers-color-scheme:dark){.file-viewer[data-viewer-theme='system'] .email-viewer{background:#172033;color:#e5eef8}.file-viewer[data-viewer-theme='system'] .email-header,.file-viewer[data-viewer-theme='system'] .attachment-item,.file-viewer[data-viewer-theme='system'] .email-html,.file-viewer[data-viewer-theme='system'] .email-text{background:#fff;color:#172033}}
43
- .file-viewer[data-viewer-theme='dark'] .email-viewer{background:#172033;color:#e5eef8}
44
- .file-viewer[data-viewer-theme='dark'] .email-header,.file-viewer[data-viewer-theme='dark'] .attachment-item,.file-viewer[data-viewer-theme='dark'] .email-html,.file-viewer[data-viewer-theme='dark'] .email-text{background:#fff;color:#172033}
45
- @media (max-width:860px){.email-meta,.email-body{grid-template-columns:1fr}.email-body{grid-template-rows:auto minmax(0,1fr)}.email-sidebar{border-right:0;border-bottom:1px solid rgba(23,32,51,.08)}}
46
- `;
47
- const formatBytes = (value) => {
48
- if (!Number.isFinite(value) || value < 0) {
49
- return '-';
50
- }
51
- if (value < 1024) {
52
- return `${value} B`;
53
- }
54
- const mb = value / 1024 / 1024;
55
- if (mb >= 1) {
56
- return `${mb.toFixed(mb < 10 ? 1 : 0)} MB`;
57
- }
58
- return `${(value / 1024).toFixed(value < 10 * 1024 ? 1 : 0)} KB`;
59
- };
60
- const normalizeAddress = (value) => {
61
- if (!value) {
62
- return [];
63
- }
64
- const source = Array.isArray(value) ? value : [value];
65
- return source.flatMap(item => {
66
- const candidate = item;
67
- if (candidate.group) {
68
- return normalizeAddress(candidate.group);
69
- }
70
- return [{
71
- name: candidate.name || '',
72
- address: candidate.address || candidate.email || '',
73
- }];
74
- });
75
- };
76
- const addressText = (items) => {
77
- return items
78
- .map(item => item.name && item.address ? `${item.name} <${item.address}>` : item.address || item.name || '')
79
- .filter(Boolean)
80
- .join(', ');
81
- };
82
- const toArrayBuffer = async (value) => {
83
- if (value instanceof ArrayBuffer) {
84
- return value;
85
- }
86
- if (value instanceof Uint8Array) {
87
- const copy = new Uint8Array(value.byteLength);
88
- copy.set(value);
89
- return copy.buffer;
90
- }
91
- return toArrayBuffer(new TextEncoder().encode(value));
92
- };
93
- const normalizeContentId = (value) => (value || '').replace(/[<>]/g, '');
94
- const createPostalAttachments = (email, objectUrls, cidUrls) => {
95
- const attachments = (email.attachments || []).map((attachment, index) => {
96
- var _a;
97
- const size = typeof attachment.content === 'string'
98
- ? attachment.content.length
99
- : ((_a = attachment.content) === null || _a === void 0 ? void 0 : _a.byteLength) || 0;
100
- const name = attachment.filename || `attachment-${index + 1}`;
101
- return {
102
- id: `${index}-${name}`,
103
- name,
104
- mimeType: attachment.mimeType,
105
- size,
106
- contentId: attachment.contentId,
107
- load: () => toArrayBuffer(attachment.content),
108
- };
109
- });
110
- return Promise.all(attachments.map(async (attachment) => {
111
- var _a;
112
- if (!attachment.contentId || !((_a = attachment.mimeType) === null || _a === void 0 ? void 0 : _a.startsWith('image/'))) {
113
- return;
114
- }
115
- const buffer = await attachment.load();
116
- const url = URL.createObjectURL(new Blob([buffer], { type: attachment.mimeType }));
117
- objectUrls.push(url);
118
- cidUrls.set(normalizeContentId(attachment.contentId), url);
119
- })).then(() => attachments);
120
- };
121
- const parseEml = async (buffer, filename, objectUrls, cidUrls) => {
122
- var _a, _b;
123
- const PostalMime = (await import('postal-mime')).default;
124
- const email = await PostalMime.parse(buffer, {
125
- attachmentEncoding: 'arraybuffer',
126
- maxNestingDepth: 24,
127
- maxHeadersSize: 2 * 1024 * 1024,
128
- });
129
- const attachments = await createPostalAttachments(email, objectUrls, cidUrls);
130
- return {
131
- kind: 'eml',
132
- subject: email.subject || filename,
133
- from: normalizeAddress(email.from),
134
- to: normalizeAddress(email.to),
135
- cc: normalizeAddress(email.cc),
136
- date: email.date,
137
- text: email.text,
138
- html: email.html,
139
- headers: ((_a = email.headerLines) === null || _a === void 0 ? void 0 : _a.map((item) => item.line).join('\n'))
140
- || ((_b = email.headers) === null || _b === void 0 ? void 0 : _b.map((item) => `${item.originalKey}: ${item.value}`).join('\n')),
141
- attachments,
142
- };
143
- };
144
- const parseMbox = async (buffer, filename, objectUrls, cidUrls) => {
145
- var _a, _b, _c, _d;
146
- const source = new TextDecoder('utf-8', { fatal: false }).decode(buffer);
147
- const starts = [...source.matchAll(/^From .*$\n/gm)].map(match => match.index || 0);
148
- const firstStart = (_a = starts[0]) !== null && _a !== void 0 ? _a : 0;
149
- const secondStart = (_b = starts[1]) !== null && _b !== void 0 ? _b : source.length;
150
- const firstMessage = source.slice(firstStart, secondStart).replace(/^From .*$\n/, '');
151
- const encoded = new TextEncoder().encode(firstMessage).buffer;
152
- const PostalMime = (await import('postal-mime')).default;
153
- const email = await PostalMime.parse(encoded, {
154
- attachmentEncoding: 'arraybuffer',
155
- maxNestingDepth: 24,
156
- maxHeadersSize: 2 * 1024 * 1024,
157
- });
158
- const attachments = await createPostalAttachments(email, objectUrls, cidUrls);
159
- return {
160
- kind: 'mbox',
161
- subject: email.subject || `${filename} · 第 1 封`,
162
- from: normalizeAddress(email.from),
163
- to: normalizeAddress(email.to),
164
- cc: normalizeAddress(email.cc),
165
- date: email.date,
166
- text: `MBOX 共识别 ${Math.max(1, starts.length)} 封邮件,当前展示第 1 封。\n\n${email.text || ''}`,
167
- html: email.html,
168
- headers: ((_c = email.headerLines) === null || _c === void 0 ? void 0 : _c.map((item) => item.line).join('\n'))
169
- || ((_d = email.headers) === null || _d === void 0 ? void 0 : _d.map((item) => `${item.originalKey}: ${item.value}`).join('\n')),
170
- attachments,
171
- };
172
- };
173
- const parseMsg = async (buffer, filename) => {
174
- var _a;
175
- const msgReaderModule = await import('@kenjiuno/msgreader');
176
- const MsgReader = (((_a = msgReaderModule.default) === null || _a === void 0 ? void 0 : _a.default) || msgReaderModule.default);
177
- const reader = new MsgReader(buffer);
178
- const fileData = reader.getFileData();
179
- const attachments = (fileData.attachments || []).map((attachment, index) => {
180
- const name = attachment.fileName || attachment.fileNameShort || attachment.name || `attachment-${index + 1}${attachment.extension || ''}`;
181
- return {
182
- id: `${index}-${name}`,
183
- name,
184
- mimeType: 'application/octet-stream',
185
- size: attachment.contentLength || attachment.size || 0,
186
- contentId: attachment.pidContentId,
187
- async load() {
188
- const file = reader.getAttachment(attachment);
189
- return toArrayBuffer(file.content);
190
- },
191
- };
192
- });
193
- return {
194
- kind: 'msg',
195
- subject: fileData.subject || filename,
196
- from: normalizeAddress({ name: fileData.senderName, address: fileData.senderEmail }),
197
- to: normalizeAddress(fileData.recipients || []).filter(item => item.name || item.address),
198
- cc: [],
199
- date: fileData.messageDeliveryTime || fileData.clientSubmitTime || fileData.creationTime,
200
- text: fileData.body,
201
- html: fileData.html || '',
202
- headers: fileData.headers,
203
- attachments,
204
- };
205
- };
206
- const parseEmail = (buffer, type, filename, objectUrls, cidUrls) => {
207
- if (type === 'msg') {
208
- return parseMsg(buffer, filename);
209
- }
210
- if (type === 'mbox') {
211
- return parseMbox(buffer, filename, objectUrls, cidUrls);
212
- }
213
- return parseEml(buffer, filename, objectUrls, cidUrls);
214
- };
215
- const createStyle = () => {
216
- const style = document.createElement('style');
217
- style.textContent = emailStyle;
218
- return style;
219
- };
220
- const createElement = (tagName, className, text) => {
221
- const element = document.createElement(tagName);
222
- if (className) {
223
- element.className = className;
224
- }
225
- if (text !== undefined) {
226
- element.textContent = text;
227
- }
228
- return element;
229
- };
230
- const getAttachmentExtension = (name) => {
231
- const index = name.lastIndexOf('.');
232
- return index >= 0 ? name.slice(index + 1).toLowerCase() : 'txt';
233
- };
234
- const createHtmlSrcdoc = (html, cidUrls) => {
235
- let next = html;
236
- cidUrls.forEach((url, cid) => {
237
- const escaped = cid.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
238
- next = next.replace(new RegExp(`cid:${escaped}`, 'gi'), url);
239
- });
240
- return `<!doctype html><html><head><meta charset="utf-8"><base target="_blank"><style>body{margin:0;padding:18px;font-family:Aptos,"Segoe UI",sans-serif;line-height:1.6;color:#172033;word-break:break-word;}img{max-width:100%;height:auto;}</style></head><body>${next}</body></html>`;
241
- };
242
- const appendMeta = (meta, label, value) => {
243
- const row = document.createElement('p');
244
- const strong = document.createElement('strong');
245
- strong.textContent = label;
246
- row.append(strong, document.createTextNode(value || '-'));
247
- meta.append(row);
248
- };
249
- export default async function renderEmail(buffer, target, type = 'eml', context) {
250
- const normalizedType = type === 'msg' ? 'msg' : type === 'mbox' ? 'mbox' : 'eml';
251
- const filename = (context === null || context === void 0 ? void 0 : context.filename) || `message.${normalizedType}`;
252
- const objectUrls = [];
253
- const cidUrls = new Map();
254
- const cleanups = [];
255
- let nestedRendered;
256
- let overlay = null;
257
- let errorElement = null;
258
- const root = createElement('section', 'email-viewer');
259
- const style = createStyle();
260
- target.replaceChildren(style, root);
261
- const listen = (element, event, listener) => {
262
- element.addEventListener(event, listener);
263
- cleanups.push(() => element.removeEventListener(event, listener));
264
- };
265
- const showLoading = (text) => {
266
- if (!overlay) {
267
- overlay = createElement('div', 'email-state');
268
- overlay.append(createElement('span'), createElement('strong', undefined, text));
269
- root.append(overlay);
270
- return;
271
- }
272
- const label = overlay.querySelector('strong');
273
- if (label) {
274
- label.textContent = text;
275
- }
276
- };
277
- const hideLoading = () => {
278
- overlay === null || overlay === void 0 ? void 0 : overlay.remove();
279
- overlay = null;
280
- };
281
- const showError = (message) => {
282
- errorElement === null || errorElement === void 0 ? void 0 : errorElement.remove();
283
- errorElement = createElement('div', 'email-error');
284
- errorElement.append(createElement('strong', undefined, '邮件预览提示'));
285
- errorElement.append(createElement('p', undefined, message));
286
- root.append(errorElement);
287
- };
288
- const clearAttachmentPreview = async () => {
289
- await disposeFileViewerRendered(nestedRendered);
290
- nestedRendered = undefined;
291
- };
292
- const downloadAttachment = async (attachment) => {
293
- const attachmentBuffer = await attachment.load();
294
- const url = URL.createObjectURL(new Blob([attachmentBuffer], { type: attachment.mimeType || 'application/octet-stream' }));
295
- objectUrls.push(url);
296
- const link = document.createElement('a');
297
- link.href = url;
298
- link.download = attachment.name;
299
- document.body.append(link);
300
- link.click();
301
- link.remove();
302
- };
303
- const renderParsedEmail = (parsed) => {
304
- let activeBody = parsed.html ? 'html' : parsed.text ? 'text' : 'headers';
305
- let activeAttachment = null;
306
- const tabButtons = [];
307
- const attachmentButtons = [];
308
- root.replaceChildren();
309
- const header = createElement('header', 'email-header');
310
- header.append(createElement('span', undefined, parsed.kind.toUpperCase()));
311
- header.append(createElement('h2', undefined, parsed.subject || filename));
312
- const meta = createElement('div', 'email-meta');
313
- appendMeta(meta, '发件人', addressText(parsed.from));
314
- appendMeta(meta, '收件人', addressText(parsed.to));
315
- if (parsed.cc.length) {
316
- appendMeta(meta, '抄送', addressText(parsed.cc));
317
- }
318
- appendMeta(meta, '时间', parsed.date || '-');
319
- header.append(meta);
320
- const body = createElement('div', 'email-body');
321
- const sidebar = createElement('aside', 'email-sidebar');
322
- const tabs = createElement('div', 'body-tabs');
323
- const messagePanel = createElement('main', 'message-panel');
324
- const messageContent = createElement('div', 'email-message-content');
325
- const attachmentPreview = createElement('section', 'attachment-preview');
326
- attachmentPreview.hidden = true;
327
- const attachmentPreviewHead = createElement('div', 'attachment-preview-head');
328
- const attachmentPreviewTitle = createElement('strong');
329
- const attachmentDownload = createElement('button', undefined, '下载附件');
330
- attachmentDownload.type = 'button';
331
- const attachmentTarget = createElement('div', 'attachment-target');
332
- attachmentPreviewHead.append(attachmentPreviewTitle, attachmentDownload);
333
- attachmentPreview.append(attachmentPreviewHead, attachmentTarget);
334
- const renderMessageContent = () => {
335
- messageContent.replaceChildren();
336
- if (activeBody === 'html' && parsed.html) {
337
- const iframe = createElement('iframe', 'email-html');
338
- iframe.setAttribute('sandbox', '');
339
- iframe.srcdoc = createHtmlSrcdoc(parsed.html, cidUrls);
340
- messageContent.append(iframe);
341
- return;
342
- }
343
- const pre = createElement('pre', 'email-text');
344
- pre.textContent = activeBody === 'text' ? parsed.text || '' : parsed.headers || '';
345
- messageContent.append(pre);
346
- };
347
- const syncTabState = () => {
348
- tabButtons.forEach(({ mode, button }) => {
349
- button.classList.toggle('active', mode === activeBody);
350
- });
351
- };
352
- const bodyModes = [
353
- { key: 'html', label: 'HTML', disabled: !parsed.html },
354
- { key: 'text', label: '正文', disabled: !parsed.text },
355
- { key: 'headers', label: '头信息', disabled: !parsed.headers },
356
- ];
357
- bodyModes.forEach(mode => {
358
- const button = createElement('button', undefined, mode.label);
359
- button.type = 'button';
360
- button.disabled = mode.disabled;
361
- listen(button, 'click', () => {
362
- if (button.disabled) {
363
- return;
364
- }
365
- activeBody = mode.key;
366
- syncTabState();
367
- renderMessageContent();
368
- });
369
- tabButtons.push({ mode: mode.key, button });
370
- tabs.append(button);
371
- });
372
- syncTabState();
373
- const attachmentPanel = createElement('section', 'attachment-panel');
374
- const attachmentTitle = createElement('div', 'attachment-title');
375
- attachmentTitle.append(createElement('strong', undefined, '附件'), createElement('span', undefined, String(parsed.attachments.length)));
376
- attachmentPanel.append(attachmentTitle);
377
- if (!parsed.attachments.length) {
378
- attachmentPanel.append(createElement('p', 'attachment-empty', '暂无附件'));
379
- }
380
- const syncAttachmentState = () => {
381
- attachmentButtons.forEach(({ id, button }) => {
382
- button.classList.toggle('active', id === (activeAttachment === null || activeAttachment === void 0 ? void 0 : activeAttachment.id));
383
- });
384
- };
385
- const previewAttachment = async (attachment) => {
386
- activeAttachment = attachment;
387
- syncAttachmentState();
388
- attachmentPreview.hidden = false;
389
- attachmentPreviewTitle.textContent = attachment.name;
390
- showLoading(`正在打开附件 ${attachment.name}...`);
391
- try {
392
- await clearAttachmentPreview();
393
- attachmentTarget.replaceChildren();
394
- const attachmentBuffer = await attachment.load();
395
- const child = createElement('div', 'email-attachment-render');
396
- attachmentTarget.append(child);
397
- const extension = getAttachmentExtension(attachment.name);
398
- if (context === null || context === void 0 ? void 0 : context.renderNestedBuffer) {
399
- nestedRendered = await context.renderNestedBuffer(attachmentBuffer, extension, child, {
400
- ...context,
401
- filename: attachment.name,
402
- options: context.options,
403
- });
404
- }
405
- else {
406
- child.append(createElement('div', undefined, `当前运行环境未提供附件嵌套预览入口,请下载 ${attachment.name} 后查看。`));
407
- }
408
- }
409
- catch (nextError) {
410
- console.error(nextError);
411
- showError(nextError instanceof Error ? nextError.message : String(nextError));
412
- }
413
- finally {
414
- hideLoading();
415
- }
416
- };
417
- parsed.attachments.forEach(attachment => {
418
- const button = createElement('button', 'attachment-item');
419
- button.type = 'button';
420
- const icon = createElement('span', undefined, getAttachmentExtension(attachment.name).toUpperCase() || 'FILE');
421
- const name = createElement('strong', undefined, attachment.name);
422
- const size = createElement('em', undefined, formatBytes(attachment.size));
423
- button.append(icon, name, size);
424
- listen(button, 'click', () => {
425
- void previewAttachment(attachment);
426
- });
427
- attachmentButtons.push({ id: attachment.id, button });
428
- attachmentPanel.append(button);
429
- });
430
- listen(attachmentDownload, 'click', () => {
431
- if (activeAttachment) {
432
- void downloadAttachment(activeAttachment);
433
- }
434
- });
435
- renderMessageContent();
436
- sidebar.append(tabs, attachmentPanel);
437
- messagePanel.append(messageContent, attachmentPreview);
438
- body.append(sidebar, messagePanel);
439
- root.append(header, body);
440
- };
441
- showLoading('正在解析邮件...');
442
- try {
443
- const parsed = await parseEmail(buffer, normalizedType, filename, objectUrls, cidUrls);
444
- renderParsedEmail(parsed);
445
- }
446
- catch (nextError) {
447
- console.error(nextError);
448
- root.replaceChildren();
449
- showError(nextError instanceof Error ? nextError.message : String(nextError));
450
- }
451
- finally {
452
- hideLoading();
453
- }
454
- return {
455
- $el: root,
456
- async unmount() {
457
- await clearAttachmentPreview();
458
- cleanups.splice(0).forEach(cleanup => cleanup());
459
- objectUrls.forEach(url => URL.revokeObjectURL(url));
460
- target.replaceChildren();
461
- },
462
- };
463
- }
@@ -1,2 +0,0 @@
1
- import type { FileViewerRenderedInstance } from '../contracts/types';
2
- export default function renderEpub(buffer: ArrayBuffer, target: HTMLDivElement): Promise<FileViewerRenderedInstance>;