@file-viewer/core 2.0.11 → 2.1.1

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 (116) 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 +76 -1
  6. package/dist/headless.d.ts +3 -3
  7. package/dist/headless.js +2 -2
  8. package/dist/index.d.ts +10 -7
  9. package/dist/index.js +106 -49
  10. package/dist/lifecycle/operations.d.ts +1 -0
  11. package/dist/lifecycle/operations.js +65 -6
  12. package/dist/platform/assets.d.ts +3 -1
  13. package/dist/platform/assets.js +43 -6
  14. package/dist/registry/capabilities.d.ts +2 -2
  15. package/dist/registry/capabilities.js +2 -1
  16. package/dist/registry/formats.d.ts +20 -7
  17. package/dist/registry/formats.js +14 -5
  18. package/dist/registry/registry.d.ts +8 -1
  19. package/dist/registry/registry.js +29 -0
  20. package/dist/renderers/image.js +1 -10
  21. package/dist/renderers/index.d.ts +320 -2
  22. package/dist/renderers/index.js +27 -157
  23. package/dist/viewer/createViewer.js +86 -3
  24. package/package.json +17 -44
  25. package/dist/renderers/archive.d.ts +0 -2
  26. package/dist/renderers/archive.js +0 -547
  27. package/dist/renderers/archiveCache.d.ts +0 -10
  28. package/dist/renderers/archiveCache.js +0 -96
  29. package/dist/renderers/archiveFallback.d.ts +0 -7
  30. package/dist/renderers/archiveFallback.js +0 -166
  31. package/dist/renderers/archiveShared.d.ts +0 -23
  32. package/dist/renderers/archiveShared.js +0 -71
  33. package/dist/renderers/audio.d.ts +0 -8
  34. package/dist/renderers/audio.js +0 -219
  35. package/dist/renderers/cad.d.ts +0 -2
  36. package/dist/renderers/cad.js +0 -446
  37. package/dist/renderers/code.d.ts +0 -11
  38. package/dist/renderers/code.js +0 -233
  39. package/dist/renderers/data.d.ts +0 -7
  40. package/dist/renderers/data.js +0 -370
  41. package/dist/renderers/drawing.d.ts +0 -10
  42. package/dist/renderers/drawing.js +0 -882
  43. package/dist/renderers/eda.d.ts +0 -2
  44. package/dist/renderers/eda.js +0 -434
  45. package/dist/renderers/edaParser.d.ts +0 -77
  46. package/dist/renderers/edaParser.js +0 -569
  47. package/dist/renderers/email.d.ts +0 -2
  48. package/dist/renderers/email.js +0 -463
  49. package/dist/renderers/epub.d.ts +0 -2
  50. package/dist/renderers/epub.js +0 -331
  51. package/dist/renderers/geo.d.ts +0 -2
  52. package/dist/renderers/geo.js +0 -284
  53. package/dist/renderers/markdown.d.ts +0 -2
  54. package/dist/renderers/markdown.js +0 -83
  55. package/dist/renderers/model.d.ts +0 -2
  56. package/dist/renderers/model.js +0 -567
  57. package/dist/renderers/ofd.d.ts +0 -2
  58. package/dist/renderers/ofd.js +0 -256
  59. package/dist/renderers/openDocument.d.ts +0 -2
  60. package/dist/renderers/openDocument.js +0 -122
  61. package/dist/renderers/pdf.d.ts +0 -3
  62. package/dist/renderers/pdf.js +0 -1001
  63. package/dist/renderers/pdfStyles.d.ts +0 -1
  64. package/dist/renderers/pdfStyles.js +0 -1
  65. package/dist/renderers/pptx.d.ts +0 -2
  66. package/dist/renderers/pptx.js +0 -217
  67. package/dist/renderers/spreadsheet/state.d.ts +0 -80
  68. package/dist/renderers/spreadsheet/state.js +0 -96
  69. package/dist/renderers/spreadsheet/view.d.ts +0 -25
  70. package/dist/renderers/spreadsheet/view.js +0 -833
  71. package/dist/renderers/spreadsheet/worker/index.d.ts +0 -2
  72. package/dist/renderers/spreadsheet/worker/index.js +0 -1
  73. package/dist/renderers/spreadsheet/worker/sheetjs/SheetJsModel.d.ts +0 -73
  74. package/dist/renderers/spreadsheet/worker/sheetjs/SheetJsModel.js +0 -623
  75. package/dist/renderers/spreadsheet/worker/sheetjs/color.d.ts +0 -2
  76. package/dist/renderers/spreadsheet/worker/sheetjs/color.js +0 -73
  77. package/dist/renderers/spreadsheet/worker/sheetjs/index.d.ts +0 -1
  78. package/dist/renderers/spreadsheet/worker/sheetjs/index.js +0 -1
  79. package/dist/renderers/spreadsheet/worker/sheetjs/parser.d.ts +0 -18
  80. package/dist/renderers/spreadsheet/worker/sheetjs/parser.js +0 -106
  81. package/dist/renderers/spreadsheet/worker/sheetjs/sheet.worker.d.ts +0 -1
  82. package/dist/renderers/spreadsheet/worker/sheetjs/sheet.worker.js +0 -11
  83. package/dist/renderers/spreadsheet/worker/type.d.ts +0 -57
  84. package/dist/renderers/spreadsheet/worker/type.js +0 -1
  85. package/dist/renderers/spreadsheet.d.ts +0 -3
  86. package/dist/renderers/spreadsheet.js +0 -929
  87. package/dist/renderers/typst.d.ts +0 -8
  88. package/dist/renderers/typst.js +0 -547
  89. package/dist/renderers/umd/parser.d.ts +0 -30
  90. package/dist/renderers/umd/parser.js +0 -408
  91. package/dist/renderers/umd.d.ts +0 -2
  92. package/dist/renderers/umd.js +0 -297
  93. package/dist/renderers/video.d.ts +0 -8
  94. package/dist/renderers/video.js +0 -108
  95. package/dist/renderers/wordDoc.d.ts +0 -5
  96. package/dist/renderers/wordDoc.js +0 -284
  97. package/dist/renderers/wordDocx.d.ts +0 -5
  98. package/dist/renderers/wordDocx.js +0 -985
  99. package/dist/renderers/wordDocx.worker.d.ts +0 -1
  100. package/dist/renderers/wordDocx.worker.js +0 -96
  101. package/vendor/ofd/dltech/jbig2/arithmetic_decoder.js +0 -183
  102. package/vendor/ofd/dltech/jbig2/ccitt.js +0 -1070
  103. package/vendor/ofd/dltech/jbig2/compatibility.js +0 -12
  104. package/vendor/ofd/dltech/jbig2/core_utils.js +0 -180
  105. package/vendor/ofd/dltech/jbig2/is_node.js +0 -27
  106. package/vendor/ofd/dltech/jbig2/jbig2.js +0 -2589
  107. package/vendor/ofd/dltech/jbig2/jbig2_stream.js +0 -81
  108. package/vendor/ofd/dltech/jbig2/primitives.js +0 -387
  109. package/vendor/ofd/dltech/jbig2/stream.js +0 -1348
  110. package/vendor/ofd/dltech/jbig2/util.js +0 -972
  111. package/vendor/ofd/dltech/ofd/ofd.d.ts +0 -11
  112. package/vendor/ofd/dltech/ofd/ofd.js +0 -100
  113. package/vendor/ofd/dltech/ofd/ofd_parser.js +0 -395
  114. package/vendor/ofd/dltech/ofd/ofd_render.js +0 -473
  115. package/vendor/ofd/dltech/ofd/ofd_util.js +0 -350
  116. 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>;