From a3d0d2ed34932699d685c2bc0fc6b656deb4de6b Mon Sep 17 00:00:00 2001 From: Vben Date: Sun, 10 Nov 2024 11:50:06 +0800 Subject: [PATCH] feat: added file download examples (#4853) --- .../@core/base/shared/src/utils/download.ts | 160 ++++++++++++++++++ packages/@core/base/shared/src/utils/index.ts | 1 + playground/src/locales/langs/en-US/demos.json | 3 +- playground/src/locales/langs/zh-CN/demos.json | 3 +- playground/src/router/routes/modules/demos.ts | 10 ++ .../demos/features/file-download/base64.ts | 1 + .../demos/features/file-download/index.vue | 74 ++++++++ 7 files changed, 250 insertions(+), 2 deletions(-) create mode 100644 packages/@core/base/shared/src/utils/download.ts create mode 100644 playground/src/views/demos/features/file-download/base64.ts create mode 100644 playground/src/views/demos/features/file-download/index.vue diff --git a/packages/@core/base/shared/src/utils/download.ts b/packages/@core/base/shared/src/utils/download.ts new file mode 100644 index 00000000..d0aa2f09 --- /dev/null +++ b/packages/@core/base/shared/src/utils/download.ts @@ -0,0 +1,160 @@ +import { openWindow } from './window'; + +interface DownloadOptions { + fileName?: string; + source: T; + target?: string; +} + +const DEFAULT_FILENAME = 'downloaded_file'; + +/** + * 通过 URL 下载文件,支持跨域 + * @throws {Error} - 当下载失败时抛出错误 + */ +export async function downloadFileFromUrl({ + fileName, + source, + target = '_blank', +}: DownloadOptions): Promise { + if (!source || typeof source !== 'string') { + throw new Error('Invalid URL.'); + } + + const isChrome = window.navigator.userAgent.toLowerCase().includes('chrome'); + const isSafari = window.navigator.userAgent.toLowerCase().includes('safari'); + + if (/iP/.test(window.navigator.userAgent)) { + console.error('Your browser does not support download!'); + return; + } + + if (isChrome || isSafari) { + triggerDownload(source, resolveFileName(source, fileName)); + } + if (!source.includes('?')) { + source += '?download'; + } + + openWindow(source, { target }); +} + +/** + * 通过 Base64 下载文件 + */ +export function downloadFileFromBase64({ fileName, source }: DownloadOptions) { + if (!source || typeof source !== 'string') { + throw new Error('Invalid Base64 data.'); + } + + const resolvedFileName = fileName || DEFAULT_FILENAME; + triggerDownload(source, resolvedFileName); +} + +/** + * 通过图片 URL 下载图片文件 + */ +export async function downloadFileFromImageUrl({ + fileName, + source, +}: DownloadOptions) { + const base64 = await urlToBase64(source); + downloadFileFromBase64({ fileName, source: base64 }); +} + +/** + * 通过 Blob 下载文件 + * @param blob - 文件的 Blob 对象 + * @param fileName - 可选,下载的文件名称 + */ +export function downloadFileFromBlob({ + fileName = DEFAULT_FILENAME, + source, +}: DownloadOptions): void { + if (!(source instanceof Blob)) { + throw new TypeError('Invalid Blob data.'); + } + + const url = URL.createObjectURL(source); + triggerDownload(url, fileName); +} + +/** + * 下载文件,支持 Blob、字符串和其他 BlobPart 类型 + * @param data - 文件的 BlobPart 数据 + * @param fileName - 下载的文件名称 + */ +export function downloadFileFromBlobPart({ + fileName = DEFAULT_FILENAME, + source, +}: DownloadOptions): void { + // 如果 data 不是 Blob,则转换为 Blob + const blob = + source instanceof Blob + ? source + : new Blob([source], { type: 'application/octet-stream' }); + + // 创建对象 URL 并触发下载 + const url = URL.createObjectURL(blob); + triggerDownload(url, fileName); +} + +/** + * img url to base64 + * @param url + */ +export function urlToBase64(url: string, mineType?: string): Promise { + return new Promise((resolve, reject) => { + let canvas = document.createElement('CANVAS') as HTMLCanvasElement | null; + const ctx = canvas?.getContext('2d'); + const img = new Image(); + img.crossOrigin = ''; + img.addEventListener('load', () => { + if (!canvas || !ctx) { + return reject(new Error('Failed to create canvas.')); + } + canvas.height = img.height; + canvas.width = img.width; + ctx.drawImage(img, 0, 0); + const dataURL = canvas.toDataURL(mineType || 'image/png'); + canvas = null; + resolve(dataURL); + }); + img.src = url; + }); +} + +/** + * 通用下载触发函数 + * @param href - 文件下载的 URL + * @param fileName - 下载文件的名称,如果未提供则自动识别 + * @param revokeDelay - 清理 URL 的延迟时间 (毫秒) + */ +export function triggerDownload( + href: string, + fileName: string | undefined, + revokeDelay: number = 100, +): void { + const defaultFileName = 'downloaded_file'; + const finalFileName = fileName || defaultFileName; + + const link = document.createElement('a'); + link.href = href; + link.download = finalFileName; + link.style.display = 'none'; + + if (link.download === undefined) { + link.setAttribute('target', '_blank'); + } + + document.body.append(link); + link.click(); + link.remove(); + + // 清理临时 URL 以释放内存 + setTimeout(() => URL.revokeObjectURL(href), revokeDelay); +} + +function resolveFileName(url: string, fileName?: string): string { + return fileName || url.slice(url.lastIndexOf('/') + 1) || DEFAULT_FILENAME; +} diff --git a/packages/@core/base/shared/src/utils/index.ts b/packages/@core/base/shared/src/utils/index.ts index 8b8da01b..2f56c601 100644 --- a/packages/@core/base/shared/src/utils/index.ts +++ b/packages/@core/base/shared/src/utils/index.ts @@ -2,6 +2,7 @@ export * from './cn'; export * from './date'; export * from './diff'; export * from './dom'; +export * from './download'; export * from './inference'; export * from './letter'; export * from './merge'; diff --git a/playground/src/locales/langs/en-US/demos.json b/playground/src/locales/langs/en-US/demos.json index 86dafca5..44b12780 100644 --- a/playground/src/locales/langs/en-US/demos.json +++ b/playground/src/locales/langs/en-US/demos.json @@ -49,7 +49,8 @@ "fullScreen": "FullScreen", "clipboard": "Clipboard", "menuWithQuery": "Menu With Query", - "openInNewWindow": "Open in New Window" + "openInNewWindow": "Open in New Window", + "fileDownload": "File Download" }, "breadcrumb": { "navigation": "Breadcrumb Navigation", diff --git a/playground/src/locales/langs/zh-CN/demos.json b/playground/src/locales/langs/zh-CN/demos.json index 908449c6..254e072b 100644 --- a/playground/src/locales/langs/zh-CN/demos.json +++ b/playground/src/locales/langs/zh-CN/demos.json @@ -49,7 +49,8 @@ "fullScreen": "全屏", "clipboard": "剪贴板", "menuWithQuery": "带参菜单", - "openInNewWindow": "新窗口打开" + "openInNewWindow": "新窗口打开", + "fileDownload": "文件下载" }, "breadcrumb": { "navigation": "面包屑导航", diff --git a/playground/src/router/routes/modules/demos.ts b/playground/src/router/routes/modules/demos.ts index f587e0be..10ffd299 100644 --- a/playground/src/router/routes/modules/demos.ts +++ b/playground/src/router/routes/modules/demos.ts @@ -177,6 +177,16 @@ const routes: RouteRecordRaw[] = [ title: $t('demos.features.fullScreen'), }, }, + { + name: 'FileDownloadDemo', + path: '/demos/features/file-download', + component: () => + import('#/views/demos/features/file-download/index.vue'), + meta: { + icon: 'lucide:hard-drive-download', + title: $t('demos.features.fileDownload'), + }, + }, { name: 'ClipboardDemo', path: '/demos/features/clipboard', diff --git a/playground/src/views/demos/features/file-download/base64.ts b/playground/src/views/demos/features/file-download/base64.ts new file mode 100644 index 00000000..ee6ac2bc --- /dev/null +++ b/playground/src/views/demos/features/file-download/base64.ts @@ -0,0 +1 @@ +export default ``; diff --git a/playground/src/views/demos/features/file-download/index.vue b/playground/src/views/demos/features/file-download/index.vue new file mode 100644 index 00000000..b4125b8a --- /dev/null +++ b/playground/src/views/demos/features/file-download/index.vue @@ -0,0 +1,74 @@ + + +