UNPKG

@mt-kit/utils

Version:
2,427 lines (1,796 loc) 57.7 kB
# @mt-kit/utils ## 下载 ```bash npm i @mt-kit/utils ``` ## 使用 [See](https://github.com/Not-have/micro-tools/tree/main/packages-utils/stories) ## API ### animationFrameThrottle - 使用 `requestAnimationFrame` 创建函数节流版本,限制函数在特定时间内的调用次数 - 适用于高频事件如滚动、窗口大小改变等场景 | 参数名 | 说明 | 是否必传 | 类型 | |--------|------|----------|------| | fn | 需要节流的回调函数 | 是 | `Function` | **返回值:** `Function` - 节流后的函数 **使用场景:** - 滚动事件处理 - 窗口大小改变监听 - 动画更新函数 - 鼠标移动事件 ```ts import { animationFrameThrottle } from "@mt-kit/utils"; // 滚动事件处理 const handleScroll = () => { console.log('页面滚动中...'); }; window.addEventListener('scroll', animationFrameThrottle(handleScroll)); // 窗口大小改变 const handleResize = () => { console.log('窗口大小改变'); }; window.addEventListener('resize', animationFrameThrottle(handleResize)); // 动画效果 const updateAnimation = () => { // 更新动画逻辑 }; const throttledAnimation = animationFrameThrottle(updateAnimation); ``` ### imageBase64ToBlob - 将 Base64 格式的图片转换为 Blob 对象 - 支持各种图片格式(PNG、JPEG、GIF、WebP 等) | 参数名 | 说明 | 是否必传 | 类型 | |--------|------|----------|------| | base64 | Base64 格式的图片数据 | 是 | `string` | **返回值:** `Blob` - 转换后的 Blob 对象 **使用场景:** - 图片上传前的格式转换 - Canvas 导出图片 - 图片压缩处理 - 文件下载 ```ts import { imageBase64ToBlob } from "@mt-kit/utils"; // 转换 Base64 图片为 Blob const base64Data = '...'; const blob = imageBase64ToBlob(base64Data); // 上传图片 const formData = new FormData(); formData.append('image', blob, 'image.png'); // 创建下载链接 const url = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = url; a.download = 'image.png'; a.click(); ``` ### imageUrlToBase64 - 将图片 URL 转换为 Base64 格式 - 支持跨域图片(需要服务器支持 CORS) | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | url | 图片的 URL 地址 | 是 | `string` | - | | mimeType | 生成的 Base64 字符串的 MIME 类型 | 否 | `string` | `"image/png"` | **返回值:** `Promise<string>` - Base64 格式的图片数据 **使用场景:** - 图片缓存处理 - 离线图片存储 - 图片压缩上传 - 图片预览功能 ```ts import { imageUrlToBase64 } from "@mt-kit/utils"; // 转换图片 URL 为 Base64 imageUrlToBase64('https://example.com/image.png') .then(base64 => { console.log(base64); // ... // 在页面中显示 const img = document.createElement('img'); img.src = base64; document.body.appendChild(img); }) .catch(error => { console.error('图片转换失败:', error); }); // 指定 MIME 类型 imageUrlToBase64('https://example.com/image.jpg', 'image/jpeg') .then(base64 => { console.log(base64); // ... }); ``` ### downloadByUrl - 根据文件地址进行下载 - 支持各种文件格式(PDF、图片、文档等) | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | url | 文件的 URL 地址 | 是 | `string` | - | | target | 链接的打开方式 | 否 | `string` | `"_blank"` | | fileName | 下载的文件名 | 否 | `string` | 自动从 URL 提取 | **返回值:** `void` **使用场景:** - 文件下载功能 - 文档导出 - 图片保存 - 资源下载 ```ts import { downloadByUrl } from "@mt-kit/utils"; // 下载 PDF 文档 downloadByUrl({ url: 'https://example.com/document.pdf', target: '_blank', fileName: 'my-document.pdf' }); // 下载图片 downloadByUrl({ url: 'https://example.com/image.jpg', fileName: 'beautiful-image.jpg' }); // 下载 Excel 文件 downloadByUrl({ url: 'https://example.com/data.xlsx', fileName: 'report-2024.xlsx' }); ``` ### downloadDataFile - 根据文件数据进行下载 - 支持各种数据类型(文本、二进制、ArrayBuffer 等) | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | data | 文件数据(BlobPart 类型) | 是 | `BlobPart` | - | | filename | 保存的文件名 | 是 | `string` | - | | mime | 文件的 MIME 类型 | 否 | `string` | 自动检测 | **返回值:** `void` **使用场景:** - 动态生成文件下载 - 文本文件导出 - 二进制数据处理 - 数据备份功能 ```ts import { downloadDataFile } from "@mt-kit/utils"; // 下载文本文件 const textData = "Hello, World!"; downloadDataFile(textData, 'hello.txt', 'text/plain'); // 下载 JSON 数据 const jsonData = JSON.stringify({ name: 'John', age: 30 }, null, 2); downloadDataFile(jsonData, 'data.json', 'application/json'); // 下载二进制数据 const binaryData = new Uint8Array([72, 101, 108, 108, 111]); // "Hello" downloadDataFile(binaryData, 'hello.bin', 'application/octet-stream'); // 下载 CSV 数据 const csvData = "Name,Age,City\nJohn,30,New York\nJane,25,London"; downloadDataFile(csvData, 'users.csv', 'text/csv'); ``` ### downloadBase64File - 根据 Base64 数据下载文件 - 支持各种文件格式的 Base64 数据 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | base64 | Base64 格式的文件数据 | 是 | `string` | - | | filename | 保存的文件名 | 是 | `string` | - | | mime | 文件的 MIME 类型 | 否 | `string` | 自动检测 | **返回值:** `void` **使用场景:** - Base64 图片下载 - 编码文件导出 - 数据恢复功能 - 文件格式转换 ```ts import { downloadBase64File } from "@mt-kit/utils"; // 下载 Base64 文本文件 const base64Text = 'data:text/plain;base64,SGVsbG8gV29ybGQh'; // "Hello World!" downloadBase64File(base64Text, 'hello.txt'); // 下载 Base64 图片 const base64Image = '...'; downloadBase64File(base64Image, 'image.png', 'image/png'); // 下载 Base64 PDF const base64Pdf = 'data:application/pdf;base64,JVBERi0xLjQKJcfsj6IK...'; downloadBase64File(base64Pdf, 'document.pdf', 'application/pdf'); ``` ### downloadUrlFile - 根据在线文件的 URL 进行下载 - 支持各种文件格式(图片、文档、视频等) | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | url | 文件的 URL 地址 | 是 | `string` | - | | filename | 保存的文件名 | 是 | `string` | - | | mime | 文件的 MIME 类型 | 否 | `string` | 自动检测 | **返回值:** `Promise<void>` **使用场景:** - 在线图片保存 - 文档下载 - 媒体文件下载 - 资源备份 ```ts import { downloadUrlFile } from "@mt-kit/utils"; // 下载图片 downloadUrlFile( 'https://example.com/image.png', 'my-image.png', 'image/png' ); // 下载 PDF 文档 downloadUrlFile( 'https://example.com/document.pdf', 'report.pdf', 'application/pdf' ); // 下载视频文件 downloadUrlFile( 'https://example.com/video.mp4', 'tutorial.mp4', 'video/mp4' ); ``` ### copyText - 复制文本到剪贴板 - 支持现代浏览器的 Clipboard API | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | text | 要复制的文本内容 | 是 | `string` | - | | fn | 复制成功的回调函数 | 否 | `Function` | - | **返回值:** `Promise<void>` **使用场景:** - 一键复制功能 - 分享链接复制 - 代码片段复制 - 用户反馈功能 ```ts import { copyText } from "@mt-kit/utils"; // 基本使用 copyText('Hello World!').then(() => { console.log('文本已复制到剪贴板'); }); // 带回调函数 copyText('https://example.com', () => { alert('链接已复制!'); }); // 错误处理 copyText('Some text') .then(() => { console.log('复制成功'); }) .catch(error => { console.error('复制失败:', error); }); ``` ### queryStringToObject - 将查询字符串转换为对象 - 支持 URL 参数解析和对象化处理 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | queryString | 查询字符串(包含 ? 或不包含都可以) | 是 | `string` | - | **返回值:** `Record<string, string>` - 解析后的参数对象 **使用场景:** - URL 参数解析 - 表单数据处理 - 路由参数提取 - API 参数构建 ```ts import { queryStringToObject } from "@mt-kit/utils"; // 基本使用 const query = queryStringToObject('?name=John&age=30'); console.log(query); // { name: 'John', age: '30' } // 不包含 ? 的字符串 const query2 = queryStringToObject('name=John&age=30'); console.log(query2); // { name: 'John', age: '30' } // 空查询字符串 const query3 = queryStringToObject(''); console.log(query3); // {} // 复杂参数 const query4 = queryStringToObject('?search=hello%20world&page=1&sort=desc'); console.log(query4); // { search: 'hello world', page: '1', sort: 'desc' } ``` ### openWindow - 打开新窗口 - 支持自定义窗口属性和打开方式 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | url | 要打开的链接地址 | 是 | `string` | - | | options | 窗口打开选项 | 否 | `object` | `{ target: '_blank' }` | **返回值:** `Window | null` - 新打开的窗口对象 **Options 配置项:** - `target`: 窗口打开方式(`_blank`, `_self`, `_parent`, `_top`) - `features`: 窗口特性(宽度、高度、位置等) **使用场景:** - 外部链接打开 - 弹窗功能 - 新页面跳转 - 第三方集成 ```ts import { openWindow } from "@mt-kit/utils"; // 基本使用 openWindow('https://example.com'); // 自定义窗口属性 openWindow('https://example.com', { target: '_blank', features: 'width=800,height=600,scrollbars=yes,resizable=yes' }); // 居中打开窗口 const windowFeatures = 'width=600,height=400,left=' + (screen.width / 2 - 300) + ',top=' + (screen.height / 2 - 200); openWindow('https://example.com', { target: '_blank', features: windowFeatures }); ``` ### cookieHelper - Cookie 操作助手 - 提供完整的 Cookie 增删改查功能 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | name | Cookie 名称 | 是 | `string` | - | | value | Cookie 值 | 是 | `string` | - | | options | Cookie 配置选项 | 否 | `object` | `{}` | **方法列表:** - `get(name)`: 获取指定名称的 Cookie - `set(name, value, options)`: 设置 Cookie - `remove(name)`: 删除指定名称的 Cookie **Options 配置项:** - `expires`: 过期时间(天数) - `path`: Cookie 路径 - `domain`: Cookie 域名 - `secure`: 是否只在 HTTPS 下传输 - `sameSite`: SameSite 属性 **使用场景:** - 用户登录状态 - 用户偏好设置 - 购物车数据 - 主题配置 ```ts import { cookieHelper } from "@mt-kit/utils"; // 设置 Cookie cookieHelper.set('username', 'John', { expires: 7 }); // 7天过期 cookieHelper.set('theme', 'dark', { path: '/', secure: true }); // 获取 Cookie const username = cookieHelper.get('username'); const theme = cookieHelper.get('theme'); // 删除 Cookie cookieHelper.remove('username'); // 设置带路径的 Cookie cookieHelper.set('session', 'abc123', { expires: 1, path: '/admin', secure: true, sameSite: 'strict' }); ``` ### localStorageHelper - localStorage 操作助手 - 提供本地存储的增删改查功能,支持 JSON 序列化 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | name | 本地存储的键名 | 是 | `string` | - | | value | 本地存储的值 | 是 | `any` | - | **方法列表:** - `get(name)`: 获取指定名称的本地存储 - `set(name, value)`: 设置本地存储 - `remove(name)`: 删除指定名称的本地存储 - `clear()`: 清空所有数据 **使用场景:** - 用户数据持久化 - 应用配置存储 - 缓存数据管理 - 离线数据存储 ```ts import { localStorageHelper } from "@mt-kit/utils"; // 设置数据 localStorageHelper.set('user', { name: 'John', age: 30 }); localStorageHelper.set('theme', 'dark'); localStorageHelper.set('settings', { language: 'zh-CN', currency: 'CNY' }); // 获取数据 const user = localStorageHelper.get('user'); const theme = localStorageHelper.get('theme'); const settings = localStorageHelper.get('settings'); // 删除数据 localStorageHelper.remove('user'); // 清空所有数据 localStorageHelper.clear(); // 处理不存在的键 const nonExistent = localStorageHelper.get('nonExistent'); console.log(nonExistent); // null ``` ### isElement - 判断是否为 DOM 元素 - 支持各种 DOM 元素类型检测 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | element | 要检测的元素 | 是 | `any` | - | **返回值:** `boolean` - 是否为 DOM 元素 **使用场景:** - DOM 操作前的类型检查 - 事件处理中的元素验证 - 组件库开发 - 工具函数开发 ```ts import { isElement } from "@mt-kit/utils"; // 检测 DOM 元素 const div = document.createElement('div'); console.log(isElement(div)); // true const span = document.createElement('span'); console.log(isElement(span)); // true // 检测非 DOM 元素 console.log(isElement({})); // false console.log(isElement('string')); // false console.log(isElement(123)); // false console.log(isElement(null)); // false console.log(isElement(undefined)); // false // 在事件处理中使用 document.addEventListener('click', (event) => { if (isElement(event.target)) { console.log('点击了 DOM 元素:', event.target); } }); ``` ### isFunction - 判断是否为函数 - 支持各种函数类型检测(普通函数、箭头函数、构造函数等) | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | fn | 要检测的值 | 是 | `any` | - | **返回值:** `boolean` - 是否为函数 **使用场景:** - 回调函数验证 - 高阶函数开发 - 事件处理器检查 - 工具函数开发 ```ts import { isFunction } from "@mt-kit/utils"; // 检测各种函数类型 console.log(isFunction(() => {})); // true - 箭头函数 console.log(isFunction(function() {})); // true - 普通函数 console.log(isFunction(async () => {})); // true - 异步函数 console.log(isFunction(function*() {})); // true - 生成器函数 console.log(isFunction(Array)); // true - 构造函数 console.log(isFunction(Math.max)); // true - 内置函数 // 检测非函数 console.log(isFunction({})); // false console.log(isFunction('string')); // false console.log(isFunction(123)); // false console.log(isFunction(null)); // false console.log(isFunction(undefined)); // false // 在回调中使用 function processCallback(callback) { if (isFunction(callback)) { callback('处理完成'); } else { console.error('回调必须是一个函数'); } } ``` ### isEqual - 深度比较两个值是否相等 - 支持对象、数组、基本类型的深度比较 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | a | 要比较的第一个值 | 是 | `any` | - | | b | 要比较的第二个值 | 是 | `any` | - | **返回值:** `boolean` - 两个值是否相等 **使用场景:** - 对象状态比较 - 表单数据验证 - 缓存数据检查 - 组件更新判断 ```ts import { isEqual } from "@mt-kit/utils"; // 基本类型比较 console.log(isEqual(1, 1)); // true console.log(isEqual('hello', 'hello')); // true console.log(isEqual(true, true)); // true console.log(isEqual(null, null)); // true console.log(isEqual(undefined, undefined)); // true // 对象深度比较 const objA = { a: 1, b: { c: 2 } }; const objB = { a: 1, b: { c: 2 } }; const objC = { a: 1, b: { c: 3 } }; console.log(isEqual(objA, objB)); // true console.log(isEqual(objA, objC)); // false // 数组比较 const arr1 = [1, 2, { a: 3 }]; const arr2 = [1, 2, { a: 3 }]; const arr3 = [1, 2, { a: 4 }]; console.log(isEqual(arr1, arr2)); // true console.log(isEqual(arr1, arr3)); // false // 复杂嵌套比较 const complex1 = { users: [ { id: 1, name: 'John', settings: { theme: 'dark' } } ], config: { api: 'https://api.example.com' } }; const complex2 = { users: [ { id: 1, name: 'John', settings: { theme: 'dark' } } ], config: { api: 'https://api.example.com' } }; console.log(isEqual(complex1, complex2)); // true ``` ### isNull - 判断是否为 null - 严格检查 null 值,不包括 undefined | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | value | 要检测的值 | 是 | `any` | - | **返回值:** `boolean` - 是否为 null **使用场景:** - 数据验证 - 空值检查 - 条件判断 - 类型守卫 ```ts import { isNull } from "@mt-kit/utils"; // 检测 null console.log(isNull(null)); // true // 检测非 null 值 console.log(isNull(undefined)); // false console.log(isNull(0)); // false console.log(isNull('')); // false console.log(isNull(false)); // false console.log(isNull({})); // false console.log(isNull([])); // false // 在条件判断中使用 function processValue(value) { if (isNull(value)) { console.log('值为 null'); } else { console.log('值不为 null:', value); } } processValue(null); // 值为 null processValue('hello'); // 值不为 null: hello ``` ### isObject - 判断是否为对象类型 - 包括普通对象、数组、函数等,但不包括 null | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | value | 要检测的值 | 是 | `any` | - | **返回值:** `boolean` - 是否为对象类型 **使用场景:** - 类型检查 - 对象操作前的验证 - 数据结构判断 - 工具函数开发 ```ts import { isObject } from "@mt-kit/utils"; // 检测对象类型 console.log(isObject({})); // true - 普通对象 console.log(isObject([])); // true - 数组 console.log(isObject(() => {})); // true - 函数 console.log(isObject(new Date())); // true - 日期对象 console.log(isObject(/regex/)); // true - 正则表达式 // 检测非对象类型 console.log(isObject(null)); // false console.log(isObject(undefined)); // false console.log(isObject('string')); // false console.log(isObject(123)); // false console.log(isObject(true)); // false // 在对象操作中使用 function processData(data) { if (isObject(data)) { console.log('处理对象数据:', data); // 安全地进行对象操作 Object.keys(data).forEach(key => { console.log(`${key}: ${data[key]}`); }); } else { console.log('数据不是对象类型'); } } ``` ### isUndefined - 判断是否为 undefined - 严格检查 undefined 值,不包括 null | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | value | 要检测的值 | 是 | `any` | - | **返回值:** `boolean` - 是否为 undefined **使用场景:** - 参数验证 - 可选属性检查 - 条件判断 - 类型守卫 ```ts import { isUndefined } from "@mt-kit/utils"; // 检测 undefined console.log(isUndefined(undefined)); // true // 检测非 undefined 值 console.log(isUndefined(null)); // false console.log(isUndefined(0)); // false console.log(isUndefined('')); // false console.log(isUndefined(false)); // false console.log(isUndefined({})); // false console.log(isUndefined([])); // false // 在函数参数中使用 function greet(name, greeting) { if (isUndefined(name)) { console.log('请提供姓名'); return; } const message = isUndefined(greeting) ? 'Hello' : greeting; console.log(`${message}, ${name}!`); } greet(); // 请提供姓名 greet('John'); // Hello, John! greet('John', 'Hi'); // Hi, John! // 在对象属性检查中使用 const user = { name: 'John' }; if (isUndefined(user.age)) { console.log('年龄未设置'); } ``` ### clone - 浅拷贝对象 - 只复制对象的第一层属性,嵌套对象保持引用 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | value | 要拷贝的值 | 是 | `any` | - | **返回值:** `any` - 拷贝后的对象 **使用场景:** - 对象状态备份 - 避免直接修改原对象 - 快速对象复制 - 性能要求较高的场景 ```ts import { clone } from "@mt-kit/utils"; // 基本对象拷贝 const obj = { a: 1, b: { c: 2 } }; const cloned = clone(obj); console.log(cloned); // { a: 1, b: { c: 2 } } console.log(cloned === obj); // false - 不是同一个对象 console.log(cloned.b === obj.b); // true - 嵌套对象引用相同 // 数组拷贝 const arr = [1, 2, { x: 3 }]; const clonedArr = clone(arr); console.log(clonedArr); // [1, 2, { x: 3 }] console.log(clonedArr === arr); // false console.log(clonedArr[2] === arr[2]); // true - 嵌套对象引用相同 // 修改拷贝后的对象 cloned.a = 10; console.log(obj.a); // 1 - 原对象不受影响 console.log(cloned.a); // 10 - 拷贝对象已修改 // 修改嵌套对象 cloned.b.c = 20; console.log(obj.b.c); // 20 - 原对象的嵌套对象也被修改 console.log(cloned.b.c); // 20 - 拷贝对象的嵌套对象也被修改 ``` ### cloneDeep - 深拷贝对象 - 递归复制所有嵌套对象,完全独立于原对象 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | value | 要拷贝的值 | 是 | `any` | - | **返回值:** `any` - 深拷贝后的对象 **使用场景:** - 完全独立的对象复制 - 状态管理 - 数据备份 - 避免引用污染 ```ts import { cloneDeep } from "@mt-kit/utils"; // 基本对象深拷贝 const obj = { a: 1, b: { c: 2 } }; const cloned = cloneDeep(obj); console.log(cloned); // { a: 1, b: { c: 2 } } console.log(cloned === obj); // false - 不是同一个对象 console.log(cloned.b === obj.b); // false - 嵌套对象也是独立的 // 复杂嵌套对象 const complex = { users: [ { id: 1, profile: { name: 'John', settings: { theme: 'dark' } } } ], config: { api: 'https://api.example.com' } }; const clonedComplex = cloneDeep(complex); console.log(clonedComplex.users[0] === complex.users[0]); // false console.log(clonedComplex.users[0].profile === complex.users[0].profile); // false // 修改深拷贝后的对象 cloned.b.c = 20; console.log(obj.b.c); // 2 - 原对象不受影响 console.log(cloned.b.c); // 20 - 拷贝对象已修改 // 数组深拷贝 const arr = [1, 2, { x: 3, y: { z: 4 } }]; const clonedArr = cloneDeep(arr); clonedArr[2].y.z = 40; console.log(arr[2].y.z); // 4 - 原数组不受影响 console.log(clonedArr[2].y.z); // 40 - 拷贝数组已修改 ``` ### debounce - 创建一个防抖函数,延迟调用函数直到上一次调用后的一段时间已经过去 - 适用于频繁触发的事件,如搜索输入、窗口调整等 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | fn | 要防抖的函数 | 是 | `Function` | - | | wait | 延迟时间(毫秒) | 是 | `number` | - | | options | 配置选项 | 否 | `object` | `{}` | **返回值:** `Function` - 防抖后的函数 **Options 配置项:** - `leading`: 是否在延迟开始前调用 - `trailing`: 是否在延迟结束后调用 - `maxWait`: 最大等待时间 **使用场景:** - 搜索输入防抖 - 窗口大小调整 - 按钮点击防抖 - API 请求优化 ```ts import { debounce } from "@mt-kit/utils"; // 搜索输入防抖 const debouncedSearch = debounce((query) => { console.log('搜索:', query); fetchSearchResults(query); }, 300); // 在输入框中使用 const input = document.getElementById('search'); input.addEventListener('input', (e) => { debouncedSearch(e.target.value); }); // 窗口大小调整防抖 const debouncedResize = debounce(() => { console.log('窗口大小已调整'); updateLayout(); }, 250); window.addEventListener('resize', debouncedResize); // 按钮点击防抖 const debouncedClick = debounce(() => { console.log('按钮被点击'); submitForm(); }, 1000); document.getElementById('submitBtn').addEventListener('click', debouncedClick); // 带配置选项的防抖 const debouncedWithOptions = debounce((data) => { console.log('处理数据:', data); }, 500, { leading: true, // 立即执行第一次 trailing: true, // 延迟结束后执行 maxWait: 2000 // 最大等待 2 秒 }); ``` ### throttle - 创建一个节流函数,限制函数在一段时间内只能调用一次 - 适用于需要定期执行但不想过于频繁的场景 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | fn | 要节流的函数 | 是 | `Function` | - | | wait | 节流时间(毫秒) | 是 | `number` | - | | options | 配置选项 | 否 | `object` | `{}` | **返回值:** `Function` - 节流后的函数 **Options 配置项:** - `leading`: 是否在节流开始前调用 - `trailing`: 是否在节流结束后调用 **使用场景:** - 滚动事件节流 - 鼠标移动事件 - 动画更新 - 性能优化 ```ts import { throttle } from "@mt-kit/utils"; // 滚动事件节流 const throttledScroll = throttle(() => { console.log('滚动位置更新'); updateScrollPosition(); }, 100); // 在滚动事件中使用 window.addEventListener('scroll', throttledScroll); // 鼠标移动节流 const throttledMouseMove = throttle((event) => { console.log('鼠标位置:', event.clientX, event.clientY); updateMousePosition(event); }, 16); // 约 60fps document.addEventListener('mousemove', throttledMouseMove); // 窗口大小调整节流 const throttledResize = throttle(() => { console.log('窗口大小调整'); updateLayout(); }, 200); window.addEventListener('resize', throttledResize); // 带配置选项的节流 const throttledWithOptions = throttle((data) => { console.log('处理数据:', data); }, 500, { leading: true, // 立即执行第一次 trailing: true // 节流结束后执行 }); // 动画更新节流 const throttledAnimation = throttle(() => { updateAnimationFrame(); }, 16); // 60fps function startAnimation() { const animate = () => { throttledAnimation(); requestAnimationFrame(animate); }; animate(); } ``` ### omitBy - 创建一个从对象中排除满足条件的属性的新对象 - 根据条件函数过滤对象属性 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | obj | 源对象 | 是 | `object` | - | | condition | 用于判断是否排除属性的条件函数 | 是 | `Function` | - | **返回值:** `object` - 过滤后的新对象 **使用场景:** - 对象属性过滤 - 数据清理 - 条件性属性排除 - 对象转换 ```ts import { omitBy } from "@mt-kit/utils"; // 基本使用 - 排除值大于 2 的属性 const sampleObject = { a: 1, b: 2, c: 3, d: 4, }; const result = omitBy(sampleObject, (value) => value > 2); console.log(result); // { a: 1, b: 2 } // 排除空值 const dataWithEmpty = { name: 'John', age: 30, email: '', phone: null, address: undefined }; const cleanedData = omitBy(dataWithEmpty, (value) => value === '' || value === null || value === undefined ); console.log(cleanedData); // { name: 'John', age: 30 } // 排除特定类型的属性 const mixedData = { id: 1, name: 'John', isActive: true, createdAt: new Date(), tags: ['admin', 'user'] }; const stringOnly = omitBy(mixedData, (value) => typeof value !== 'string'); console.log(stringOnly); // { name: 'John' } // 排除特定键名的属性 const userData = { id: 1, name: 'John', password: 'secret', email: 'john@example.com', token: 'abc123' }; const publicData = omitBy(userData, (value, key) => key === 'password' || key === 'token' ); console.log(publicData); // { id: 1, name: 'John', email: 'john@example.com' } ``` ### objectValueToString - 将对象的值转换为字符串 - 保持对象结构不变,只转换值类型 | 参数名 | 说明 | 是否必传 | 类型 | 默认值 | |--------|------|----------|------|--------| | obj | 源对象 | 是 | `object` | - | **返回值:** `object` - 值转换为字符串的新对象 **使用场景:** - 数据序列化准备 - 表单数据处理 - API 参数转换 - 数据格式统一 ```ts import { objectValueToString } from "@mt-kit/utils"; // 基本使用 const obj = { id: 123, active: true, score: 98.5 }; const stringified = objectValueToString(obj); console.log(stringified); // 输出: { id: '123', active: 'true', score: '98.5' } // 复杂对象 const complexObj = { user: { id: 1, name: 'John', age: 30, isActive: true }, settings: { theme: 'dark', notifications: false, language: 'zh-CN' }, scores: [85, 92, 78] }; const stringifiedComplex = objectValueToString(complexObj); console.log(stringifiedComplex); // 输出: { // user: { id: '1', name: 'John', age: '30', isActive: 'true' }, // settings: { theme: 'dark', notifications: 'false', language: 'zh-CN' }, // scores: ['85', '92', '78'] // } // 处理特殊值 const specialObj = { nullValue: null, undefinedValue: undefined, emptyString: '', zero: 0, falseValue: false }; const stringifiedSpecial = objectValueToString(specialObj); console.log(stringifiedSpecial); // 输出: { // nullValue: 'null', // undefinedValue: 'undefined', // emptyString: '', // zero: '0', // falseValue: 'false' // } ``` ### IframeMessage - 创建一个 iframe 通信工具,与内嵌的页面进行双向通讯 - 支持父子页面之间的消息传递和事件监听 **主要方法:** - `createIframe(url)`: 创建 iframe 元素 - `postMessage(message)`: 发送消息 - `onMessage(callback)`: 监听消息 - `removeMessageListener(callback)`: 移除消息监听 **使用场景:** - 跨域页面通信 - 微前端架构 - 嵌入式应用 - 父子页面数据同步 **调用界面(父页面):** ```tsx import { JSX, useCallback, useMemo } from "react"; import { IframeMessage } from "@mt-kit/utils"; export default function DemoIframeMessage(): JSX.Element { const url = "http://localhost:5173/"; const iframe = useMemo(() => { const communicator = new IframeMessage(); communicator.createIframe(url); return communicator; }, [url]); const handleClick = useCallback(() => { iframe.postMessage({ type: "message", data: { name: "hello", age: 18 } }); }, [iframe]); return ( <div> <p>iframe 通信</p> <button onClick={handleClick}>传值</button> </div> ); } ``` **接收通信(子页面):** ```vue <script setup lang="ts"> import { onMounted, onUnmounted } from "vue"; import { Message, IframeMessage } from "@mt-kit/utils"; const iframe = new IframeMessage(); const handleMessage = (e: Message): void => { console.log('收到消息:', e); // 处理不同类型的消息 switch (e.type) { case 'message': console.log('数据:', e.data); break; case 'command': console.log('命令:', e.data); break; } }; onMounted(() => { iframe.onMessage(handleMessage); }); onUnmounted(() => { iframe.removeMessageListener(handleMessage); }); </script> <template> <div> <h1>子页面</h1> <p>等待接收消息...</p> </div> </template> ``` **双向通信示例:** ```ts // 父页面 const iframe = new IframeMessage(); iframe.createIframe('http://localhost:5173/'); // 监听子页面消息 iframe.onMessage((message) => { console.log('父页面收到:', message); }); // 发送消息到子页面 iframe.postMessage({ type: 'userData', data: { userId: 123, name: 'John' } }); // 子页面 const iframe = new IframeMessage(); // 监听父页面消息 iframe.onMessage((message) => { console.log('子页面收到:', message); // 回复消息给父页面 iframe.postMessage({ type: 'response', data: { status: 'received' } }); }); ``` ### flattenAndSort 将对象扁平化为键值对并按键排序,确保输出结果的确定性。 **参数:** | 参数 | 类型 | 必填 | 默认值 | 说明 | |------|------|------|--------|------| | `obj` | `Record<string, unknown>` | ✅ | - | 要扁平化的对象 | | `prefix` | `string` | ❌ | `""` | 键前缀,用于嵌套对象的键名 | **返回值:** | 类型 | 说明 | |------|------| | `Record<string, string>` | 扁平化的对象,所有值都转换为字符串 | **特性:** - **确定性输出**: 通过按键排序确保相同输入产生相同输出 - **递归处理**: 自动处理嵌套对象 - **类型安全**: 所有值统一转换为字符串 - **空值处理**: `undefined` 和 `null` 转换为空字符串 **使用示例:** ```typescript import { flattenAndSort } from '@mt-kit/utils'; // 基础用法 const obj = { name: 'John', age: 30, address: { city: 'New York', country: 'USA' } }; const result = flattenAndSort(obj); // 结果: { // "address.city": "New York", // "address.country": "USA", // "age": "30", // "name": "John" // } // 带前缀 const resultWithPrefix = flattenAndSort(obj, 'user'); // 结果: { // "user.address.city": "New York", // "user.address.country": "USA", // "user.age": "30", // "user.name": "John" // } ``` **应用场景:** - 对象序列化前的预处理 - 生成确定性哈希值 - 对象比较和去重 - 配置文件的扁平化处理 --- ### arrayBufferToBase64 将 ArrayBuffer 转换为 Base64 字符串,提供安全高效的转换。 **参数:** | 参数 | 类型 | 必填 | 说明 | |------|------|------|------| | `buffer` | `ArrayBuffer` | ✅ | 要转换的 ArrayBuffer | **返回值:** | 类型 | 说明 | |------|------| | `string` | Base64 编码的字符串,失败时返回空字符串 | **特性:** - **容错处理**: 输入无效时返回空字符串,不抛出异常 - **高效转换**: 使用 `String.fromCharCode` 进行安全转换 - **类型验证**: 自动验证输入是否为有效的 ArrayBuffer - **错误日志**: 失败时输出警告信息便于调试 **使用示例:** ```typescript import { arrayBufferToBase64 } from '@mt-kit/utils'; // 基础用法 const buffer = new ArrayBuffer(8); const view = new Uint8Array(buffer); view[0] = 72; // 'H' view[1] = 101; // 'e' view[2] = 108; // 'l' view[3] = 108; // 'l' view[4] = 111; // 'o' const base64 = arrayBufferToBase64(buffer); console.log(base64); // "SGVsbG8=" // 错误处理 const invalid = arrayBufferToBase64(null as any); console.log(invalid); // "" // 从文件读取 const file = new File(['Hello'], 'test.txt'); const reader = new FileReader(); reader.onload = (e) => { const base64 = arrayBufferToBase64(e.target?.result as ArrayBuffer); console.log(base64); }; reader.readAsArrayBuffer(file); ``` **应用场景:** - 文件上传前的编码 - 图片数据处理 - 二进制数据传输 - 加密数据存储 --- ### sha256Base64 将字符串转换为 SHA-256 哈希的 Base64 编码,提供安全的哈希功能。 **参数:** | 参数 | 类型 | 必填 | 说明 | |------|------|------|------| | `str` | `string` | ✅ | 要哈希的字符串 | **返回值:** | 类型 | 说明 | |------|------| | `Promise<string>` | Base64 编码的 SHA-256 哈希值 | **特性:** - **异步处理**: 使用 Web Crypto API 进行异步哈希计算 - **安全算法**: 使用 SHA-256 算法确保安全性 - **Base64 输出**: 直接输出 Base64 编码,便于存储和传输 - **错误处理**: 哈希失败时抛出错误,便于上层处理 **使用示例:** ```typescript import { sha256Base64 } from '@mt-kit/utils'; // 基础用法 const hash = await sha256Base64('Hello World'); console.log(hash); // "pZGm1Av0IEBKARCz7JZcLAuUx7QgTz+ekpJZeBc7uT0=" // 密码哈希 const password = 'myPassword123'; const passwordHash = await sha256Base64(password); console.log(passwordHash); // 文件内容哈希 const fileContent = 'file content here'; const fileHash = await sha256Base64(fileContent); console.log(fileHash); // 错误处理 try { const hash = await sha256Base64('test'); console.log(hash); } catch (error) { console.error('哈希生成失败:', error); } ``` **应用场景:** - 密码存储和验证 - 数据完整性校验 - 唯一标识符生成 - 缓存键生成 --- ### getAvailableFonts 检测系统中可用的字体列表,支持批量检测和容错处理。 **参数:** | 参数 | 类型 | 必填 | 默认值 | 说明 | |------|------|------|--------|------| | `fontList` | `string[]` | ❌ | `COMMON_FONTS` | 要检测的字体列表 | **返回值:** | 类型 | 说明 | |------|------| | `string[]` | 可用的字体列表 | **特性:** - **Canvas 检测**: 使用 Canvas 2D 上下文进行字体检测 - **批量处理**: 支持同时检测多个字体 - **容错机制**: 单个字体检测失败不影响其他字体 - **精度控制**: 使用宽度差异阈值避免浮点数精度问题 - **详细日志**: 提供检测过程和结果的详细日志 **使用示例:** ```typescript import { getAvailableFonts } from '@mt-kit/utils'; // 检测常用字体 const availableFonts = getAvailableFonts(); console.log(availableFonts); // ['Arial', 'Helvetica', 'Times New Roman', ...] // 检测自定义字体列表 const customFonts = ['CustomFont', 'AnotherFont', 'Arial']; const available = getAvailableFonts(customFonts); console.log(available); // ['Arial'] (假设其他字体不可用) // 空列表处理 const empty = getAvailableFonts([]); console.log(empty); // [] ``` **检测原理:** 1. 使用回退字体测量基准宽度 2. 使用目标字体测量文本宽度 3. 比较宽度差异,超过阈值则认为字体可用 4. 阈值设置为 0.1px,避免浮点数精度问题 **应用场景:** - 字体回退策略 - 设备指纹生成 - 字体兼容性检测 - 动态字体加载 --- ### measureTextWidthDom 使用 DOM 元素测量文本宽度,提供精确的文本尺寸测量。 **参数:** | 参数 | 类型 | 必填 | 默认值 | 说明 | |------|------|------|--------|------| | `text` | `string` | ✅ | - | 要测量的文本 | | `font` | `string` | ✅ | - | 字体样式 | | `size` | `string` | ❌ | `FONT_CONFIG.size` | 字体大小 | **返回值:** | 类型 | 说明 | |------|------| | `number` | 文本宽度(像素),测量失败返回 -1 | **特性:** - **DOM 测量**: 使用真实的 DOM 元素进行测量 - **样式隔离**: 创建隐藏元素避免影响页面布局 - **强制重排**: 确保样式生效后再测量 - **自动清理**: 测量完成后自动移除临时元素 - **错误处理**: 测量失败时返回 -1 **使用示例:** ```typescript import { measureTextWidthDom } from '@mt-kit/utils'; // 基础用法 const width = measureTextWidthDom('Hello World', 'Arial'); console.log(width); // 82.5 // 自定义字体大小 const width2 = measureTextWidthDom('Test', 'Times New Roman', '16px'); console.log(width2); // 24.8 // 错误处理 const invalid = measureTextWidthDom('', 'InvalidFont'); console.log(invalid); // -1 ``` **应用场景:** - 文本布局计算 - 字体检测 - 动态宽度调整 - 文本溢出处理 --- ### measureTextWidthCanvas 使用 Canvas 2D 上下文测量文本宽度,提供高性能的文本尺寸测量。 **参数:** | 参数 | 类型 | 必填 | 默认值 | 说明 | |------|------|------|--------|------| | `text` | `string` | ✅ | - | 要测量的文本 | | `font` | `string` | ✅ | - | 字体样式 | | `size` | `string` | ❌ | `FONT_CONFIG.size` | 字体大小 | **返回值:** | 类型 | 说明 | |------|------| | `number` | 文本宽度(像素),测量失败返回 -1 | **特性:** - **Canvas 测量**: 使用 Canvas 2D 上下文进行测量 - **高性能**: 不操作 DOM,性能优于 DOM 方法 - **高精度**: 设置固定 Canvas 尺寸确保测量精度 - **错误处理**: 测量失败时返回 -1 - **类型安全**: 验证测量结果的数值有效性 **使用示例:** ```typescript import { measureTextWidthCanvas } from '@mt-kit/utils'; // 基础用法 const width = measureTextWidthCanvas('Hello World', 'Arial'); console.log(width); // 82.5 // 自定义字体大小 const width2 = measureTextWidthCanvas('Test', 'Times New Roman', '16px'); console.log(width2); // 24.8 // 批量测量 const texts = ['Short', 'Medium Length Text', 'Very Long Text Here']; const widths = texts.map(text => measureTextWidthCanvas(text, 'Arial')); console.log(widths); // [32.5, 95.2, 158.7] ``` **Canvas vs DOM 对比:** | 特性 | Canvas | DOM | |------|--------|-----| | 性能 | ✅ 高 | ❌ 低 | | 精度 | ✅ 高 | ✅ 高 | | 兼容性 | ✅ 好 | ✅ 好 | | 内存使用 | ✅ 低 | ❌ 高 | | 布局影响 | ✅ 无 | ❌ 有 | **应用场景:** - 字体检测(推荐) - 批量文本测量 - 性能敏感场景 - 字体指纹生成 ### createContainer - 创建容器 - 支持自定义元素插入到容器中 - 支持追加到容器中 - 支持追加到 body - 支持追加到指定元素 **参数:** | 参数 | 说明 | 是否必传 | 默认值 | |--------|------|----------|--------| | positionOptions | 位置选项 | 否 | 所有功能启用 | | append | 是否追加到容器中 | 否 | false | | customElement | 自定义元素 | 否 | 无 | **返回值:** | 类型 | 说明 | |--------|------| | `{ remove: () => void; element: HTMLDivElement; }` | 容器 | **positionOptions 参数:** | 参数 | 说明 | 是否必传 | 默认值 | |--------|------|----------|--------| | position | 位置 | 否 | 无 | **append 参数:** | 参数 | 说明 | 是否必传 | 默认值 | |--------|------|----------|--------| | append | 是否追加到容器中 | 否 | false | **customElement 参数:** | 参数 | 说明 | 是否必传 | 默认值 | |--------|------|----------|--------| | customElement | 自定义元素 | 否 | 无 | **使用示例:** ```ts import { createContainer } from "@mt-kit/utils"; const container = createContainer(); console.log(container); ``` ### uuid - 生成唯一的 ID - 使用 Date.now() 和 Math.random() 生成唯一的 ID - 使用 slice(2, 15) 生成唯一的 ID - 使用 _ 连接 - 使用 toString(36) 生成唯一的 ID ```ts import { uuid } from "@mt-kit/utils"; const id = uuid(); console.log(id); ``` ## 设备信息 ### deviceAll - 获取所有设备信息(包括同步和异步信息) | 参数名 | 说明 | 是否必传 | 默认值 | |--------|------|----------|--------| | options | 配置选项,可禁用某些功能 | 否 | 所有功能启用 | 返回类型:`Promise<IDeviceAll>` **Options 配置项:** - `operatingSystem`: 是否获取操作系统信息 - `browser`: 是否获取浏览器信息 - `language`: 是否获取语言信息 - `onLine`: 是否获取在线状态 - `screen`: 是否获取屏幕信息 - `cpuCores`: 是否获取CPU核心数 - `hardwareConcurrency`: 是否获取硬件并发数 - `memory`: 是否获取内存信息 - `features`: 是否获取设备特性 - `sensor`: 是否获取传感器信息 - `i18n`: 是否获取国际化信息 - `location`: 是否获取地理位置(异步) - `publicIp`: 是否获取公网IP(异步) ```ts import { deviceAll } from "@mt-kit/utils"; // 获取完整设备信息 const allDeviceInfo = await deviceAll(); console.log(allDeviceInfo); // 禁用某些功能 const basicInfo = await deviceAll({ location: false, // 不获取地理位置 publicIp: false, // 不获取公网IP memory: false // 不获取内存信息 }); // 只获取基础信息 const minimalInfo = await deviceAll({ operatingSystem: true, browser: true, language: true, onLine: true, screen: false, cpuCores: false, hardwareConcurrency: false, memory: false, features: false, sensor: false, i18n: false, location: false, publicIp: false }); ``` ### deviceOperatingSystem - 获取操作系统信息 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | ```ts import { deviceOperatingSystem } from "@mt-kit/utils"; const os = deviceOperatingSystem(); console.log(os); // "Windows", "macOS", "Linux", "Android", "iOS", "unknown" ``` ### deviceBrowser - 获取浏览器信息 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | 返回类型:`IBrowser` ```ts import { deviceBrowser } from "@mt-kit/utils"; const browser = deviceBrowser(); console.log(browser); // { name: "Chrome", version: "120.0.0.0" } ``` ### deviceLanguage - 获取浏览器语言 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | ```ts import { deviceLanguage } from "@mt-kit/utils"; const language = deviceLanguage(); console.log(language); // "zh-CN", "en-US", etc. ``` ### deviceOnLine - 获取在线状态 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | ```ts import { deviceOnLine } from "@mt-kit/utils"; const isOnline = deviceOnLine(); console.log(isOnline); // true or false ``` ### deviceScreen - 获取屏幕信息 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | 返回类型:`IScreen` ```ts import { deviceScreen } from "@mt-kit/utils"; const screen = deviceScreen(); console.log(screen); // { // width: 1920, // height: 1080, // availWidth: 1920, // availHeight: 1040, // colorDepth: 24, // pixelDepth: 24, // devicePixelRatio: 1 // } ``` ### deviceLocation - 获取地理位置信息(需要用户授权) | 参数名 | 说明 | 是否必传 | 返回值 | |--------|------|----------|--------| | 无 | 无参数 | - | `{ latitude: number, longitude: number }` | 返回类型:`Promise<ILocation>` **返回值说明:** - `latitude`: 纬度,范围 -9090 - `longitude`: 经度,范围 -180180 **失败情况:** - **定位服务不可用**:返回 `{ latitude: 0, longitude: 0 }` - **用户拒绝授权**:返回 `{ latitude: -1, longitude: -1 }` - **超时**:返回 `{ latitude: -1, longitude: -1 }` - **其他错误**:返回 `{ latitude: -1, longitude: -1 }` ```ts import { deviceLocation } from "@mt-kit/utils"; deviceLocation().then(location => { if (location.latitude === -1 && location.longitude === -1) { console.log("定位失败或用户拒绝授权"); } else { console.log(`位置: ${location.latitude}, ${location.longitude}`); } }); ``` ### devicePublicIp **IP查询服务列表:** 1. `https://api.ipify.org?format=json` - 官方服务,稳定可靠 2. `https://ipapi.co/json/` - 官方 API,功能丰富 3. `https://ipinfo.io/json` - 官方服务,数据准确 4. `https://api.ip.sb/geoip` - 开源项目,稳定 5. `https://httpbin.org/ip` - 官方测试服务 **服务特点:** - 按顺序尝试,成功即停止 - 总超时时间:15秒 - 单个服务超时:5秒 - 全部失败时返回 `127.0.0.1` | 参数名 | 说明 | 是否必传 | 返回值 | |--------|------|----------|--------| | 无 | 无参数 | - | `string` | 返回类型:`Promise<string>` **返回值说明:** - 成功时:返回公网IP地址字符串,如 `"203.208.60.1"` - 失败时:返回本地IP地址 `"127.0.0.1"` 作为容错处理 - 可能包含位置信息:`"203.208.60.1 (北京市)"` ```ts import { devicePublicIp } from "@mt-kit/utils"; devicePublicIp().then(ip => { console.log(ip); // "127.0.0.1" (获取失败时返回本地IP) }); ``` ### deviceCpuCores - 获取CPU核心数 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | ```ts import { deviceCpuCores } from "@mt-kit/utils"; const cores = deviceCpuCores(); console.log(cores); // 8 ``` ### deviceMemory - 获取内存信息 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | 返回类型:`IMemory` ```ts import { deviceMemory } from "@mt-kit/utils"; const memory = deviceMemory(); console.log(memory); // { // usedJSHeapSize: 43.18, // totalJSHeapSize: 46.07, // jsHeapSizeLimit: 4095.75, // deviceMemory: 8 // } ``` ### deviceHardwareConcurrency - 获取硬件并发数 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | ```ts import { deviceHardwareConcurrency } from "@mt-kit/utils"; const concurrency = deviceHardwareConcurrency(); console.log(concurrency); // 8 ``` ### deviceFeatures - 获取设备特性支持 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | 返回类型:`IFeatures` ```ts import { deviceFeatures } from "@mt-kit/utils"; const features = deviceFeatures(); console.log(features); // { // touchSupport: true, // maxTouchPoints: 5, // vibrate: true, // webglSupport: true, // webrtcSupport: true // } ``` ### deviceSensor - 获取传感器支持 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | 返回类型:`ISensor` ```ts import { deviceSensor } from "@mt-kit/utils"; const sensor = deviceSensor(); console.log(sensor); // { // accelerometer: true, // gyroscope: true, // magnetometer: false, // ambientLight: false, // barometer: false // } ``` ### deviceI18n - 获取国际化信息 | 参数名 | 说明 | 是否必传 | |--------|------|----------| | 无 | 无参数 | - | 返回类型:`II18n` ```ts import { deviceI18n } from "@mt-kit/utils"; const i18n = deviceI18n(); console.log(i18n); // { // timeZone: "Asia/Shanghai", // daylightSaving: false, // numberFormat: "1,234.56", // currencyFormat: "¥1,234.56", // dateFormat: "2024/1/1", // timeFormat: "14:30:00" // } ``` ## 获取设备指纹 ### fingerprint 生成综合设备指纹,整合多种指纹信息创建唯一标识符。 **参数:** | 参数 | 类型 | 必填 | 默认值 | 说明 | |------|------|------|--------|------| | `strength` | `boolean` | ❌ | `true` | 是否生成强指纹(包含屏幕信息) | **返回值:** | 类型 | 说明 | |------|------| | `Promise<string>` | SHA-256 哈希的 Base64 编码指纹字符串 | **指纹组成:** - **设备信息**: 操作系统、浏览器、屏幕等基础信息 - **Canvas 指纹**: 基于 Canvas 渲染的像素数据 - **WebGL 指纹**: 显卡硬件信息 - **音频指纹**: 音频处理特征 - **字体指纹**: 可用字体列表 - **国际化信息**: 时区、语言、数字格式等 **特性:** - **综合指纹**: 整合多种指纹信息提高唯一性 - **确定性**: 相同环境产生相同指纹 - **容错处理**: 单个指纹失败不影响整体结果 - **强度控制**: 可选择是否包含屏幕信息 **使用示例:** ```typescript import { fingerprint } from '@mt-kit/utils'; // 生成强指纹(包含屏幕信息) const strongFingerprint = await fingerprint(true); console.log(strongFingerprint); // "a1b2c3d4e5f6..." // 生成弱指纹(不包含屏幕信息) const weakFingerprint = await fingerprint(false); console.log(weakFingerprint); // "x1y2z3a4b5c6..." // 默认强指纹 const defaultFingerprint = await fingerprint(); console.log(defaultFingerprint); // "m1n2o3p4q5r6..." ``` **指纹稳定性:** | 因素 | 影响程度 | 说明 | |------|----------|------| | 屏幕分辨率 | 🔴 高 | 改变分辨率会改变指纹 | | 浏览器版本 | 🟡 中 | 更新可能影响指纹 | | 操作系统 | 🔴 高 | 不同系统指纹不同 | | 硬件配置 | 🔴 高 | 显卡、音频设备影响指纹 | | 字体安装 | 🟡 中 | 字体变化影响指纹 | **应用场景:** - 用户身份识别 - 设备唯一标识 - 反爬虫检测 - 用户行为分析 - 安全验证 **注意事项:** - 指纹可能因环境变化而改变 - 建议结合其他身份验证方式 - 遵守隐私法规,明确告知用户 - 定期更新指纹算法以提高安全性 <details> <summary>生成指纹的原理 API 列表</summary> #### fingerprintFonts 生成字体指纹,通过检测可用字体列表创建设备指纹。 **参数:** | 参数 | 类型 | 必填 | 默认值 | 说明 | |------|------|------|--------|------| | `fonts` | `string[]` | ❌ | `COMMON_FONTS` | 要检测的字体列表 | **返回值:** | 类型 | 说明 | |------|------| | `string` | 可用字体列表的逗号分隔字符串 | **特性:** - **字体检测**: 基于 Canvas 2D 上下文检测字体可用性 - **批量处理**: 支持检测多个字体 - **容错机制**: 单个字体检测失败不影响整体结果 - **确定性输出**: 相同字体列表产生相同结果 **使用示例:** ```typescript import { fingerprintFonts } from '@mt-kit/utils'; // 检测默认字体列表 const fontFingerprint = fingerprintFonts(); console.log(fontFingerprint); // "Arial,Helvetica,Times New Roman,Georgia" // 检测自定义字体列表 const customFonts = ['CustomFont', 'AnotherFont', 'Arial']; const customFingerprint = fingerprintFonts(customFonts); console.log(customFingerprint); // "Arial" (假设其他字体不可用) ``` **应用场景:** - 设备指纹生成 - 字体兼容性检测 - 浏览器环境识别 - 反爬虫检测 --- #### fingerprintCanvas 生成 Canvas 指纹,通过渲染特定内容获取设备特定的渲染特征。 **参数:** 无参数 **返回值:** | 类型 | 说明 | |------|------| | `string` | Canvas 渲染数据的像素值字符串,失败时返回空字符串 | **特性:** - **渲染指纹**: 基于 Canvas 2D 渲染结果生成指纹 - **固定内容**: 使用固定的文本和样式确保一致性 - **像素采样**: 提取特定位置的像素值作为指纹 - **容错处理**: 渲染失败时返回空字符串 **渲染内容:** - 文本: 使用固定字体渲染特定文本 - 图形: 绘制矩形和半透明文本 - 采样: 提取前 100 个像素的 RGB 值 **使用示例:** ```typescript import { fingerprintCanvas } from '@mt-kit/utils'; // 生成 Canvas 指纹 const canvasFingerprint = fingerprintCanvas(); console.log(canvasFingerprint); // "255,0,0,0,0,255,102,204,0,..." // 错误处理 if (!canvasFingerprint) { console.warn('Canvas 指纹生成失败'); } ``` **技术原理:** 1. 创建 200x50 像素的 Canvas 2. 设置固定的字体和样式 3. 渲染文本和图形内容 4. 提取图像数据的 RGB 像素值 5. 按固定间隔采样并拼接成字符串 **应用场景:** - 设备指纹生成 - 浏览器环境识别 - 反爬虫检测 - 用户行为分析 --- #### fingerprintWebgl 生成 WebGL 指纹,通过获取显卡信息创建硬件指纹。 **参数:** 无参数 **返回值:** | 类型 | 说明 | |------|------| | `string` | WebGL 厂商和渲染器信息,失败时返回空字符串 | **特性:** - **硬件指纹**: 基于显卡硬件信息生成指纹 - **调试扩展**: 使用 `WEBGL_debug_renderer_info` 扩展获取详细信息 - **厂商信息**: 包含显卡厂商和渲染器型号 - **容错处理**: 不支持时返回空字符串 **获取信息:** - **厂商**: 显卡厂商名称(如 NVIDIA、AMD、Intel) - **渲染器**: 显卡型号和驱动信息 **使用示例:** ```typescript import { fingerprintWebgl } from '@mt-kit/utils'; // 生成 WebGL 指纹 const webglFingerprint = fingerprintWebgl(); console.log(webglFingerprint); // "NVIDIA Corporation~NVIDIA GeForce RTX 3080/PCIe/SSE2" // 检查支持情况 if (!webglFingerprint) { console.warn('WebGL 调试扩展不支持'); } ``` **兼容性:** - **桌面浏览器**: Chrome、Firefox、Safari 支持较好 - **移动浏览器**: 支持有限,很多移动浏览器禁用此扩展 - **隐私模式**: 可能被禁用或返回受限信息 **应用场景:** - 设备指纹生成 - 硬件环境识别 - 反爬虫检测 - 用户设备分析 --- #### fingerprintAudio 生成音频指纹,通过音频处理获取设备特定的音频特征。 **参数:** 无参数 **返回值:** | 类型 | 说明 | |------|------| | `Promise<string>` | 音频处理数据的样本值字符串,失败时返回空字符串 | **特性:** - **音频处理**: 基于 Web Audio API 生成音频指纹 - **离线渲染**: 使用 OfflineAudioContext 避免播放音频 - **固定参数**: 使用固定的音频参数确保一致性 - **样本采样**: 提取特定位置的音频样本值 **音频配置:** - **采样率**: 44.1kHz - **声道数**: 单声道 - **振荡器**: 三角波,1000Hz - **压缩器**: 固定参数配置 **使用示例:** ```typescript import { fingerprintAudio } from '@mt-kit/utils'; // 生成音频指纹 const audioFingerprint = await fingerprintAudio(); console.log(audioFingerprint); // "0.1,0.2,0.3,0.4,0.5,..." // 错误处理 try { const fingerprint = await fingerprintAudio(); if (!fingerprint) { console.warn('音频指纹生成失败'); } } catch (error) { console.error('音频处理错误:', error); } ``` **技术原理:** 1. 创建 OfflineAudioContext 2. 生成三角波振荡器 3. 通过动态压缩器处理音频 4. 离线渲染音频数据 5. 提取特定间隔的样本值 **应用场景:** - 设备指纹生成 - 音频环境识别 - 反爬虫检测 - 用户设备分析 --- </details>