@base-web-kits/base-tools-web 1.2.4 → 1.2.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.
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/web/clipboard/index.ts","../src/web/config/index.ts","../src/web/cookie/index.ts","../src/web/device/index.ts","../src/web/dom/index.ts","../src/web/network/download.ts","../src/ts/day/index.ts","../src/ts/es-toolkit/index.ts","../src/ts/object/index.ts","../src/ts/url/param/index.ts","../src/web/network/request.ts","../src/web/async/index.ts","../src/web/network/uploadFile.ts","../src/web/storage/index.ts","../src/web/url/index.ts"],"sourcesContent":["/**\r\n * 复制文本到剪贴板(兼容移动端和PC)\r\n * @returns Promise<void> 复制成功时 resolve,失败时 reject。\r\n * @example\r\n * await copyText('hello');\r\n * toast('复制成功');\r\n */\r\nexport async function copyText(text: string): Promise<void> {\r\n if (typeof text !== 'string') text = String(text ?? '');\r\n\r\n // 现代 API\r\n if (navigator.clipboard && typeof navigator.clipboard.writeText === 'function') {\r\n try {\r\n await navigator.clipboard.writeText(text);\r\n return;\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n } catch (e) {\r\n // 继续尝试回退方案\r\n }\r\n }\r\n\r\n // 回退方案:使用隐藏 textarea + execCommand('copy')\r\n return new Promise<void>((resolve, reject) => {\r\n try {\r\n const textarea = document.createElement('textarea');\r\n textarea.value = text;\r\n\r\n // 避免视觉影响与页面布局影响\r\n textarea.setAttribute('readonly', '');\r\n textarea.style.position = 'fixed';\r\n textarea.style.top = '0';\r\n textarea.style.right = '-9999px';\r\n textarea.style.opacity = '0';\r\n textarea.style.pointerEvents = 'none';\r\n\r\n document.body.appendChild(textarea);\r\n\r\n // 选中文本(移动端兼容)\r\n textarea.focus();\r\n textarea.select();\r\n\r\n // iOS 兼容:明确选区\r\n textarea.setSelectionRange(0, textarea.value.length);\r\n\r\n const ok = document.execCommand('copy');\r\n document.body.removeChild(textarea);\r\n\r\n if (ok) {\r\n resolve();\r\n } else {\r\n reject(new Error('Copy failed: clipboard unavailable'));\r\n }\r\n } catch (e) {\r\n reject(e);\r\n }\r\n });\r\n}\r\n\r\n/**\r\n * 复制富文本 HTML 到剪贴板(移动端与 PC)\r\n * 使用场景:图文混排文章、带样式段落,保留格式粘贴。\r\n * @param html HTML字符串\r\n * @example\r\n * await copyHtml('<p><b>加粗</b> 与 <i>斜体</i></p>');\r\n */\r\nexport async function copyHtml(html: string): Promise<void> {\r\n const s = String(html ?? '');\r\n if (canWriteClipboard()) {\r\n const plain = htmlToText(s);\r\n await writeClipboard({\r\n 'text/html': new Blob([s], { type: 'text/html' }),\r\n 'text/plain': new Blob([plain], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n return execCopyFromHtml(s);\r\n}\r\n\r\n/**\r\n * 复制 DOM 节点到剪贴板(移动端与 PC)\r\n * 使用场景:页面已有区域的可视化复制;元素使用 `outerHTML`,非元素使用其文本内容。\r\n * @param node DOM 节点(元素或文本节点)\r\n * @example\r\n * const el = document.querySelector('#article')!;\r\n * await copyNode(el);\r\n */\r\nexport async function copyNode(node: Node): Promise<void> {\r\n if (canWriteClipboard()) {\r\n const { html, text } = nodeToHtmlText(node);\r\n await writeClipboard({\r\n 'text/html': new Blob([html], { type: 'text/html' }),\r\n 'text/plain': new Blob([text], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n const { html } = nodeToHtmlText(node);\r\n return execCopyFromHtml(html);\r\n}\r\n\r\n/**\r\n * 复制单张图片到剪贴板(移动端与 PC,需浏览器支持 `ClipboardItem`)\r\n * 使用场景:把本地 `canvas` 或 `Blob` 生成的图片直接粘贴到聊天/文档。\r\n * @param image 图片源(Blob/Canvas/ImageBitmap)\r\n * @example\r\n * const canvas = document.querySelector('canvas')!;\r\n * await copyImage(canvas);\r\n */\r\nexport async function copyImage(image: Blob | HTMLCanvasElement | ImageBitmap): Promise<void> {\r\n const blob = await toImageBlob(image);\r\n if (!blob) throw new Error('Unsupported image source');\r\n if (canWriteClipboard()) {\r\n const type = blob.type || 'image/png';\r\n await writeClipboard({ [type]: blob });\r\n return;\r\n }\r\n throw new Error('Clipboard image write not supported');\r\n}\r\n\r\n/**\r\n * 复制 URL 到剪贴板(移动端与 PC)\r\n * 写入 `text/uri-list` 与 `text/plain`,在支持 URI 列表的应用中可识别为链接。\r\n * @param url 完整的 URL 字符串\r\n * @example\r\n * await copyUrl('https://example.com/page');\r\n */\r\nexport async function copyUrl(url: string): Promise<void> {\r\n const s = String(url ?? '');\r\n if (canWriteClipboard()) {\r\n await writeClipboard({\r\n 'text/uri-list': new Blob([s], { type: 'text/uri-list' }),\r\n 'text/plain': new Blob([s], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n await copyText(s);\r\n}\r\n\r\n/**\r\n * 复制任意 Blob 到剪贴板(移动端与 PC,需 `ClipboardItem`)\r\n * 使用场景:原生格式粘贴(如 `image/svg+xml`、`application/pdf` 等)。\r\n * @param blob 任意 Blob 数据\r\n * @example\r\n * const svg = new Blob(['<svg></svg>'], { type: 'image/svg+xml' });\r\n * await copyBlob(svg);\r\n */\r\nexport async function copyBlob(blob: Blob): Promise<void> {\r\n if (canWriteClipboard()) {\r\n const type = blob.type || 'application/octet-stream';\r\n await writeClipboard({ [type]: blob });\r\n return;\r\n }\r\n throw new Error('Clipboard blob write not supported');\r\n}\r\n\r\n/**\r\n * 复制 RTF 富文本到剪贴板(移动端与 PC)\r\n * 同时写入 `text/plain`,增强与 Office/富文本编辑器的兼容性。\r\n * @param rtf RTF 字符串(如:`{\\\\rtf1\\\\ansi ...}`)\r\n * @example\r\n * await copyRtf('{\\\\rtf1\\\\ansi Hello \\\\b World}');\r\n */\r\nexport async function copyRtf(rtf: string): Promise<void> {\r\n const s = String(rtf ?? '');\r\n if (canWriteClipboard()) {\r\n const plain = s\r\n .replace(/\\\\par[\\s]?/g, '\\n')\r\n .replace(/\\{[^}]*\\}/g, '')\r\n .replace(/\\\\[a-zA-Z]+[0-9'-]*/g, '')\r\n .replace(/\\r?\\n/g, '\\n')\r\n .trim();\r\n await writeClipboard({\r\n 'text/rtf': new Blob([s], { type: 'text/rtf' }),\r\n 'text/plain': new Blob([plain], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n await copyText(s);\r\n}\r\n\r\n/**\r\n * 复制表格到剪贴板(移动端与 PC)\r\n * 同时写入多种 MIME:`text/html`(表格)、`text/tab-separated-values`(TSV)、`text/csv`、`text/plain`(TSV)。\r\n * 使用场景:优化粘贴到 Excel/Google Sheets/Docs 的体验\r\n * @param rows 二维数组,每行一个数组(字符串/数字)\r\n * @example\r\n * await copyTable([\r\n * ['姓名', '分数'],\r\n * ['张三', 95],\r\n * ['李四', 88],\r\n * ]);\r\n */\r\nexport async function copyTable(rows: Array<Array<string | number>>): Promise<void> {\r\n const data = Array.isArray(rows) ? rows : [];\r\n const escapeHtml = (t: string) =>\r\n t\r\n .replace(/&/g, '&amp;')\r\n .replace(/</g, '&lt;')\r\n .replace(/>/g, '&gt;')\r\n .replace(/\"/g, '&quot;')\r\n .replace(/'/g, '&#39;');\r\n const html = (() => {\r\n const trs = data\r\n .map((r) => `<tr>${r.map((c) => `<td>${escapeHtml(String(c))}</td>`).join('')}</tr>`)\r\n .join('');\r\n return `<table>${trs}</table>`;\r\n })();\r\n const tsv = data.map((r) => r.map((c) => String(c)).join('\\t')).join('\\n');\r\n const csv = data\r\n .map((r) =>\r\n r\r\n .map((c) => {\r\n const s = String(c);\r\n const needQuote = /[\",\\n]/.test(s);\r\n const escaped = s.replace(/\"/g, '\"\"');\r\n return needQuote ? `\"${escaped}\"` : escaped;\r\n })\r\n .join(','),\r\n )\r\n .join('\\n');\r\n if (canWriteClipboard()) {\r\n await writeClipboard({\r\n 'text/html': new Blob([html], { type: 'text/html' }),\r\n 'text/tab-separated-values': new Blob([tsv], { type: 'text/tab-separated-values' }),\r\n 'text/csv': new Blob([csv], { type: 'text/csv' }),\r\n 'text/plain': new Blob([tsv], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n await copyText(tsv);\r\n}\r\n\r\nasync function toImageBlob(image: Blob | HTMLCanvasElement | ImageBitmap) {\r\n if (image instanceof Blob) return image;\r\n if (image instanceof HTMLCanvasElement)\r\n return await new Promise<Blob>((resolve, reject) => {\r\n image.toBlob(\r\n (b) => (b ? resolve(b) : reject(new Error('Canvas toBlob failed'))),\r\n 'image/png',\r\n );\r\n });\r\n const isBitmap = typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap;\r\n if (isBitmap) {\r\n const cnv = document.createElement('canvas');\r\n cnv.width = (image as ImageBitmap).width;\r\n cnv.height = (image as ImageBitmap).height;\r\n const ctx = cnv.getContext('2d');\r\n ctx?.drawImage(image as ImageBitmap, 0, 0);\r\n return await new Promise<Blob>((resolve, reject) => {\r\n cnv.toBlob((b) => (b ? resolve(b) : reject(new Error('Canvas toBlob failed'))), 'image/png');\r\n });\r\n }\r\n return null;\r\n}\r\n\r\nfunction canWriteClipboard() {\r\n return !!(\r\n navigator.clipboard &&\r\n typeof navigator.clipboard.write === 'function' &&\r\n typeof ClipboardItem !== 'undefined'\r\n );\r\n}\r\n\r\nasync function writeClipboard(items: Record<string, Blob>) {\r\n await navigator.clipboard!.write([new ClipboardItem(items)]);\r\n}\r\n\r\nfunction htmlToText(html: string) {\r\n const div = document.createElement('div');\r\n div.innerHTML = html;\r\n return div.textContent || '';\r\n}\r\n\r\nfunction nodeToHtmlText(node: Node) {\r\n const container = document.createElement('div');\r\n container.appendChild(node.cloneNode(true));\r\n const html =\r\n node instanceof Element ? (node.outerHTML ?? container.innerHTML) : container.innerHTML;\r\n const text = container.textContent || '';\r\n return { html, text };\r\n}\r\n\r\nfunction execCopyFromHtml(html: string) {\r\n return new Promise<void>((resolve, reject) => {\r\n try {\r\n const div = document.createElement('div');\r\n div.contentEditable = 'true';\r\n div.style.position = 'fixed';\r\n div.style.top = '0';\r\n div.style.right = '-9999px';\r\n div.style.opacity = '0';\r\n div.style.pointerEvents = 'none';\r\n div.innerHTML = html;\r\n document.body.appendChild(div);\r\n const selection = window.getSelection();\r\n const range = document.createRange();\r\n range.selectNodeContents(div);\r\n selection?.removeAllRanges();\r\n selection?.addRange(range);\r\n const ok = document.execCommand('copy');\r\n document.body.removeChild(div);\r\n selection?.removeAllRanges();\r\n if (ok) {\r\n resolve();\r\n } else {\r\n reject(new Error('Copy failed: clipboard unavailable'));\r\n }\r\n } catch (e) {\r\n reject(e);\r\n }\r\n });\r\n}\r\n","export type AppConfig = {\r\n /** 全局 Toast 提示 */\r\n toast?: (option: { msg: string; status: 'success' | 'fail' }) => void;\r\n /** 显示全局 Loading */\r\n showLoading?: (option?: { title?: string }) => void;\r\n /** 隐藏全局 Loading */\r\n hideLoading?: () => void;\r\n /** 跳转登录页的方法 */\r\n toLogin?: () => void;\r\n /** 日志记录函数 */\r\n log?: (level: 'info' | 'error' | 'warn' | 'debug', data: AppLogInfo) => void;\r\n};\r\n\r\nexport type AppLogInfo = {\r\n /** 调用函数的名称 */\r\n name: string;\r\n\r\n /** 函数的调用状态 */\r\n status?: 'success' | 'fail';\r\n\r\n /** 函数的调用参数 */\r\n option?: unknown;\r\n\r\n /** 函数的调用结果 */\r\n res?: unknown;\r\n\r\n /** 函数的调用错误 */\r\n e?: unknown;\r\n\r\n /** 日志描述 */\r\n desc?: string;\r\n\r\n // 其他自定义属性\r\n [key: string]: unknown;\r\n};\r\n\r\nconst appConfig: AppConfig = {};\r\n\r\n/**\r\n * 获取应用配置\r\n */\r\nexport function getBaseToolsConfig() {\r\n return appConfig;\r\n}\r\n\r\n/**\r\n * 初始化应用配置 (在入口文件设置)\r\n * @example\r\n * setBaseToolsConfig({\r\n * toast: ({ msg, status }) => (status === 'fail' ? message.error(msg) : message.success(msg)),\r\n * showLoading: () => message.loading('加载中...'),\r\n * hideLoading: () => message.destroy(),\r\n * toLogin: () => reLogin(),\r\n * log(level, data) {\r\n * if (data.name === 'request') {\r\n * sendLog('request', data); // 请求日志\r\n * } else if (level === 'error') {\r\n * sendLog('error', data); // 错误日志\r\n * } else {\r\n * sendLog('action', data); // 操作日志\r\n * }\r\n * },\r\n * });\r\n */\r\nexport function setBaseToolsConfig(newConfig: AppConfig) {\r\n Object.assign(appConfig, newConfig);\r\n}\r\n","/**\r\n * 设置 Cookie(路径默认为 `/`)\r\n * @param name Cookie 名称\r\n * @param value Cookie 值(内部已使用 `encodeURIComponent` 编码)\r\n * @param days 过期天数(从当前时间起算)\r\n * @example\r\n * setCookie('token', 'abc', 7);\r\n */\r\nexport function setCookie(name: string, value: string, days: number) {\r\n const date = new Date();\r\n date.setTime(date.getTime() + days * 24 * 60 * 60 * 1000);\r\n const expires = `expires=${date.toUTCString()}; path=/`;\r\n document.cookie = `${name}=${encodeURIComponent(value)}; ${expires}`;\r\n}\r\n\r\n/**\r\n * 获取 Cookie\r\n * @param name Cookie 名称\r\n * @returns 若存在返回解码后的值,否则 `null`\r\n * @example\r\n * const token = getCookie('token');\r\n */\r\nexport function getCookie(name: string): string | null {\r\n const value = `; ${document.cookie}`;\r\n const parts = value.split(`; ${name}=`);\r\n if (parts.length === 2) {\r\n const v = parts.pop()?.split(';').shift();\r\n return v ? decodeURIComponent(v) : null;\r\n }\r\n return null;\r\n}\r\n\r\n/**\r\n * 移除 Cookie(通过设置过期时间为过去)\r\n * 路径固定为 `/`,确保与默认写入路径一致。\r\n * @param name Cookie 名称\r\n * @example\r\n * removeCookie('token');\r\n */\r\nexport function removeCookie(name: string) {\r\n document.cookie = `${name}=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/`;\r\n}\r\n","/**\r\n * 获取用户代理字符串(UA)\r\n * @returns navigator.userAgent.toLowerCase();\r\n */\r\nexport function getUA(): string {\r\n if (typeof navigator === 'undefined') return ''; // SSR无 navigator\r\n return (navigator.userAgent || '').toLowerCase();\r\n}\r\n\r\n/**\r\n * 是否为移动端设备(含平板)\r\n */\r\nexport function isMobile(): boolean {\r\n const ua = getUA();\r\n return /android|webos|iphone|ipod|blackberry|iemobile|opera mini|mobile/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为平板设备\r\n */\r\nexport function isTablet(): boolean {\r\n const ua = getUA();\r\n return /ipad|android(?!.*mobile)|tablet/i.test(ua) && !/mobile/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为 PC 设备\r\n */\r\nexport function isPC(): boolean {\r\n return !isMobile() && !isTablet();\r\n}\r\n\r\n/**\r\n * 是否为 iOS 系统\r\n */\r\nexport function isIOS(): boolean {\r\n const ua = getUA();\r\n return /iphone|ipad|ipod/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为 Android 系统\r\n */\r\nexport function isAndroid(): boolean {\r\n const ua = getUA();\r\n return /android/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否微信内置浏览器\r\n */\r\nexport function isWeChat(): boolean {\r\n const ua = getUA();\r\n return /micromessenger/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为 Chrome 浏览器\r\n * 已排除 Edge、Opera 等基于 Chromium 的浏览器\r\n */\r\nexport function isChrome(): boolean {\r\n const ua = getUA();\r\n return /chrome\\//i.test(ua) && !/edg\\//i.test(ua) && !/opr\\//i.test(ua) && !/whale\\//i.test(ua);\r\n}\r\n\r\n/**\r\n * 检测是否支持触摸事件\r\n */\r\nexport function isTouchSupported(): boolean {\r\n if (typeof window === 'undefined') return false;\r\n return 'ontouchstart' in window || navigator.maxTouchPoints > 0;\r\n}\r\n\r\n/**\r\n * 获取设备像素比\r\n */\r\nexport function getDevicePixelRatio(): number {\r\n if (typeof window === 'undefined') return 1;\r\n return window.devicePixelRatio || 1;\r\n}\r\n\r\n/**\r\n * 获取浏览器名字\r\n */\r\nexport function getBrowserName(): string | null {\r\n const ua = getUA();\r\n\r\n if (/chrome\\//i.test(ua)) return 'chrome';\r\n if (/safari\\//i.test(ua)) return 'safari';\r\n if (/firefox\\//i.test(ua)) return 'firefox';\r\n if (/opr\\//i.test(ua)) return 'opera';\r\n if (/edg\\//i.test(ua)) return 'edge';\r\n if (/msie|trident/i.test(ua)) return 'ie';\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * 获取浏览器版本号\r\n */\r\nexport function getBrowserVersion(): string | null {\r\n const ua = getUA();\r\n\r\n const versionPatterns = [\r\n /(?:edg|edge)\\/([0-9.]+)/i,\r\n /(?:opr|opera)\\/([0-9.]+)/i,\r\n /chrome\\/([0-9.]+)/i,\r\n /firefox\\/([0-9.]+)/i,\r\n /version\\/([0-9.]+).*safari/i,\r\n /(?:msie |rv:)([0-9.]+)/i,\r\n ];\r\n\r\n for (const pattern of versionPatterns) {\r\n const matches = ua.match(pattern);\r\n if (matches && matches[1]) {\r\n return matches[1];\r\n }\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * 获取操作系统信息\r\n */\r\nexport function getOS(): string {\r\n const ua = getUA();\r\n\r\n if (/windows/i.test(ua)) return 'windows';\r\n if (/mac os/i.test(ua)) return 'macos';\r\n if (/linux/i.test(ua)) return 'linux';\r\n if (/iphone|ipad|ipod/i.test(ua)) return 'ios';\r\n if (/android/i.test(ua)) return 'android';\r\n\r\n return 'unknown';\r\n}\r\n","/**\r\n * 获取窗口宽度(不含滚动条)\r\n * @returns 窗口宽度\r\n */\r\nexport function getWindowWidth() {\r\n return window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth;\r\n}\r\n\r\n/**\r\n * 获取窗口高度(不含滚动条)\r\n * @returns 窗口高度\r\n */\r\nexport function getWindowHeight() {\r\n return window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight;\r\n}\r\n\r\n/**\r\n * 获取文档垂直滚动位置\r\n * @example\r\n * const top = getWindowScrollTop();\r\n */\r\nexport function getWindowScrollTop() {\r\n const doc = document.documentElement;\r\n const body = document.body;\r\n return window.pageYOffset || doc.scrollTop || body.scrollTop || 0;\r\n}\r\n\r\n/**\r\n * 获取文档水平滚动位置\r\n * @example\r\n * const left = getWindowScrollLeft();\r\n */\r\nexport function getWindowScrollLeft() {\r\n const doc = document.documentElement;\r\n const body = document.body;\r\n return window.pageXOffset || doc.scrollLeft || body.scrollLeft || 0;\r\n}\r\n\r\n/**\r\n * 平滑滚动到指定位置\r\n * @param top 目标纵向滚动位置\r\n * @param behavior 滚动行为,默认 'smooth'\r\n * @example\r\n * windowScrollTo(0);\r\n */\r\nexport function windowScrollTo(top: number, behavior: ScrollBehavior = 'smooth') {\r\n if ('scrollBehavior' in document.documentElement.style) {\r\n window.scrollTo({ top, behavior });\r\n } else {\r\n window.scrollTo(0, top);\r\n }\r\n}\r\n\r\n/**\r\n * 元素是否在视口内(可设置阈值)\r\n * @param el 目标元素\r\n * @param offset 额外判定偏移(像素,正数放宽,负数收紧)\r\n * @returns 是否在视口内\r\n */\r\nexport function isInViewport(el: Element, offset = 0) {\r\n const rect = el.getBoundingClientRect();\r\n const width = getWindowWidth();\r\n const height = getWindowHeight();\r\n return (\r\n rect.bottom >= -offset &&\r\n rect.right >= -offset &&\r\n rect.top <= height + offset &&\r\n rect.left <= width + offset\r\n );\r\n}\r\n\r\n/**\r\n * 锁定页面滚动(移动端/PC)\r\n * 使用 `body{ position: fixed }` 技术消除滚动条抖动,记录并恢复滚动位置。\r\n * @example\r\n * lockBodyScroll();\r\n */\r\nexport function lockBodyScroll() {\r\n const body = document.body;\r\n if (body.dataset.scrollLock === 'true') return;\r\n const y = Math.round(window.scrollY || window.pageYOffset || 0);\r\n body.dataset.scrollLock = 'true';\r\n body.dataset.scrollLockY = String(y);\r\n body.style.position = 'fixed';\r\n body.style.top = `-${y}px`;\r\n body.style.left = '0';\r\n body.style.right = '0';\r\n body.style.width = '100%';\r\n}\r\n\r\n/**\r\n * 解除页面滚动锁定,恢复原始滚动位置\r\n * @example\r\n * unlockBodyScroll();\r\n */\r\nexport function unlockBodyScroll() {\r\n const body = document.body;\r\n if (body.dataset.scrollLock !== 'true') return;\r\n const y = Number(body.dataset.scrollLockY || 0);\r\n body.style.position = '';\r\n body.style.top = '';\r\n body.style.left = '';\r\n body.style.right = '';\r\n body.style.width = '';\r\n delete body.dataset.scrollLock;\r\n delete body.dataset.scrollLockY;\r\n window.scrollTo(0, y);\r\n}\r\n","import type { AxiosResponse } from 'axios';\r\n\r\n/**\r\n * 下载文件\r\n * @param url 完整的下载地址 | base64字符串 | Blob对象\r\n * @param fileName 自定义文件名(需含后缀)\r\n * @example\r\n * download('https://xx/xx.pdf');\r\n * download('https://xx/xx.pdf', 'xx.pdf');\r\n * download(blob, '图片.jpg');\r\n */\r\nexport async function download(url: string | Blob, fileName = '') {\r\n if (!url) return;\r\n\r\n let blobUrl = '';\r\n let needRevoke = false; // createObjectURL必须revoke,否则内存泄露,刷新页面都不释放\r\n try {\r\n if (url instanceof Blob) {\r\n // Blob对象\r\n blobUrl = URL.createObjectURL(url);\r\n needRevoke = true;\r\n } else if (url.includes(';base64,')) {\r\n // base64字符串\r\n blobUrl = url;\r\n } else {\r\n if (fileName) {\r\n // 自定义文件名:跨域的url无法自定义文件名,此处统一转为blob\r\n const res = await fetch(url);\r\n if (!res.ok) throw new Error(`fetch error ${res.status}:${url}`); // 拦截错误页(404/500 等 HTML)\r\n const blob = await res.blob();\r\n blobUrl = URL.createObjectURL(blob);\r\n needRevoke = true;\r\n } else {\r\n // 非自定义文件名的普通链接\r\n blobUrl = url;\r\n }\r\n }\r\n\r\n // window.location.href = fileUrl // 可能会关闭当前页面\r\n // window.open(fileUrl, '_blank') // 不支持下载图片\r\n // 通过a标签模拟点击下载\r\n const a = document.createElement('a');\r\n a.href = blobUrl;\r\n a.download = fileName; // 若为空字符串,则会自动取url的文件名(跨域url无法自定义文件名,需转为blob)\r\n document.body.appendChild(a);\r\n a.click();\r\n document.body.removeChild(a);\r\n } finally {\r\n if (needRevoke) {\r\n setTimeout(() => URL.revokeObjectURL(blobUrl), 100); // Safari 需要延迟 revoke\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * 解析Axios返回的Blob数据\r\n * @param res Axios响应对象 (responseType='blob')\r\n * @returns 包含blob数据和文件名的对象 { blob, fileName }\r\n * @example\r\n * const res = await axios.get(url, { responseType: 'blob' });\r\n * const { blob, fileName } = await parseAxiosBlob(res);\r\n * download(blob, fileName);\r\n */\r\nexport async function parseAxiosBlob(res: AxiosResponse<Blob>) {\r\n const { data, headers, status, statusText, config } = res;\r\n\r\n if (status < 200 || status >= 300) throw new Error(`${status},${statusText}:${config.url}`);\r\n\r\n // 抛出json错误\r\n if (data.type.includes('application/json')) {\r\n const txt = await data.text();\r\n throw JSON.parse(txt);\r\n }\r\n\r\n // 解析文件名\r\n const fileName = getDispositionFileName(headers['content-disposition']);\r\n return { blob: data, fileName };\r\n}\r\n\r\n/**\r\n * 获取文件名\r\n * @param disposition content-disposition头值\r\n * @returns content-disposition中的filename\r\n * @example\r\n * const fileName = getDispositionFileName(headers['content-disposition']);\r\n */\r\nexport function getDispositionFileName(disposition?: string) {\r\n if (!disposition) return '';\r\n\r\n // 1. RFC5987 filename* 优先\r\n const rfc5987 = /filename\\*\\s*=\\s*([^']*)''([^;]*)/i.exec(disposition);\r\n if (rfc5987?.[2]) {\r\n try {\r\n return decodeURIComponent(rfc5987[2].trim()).replace(/[\\r\\n]+/g, '');\r\n } catch {\r\n return rfc5987[2].trim().replace(/[\\r\\n]+/g, '');\r\n }\r\n }\r\n\r\n // 2. 旧式 filename=\r\n const old = /filename\\s*=\\s*(?:\"([^\"]*)\"|([^\";]*))(?=;|$)/i.exec(disposition);\r\n if (old) return (old[1] ?? old[2]).trim().replace(/[\\r\\n]+/g, '');\r\n\r\n return '';\r\n}\r\n\r\n/**\r\n * 动态加载 JS(重复执行不会重复加载,内部已排重)\r\n * @param src js 文件路径\r\n * @param attrs 可选的脚本属性,如 async、defer、crossOrigin\r\n * @example\r\n * await loadJs('https://xx/xx.js');\r\n * await loadJs('/a.js', { defer: true });\r\n */\r\nexport async function loadJs(\r\n src: string,\r\n attrs?: Pick<HTMLScriptElement, 'async' | 'defer' | 'crossOrigin'>,\r\n) {\r\n return new Promise<void>((resolve, reject) => {\r\n if (hasJs(src)) return resolve();\r\n\r\n const script = document.createElement('script');\r\n script.type = 'text/javascript';\r\n script.src = src;\r\n\r\n if (attrs) {\r\n const keys = Object.keys(attrs) as Array<keyof typeof attrs>;\r\n keys.forEach((key) => {\r\n const v = attrs[key];\r\n if (v === null || v === undefined || v === false) return;\r\n script.setAttribute(key, typeof v === 'boolean' ? '' : v);\r\n });\r\n }\r\n\r\n script.onload = () => resolve();\r\n script.onerror = (e) => reject(e);\r\n\r\n document.head.appendChild(script);\r\n });\r\n}\r\n\r\n/**\r\n * 判断某个 JS 地址是否已在页面中加载过\r\n * @param src 相对、绝对路径的 JS 地址\r\n * @returns 是否已加载过\r\n * @example\r\n * hasJs('https://xx/xx.js'); // boolean\r\n * hasJs('/xx.js'); // boolean\r\n * hasJs('xx.js'); // boolean\r\n */\r\nexport function hasJs(src: string) {\r\n const target = new URL(src, document.baseURI).href;\r\n const jsList = Array.from(document.querySelectorAll('script[src]'));\r\n return jsList.some((e) => {\r\n const src = e.getAttribute('src');\r\n return src && new URL(src, document.baseURI).href === target;\r\n });\r\n}\r\n\r\n/**\r\n * 动态加载 CSS(重复执行不会重复加载,内部已排重)\r\n * @param href css 文件地址\r\n * @param attrs 可选属性,如 crossOrigin、media\r\n * @example\r\n * await loadCss('https://xx/xx.css');\r\n * await loadCss('/a.css', { media: 'print' });\r\n */\r\nexport async function loadCss(\r\n href: string,\r\n attrs?: Pick<HTMLLinkElement, 'crossOrigin' | 'media'>,\r\n) {\r\n return new Promise<void>((resolve, reject) => {\r\n if (hasCss(href)) return resolve();\r\n\r\n const link = document.createElement('link');\r\n link.rel = 'stylesheet';\r\n link.href = href;\r\n\r\n if (attrs) {\r\n const keys = Object.keys(attrs) as Array<keyof typeof attrs>;\r\n keys.forEach((key) => {\r\n const v = attrs[key];\r\n if (v === null || v === undefined) return;\r\n link.setAttribute(key, String(v));\r\n });\r\n }\r\n\r\n link.onload = () => resolve();\r\n link.onerror = (e) => reject(e);\r\n\r\n document.head.appendChild(link);\r\n });\r\n}\r\n\r\n/**\r\n * 判断某个 CSS 地址是否已在页面中加载过\r\n * @param href 相对、绝对路径的 CSS 地址\r\n * @returns 是否已加载过\r\n * @example\r\n * hasCss('https://xx/xx.css'); // boolean\r\n */\r\nexport function hasCss(href: string) {\r\n const target = new URL(href, document.baseURI).href;\r\n const list = Array.from(document.querySelectorAll('link[rel=\"stylesheet\"][href]'));\r\n return list.some((e) => {\r\n const h = e.getAttribute('href');\r\n return h && new URL(h, document.baseURI).href === target;\r\n });\r\n}\r\n\r\n/**\r\n * 预加载图片\r\n * @param src 图片地址\r\n * @returns Promise<HTMLImageElement>\r\n * @example\r\n * await preloadImage('/a.png');\r\n */\r\nexport function preloadImage(src: string) {\r\n return new Promise<HTMLImageElement>((resolve, reject) => {\r\n const img = new Image();\r\n img.onload = () => resolve(img);\r\n img.onerror = (e) => reject(e);\r\n img.src = src;\r\n });\r\n}\r\n","import dayjs from 'dayjs';\r\nimport customParseFormat from 'dayjs/plugin/customParseFormat';\r\nimport utc from 'dayjs/plugin/utc';\r\nimport timezone from 'dayjs/plugin/timezone';\r\nimport relativeTime from 'dayjs/plugin/relativeTime';\r\nimport advancedFormat from 'dayjs/plugin/advancedFormat';\r\nimport 'dayjs/locale/zh-cn';\r\nimport { zeroPad } from '../number';\r\n\r\ndayjs.extend(customParseFormat);\r\ndayjs.extend(utc);\r\ndayjs.extend(timezone);\r\ndayjs.extend(relativeTime);\r\ndayjs.extend(advancedFormat);\r\ndayjs.locale('zh-cn');\r\n\r\ntype BaseTime = number | string | Date | dayjs.Dayjs | null | undefined;\r\n\r\n/**\r\n * 创建 dayjs 实例\r\n * 文档: https://day.js.org/zh-CN/\r\n * @param t 各种规范或不规范的时间\r\n * @returns dayjs 实例\r\n * @example\r\n * const d = toDayjs('2021-01-01'); // dayjs 实例 (无参,则默认当前时间)\r\n * d.format('YYYY-MM-DD HH:mm:ss'); // \"2025-12-10 11:33:16\"\r\n * d.valueOf(); // 毫秒时间戳,如 1765337596913\r\n * d.unix(); // 秒时间戳,如 1765337596\r\n * d.millisecond(); // 毫秒 913\r\n * d.second(); // 秒 16\r\n * d.minute(); // 分 33\r\n * d.hour(); // 时 11\r\n * d.date(); // 日 10\r\n * d.day(); // 星期几 5(周日=0)\r\n * d.month() + 1; // 月 12\r\n * d.year(); // 年 2025\r\n * d.startOf('day').valueOf(); // 当日零点\r\n * d.startOf('month').format('YYYY-MM-DD HH:mm:ss'); // 月初 \"2025-12-01 00:00:00\"\r\n * d.endOf('month').format('YYYY-MM-DD HH:mm:ss'); // 月末 \"2025-12-31 23:59:59\"\r\n * d.fromNow(); // “刚刚”、“x分钟前/后”、“x小时前/后”、“x天前/后”、“x月前/后”、“x年前/后”\r\n * d.isSame(t, 'day'); // 是否与t在同一天\r\n * d.diff(); // 与当前时间相差的毫秒数\r\n * d.diff(t); // 与t相差的毫秒数\r\n * d.diff(t, 'second'); // 与t相差的秒数\r\n * d.diff(t, 'minute'); // 与t相差的分钟数\r\n * d.diff(t, 'hour'); // 与t相差的小时数\r\n * d.diff(t, 'day'); // 与t相差的天数\r\n * d.diff(t, 'week'); // 与t相差的周数\r\n * d.diff(t, 'month'); // 与t相差的月数\r\n * d.diff(t, 'quarter'); // 与t相差的季度数\r\n * d.diff(t, 'year'); // 与t相差的年数\r\n */\r\nexport function toDayjs(t?: BaseTime, fmt?: dayjs.OptionType) {\r\n if (t === null || t === undefined) return dayjs();\r\n if (typeof t === 'number') {\r\n const s = String(Math.trunc(t));\r\n return dayjs(s.length === 10 ? t * 1000 : t, fmt);\r\n }\r\n if (typeof t === 'string') {\r\n const s = t.trim();\r\n if (/^\\d{10}$/.test(s)) return dayjs(Number(s) * 1000, fmt);\r\n if (/^\\d{13}$/.test(s)) return dayjs(Number(s), fmt);\r\n if (/^\\d{4}-\\d{2}-\\d{2}$/.test(s)) return dayjs(s, fmt || 'YYYY-MM-DD');\r\n if (/^\\d{4}\\/\\d{2}\\/\\d{2}$/.test(s)) return dayjs(s, fmt || 'YYYY/MM/DD');\r\n if (/^\\d{4}-\\d{2}-\\d{2}\\s+\\d{2}:\\d{2}:\\d{2}$/.test(s))\r\n return dayjs(s, fmt || 'YYYY-MM-DD HH:mm:ss');\r\n if (/^\\d{4}\\/\\d{2}\\/\\d{2}\\s+\\d{2}:\\d{2}:\\d{2}$/.test(s))\r\n return dayjs(s, fmt || 'YYYY/MM/DD HH:mm:ss');\r\n return dayjs(s, fmt);\r\n }\r\n return dayjs(t, fmt);\r\n}\r\n\r\n/**\r\n * 获取“前几天”的日期范围\r\n * @param offset 正整数天数\r\n * @param fmt 日期格式,默认 `YYYY-MM-DD`\r\n * @returns `[start, end]` 日期字符串数组\r\n * @example\r\n * 若今天为 2025-11-19:\r\n * getDateRangeBefore(1) // ['2025-11-18', '2025-11-19']\r\n * getDateRangeBefore(1, 'YYYY-MM-DD HH:mm:ss') // ['2025-11-18 00:00:00', '2025-11-19 23:59:59']\r\n */\r\nexport function getDateRangeBefore(offset: number, fmt = 'YYYY-MM-DD') {\r\n const now = toDayjs(Date.now());\r\n const n = Math.max(0, Math.trunc(offset));\r\n const hasTime = /H|h|m|s|S|A|a|x|X/.test(fmt);\r\n const startDay = now.add(-n, 'day');\r\n const endDay = now;\r\n const start = (hasTime ? startDay.startOf('day') : startDay).format(fmt);\r\n const end = (hasTime ? endDay.endOf('day') : endDay).format(fmt);\r\n return [start, end];\r\n}\r\n\r\n/**\r\n * 获取“后几天”的日期范围\r\n * - 起点:今天;终点:`offset` 天后\r\n * - 若 `fmt` 含时间令牌(如 `HH:mm:ss`),则返回整日范围:起点为当日零点,终点为当日末尾\r\n * @param offset 正整数天数\r\n * @param fmt 日期格式,默认 `YYYY-MM-DD`\r\n * @returns `[start, end]` 日期字符串数组\r\n * @example\r\n * 若今天为 2025-11-19:\r\n * getDateRangeAfter(1) // ['2025-11-19', '2025-11-20']\r\n * getDateRangeAfter(1, 'YYYY-MM-DD HH:mm:ss') // ['2025-11-19 00:00:00', '2025-11-20 23:59:59']\r\n */\r\nexport function getDateRangeAfter(offset: number, fmt = 'YYYY-MM-DD') {\r\n const now = toDayjs(Date.now());\r\n const n = Math.max(0, Math.trunc(offset));\r\n const hasTime = /H|h|m|s|S|A|a|x|X/.test(fmt);\r\n const startDay = now;\r\n const endDay = now.add(n, 'day');\r\n const start = (hasTime ? startDay.startOf('day') : startDay).format(fmt);\r\n const end = (hasTime ? endDay.endOf('day') : endDay).format(fmt);\r\n return [start, end];\r\n}\r\n\r\n/**\r\n * 获取倒计时的时间分解(零填充字符串)\r\n * @param diff 毫秒差值(正数表示剩余时间,负数/0表示已到期)\r\n * @returns 包含天、时、分、秒、毫秒的零填充对象\r\n * @example\r\n * const diff = toDayjs(t).diff(); // 毫秒差值\r\n * const parts = getCountdownParts(diff); // { d: '00', h: '00', m: '00', s: '00', ms: '000' }\r\n */\r\nexport function getCountdownParts(diff: number) {\r\n if (diff <= 0) return { d: '00', h: '00', m: '00', s: '00', ms: '000' };\r\n\r\n const d = Math.floor(diff / (1000 * 60 * 60 * 24));\r\n const h = Math.floor((diff / (1000 * 60 * 60)) % 24);\r\n const m = Math.floor((diff / (1000 * 60)) % 60);\r\n const s = Math.floor((diff / 1000) % 60);\r\n const ms = diff % 1000;\r\n\r\n return {\r\n d: zeroPad(d),\r\n h: zeroPad(h),\r\n m: zeroPad(m),\r\n s: zeroPad(s),\r\n ms: zeroPad(ms, 3),\r\n };\r\n}\r\n\r\n/**\r\n * 通过出生日期计算年龄\r\n * @param birthdate 生日日期,支持多种格式(会被自动解析)\r\n * @returns 年龄对象,包含 `age`(年龄数值)和 `type`(年龄单位,'year' 表示年,'month' 表示月)\r\n * @example\r\n * // 假设当前日期为 2025-11-19\r\n * getAgeByBirthdate('2025-05-10'); // { age: 6, type: 'month' }\r\n * getAgeByBirthdate('2020-11-19'); // { age: 5, type: 'year' }\r\n * getAgeByBirthdate('2020-12-01'); // { age: 4, type: 'year' }(生日还没到, 所以年龄是4岁)\r\n */\r\nexport function getAgeByBirthdate(birthdate: string) {\r\n const birth = toDayjs(birthdate, 'YYYY-MM-DD');\r\n const now = toDayjs(Date.now());\r\n\r\n // 精确的月份计算\r\n const totalMonths = (now.year() - birth.year()) * 12 + (now.month() - birth.month());\r\n\r\n // 如果当前日期小于出生日期,月份减1\r\n const adjustedMonths = now.date() < birth.date() ? totalMonths - 1 : totalMonths;\r\n\r\n if (adjustedMonths >= 12) {\r\n let age = Math.floor(adjustedMonths / 12);\r\n // 检查生日是否已过\r\n const birthdayThisYear = birth.add(age, 'year');\r\n if (now.isBefore(birthdayThisYear)) {\r\n age--;\r\n }\r\n return { age, type: 'year' };\r\n }\r\n\r\n return { age: adjustedMonths, type: 'month' };\r\n}\r\n\r\n/**\r\n * 对外抛出 dayjs 以便全局配置\r\n * @example\r\n * 切换语言\r\n * dayjs.locale('en'); // 切换为英文\r\n * dayjs.locale('zh-cn'); // 切换为中文 (默认)\r\n */\r\nexport { dayjs };\r\n","/**\r\n * re-export 全量 es-toolkit\r\n * 版本: 1.44.0\r\n * 文档: https://es-toolkit.dev/\r\n * 目的: 提供常用工具,收敛依赖版本\r\n *\r\n * 注意: 此文件由 scripts/update-es-toolkit-exports.ts 生成, 请勿手动修改.\r\n * 如需更新, 请先升级es-toolkit, 然后运行: npm run update:es-toolkit\r\n */\r\nexport {\r\n AbortError,\r\n Mutex,\r\n Semaphore,\r\n TimeoutError,\r\n after,\r\n ary,\r\n assert,\r\n asyncNoop,\r\n at,\r\n attempt,\r\n attemptAsync,\r\n before,\r\n camelCase,\r\n capitalize,\r\n chunk,\r\n clamp,\r\n clone,\r\n cloneDeep,\r\n cloneDeepWith,\r\n compact,\r\n constantCase,\r\n countBy,\r\n curry,\r\n curryRight,\r\n debounce,\r\n deburr,\r\n delay,\r\n difference,\r\n differenceBy,\r\n differenceWith,\r\n drop,\r\n dropRight,\r\n dropRightWhile,\r\n dropWhile,\r\n escape,\r\n escapeRegExp,\r\n fill,\r\n filterAsync,\r\n findKey,\r\n flatMap,\r\n flatMapAsync,\r\n flatMapDeep,\r\n flatten,\r\n flattenDeep,\r\n flattenObject,\r\n flow,\r\n flowRight,\r\n forEachAsync,\r\n forEachRight,\r\n groupBy,\r\n head,\r\n identity,\r\n inRange,\r\n initial,\r\n intersection,\r\n intersectionBy,\r\n intersectionWith,\r\n invariant,\r\n invert,\r\n isArrayBuffer,\r\n isBlob,\r\n isBoolean,\r\n isBrowser,\r\n isBuffer,\r\n isDate,\r\n isEmptyObject,\r\n isEqual,\r\n isEqualWith,\r\n isError,\r\n isFile,\r\n isFunction,\r\n isJSON,\r\n isJSONArray,\r\n isJSONObject,\r\n isJSONValue,\r\n isLength,\r\n isMap,\r\n isNil,\r\n isNode,\r\n isNotNil,\r\n isNull,\r\n isNumber,\r\n isPlainObject,\r\n isPrimitive,\r\n isPromise,\r\n isRegExp,\r\n isSet,\r\n isString,\r\n isSubset,\r\n isSubsetWith,\r\n isSymbol,\r\n isTypedArray,\r\n isUndefined,\r\n isWeakMap,\r\n isWeakSet,\r\n kebabCase,\r\n keyBy,\r\n last,\r\n limitAsync,\r\n lowerCase,\r\n lowerFirst,\r\n mapAsync,\r\n mapKeys,\r\n mapValues,\r\n maxBy,\r\n mean,\r\n meanBy,\r\n median,\r\n medianBy,\r\n memoize,\r\n merge,\r\n mergeWith,\r\n minBy,\r\n negate,\r\n noop,\r\n omit,\r\n omitBy,\r\n once,\r\n orderBy,\r\n pad,\r\n partial,\r\n partialRight,\r\n partition,\r\n pascalCase,\r\n pick,\r\n pickBy,\r\n pull,\r\n pullAt,\r\n random,\r\n randomInt,\r\n range,\r\n rangeRight,\r\n reduceAsync,\r\n remove,\r\n rest,\r\n retry,\r\n reverseString,\r\n round,\r\n sample,\r\n sampleSize,\r\n shuffle,\r\n snakeCase,\r\n sortBy,\r\n spread,\r\n startCase,\r\n sum,\r\n sumBy,\r\n tail,\r\n take,\r\n takeRight,\r\n takeRightWhile,\r\n takeWhile,\r\n throttle,\r\n timeout,\r\n toCamelCaseKeys,\r\n toFilled,\r\n toMerged,\r\n toSnakeCaseKeys,\r\n trim,\r\n trimEnd,\r\n trimStart,\r\n unary,\r\n unescape,\r\n union,\r\n unionBy,\r\n unionWith,\r\n uniq,\r\n uniqBy,\r\n uniqWith,\r\n unzip,\r\n unzipWith,\r\n upperCase,\r\n upperFirst,\r\n windowed,\r\n withTimeout,\r\n without,\r\n words,\r\n xor,\r\n xorBy,\r\n xorWith,\r\n zip,\r\n zipObject,\r\n zipWith,\r\n} from 'es-toolkit';\r\n\r\n// 导出类型\r\nexport type * from 'es-toolkit';\r\n","import { get, set } from 'es-toolkit/compat';\r\n\r\n/**\r\n * 获取对象键名数组(类型安全)。\r\n * 注:内置 `Object.keys` 与 `es-toolkit` 的 `keys` 在 TS 中通常返回 `string[]`,无法精确到 `keyof T`。\r\n * @param obj 目标对象\r\n * @returns 类型精确的 `Array<keyof T>`\r\n * @example\r\n * const o = { a: 1, b: 'x' };\r\n * const keys = getObjectKeys(o); // type: ('a' | 'b')[], value: ['a','b']\r\n */\r\nexport function getObjectKeys<T extends object>(obj: T): Array<keyof T> {\r\n return Object.keys(obj) as (keyof T)[];\r\n}\r\n\r\n/**\r\n * 获取对象值。\r\n * @param obj 目标对象\r\n * @param path 路径。(点路径:'a.b'、'a[0].b';数组路径:['a',0,'b'])\r\n * @returns 值\r\n * @example\r\n * const o = { b: { c: 'x' }, users: [{ name: 'john' }, { name: 'jane' }] };\r\n * const c = getObjectValue(o, 'b.c'); // 点路径: 'x'\r\n * const name0 = getObjectValue(o, 'users[0].name'); // 数组字符: 'john'\r\n * const name1 = getObjectValue(o, ['users', 1, 'name']); // 数组路径: 'jane'\r\n */\r\nexport const getObjectValue = get;\r\n\r\n/**\r\n * 设置对象值。\r\n * @param obj 目标对象\r\n * @param path 路径。(点路径:'a.b'、'a[0].b';数组路径:['a',0,'b'])\r\n * @param value 值\r\n * @example\r\n * const o = { b: { c: 'x' }, users: [{ name: 'john' }, { name: 'jane' }] };\r\n * setObjectValue(o, 'b.c', 'y'); // 点路径\r\n * setObjectValue(o, ['users', 1, 'name'], 'jane-doe'); // 数组路径\r\n */\r\nexport const setObjectValue = set;\r\n","/**\r\n * 将对象参数拼接到 URL\r\n * - 采用纯JS拼接,因为小程序不支持URLSearchParams\r\n * @param url 基础地址\r\n * @param param 将要追加的参数对象;`null/undefined` 值会被忽略,Object 会使用 `JSON.stringify`\r\n * @returns 拼接后的完整 URL(保留原有哈希片段)\r\n * @example\r\n * const url = appendUrlParam('https://a.com', { q: '测试', list: [1, 2], a: null, b: undefined }); // 'https://a.com/?q=%E6%B5%8B%E8%AF%95&list=[1,2]'\r\n */\r\nexport function appendUrlParam(url: string, param: Record<string, unknown>) {\r\n if (!param || typeof param !== 'object') return url;\r\n\r\n const hashIndex = url.indexOf('#');\r\n const baseWithoutHash = hashIndex >= 0 ? url.slice(0, hashIndex) : url;\r\n const hash = hashIndex >= 0 ? url.slice(hashIndex) : '';\r\n\r\n const [base, existingQs] = baseWithoutHash.split('?');\r\n const parts: string[] = [];\r\n if (existingQs) parts.push(existingQs);\r\n for (const key in param) {\r\n const rawVal = param[key];\r\n if (rawVal === null || rawVal === undefined) continue;\r\n const val = typeof rawVal === 'object' ? JSON.stringify(rawVal) : String(rawVal);\r\n parts.push(`${encodeURIComponent(key)}=${encodeURIComponent(val)}`);\r\n }\r\n const qs = parts.filter(Boolean).join('&');\r\n return base + (qs ? `?${qs}` : '') + hash;\r\n}\r\n","import {\r\n appendUrlParam,\r\n cloneDeep,\r\n getObjectValue,\r\n isPlainObject,\r\n toDayjs,\r\n} from '@base-web-kits/base-tools-ts';\r\nimport { getBaseToolsConfig } from '../config';\r\nimport type { AppLogInfo } from '../config';\r\n\r\n/** 请求方法类型 */\r\nexport type RequestMethod =\r\n | 'GET'\r\n | 'POST'\r\n | 'PUT'\r\n | 'DELETE'\r\n | 'CONNECT'\r\n | 'HEAD'\r\n | 'OPTIONS'\r\n | 'TRACE'\r\n | 'PATCH';\r\n\r\n/**\r\n * 请求参数类型\r\n * 包含 fetch 原生支持的 BodyInit 类型,以及支持自动 JSON 序列化的对象和数组\r\n */\r\nexport type RequestData =\r\n | string\r\n | ArrayBuffer\r\n | ArrayBufferView\r\n | Blob\r\n | FormData\r\n | URLSearchParams\r\n | ReadableStream<Uint8Array>\r\n | Record<string, unknown>\r\n | unknown[]\r\n | null;\r\n\r\n/**\r\n * 响应数据类型\r\n */\r\nexport type ResponseData = string | ArrayBuffer | Blob | Record<string, unknown> | unknown[] | null;\r\n\r\n/**\r\n * 发起请求的配置 (对外,参数可选)\r\n */\r\nexport type RequestConfig<D extends RequestData = RequestData> = Partial<RequestConfigBase<D>>;\r\n\r\n/**\r\n * 自定义请求的配置 (接口字段参数必填)\r\n */\r\nexport type RequestConfigBase<D extends RequestData = RequestData> = {\r\n /** 接口地址 */\r\n url: string;\r\n\r\n /** 请求方法 */\r\n method?: RequestMethod;\r\n\r\n /** 请求头(会自动过滤undefined, null, \"\";不过滤0和false; 数字和布尔值会自动转换为字符串) */\r\n header?: Record<string, string | number | boolean | null | undefined>;\r\n\r\n /** 请求参数 */\r\n data?: D;\r\n\r\n /** 超时时间 (毫秒), 默认 60000 */\r\n timeout?: number;\r\n\r\n /** 接口返回响应数据的字段, 支持\"a[0].b.c\"的格式, 当配置false时返回完整的响应数据 */\r\n resKey: string | false;\r\n\r\n /** 接口返回响应消息的字段, 支持\"a[0].b.c\"的格式 */\r\n msgKey: string;\r\n\r\n /** 接口返回响应状态码的字段, 支持\"a[0].b.c\"的格式 */\r\n codeKey: string;\r\n\r\n /** 接口返回成功状态码的字段, 支持\"a[0].b.c\"的格式 (默认取 codeKey) */\r\n successKey?: string;\r\n\r\n /** 成功状态码 */\r\n successCode: (number | string)[];\r\n\r\n /** 登录过期状态码 */\r\n reloginCode: (number | string)[];\r\n\r\n /** 是否开启流式传输 (如 SSE) */\r\n enableChunked?: boolean;\r\n\r\n /** 响应类型 (默认 json, enableChunked为true时忽略) */\r\n responseType?: 'text' | 'arraybuffer' | 'json';\r\n\r\n /** 响应数据的缓存时间, 单位毫秒。仅在成功时缓存;仅缓存在内存,应用退出,缓存消失。(默认0,不开启缓存) */\r\n cacheTime?: number;\r\n\r\n /** 是否提示接口异常 (默认true) */\r\n toastError?: boolean;\r\n\r\n /** 是否显示进度条: 支持字符串,自定义文本 (默认true) */\r\n showLoading?: boolean | string;\r\n\r\n /** 是否输出日志 (默认true) */\r\n showLog?: boolean;\r\n\r\n /** 成功和失败时,额外输出的日志数据 (可覆盖内部log参数,如'name') */\r\n logExtra?: Record<string, unknown>;\r\n\r\n /** 响应数据的转换 */\r\n resMap?: (data: ResponseData) => ResponseData;\r\n\r\n /** 获取task对象, 用于取消请求或监听流式数据 */\r\n onTaskReady?: (task: RequestTask) => void;\r\n};\r\n\r\n/**\r\n * 请求任务对象 (用于取消请求或监听流式数据)\r\n */\r\nexport interface RequestTask {\r\n /** 取消请求 */\r\n abort: () => void;\r\n\r\n /** 监听流式数据块接收事件 */\r\n onChunkReceived: (callback: ChunkCallback) => void;\r\n\r\n /** 取消监听流式数据块接收事件 */\r\n offChunkReceived: () => void;\r\n}\r\n\r\n/**\r\n * 流式数据块接收事件回调\r\n */\r\nexport type ChunkCallback = (response: { data: ArrayBuffer }) => void;\r\n\r\n/** 请求缓存 */\r\nconst requestCache = new Map<string, { res: unknown; expire: number }>();\r\n\r\n/**\r\n * 基础请求 (返回 Promise 和 Task 对象)\r\n * 基于 fetch API 封装,支持流式请求\r\n * @param config 请求配置\r\n * @example\r\n * // 在入口文件完成配置 (确保请求失败有toast提示,登录过期能够触发重新登录,log有日志输出)\r\n * setBaseToolsConfig({\r\n * toast: ({ msg, status }) => (status === 'fail' ? message.error(msg) : message.success(msg)),\r\n * showLoading: ({ title }) => message.loading(title || '加载中...'),\r\n * hideLoading: () => message.destroy(),\r\n * toLogin: () => reLogin(),\r\n * log(level, data) {\r\n * if (data.name === 'request') {\r\n * sendLog('request', data); // 请求日志\r\n * } else if (level === 'error') {\r\n * sendLog('error', data); // 错误日志\r\n * } else {\r\n * sendLog('action', data); // 操作日志\r\n * }\r\n * },\r\n * });\r\n *\r\n * // 封装项目的基础请求\r\n * export function requestApi<T>(config: RequestConfig) {\r\n * return request<T>({\r\n * header: { token: 'xx', version: 'xx', tid: 'xx' }, // 会自动过滤空值\r\n * // resMap: (res) => res, // 响应数据的转换, 如解密操作 (可选)\r\n * resKey: 'data',\r\n * msgKey: 'message',\r\n * codeKey: 'status',\r\n * successCode: [1],\r\n * reloginCode: [-10],\r\n * ...config,\r\n * });\r\n * }\r\n *\r\n * // 1. 基于上面 requestApi 的普通接口\r\n * export function apiGoodList(data: { page: number, size: number }) {\r\n * return requestApi<GoodItem[]>({ url: '/goods/list', data, resKey: 'data.list' });\r\n * }\r\n *\r\n * const goodList = await apiGoodList({ page:1, size:10 });\r\n *\r\n * // 2. 参数泛型的写法\r\n * export function apiGoodList(config: RequestConfig<{ page: number, size: number }>) {\r\n * return requestApi<GoodItem[]>({ url: '/goods/list', resKey: 'data.list', ...config });\r\n * }\r\n *\r\n * const goodList = await apiGoodList({ data: { page:1, size:10 } });\r\n *\r\n * // 3. 基于上面 requestApi 的流式接口\r\n * export function apiChatStream(config: RequestConfig) {\r\n * return requestApi({\r\n * ...config,\r\n * url: '/sse/chatStream',\r\n * resKey: false,\r\n * showLoading: false,\r\n * responseType: 'arraybuffer', // 流式响应类型\r\n * enableChunked: true, // 开启分块传输\r\n * });\r\n * }\r\n *\r\n * // 流式监听\r\n * const onTaskReady = (task: RequestTask) => {\r\n * task.onChunkReceived((res) => {\r\n * console.log('ArrayBuffer', res.data);\r\n * });\r\n * }\r\n *\r\n * // 流式发起\r\n * const data = { content: '你好', chatId: 123 };\r\n * await apiChatStream({ data, onTaskReady });\r\n *\r\n * // 流式取消 (在组件销毁或页面关闭时调用)\r\n * task?.offChunkReceived(); // 取消监听,中断流式接收\r\n * task?.abort(); // 取消请求 (若流式已生成,此时abort无效,因为请求已成功)\r\n */\r\nexport function request<T, D extends RequestData = RequestData>(config: RequestConfigBase<D>) {\r\n const {\r\n url,\r\n data,\r\n header,\r\n method = 'GET',\r\n resKey,\r\n msgKey,\r\n codeKey,\r\n successKey,\r\n successCode,\r\n reloginCode,\r\n showLoading = true,\r\n toastError = true,\r\n enableChunked = false,\r\n cacheTime,\r\n resMap,\r\n responseType = 'json',\r\n timeout = 60000,\r\n onTaskReady,\r\n } = config;\r\n\r\n // 1. 初始化控制对象\r\n const controller = new AbortController();\r\n const signal = controller.signal;\r\n let chunkCallback: ChunkCallback | null = null;\r\n\r\n // 构造 Task 对象\r\n const task: RequestTask = {\r\n abort: () => controller.abort(),\r\n onChunkReceived: (cb) => {\r\n chunkCallback = cb;\r\n },\r\n offChunkReceived: () => {\r\n chunkCallback = null;\r\n },\r\n };\r\n onTaskReady?.(task);\r\n\r\n // 2. 创建 Promise\r\n return new Promise<T>((resolve, reject) => {\r\n const execute = async () => {\r\n const isGet = method === 'GET';\r\n const isObjectData = isPlainObject(data);\r\n const isArrayData = !isObjectData && Array.isArray(data);\r\n\r\n // 2.1 参数处理\r\n // 参数过滤 undefined\r\n const fillData = isObjectData ? filterRequestData(data) : data;\r\n\r\n // 请求头过滤空值 (undefined, null, \"\")\r\n const fillHeader = filterRequestHeader(header);\r\n\r\n // 获取 Content-Type (忽略大小写)\r\n const contentTypeKey = Object.keys(fillHeader).find(\r\n (k) => k.toLowerCase() === 'content-type',\r\n );\r\n const contentType = contentTypeKey ? String(fillHeader[contentTypeKey]).toLowerCase() : '';\r\n\r\n if (!isGet && fillData && (isObjectData || isArrayData) && !contentType) {\r\n fillHeader['Content-Type'] = 'application/json';\r\n }\r\n\r\n // 2.2 处理 URL 和 Body\r\n const fillUrl =\r\n isGet && isObjectData ? appendUrlParam(url, fillData as Record<string, unknown>) : url;\r\n\r\n let fillBody: BodyInit | null | undefined;\r\n\r\n if (!isGet && fillData) {\r\n if (isObjectData && contentType.includes('application/x-www-form-urlencoded')) {\r\n // application/x-www-form-urlencoded: 转换为 URLSearchParams\r\n fillBody = toSearchParams(fillData as Record<string, unknown>);\r\n } else if (isObjectData && contentType.includes('multipart/form-data')) {\r\n // multipart/form-data: 转换为 FormData\r\n fillBody = toFormData(fillData as Record<string, unknown>);\r\n // 删除 Content-Type, 让 fetch 自动生成 boundary\r\n if (contentTypeKey) delete fillHeader[contentTypeKey];\r\n } else if (isObjectData || isArrayData) {\r\n fillBody = JSON.stringify(fillData);\r\n } else {\r\n fillBody = fillData as BodyInit;\r\n }\r\n }\r\n\r\n // 2.3 日志与缓存配置\r\n const logConfig = { ...config, data: fillData, header: fillHeader, url: fillUrl };\r\n const startTime = Date.now();\r\n\r\n // 2.4 检查缓存\r\n const isCache = cacheTime && cacheTime > 0;\r\n const cacheKey = isCache ? JSON.stringify({ url: fillUrl, data: fillData }) : '';\r\n\r\n if (isCache) {\r\n const res = checkCache(cacheKey);\r\n if (res) {\r\n logRequestInfo({\r\n status: 'success',\r\n config: logConfig,\r\n fromCache: true,\r\n startTime,\r\n res,\r\n });\r\n resolve(getResult(res, resKey) as T);\r\n return;\r\n }\r\n }\r\n\r\n // 2.5 UI 反馈\r\n const appConfig = getBaseToolsConfig();\r\n if (showLoading)\r\n appConfig.showLoading?.(typeof showLoading === 'string' ? { title: showLoading } : {});\r\n\r\n // 2.6 设置超时\r\n let isTimeout = false;\r\n const timeoutId = setTimeout(() => {\r\n isTimeout = true;\r\n controller.abort();\r\n }, timeout);\r\n\r\n try {\r\n // 2.7 发起请求\r\n const response = await fetch(fillUrl, {\r\n method,\r\n headers: fillHeader,\r\n body: fillBody,\r\n signal,\r\n });\r\n\r\n if (!response.ok) {\r\n if (showLoading) appConfig.hideLoading?.();\r\n throw new Error(`HTTP Error ${response.status}: ${response.statusText}`);\r\n }\r\n\r\n // 2.8 处理流式响应\r\n if (enableChunked) {\r\n if (showLoading) appConfig.hideLoading?.();\r\n\r\n const res = await handleStreamResponse(response, chunkCallback);\r\n\r\n logRequestInfo({ status: 'success', config: logConfig, startTime, res });\r\n\r\n resolve(res as T);\r\n return;\r\n }\r\n\r\n // 2.9 处理普通响应\r\n const resData = await parseResponse(response, responseType);\r\n\r\n // 隐藏 Loading\r\n if (showLoading) appConfig.hideLoading?.();\r\n\r\n // 响应拦截\r\n const res = resMap ? resMap(resData) : resData;\r\n\r\n // 2.10 业务状态码解析\r\n const code = getObjectValue(res, codeKey);\r\n const scode = successKey ? getObjectValue(res, successKey) : code;\r\n const msg = getObjectValue(res, msgKey);\r\n const isSuccess = successCode.includes(scode);\r\n const isRelogin = reloginCode.includes(code);\r\n\r\n logRequestInfo({ status: 'success', config: logConfig, startTime, res });\r\n\r\n // 2.11 结果处理\r\n if (isSuccess) {\r\n // 业务正常\r\n if (isCache) requestCache.set(cacheKey, { res, expire: Date.now() + cacheTime });\r\n resolve(getResult(res, resKey) as T);\r\n } else if (isRelogin) {\r\n // 登录失效\r\n reject(res);\r\n appConfig.toLogin?.(); // 放在后面,确保reject执行后再跳转登录\r\n } else {\r\n // 业务错误\r\n if (toastError && msg) appConfig.toast?.({ status: 'fail', msg });\r\n reject(res);\r\n }\r\n } catch (e) {\r\n const status = 'fail';\r\n const isAbortError = e instanceof DOMException && e.name === 'AbortError'; // 取消请求不视为错误\r\n\r\n if (isAbortError && isTimeout) {\r\n if (toastError) appConfig.toast?.({ status, msg: '请求超时' });\r\n const timeoutError = new Error('Request Timeout');\r\n logRequestInfo({ status, config: logConfig, startTime, e: timeoutError });\r\n reject(timeoutError);\r\n return;\r\n }\r\n\r\n if (!isAbortError && toastError) appConfig.toast?.({ status, msg: '网络请求失败' });\r\n logRequestInfo({ status, config: logConfig, startTime, e });\r\n reject(e);\r\n } finally {\r\n if (timeoutId) clearTimeout(timeoutId);\r\n }\r\n };\r\n\r\n execute();\r\n });\r\n}\r\n\r\n/**\r\n * 参数过滤undefined, 避免接口处理异常 (不可过滤 null 、 \"\" 、 false 、 0 这些有效值)\r\n */\r\nexport function filterRequestData(data: Record<string, any>) {\r\n const res: Record<string, any> = {};\r\n Object.entries(data).forEach(([k, v]) => {\r\n if (v !== undefined) res[k] = v;\r\n });\r\n return res;\r\n}\r\n\r\n/**\r\n * 请求头过滤空值 (undefined, null, \"\"), 不过滤0和false\r\n */\r\nexport function filterRequestHeader(header: RequestConfigBase['header']) {\r\n const newHeader: Record<string, string> = {};\r\n if (header) {\r\n Object.entries(header).forEach(([k, v]) => {\r\n if (v !== undefined && v !== null && v !== '') newHeader[k] = String(v);\r\n });\r\n }\r\n return newHeader;\r\n}\r\n\r\n/**\r\n * 日志输出\r\n */\r\nfunction logRequestInfo(options: {\r\n config: RequestConfigBase<RequestData> & { url?: string };\r\n fromCache?: boolean;\r\n startTime: number;\r\n status: 'success' | 'fail';\r\n res?: unknown;\r\n e?: unknown;\r\n}) {\r\n const { log } = getBaseToolsConfig();\r\n const { showLog = true } = options.config;\r\n\r\n if (!log || !showLog) return;\r\n\r\n const { config, res, fromCache = false, startTime, status, e } = options;\r\n const { url, data, header, method, logExtra } = config;\r\n const endTime = Date.now();\r\n const fmt = 'YYYY-MM-DD HH:mm:ss.SSS';\r\n\r\n const info: AppLogInfo = {\r\n name: 'request',\r\n status,\r\n url,\r\n data,\r\n method,\r\n header,\r\n fromCache,\r\n startTime: toDayjs(startTime).format(fmt),\r\n endTime: toDayjs(endTime).format(fmt),\r\n duration: endTime - startTime,\r\n ...logExtra,\r\n };\r\n\r\n if (status === 'success') {\r\n info.res = cloneDeep(res); // 深拷贝,避免外部修改对象,造成输出不一致\r\n log('info', info);\r\n } else {\r\n info.e = e;\r\n log('error', info);\r\n }\r\n}\r\n\r\n/**\r\n * 获取 resKey 对应的数据\r\n */\r\nfunction getResult(res: unknown, resKey?: RequestConfigBase['resKey']) {\r\n if (!res || !resKey || typeof res !== 'object') return res;\r\n return getObjectValue(res, resKey);\r\n}\r\n\r\n/**\r\n * 检查缓存\r\n */\r\nfunction checkCache(cacheKey: string) {\r\n const cached = requestCache.get(cacheKey);\r\n if (!cached) return null;\r\n if (cached.expire <= Date.now()) {\r\n requestCache.delete(cacheKey);\r\n return null;\r\n }\r\n return cached.res;\r\n}\r\n\r\n/**\r\n * 处理流式响应\r\n */\r\nasync function handleStreamResponse(response: Response, chunkCallback: ChunkCallback | null) {\r\n if (!response.body) throw new Error('Response body is null');\r\n\r\n const reader = response.body.getReader();\r\n\r\n while (true) {\r\n const { done, value } = await reader.read();\r\n if (done) break;\r\n if (chunkCallback && value) {\r\n chunkCallback({ data: value.buffer });\r\n }\r\n }\r\n\r\n return 'Stream Finished';\r\n}\r\n\r\n/**\r\n * 解析响应数据\r\n */\r\nasync function parseResponse(response: Response, responseType: string) {\r\n let resData: ResponseData;\r\n if (responseType === 'arraybuffer') {\r\n resData = await response.arrayBuffer();\r\n } else if (responseType === 'text') {\r\n resData = await response.text();\r\n } else {\r\n const text = await response.text();\r\n try {\r\n resData = JSON.parse(text);\r\n } catch {\r\n resData = text;\r\n }\r\n }\r\n return resData;\r\n}\r\n\r\n/**\r\n * 转换为 URLSearchParams\r\n */\r\nfunction toSearchParams(data: Record<string, unknown>) {\r\n const params = new URLSearchParams();\r\n for (const key in data) {\r\n const val = data[key];\r\n // undefined 已在 fillData 阶段过滤,此处仅需判断 null\r\n // null 在 Form 中会被转为字符串 \"null\",通常不符合预期,故过滤\r\n if (val === null) continue;\r\n if (Array.isArray(val)) {\r\n val.forEach((v) => params.append(key, typeof v === 'object' ? JSON.stringify(v) : String(v)));\r\n } else {\r\n params.append(key, typeof val === 'object' ? JSON.stringify(val) : String(val));\r\n }\r\n }\r\n return params;\r\n}\r\n\r\n/**\r\n * 转换为 FormData\r\n */\r\nfunction toFormData(data: Record<string, unknown>) {\r\n const formData = new FormData();\r\n for (const key in data) {\r\n const val = data[key];\r\n // undefined 已在 fillData 阶段过滤,此处仅需判断 null\r\n // null 在 Form 中会被转为字符串 \"null\",通常不符合预期,故过滤\r\n if (val === null) continue;\r\n if (Array.isArray(val)) {\r\n val.forEach((v) =>\r\n formData.append(\r\n key,\r\n v instanceof Blob ? v : typeof v === 'object' ? JSON.stringify(v) : String(v),\r\n ),\r\n );\r\n } else {\r\n formData.append(\r\n key,\r\n val instanceof Blob ? val : typeof val === 'object' ? JSON.stringify(val) : String(val),\r\n );\r\n }\r\n }\r\n return formData;\r\n}\r\n","import { cloneDeep } from '@base-web-kits/base-tools-ts';\r\nimport { getBaseToolsConfig } from '../index';\r\nimport type { AppLogInfo } from '../index';\r\n\r\ntype WebApi<Option = any, Res = any, Config = any> = (\r\n option: Option,\r\n config?: Config,\r\n) => Promise<Res>;\r\n\r\n/**\r\n * web api 的调用配置\r\n */\r\nexport type WebApiConfig<Res = any, Err = any> = {\r\n /** 是否显示加载提示, 默认 false. (支持字符串,自定义文本) */\r\n showLoading?: boolean | string;\r\n\r\n /** 操作成功的toast提示, 默认不显示 */\r\n toastSuccess?: ((res: Res) => false | string) | false | string;\r\n\r\n /** 是否显示操作失败的详细错误信息, 默认 true. (支持字符串,自定义文本; 支持根据errMsg判断是否显示, 例如: (e) => !e.errMsg.includes('cancel') */\r\n toastError?: ((e: Err) => boolean | string) | boolean | string;\r\n\r\n /** 是否显示日志, 默认 true */\r\n showLog?: boolean;\r\n\r\n /** 响应数据的转换, 如解密操作 (返回值在成功日志中输出'resMap'字段) */\r\n resMap?: (res: any) => Res;\r\n\r\n /** 成功和失败时,额外输出的日志数据 (可覆盖内部log参数,如'name') */\r\n logExtra?: Record<string, unknown>;\r\n};\r\n\r\n/**\r\n * 拓展 web api, 使其支持loading,toast,log能力\r\n * @param webApi web api\r\n * @param apiName web api 名称 (可选, 用于日志输出, 默认'enhanceWebApi')\r\n * @return 注入拓展能力的promise (默认提示异常和输出日志,不显示进度条和操作成功)\r\n * @example\r\n * const promise = enhanceWebApi(downloadFile, 'downloadFile');\r\n * await promise({ url: 'xx' }, {showLoading: '下载中', toastSuccess: '下载成功'});\r\n */\r\nexport function enhanceWebApi<Option = any, Res = any, Err = any, Config = any>(\r\n webApi: WebApi<Option, Res, Config>,\r\n apiName?: string,\r\n) {\r\n return (option: Option, config?: WebApiConfig<Res, Err> & Config) => {\r\n const finalConfig = config || ({} as WebApiConfig<Res, Err> & Config);\r\n const {\r\n showLoading = false,\r\n toastSuccess = false,\r\n toastError = true,\r\n showLog = true,\r\n resMap,\r\n logExtra,\r\n } = finalConfig;\r\n\r\n const {\r\n log,\r\n toast,\r\n showLoading: showLoadingFn,\r\n hideLoading: hideLoadingFn,\r\n } = getBaseToolsConfig();\r\n const fname = apiName || 'enhanceWebApi'; // webApi.name经过打包后取不到原函数名,不如默认'enhanceWebApi'\r\n\r\n if (showLoading) {\r\n const title = typeof showLoading === 'string' ? showLoading : '';\r\n showLoadingFn?.({ title });\r\n }\r\n\r\n return new Promise<Res>((resolve, reject) => {\r\n webApi(option, finalConfig)\r\n .then((res) => {\r\n if (showLoading) hideLoadingFn?.();\r\n\r\n const finalRes = resMap ? resMap(res) : res;\r\n\r\n if (showLog) {\r\n const logData: AppLogInfo = { name: fname, status: 'success', option, ...logExtra };\r\n\r\n if (resMap) {\r\n logData.res = res; // 输出原始数据\r\n logData.resMap = cloneDeep(finalRes); // 深拷贝处理后数据,避免外部修改对象,造成输出不一致\r\n } else {\r\n logData.res = cloneDeep(res); // 深拷贝原始数据,避免外部修改对象,造成输出不一致\r\n }\r\n\r\n log?.('info', logData);\r\n }\r\n\r\n resolve(finalRes);\r\n\r\n const msg = typeof toastSuccess === 'function' ? toastSuccess(finalRes) : toastSuccess;\r\n if (msg) toast?.({ msg, status: 'success' });\r\n })\r\n .catch((e) => {\r\n if (showLoading) hideLoadingFn?.();\r\n if (showLog) log?.('error', { name: fname, status: 'fail', option, e, ...logExtra });\r\n\r\n const msg = typeof toastError === 'function' ? toastError(e) : toastError;\r\n if (msg) {\r\n toast?.({\r\n msg: typeof msg === 'string' ? msg : `${fname} fail: ${JSON.stringify(e)}`,\r\n status: 'fail',\r\n });\r\n }\r\n\r\n reject(e);\r\n });\r\n });\r\n };\r\n}\r\n","import { enhanceWebApi } from '../async';\r\nimport type { WebApiConfig } from '../async';\r\n\r\n/**\r\n * 上传文件的选项\r\n */\r\nexport type UploadFileOption = {\r\n /** 上传接口地址 */\r\n url: string;\r\n\r\n /** 要上传的文件对象 */\r\n file: File;\r\n\r\n /** 文件对应的 key, 默认'file' (服务端通过这个 key 获取文件的二进制内容) */\r\n name?: string;\r\n\r\n /** 请求头 */\r\n header?: Record<string, string | number>;\r\n\r\n /** 额外的formData参数 */\r\n formData?: Record<string, string | number>;\r\n\r\n /** 超时时间,单位 ms,默认 0(不超时) */\r\n timeout?: number;\r\n};\r\n\r\nexport type OnUploadProgressUpdate = (res: UploadProgressEvent) => void;\r\n\r\nexport type UploadProgressEvent = {\r\n /** 上传进度百分比: 0-100 */\r\n progress: number;\r\n /** 已上传字节数 */\r\n loaded: number;\r\n /** 总字节数 */\r\n total: number;\r\n};\r\n\r\nexport type UploadTask = {\r\n /** 上传进度 */\r\n onProgressUpdate: (callback: OnUploadProgressUpdate) => void;\r\n /** 取消上传 */\r\n abort: () => void;\r\n};\r\n\r\nexport type UploadConfig = {\r\n /** 获取task对象 */\r\n onTaskReady?: (task: UploadTask) => void;\r\n};\r\n\r\nexport type UploadFail = {\r\n message: string;\r\n status: number;\r\n};\r\n\r\nfunction upload(option: UploadFileOption, config?: UploadConfig) {\r\n return new Promise<string>((resolve, reject) => {\r\n const xhr = new XMLHttpRequest();\r\n const { url, file, name = 'file', header, formData, timeout = 0 } = option;\r\n\r\n const fail = (error: UploadFail) => reject(error);\r\n\r\n const success = (responseText: string) => {\r\n resolve(responseText);\r\n };\r\n\r\n // 构造任务对象\r\n let onProgressUpdate: OnUploadProgressUpdate;\r\n const task: UploadTask = {\r\n onProgressUpdate: (callback) => {\r\n onProgressUpdate = callback;\r\n },\r\n abort: () => xhr.abort(),\r\n };\r\n config?.onTaskReady?.(task);\r\n\r\n // 监听进度\r\n xhr.upload.onprogress = (e) => {\r\n if (!e.lengthComputable) return;\r\n const ev: UploadProgressEvent = {\r\n progress: Math.round((e.loaded / e.total) * 100),\r\n loaded: e.loaded,\r\n total: e.total,\r\n };\r\n onProgressUpdate?.(ev);\r\n };\r\n\r\n // 监听事件\r\n xhr.onload = () => {\r\n if (xhr.status >= 200 && xhr.status < 300) {\r\n success(xhr.responseText);\r\n } else {\r\n fail({ message: `上传失败`, status: xhr.status });\r\n }\r\n };\r\n xhr.onerror = () => fail({ message: '网络错误', status: 0 });\r\n xhr.ontimeout = () => fail({ message: '上传超时', status: -1 });\r\n xhr.onabort = () => fail({ message: '用户取消', status: -2 });\r\n\r\n // 设置请求方法和 URL\r\n xhr.open('POST', url);\r\n\r\n // 设置请求头\r\n if (header) {\r\n Object.entries(header).forEach(([k, v]) => {\r\n if (v !== undefined && v !== null && v !== '') xhr.setRequestHeader(k, String(v));\r\n });\r\n }\r\n\r\n // 设置超时时间\r\n xhr.timeout = timeout;\r\n\r\n // 组装 FormData\r\n const data = new FormData();\r\n data.append(name, file);\r\n if (formData) {\r\n Object.entries(formData).forEach(([k, v]) => {\r\n if (v !== undefined && v !== null) data.append(k, String(v));\r\n });\r\n }\r\n\r\n // 发送请求\r\n xhr.send(data);\r\n });\r\n}\r\n\r\n/**\r\n * 上传文件\r\n * @param option 上传文件的选项\r\n * @param config 配置项\r\n * @example\r\n * // 上传\r\n * const res = await uploadFile({ url: 'https://xx', file: file});\r\n *\r\n * // 监听上传进度\r\n * const res = await uploadFile({ url: 'https://xx', file: file}, {\r\n * onTaskReady: (task) =>\r\n * task.onProgressUpdate((res) => console.log('上传进度:', res.progress)),\r\n * });\r\n *\r\n * // 解析上传结果\r\n * console.log('uploadFile ok', JSON.parse(res));\r\n */\r\nexport function uploadFile(option: UploadFileOption, config?: UploadConfig & WebApiConfig) {\r\n return enhanceWebApi(upload, 'uploadFile')(option, config);\r\n}\r\n","const WK = {\r\n val: '__l_val',\r\n exp: '__l_exp',\r\n wrap: '__l_wrap',\r\n} as const;\r\n\r\n/**\r\n * 写入 localStorage(自动 JSON 序列化)\r\n * @param key 键名\r\n * @param value 任意可序列化的值:对象、数组、字符串、数字、布尔值。(`null` 或 `undefined` 会自动移除该键)\r\n * @param days 过期天数(从当前时间起算)\r\n * @example\r\n * setLocalStorage('user', { id: 1, name: 'Alice' }); // 对象\r\n * setLocalStorage('age', 18); // 数字\r\n * setLocalStorage('vip', true); // 布尔值\r\n * setLocalStorage('token', 'abc123', 7); // 7 天后过期\r\n */\r\nexport function setLocalStorage(key: string, value: unknown, days?: number) {\r\n if (value === undefined || value === null) {\r\n removeLocalStorage(key);\r\n return;\r\n }\r\n\r\n let toStore: unknown = value;\r\n if (typeof days === 'number' && days > 0) {\r\n const ms = days * 24 * 60 * 60 * 1000;\r\n toStore = {\r\n [WK.wrap]: true,\r\n [WK.val]: value,\r\n [WK.exp]: Date.now() + ms,\r\n };\r\n }\r\n\r\n localStorage.setItem(key, JSON.stringify(toStore));\r\n}\r\n\r\n/**\r\n * 读取 localStorage(自动 JSON 反序列化)\r\n * 若值为合法 JSON,则返回反序列化后的数据;\r\n * 若值非 JSON(如外部写入的纯字符串),则原样返回字符串。\r\n * 不存在时返回 `null`。\r\n * @param key 键名\r\n * @returns 解析后的值或 `null`\r\n * @example\r\n * const user = getLocalStorage<{ id: number; name: string }>('user');\r\n * const age = getLocalStorage<number>('age');\r\n * const vip = getLocalStorage<boolean>('vip');\r\n */\r\nexport function getLocalStorage<T = unknown>(key: string): T | null {\r\n const raw = localStorage.getItem(key);\r\n if (raw === null) return null;\r\n try {\r\n const parsed = JSON.parse(raw);\r\n\r\n if (parsed && typeof parsed === 'object' && WK.wrap in parsed && WK.exp in parsed) {\r\n if (Date.now() > parsed[WK.exp]) {\r\n removeLocalStorage(key);\r\n return null;\r\n }\r\n return parsed[WK.val] as T;\r\n }\r\n return parsed as T;\r\n } catch {\r\n return raw as T;\r\n }\r\n}\r\n\r\n/**\r\n * 移除 localStorage 指定键\r\n * @param key 键名\r\n * @example\r\n * removeLocalStorage('token');\r\n */\r\nexport function removeLocalStorage(key: string) {\r\n localStorage.removeItem(key);\r\n}\r\n","/**\r\n * 获取url的查询参数值\r\n * @param key 参数名\r\n * @param url 完整 URL 或仅查询串(如 \"a=1&b=2\")\r\n * @returns 解码后的参数值 (若不存在|\"null\"|\"undefined\",则返回 null)\r\n * @example\r\n * const q = getUrlParam('q'); // 默认当前地址\r\n * const q = getUrlParam('q', 'https://a.com/?q=%E6%B5%8B%E8%AF%95'); // \"测试\"\r\n * const a = getUrlParam('a', 'a=1'); // \"1\"\r\n * const list = getUrlParam('list', 'list=[1,2]'); // \"[1,2]\"\r\n * const list = getUrlParam('list', 'list=null'); // null\r\n * const list = getUrlParam('list', 'list=undefined'); // null\r\n */\r\nexport function getUrlParam(key: string, url = window.location.href) {\r\n const searchParams = new URL(url.includes('?') ? url : `?${url}`, 'http://localhost')\r\n .searchParams;\r\n const value = searchParams.get(key);\r\n return value === 'null' || value === 'undefined' ? null : value;\r\n}\r\n\r\n/**\r\n * 获取url的查询参数值,并转为number类型\r\n * @param key 参数名\r\n * @param url 完整 URL 或仅查询串(如 \"a=1&b=2\")\r\n * @returns 解码后的参数值 (若不存在|\"非数字字符串\",则返回 null)\r\n * @example\r\n * const a = getUrlNumber('a'); // 默认当前地址\r\n * const a = getUrlNumber('a', 'https://a.com/?a=1'); // 1\r\n * const a = getUrlNumber('a', 'a=1'); // 1\r\n * const a = getUrlNumber('a', 'a=1.2'); // 1.2\r\n * const a = getUrlNumber('a', 'a=abc'); // null\r\n */\r\nexport function getUrlNumber(key: string, url = window.location.href) {\r\n const str = getUrlParam(key, url);\r\n if (!str) return null;\r\n\r\n const num = Number(str);\r\n return isNaN(num) ? null : num;\r\n}\r\n\r\n/**\r\n * 获取url的所有查询参数值\r\n * @param url 完整 URL 或仅查询串(如 \"a=1&b=2\")\r\n * @returns 解码后的键值对象(无参数返回空对象; \"null\"|\"undefined\"的参数会被忽略)\r\n * @example\r\n * const params = getUrlParams(); // 默认当前地址\r\n * const params = getUrlParams('a=1&b=2'); // { a: \"1\", b: \"2\" }\r\n * const params = getUrlParams('https://a.com/?a=1&b=2'); // { a: \"1\", b: \"2\" }\r\n * const params = getUrlParams('a=1&b=null'); // { a: \"1\" }\r\n * const params = getUrlParams('a=1&b=undefined'); // { a: \"1\" }\r\n */\r\nexport function getUrlParams(url = window.location.href) {\r\n const searchParams = new URL(url.includes('?') ? url : `?${url}`, 'http://localhost')\r\n .searchParams;\r\n const result: Record<string, string> = {};\r\n\r\n for (const [key, value] of searchParams.entries()) {\r\n if (value !== 'null' && value !== 'undefined') {\r\n result[key] = value;\r\n }\r\n }\r\n\r\n return result;\r\n}\r\n"],"mappings":"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AAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;A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","names":["html","range","src","timeout","appConfig","res","timeout"]}
1
+ {"version":3,"sources":["../src/web/clipboard/index.ts","../src/web/config/index.ts","../src/web/cookie/index.ts","../src/web/device/index.ts","../src/web/dom/index.ts","../src/web/network/download.ts","../src/ts/day/index.ts","../src/ts/es-toolkit/index.ts","../src/ts/object/index.ts","../src/ts/url/param/index.ts","../src/web/network/request.ts","../src/web/async/index.ts","../src/web/network/uploadFile.ts","../src/web/storage/index.ts","../src/web/url/index.ts"],"sourcesContent":["/**\r\n * 复制文本到剪贴板(兼容移动端和PC)\r\n * @returns Promise<void> 复制成功时 resolve,失败时 reject。\r\n * @example\r\n * await copyText('hello');\r\n * toast('复制成功');\r\n */\r\nexport async function copyText(text: string): Promise<void> {\r\n if (typeof text !== 'string') text = String(text ?? '');\r\n\r\n // 现代 API\r\n if (navigator.clipboard && typeof navigator.clipboard.writeText === 'function') {\r\n try {\r\n await navigator.clipboard.writeText(text);\r\n return;\r\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\r\n } catch (e) {\r\n // 继续尝试回退方案\r\n }\r\n }\r\n\r\n // 回退方案:使用隐藏 textarea + execCommand('copy')\r\n return new Promise<void>((resolve, reject) => {\r\n try {\r\n const textarea = document.createElement('textarea');\r\n textarea.value = text;\r\n\r\n // 避免视觉影响与页面布局影响\r\n textarea.setAttribute('readonly', '');\r\n textarea.style.position = 'fixed';\r\n textarea.style.top = '0';\r\n textarea.style.right = '-9999px';\r\n textarea.style.opacity = '0';\r\n textarea.style.pointerEvents = 'none';\r\n\r\n document.body.appendChild(textarea);\r\n\r\n // 选中文本(移动端兼容)\r\n textarea.focus();\r\n textarea.select();\r\n\r\n // iOS 兼容:明确选区\r\n textarea.setSelectionRange(0, textarea.value.length);\r\n\r\n const ok = document.execCommand('copy');\r\n document.body.removeChild(textarea);\r\n\r\n if (ok) {\r\n resolve();\r\n } else {\r\n reject(new Error('Copy failed: clipboard unavailable'));\r\n }\r\n } catch (e) {\r\n reject(e);\r\n }\r\n });\r\n}\r\n\r\n/**\r\n * 复制富文本 HTML 到剪贴板(移动端与 PC)\r\n * 使用场景:图文混排文章、带样式段落,保留格式粘贴。\r\n * @param html HTML字符串\r\n * @example\r\n * await copyHtml('<p><b>加粗</b> 与 <i>斜体</i></p>');\r\n */\r\nexport async function copyHtml(html: string): Promise<void> {\r\n const s = String(html ?? '');\r\n if (canWriteClipboard()) {\r\n const plain = htmlToText(s);\r\n await writeClipboard({\r\n 'text/html': new Blob([s], { type: 'text/html' }),\r\n 'text/plain': new Blob([plain], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n return execCopyFromHtml(s);\r\n}\r\n\r\n/**\r\n * 复制 DOM 节点到剪贴板(移动端与 PC)\r\n * 使用场景:页面已有区域的可视化复制;元素使用 `outerHTML`,非元素使用其文本内容。\r\n * @param node DOM 节点(元素或文本节点)\r\n * @example\r\n * const el = document.querySelector('#article')!;\r\n * await copyNode(el);\r\n */\r\nexport async function copyNode(node: Node): Promise<void> {\r\n if (canWriteClipboard()) {\r\n const { html, text } = nodeToHtmlText(node);\r\n await writeClipboard({\r\n 'text/html': new Blob([html], { type: 'text/html' }),\r\n 'text/plain': new Blob([text], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n const { html } = nodeToHtmlText(node);\r\n return execCopyFromHtml(html);\r\n}\r\n\r\n/**\r\n * 复制单张图片到剪贴板(移动端与 PC,需浏览器支持 `ClipboardItem`)\r\n * 使用场景:把本地 `canvas` 或 `Blob` 生成的图片直接粘贴到聊天/文档。\r\n * @param image 图片源(Blob/Canvas/ImageBitmap)\r\n * @example\r\n * const canvas = document.querySelector('canvas')!;\r\n * await copyImage(canvas);\r\n */\r\nexport async function copyImage(image: Blob | HTMLCanvasElement | ImageBitmap): Promise<void> {\r\n const blob = await toImageBlob(image);\r\n if (!blob) throw new Error('Unsupported image source');\r\n if (canWriteClipboard()) {\r\n const type = blob.type || 'image/png';\r\n await writeClipboard({ [type]: blob });\r\n return;\r\n }\r\n throw new Error('Clipboard image write not supported');\r\n}\r\n\r\n/**\r\n * 复制 URL 到剪贴板(移动端与 PC)\r\n * 写入 `text/uri-list` 与 `text/plain`,在支持 URI 列表的应用中可识别为链接。\r\n * @param url 完整的 URL 字符串\r\n * @example\r\n * await copyUrl('https://example.com/page');\r\n */\r\nexport async function copyUrl(url: string): Promise<void> {\r\n const s = String(url ?? '');\r\n if (canWriteClipboard()) {\r\n await writeClipboard({\r\n 'text/uri-list': new Blob([s], { type: 'text/uri-list' }),\r\n 'text/plain': new Blob([s], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n await copyText(s);\r\n}\r\n\r\n/**\r\n * 复制任意 Blob 到剪贴板(移动端与 PC,需 `ClipboardItem`)\r\n * 使用场景:原生格式粘贴(如 `image/svg+xml`、`application/pdf` 等)。\r\n * @param blob 任意 Blob 数据\r\n * @example\r\n * const svg = new Blob(['<svg></svg>'], { type: 'image/svg+xml' });\r\n * await copyBlob(svg);\r\n */\r\nexport async function copyBlob(blob: Blob): Promise<void> {\r\n if (canWriteClipboard()) {\r\n const type = blob.type || 'application/octet-stream';\r\n await writeClipboard({ [type]: blob });\r\n return;\r\n }\r\n throw new Error('Clipboard blob write not supported');\r\n}\r\n\r\n/**\r\n * 复制 RTF 富文本到剪贴板(移动端与 PC)\r\n * 同时写入 `text/plain`,增强与 Office/富文本编辑器的兼容性。\r\n * @param rtf RTF 字符串(如:`{\\\\rtf1\\\\ansi ...}`)\r\n * @example\r\n * await copyRtf('{\\\\rtf1\\\\ansi Hello \\\\b World}');\r\n */\r\nexport async function copyRtf(rtf: string): Promise<void> {\r\n const s = String(rtf ?? '');\r\n if (canWriteClipboard()) {\r\n const plain = s\r\n .replace(/\\\\par[\\s]?/g, '\\n')\r\n .replace(/\\{[^}]*\\}/g, '')\r\n .replace(/\\\\[a-zA-Z]+[0-9'-]*/g, '')\r\n .replace(/\\r?\\n/g, '\\n')\r\n .trim();\r\n await writeClipboard({\r\n 'text/rtf': new Blob([s], { type: 'text/rtf' }),\r\n 'text/plain': new Blob([plain], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n await copyText(s);\r\n}\r\n\r\n/**\r\n * 复制表格到剪贴板(移动端与 PC)\r\n * 同时写入多种 MIME:`text/html`(表格)、`text/tab-separated-values`(TSV)、`text/csv`、`text/plain`(TSV)。\r\n * 使用场景:优化粘贴到 Excel/Google Sheets/Docs 的体验\r\n * @param rows 二维数组,每行一个数组(字符串/数字)\r\n * @example\r\n * await copyTable([\r\n * ['姓名', '分数'],\r\n * ['张三', 95],\r\n * ['李四', 88],\r\n * ]);\r\n */\r\nexport async function copyTable(rows: Array<Array<string | number>>): Promise<void> {\r\n const data = Array.isArray(rows) ? rows : [];\r\n const escapeHtml = (t: string) =>\r\n t\r\n .replace(/&/g, '&amp;')\r\n .replace(/</g, '&lt;')\r\n .replace(/>/g, '&gt;')\r\n .replace(/\"/g, '&quot;')\r\n .replace(/'/g, '&#39;');\r\n const html = (() => {\r\n const trs = data\r\n .map((r) => `<tr>${r.map((c) => `<td>${escapeHtml(String(c))}</td>`).join('')}</tr>`)\r\n .join('');\r\n return `<table>${trs}</table>`;\r\n })();\r\n const tsv = data.map((r) => r.map((c) => String(c)).join('\\t')).join('\\n');\r\n const csv = data\r\n .map((r) =>\r\n r\r\n .map((c) => {\r\n const s = String(c);\r\n const needQuote = /[\",\\n]/.test(s);\r\n const escaped = s.replace(/\"/g, '\"\"');\r\n return needQuote ? `\"${escaped}\"` : escaped;\r\n })\r\n .join(','),\r\n )\r\n .join('\\n');\r\n if (canWriteClipboard()) {\r\n await writeClipboard({\r\n 'text/html': new Blob([html], { type: 'text/html' }),\r\n 'text/tab-separated-values': new Blob([tsv], { type: 'text/tab-separated-values' }),\r\n 'text/csv': new Blob([csv], { type: 'text/csv' }),\r\n 'text/plain': new Blob([tsv], { type: 'text/plain' }),\r\n });\r\n return;\r\n }\r\n await copyText(tsv);\r\n}\r\n\r\nasync function toImageBlob(image: Blob | HTMLCanvasElement | ImageBitmap) {\r\n if (image instanceof Blob) return image;\r\n if (image instanceof HTMLCanvasElement)\r\n return await new Promise<Blob>((resolve, reject) => {\r\n image.toBlob(\r\n (b) => (b ? resolve(b) : reject(new Error('Canvas toBlob failed'))),\r\n 'image/png',\r\n );\r\n });\r\n const isBitmap = typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap;\r\n if (isBitmap) {\r\n const cnv = document.createElement('canvas');\r\n cnv.width = (image as ImageBitmap).width;\r\n cnv.height = (image as ImageBitmap).height;\r\n const ctx = cnv.getContext('2d');\r\n ctx?.drawImage(image as ImageBitmap, 0, 0);\r\n return await new Promise<Blob>((resolve, reject) => {\r\n cnv.toBlob((b) => (b ? resolve(b) : reject(new Error('Canvas toBlob failed'))), 'image/png');\r\n });\r\n }\r\n return null;\r\n}\r\n\r\nfunction canWriteClipboard() {\r\n return !!(\r\n navigator.clipboard &&\r\n typeof navigator.clipboard.write === 'function' &&\r\n typeof ClipboardItem !== 'undefined'\r\n );\r\n}\r\n\r\nasync function writeClipboard(items: Record<string, Blob>) {\r\n await navigator.clipboard!.write([new ClipboardItem(items)]);\r\n}\r\n\r\nfunction htmlToText(html: string) {\r\n const div = document.createElement('div');\r\n div.innerHTML = html;\r\n return div.textContent || '';\r\n}\r\n\r\nfunction nodeToHtmlText(node: Node) {\r\n const container = document.createElement('div');\r\n container.appendChild(node.cloneNode(true));\r\n const html =\r\n node instanceof Element ? (node.outerHTML ?? container.innerHTML) : container.innerHTML;\r\n const text = container.textContent || '';\r\n return { html, text };\r\n}\r\n\r\nfunction execCopyFromHtml(html: string) {\r\n return new Promise<void>((resolve, reject) => {\r\n try {\r\n const div = document.createElement('div');\r\n div.contentEditable = 'true';\r\n div.style.position = 'fixed';\r\n div.style.top = '0';\r\n div.style.right = '-9999px';\r\n div.style.opacity = '0';\r\n div.style.pointerEvents = 'none';\r\n div.innerHTML = html;\r\n document.body.appendChild(div);\r\n const selection = window.getSelection();\r\n const range = document.createRange();\r\n range.selectNodeContents(div);\r\n selection?.removeAllRanges();\r\n selection?.addRange(range);\r\n const ok = document.execCommand('copy');\r\n document.body.removeChild(div);\r\n selection?.removeAllRanges();\r\n if (ok) {\r\n resolve();\r\n } else {\r\n reject(new Error('Copy failed: clipboard unavailable'));\r\n }\r\n } catch (e) {\r\n reject(e);\r\n }\r\n });\r\n}\r\n","export type AppConfig = {\r\n /** 全局 Toast 提示 */\r\n toast?: (option: { msg: string; status: 'success' | 'fail' }) => void;\r\n /** 显示全局 Loading */\r\n showLoading?: (option?: { title?: string }) => void;\r\n /** 隐藏全局 Loading */\r\n hideLoading?: () => void;\r\n /** 跳转登录页的方法 */\r\n toLogin?: () => void;\r\n /** 日志记录函数 */\r\n log?: (level: 'info' | 'error' | 'warn' | 'debug', data: AppLogInfo) => void;\r\n};\r\n\r\nexport type AppLogInfo = {\r\n /** 调用函数的名称 */\r\n name: string;\r\n\r\n /** 函数的调用状态 */\r\n status?: 'success' | 'fail';\r\n\r\n /** 函数的调用参数 */\r\n option?: unknown;\r\n\r\n /** 函数的调用结果 */\r\n res?: unknown;\r\n\r\n /** 函数的调用错误 */\r\n e?: unknown;\r\n\r\n /** 日志描述 */\r\n desc?: string;\r\n\r\n // 其他自定义属性\r\n [key: string]: unknown;\r\n};\r\n\r\nconst appConfig: AppConfig = {};\r\n\r\n/**\r\n * 获取应用配置\r\n */\r\nexport function getBaseToolsConfig() {\r\n return appConfig;\r\n}\r\n\r\n/**\r\n * 初始化应用配置 (在入口文件设置)\r\n * @example\r\n * setBaseToolsConfig({\r\n * toast: ({ msg, status }) => (status === 'fail' ? message.error(msg) : message.success(msg)),\r\n * showLoading: () => message.loading('加载中...'),\r\n * hideLoading: () => message.destroy(),\r\n * toLogin: () => reLogin(),\r\n * log(level, data) {\r\n * if (data.name === 'request') {\r\n * sendLog('request', data); // 请求日志\r\n * } else if (level === 'error') {\r\n * sendLog('error', data); // 错误日志\r\n * } else {\r\n * sendLog('action', data); // 操作日志\r\n * }\r\n * },\r\n * });\r\n */\r\nexport function setBaseToolsConfig(newConfig: AppConfig) {\r\n Object.assign(appConfig, newConfig);\r\n}\r\n","/**\r\n * 设置 Cookie(路径默认为 `/`)\r\n * @param name Cookie 名称\r\n * @param value Cookie 值(内部已使用 `encodeURIComponent` 编码)\r\n * @param days 过期天数(从当前时间起算)\r\n * @example\r\n * setCookie('token', 'abc', 7);\r\n */\r\nexport function setCookie(name: string, value: string, days: number) {\r\n const date = new Date();\r\n date.setTime(date.getTime() + days * 24 * 60 * 60 * 1000);\r\n const expires = `expires=${date.toUTCString()}; path=/`;\r\n document.cookie = `${name}=${encodeURIComponent(value)}; ${expires}`;\r\n}\r\n\r\n/**\r\n * 获取 Cookie\r\n * @param name Cookie 名称\r\n * @returns 若存在返回解码后的值,否则 `null`\r\n * @example\r\n * const token = getCookie('token');\r\n */\r\nexport function getCookie(name: string): string | null {\r\n const value = `; ${document.cookie}`;\r\n const parts = value.split(`; ${name}=`);\r\n if (parts.length === 2) {\r\n const v = parts.pop()?.split(';').shift();\r\n return v ? decodeURIComponent(v) : null;\r\n }\r\n return null;\r\n}\r\n\r\n/**\r\n * 移除 Cookie(通过设置过期时间为过去)\r\n * 路径固定为 `/`,确保与默认写入路径一致。\r\n * @param name Cookie 名称\r\n * @example\r\n * removeCookie('token');\r\n */\r\nexport function removeCookie(name: string) {\r\n document.cookie = `${name}=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/`;\r\n}\r\n","/**\r\n * 获取用户代理字符串(UA)\r\n * @returns navigator.userAgent.toLowerCase();\r\n */\r\nexport function getUA(): string {\r\n if (typeof navigator === 'undefined') return ''; // SSR无 navigator\r\n return (navigator.userAgent || '').toLowerCase();\r\n}\r\n\r\n/**\r\n * 是否为移动端设备(含平板)\r\n */\r\nexport function isMobile(): boolean {\r\n const ua = getUA();\r\n return /android|webos|iphone|ipod|blackberry|iemobile|opera mini|mobile/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为平板设备\r\n */\r\nexport function isTablet(): boolean {\r\n const ua = getUA();\r\n return /ipad|android(?!.*mobile)|tablet/i.test(ua) && !/mobile/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为 PC 设备\r\n */\r\nexport function isPC(): boolean {\r\n return !isMobile() && !isTablet();\r\n}\r\n\r\n/**\r\n * 是否为 iOS 系统\r\n */\r\nexport function isIOS(): boolean {\r\n const ua = getUA();\r\n return /iphone|ipad|ipod/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为 Android 系统\r\n */\r\nexport function isAndroid(): boolean {\r\n const ua = getUA();\r\n return /android/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否微信内置浏览器\r\n */\r\nexport function isWeChat(): boolean {\r\n const ua = getUA();\r\n return /micromessenger/i.test(ua);\r\n}\r\n\r\n/**\r\n * 是否为 Chrome 浏览器\r\n * 已排除 Edge、Opera 等基于 Chromium 的浏览器\r\n */\r\nexport function isChrome(): boolean {\r\n const ua = getUA();\r\n return /chrome\\//i.test(ua) && !/edg\\//i.test(ua) && !/opr\\//i.test(ua) && !/whale\\//i.test(ua);\r\n}\r\n\r\n/**\r\n * 检测是否支持触摸事件\r\n */\r\nexport function isTouchSupported(): boolean {\r\n if (typeof window === 'undefined') return false;\r\n return 'ontouchstart' in window || navigator.maxTouchPoints > 0;\r\n}\r\n\r\n/**\r\n * 获取设备像素比\r\n */\r\nexport function getDevicePixelRatio(): number {\r\n if (typeof window === 'undefined') return 1;\r\n return window.devicePixelRatio || 1;\r\n}\r\n\r\n/**\r\n * 获取浏览器名字\r\n */\r\nexport function getBrowserName(): string | null {\r\n const ua = getUA();\r\n\r\n if (/chrome\\//i.test(ua)) return 'chrome';\r\n if (/safari\\//i.test(ua)) return 'safari';\r\n if (/firefox\\//i.test(ua)) return 'firefox';\r\n if (/opr\\//i.test(ua)) return 'opera';\r\n if (/edg\\//i.test(ua)) return 'edge';\r\n if (/msie|trident/i.test(ua)) return 'ie';\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * 获取浏览器版本号\r\n */\r\nexport function getBrowserVersion(): string | null {\r\n const ua = getUA();\r\n\r\n const versionPatterns = [\r\n /(?:edg|edge)\\/([0-9.]+)/i,\r\n /(?:opr|opera)\\/([0-9.]+)/i,\r\n /chrome\\/([0-9.]+)/i,\r\n /firefox\\/([0-9.]+)/i,\r\n /version\\/([0-9.]+).*safari/i,\r\n /(?:msie |rv:)([0-9.]+)/i,\r\n ];\r\n\r\n for (const pattern of versionPatterns) {\r\n const matches = ua.match(pattern);\r\n if (matches && matches[1]) {\r\n return matches[1];\r\n }\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * 获取操作系统信息\r\n */\r\nexport function getOS(): string {\r\n const ua = getUA();\r\n\r\n if (/windows/i.test(ua)) return 'windows';\r\n if (/mac os/i.test(ua)) return 'macos';\r\n if (/linux/i.test(ua)) return 'linux';\r\n if (/iphone|ipad|ipod/i.test(ua)) return 'ios';\r\n if (/android/i.test(ua)) return 'android';\r\n\r\n return 'unknown';\r\n}\r\n","/**\r\n * 获取窗口宽度(不含滚动条)\r\n * @returns 窗口宽度\r\n */\r\nexport function getWindowWidth() {\r\n return window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth;\r\n}\r\n\r\n/**\r\n * 获取窗口高度(不含滚动条)\r\n * @returns 窗口高度\r\n */\r\nexport function getWindowHeight() {\r\n return window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight;\r\n}\r\n\r\n/**\r\n * 获取文档垂直滚动位置\r\n * @example\r\n * const top = getWindowScrollTop();\r\n */\r\nexport function getWindowScrollTop() {\r\n const doc = document.documentElement;\r\n const body = document.body;\r\n return window.pageYOffset || doc.scrollTop || body.scrollTop || 0;\r\n}\r\n\r\n/**\r\n * 获取文档水平滚动位置\r\n * @example\r\n * const left = getWindowScrollLeft();\r\n */\r\nexport function getWindowScrollLeft() {\r\n const doc = document.documentElement;\r\n const body = document.body;\r\n return window.pageXOffset || doc.scrollLeft || body.scrollLeft || 0;\r\n}\r\n\r\n/**\r\n * 平滑滚动到指定位置\r\n * @param top 目标纵向滚动位置\r\n * @param behavior 滚动行为,默认 'smooth'\r\n * @example\r\n * windowScrollTo(0);\r\n */\r\nexport function windowScrollTo(top: number, behavior: ScrollBehavior = 'smooth') {\r\n if ('scrollBehavior' in document.documentElement.style) {\r\n window.scrollTo({ top, behavior });\r\n } else {\r\n window.scrollTo(0, top);\r\n }\r\n}\r\n\r\n/**\r\n * 元素是否在视口内(可设置阈值)\r\n * @param el 目标元素\r\n * @param offset 额外判定偏移(像素,正数放宽,负数收紧)\r\n * @returns 是否在视口内\r\n */\r\nexport function isInViewport(el: Element, offset = 0) {\r\n const rect = el.getBoundingClientRect();\r\n const width = getWindowWidth();\r\n const height = getWindowHeight();\r\n return (\r\n rect.bottom >= -offset &&\r\n rect.right >= -offset &&\r\n rect.top <= height + offset &&\r\n rect.left <= width + offset\r\n );\r\n}\r\n\r\n/**\r\n * 锁定页面滚动(移动端/PC)\r\n * 使用 `body{ position: fixed }` 技术消除滚动条抖动,记录并恢复滚动位置。\r\n * @example\r\n * lockBodyScroll();\r\n */\r\nexport function lockBodyScroll() {\r\n const body = document.body;\r\n if (body.dataset.scrollLock === 'true') return;\r\n const y = Math.round(window.scrollY || window.pageYOffset || 0);\r\n body.dataset.scrollLock = 'true';\r\n body.dataset.scrollLockY = String(y);\r\n body.style.position = 'fixed';\r\n body.style.top = `-${y}px`;\r\n body.style.left = '0';\r\n body.style.right = '0';\r\n body.style.width = '100%';\r\n}\r\n\r\n/**\r\n * 解除页面滚动锁定,恢复原始滚动位置\r\n * @example\r\n * unlockBodyScroll();\r\n */\r\nexport function unlockBodyScroll() {\r\n const body = document.body;\r\n if (body.dataset.scrollLock !== 'true') return;\r\n const y = Number(body.dataset.scrollLockY || 0);\r\n body.style.position = '';\r\n body.style.top = '';\r\n body.style.left = '';\r\n body.style.right = '';\r\n body.style.width = '';\r\n delete body.dataset.scrollLock;\r\n delete body.dataset.scrollLockY;\r\n window.scrollTo(0, y);\r\n}\r\n","import type { AxiosResponse } from 'axios';\r\n\r\n/**\r\n * 下载文件\r\n * @param url 完整的下载地址 | base64字符串 | Blob对象\r\n * @param fileName 自定义文件名(需含后缀)\r\n * @example\r\n * download('https://xx/xx.pdf');\r\n * download('https://xx/xx.pdf', 'xx.pdf');\r\n * download(blob, '图片.jpg');\r\n */\r\nexport async function download(url: string | Blob, fileName = '') {\r\n if (!url) return;\r\n\r\n let blobUrl = '';\r\n let needRevoke = false; // createObjectURL必须revoke,否则内存泄露,刷新页面都不释放\r\n try {\r\n if (url instanceof Blob) {\r\n // Blob对象\r\n blobUrl = URL.createObjectURL(url);\r\n needRevoke = true;\r\n } else if (url.includes(';base64,')) {\r\n // base64字符串\r\n blobUrl = url;\r\n } else {\r\n if (fileName) {\r\n // 自定义文件名:跨域的url无法自定义文件名,此处统一转为blob\r\n const res = await fetch(url);\r\n if (!res.ok) throw new Error(`fetch error ${res.status}:${url}`); // 拦截错误页(404/500 等 HTML)\r\n const blob = await res.blob();\r\n blobUrl = URL.createObjectURL(blob);\r\n needRevoke = true;\r\n } else {\r\n // 非自定义文件名的普通链接\r\n blobUrl = url;\r\n }\r\n }\r\n\r\n // window.location.href = fileUrl // 可能会关闭当前页面\r\n // window.open(fileUrl, '_blank') // 不支持下载图片\r\n // 通过a标签模拟点击下载\r\n const a = document.createElement('a');\r\n a.href = blobUrl;\r\n a.download = fileName; // 若为空字符串,则会自动取url的文件名(跨域url无法自定义文件名,需转为blob)\r\n document.body.appendChild(a);\r\n a.click();\r\n document.body.removeChild(a);\r\n } finally {\r\n if (needRevoke) {\r\n setTimeout(() => URL.revokeObjectURL(blobUrl), 100); // Safari 需要延迟 revoke\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * 解析Axios返回的Blob数据\r\n * @param res Axios响应对象 (responseType='blob')\r\n * @returns 包含blob数据和文件名的对象 { blob, fileName }\r\n * @example\r\n * const res = await axios.get(url, { responseType: 'blob' });\r\n * const { blob, fileName } = await parseAxiosBlob(res);\r\n * download(blob, fileName);\r\n */\r\nexport async function parseAxiosBlob(res: AxiosResponse<Blob>) {\r\n const { data, headers, status, statusText, config } = res;\r\n\r\n if (status < 200 || status >= 300) throw new Error(`${status},${statusText}:${config.url}`);\r\n\r\n // 抛出json错误\r\n if (data.type.includes('application/json')) {\r\n const txt = await data.text();\r\n throw JSON.parse(txt);\r\n }\r\n\r\n // 解析文件名\r\n const fileName = getDispositionFileName(headers['content-disposition']);\r\n return { blob: data, fileName };\r\n}\r\n\r\n/**\r\n * 获取文件名\r\n * @param disposition content-disposition头值\r\n * @returns content-disposition中的filename\r\n * @example\r\n * const fileName = getDispositionFileName(headers['content-disposition']);\r\n */\r\nexport function getDispositionFileName(disposition?: string) {\r\n if (!disposition) return '';\r\n\r\n // 1. RFC5987 filename* 优先\r\n const rfc5987 = /filename\\*\\s*=\\s*([^']*)''([^;]*)/i.exec(disposition);\r\n if (rfc5987?.[2]) {\r\n try {\r\n return decodeURIComponent(rfc5987[2].trim()).replace(/[\\r\\n]+/g, '');\r\n } catch {\r\n return rfc5987[2].trim().replace(/[\\r\\n]+/g, '');\r\n }\r\n }\r\n\r\n // 2. 旧式 filename=\r\n const old = /filename\\s*=\\s*(?:\"([^\"]*)\"|([^\";]*))(?=;|$)/i.exec(disposition);\r\n if (old) return (old[1] ?? old[2]).trim().replace(/[\\r\\n]+/g, '');\r\n\r\n return '';\r\n}\r\n\r\n/**\r\n * 动态加载 JS(重复执行不会重复加载,内部已排重)\r\n * @param src js 文件路径\r\n * @param attrs 可选的脚本属性,如 async、defer、crossOrigin\r\n * @example\r\n * await loadJs('https://xx/xx.js');\r\n * await loadJs('/a.js', { defer: true });\r\n */\r\nexport async function loadJs(\r\n src: string,\r\n attrs?: Pick<HTMLScriptElement, 'async' | 'defer' | 'crossOrigin'>,\r\n) {\r\n return new Promise<void>((resolve, reject) => {\r\n if (hasJs(src)) return resolve();\r\n\r\n const script = document.createElement('script');\r\n script.type = 'text/javascript';\r\n script.src = src;\r\n\r\n if (attrs) {\r\n const keys = Object.keys(attrs) as Array<keyof typeof attrs>;\r\n keys.forEach((key) => {\r\n const v = attrs[key];\r\n if (v === null || v === undefined || v === false) return;\r\n script.setAttribute(key, typeof v === 'boolean' ? '' : v);\r\n });\r\n }\r\n\r\n script.onload = () => resolve();\r\n script.onerror = (e) => reject(e);\r\n\r\n document.head.appendChild(script);\r\n });\r\n}\r\n\r\n/**\r\n * 判断某个 JS 地址是否已在页面中加载过\r\n * @param src 相对、绝对路径的 JS 地址\r\n * @returns 是否已加载过\r\n * @example\r\n * hasJs('https://xx/xx.js'); // boolean\r\n * hasJs('/xx.js'); // boolean\r\n * hasJs('xx.js'); // boolean\r\n */\r\nexport function hasJs(src: string) {\r\n const target = new URL(src, document.baseURI).href;\r\n const jsList = Array.from(document.querySelectorAll('script[src]'));\r\n return jsList.some((e) => {\r\n const src = e.getAttribute('src');\r\n return src && new URL(src, document.baseURI).href === target;\r\n });\r\n}\r\n\r\n/**\r\n * 动态加载 CSS(重复执行不会重复加载,内部已排重)\r\n * @param href css 文件地址\r\n * @param attrs 可选属性,如 crossOrigin、media\r\n * @example\r\n * await loadCss('https://xx/xx.css');\r\n * await loadCss('/a.css', { media: 'print' });\r\n */\r\nexport async function loadCss(\r\n href: string,\r\n attrs?: Pick<HTMLLinkElement, 'crossOrigin' | 'media'>,\r\n) {\r\n return new Promise<void>((resolve, reject) => {\r\n if (hasCss(href)) return resolve();\r\n\r\n const link = document.createElement('link');\r\n link.rel = 'stylesheet';\r\n link.href = href;\r\n\r\n if (attrs) {\r\n const keys = Object.keys(attrs) as Array<keyof typeof attrs>;\r\n keys.forEach((key) => {\r\n const v = attrs[key];\r\n if (v === null || v === undefined) return;\r\n link.setAttribute(key, String(v));\r\n });\r\n }\r\n\r\n link.onload = () => resolve();\r\n link.onerror = (e) => reject(e);\r\n\r\n document.head.appendChild(link);\r\n });\r\n}\r\n\r\n/**\r\n * 判断某个 CSS 地址是否已在页面中加载过\r\n * @param href 相对、绝对路径的 CSS 地址\r\n * @returns 是否已加载过\r\n * @example\r\n * hasCss('https://xx/xx.css'); // boolean\r\n */\r\nexport function hasCss(href: string) {\r\n const target = new URL(href, document.baseURI).href;\r\n const list = Array.from(document.querySelectorAll('link[rel=\"stylesheet\"][href]'));\r\n return list.some((e) => {\r\n const h = e.getAttribute('href');\r\n return h && new URL(h, document.baseURI).href === target;\r\n });\r\n}\r\n\r\n/**\r\n * 预加载图片\r\n * @param src 图片地址\r\n * @returns Promise<HTMLImageElement>\r\n * @example\r\n * await preloadImage('/a.png');\r\n */\r\nexport function preloadImage(src: string) {\r\n return new Promise<HTMLImageElement>((resolve, reject) => {\r\n const img = new Image();\r\n img.onload = () => resolve(img);\r\n img.onerror = (e) => reject(e);\r\n img.src = src;\r\n });\r\n}\r\n","import dayjs from 'dayjs';\r\nimport customParseFormat from 'dayjs/plugin/customParseFormat';\r\nimport utc from 'dayjs/plugin/utc';\r\nimport timezone from 'dayjs/plugin/timezone';\r\nimport relativeTime from 'dayjs/plugin/relativeTime';\r\nimport advancedFormat from 'dayjs/plugin/advancedFormat';\r\nimport 'dayjs/locale/zh-cn';\r\nimport { zeroPad } from '../number';\r\n\r\ndayjs.extend(customParseFormat);\r\ndayjs.extend(utc);\r\ndayjs.extend(timezone);\r\ndayjs.extend(relativeTime);\r\ndayjs.extend(advancedFormat);\r\ndayjs.locale('zh-cn');\r\n\r\ntype BaseTime = number | string | Date | dayjs.Dayjs | null | undefined;\r\n\r\n/**\r\n * 创建 dayjs 实例\r\n * 文档: https://day.js.org/zh-CN/\r\n * @param t 各种规范或不规范的时间\r\n * @returns dayjs 实例\r\n * @example\r\n * const d = toDayjs('2021-01-01'); // dayjs 实例 (无参,则默认当前时间)\r\n * d.format('YYYY-MM-DD HH:mm:ss'); // \"2025-12-10 11:33:16\"\r\n * d.valueOf(); // 毫秒时间戳,如 1765337596913\r\n * d.unix(); // 秒时间戳,如 1765337596\r\n * d.millisecond(); // 毫秒 913\r\n * d.second(); // 秒 16\r\n * d.minute(); // 分 33\r\n * d.hour(); // 时 11\r\n * d.date(); // 日 10\r\n * d.day(); // 星期几 5(周日=0)\r\n * d.month() + 1; // 月 12\r\n * d.year(); // 年 2025\r\n * d.startOf('day').valueOf(); // 当日零点\r\n * d.startOf('month').format('YYYY-MM-DD HH:mm:ss'); // 月初 \"2025-12-01 00:00:00\"\r\n * d.endOf('month').format('YYYY-MM-DD HH:mm:ss'); // 月末 \"2025-12-31 23:59:59\"\r\n * d.fromNow(); // “刚刚”、“x分钟前/后”、“x小时前/后”、“x天前/后”、“x月前/后”、“x年前/后”\r\n * d.isSame(t, 'day'); // 是否与t在同一天\r\n * d.diff(); // 与当前时间相差的毫秒数\r\n * d.diff(t); // 与t相差的毫秒数\r\n * d.diff(t, 'second'); // 与t相差的秒数\r\n * d.diff(t, 'minute'); // 与t相差的分钟数\r\n * d.diff(t, 'hour'); // 与t相差的小时数\r\n * d.diff(t, 'day'); // 与t相差的天数\r\n * d.diff(t, 'week'); // 与t相差的周数\r\n * d.diff(t, 'month'); // 与t相差的月数\r\n * d.diff(t, 'quarter'); // 与t相差的季度数\r\n * d.diff(t, 'year'); // 与t相差的年数\r\n */\r\nexport function toDayjs(t?: BaseTime, fmt?: dayjs.OptionType) {\r\n if (t === null || t === undefined) return dayjs();\r\n if (typeof t === 'number') {\r\n const s = String(Math.trunc(t));\r\n return dayjs(s.length === 10 ? t * 1000 : t, fmt);\r\n }\r\n if (typeof t === 'string') {\r\n const s = t.trim();\r\n if (/^\\d{10}$/.test(s)) return dayjs(Number(s) * 1000, fmt);\r\n if (/^\\d{13}$/.test(s)) return dayjs(Number(s), fmt);\r\n if (/^\\d{4}-\\d{2}-\\d{2}$/.test(s)) return dayjs(s, fmt || 'YYYY-MM-DD');\r\n if (/^\\d{4}\\/\\d{2}\\/\\d{2}$/.test(s)) return dayjs(s, fmt || 'YYYY/MM/DD');\r\n if (/^\\d{4}-\\d{2}-\\d{2}\\s+\\d{2}:\\d{2}:\\d{2}$/.test(s))\r\n return dayjs(s, fmt || 'YYYY-MM-DD HH:mm:ss');\r\n if (/^\\d{4}\\/\\d{2}\\/\\d{2}\\s+\\d{2}:\\d{2}:\\d{2}$/.test(s))\r\n return dayjs(s, fmt || 'YYYY/MM/DD HH:mm:ss');\r\n return dayjs(s, fmt);\r\n }\r\n return dayjs(t, fmt);\r\n}\r\n\r\n/**\r\n * 获取“前几天”的日期范围\r\n * @param offset 正整数天数\r\n * @param fmt 日期格式,默认 `YYYY-MM-DD`\r\n * @returns `[start, end]` 日期字符串数组\r\n * @example\r\n * 若今天为 2025-11-19:\r\n * getDateRangeBefore(1) // ['2025-11-18', '2025-11-19']\r\n * getDateRangeBefore(1, 'YYYY-MM-DD HH:mm:ss') // ['2025-11-18 00:00:00', '2025-11-19 23:59:59']\r\n */\r\nexport function getDateRangeBefore(offset: number, fmt = 'YYYY-MM-DD') {\r\n const now = toDayjs(Date.now());\r\n const n = Math.max(0, Math.trunc(offset));\r\n const hasTime = /H|h|m|s|S|A|a|x|X/.test(fmt);\r\n const startDay = now.add(-n, 'day');\r\n const endDay = now;\r\n const start = (hasTime ? startDay.startOf('day') : startDay).format(fmt);\r\n const end = (hasTime ? endDay.endOf('day') : endDay).format(fmt);\r\n return [start, end];\r\n}\r\n\r\n/**\r\n * 获取“后几天”的日期范围\r\n * - 起点:今天;终点:`offset` 天后\r\n * - 若 `fmt` 含时间令牌(如 `HH:mm:ss`),则返回整日范围:起点为当日零点,终点为当日末尾\r\n * @param offset 正整数天数\r\n * @param fmt 日期格式,默认 `YYYY-MM-DD`\r\n * @returns `[start, end]` 日期字符串数组\r\n * @example\r\n * 若今天为 2025-11-19:\r\n * getDateRangeAfter(1) // ['2025-11-19', '2025-11-20']\r\n * getDateRangeAfter(1, 'YYYY-MM-DD HH:mm:ss') // ['2025-11-19 00:00:00', '2025-11-20 23:59:59']\r\n */\r\nexport function getDateRangeAfter(offset: number, fmt = 'YYYY-MM-DD') {\r\n const now = toDayjs(Date.now());\r\n const n = Math.max(0, Math.trunc(offset));\r\n const hasTime = /H|h|m|s|S|A|a|x|X/.test(fmt);\r\n const startDay = now;\r\n const endDay = now.add(n, 'day');\r\n const start = (hasTime ? startDay.startOf('day') : startDay).format(fmt);\r\n const end = (hasTime ? endDay.endOf('day') : endDay).format(fmt);\r\n return [start, end];\r\n}\r\n\r\n/**\r\n * 获取倒计时的时间分解(零填充字符串)\r\n * @param diff 毫秒差值(正数表示剩余时间,负数/0表示已到期)\r\n * @returns 包含天、时、分、秒、毫秒的零填充对象\r\n * @example\r\n * const diff = toDayjs(t).diff(); // 毫秒差值\r\n * const parts = getCountdownParts(diff); // { d: '00', h: '00', m: '00', s: '00', ms: '000' }\r\n */\r\nexport function getCountdownParts(diff: number) {\r\n if (diff <= 0) return { d: '00', h: '00', m: '00', s: '00', ms: '000' };\r\n\r\n const d = Math.floor(diff / (1000 * 60 * 60 * 24));\r\n const h = Math.floor((diff / (1000 * 60 * 60)) % 24);\r\n const m = Math.floor((diff / (1000 * 60)) % 60);\r\n const s = Math.floor((diff / 1000) % 60);\r\n const ms = diff % 1000;\r\n\r\n return {\r\n d: zeroPad(d),\r\n h: zeroPad(h),\r\n m: zeroPad(m),\r\n s: zeroPad(s),\r\n ms: zeroPad(ms, 3),\r\n };\r\n}\r\n\r\n/**\r\n * 通过出生日期计算年龄\r\n * @param birthdate 生日日期,支持多种格式(会被自动解析)\r\n * @returns 年龄对象,包含 `age`(年龄数值)和 `type`(年龄单位,'year' 表示年,'month' 表示月)\r\n * @example\r\n * // 假设当前日期为 2025-11-19\r\n * getAgeByBirthdate('2025-05-10'); // { age: 6, type: 'month' }\r\n * getAgeByBirthdate('2020-11-19'); // { age: 5, type: 'year' }\r\n * getAgeByBirthdate('2020-12-01'); // { age: 4, type: 'year' }(生日还没到, 所以年龄是4岁)\r\n */\r\nexport function getAgeByBirthdate(birthdate: string) {\r\n const birth = toDayjs(birthdate, 'YYYY-MM-DD');\r\n const now = toDayjs(Date.now());\r\n\r\n // 精确的月份计算\r\n const totalMonths = (now.year() - birth.year()) * 12 + (now.month() - birth.month());\r\n\r\n // 如果当前日期小于出生日期,月份减1\r\n const adjustedMonths = now.date() < birth.date() ? totalMonths - 1 : totalMonths;\r\n\r\n if (adjustedMonths >= 12) {\r\n let age = Math.floor(adjustedMonths / 12);\r\n // 检查生日是否已过\r\n const birthdayThisYear = birth.add(age, 'year');\r\n if (now.isBefore(birthdayThisYear)) {\r\n age--;\r\n }\r\n return { age, type: 'year' };\r\n }\r\n\r\n return { age: adjustedMonths, type: 'month' };\r\n}\r\n\r\n/**\r\n * 对外抛出 dayjs 以便全局配置\r\n * @example\r\n * 切换语言\r\n * dayjs.locale('en'); // 切换为英文\r\n * dayjs.locale('zh-cn'); // 切换为中文 (默认)\r\n */\r\nexport { dayjs };\r\n","/**\r\n * re-export 全量 es-toolkit\r\n * 版本: 1.44.0\r\n * 文档: https://es-toolkit.dev/\r\n * 目的: 提供常用工具,收敛依赖版本\r\n *\r\n * 注意: 此文件由 scripts/update-es-toolkit-exports.ts 生成, 请勿手动修改.\r\n * 如需更新, 请先升级es-toolkit, 然后运行: npm run update:es-toolkit\r\n */\r\nexport {\r\n AbortError,\r\n Mutex,\r\n Semaphore,\r\n TimeoutError,\r\n after,\r\n ary,\r\n assert,\r\n asyncNoop,\r\n at,\r\n attempt,\r\n attemptAsync,\r\n before,\r\n camelCase,\r\n capitalize,\r\n chunk,\r\n clamp,\r\n clone,\r\n cloneDeep,\r\n cloneDeepWith,\r\n compact,\r\n constantCase,\r\n countBy,\r\n curry,\r\n curryRight,\r\n debounce,\r\n deburr,\r\n delay,\r\n difference,\r\n differenceBy,\r\n differenceWith,\r\n drop,\r\n dropRight,\r\n dropRightWhile,\r\n dropWhile,\r\n escape,\r\n escapeRegExp,\r\n fill,\r\n filterAsync,\r\n findKey,\r\n flatMap,\r\n flatMapAsync,\r\n flatMapDeep,\r\n flatten,\r\n flattenDeep,\r\n flattenObject,\r\n flow,\r\n flowRight,\r\n forEachAsync,\r\n forEachRight,\r\n groupBy,\r\n head,\r\n identity,\r\n inRange,\r\n initial,\r\n intersection,\r\n intersectionBy,\r\n intersectionWith,\r\n invariant,\r\n invert,\r\n isArrayBuffer,\r\n isBlob,\r\n isBoolean,\r\n isBrowser,\r\n isBuffer,\r\n isDate,\r\n isEmptyObject,\r\n isEqual,\r\n isEqualWith,\r\n isError,\r\n isFile,\r\n isFunction,\r\n isJSON,\r\n isJSONArray,\r\n isJSONObject,\r\n isJSONValue,\r\n isLength,\r\n isMap,\r\n isNil,\r\n isNode,\r\n isNotNil,\r\n isNull,\r\n isNumber,\r\n isPlainObject,\r\n isPrimitive,\r\n isPromise,\r\n isRegExp,\r\n isSet,\r\n isString,\r\n isSubset,\r\n isSubsetWith,\r\n isSymbol,\r\n isTypedArray,\r\n isUndefined,\r\n isWeakMap,\r\n isWeakSet,\r\n kebabCase,\r\n keyBy,\r\n last,\r\n limitAsync,\r\n lowerCase,\r\n lowerFirst,\r\n mapAsync,\r\n mapKeys,\r\n mapValues,\r\n maxBy,\r\n mean,\r\n meanBy,\r\n median,\r\n medianBy,\r\n memoize,\r\n merge,\r\n mergeWith,\r\n minBy,\r\n negate,\r\n noop,\r\n omit,\r\n omitBy,\r\n once,\r\n orderBy,\r\n pad,\r\n partial,\r\n partialRight,\r\n partition,\r\n pascalCase,\r\n pick,\r\n pickBy,\r\n pull,\r\n pullAt,\r\n random,\r\n randomInt,\r\n range,\r\n rangeRight,\r\n reduceAsync,\r\n remove,\r\n rest,\r\n retry,\r\n reverseString,\r\n round,\r\n sample,\r\n sampleSize,\r\n shuffle,\r\n snakeCase,\r\n sortBy,\r\n spread,\r\n startCase,\r\n sum,\r\n sumBy,\r\n tail,\r\n take,\r\n takeRight,\r\n takeRightWhile,\r\n takeWhile,\r\n throttle,\r\n timeout,\r\n toCamelCaseKeys,\r\n toFilled,\r\n toMerged,\r\n toSnakeCaseKeys,\r\n trim,\r\n trimEnd,\r\n trimStart,\r\n unary,\r\n unescape,\r\n union,\r\n unionBy,\r\n unionWith,\r\n uniq,\r\n uniqBy,\r\n uniqWith,\r\n unzip,\r\n unzipWith,\r\n upperCase,\r\n upperFirst,\r\n windowed,\r\n withTimeout,\r\n without,\r\n words,\r\n xor,\r\n xorBy,\r\n xorWith,\r\n zip,\r\n zipObject,\r\n zipWith,\r\n} from 'es-toolkit';\r\n\r\n// 导出类型\r\nexport type * from 'es-toolkit';\r\n","import { get, set } from 'es-toolkit/compat';\r\n\r\n/**\r\n * 获取对象键名数组(类型安全)。\r\n * 注:内置 `Object.keys` 与 `es-toolkit` 的 `keys` 在 TS 中通常返回 `string[]`,无法精确到 `keyof T`。\r\n * @param obj 目标对象\r\n * @returns 类型精确的 `Array<keyof T>`\r\n * @example\r\n * const o = { a: 1, b: 'x' };\r\n * const keys = getObjectKeys(o); // type: ('a' | 'b')[], value: ['a','b']\r\n */\r\nexport function getObjectKeys<T extends object>(obj: T): Array<keyof T> {\r\n return Object.keys(obj) as (keyof T)[];\r\n}\r\n\r\n/**\r\n * 获取对象值。\r\n * @param obj 目标对象\r\n * @param path 路径。(点路径:'a.b'、'a[0].b';数组路径:['a',0,'b'])\r\n * @returns 值\r\n * @example\r\n * const o = { b: { c: 'x' }, users: [{ name: 'john' }, { name: 'jane' }] };\r\n * const c = getObjectValue(o, 'b.c'); // 点路径: 'x'\r\n * const name0 = getObjectValue(o, 'users[0].name'); // 数组字符: 'john'\r\n * const name1 = getObjectValue(o, ['users', 1, 'name']); // 数组路径: 'jane'\r\n */\r\nexport const getObjectValue = get;\r\n\r\n/**\r\n * 设置对象值。\r\n * @param obj 目标对象\r\n * @param path 路径。(点路径:'a.b'、'a[0].b';数组路径:['a',0,'b'])\r\n * @param value 值\r\n * @example\r\n * const o = { b: { c: 'x' }, users: [{ name: 'john' }, { name: 'jane' }] };\r\n * setObjectValue(o, 'b.c', 'y'); // 点路径\r\n * setObjectValue(o, ['users', 1, 'name'], 'jane-doe'); // 数组路径\r\n */\r\nexport const setObjectValue = set;\r\n","/**\r\n * 将对象参数拼接到 URL\r\n * - 采用纯JS拼接,因为小程序不支持URLSearchParams\r\n * @param url 基础地址\r\n * @param param 将要追加的参数对象;`null/undefined` 值会被忽略,Object 会使用 `JSON.stringify`\r\n * @returns 拼接后的完整 URL(保留原有哈希片段)\r\n * @example\r\n * const url = appendUrlParam('https://a.com', { q: '测试', list: [1, 2], a: null, b: undefined }); // 'https://a.com/?q=%E6%B5%8B%E8%AF%95&list=[1,2]'\r\n */\r\nexport function appendUrlParam(url: string, param: Record<string, unknown>) {\r\n if (!param || typeof param !== 'object') return url;\r\n\r\n const hashIndex = url.indexOf('#');\r\n const baseWithoutHash = hashIndex >= 0 ? url.slice(0, hashIndex) : url;\r\n const hash = hashIndex >= 0 ? url.slice(hashIndex) : '';\r\n\r\n const [base, existingQs] = baseWithoutHash.split('?');\r\n const parts: string[] = [];\r\n if (existingQs) parts.push(existingQs);\r\n for (const key in param) {\r\n const rawVal = param[key];\r\n if (rawVal === null || rawVal === undefined) continue;\r\n const val = typeof rawVal === 'object' ? JSON.stringify(rawVal) : String(rawVal);\r\n parts.push(`${encodeURIComponent(key)}=${encodeURIComponent(val)}`);\r\n }\r\n const qs = parts.filter(Boolean).join('&');\r\n return base + (qs ? `?${qs}` : '') + hash;\r\n}\r\n","import {\r\n appendUrlParam,\r\n cloneDeep,\r\n getObjectValue,\r\n isPlainObject,\r\n toDayjs,\r\n} from '@base-web-kits/base-tools-ts';\r\nimport { getBaseToolsConfig } from '../config';\r\nimport type { AppLogInfo } from '../config';\r\n\r\n/** 请求方法类型 */\r\nexport type RequestMethod =\r\n | 'GET'\r\n | 'POST'\r\n | 'PUT'\r\n | 'DELETE'\r\n | 'CONNECT'\r\n | 'HEAD'\r\n | 'OPTIONS'\r\n | 'TRACE'\r\n | 'PATCH';\r\n\r\n/**\r\n * 请求参数类型\r\n * 包含 fetch 原生支持的 BodyInit 类型,以及支持自动 JSON 序列化的对象和数组\r\n */\r\nexport type RequestData =\r\n | string\r\n | ArrayBuffer\r\n | ArrayBufferView\r\n | Blob\r\n | FormData\r\n | URLSearchParams\r\n | ReadableStream<Uint8Array>\r\n | Record<string, unknown>\r\n | unknown[]\r\n | null;\r\n\r\n/**\r\n * 响应数据类型\r\n */\r\nexport type ResponseData = string | ArrayBuffer | Blob | Record<string, unknown> | unknown[] | null;\r\n\r\n/**\r\n * 发起请求的配置 (对外,参数可选)\r\n */\r\nexport type RequestConfig<D extends RequestData = RequestData> = Partial<RequestConfigBase<D>>;\r\n\r\n/**\r\n * 自定义请求的配置 (接口字段参数必填)\r\n */\r\nexport type RequestConfigBase<D extends RequestData = RequestData> = {\r\n /** 接口地址 */\r\n url: string;\r\n\r\n /** 请求方法 */\r\n method?: RequestMethod;\r\n\r\n /** 请求头(会自动过滤undefined, null, \"\";不过滤0和false; 数字和布尔值会自动转换为字符串) */\r\n header?: Record<string, string | number | boolean | null | undefined>;\r\n\r\n /** 请求参数 */\r\n data?: D;\r\n\r\n /** 超时时间 (毫秒), 默认 60000 */\r\n timeout?: number;\r\n\r\n /** 接口返回响应数据的字段, 支持\"a[0].b.c\"的格式, 当配置false时返回完整的响应数据 */\r\n resKey: string | false;\r\n\r\n /** 接口返回响应消息的字段, 支持\"a[0].b.c\"的格式 */\r\n msgKey: string;\r\n\r\n /** 接口返回响应状态码的字段, 支持\"a[0].b.c\"的格式 */\r\n codeKey: string;\r\n\r\n /** 接口返回成功状态码的字段, 支持\"a[0].b.c\"的格式 (默认取 codeKey) */\r\n successKey?: string;\r\n\r\n /** 成功状态码 */\r\n successCode: (number | string)[];\r\n\r\n /** 登录过期状态码 */\r\n reloginCode: (number | string)[];\r\n\r\n /** 是否开启流式传输 (如 SSE) */\r\n enableChunked?: boolean;\r\n\r\n /** 响应类型 (默认 json, enableChunked为true时忽略) */\r\n responseType?: 'text' | 'arraybuffer' | 'json';\r\n\r\n /** 响应数据的缓存时间, 单位毫秒。仅在成功时缓存;仅缓存在内存,应用退出,缓存消失。(默认0,不开启缓存) */\r\n cacheTime?: number;\r\n\r\n /** 是否提示接口异常 (默认true) */\r\n toastError?: boolean;\r\n\r\n /** 是否显示进度条: 支持字符串,自定义文本 (默认true) */\r\n showLoading?: boolean | string;\r\n\r\n /** 是否输出日志 (默认true) */\r\n showLog?: boolean;\r\n\r\n /** 成功和失败时,额外输出的日志数据 (可覆盖内部log参数,如'name') */\r\n logExtra?: Record<string, unknown>;\r\n\r\n /** 响应数据的转换 */\r\n resMap?: (data: ResponseData) => ResponseData;\r\n\r\n /** 获取task对象, 用于取消请求或监听流式数据 */\r\n onTaskReady?: (task: RequestTask) => void;\r\n};\r\n\r\n/**\r\n * 请求任务对象 (用于取消请求或监听流式数据)\r\n */\r\nexport interface RequestTask {\r\n /** 取消请求 */\r\n abort: () => void;\r\n\r\n /** 监听流式数据块接收事件 */\r\n onChunkReceived: (callback: ChunkCallback) => void;\r\n\r\n /** 取消监听流式数据块接收事件 */\r\n offChunkReceived: () => void;\r\n}\r\n\r\n/**\r\n * 流式数据块接收事件回调\r\n */\r\nexport type ChunkCallback = (response: { data: ArrayBuffer }) => void;\r\n\r\n/** 请求缓存 */\r\nconst requestCache = new Map<string, { res: unknown; expire: number }>();\r\n\r\n/**\r\n * 基础请求 (返回 Promise 和 Task 对象)\r\n * 基于 fetch API 封装,支持流式请求\r\n * @param config 请求配置\r\n * @example\r\n * // 在入口文件完成配置 (确保请求失败有toast提示,登录过期能够触发重新登录,log有日志输出)\r\n * setBaseToolsConfig({\r\n * toast: ({ msg, status }) => (status === 'fail' ? message.error(msg) : message.success(msg)),\r\n * showLoading: ({ title }) => message.loading(title || '加载中...'),\r\n * hideLoading: () => message.destroy(),\r\n * toLogin: () => reLogin(),\r\n * log(level, data) {\r\n * if (data.name === 'request') {\r\n * sendLog('request', data); // 请求日志\r\n * } else if (level === 'error') {\r\n * sendLog('error', data); // 错误日志\r\n * } else {\r\n * sendLog('action', data); // 操作日志\r\n * }\r\n * },\r\n * });\r\n *\r\n * // 封装项目的基础请求\r\n * export function requestApi<T>(config: RequestConfig) {\r\n * return request<T>({\r\n * header: { token: 'xx', version: 'xx', tid: 'xx' }, // 会自动过滤空值\r\n * // resMap: (res) => res, // 响应数据的转换, 如解密操作 (可选)\r\n * resKey: 'data',\r\n * msgKey: 'message',\r\n * codeKey: 'status',\r\n * successCode: [1],\r\n * reloginCode: [-10],\r\n * ...config,\r\n * });\r\n * }\r\n *\r\n * // 1. 基于上面 requestApi 的普通接口\r\n * export function apiGoodList(data: { page: number, size: number }) {\r\n * return requestApi<GoodItem[]>({ url: '/goods/list', data, resKey: 'data.list' });\r\n * }\r\n *\r\n * const goodList = await apiGoodList({ page:1, size:10 });\r\n *\r\n * // 2. 参数泛型的写法\r\n * export function apiGoodList(config: RequestConfig<{ page: number, size: number }>) {\r\n * return requestApi<GoodItem[]>({ url: '/goods/list', resKey: 'data.list', ...config });\r\n * }\r\n *\r\n * const goodList = await apiGoodList({ data: { page:1, size:10 } });\r\n *\r\n * // 3. 基于上面 requestApi 的流式接口\r\n * export function apiChatStream(config: RequestConfig) {\r\n * return requestApi({\r\n * ...config,\r\n * url: '/sse/chatStream',\r\n * resKey: false,\r\n * showLoading: false,\r\n * responseType: 'arraybuffer', // 流式响应类型\r\n * enableChunked: true, // 开启分块传输\r\n * });\r\n * }\r\n *\r\n * // 流式监听\r\n * const onTaskReady = (task: RequestTask) => {\r\n * task.onChunkReceived((res) => {\r\n * console.log('ArrayBuffer', res.data);\r\n * });\r\n * }\r\n *\r\n * // 流式发起\r\n * const data = { content: '你好', chatId: 123 };\r\n * await apiChatStream({ data, onTaskReady });\r\n *\r\n * // 流式取消 (在组件销毁或页面关闭时调用)\r\n * task?.offChunkReceived(); // 取消监听,中断流式接收\r\n * task?.abort(); // 取消请求 (若流式已生成,此时abort无效,因为请求已成功)\r\n */\r\nexport function request<T, D extends RequestData = RequestData>(config: RequestConfigBase<D>) {\r\n const {\r\n url,\r\n data,\r\n header,\r\n method = 'GET',\r\n resKey,\r\n msgKey,\r\n codeKey,\r\n successKey,\r\n successCode,\r\n reloginCode,\r\n showLoading = true,\r\n toastError = true,\r\n enableChunked = false,\r\n cacheTime,\r\n resMap,\r\n responseType = 'json',\r\n timeout = 60000,\r\n onTaskReady,\r\n } = config;\r\n\r\n // 1. 初始化控制对象\r\n const controller = new AbortController();\r\n const signal = controller.signal;\r\n let chunkCallback: ChunkCallback | null = null;\r\n\r\n // 构造 Task 对象\r\n const task: RequestTask = {\r\n abort: () => controller.abort(),\r\n onChunkReceived: (cb) => {\r\n chunkCallback = cb;\r\n },\r\n offChunkReceived: () => {\r\n chunkCallback = null;\r\n },\r\n };\r\n onTaskReady?.(task);\r\n\r\n // 2. 创建 Promise\r\n return new Promise<T>((resolve, reject) => {\r\n const execute = async () => {\r\n const isGet = method === 'GET';\r\n const isObjectData = isPlainObject(data);\r\n const isArrayData = !isObjectData && Array.isArray(data);\r\n\r\n // 2.1 参数处理\r\n // 参数过滤 undefined\r\n const fillData = isObjectData ? filterRequestData(data) : data;\r\n\r\n // 请求头过滤空值 (undefined, null, \"\")\r\n const fillHeader = filterRequestHeader(header);\r\n\r\n // 获取 Content-Type (忽略大小写)\r\n const contentTypeKey = Object.keys(fillHeader).find(\r\n (k) => k.toLowerCase() === 'content-type',\r\n );\r\n const contentType = contentTypeKey ? String(fillHeader[contentTypeKey]).toLowerCase() : '';\r\n\r\n if (!isGet && fillData && (isObjectData || isArrayData) && !contentType) {\r\n fillHeader['Content-Type'] = 'application/json';\r\n }\r\n\r\n // 2.2 处理 URL 和 Body\r\n const fillUrl =\r\n isGet && isObjectData ? appendUrlParam(url, fillData as Record<string, unknown>) : url;\r\n\r\n let fillBody: BodyInit | null | undefined;\r\n\r\n if (!isGet && fillData) {\r\n if (isObjectData && contentType.includes('application/x-www-form-urlencoded')) {\r\n // application/x-www-form-urlencoded: 转换为 URLSearchParams\r\n fillBody = toSearchParams(fillData as Record<string, unknown>);\r\n } else if (isObjectData && contentType.includes('multipart/form-data')) {\r\n // multipart/form-data: 转换为 FormData\r\n fillBody = toFormData(fillData as Record<string, unknown>);\r\n // 删除 Content-Type, 让 fetch 自动生成 boundary\r\n if (contentTypeKey) delete fillHeader[contentTypeKey];\r\n } else if (isObjectData || isArrayData) {\r\n fillBody = JSON.stringify(fillData);\r\n } else {\r\n fillBody = fillData as BodyInit;\r\n }\r\n }\r\n\r\n // 2.3 日志与缓存配置\r\n const logConfig = { ...config, data: fillData, header: fillHeader, url: fillUrl };\r\n const startTime = Date.now();\r\n\r\n // 2.4 检查缓存\r\n const isCache = cacheTime && cacheTime > 0;\r\n const cacheKey = isCache ? JSON.stringify({ url: fillUrl, data: fillData }) : '';\r\n\r\n if (isCache) {\r\n const res = checkCache(cacheKey);\r\n if (res) {\r\n logRequestInfo({\r\n status: 'success',\r\n config: logConfig,\r\n fromCache: true,\r\n startTime,\r\n res,\r\n });\r\n resolve(getResult(res, resKey) as T);\r\n return;\r\n }\r\n }\r\n\r\n // 2.5 UI 反馈\r\n const appConfig = getBaseToolsConfig();\r\n if (showLoading)\r\n appConfig.showLoading?.(typeof showLoading === 'string' ? { title: showLoading } : {});\r\n\r\n // 2.6 设置超时\r\n let isTimeout = false;\r\n const timeoutId = setTimeout(() => {\r\n isTimeout = true;\r\n controller.abort();\r\n }, timeout);\r\n\r\n try {\r\n // 2.7 发起请求\r\n const response = await fetch(fillUrl, {\r\n method,\r\n headers: fillHeader,\r\n body: fillBody,\r\n signal,\r\n });\r\n\r\n if (!response.ok) {\r\n if (showLoading) appConfig.hideLoading?.();\r\n throw new Error(`HTTP Error ${response.status}: ${response.statusText}`);\r\n }\r\n\r\n // 2.8 处理流式响应\r\n if (enableChunked) {\r\n if (showLoading) appConfig.hideLoading?.();\r\n\r\n const res = await handleStreamResponse(response, chunkCallback);\r\n\r\n logRequestInfo({ status: 'success', config: logConfig, startTime, res });\r\n\r\n resolve(res as T);\r\n return;\r\n }\r\n\r\n // 2.9 处理普通响应\r\n const resData = await parseResponse(response, responseType);\r\n\r\n // 隐藏 Loading\r\n if (showLoading) appConfig.hideLoading?.();\r\n\r\n // 响应拦截\r\n const res = resMap ? resMap(resData) : resData;\r\n\r\n // 2.10 业务状态码解析\r\n const code = getObjectValue(res, codeKey);\r\n const scode = successKey ? getObjectValue(res, successKey) : code;\r\n const msg = getObjectValue(res, msgKey);\r\n const isSuccess = successCode.includes(scode);\r\n const isRelogin = reloginCode.includes(code);\r\n\r\n logRequestInfo({ status: 'success', config: logConfig, startTime, res });\r\n\r\n // 2.11 结果处理\r\n if (isSuccess) {\r\n // 业务正常\r\n if (isCache) requestCache.set(cacheKey, { res, expire: Date.now() + cacheTime });\r\n resolve(getResult(res, resKey) as T);\r\n } else if (isRelogin) {\r\n // 登录失效\r\n reject(res);\r\n appConfig.toLogin?.(); // 放在后面,确保reject执行后再跳转登录\r\n } else {\r\n // 业务错误\r\n if (toastError && msg) appConfig.toast?.({ status: 'fail', msg });\r\n reject(res);\r\n }\r\n } catch (e) {\r\n const status = 'fail';\r\n const isAbortError = e instanceof DOMException && e.name === 'AbortError'; // 取消请求不视为错误\r\n\r\n if (isAbortError && isTimeout) {\r\n if (toastError) appConfig.toast?.({ status, msg: '请求超时' });\r\n const timeoutError = new Error('Request Timeout');\r\n logRequestInfo({ status, config: logConfig, startTime, e: timeoutError });\r\n reject(timeoutError);\r\n return;\r\n }\r\n\r\n if (!isAbortError && toastError) appConfig.toast?.({ status, msg: '网络请求失败' });\r\n logRequestInfo({ status, config: logConfig, startTime, e });\r\n reject(e);\r\n } finally {\r\n if (timeoutId) clearTimeout(timeoutId);\r\n }\r\n };\r\n\r\n execute();\r\n });\r\n}\r\n\r\n/**\r\n * 参数过滤undefined, 避免接口处理异常 (不可过滤 null 、 \"\" 、 false 、 0 这些有效值)\r\n */\r\nexport function filterRequestData(data: Record<string, any>) {\r\n const res: Record<string, any> = {};\r\n Object.entries(data).forEach(([k, v]) => {\r\n if (v !== undefined) res[k] = v;\r\n });\r\n return res;\r\n}\r\n\r\n/**\r\n * 请求头过滤空值 (undefined, null, \"\"), 不过滤0和false\r\n */\r\nexport function filterRequestHeader(header: RequestConfigBase['header']) {\r\n const newHeader: Record<string, string> = {};\r\n if (header) {\r\n Object.entries(header).forEach(([k, v]) => {\r\n if (v !== undefined && v !== null && v !== '') newHeader[k] = String(v);\r\n });\r\n }\r\n return newHeader;\r\n}\r\n\r\n/**\r\n * 日志输出\r\n */\r\nfunction logRequestInfo(options: {\r\n config: RequestConfigBase<RequestData> & { url?: string };\r\n fromCache?: boolean;\r\n startTime: number;\r\n status: 'success' | 'fail';\r\n res?: unknown;\r\n e?: unknown;\r\n}) {\r\n const { log } = getBaseToolsConfig();\r\n const { showLog = true } = options.config;\r\n\r\n if (!log || !showLog) return;\r\n\r\n const { config, res, fromCache = false, startTime, status, e } = options;\r\n const { url, data, header, method, logExtra } = config;\r\n const endTime = Date.now();\r\n const fmt = 'YYYY-MM-DD HH:mm:ss.SSS';\r\n\r\n const info: AppLogInfo = {\r\n name: 'request',\r\n status,\r\n url,\r\n data,\r\n method,\r\n header,\r\n fromCache,\r\n startTime: toDayjs(startTime).format(fmt),\r\n endTime: toDayjs(endTime).format(fmt),\r\n duration: endTime - startTime,\r\n ...logExtra,\r\n };\r\n\r\n if (status === 'success') {\r\n info.res = cloneDeep(res); // 深拷贝,避免外部修改对象,造成输出不一致\r\n log('info', info);\r\n } else {\r\n info.e = e;\r\n log('error', info);\r\n }\r\n}\r\n\r\n/**\r\n * 获取 resKey 对应的数据\r\n */\r\nfunction getResult(res: unknown, resKey?: RequestConfigBase['resKey']) {\r\n if (!res || !resKey || typeof res !== 'object') return res;\r\n return getObjectValue(res, resKey);\r\n}\r\n\r\n/**\r\n * 检查缓存\r\n */\r\nfunction checkCache(cacheKey: string) {\r\n const cached = requestCache.get(cacheKey);\r\n if (!cached) return null;\r\n if (cached.expire <= Date.now()) {\r\n requestCache.delete(cacheKey);\r\n return null;\r\n }\r\n return cached.res;\r\n}\r\n\r\n/**\r\n * 处理流式响应\r\n */\r\nasync function handleStreamResponse(response: Response, chunkCallback: ChunkCallback | null) {\r\n if (!response.body) throw new Error('Response body is null');\r\n\r\n const reader = response.body.getReader();\r\n\r\n while (true) {\r\n const { done, value } = await reader.read();\r\n if (done) break;\r\n if (chunkCallback && value) {\r\n chunkCallback({ data: value.buffer });\r\n }\r\n }\r\n\r\n return 'Stream Finished';\r\n}\r\n\r\n/**\r\n * 解析响应数据\r\n */\r\nasync function parseResponse(response: Response, responseType: string) {\r\n let resData: ResponseData;\r\n if (responseType === 'arraybuffer') {\r\n resData = await response.arrayBuffer();\r\n } else if (responseType === 'text') {\r\n resData = await response.text();\r\n } else {\r\n const text = await response.text();\r\n try {\r\n resData = JSON.parse(text);\r\n } catch {\r\n resData = text;\r\n }\r\n }\r\n return resData;\r\n}\r\n\r\n/**\r\n * 转换为 URLSearchParams\r\n */\r\nfunction toSearchParams(data: Record<string, unknown>) {\r\n const params = new URLSearchParams();\r\n for (const key in data) {\r\n const val = data[key];\r\n // undefined 已在 fillData 阶段过滤,此处仅需判断 null\r\n // null 在 Form 中会被转为字符串 \"null\",通常不符合预期,故过滤\r\n if (val === null) continue;\r\n if (Array.isArray(val)) {\r\n val.forEach((v) => params.append(key, typeof v === 'object' ? JSON.stringify(v) : String(v)));\r\n } else {\r\n params.append(key, typeof val === 'object' ? JSON.stringify(val) : String(val));\r\n }\r\n }\r\n return params;\r\n}\r\n\r\n/**\r\n * 转换为 FormData\r\n */\r\nfunction toFormData(data: Record<string, unknown>) {\r\n const formData = new FormData();\r\n for (const key in data) {\r\n const val = data[key];\r\n // undefined 已在 fillData 阶段过滤,此处仅需判断 null\r\n // null 在 Form 中会被转为字符串 \"null\",通常不符合预期,故过滤\r\n if (val === null) continue;\r\n if (Array.isArray(val)) {\r\n val.forEach((v) =>\r\n formData.append(\r\n key,\r\n v instanceof Blob ? v : typeof v === 'object' ? JSON.stringify(v) : String(v),\r\n ),\r\n );\r\n } else {\r\n formData.append(\r\n key,\r\n val instanceof Blob ? val : typeof val === 'object' ? JSON.stringify(val) : String(val),\r\n );\r\n }\r\n }\r\n return formData;\r\n}\r\n","import { cloneDeep } from '@base-web-kits/base-tools-ts';\r\nimport { getBaseToolsConfig } from '../index';\r\nimport type { AppLogInfo } from '../index';\r\n\r\ntype WebApi<Option = any, Res = any, Config = any> = (\r\n option: Option,\r\n config?: Config,\r\n) => Promise<Res>;\r\n\r\n/**\r\n * web api 的调用配置\r\n */\r\nexport type WebApiConfig<Res = any, Err = any> = {\r\n /** 是否显示加载提示, 默认 false. (支持字符串,自定义文本) */\r\n showLoading?: boolean | string;\r\n\r\n /** 操作成功的toast提示, 默认不显示 */\r\n toastSuccess?: ((res: Res) => false | string) | false | string;\r\n\r\n /** 是否显示操作失败的详细错误信息, 默认 true. (支持字符串,自定义文本; 支持根据errMsg判断是否显示, 例如: (e) => !e.errMsg.includes('cancel') */\r\n toastError?: ((e: Err) => boolean | string) | boolean | string;\r\n\r\n /** 是否显示日志, 默认 true */\r\n showLog?: boolean;\r\n\r\n /** 响应数据的转换, 如解密操作 (返回值在成功日志中输出'resMap'字段) */\r\n resMap?: (res: any) => Res;\r\n\r\n /** 成功和失败时,额外输出的日志数据 (可覆盖内部log参数,如'name') */\r\n logExtra?: Record<string, unknown>;\r\n};\r\n\r\n/**\r\n * 拓展 web api, 使其支持loading,toast,log能力\r\n * @param webApi web api\r\n * @param apiName web api 名称 (可选, 用于日志输出, 默认'enhanceWebApi')\r\n * @return 注入拓展能力的promise (默认提示异常和输出日志,不显示进度条和操作成功)\r\n * @example\r\n * const promise = enhanceWebApi(downloadFile, 'downloadFile');\r\n * await promise({ url: 'xx' }, {showLoading: '下载中', toastSuccess: '下载成功'});\r\n */\r\nexport function enhanceWebApi<Option = any, Res = any, Err = any, Config = any>(\r\n webApi: WebApi<Option, Res, Config>,\r\n apiName?: string,\r\n) {\r\n return (option: Option, config?: WebApiConfig<Res, Err> & Config) => {\r\n const finalConfig = config || ({} as WebApiConfig<Res, Err> & Config);\r\n const {\r\n showLoading = false,\r\n toastSuccess = false,\r\n toastError = true,\r\n showLog = true,\r\n resMap,\r\n logExtra,\r\n } = finalConfig;\r\n\r\n const {\r\n log,\r\n toast,\r\n showLoading: showLoadingFn,\r\n hideLoading: hideLoadingFn,\r\n } = getBaseToolsConfig();\r\n const fname = apiName || 'enhanceWebApi'; // webApi.name经过打包后取不到原函数名,不如默认'enhanceWebApi'\r\n\r\n if (showLoading) {\r\n const title = typeof showLoading === 'string' ? showLoading : '';\r\n showLoadingFn?.({ title });\r\n }\r\n\r\n return new Promise<Res>((resolve, reject) => {\r\n webApi(option, finalConfig)\r\n .then((res) => {\r\n if (showLoading) hideLoadingFn?.();\r\n\r\n const finalRes = resMap ? resMap(res) : res;\r\n\r\n if (showLog) {\r\n const logData: AppLogInfo = { name: fname, status: 'success', option, ...logExtra };\r\n\r\n if (resMap) {\r\n logData.res = res; // 输出原始数据\r\n logData.resMap = cloneDeep(finalRes); // 深拷贝处理后数据,避免外部修改对象,造成输出不一致\r\n } else {\r\n logData.res = cloneDeep(res); // 深拷贝原始数据,避免外部修改对象,造成输出不一致\r\n }\r\n\r\n log?.('info', logData);\r\n }\r\n\r\n resolve(finalRes);\r\n\r\n const msg = typeof toastSuccess === 'function' ? toastSuccess(finalRes) : toastSuccess;\r\n if (msg) toast?.({ msg, status: 'success' });\r\n })\r\n .catch((e) => {\r\n if (showLoading) hideLoadingFn?.();\r\n if (showLog) log?.('error', { name: fname, status: 'fail', option, e, ...logExtra });\r\n\r\n const msg = typeof toastError === 'function' ? toastError(e) : toastError;\r\n if (msg) {\r\n toast?.({\r\n msg: typeof msg === 'string' ? msg : `${fname} fail: ${JSON.stringify(e)}`,\r\n status: 'fail',\r\n });\r\n }\r\n\r\n reject(e);\r\n });\r\n });\r\n };\r\n}\r\n","import { enhanceWebApi } from '../async';\r\nimport type { WebApiConfig } from '../async';\r\n\r\n/**\r\n * 上传文件的选项\r\n */\r\nexport type UploadFileOption = {\r\n /** 上传接口地址 */\r\n url: string;\r\n\r\n /** 要上传的文件对象 */\r\n file: File;\r\n\r\n /** 文件对应的 key, 默认'file' (服务端通过这个 key 获取文件的二进制内容) */\r\n name?: string;\r\n\r\n /** 请求头 */\r\n header?: Record<string, string | number>;\r\n\r\n /** 额外的formData参数 */\r\n formData?: Record<string, string | number>;\r\n\r\n /** 超时时间,单位 ms,默认 0(不超时) */\r\n timeout?: number;\r\n};\r\n\r\nexport type OnUploadProgressUpdate = (res: UploadProgressEvent) => void;\r\n\r\nexport type UploadProgressEvent = {\r\n /** 上传进度百分比: 0-100 */\r\n progress: number;\r\n /** 已上传字节数 */\r\n loaded: number;\r\n /** 总字节数 */\r\n total: number;\r\n};\r\n\r\nexport type UploadTask = {\r\n /** 上传进度 */\r\n onProgressUpdate: (callback: OnUploadProgressUpdate) => void;\r\n /** 取消上传 */\r\n abort: () => void;\r\n};\r\n\r\nexport type UploadConfig = {\r\n /** 获取task对象 */\r\n onTaskReady?: (task: UploadTask) => void;\r\n};\r\n\r\nexport type UploadFail = {\r\n message: string;\r\n status: number;\r\n};\r\n\r\nfunction upload(option: UploadFileOption, config?: UploadConfig) {\r\n return new Promise<string>((resolve, reject) => {\r\n const xhr = new XMLHttpRequest();\r\n const { url, file, name = 'file', header, formData, timeout = 0 } = option;\r\n\r\n const fail = (error: UploadFail) => reject(error);\r\n\r\n const success = (responseText: string) => {\r\n resolve(responseText);\r\n };\r\n\r\n // 构造任务对象\r\n let onProgressUpdate: OnUploadProgressUpdate;\r\n const task: UploadTask = {\r\n onProgressUpdate: (callback) => {\r\n onProgressUpdate = callback;\r\n },\r\n abort: () => xhr.abort(),\r\n };\r\n config?.onTaskReady?.(task);\r\n\r\n // 监听进度\r\n xhr.upload.onprogress = (e) => {\r\n if (!e.lengthComputable) return;\r\n const ev: UploadProgressEvent = {\r\n progress: Math.round((e.loaded / e.total) * 100),\r\n loaded: e.loaded,\r\n total: e.total,\r\n };\r\n onProgressUpdate?.(ev);\r\n };\r\n\r\n // 监听事件\r\n xhr.onload = () => {\r\n if (xhr.status >= 200 && xhr.status < 300) {\r\n success(xhr.responseText);\r\n } else {\r\n fail({ message: `上传失败`, status: xhr.status });\r\n }\r\n };\r\n xhr.onerror = () => fail({ message: '网络错误', status: 0 });\r\n xhr.ontimeout = () => fail({ message: '上传超时', status: -1 });\r\n xhr.onabort = () => fail({ message: '用户取消', status: -2 });\r\n\r\n // 设置请求方法和 URL\r\n xhr.open('POST', url);\r\n\r\n // 设置请求头\r\n if (header) {\r\n Object.entries(header).forEach(([k, v]) => {\r\n if (v !== undefined && v !== null && v !== '') xhr.setRequestHeader(k, String(v));\r\n });\r\n }\r\n\r\n // 设置超时时间\r\n xhr.timeout = timeout;\r\n\r\n // 组装 FormData\r\n const data = new FormData();\r\n if (formData) {\r\n Object.entries(formData).forEach(([k, v]) => {\r\n if (v !== undefined && v !== null) data.append(k, String(v));\r\n });\r\n }\r\n\r\n // OSS直传的file字段必须写在最后\r\n data.append(name, file);\r\n\r\n // 发送请求\r\n xhr.send(data);\r\n });\r\n}\r\n\r\n/**\r\n * 上传文件\r\n * @param option 上传文件的选项\r\n * @param config 配置项\r\n * @example\r\n * // 上传\r\n * const res = await uploadFile({ url: 'https://xx', file: file});\r\n *\r\n * // 监听上传进度\r\n * const res = await uploadFile({ url: 'https://xx', file: file}, {\r\n * onTaskReady: (task) =>\r\n * task.onProgressUpdate((res) => console.log('上传进度:', res.progress)),\r\n * });\r\n *\r\n * // 解析上传结果\r\n * console.log('uploadFile ok', JSON.parse(res));\r\n */\r\nexport function uploadFile(option: UploadFileOption, config?: UploadConfig & WebApiConfig) {\r\n return enhanceWebApi(upload, 'uploadFile')(option, config);\r\n}\r\n","const WK = {\r\n val: '__l_val',\r\n exp: '__l_exp',\r\n wrap: '__l_wrap',\r\n} as const;\r\n\r\n/**\r\n * 写入 localStorage(自动 JSON 序列化)\r\n * @param key 键名\r\n * @param value 任意可序列化的值:对象、数组、字符串、数字、布尔值。(`null` 或 `undefined` 会自动移除该键)\r\n * @param days 过期天数(从当前时间起算)\r\n * @example\r\n * setLocalStorage('user', { id: 1, name: 'Alice' }); // 对象\r\n * setLocalStorage('age', 18); // 数字\r\n * setLocalStorage('vip', true); // 布尔值\r\n * setLocalStorage('token', 'abc123', 7); // 7 天后过期\r\n */\r\nexport function setLocalStorage(key: string, value: unknown, days?: number) {\r\n if (value === undefined || value === null) {\r\n removeLocalStorage(key);\r\n return;\r\n }\r\n\r\n let toStore: unknown = value;\r\n if (typeof days === 'number' && days > 0) {\r\n const ms = days * 24 * 60 * 60 * 1000;\r\n toStore = {\r\n [WK.wrap]: true,\r\n [WK.val]: value,\r\n [WK.exp]: Date.now() + ms,\r\n };\r\n }\r\n\r\n localStorage.setItem(key, JSON.stringify(toStore));\r\n}\r\n\r\n/**\r\n * 读取 localStorage(自动 JSON 反序列化)\r\n * 若值为合法 JSON,则返回反序列化后的数据;\r\n * 若值非 JSON(如外部写入的纯字符串),则原样返回字符串。\r\n * 不存在时返回 `null`。\r\n * @param key 键名\r\n * @returns 解析后的值或 `null`\r\n * @example\r\n * const user = getLocalStorage<{ id: number; name: string }>('user');\r\n * const age = getLocalStorage<number>('age');\r\n * const vip = getLocalStorage<boolean>('vip');\r\n */\r\nexport function getLocalStorage<T = unknown>(key: string): T | null {\r\n const raw = localStorage.getItem(key);\r\n if (raw === null) return null;\r\n try {\r\n const parsed = JSON.parse(raw);\r\n\r\n if (parsed && typeof parsed === 'object' && WK.wrap in parsed && WK.exp in parsed) {\r\n if (Date.now() > parsed[WK.exp]) {\r\n removeLocalStorage(key);\r\n return null;\r\n }\r\n return parsed[WK.val] as T;\r\n }\r\n return parsed as T;\r\n } catch {\r\n return raw as T;\r\n }\r\n}\r\n\r\n/**\r\n * 移除 localStorage 指定键\r\n * @param key 键名\r\n * @example\r\n * removeLocalStorage('token');\r\n */\r\nexport function removeLocalStorage(key: string) {\r\n localStorage.removeItem(key);\r\n}\r\n","/**\r\n * 获取url的查询参数值\r\n * @param key 参数名\r\n * @param url 完整 URL 或仅查询串(如 \"a=1&b=2\")\r\n * @returns 解码后的参数值 (若不存在|\"null\"|\"undefined\",则返回 null)\r\n * @example\r\n * const q = getUrlParam('q'); // 默认当前地址\r\n * const q = getUrlParam('q', 'https://a.com/?q=%E6%B5%8B%E8%AF%95'); // \"测试\"\r\n * const a = getUrlParam('a', 'a=1'); // \"1\"\r\n * const list = getUrlParam('list', 'list=[1,2]'); // \"[1,2]\"\r\n * const list = getUrlParam('list', 'list=null'); // null\r\n * const list = getUrlParam('list', 'list=undefined'); // null\r\n */\r\nexport function getUrlParam(key: string, url = window.location.href) {\r\n const searchParams = new URL(url.includes('?') ? url : `?${url}`, 'http://localhost')\r\n .searchParams;\r\n const value = searchParams.get(key);\r\n return value === 'null' || value === 'undefined' ? null : value;\r\n}\r\n\r\n/**\r\n * 获取url的查询参数值,并转为number类型\r\n * @param key 参数名\r\n * @param url 完整 URL 或仅查询串(如 \"a=1&b=2\")\r\n * @returns 解码后的参数值 (若不存在|\"非数字字符串\",则返回 null)\r\n * @example\r\n * const a = getUrlNumber('a'); // 默认当前地址\r\n * const a = getUrlNumber('a', 'https://a.com/?a=1'); // 1\r\n * const a = getUrlNumber('a', 'a=1'); // 1\r\n * const a = getUrlNumber('a', 'a=1.2'); // 1.2\r\n * const a = getUrlNumber('a', 'a=abc'); // null\r\n */\r\nexport function getUrlNumber(key: string, url = window.location.href) {\r\n const str = getUrlParam(key, url);\r\n if (!str) return null;\r\n\r\n const num = Number(str);\r\n return isNaN(num) ? null : num;\r\n}\r\n\r\n/**\r\n * 获取url的所有查询参数值\r\n * @param url 完整 URL 或仅查询串(如 \"a=1&b=2\")\r\n * @returns 解码后的键值对象(无参数返回空对象; \"null\"|\"undefined\"的参数会被忽略)\r\n * @example\r\n * const params = getUrlParams(); // 默认当前地址\r\n * const params = getUrlParams('a=1&b=2'); // { a: \"1\", b: \"2\" }\r\n * const params = getUrlParams('https://a.com/?a=1&b=2'); // { a: \"1\", b: \"2\" }\r\n * const params = getUrlParams('a=1&b=null'); // { a: \"1\" }\r\n * const params = getUrlParams('a=1&b=undefined'); // { a: \"1\" }\r\n */\r\nexport function getUrlParams(url = window.location.href) {\r\n const searchParams = new URL(url.includes('?') ? url : `?${url}`, 'http://localhost')\r\n .searchParams;\r\n const result: Record<string, string> = {};\r\n\r\n for (const [key, value] of searchParams.entries()) {\r\n if (value !== 'null' && value !== 'undefined') {\r\n result[key] = value;\r\n }\r\n }\r\n\r\n return result;\r\n}\r\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAOA,SAAsB,SAAS,MAA6B;AAAA;AAC1D,QAAI,OAAO,SAAS,SAAU,QAAO,OAAO,sBAAQ,EAAE;AAGtD,QAAI,UAAU,aAAa,OAAO,UAAU,UAAU,cAAc,YAAY;AAC9E,UAAI;AACF,cAAM,UAAU,UAAU,UAAU,IAAI;AACxC;AAAA,MAEF,SAAS,GAAG;AAAA,MAEZ;AAAA,IACF;AAGA,WAAO,IAAI,QAAc,CAAC,SAAS,WAAW;AAC5C,UAAI;AACF,cAAM,WAAW,SAAS,cAAc,UAAU;AAClD,iBAAS,QAAQ;AAGjB,iBAAS,aAAa,YAAY,EAAE;AACpC,iBAAS,MAAM,WAAW;AAC1B,iBAAS,MAAM,MAAM;AACrB,iBAAS,MAAM,QAAQ;AACvB,iBAAS,MAAM,UAAU;AACzB,iBAAS,MAAM,gBAAgB;AAE/B,iBAAS,KAAK,YAAY,QAAQ;AAGlC,iBAAS,MAAM;AACf,iBAAS,OAAO;AAGhB,iBAAS,kBAAkB,GAAG,SAAS,MAAM,MAAM;AAEnD,cAAM,KAAK,SAAS,YAAY,MAAM;AACtC,iBAAS,KAAK,YAAY,QAAQ;AAElC,YAAI,IAAI;AACN,kBAAQ;AAAA,QACV,OAAO;AACL,iBAAO,IAAI,MAAM,oCAAoC,CAAC;AAAA,QACxD;AAAA,MACF,SAAS,GAAG;AACV,eAAO,CAAC;AAAA,MACV;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AASA,SAAsB,SAAS,MAA6B;AAAA;AAC1D,UAAM,IAAI,OAAO,sBAAQ,EAAE;AAC3B,QAAI,kBAAkB,GAAG;AACvB,YAAM,QAAQ,WAAW,CAAC;AAC1B,YAAM,eAAe;AAAA,QACnB,aAAa,IAAI,KAAK,CAAC,CAAC,GAAG,EAAE,MAAM,YAAY,CAAC;AAAA,QAChD,cAAc,IAAI,KAAK,CAAC,KAAK,GAAG,EAAE,MAAM,aAAa,CAAC;AAAA,MACxD,CAAC;AACD;AAAA,IACF;AACA,WAAO,iBAAiB,CAAC;AAAA,EAC3B;AAAA;AAUA,SAAsB,SAAS,MAA2B;AAAA;AACxD,QAAI,kBAAkB,GAAG;AACvB,YAAM,EAAE,MAAAA,OAAM,KAAK,IAAI,eAAe,IAAI;AAC1C,YAAM,eAAe;AAAA,QACnB,aAAa,IAAI,KAAK,CAACA,KAAI,GAAG,EAAE,MAAM,YAAY,CAAC;AAAA,QACnD,cAAc,IAAI,KAAK,CAAC,IAAI,GAAG,EAAE,MAAM,aAAa,CAAC;AAAA,MACvD,CAAC;AACD;AAAA,IACF;AACA,UAAM,EAAE,KAAK,IAAI,eAAe,IAAI;AACpC,WAAO,iBAAiB,IAAI;AAAA,EAC9B;AAAA;AAUA,SAAsB,UAAU,OAA8D;AAAA;AAC5F,UAAM,OAAO,MAAM,YAAY,KAAK;AACpC,QAAI,CAAC,KAAM,OAAM,IAAI,MAAM,0BAA0B;AACrD,QAAI,kBAAkB,GAAG;AACvB,YAAM,OAAO,KAAK,QAAQ;AAC1B,YAAM,eAAe,EAAE,CAAC,IAAI,GAAG,KAAK,CAAC;AACrC;AAAA,IACF;AACA,UAAM,IAAI,MAAM,qCAAqC;AAAA,EACvD;AAAA;AASA,SAAsB,QAAQ,KAA4B;AAAA;AACxD,UAAM,IAAI,OAAO,oBAAO,EAAE;AAC1B,QAAI,kBAAkB,GAAG;AACvB,YAAM,eAAe;AAAA,QACnB,iBAAiB,IAAI,KAAK,CAAC,CAAC,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAAA,QACxD,cAAc,IAAI,KAAK,CAAC,CAAC,GAAG,EAAE,MAAM,aAAa,CAAC;AAAA,MACpD,CAAC;AACD;AAAA,IACF;AACA,UAAM,SAAS,CAAC;AAAA,EAClB;AAAA;AAUA,SAAsB,SAAS,MAA2B;AAAA;AACxD,QAAI,kBAAkB,GAAG;AACvB,YAAM,OAAO,KAAK,QAAQ;AAC1B,YAAM,eAAe,EAAE,CAAC,IAAI,GAAG,KAAK,CAAC;AACrC;AAAA,IACF;AACA,UAAM,IAAI,MAAM,oCAAoC;AAAA,EACtD;AAAA;AASA,SAAsB,QAAQ,KAA4B;AAAA;AACxD,UAAM,IAAI,OAAO,oBAAO,EAAE;AAC1B,QAAI,kBAAkB,GAAG;AACvB,YAAM,QAAQ,EACX,QAAQ,eAAe,IAAI,EAC3B,QAAQ,cAAc,EAAE,EACxB,QAAQ,wBAAwB,EAAE,EAClC,QAAQ,UAAU,IAAI,EACtB,KAAK;AACR,YAAM,eAAe;AAAA,QACnB,YAAY,IAAI,KAAK,CAAC,CAAC,GAAG,EAAE,MAAM,WAAW,CAAC;AAAA,QAC9C,cAAc,IAAI,KAAK,CAAC,KAAK,GAAG,EAAE,MAAM,aAAa,CAAC;AAAA,MACxD,CAAC;AACD;AAAA,IACF;AACA,UAAM,SAAS,CAAC;AAAA,EAClB;AAAA;AAcA,SAAsB,UAAU,MAAoD;AAAA;AAClF,UAAM,OAAO,MAAM,QAAQ,IAAI,IAAI,OAAO,CAAC;AAC3C,UAAM,aAAa,CAAC,MAClB,EACG,QAAQ,MAAM,OAAO,EACrB,QAAQ,MAAM,MAAM,EACpB,QAAQ,MAAM,MAAM,EACpB,QAAQ,MAAM,QAAQ,EACtB,QAAQ,MAAM,OAAO;AAC1B,UAAM,QAAQ,MAAM;AAClB,YAAM,MAAM,KACT,IAAI,CAAC,MAAM,OAAO,EAAE,IAAI,CAAC,MAAM,OAAO,WAAW,OAAO,CAAC,CAAC,CAAC,OAAO,EAAE,KAAK,EAAE,CAAC,OAAO,EACnF,KAAK,EAAE;AACV,aAAO,UAAU,GAAG;AAAA,IACtB,GAAG;AACH,UAAM,MAAM,KAAK,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,MAAM,OAAO,CAAC,CAAC,EAAE,KAAK,GAAI,CAAC,EAAE,KAAK,IAAI;AACzE,UAAM,MAAM,KACT;AAAA,MAAI,CAAC,MACJ,EACG,IAAI,CAAC,MAAM;AACV,cAAM,IAAI,OAAO,CAAC;AAClB,cAAM,YAAY,SAAS,KAAK,CAAC;AACjC,cAAM,UAAU,EAAE,QAAQ,MAAM,IAAI;AACpC,eAAO,YAAY,IAAI,OAAO,MAAM;AAAA,MACtC,CAAC,EACA,KAAK,GAAG;AAAA,IACb,EACC,KAAK,IAAI;AACZ,QAAI,kBAAkB,GAAG;AACvB,YAAM,eAAe;AAAA,QACnB,aAAa,IAAI,KAAK,CAAC,IAAI,GAAG,EAAE,MAAM,YAAY,CAAC;AAAA,QACnD,6BAA6B,IAAI,KAAK,CAAC,GAAG,GAAG,EAAE,MAAM,4BAA4B,CAAC;AAAA,QAClF,YAAY,IAAI,KAAK,CAAC,GAAG,GAAG,EAAE,MAAM,WAAW,CAAC;AAAA,QAChD,cAAc,IAAI,KAAK,CAAC,GAAG,GAAG,EAAE,MAAM,aAAa,CAAC;AAAA,MACtD,CAAC;AACD;AAAA,IACF;AACA,UAAM,SAAS,GAAG;AAAA,EACpB;AAAA;AAEA,SAAe,YAAY,OAA+C;AAAA;AACxE,QAAI,iBAAiB,KAAM,QAAO;AAClC,QAAI,iBAAiB;AACnB,aAAO,MAAM,IAAI,QAAc,CAAC,SAAS,WAAW;AAClD,cAAM;AAAA,UACJ,CAAC,MAAO,IAAI,QAAQ,CAAC,IAAI,OAAO,IAAI,MAAM,sBAAsB,CAAC;AAAA,UACjE;AAAA,QACF;AAAA,MACF,CAAC;AACH,UAAM,WAAW,OAAO,gBAAgB,eAAe,iBAAiB;AACxE,QAAI,UAAU;AACZ,YAAM,MAAM,SAAS,cAAc,QAAQ;AAC3C,UAAI,QAAS,MAAsB;AACnC,UAAI,SAAU,MAAsB;AACpC,YAAM,MAAM,IAAI,WAAW,IAAI;AAC/B,iCAAK,UAAU,OAAsB,GAAG;AACxC,aAAO,MAAM,IAAI,QAAc,CAAC,SAAS,WAAW;AAClD,YAAI,OAAO,CAAC,MAAO,IAAI,QAAQ,CAAC,IAAI,OAAO,IAAI,MAAM,sBAAsB,CAAC,GAAI,WAAW;AAAA,MAC7F,CAAC;AAAA,IACH;AACA,WAAO;AAAA,EACT;AAAA;AAEA,SAAS,oBAAoB;AAC3B,SAAO,CAAC,EACN,UAAU,aACV,OAAO,UAAU,UAAU,UAAU,cACrC,OAAO,kBAAkB;AAE7B;AAEA,SAAe,eAAe,OAA6B;AAAA;AACzD,UAAM,UAAU,UAAW,MAAM,CAAC,IAAI,cAAc,KAAK,CAAC,CAAC;AAAA,EAC7D;AAAA;AAEA,SAAS,WAAW,MAAc;AAChC,QAAM,MAAM,SAAS,cAAc,KAAK;AACxC,MAAI,YAAY;AAChB,SAAO,IAAI,eAAe;AAC5B;AAEA,SAAS,eAAe,MAAY;AAhRpC;AAiRE,QAAM,YAAY,SAAS,cAAc,KAAK;AAC9C,YAAU,YAAY,KAAK,UAAU,IAAI,CAAC;AAC1C,QAAM,OACJ,gBAAgB,WAAW,UAAK,cAAL,YAAkB,UAAU,YAAa,UAAU;AAChF,QAAM,OAAO,UAAU,eAAe;AACtC,SAAO,EAAE,MAAM,KAAK;AACtB;AAEA,SAAS,iBAAiB,MAAc;AACtC,SAAO,IAAI,QAAc,CAAC,SAAS,WAAW;AAC5C,QAAI;AACF,YAAM,MAAM,SAAS,cAAc,KAAK;AACxC,UAAI,kBAAkB;AACtB,UAAI,MAAM,WAAW;AACrB,UAAI,MAAM,MAAM;AAChB,UAAI,MAAM,QAAQ;AAClB,UAAI,MAAM,UAAU;AACpB,UAAI,MAAM,gBAAgB;AAC1B,UAAI,YAAY;AAChB,eAAS,KAAK,YAAY,GAAG;AAC7B,YAAM,YAAY,OAAO,aAAa;AACtC,YAAMC,SAAQ,SAAS,YAAY;AACnC,MAAAA,OAAM,mBAAmB,GAAG;AAC5B,6CAAW;AACX,6CAAW,SAASA;AACpB,YAAM,KAAK,SAAS,YAAY,MAAM;AACtC,eAAS,KAAK,YAAY,GAAG;AAC7B,6CAAW;AACX,UAAI,IAAI;AACN,gBAAQ;AAAA,MACV,OAAO;AACL,eAAO,IAAI,MAAM,oCAAoC,CAAC;AAAA,MACxD;AAAA,IACF,SAAS,GAAG;AACV,aAAO,CAAC;AAAA,IACV;AAAA,EACF,CAAC;AACH;;;AClRA,IAAM,YAAuB,CAAC;AAKvB,SAAS,qBAAqB;AACnC,SAAO;AACT;AAqBO,SAAS,mBAAmB,WAAsB;AACvD,SAAO,OAAO,WAAW,SAAS;AACpC;;;AC1DO,SAAS,UAAU,MAAc,OAAe,MAAc;AACnE,QAAM,OAAO,oBAAI,KAAK;AACtB,OAAK,QAAQ,KAAK,QAAQ,IAAI,OAAO,KAAK,KAAK,KAAK,GAAI;AACxD,QAAM,UAAU,WAAW,KAAK,YAAY,CAAC;AAC7C,WAAS,SAAS,GAAG,IAAI,IAAI,mBAAmB,KAAK,CAAC,KAAK,OAAO;AACpE;AASO,SAAS,UAAU,MAA6B;AAtBvD;AAuBE,QAAM,QAAQ,KAAK,SAAS,MAAM;AAClC,QAAM,QAAQ,MAAM,MAAM,KAAK,IAAI,GAAG;AACtC,MAAI,MAAM,WAAW,GAAG;AACtB,UAAM,KAAI,WAAM,IAAI,MAAV,mBAAa,MAAM,KAAK;AAClC,WAAO,IAAI,mBAAmB,CAAC,IAAI;AAAA,EACrC;AACA,SAAO;AACT;AASO,SAAS,aAAa,MAAc;AACzC,WAAS,SAAS,GAAG,IAAI;AAC3B;;;ACrCO,SAAS,QAAgB;AAC9B,MAAI,OAAO,cAAc,YAAa,QAAO;AAC7C,UAAQ,UAAU,aAAa,IAAI,YAAY;AACjD;AAKO,SAAS,WAAoB;AAClC,QAAM,KAAK,MAAM;AACjB,SAAO,mEAAmE,KAAK,EAAE;AACnF;AAKO,SAAS,WAAoB;AAClC,QAAM,KAAK,MAAM;AACjB,SAAO,mCAAmC,KAAK,EAAE,KAAK,CAAC,UAAU,KAAK,EAAE;AAC1E;AAKO,SAAS,OAAgB;AAC9B,SAAO,CAAC,SAAS,KAAK,CAAC,SAAS;AAClC;AAKO,SAAS,QAAiB;AAC/B,QAAM,KAAK,MAAM;AACjB,SAAO,oBAAoB,KAAK,EAAE;AACpC;AAKO,SAAS,YAAqB;AACnC,QAAM,KAAK,MAAM;AACjB,SAAO,WAAW,KAAK,EAAE;AAC3B;AAKO,SAAS,WAAoB;AAClC,QAAM,KAAK,MAAM;AACjB,SAAO,kBAAkB,KAAK,EAAE;AAClC;AAMO,SAAS,WAAoB;AAClC,QAAM,KAAK,MAAM;AACjB,SAAO,YAAY,KAAK,EAAE,KAAK,CAAC,SAAS,KAAK,EAAE,KAAK,CAAC,SAAS,KAAK,EAAE,KAAK,CAAC,WAAW,KAAK,EAAE;AAChG;AAKO,SAAS,mBAA4B;AAC1C,MAAI,OAAO,WAAW,YAAa,QAAO;AAC1C,SAAO,kBAAkB,UAAU,UAAU,iBAAiB;AAChE;AAKO,SAAS,sBAA8B;AAC5C,MAAI,OAAO,WAAW,YAAa,QAAO;AAC1C,SAAO,OAAO,oBAAoB;AACpC;AAKO,SAAS,iBAAgC;AAC9C,QAAM,KAAK,MAAM;AAEjB,MAAI,YAAY,KAAK,EAAE,EAAG,QAAO;AACjC,MAAI,YAAY,KAAK,EAAE,EAAG,QAAO;AACjC,MAAI,aAAa,KAAK,EAAE,EAAG,QAAO;AAClC,MAAI,SAAS,KAAK,EAAE,EAAG,QAAO;AAC9B,MAAI,SAAS,KAAK,EAAE,EAAG,QAAO;AAC9B,MAAI,gBAAgB,KAAK,EAAE,EAAG,QAAO;AAErC,SAAO;AACT;AAKO,SAAS,oBAAmC;AACjD,QAAM,KAAK,MAAM;AAEjB,QAAM,kBAAkB;AAAA,IACtB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AAEA,aAAW,WAAW,iBAAiB;AACrC,UAAM,UAAU,GAAG,MAAM,OAAO;AAChC,QAAI,WAAW,QAAQ,CAAC,GAAG;AACzB,aAAO,QAAQ,CAAC;AAAA,IAClB;AAAA,EACF;AAEA,SAAO;AACT;AAKO,SAAS,QAAgB;AAC9B,QAAM,KAAK,MAAM;AAEjB,MAAI,WAAW,KAAK,EAAE,EAAG,QAAO;AAChC,MAAI,UAAU,KAAK,EAAE,EAAG,QAAO;AAC/B,MAAI,SAAS,KAAK,EAAE,EAAG,QAAO;AAC9B,MAAI,oBAAoB,KAAK,EAAE,EAAG,QAAO;AACzC,MAAI,WAAW,KAAK,EAAE,EAAG,QAAO;AAEhC,SAAO;AACT;;;ACnIO,SAAS,iBAAiB;AAC/B,SAAO,OAAO,cAAc,SAAS,gBAAgB,eAAe,SAAS,KAAK;AACpF;AAMO,SAAS,kBAAkB;AAChC,SAAO,OAAO,eAAe,SAAS,gBAAgB,gBAAgB,SAAS,KAAK;AACtF;AAOO,SAAS,qBAAqB;AACnC,QAAM,MAAM,SAAS;AACrB,QAAM,OAAO,SAAS;AACtB,SAAO,OAAO,eAAe,IAAI,aAAa,KAAK,aAAa;AAClE;AAOO,SAAS,sBAAsB;AACpC,QAAM,MAAM,SAAS;AACrB,QAAM,OAAO,SAAS;AACtB,SAAO,OAAO,eAAe,IAAI,cAAc,KAAK,cAAc;AACpE;AASO,SAAS,eAAe,KAAa,WAA2B,UAAU;AAC/E,MAAI,oBAAoB,SAAS,gBAAgB,OAAO;AACtD,WAAO,SAAS,EAAE,KAAK,SAAS,CAAC;AAAA,EACnC,OAAO;AACL,WAAO,SAAS,GAAG,GAAG;AAAA,EACxB;AACF;AAQO,SAAS,aAAa,IAAa,SAAS,GAAG;AACpD,QAAM,OAAO,GAAG,sBAAsB;AACtC,QAAM,QAAQ,eAAe;AAC7B,QAAM,SAAS,gBAAgB;AAC/B,SACE,KAAK,UAAU,CAAC,UAChB,KAAK,SAAS,CAAC,UACf,KAAK,OAAO,SAAS,UACrB,KAAK,QAAQ,QAAQ;AAEzB;AAQO,SAAS,iBAAiB;AAC/B,QAAM,OAAO,SAAS;AACtB,MAAI,KAAK,QAAQ,eAAe,OAAQ;AACxC,QAAM,IAAI,KAAK,MAAM,OAAO,WAAW,OAAO,eAAe,CAAC;AAC9D,OAAK,QAAQ,aAAa;AAC1B,OAAK,QAAQ,cAAc,OAAO,CAAC;AACnC,OAAK,MAAM,WAAW;AACtB,OAAK,MAAM,MAAM,IAAI,CAAC;AACtB,OAAK,MAAM,OAAO;AAClB,OAAK,MAAM,QAAQ;AACnB,OAAK,MAAM,QAAQ;AACrB;AAOO,SAAS,mBAAmB;AACjC,QAAM,OAAO,SAAS;AACtB,MAAI,KAAK,QAAQ,eAAe,OAAQ;AACxC,QAAM,IAAI,OAAO,KAAK,QAAQ,eAAe,CAAC;AAC9C,OAAK,MAAM,WAAW;AACtB,OAAK,MAAM,MAAM;AACjB,OAAK,MAAM,OAAO;AAClB,OAAK,MAAM,QAAQ;AACnB,OAAK,MAAM,QAAQ;AACnB,SAAO,KAAK,QAAQ;AACpB,SAAO,KAAK,QAAQ;AACpB,SAAO,SAAS,GAAG,CAAC;AACtB;;;AChGA,SAAsB,SAAS,KAAoB,WAAW,IAAI;AAAA;AAChE,QAAI,CAAC,IAAK;AAEV,QAAI,UAAU;AACd,QAAI,aAAa;AACjB,QAAI;AACF,UAAI,eAAe,MAAM;AAEvB,kBAAU,IAAI,gBAAgB,GAAG;AACjC,qBAAa;AAAA,MACf,WAAW,IAAI,SAAS,UAAU,GAAG;AAEnC,kBAAU;AAAA,MACZ,OAAO;AACL,YAAI,UAAU;AAEZ,gBAAM,MAAM,MAAM,MAAM,GAAG;AAC3B,cAAI,CAAC,IAAI,GAAI,OAAM,IAAI,MAAM,eAAe,IAAI,MAAM,SAAI,GAAG,EAAE;AAC/D,gBAAM,OAAO,MAAM,IAAI,KAAK;AAC5B,oBAAU,IAAI,gBAAgB,IAAI;AAClC,uBAAa;AAAA,QACf,OAAO;AAEL,oBAAU;AAAA,QACZ;AAAA,MACF;AAKA,YAAM,IAAI,SAAS,cAAc,GAAG;AACpC,QAAE,OAAO;AACT,QAAE,WAAW;AACb,eAAS,KAAK,YAAY,CAAC;AAC3B,QAAE,MAAM;AACR,eAAS,KAAK,YAAY,CAAC;AAAA,IAC7B,UAAE;AACA,UAAI,YAAY;AACd,mBAAW,MAAM,IAAI,gBAAgB,OAAO,GAAG,GAAG;AAAA,MACpD;AAAA,IACF;AAAA,EACF;AAAA;AAWA,SAAsB,eAAe,KAA0B;AAAA;AAC7D,UAAM,EAAE,MAAM,SAAS,QAAQ,YAAY,OAAO,IAAI;AAEtD,QAAI,SAAS,OAAO,UAAU,IAAK,OAAM,IAAI,MAAM,GAAG,MAAM,SAAI,UAAU,SAAI,OAAO,GAAG,EAAE;AAG1F,QAAI,KAAK,KAAK,SAAS,kBAAkB,GAAG;AAC1C,YAAM,MAAM,MAAM,KAAK,KAAK;AAC5B,YAAM,KAAK,MAAM,GAAG;AAAA,IACtB;AAGA,UAAM,WAAW,uBAAuB,QAAQ,qBAAqB,CAAC;AACtE,WAAO,EAAE,MAAM,MAAM,SAAS;AAAA,EAChC;AAAA;AASO,SAAS,uBAAuB,aAAsB;AAtF7D;AAuFE,MAAI,CAAC,YAAa,QAAO;AAGzB,QAAM,UAAU,qCAAqC,KAAK,WAAW;AACrE,MAAI,mCAAU,IAAI;AAChB,QAAI;AACF,aAAO,mBAAmB,QAAQ,CAAC,EAAE,KAAK,CAAC,EAAE,QAAQ,YAAY,EAAE;AAAA,IACrE,SAAQ;AACN,aAAO,QAAQ,CAAC,EAAE,KAAK,EAAE,QAAQ,YAAY,EAAE;AAAA,IACjD;AAAA,EACF;AAGA,QAAM,MAAM,gDAAgD,KAAK,WAAW;AAC5E,MAAI,IAAK,UAAQ,SAAI,CAAC,MAAL,YAAU,IAAI,CAAC,GAAG,KAAK,EAAE,QAAQ,YAAY,EAAE;AAEhE,SAAO;AACT;AAUA,SAAsB,OACpB,KACA,OACA;AAAA;AACA,WAAO,IAAI,QAAc,CAAC,SAAS,WAAW;AAC5C,UAAI,MAAM,GAAG,EAAG,QAAO,QAAQ;AAE/B,YAAM,SAAS,SAAS,cAAc,QAAQ;AAC9C,aAAO,OAAO;AACd,aAAO,MAAM;AAEb,UAAI,OAAO;AACT,cAAM,OAAO,OAAO,KAAK,KAAK;AAC9B,aAAK,QAAQ,CAAC,QAAQ;AACpB,gBAAM,IAAI,MAAM,GAAG;AACnB,cAAI,MAAM,QAAQ,MAAM,UAAa,MAAM,MAAO;AAClD,iBAAO,aAAa,KAAK,OAAO,MAAM,YAAY,KAAK,CAAC;AAAA,QAC1D,CAAC;AAAA,MACH;AAEA,aAAO,SAAS,MAAM,QAAQ;AAC9B,aAAO,UAAU,CAAC,MAAM,OAAO,CAAC;AAEhC,eAAS,KAAK,YAAY,MAAM;AAAA,IAClC,CAAC;AAAA,EACH;AAAA;AAWO,SAAS,MAAM,KAAa;AACjC,QAAM,SAAS,IAAI,IAAI,KAAK,SAAS,OAAO,EAAE;AAC9C,QAAM,SAAS,MAAM,KAAK,SAAS,iBAAiB,aAAa,CAAC;AAClE,SAAO,OAAO,KAAK,CAAC,MAAM;AACxB,UAAMC,OAAM,EAAE,aAAa,KAAK;AAChC,WAAOA,QAAO,IAAI,IAAIA,MAAK,SAAS,OAAO,EAAE,SAAS;AAAA,EACxD,CAAC;AACH;AAUA,SAAsB,QACpB,MACA,OACA;AAAA;AACA,WAAO,IAAI,QAAc,CAAC,SAAS,WAAW;AAC5C,UAAI,OAAO,IAAI,EAAG,QAAO,QAAQ;AAEjC,YAAM,OAAO,SAAS,cAAc,MAAM;AAC1C,WAAK,MAAM;AACX,WAAK,OAAO;AAEZ,UAAI,OAAO;AACT,cAAM,OAAO,OAAO,KAAK,KAAK;AAC9B,aAAK,QAAQ,CAAC,QAAQ;AACpB,gBAAM,IAAI,MAAM,GAAG;AACnB,cAAI,MAAM,QAAQ,MAAM,OAAW;AACnC,eAAK,aAAa,KAAK,OAAO,CAAC,CAAC;AAAA,QAClC,CAAC;AAAA,MACH;AAEA,WAAK,SAAS,MAAM,QAAQ;AAC5B,WAAK,UAAU,CAAC,MAAM,OAAO,CAAC;AAE9B,eAAS,KAAK,YAAY,IAAI;AAAA,IAChC,CAAC;AAAA,EACH;AAAA;AASO,SAAS,OAAO,MAAc;AACnC,QAAM,SAAS,IAAI,IAAI,MAAM,SAAS,OAAO,EAAE;AAC/C,QAAM,OAAO,MAAM,KAAK,SAAS,iBAAiB,8BAA8B,CAAC;AACjF,SAAO,KAAK,KAAK,CAAC,MAAM;AACtB,UAAM,IAAI,EAAE,aAAa,MAAM;AAC/B,WAAO,KAAK,IAAI,IAAI,GAAG,SAAS,OAAO,EAAE,SAAS;AAAA,EACpD,CAAC;AACH;AASO,SAAS,aAAa,KAAa;AACxC,SAAO,IAAI,QAA0B,CAAC,SAAS,WAAW;AACxD,UAAM,MAAM,IAAI,MAAM;AACtB,QAAI,SAAS,MAAM,QAAQ,GAAG;AAC9B,QAAI,UAAU,CAAC,MAAM,OAAO,CAAC;AAC7B,QAAI,MAAM;AAAA,EACZ,CAAC;AACH;;;AChOA,OAAO,WAAW;AAClB,OAAO,uBAAuB;AAC9B,OAAO,SAAS;AAChB,OAAO,cAAc;AACrB,OAAO,kBAAkB;AACzB,OAAO,oBAAoB;AAC3B,OAAO;AAGP,MAAM,OAAO,iBAAiB;AAC9B,MAAM,OAAO,GAAG;AAChB,MAAM,OAAO,QAAQ;AACrB,MAAM,OAAO,YAAY;AACzB,MAAM,OAAO,cAAc;AAC3B,MAAM,OAAO,OAAO;AAsCb,SAAS,QAAQ,GAAc,KAAwB;AAC5D,MAAI,MAAM,QAAQ,MAAM,OAAW,QAAO,MAAM;AAChD,MAAI,OAAO,MAAM,UAAU;AACzB,UAAM,IAAI,OAAO,KAAK,MAAM,CAAC,CAAC;AAC9B,WAAO,MAAM,EAAE,WAAW,KAAK,IAAI,MAAO,GAAG,GAAG;AAAA,EAClD;AACA,MAAI,OAAO,MAAM,UAAU;AACzB,UAAM,IAAI,EAAE,KAAK;AACjB,QAAI,WAAW,KAAK,CAAC,EAAG,QAAO,MAAM,OAAO,CAAC,IAAI,KAAM,GAAG;AAC1D,QAAI,WAAW,KAAK,CAAC,EAAG,QAAO,MAAM,OAAO,CAAC,GAAG,GAAG;AACnD,QAAI,sBAAsB,KAAK,CAAC,EAAG,QAAO,MAAM,GAAG,OAAO,YAAY;AACtE,QAAI,wBAAwB,KAAK,CAAC,EAAG,QAAO,MAAM,GAAG,OAAO,YAAY;AACxE,QAAI,0CAA0C,KAAK,CAAC;AAClD,aAAO,MAAM,GAAG,OAAO,qBAAqB;AAC9C,QAAI,4CAA4C,KAAK,CAAC;AACpD,aAAO,MAAM,GAAG,OAAO,qBAAqB;AAC9C,WAAO,MAAM,GAAG,GAAG;AAAA,EACrB;AACA,SAAO,MAAM,GAAG,GAAG;AACrB;;;AC9DA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;A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","names":["html","range","src","timeout","appConfig","res","timeout"]}
@@ -1 +1 @@
1
- {"version":3,"file":"uploadFile.d.ts","sourceRoot":"","sources":["../../src/web/network/uploadFile.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,UAAU,CAAC;AAE7C;;GAEG;AACH,MAAM,MAAM,gBAAgB,GAAG;IAC7B,aAAa;IACb,GAAG,EAAE,MAAM,CAAC;IAEZ,eAAe;IACf,IAAI,EAAE,IAAI,CAAC;IAEX,mDAAmD;IACnD,IAAI,CAAC,EAAE,MAAM,CAAC;IAEd,UAAU;IACV,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAAC,CAAC;IAEzC,oBAAoB;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAAC,CAAC;IAE3C,2BAA2B;IAC3B,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB,CAAC;AAEF,MAAM,MAAM,sBAAsB,GAAG,CAAC,GAAG,EAAE,mBAAmB,KAAK,IAAI,CAAC;AAExE,MAAM,MAAM,mBAAmB,GAAG;IAChC,qBAAqB;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa;IACb,MAAM,EAAE,MAAM,CAAC;IACf,WAAW;IACX,KAAK,EAAE,MAAM,CAAC;CACf,CAAC;AAEF,MAAM,MAAM,UAAU,GAAG;IACvB,WAAW;IACX,gBAAgB,EAAE,CAAC,QAAQ,EAAE,sBAAsB,KAAK,IAAI,CAAC;IAC7D,WAAW;IACX,KAAK,EAAE,MAAM,IAAI,CAAC;CACnB,CAAC;AAEF,MAAM,MAAM,YAAY,GAAG;IACzB,eAAe;IACf,WAAW,CAAC,EAAE,CAAC,IAAI,EAAE,UAAU,KAAK,IAAI,CAAC;CAC1C,CAAC;AAEF,MAAM,MAAM,UAAU,GAAG;IACvB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;CAChB,CAAC;AAyEF;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,UAAU,CAAC,MAAM,EAAE,gBAAgB,EAAE,MAAM,CAAC,EAAE,YAAY,GAAG,YAAY,mBAExF"}
1
+ {"version":3,"file":"uploadFile.d.ts","sourceRoot":"","sources":["../../src/web/network/uploadFile.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,UAAU,CAAC;AAE7C;;GAEG;AACH,MAAM,MAAM,gBAAgB,GAAG;IAC7B,aAAa;IACb,GAAG,EAAE,MAAM,CAAC;IAEZ,eAAe;IACf,IAAI,EAAE,IAAI,CAAC;IAEX,mDAAmD;IACnD,IAAI,CAAC,EAAE,MAAM,CAAC;IAEd,UAAU;IACV,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAAC,CAAC;IAEzC,oBAAoB;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAAC,CAAC;IAE3C,2BAA2B;IAC3B,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB,CAAC;AAEF,MAAM,MAAM,sBAAsB,GAAG,CAAC,GAAG,EAAE,mBAAmB,KAAK,IAAI,CAAC;AAExE,MAAM,MAAM,mBAAmB,GAAG;IAChC,qBAAqB;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa;IACb,MAAM,EAAE,MAAM,CAAC;IACf,WAAW;IACX,KAAK,EAAE,MAAM,CAAC;CACf,CAAC;AAEF,MAAM,MAAM,UAAU,GAAG;IACvB,WAAW;IACX,gBAAgB,EAAE,CAAC,QAAQ,EAAE,sBAAsB,KAAK,IAAI,CAAC;IAC7D,WAAW;IACX,KAAK,EAAE,MAAM,IAAI,CAAC;CACnB,CAAC;AAEF,MAAM,MAAM,YAAY,GAAG;IACzB,eAAe;IACf,WAAW,CAAC,EAAE,CAAC,IAAI,EAAE,UAAU,KAAK,IAAI,CAAC;CAC1C,CAAC;AAEF,MAAM,MAAM,UAAU,GAAG;IACvB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;CAChB,CAAC;AA2EF;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,UAAU,CAAC,MAAM,EAAE,gBAAgB,EAAE,MAAM,CAAC,EAAE,YAAY,GAAG,YAAY,mBAExF"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@base-web-kits/base-tools-web",
3
- "version": "1.2.4",
3
+ "version": "1.2.5",
4
4
  "sideEffects": false,
5
5
  "description": "Independent Web utilities package built from src/web.",
6
6
  "keywords": [
@@ -111,13 +111,15 @@ function upload(option: UploadFileOption, config?: UploadConfig) {
111
111
 
112
112
  // 组装 FormData
113
113
  const data = new FormData();
114
- data.append(name, file);
115
114
  if (formData) {
116
115
  Object.entries(formData).forEach(([k, v]) => {
117
116
  if (v !== undefined && v !== null) data.append(k, String(v));
118
117
  });
119
118
  }
120
119
 
120
+ // OSS直传的file字段必须写在最后
121
+ data.append(name, file);
122
+
121
123
  // 发送请求
122
124
  xhr.send(data);
123
125
  });