rx-util

rxliuli 在浏览器上使用的 js 工具集

目前为 2020-01-22 提交的版本。查看 最新版本

此脚本不应直接安装,它是一个供其他脚本使用的外部库。如果您需要使用该库,请在脚本元属性加入:// @require https://update.cn-greasyfork.org/scripts/382120/767256/rx-util.js

  1. (function (global, factory) {
  2. typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
  3. typeof define === 'function' && define.amd ? define(['exports'], factory) :
  4. (global = global || self, factory(global.rx = {}));
  5. }(this, function (exports) { 'use strict';
  6.  
  7. /**
  8. * 在浏览器上下载二进制资源
  9. * @param blob 要下载的二进制资源
  10. * @param filename 文件名
  11. */
  12. function download(blob, filename = 'unknown') {
  13. // 创建隐藏的可下载链接
  14. const eleLink = document.createElement('a');
  15. eleLink.download = filename;
  16. eleLink.style.display = 'none';
  17. // 为 link 赋值
  18. eleLink.href = URL.createObjectURL(blob);
  19. // 触发点击
  20. document.body.appendChild(eleLink);
  21. eleLink.click();
  22. // 然后移除
  23. document.body.removeChild(eleLink);
  24. }
  25.  
  26. /*! *****************************************************************************
  27. Copyright (c) Microsoft Corporation. All rights reserved.
  28. Licensed under the Apache License, Version 2.0 (the "License"); you may not use
  29. this file except in compliance with the License. You may obtain a copy of the
  30. License at http://www.apache.org/licenses/LICENSE-2.0
  31.  
  32. THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
  33. KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED
  34. WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE,
  35. MERCHANTABLITY OR NON-INFRINGEMENT.
  36.  
  37. See the Apache Version 2.0 License for specific language governing permissions
  38. and limitations under the License.
  39. ***************************************************************************** */
  40.  
  41. function __awaiter(thisArg, _arguments, P, generator) {
  42. return new (P || (P = Promise))(function (resolve, reject) {
  43. function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
  44. function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
  45. function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
  46. step((generator = generator.apply(thisArg, _arguments || [])).next());
  47. });
  48. }
  49.  
  50. /**
  51. * 将字符串转化为 Blob 类型
  52. * @param str 字符串
  53. * @returns Blob 数据
  54. */
  55. function strToBlob(str) {
  56. return new Blob([str], {
  57. type: 'text/plain',
  58. });
  59. }
  60.  
  61. /**
  62. * 在浏览器上下载文本内容
  63. * @param str 字符串内容
  64. * @param filename 下载文件名,没有则默认为链接中的文件名
  65. */
  66. function downloadString(str, filename = 'unknown.txt') {
  67. return __awaiter(this, void 0, void 0, function* () {
  68. download(strToBlob(str), filename);
  69. });
  70. }
  71.  
  72. /**
  73. * 根据 url 下载二进制资源
  74. * @param url 下载请求信息
  75. * @param filename 下载文件名,没有则默认为链接中的文件名
  76. */
  77. function downloadUrl(url, filename = url.substr(url.lastIndexOf('/'))) {
  78. return __awaiter(this, void 0, void 0, function* () {
  79. // 创建隐藏的可下载链接
  80. const eleLink = document.createElement('a');
  81. eleLink.download = filename;
  82. eleLink.style.display = 'none';
  83. // 为 link 赋值
  84. eleLink.href = url;
  85. // 触发点击
  86. document.body.appendChild(eleLink);
  87. eleLink.click();
  88. // 然后移除
  89. document.body.removeChild(eleLink);
  90. });
  91. }
  92.  
  93. /**
  94. * 获取 cookie 键值映射 Map
  95. * @returns cookie 键值映射 Map
  96. */
  97. function getCookies() {
  98. return document.cookie.split(';').reduce((res, str) => {
  99. const [k, v] = str.split('=');
  100. res.set(k, v);
  101. return res;
  102. }, new Map());
  103. }
  104.  
  105. /**
  106. * 将 url 中的内容加载到元素上
  107. * 注:domSelector 必须有 src 属性用以将加载完成的资源赋值给其,加载默认是异步的
  108. * @param url url 资源
  109. * @param dom dom 元素
  110. * @param init 初始化参数, 实为 fetch() 的参数以及一些自定义的参数,默认 {}
  111. * 关于 fetch 具体可以参考 <https://developer.mozilla.org/zh-CN/docs/Web/API/Fetch_API/Using_Fetch>
  112. */
  113. function loadResource(url, dom, init = {}) {
  114. return __awaiter(this, void 0, void 0, function* () {
  115. const res = yield fetch(url, init);
  116. const blob = yield res.blob();
  117. // 生成一个本地的 url 并赋值给 src 属性
  118. dom.src = window.URL.createObjectURL(blob);
  119. });
  120. }
  121.  
  122. /**
  123. * 协议与默认端口映射表
  124. */
  125. const protocolPortMap = new Map()
  126. .set('http', 80)
  127. .set('https', 443)
  128. .set('ssh', 22)
  129. .set('ftp', 21);
  130. /**
  131. * 解析 url 字符串
  132. * @param url url 字符串,不能为空
  133. * @returns url 对象
  134. * @deprecated 请使用原生 API URL 类,可以通过 new URL(url) 将 URL 字符串转换为 URL 对象,并获取指定的信息
  135. */
  136. function parseUrl(url) {
  137. if (!url) {
  138. throw new Error('Url cannot be empty');
  139. }
  140. const regexp = new RegExp('^((\\w+)://([\\w\\.]*)(:(\\d+))?)(.*)');
  141. const temps = regexp.exec(url);
  142. if (temps === null) {
  143. return null;
  144. }
  145. const website = temps[1];
  146. const protocol = temps[2];
  147. const domain = temps[3];
  148. const portStr = temps[5];
  149. const href = temps[6];
  150. // 截取域名之后的内容
  151. const temp = url.substr(website.length);
  152. const markIndex = temp.indexOf('?');
  153. // 如果没有携带参数则直接返回
  154. if (markIndex === -1) {
  155. const accessPath = temp;
  156. return {
  157. url,
  158. website,
  159. protocol,
  160. domain,
  161. // tslint:disable-next-line:radix
  162. port: parseInt(portStr),
  163. href,
  164. accessPath,
  165. params: new Map(),
  166. };
  167. }
  168. let accessPath = temp.substr(0, markIndex);
  169. if (accessPath.endsWith('/')) {
  170. accessPath = accessPath.substring(0, accessPath.length - 1);
  171. }
  172. const port = portStr || protocolPortMap.get(protocol) || 0;
  173. // 解析参数列表
  174. const params = temp
  175. .substr(markIndex + 1)
  176. .split('&')
  177. .map(str => str.split('='))
  178. .filter(arr => arr[0] !== '')
  179. .reduce((params, arr) => {
  180. const k = decodeURIComponent(arr[0]);
  181. const v = decodeURIComponent(arr.length === 1 ? '' : arr[1]);
  182. // 如果已经存在了就认为是数组参数
  183. const vs = params.get(k);
  184. if (vs === undefined) {
  185. params.set(k, v);
  186. }
  187. else {
  188. if (!Array.isArray(vs)) {
  189. params.set(k, [vs]);
  190. }
  191. if (params.get(k).length !== undefined) {
  192. params.get(k).push(v);
  193. }
  194. }
  195. return params;
  196. }, new Map());
  197. return {
  198. url,
  199. website,
  200. protocol,
  201. domain,
  202. port,
  203. href,
  204. accessPath,
  205. params,
  206. };
  207. }
  208.  
  209. /**
  210. * 读取文件类型
  211. */
  212. var ReadType;
  213. (function (ReadType) {
  214. /**
  215. * 以 data url 读取
  216. */
  217. ReadType["DataURL"] = "readAsDataURL";
  218. /**
  219. * 以文本读取
  220. */
  221. ReadType["Text"] = "readAsText";
  222. /**
  223. * 以二进制文件读取
  224. */
  225. ReadType["BinaryString"] = "readAsBinaryString";
  226. /**
  227. * 以 ArrayBuffer 读取
  228. */
  229. ReadType["ArrayBuffer"] = "readAsArrayBuffer";
  230. })(ReadType || (ReadType = {}));
  231. /**
  232. * 读取本地浏览器选择的文件
  233. * @param file 选择的文件
  234. * @param option 可选项参数
  235. * @param option.type 读取的类型,默认按照二进制 url 读取
  236. * @param option.encoding 读取的编码格式,默认为 UTF-8
  237. * @returns 返回了读取到的内容(异步)
  238. */
  239. function readLocal(file, { type = readLocal.DataURL, encoding = 'UTF-8', } = {}) {
  240. return new Promise((resolve, reject) => {
  241. if (!file) {
  242. reject(new Error('file not exists'));
  243. }
  244. const fr = new FileReader();
  245. fr.onload = () => {
  246. resolve(fr.result);
  247. };
  248. fr.onerror = error => {
  249. reject(error);
  250. };
  251. new Map()
  252. .set(ReadType.DataURL, () => fr.readAsDataURL(file))
  253. .set(ReadType.Text, () => fr.readAsText(file, encoding))
  254. .set(ReadType.BinaryString, () => fr.readAsBinaryString(file))
  255. .set(ReadType.ArrayBuffer, () => fr.readAsArrayBuffer(file))
  256. .get(type)();
  257. });
  258. }
  259. /**
  260. * 以 data url 读取
  261. * @deprecated 已废弃,请使用枚举类 ReadType
  262. */
  263. readLocal.DataURL = ReadType.DataURL;
  264. /**
  265. * 以文本读取
  266. * @deprecated 已废弃,请使用枚举类 ReadType
  267. */
  268. readLocal.Text = ReadType.Text;
  269. /**
  270. * 以二进制文件读取
  271. * @deprecated 已废弃,请使用枚举类 ReadType
  272. */
  273. readLocal.BinaryString = ReadType.BinaryString;
  274. /**
  275. * 以 ArrayBuffer 读取
  276. * @deprecated 已废弃,请使用枚举类 ReadType
  277. */
  278. readLocal.ArrayBuffer = ReadType.ArrayBuffer;
  279.  
  280. /**
  281. * 为 js 中的 Date 对象原型添加 format 格式化方法
  282. * @param date 要进行格式化的日期
  283. * @param fmt 日期的格式
  284. * @returns 格式化得到的结果
  285. */
  286. function dateFormat(date, fmt) {
  287. const timeFormatDefaults = {
  288. 'Y+|y+': date.getFullYear(),
  289. 'M+': date.getMonth() + 1,
  290. 'D+|d+': date.getDate(),
  291. 'h+': date.getHours(),
  292. 'm+': date.getMinutes(),
  293. 's+': date.getSeconds(),
  294. 'q+': Math.floor((date.getMonth() + 3) / 3),
  295. 'S+': date.getMilliseconds(),
  296. };
  297. for (const k in timeFormatDefaults) {
  298. if (!new RegExp('(' + k + ')').test(fmt)) {
  299. continue;
  300. }
  301. if (k === 'Y+|y+') {
  302. fmt = fmt.replace(RegExp.$1, ('' + timeFormatDefaults[k]).substr(4 - RegExp.$1.length));
  303. }
  304. else if (k === 'S+') {
  305. let lens = RegExp.$1.length;
  306. lens = lens === 1 ? 3 : lens;
  307. fmt = fmt.replace(RegExp.$1, ('00' + timeFormatDefaults[k]).substr(('' + timeFormatDefaults[k]).length - 1, lens));
  308. }
  309. else {
  310. const v = Reflect.get(timeFormatDefaults, k);
  311. fmt = fmt.replace(RegExp.$1, RegExp.$1.length === 1 ? v : ('00' + v).substr(('' + v).length));
  312. }
  313. }
  314. return fmt;
  315. }
  316.  
  317. /**
  318. * 默认的日期格式
  319. * 不加 Z 为本地日期时间,避免出现时区的问题
  320. */
  321. const dateFormatter = 'yyyy-MM-ddThh:mm:ss.SSS';
  322. /**
  323. * 将参数 key 与 value 进行 url 编码
  324. * @param k 参数的名字
  325. * @param v 参数的值
  326. * @returns 编码后的字符串
  327. */
  328. const encode = (k, v) => encodeURIComponent(k) + '=' + encodeURIComponent(v);
  329. /**
  330. * 拼接参数字符串
  331. * @param params 参数对象
  332. * @returns 拼接后的字符串
  333. */
  334. function spliceParams(params = {}) {
  335. return Array.from(Object.entries(params)).reduce((res, [k, v]) => {
  336. if (v === undefined || v === null) {
  337. return res;
  338. }
  339. else if (v instanceof Date) {
  340. res += encode(k, dateFormat(v, dateFormatter));
  341. }
  342. else if (v instanceof Array) {
  343. res += v
  344. .map(item => encode(k, item instanceof Date ? dateFormat(item, dateFormatter) : item))
  345. .join('&');
  346. }
  347. else {
  348. res += encode(k, v);
  349. }
  350. return (res += '&');
  351. }, '');
  352. }
  353.  
  354. /**
  355. * 等待指定的时间/等待指定表达式成立
  356. * 如果未指定等待条件则立刻执行
  357. * 注: 此实现在 nodejs 10- 会存在宏任务与微任务的问题,切记 async-await 本质上还是 Promise 的语法糖,实际上并非真正的同步函数!!!即便在浏览器,也不要依赖于这种特性。
  358. * @param param 等待时间/等待条件
  359. * @returns Promise 对象
  360. */
  361. function wait(param) {
  362. return new Promise(resolve => {
  363. if (typeof param === 'number') {
  364. setTimeout(resolve, param);
  365. }
  366. else if (typeof param === 'function') {
  367. const timer = setInterval(() => {
  368. if (param()) {
  369. clearInterval(timer);
  370. resolve();
  371. }
  372. }, 100);
  373. }
  374. else {
  375. resolve();
  376. }
  377. });
  378. }
  379.  
  380. /**
  381. * 为 fetch 请求添加超时选项
  382. * 注:超时选项并非真正意义上的超时即取消请求,请求依旧正常执行完成,但会提前返回 reject 结果
  383. * @param fetchPromise fetch 请求的 Promise
  384. * @param timeout 超时时间
  385. * @returns 如果超时就提前返回 reject, 否则正常返回 fetch 结果
  386. */
  387. function fetchTimeout(fetchPromise, timeout) {
  388. return Promise.race([
  389. fetchPromise,
  390. wait(timeout).then(() => {
  391. throw new Error('timeout');
  392. }),
  393. ]);
  394. }
  395.  
  396. /**
  397. * 将字符串转为字符流
  398. * @param str 字符串
  399. * @returns 字符流对象
  400. */
  401. function strToArrayBuffer(str) {
  402. const buf = new ArrayBuffer(str.length);
  403. const view = new Uint8Array(buf);
  404. for (let i = 0; i < str.length; ++i) {
  405. view[i] = str.charCodeAt(i) & 0xff;
  406. }
  407. return buf;
  408. }
  409.  
  410. /**
  411. * 限制并发请求数量的 fetch 封装
  412. * @class FetchLimiting
  413. * @example
  414. * const fetchLimiting = new FetchLimiting()
  415. * fetchLimiting._fetch('/')
  416. * .then(res => res.json())
  417. * .then(json => console.log(json))
  418. * @deprecated 已废弃,请使用 {@link asyncLimiting} 函数
  419. */
  420. class FetchLimiting {
  421. /**
  422. * 构造函数
  423. * @param option 可选配置项
  424. * @param option.timeout 超时毫秒数
  425. * @param option.limit 最大并发数限制
  426. */
  427. constructor({ timeout = 10000, limit = 10, } = {}) {
  428. /**
  429. * @field timeout 超时毫秒数
  430. */
  431. this.timeout = timeout;
  432. /**
  433. * @field limit 最大并发数限制
  434. */
  435. this.limit = limit;
  436. /**
  437. * @field execCount 当前正在执行异步的数量
  438. */
  439. this.execCount = 0;
  440. /**
  441. * @field waitArr 等待的队列
  442. * @type {Array.<IArguments>}
  443. */
  444. this.waitArr = [];
  445. }
  446. /**
  447. * 执行一个请求
  448. * 如果到达最大并发限制时就进行等待
  449. * @param url 请求 url 信息
  450. * @param init 请求的其他可选项,默认为 undefined
  451. * @returns 如果超时就提前返回 reject, 否则正常返回 fetch 结果
  452. */
  453. fetch(input, init) {
  454. return __awaiter(this, void 0, void 0, function* () {
  455. const _innerFetch = () => __awaiter(this, void 0, void 0, function* () {
  456. this.execCount++;
  457. const args = this.waitArr.shift();
  458. try {
  459. // 这里的 args 实际上就是 arguments 对象,即上面的 url 和 init
  460. return yield fetchTimeout(fetch(args[0], args[1]), this.timeout);
  461. }
  462. finally {
  463. this.execCount--;
  464. }
  465. });
  466. this.waitArr.push([input, init]);
  467. yield wait(() => this.execCount < this.limit);
  468. // 尝试启动等待队列
  469. return _innerFetch();
  470. });
  471. }
  472. }
  473.  
  474. /**
  475. * 将一个 Iterator 迭代器转换为一个 Array
  476. * @param iterator Iterator 迭代器
  477. * @return Iterator 中每一项元素转换而得到的 Array
  478. * @deprecated 已废弃,请使用 ES6 原生函数 {@see Array.from} 替代
  479. */
  480. function asIterator(iterator) {
  481. const arr = [];
  482. while (true) {
  483. const next = iterator.next();
  484. if (next.done) {
  485. break;
  486. }
  487. arr.push(next.value);
  488. }
  489. return arr;
  490. }
  491.  
  492. /**
  493. * 判断一个对象是否是无效的
  494. * 无效的值仅包含 null/undefined
  495. * @param object 任何一个对象
  496. * @returns 是否无效的值
  497. */
  498. function isNullOrUndefined(object) {
  499. return object === undefined || object === null;
  500. }
  501.  
  502. /**
  503. * 返回第一个参数的函数
  504. * 注: 一般可以当作返回参数自身的函数,如果你只关注第一个参数的话
  505. * @param obj 任何对象
  506. * @typeparam T 传入参数的类型
  507. * @typeparam R 返回结果的类型,默认为 T,只是为了兼容该函数当参数被传递时可能出现需要类型不一致的问题
  508. * @returns 传入的第一个参数
  509. */
  510. function returnItself(obj) {
  511. return obj;
  512. }
  513.  
  514. /**
  515. * 兼容异步函数的返回值
  516. * @param res 返回值
  517. * @param callback 同步/异步结果的回调函数
  518. * @typeparam T 处理参数的类型,如果是 Promise 类型,则取出其泛型类型
  519. * @typeparam Param 处理参数具体的类型,如果是 Promise 类型,则指定为原类型
  520. * @typeparam R 返回值具体的类型,如果是 Promise 类型,则指定为 Promise 类型,否则为原类型
  521. * @returns 处理后的结果,如果是同步的,则返回结果是同步的,否则为异步的
  522. */
  523. function compatibleAsync(res, callback) {
  524. return (res instanceof Promise
  525. ? res.then(callback)
  526. : callback(res));
  527. }
  528.  
  529. /**
  530. * 内部使用的函数
  531. * 注: 如果谓词中包含任意一个异步(返回 Promise)函数,则整个返回结果将变成异步的,否则默认为同步操作.
  532. * @param fns 谓词数组
  533. * @param args 谓词应用的参数列表
  534. * @param condition 临界条件
  535. * @returns 返回结果
  536. */
  537. function _inner(fns, args, condition) {
  538. const fn = fns[0];
  539. const res = fn(...args);
  540. function _call(res) {
  541. if (condition(res)) {
  542. return res;
  543. }
  544. const others = fns.slice(1);
  545. if (others.length === 0) {
  546. return res;
  547. }
  548. return _inner(others, args, condition);
  549. }
  550. return compatibleAsync(res, _call);
  551. }
  552. /**
  553. * 连接谓词函数
  554. */
  555. class CombinedPredicate {
  556. /**
  557. * 使用 && 进行连接
  558. * @param fns 连接任意多个谓词
  559. * @returns 连接后的新谓词
  560. */
  561. static and(...fns) {
  562. return function (...args) {
  563. return _inner(fns, args, res => !res);
  564. };
  565. }
  566. /**
  567. * 使用 || 进行连接
  568. * @param fns 连接任意多个谓词
  569. * @returns 连接后的新谓词
  570. */
  571. static or(...fns) {
  572. return function (...args) {
  573. return _inner(fns, args, res => res);
  574. };
  575. }
  576. /**
  577. * 对谓词进行取反
  578. * @param fn 谓词
  579. * @returns 取反后的谓词
  580. */
  581. static not(fn) {
  582. return new Proxy(fn, {
  583. apply(_, _this, args) {
  584. return compatibleAsync(Reflect.apply(_, this, args), res => !res);
  585. },
  586. });
  587. }
  588. }
  589. const and = CombinedPredicate.and;
  590. const or = CombinedPredicate.or;
  591. const not = CombinedPredicate.not;
  592.  
  593. /**
  594. * 操作类型
  595. */
  596. var ActionType;
  597. (function (ActionType) {
  598. ActionType["forEach"] = "forEach";
  599. ActionType["filter"] = "filter";
  600. ActionType["map"] = "map";
  601. ActionType["flatMap"] = "flatMap";
  602. ActionType["sort"] = "sort";
  603. ActionType["reduce"] = "reduce";
  604. ActionType["reduceRight"] = "reduceRight";
  605. ActionType["findIndex"] = "findIndex";
  606. ActionType["find"] = "find";
  607. ActionType["every"] = "every";
  608. ActionType["some"] = "some";
  609. ActionType["parallel"] = "parallel";
  610. ActionType["serial"] = "serial";
  611. })(ActionType || (ActionType = {}));
  612. /**
  613. * 保存高阶函数传入的异步操作
  614. * @field 异步操作的类型
  615. * @field 异步操作
  616. */
  617. class Action {
  618. constructor(type, args) {
  619. this.type = type;
  620. this.args = args;
  621. this.type = type;
  622. this.args = args;
  623. }
  624. }
  625. Action.Type = ActionType;
  626. /**
  627. * 抽象异步数组,实现了一些公共的函数
  628. */
  629. class InnerBaseAsyncArray {
  630. /**
  631. * 构造函数
  632. * @param args 数组初始元素
  633. */
  634. constructor(args = []) {
  635. this._arr = args;
  636. }
  637. /**
  638. * 将整个数组排序
  639. * @param fn 比较函数
  640. * @returns 排序后的数组
  641. */
  642. sort(fn) {
  643. return __awaiter(this, void 0, void 0, function* () {
  644. if (fn === undefined) {
  645. return new InnerAsyncArray(this._arr.sort());
  646. }
  647. // TODO 此处为了让 type-doc 能生成文档而不得不加上类型
  648. const arr = this._arr.map((v, i) => [v, i]);
  649. function _sort(arr, fn) {
  650. return __awaiter(this, void 0, void 0, function* () {
  651. // 边界条件,如果传入数组的值
  652. if (arr.length <= 1) {
  653. return arr;
  654. }
  655. // 根据中间值对数组分治为两个数组
  656. const medianIndex = Math.floor(arr.length / 2);
  657. const medianValue = arr[medianIndex];
  658. const left = [];
  659. const right = [];
  660. for (let i = 0, len = arr.length; i < len; i++) {
  661. if (i === medianIndex) {
  662. continue;
  663. }
  664. const v = arr[i];
  665. if ((yield fn(v, medianValue)) <= 0) {
  666. left.push(v);
  667. }
  668. else {
  669. right.push(v);
  670. }
  671. }
  672. return (yield _sort(left, fn))
  673. .concat([medianValue])
  674. .concat(yield _sort(right, fn));
  675. });
  676. }
  677. return new InnerAsyncArray(yield (yield _sort(arr, ([t1], [t2]) => fn(t1, t2))).map(([_v, i]) => this._arr[i]));
  678. });
  679. }
  680. /**
  681. * 异步的 find
  682. * @param fn 异步查询函数
  683. * @returns 查询到的第一个值
  684. */
  685. find(fn) {
  686. return __awaiter(this, void 0, void 0, function* () {
  687. const i = yield this.findIndex(fn);
  688. return i === -1 ? null : this._arr[i];
  689. });
  690. }
  691. /**
  692. * 异步的 every
  693. * @param fn 异步匹配函数
  694. * @returns 是否全部匹配
  695. */
  696. every(fn) {
  697. return __awaiter(this, void 0, void 0, function* () {
  698. return (yield this.findIndex(CombinedPredicate.not(fn))) === -1;
  699. });
  700. }
  701. /**
  702. * 异步的 some
  703. * @param fn 异步匹配函数
  704. * @returns 是否有任意一个匹配
  705. */
  706. some(fn) {
  707. return __awaiter(this, void 0, void 0, function* () {
  708. return (yield this.findIndex(fn)) !== -1;
  709. });
  710. }
  711. /**
  712. * 转换为并发异步数组
  713. */
  714. parallel() {
  715. return new InnerAsyncArrayParallel(this._arr);
  716. }
  717. /**
  718. * 转换为顺序异步数组
  719. */
  720. serial() {
  721. return new InnerAsyncArray(this._arr);
  722. }
  723. /**
  724. * 获取内部数组的值,将返回一个浅复制的数组
  725. */
  726. value() {
  727. return this._arr.slice();
  728. }
  729. }
  730. /**
  731. * 串行的异步数组
  732. */
  733. class InnerAsyncArray extends InnerBaseAsyncArray {
  734. constructor(args) {
  735. super(args);
  736. }
  737. forEach(fn) {
  738. return __awaiter(this, void 0, void 0, function* () {
  739. for (let i = 0, len = this._arr.length; i < len; i++) {
  740. yield fn.call(this, this._arr[i], i, this);
  741. }
  742. });
  743. }
  744. filter(fn) {
  745. return __awaiter(this, void 0, void 0, function* () {
  746. const res = new InnerAsyncArray();
  747. for (let i = 0, len = this._arr.length; i < len; i++) {
  748. if (yield fn.call(this, this._arr[i], i, this)) {
  749. res._arr.push(this._arr[i]);
  750. }
  751. }
  752. return res;
  753. });
  754. }
  755. map(fn) {
  756. return __awaiter(this, void 0, void 0, function* () {
  757. const res = new InnerAsyncArray();
  758. for (let i = 0, len = this._arr.length; i < len; i++) {
  759. res._arr.push(yield fn.call(this, this._arr[i], i, this));
  760. }
  761. return res;
  762. });
  763. }
  764. flatMap(fn) {
  765. return __awaiter(this, void 0, void 0, function* () {
  766. const res = new InnerAsyncArray();
  767. for (let i = 0, len = this._arr.length; i < len; i++) {
  768. res._arr.push(...(yield fn.call(this, this._arr[i], i, this)));
  769. }
  770. return res;
  771. });
  772. }
  773. reduce(fn, res) {
  774. return __awaiter(this, void 0, void 0, function* () {
  775. for (let i = 0, len = this._arr.length; i < len; i++) {
  776. if (res) {
  777. res = yield fn.call(this, res, this._arr[i], i, this);
  778. }
  779. else {
  780. res = this._arr[i];
  781. }
  782. }
  783. return res;
  784. });
  785. }
  786. reduceRight(fn, res) {
  787. return __awaiter(this, void 0, void 0, function* () {
  788. for (let i = this._arr.length - 1; i >= 0; i--) {
  789. if (res) {
  790. res = yield fn.apply(this, [res, this._arr[i], i, this]);
  791. }
  792. else {
  793. res = this._arr[i];
  794. }
  795. }
  796. return res;
  797. });
  798. }
  799. findIndex(fn) {
  800. return __awaiter(this, void 0, void 0, function* () {
  801. for (let i = 0, len = this._arr.length; i < len; i++) {
  802. const res = yield fn.call(this, this._arr[i], i, this);
  803. if (res) {
  804. return i;
  805. }
  806. }
  807. return -1;
  808. });
  809. }
  810. }
  811. /**
  812. * 并发的异步数组
  813. */
  814. class InnerAsyncArrayParallel extends InnerBaseAsyncArray {
  815. constructor(args) {
  816. super(args);
  817. }
  818. forEach(fn) {
  819. return __awaiter(this, void 0, void 0, function* () {
  820. yield this._all(fn);
  821. });
  822. }
  823. filter(fn) {
  824. return __awaiter(this, void 0, void 0, function* () {
  825. const res = yield this._all(fn);
  826. const result = new InnerAsyncArrayParallel();
  827. for (let i = 0, len = res.length; i < len; i++) {
  828. if (res[i]) {
  829. result._arr.push(this._arr[i]);
  830. }
  831. }
  832. return result;
  833. });
  834. }
  835. map(fn) {
  836. return __awaiter(this, void 0, void 0, function* () {
  837. return new InnerAsyncArrayParallel(yield this._all(fn));
  838. });
  839. }
  840. flatMap(fn) {
  841. return __awaiter(this, void 0, void 0, function* () {
  842. const res = yield this._all(fn);
  843. return new InnerAsyncArrayParallel(res.flat());
  844. });
  845. }
  846. sort(fn) {
  847. throw new Error('Method not implemented.');
  848. }
  849. reduce(fn, res) {
  850. return __awaiter(this, void 0, void 0, function* () {
  851. for (let i = 0, len = this._arr.length; i < len; i++) {
  852. if (res) {
  853. res = yield fn.call(this, res, this._arr[i], i, this);
  854. }
  855. else {
  856. res = this._arr[i];
  857. }
  858. }
  859. return res;
  860. });
  861. }
  862. reduceRight(fn, res) {
  863. return __awaiter(this, void 0, void 0, function* () {
  864. for (let i = this._arr.length - 1; i >= 0; i--) {
  865. if (res) {
  866. res = yield fn.apply(this, [res, this._arr[i], i, this]);
  867. }
  868. else {
  869. res = this._arr[i];
  870. }
  871. }
  872. return res;
  873. });
  874. }
  875. findIndex(fn) {
  876. return __awaiter(this, void 0, void 0, function* () {
  877. return (yield this._all(fn)).findIndex(returnItself);
  878. });
  879. }
  880. _all(fn) {
  881. return __awaiter(this, void 0, void 0, function* () {
  882. return yield Promise.all(this._arr.map((v, i) => fn.apply(this, [v, i, this])));
  883. });
  884. }
  885. }
  886. /**
  887. * 异步数组
  888. */
  889. class AsyncArray {
  890. /**
  891. * 构造函数
  892. * @param args 任意个参数
  893. */
  894. constructor(...args) {
  895. /**
  896. * 内部数组的长度,用于让 {@link AsyncArray} 的实例能作为 {@link Array.from} 的参数
  897. */
  898. this.length = 0;
  899. this._arr = Array.from(args);
  900. /**
  901. * @field 保存异步任务
  902. * @type {Action[]}
  903. */
  904. this._tasks = [];
  905. }
  906. /**
  907. * 为内置数组赋值
  908. * 此处自动重新计算 length 属性
  909. */
  910. set _arr(arr) {
  911. this.__arr = arr;
  912. this.length = this.__arr.length;
  913. }
  914. get _arr() {
  915. return this.__arr;
  916. }
  917. /**
  918. * 提供一个函数方便根据已有的数组或类数组(Set/Map)创建 {@link AsyncArray}
  919. * @param arr 一个可迭代元素
  920. * @returns 创建一个新的异步数组包装
  921. */
  922. static from(arr) {
  923. const result = new AsyncArray();
  924. if (isNullOrUndefined(arr)) {
  925. return result;
  926. }
  927. result._arr = Array.from(arr);
  928. return result;
  929. }
  930. filter(fn) {
  931. return this._addTask(new Action(Action.Type.filter, [fn]));
  932. }
  933. map(fn) {
  934. return this._addTask(new Action(Action.Type.map, [fn]));
  935. }
  936. flatMap(fn) {
  937. return this._addTask(new Action(Action.Type.flatMap, [fn]));
  938. }
  939. sort(fn) {
  940. return this._addTask(new Action(Action.Type.sort, [fn]));
  941. }
  942. parallel() {
  943. return this._addTask(new Action(Action.Type.parallel, []));
  944. }
  945. serial() {
  946. return this._addTask(new Action(Action.Type.serial, []));
  947. }
  948. forEach(fn) {
  949. return this._addTask(new Action(Action.Type.forEach, [fn])).then();
  950. }
  951. some(fn) {
  952. return this._addTask(new Action(Action.Type.some, [fn])).then();
  953. }
  954. every(fn) {
  955. return this._addTask(new Action(Action.Type.every, [fn])).then();
  956. }
  957. find(fn) {
  958. return this._addTask(new Action(Action.Type.find, [fn])).then();
  959. }
  960. findIndex(fn) {
  961. return this._addTask(new Action(Action.Type.findIndex, [fn])).then();
  962. }
  963. reduce(fn, res) {
  964. return this._addTask(new Action(Action.Type.reduce, [fn, res])).then();
  965. }
  966. reduceRight(fn, res) {
  967. return this._addTask(new Action(Action.Type.reduceRight, [fn, res])).then();
  968. }
  969. /**
  970. * 终结整个链式操作并返回结果,可以使用 await 等待当前实例开始计算
  971. */
  972. then(onfulfilled, onrejected) {
  973. return __awaiter(this, void 0, void 0, function* () {
  974. try {
  975. let asyncArray = new InnerAsyncArray(this._arr);
  976. let result = this._arr;
  977. for (const task of this._tasks) {
  978. asyncArray = yield Reflect.apply(Reflect.get(asyncArray, task.type), asyncArray, task.args);
  979. if (asyncArray instanceof InnerBaseAsyncArray) {
  980. result = asyncArray.value();
  981. }
  982. else {
  983. if (!isNullOrUndefined(onfulfilled)) {
  984. onfulfilled(result);
  985. }
  986. return asyncArray;
  987. }
  988. }
  989. if (!isNullOrUndefined(onfulfilled)) {
  990. onfulfilled(result);
  991. }
  992. return result;
  993. }
  994. catch (err) {
  995. if (!isNullOrUndefined(onrejected)) {
  996. onrejected(err);
  997. }
  998. }
  999. });
  1000. }
  1001. /**
  1002. * @deprecated 已废弃,请直接使用 await 进行等待获取结果值即可
  1003. */
  1004. value() {
  1005. return this.then();
  1006. }
  1007. /**
  1008. * 允许使用 for-of 遍历内部的 _arr
  1009. */
  1010. *[Symbol.iterator]() {
  1011. for (const kv of this._arr) {
  1012. yield kv;
  1013. }
  1014. }
  1015. _addTask(task) {
  1016. const result = new AsyncArray(...this._arr);
  1017. result._tasks = [...this._tasks, task];
  1018. return result;
  1019. }
  1020. }
  1021.  
  1022. function asyncFlatMap(arr, fn) {
  1023. return __awaiter(this, void 0, void 0, function* () {
  1024. return new AsyncArray(...arr).flatMap(fn);
  1025. });
  1026. }
  1027.  
  1028. /**
  1029. * 校验变量的类型
  1030. */
  1031. class TypeValidator {
  1032. /**
  1033. * 判断是否为字符串
  1034. * @param val 需要判断的值
  1035. * @returns 是否为字符串
  1036. */
  1037. static isString(val) {
  1038. return typeof val === 'string';
  1039. }
  1040. /**
  1041. * 判断是否为数字
  1042. * @param val 需要判断的值
  1043. * @returns 是否为数字
  1044. */
  1045. static isNumber(val) {
  1046. return typeof val === 'number';
  1047. }
  1048. /**
  1049. * 判断是否为布尔值
  1050. * @param val 需要判断的值
  1051. * @returns 是否为布尔值
  1052. */
  1053. static isBoolean(val) {
  1054. return typeof val === 'boolean';
  1055. }
  1056. /**
  1057. * 判断是否为 undefined
  1058. * @param val 需要判断的值
  1059. * @returns 是否为 undefined
  1060. */
  1061. static isUndefined(val) {
  1062. return val === undefined;
  1063. }
  1064. /**
  1065. * 判断是否为 null
  1066. * @param val 需要判断的值
  1067. * @returns 是否为 null
  1068. */
  1069. static isNull(val) {
  1070. return val === null;
  1071. }
  1072. /**
  1073. * 判断是否为 Symbol
  1074. * @param val 需要判断的值
  1075. * @returns 是否为 Symbol
  1076. */
  1077. static isSymbol(val) {
  1078. return typeof val === 'symbol';
  1079. }
  1080. /**
  1081. * 判断是否可以作为对象的属性
  1082. * @param val 需要判断的值
  1083. * @returns 是否为对象属性
  1084. */
  1085. static isPropertyKey(val) {
  1086. return (TypeValidator.isString(val) ||
  1087. TypeValidator.isNumber(val) ||
  1088. TypeValidator.isSymbol(val));
  1089. }
  1090. /**
  1091. * 判断是否为对象
  1092. * 注: 函数(包括 ES6 箭头函数)将不被视为对象
  1093. * @param val 需要判断的值
  1094. * @returns 是否为对象
  1095. */
  1096. static isObject(val) {
  1097. return (!TypeValidator.isNull(val) &&
  1098. !TypeValidator.isUndefined(val) &&
  1099. typeof val === 'object');
  1100. }
  1101. /**
  1102. * 判断是否为数组
  1103. * @param val 需要判断的值
  1104. * @returns 是否为数组
  1105. */
  1106. static isArray(val) {
  1107. return Array.isArray(val);
  1108. }
  1109. /**
  1110. * 判断是否为数组
  1111. * @param val 需要判断的值
  1112. * @returns 是否为数组
  1113. */
  1114. static isFunction(val) {
  1115. return toString.call(val) === '[object Function]';
  1116. }
  1117. /**
  1118. * 判断是否为日期
  1119. * @param val 需要判断的值
  1120. * @returns 是否为日期
  1121. */
  1122. static isDate(val) {
  1123. return toString.call(val) === '[object Date]';
  1124. }
  1125. /**
  1126. * 判断是否为浏览器文件类型
  1127. * @param val 需要判断的值
  1128. * @returns 是否为浏览器文件类型
  1129. */
  1130. static isFile(val) {
  1131. return toString.call(val) === '[object File]';
  1132. }
  1133. /**
  1134. * 判断是否为浏览器二进制类型
  1135. * @param val 需要判断的值
  1136. * @returns 是否为浏览器二进制类型
  1137. */
  1138. static isBlob(val) {
  1139. return toString.call(val) === '[object Blob]';
  1140. }
  1141. /**
  1142. * 判断是否为浏览器流类型
  1143. * @param val 需要判断的值
  1144. * @returns 是否为浏览器流类型
  1145. */
  1146. static isStream(val) {
  1147. return TypeValidator.isObject(val) && TypeValidator.isFunction(val.pipe);
  1148. }
  1149. /**
  1150. * 判断是否为浏览器 ArrayBuffer 类型
  1151. * @param val 需要判断的值
  1152. * @returns 是否为浏览器 ArrayBuffer 类型
  1153. */
  1154. static isArrayBuffer(val) {
  1155. return toString.call(val) === '[object ArrayBuffer]';
  1156. }
  1157. /**
  1158. * 判断是否为浏览器 ArrayBufferView 类型
  1159. * @param val 需要判断的值
  1160. * @returns 是否为浏览器 ArrayBufferView 类型
  1161. */
  1162. static isArrayBufferView(val) {
  1163. return typeof ArrayBuffer !== 'undefined' && ArrayBuffer.isView
  1164. ? ArrayBuffer.isView(val)
  1165. : val && val.buffer && val.buffer instanceof ArrayBuffer;
  1166. }
  1167. /**
  1168. * 判断是否为浏览器 URLSearchParams 类型
  1169. * @param val 需要判断的值
  1170. * @returns 是否为浏览器 URLSearchParams 类型
  1171. */
  1172. static isURLSearchParams(val) {
  1173. return !TypeValidator.isUndefined(val) && val instanceof URLSearchParams;
  1174. }
  1175. /**
  1176. * 判断是否为浏览器 FormData 类型
  1177. * @param val 需要判断的值
  1178. * @returns 是否为浏览器 FormData 类型
  1179. */
  1180. static isFormData(val) {
  1181. return !TypeValidator.isUndefined(val) && val instanceof FormData;
  1182. }
  1183. }
  1184.  
  1185. /**
  1186. * 安全执行某个函数
  1187. * 支持异步函数
  1188. * @param fn 需要执行的函数
  1189. * @param defaultVal 发生异常后的默认返回值,默认为 null
  1190. * @param args 可选的函数参数
  1191. * @returns 函数执行的结果,或者其默认值
  1192. */
  1193. function safeExec(fn, defaultVal, ...args) {
  1194. const defRes = (defaultVal === undefined ? null : defaultVal);
  1195. try {
  1196. const res = fn(...args);
  1197. return res instanceof Promise ? res.catch(() => defRes) : res;
  1198. }
  1199. catch (err) {
  1200. return defRes;
  1201. }
  1202. }
  1203.  
  1204. /**
  1205. * 提取对象中的字段并封装为函数
  1206. * @param k 提取的字段,深度获取使用 . 分割不同的字段
  1207. * @returns 获取对象中指定字段的函数
  1208. */
  1209. function extractField(k) {
  1210. const fields = TypeValidator.isString(k) ? k.split('.') : [k];
  1211. return fields.reduceRight((fn, field) => {
  1212. return function (obj) {
  1213. return safeExec(() => fn(Reflect.get(obj, field)));
  1214. };
  1215. }, returnItself);
  1216. }
  1217.  
  1218. /**
  1219. * 获取提取对象属性的函数
  1220. * @param k 提取对象属性的函数或者是属性名(允许使用 . 进行分割)
  1221. * @returns 提取对象属性的函数
  1222. */
  1223. function getKFn(k) {
  1224. return k instanceof Function ? k : extractField(k);
  1225. }
  1226.  
  1227. /**
  1228. * 自行实现 flatMap,将数组压平一层
  1229. * @param arr 数组
  1230. * @param k 映射方法,将一个元素映射为一个数组
  1231. * @returns 压平一层的数组
  1232. */
  1233. function flatMap(arr, k = (v) => Array.from(v)) {
  1234. const fn = getKFn(k);
  1235. return arr.reduce((res, v, i, arr) => {
  1236. res.push(...fn(v, i, arr));
  1237. return res;
  1238. }, new Array());
  1239. }
  1240.  
  1241. function groupBy(arr, k,
  1242. /**
  1243. * 默认的值处理函数
  1244. * @param res 最终 V 集合
  1245. * @param item 当前迭代的元素
  1246. * @returns 将当前元素合并后的最终 V 集合
  1247. */
  1248. vFn = ((res, item) => {
  1249. res.push(item);
  1250. return res;
  1251. }), init = () => []) {
  1252. const kFn = getKFn(k);
  1253. // 将元素按照分组条件进行分组得到一个 条件 -> 数组 的对象
  1254. return arr.reduce((res, item, index, arr) => {
  1255. const k = kFn(item, index, arr);
  1256. // 如果已经有这个键了就直接追加, 否则先将之初始化再追加元素
  1257. if (!res.has(k)) {
  1258. res.set(k, init());
  1259. }
  1260. res.set(k, vFn(res.get(k), item, index, arr));
  1261. return res;
  1262. }, new Map());
  1263. }
  1264.  
  1265. /**
  1266. * 创建一个等差数列数组
  1267. * @param start 开始(包含)
  1268. * @param end 结束(不包含)
  1269. * @param sep 步长,默认为 1
  1270. * @returns 等差数列数组
  1271. */
  1272. function range(start, end, sep = 1) {
  1273. const arr = [];
  1274. for (let i = start; i < end; i += sep) {
  1275. arr.push(i);
  1276. }
  1277. return arr;
  1278. }
  1279.  
  1280. /**
  1281. * 将数组转化为一个 Object 对象
  1282. * @deprecated 已废弃,请使用更好的 {@link arrayToMap} 替代
  1283. * @param arr 需要进行转换的数组
  1284. * @param k 生成对象属性名的函数
  1285. * @param v 生成对象属性值的函数,默认为数组中的迭代元素
  1286. * @returns 转化得到的对象
  1287. */
  1288. function toObject(arr, k, v = returnItself) {
  1289. const kFn = getKFn(k);
  1290. const vFn = getKFn(v);
  1291. return arr.reduce((res, item, i, arr) => {
  1292. const k = kFn(item, i, arr);
  1293. if (!Reflect.has(res, k)) {
  1294. Reflect.set(res, k, vFn(item, i, arr));
  1295. }
  1296. return res;
  1297. }, {});
  1298. }
  1299.  
  1300. /**
  1301. * js 的数组去重方法
  1302. * @param arr 要进行去重的数组
  1303. * @param k 唯一标识元素的方法,默认使用 {@link returnItself}
  1304. * @returns 进行去重操作之后得到的新的数组 (原数组并未改变)
  1305. */
  1306. function uniqueBy(arr, k = returnItself) {
  1307. const kFn = getKFn(k);
  1308. const set = new Set();
  1309. return arr.filter((v, ...args) => {
  1310. const k = kFn(v, ...args);
  1311. if (set.has(k)) {
  1312. return false;
  1313. }
  1314. set.add(k);
  1315. return true;
  1316. });
  1317. }
  1318.  
  1319. /**
  1320. * 将数组映射为 Map
  1321. * @param arr 数组
  1322. * @param k 产生 Map 元素唯一标识的函数,或者对象元素中的一个属性名
  1323. * @param v 产生 Map 值的函数,默认为返回数组的元素,或者对象元素中的一个属性名
  1324. * @returns 映射产生的 map 集合
  1325. */
  1326. function arrayToMap(arr, k, v = returnItself) {
  1327. const kFn = getKFn(k);
  1328. const vFn = getKFn(v);
  1329. return arr.reduce((res, item, index, arr) => res.set(kFn(item, index, arr), vFn(item, index, arr)), new Map());
  1330. }
  1331.  
  1332. /**
  1333. * 判断数字是否在指定区间之中
  1334. * @param num 指定数字
  1335. * @param min 最小值
  1336. * @param max 最大值(不包含)
  1337. */
  1338. function isRange(num, min, max) {
  1339. return num >= min && num < max;
  1340. }
  1341.  
  1342. /**
  1343. * 日期格式化类
  1344. */
  1345. class DateFormat {
  1346. /**
  1347. * 构造函数
  1348. * @param name 日期格式的名称
  1349. * @param format 日期的格式值
  1350. * @param value 格式化得到的值
  1351. * @param index 需要替换位置的索引
  1352. */
  1353. constructor(name, format, value, index) {
  1354. this.name = name;
  1355. this.format = format;
  1356. this.value = value;
  1357. this.index = index;
  1358. }
  1359. }
  1360. /**
  1361. * 日期时间的正则表达式
  1362. */
  1363. const dateFormats = new Map()
  1364. .set('year', 'Y{4}|Y{2}|y{4}|y{2}')
  1365. .set('month', 'M{1,2}')
  1366. .set('day', 'D{1,2}|d{1,2}')
  1367. .set('hour', 'h{1,2}')
  1368. .set('minute', 'm{1,2}')
  1369. .set('second', 's{1,2}')
  1370. .set('millieSecond', 'S{1,3}');
  1371. /**
  1372. * 如果没有格式化某项的话则设置为默认时间
  1373. */
  1374. const defaultDateValues = new Map()
  1375. .set('month', '01')
  1376. .set('day', '01')
  1377. .set('hour', '00')
  1378. .set('minute', '00')
  1379. .set('second', '00')
  1380. .set('millieSecond', '000');
  1381. /**
  1382. * 月份日期校验
  1383. */
  1384. const monthDayValidate = {
  1385. 1: 31,
  1386. 3: 31,
  1387. 5: 31,
  1388. 7: 31,
  1389. 8: 31,
  1390. 10: 31,
  1391. 12: 31,
  1392. 4: 30,
  1393. 6: 30,
  1394. 9: 30,
  1395. 11: 30,
  1396. 2: 28,
  1397. };
  1398. /**
  1399. * 解析字符串为 Date 对象
  1400. * @param str 日期字符串
  1401. * @param fmt 日期字符串的格式,目前仅支持使用 y(年),M(月),d(日),h(时),m(分),s(秒),S(毫秒)
  1402. * @returns 解析得到的 Date 对象
  1403. */
  1404. function dateParse(str, fmt) {
  1405. const now = new Date();
  1406. defaultDateValues.set('year', now.getFullYear().toString());
  1407. // 保存对传入的日期字符串进行格式化的全部信息数组列表
  1408. const dateUnits = [];
  1409. for (const [fmtName, regex] of dateFormats) {
  1410. const regExp = new RegExp(regex);
  1411. if (regExp.test(fmt)) {
  1412. const matchStr = regExp.exec(fmt)[0];
  1413. const regexStr = '`'.repeat(matchStr.length);
  1414. const index = fmt.indexOf(matchStr);
  1415. fmt = fmt.replace(matchStr, regexStr);
  1416. dateUnits.push(new DateFormat(fmtName, '\\d'.repeat(matchStr.length), null, index));
  1417. }
  1418. else {
  1419. dateUnits.push(new DateFormat(fmtName, null, defaultDateValues.get(fmtName), -1));
  1420. }
  1421. }
  1422. // 进行验证是否真的是符合传入格式的字符串
  1423. fmt = fmt.replace(new RegExp('`', 'g'), '\\d');
  1424. if (!new RegExp(`^${fmt}$`).test(str)) {
  1425. return null;
  1426. }
  1427. // 进行一次排序, 依次对字符串进行截取
  1428. dateUnits
  1429. // 过滤掉没有得到格式化的对象
  1430. .filter(({ format }) => format)
  1431. // 按照字符串中日期片段的索引进行排序
  1432. .sort(function (a, b) {
  1433. return a.index - b.index;
  1434. })
  1435. // 获取到匹配的日期片段的值
  1436. .map(format => {
  1437. const matchDateUnit = new RegExp(format.format).exec(str);
  1438. if (matchDateUnit !== null && matchDateUnit.length > 0) {
  1439. str = str.replace(matchDateUnit[0], '');
  1440. format.value = matchDateUnit[0];
  1441. }
  1442. return format;
  1443. })
  1444. // 覆写到 dateStr 上面
  1445. .forEach(({ format }, i) => {
  1446. const matchDateUnit = new RegExp(format).exec(str);
  1447. if (matchDateUnit !== null && matchDateUnit.length > 0) {
  1448. str = str.replace(matchDateUnit[0], '');
  1449. dateUnits[i].value = matchDateUnit[0];
  1450. }
  1451. });
  1452. // 将截取完成的信息封装成对象并格式化标准的日期字符串
  1453. const map = arrayToMap(dateUnits, item => item.name, item => item.value);
  1454. if (map.get('year').length === 2) {
  1455. map.set('year', defaultDateValues
  1456. .get('year')
  1457. .substr(0, 2)
  1458. .concat(map.get('year')));
  1459. }
  1460. // 注意:此处使用的是本地时间而非 UTC 时间
  1461. const get = (unit) => parseInt(map.get(unit));
  1462. const year = get('year');
  1463. const month = get('month');
  1464. const day = get('day');
  1465. const hour = get('hour');
  1466. const minute = get('minute');
  1467. const second = get('second');
  1468. const millieSecond = get('millieSecond');
  1469. if (!isRange(month, 1, 12 + 1)) {
  1470. return null;
  1471. }
  1472. if (!isRange(day, 1, Reflect.get(monthDayValidate, month) +
  1473. (month === 2 && year % 4 === 0 ? 1 : 0) +
  1474. 1)) {
  1475. return null;
  1476. }
  1477. if (!isRange(hour, 0, 24 + 1) ||
  1478. !isRange(minute, 0, 60 + 1) ||
  1479. !isRange(second, 0, 60 + 1) ||
  1480. !isRange(millieSecond, 0, 999 + 1)) {
  1481. return null;
  1482. }
  1483. return new Date(year, month - 1, day, hour, minute, second, millieSecond);
  1484. }
  1485.  
  1486. /**
  1487. * 解析字符串为 Date 对象
  1488. * @deprecated 已弃用,请使用可读性更好的 {@link dateParse} 代替
  1489. * @param dateStr 日期字符串
  1490. * @param fmt 日期字符串的格式
  1491. * 目前仅支持使用 y(年),M(月),d(日),h(时),m(分),s(秒),S(毫秒)
  1492. * @returns 解析得到的 Date 对象
  1493. */
  1494. function strToDate(dateStr, fmt) {
  1495. return dateParse(dateStr, fmt);
  1496. }
  1497.  
  1498. /**
  1499. * 复制一段文本内容
  1500. * @param text 要进行复制的文本
  1501. * @returns 是否复制成功
  1502. */
  1503. function copyText(text) {
  1504. const $el = document.createElement('textarea');
  1505. $el.style.position = 'fixed';
  1506. $el.style.top = '-1000px';
  1507. document.body.appendChild($el);
  1508. $el.value = text;
  1509. $el.select();
  1510. const res = document.execCommand('copy');
  1511. document.body.removeChild($el);
  1512. return res;
  1513. }
  1514.  
  1515. /**
  1516. * 根据 html 字符串创建 Element 元素
  1517. * @param str html 字符串
  1518. * @returns 创建的 Element 元素
  1519. */
  1520. function createElByString(str) {
  1521. const root = document.createElement('div');
  1522. root.innerHTML = str;
  1523. return root.querySelector('*');
  1524. }
  1525.  
  1526. /**
  1527. * 获取输入框中光标所在位置
  1528. * @param {HTMLFormElement} el 需要获取的输入框元素
  1529. * @returns 光标所在位置的下标
  1530. */
  1531. function getCursorPosition(el) {
  1532. return el.selectionStart;
  1533. }
  1534.  
  1535. /**
  1536. * 获取输入框中光标所在位置
  1537. * @param {HTMLFormElement} el 需要获取的输入框元素
  1538. * @returns 光标所在位置的下标
  1539. * @deprecated 已废弃,请使用正确更名后的 {@link getCursorPosition} 函数
  1540. */
  1541. function getCusorPostion(el) {
  1542. return getCursorPosition(el);
  1543. }
  1544.  
  1545. /**
  1546. * 设置输入框中选中的文本/光标所在位置
  1547. * @param el 需要设置的输入框元素
  1548. * @param start 光标所在位置的下标
  1549. * @param end 结束位置,默认为输入框结束
  1550. */
  1551. function setCursorPosition(el, start, end = start) {
  1552. el.focus();
  1553. el.setSelectionRange(start, end);
  1554. }
  1555.  
  1556. /**
  1557. * 在指定位置后插入文本
  1558. * @param el 需要设置的输入框元素
  1559. * @param text 要插入的值
  1560. * @param start 开始位置,默认为当前光标处
  1561. */
  1562. function insertText(el, text, start = getCursorPosition(el)) {
  1563. const value = el.value;
  1564. el.value = value.substr(0, start) + text + value.substr(start);
  1565. setCursorPosition(el, start + text.length);
  1566. }
  1567.  
  1568. /**
  1569. * 字符串安全的转换为小写
  1570. * @param str 字符串
  1571. * @returns 转换后得到的全小写字符串
  1572. */
  1573. function toLowerCase(str) {
  1574. if (isNullOrUndefined(str) || typeof str !== 'string') {
  1575. return str;
  1576. }
  1577. return str.toLowerCase();
  1578. }
  1579.  
  1580. /**
  1581. * 判断指定元素是否是可编辑元素
  1582. * 注:可编辑元素并不一定能够进行编辑,例如只读的 input 元素
  1583. * @param el 需要进行判断的元素
  1584. * @returns 是否为可编辑元素
  1585. */
  1586. function isEditable(el) {
  1587. const inputEls = ['input', 'date', 'datetime', 'select', 'textarea'];
  1588. return (
  1589. // 此处需要判断是否存在属性 isContentEditable
  1590. // @ts-ignore
  1591. el && (el.isContentEditable || inputEls.includes(toLowerCase(el.tagName))));
  1592. }
  1593.  
  1594. let lastFocusEl;
  1595. document.addEventListener('focus', event => {
  1596. lastFocusEl = event.target;
  1597. }, true);
  1598. document.addEventListener('blur', () => {
  1599. lastFocusEl = null;
  1600. }, true);
  1601. /**
  1602. * 获取到最后一个获得焦点的元素
  1603. * @returns 最后一个获取到焦点的元素
  1604. */
  1605. function lastFocus() {
  1606. return lastFocusEl;
  1607. }
  1608.  
  1609. /**
  1610. * 直接删除指定元素
  1611. * @param el 需要删除的元素
  1612. * @returns 返回被删除的元素
  1613. */
  1614. function removeEl(el) {
  1615. const parent = el.parentElement;
  1616. if (parent == null) {
  1617. return null;
  1618. }
  1619. return parent.removeChild(el);
  1620. }
  1621.  
  1622. /**
  1623. * 在指定范围内删除文本
  1624. * @param el 需要设置的输入框元素
  1625. * @param start 开始位置,默认为当前选中开始位置
  1626. * @param end 结束位置,默认为当前选中结束位置
  1627. */
  1628. function removeText(el, start = el.selectionStart, end = el.selectionEnd) {
  1629. // 删除之前必须要 [记住] 当前光标的位置
  1630. const index = getCursorPosition(el);
  1631. const value = el.value;
  1632. el.value = value.substr(0, start) + value.substr(end, value.length);
  1633. setCursorPosition(el, index);
  1634. }
  1635.  
  1636. /**
  1637. * 设置输入框中选中的文本/光标所在位置
  1638. * @param el 需要设置的输入框元素
  1639. * @param start 光标所在位置的下标
  1640. * @param end 结束位置,默认为输入框结束
  1641. * @deprecated 已废弃,请使用正确更名后的 {@link setCursorPosition} 函数
  1642. */
  1643. function setCusorPostion(el, start, end = start) {
  1644. return setCursorPosition(el, start, end);
  1645. }
  1646.  
  1647. /**
  1648. * 监听 event 的添加/删除,使 DOM 事件是可撤销的
  1649. * 注:必须及早运行,否则无法监听之前添加的事件
  1650. */
  1651. function watchEventListener() {
  1652. /**
  1653. * 用来保存监听到的事件信息
  1654. */
  1655. class Event {
  1656. constructor(el, type, listener, useCapture) {
  1657. this.el = el;
  1658. this.type = type;
  1659. this.listener = listener;
  1660. this.useCapture = useCapture;
  1661. }
  1662. }
  1663. /**
  1664. * 监听所有的 addEventListener, removeEventListener 事件
  1665. */
  1666. const documentAddEventListener = document.addEventListener;
  1667. const eventTargetAddEventListener = EventTarget.prototype.addEventListener;
  1668. const documentRemoveEventListener = document.removeEventListener;
  1669. const eventTargetRemoveEventListener = EventTarget.prototype.removeEventListener;
  1670. const events = [];
  1671. /**
  1672. * 自定义的添加事件监听函数
  1673. * @param type 事件类型
  1674. * @param listener 事件监听函数
  1675. * @param [useCapture] 是否需要捕获事件冒泡,默认为 false
  1676. */
  1677. function addEventListener(type, listener, useCapture = false) {
  1678. const $addEventListener =
  1679. // @ts-ignore
  1680. this === document ? documentAddEventListener : eventTargetAddEventListener;
  1681. // @ts-ignore
  1682. events.push(new Event(this, type, listener, useCapture));
  1683. // @ts-ignore
  1684. $addEventListener.apply(this, arguments);
  1685. }
  1686. /**
  1687. * 自定义的根据类型删除事件函数
  1688. * 该方法会删除这个类型下面全部的监听函数,不管数量
  1689. * @param type 事件类型
  1690. */
  1691. function removeEventListenerByType(type) {
  1692. const $removeEventListener =
  1693. // @ts-ignore
  1694. this === document
  1695. ? documentRemoveEventListener
  1696. : eventTargetRemoveEventListener;
  1697. const removeIndexList = events
  1698. // @ts-ignore
  1699. .map((e, i) => (e.el === this || e.type === arguments[0] ? i : -1))
  1700. .filter(i => i !== -1);
  1701. removeIndexList.forEach(i => {
  1702. const e = events[i];
  1703. $removeEventListener.apply(e.el, [e.type, e.listener, e.useCapture]);
  1704. });
  1705. removeIndexList.sort((a, b) => b - a).forEach(i => events.splice(i, 1));
  1706. }
  1707. document.addEventListener = EventTarget.prototype.addEventListener = addEventListener;
  1708. // @ts-ignore
  1709. document.removeEventListenerByType = EventTarget.prototype.removeEventListenerByType = removeEventListenerByType;
  1710. }
  1711.  
  1712. /**
  1713. * 将任意对象转换为 String
  1714. * 主要避免原生 Object toString 遇到某些空值的时候抛异常的问题
  1715. * @param object 任意对象
  1716. * @returns 字符串
  1717. */
  1718. function toString$1(object) {
  1719. if (isNullOrUndefined(object)) {
  1720. return '';
  1721. }
  1722. if (object instanceof Date) {
  1723. return object.toISOString();
  1724. }
  1725. return object.toString();
  1726. }
  1727.  
  1728. /**
  1729. * FormData 批量添加方法
  1730. * 注:该方法不会覆盖掉原本的属性
  1731. * @param fd FormData 对象
  1732. * @param obj 键值对对象
  1733. * @returns 添加完成后的 FormData 对象
  1734. */
  1735. function appends(fd, obj) {
  1736. for (const k in obj) {
  1737. const v = obj[k];
  1738. fd.append(k, toString$1(v));
  1739. }
  1740. return fd;
  1741. }
  1742.  
  1743. /**
  1744. * FormData 批量删除方法
  1745. * @param fd FormData 对象
  1746. * @param keys 删除的 key 列表
  1747. * @returns 返回删除后的 FormData 对象
  1748. */
  1749. function deletes(fd, keys) {
  1750. keys.forEach(key => fd.delete(key));
  1751. return fd;
  1752. }
  1753.  
  1754. /**
  1755. * FormData 批量设置方法
  1756. * 注:该方法会覆盖掉原本的属性
  1757. * @param fd 表单对象
  1758. * @param obj 键值对对象
  1759. * @returns 设置完成后的 FormData 对象
  1760. */
  1761. function sets(fd, obj) {
  1762. for (const k in obj) {
  1763. fd.set(k, obj[k]);
  1764. }
  1765. return fd;
  1766. }
  1767.  
  1768. /**
  1769. * FormData 转换为包含所有键值数组的二维数组函数
  1770. *
  1771. * @param fd 需要转换的 FormData 对象
  1772. * @returns 转换后的数组
  1773. * @deprecated 已被原生函数 Array.from 取代
  1774. */
  1775. function formDataToArray(fd) {
  1776. // @ts-ignore
  1777. return Array.from(fd);
  1778. }
  1779.  
  1780. /**
  1781. * 将参数对象转换为 FormData,只转换一层
  1782. * @param data 参数对象
  1783. * @return {FormData} 转换后的表单对象
  1784. */
  1785. function objToFormData(data) {
  1786. return Object.entries(data).reduce((res, [k, v]) => {
  1787. if (v instanceof Blob) {
  1788. res.append(k, v);
  1789. }
  1790. else {
  1791. res.append(k, v && v.toString());
  1792. }
  1793. return res;
  1794. }, new FormData());
  1795. }
  1796.  
  1797. /**
  1798. * 函数去抖
  1799. * 去抖 (debounce) 去抖就是对于一定时间段的连续的函数调用,只让其执行一次
  1800. * 注: 包装后的函数如果两次操作间隔小于 delay 则不会被执行, 如果一直在操作就会一直不执行, 直到操作停止的时间大于 delay 最小间隔时间才会执行一次, 不管任何时间调用都需要停止操作等待最小延迟时间
  1801. * 应用场景主要在那些连续的操作, 例如页面滚动监听, 包装后的函数只会执行最后一次
  1802. * 注: 该函数第一次调用一定不会执行,第一次一定拿不到缓存值,后面的连续调用都会拿到上一次的缓存值。如果需要在第一次调用获取到的缓存值,则需要传入第三个参数 {@param init},默认为 {@code undefined} 的可选参数
  1803. * 注: 返回函数结果的高阶函数需要使用 {@see Proxy} 实现,以避免原函数原型链上的信息丢失
  1804. *
  1805. * @param delay 最小延迟时间,单位为 ms
  1806. * @param action 真正需要执行的操作
  1807. * @param init 初始的缓存值,不填默认为 {@see undefined}
  1808. * @return 包装后有去抖功能的函数。该函数是异步的,与需要包装的函数 {@see action} 是否异步没有太大关联
  1809. */
  1810. function debounce(delay, action, init = null) {
  1811. let flag;
  1812. let result = init;
  1813. return new Proxy(action, {
  1814. apply(_, _this, args) {
  1815. return new Promise(resolve => {
  1816. if (flag)
  1817. clearTimeout(flag);
  1818. flag = setTimeout(() => resolve((result = Reflect.apply(_, _this, args))), delay);
  1819. setTimeout(() => resolve(result), delay);
  1820. });
  1821. },
  1822. });
  1823. }
  1824.  
  1825. /**
  1826. * 使用 Proxy 实现通用的单例模式
  1827. * @param clazz 需要包装为单例的类型
  1828. * @returns 包装后的单例模式类,使用 {@code new} 创建将只在第一次有效
  1829. */
  1830. function singleModel(clazz) {
  1831. let instance;
  1832. return new Proxy(clazz, {
  1833. construct(target, args, newTarget) {
  1834. if (instance === undefined) {
  1835. instance = Reflect.construct(target, args, newTarget);
  1836. }
  1837. return instance;
  1838. },
  1839. });
  1840. }
  1841.  
  1842. /**
  1843. * 状态机
  1844. * 用于避免使用 if-else 的一种方式
  1845. * @typeparam K 状态的类型,默认为 any
  1846. * @typeparam V 构造函数返回值的类型,一般为实现子类的基类,默认为 any
  1847. * @deprecated 该类将在下个大版本进行重构,使用函数而非类作为基本单元
  1848. */
  1849. class StateMachine {
  1850. constructor() {
  1851. this.classMap = new Map();
  1852. }
  1853. /**
  1854. * 获取到一个状态工厂
  1855. * @deprecated 已废弃,请直接创建一个 StateMachine 实例
  1856. */
  1857. static getFactory() {
  1858. /**
  1859. * 状态注册器
  1860. * 更好的有限状态机,分离子类与构建的关系,无论子类如何增删该都不影响基类及工厂类
  1861. */
  1862. return new StateMachine();
  1863. }
  1864. /**
  1865. * 注册一个 class,创建子类时调用,用于记录每一个 [状态 => 子类] 对应
  1866. * 注: 此处不再默认使用单例模式,如果需要,请自行对 class 进行包装
  1867. * @param state 作为键的状态
  1868. * @param clazz 对应的子类型
  1869. * @returns 返回 clazz 本身
  1870. */
  1871. register(state, clazz) {
  1872. this.classMap.set(state, clazz);
  1873. return clazz;
  1874. }
  1875. /**
  1876. * 获取一个标签子类对象
  1877. * @param state 状态索引
  1878. * @param args 构造函数的参数
  1879. * @returns 子类对象
  1880. */
  1881. getInstance(state, ...args) {
  1882. const Class = this.classMap.get(state);
  1883. if (!Class) {
  1884. return null;
  1885. }
  1886. // 构造函数的参数
  1887. return new Class(...args);
  1888. }
  1889. /**
  1890. * 允许使用 for-of 遍历整个状态机
  1891. */
  1892. *[Symbol.iterator]() {
  1893. for (const kv of this.classMap.entries()) {
  1894. yield kv;
  1895. }
  1896. }
  1897. }
  1898.  
  1899. /**
  1900. * 函数节流
  1901. * 节流 (throttle) 让一个函数不要执行的太频繁,减少执行过快的调用,叫节流
  1902. * 类似于上面而又不同于上面的函数去抖, 包装后函数在上一次操作执行过去了最小间隔时间后会直接执行, 否则会忽略该次操作
  1903. * 与上面函数去抖的明显区别在连续操作时会按照最小间隔时间循环执行操作, 而非仅执行最后一次操作
  1904. * 注: 该函数第一次调用一定会执行,不需要担心第一次拿不到缓存值,后面的连续调用都会拿到上一次的缓存值
  1905. * 注: 返回函数结果的高阶函数需要使用 {@see Proxy} 实现,以避免原函数原型链上的信息丢失
  1906. *
  1907. * @param delay 最小间隔时间,单位为 ms
  1908. * @param action 真正需要执行的操作
  1909. * @return {Function} 包装后有节流功能的函数。该函数是异步的,与需要包装的函数 {@link action} 是否异步没有太大关联
  1910. */
  1911. function throttle(delay, action) {
  1912. let last = 0;
  1913. let result;
  1914. return new Proxy(action, {
  1915. apply(target, thisArg, args) {
  1916. return new Promise(resolve => {
  1917. const curr = Date.now();
  1918. if (curr - last > delay) {
  1919. result = Reflect.apply(target, thisArg, args);
  1920. last = curr;
  1921. resolve(result);
  1922. return;
  1923. }
  1924. resolve(result);
  1925. });
  1926. },
  1927. });
  1928. }
  1929.  
  1930. /**
  1931. * 测试函数的执行时间
  1932. * 注:如果函数返回 Promise,则该函数也会返回 Promise,否则直接返回执行时间
  1933. * @param fn 需要测试的函数
  1934. * @returns 执行的毫秒数
  1935. */
  1936. function timing(fn) {
  1937. const begin = performance.now();
  1938. const res = fn();
  1939. return compatibleAsync(res, () => performance.now() - begin);
  1940. }
  1941.  
  1942. /**
  1943. * 轮询等待指定资源加载完毕再执行操作
  1944. * 使用 Promises 实现,可以使用 ES7 的 {@see async} 和 {@see await} 调用
  1945. * @param fn 判断必须的资源是否存在的方法
  1946. * @param option 可配置项
  1947. * @returns Promise 对象
  1948. */
  1949. function waitResource(fn, { interval = 100, max = 10 } = {}) {
  1950. let current = 0;
  1951. return new Promise((resolve, reject) => {
  1952. const timer = setInterval(() => {
  1953. if (fn()) {
  1954. clearInterval(timer);
  1955. resolve();
  1956. }
  1957. current++;
  1958. if (current >= max) {
  1959. clearInterval(timer);
  1960. reject(new Error('waitResource call timeout'));
  1961. }
  1962. }, interval);
  1963. });
  1964. }
  1965.  
  1966. /**
  1967. * 监视指定函数返回值的变化
  1968. * @param fn 需要监视的函数
  1969. * @param callback 回调函数
  1970. * @param interval 每次检查的间隔时间,默认为 100ms
  1971. * @returns 关闭这个监视函数
  1972. */
  1973. function watch(fn, callback, interval = 100) {
  1974. let oldVal = fn();
  1975. const timer = setInterval(() => {
  1976. const newVal = fn();
  1977. if (oldVal !== newVal) {
  1978. callback(newVal, oldVal);
  1979. oldVal = newVal;
  1980. }
  1981. }, interval);
  1982. return () => clearInterval(timer);
  1983. }
  1984.  
  1985. /**
  1986. * 深度监听指定对象属性的变化
  1987. * 注:指定对象不能是原始类型,即不可变类型,而且对象本身的引用不能改变,最好使用 const 进行声明
  1988. * @param object 需要监视的对象
  1989. * @param callback 当代理对象发生改变时的回调函数,回调函数有三个参数,分别是对象,修改的 key,修改的 v
  1990. * @returns 返回源对象的一个代理
  1991. */
  1992. function watchObject(object, callback) {
  1993. const handler = {
  1994. get(target, k) {
  1995. try {
  1996. // 注意: 这里很关键,它为对象的字段也添加了代理
  1997. return new Proxy(Reflect.get(target, k), handler);
  1998. }
  1999. catch (err) {
  2000. return Reflect.get(target, k);
  2001. }
  2002. },
  2003. set(target, k, v) {
  2004. callback(target, k, v);
  2005. return Reflect.set(target, k, v);
  2006. },
  2007. };
  2008. return new Proxy(object, handler);
  2009. }
  2010.  
  2011. /**
  2012. * 填充字符串到指定长度
  2013. * @param item 填充的字符串
  2014. * @param len 填充的长度
  2015. * @returns 填充完成的字符串
  2016. * @deprecated 已废弃,请使用 ES6 {@link String.prototype.repeat} 函数
  2017. * 具体请参考 MDN {@url(https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/String/repeat)}
  2018. */
  2019. function fill(item, len) {
  2020. if (len <= 0) {
  2021. return '';
  2022. }
  2023. return item + fill(item, len - 1);
  2024. }
  2025.  
  2026. /**
  2027. * 字符串格式化
  2028. *
  2029. * @param str 要进行格式化的值
  2030. * @param args 格式化参数值,替换字符串中的 {} 的值
  2031. * @returns 替换完成的字符串
  2032. * @deprecated 已废弃,请使用 ES6 模板字符串 {@url(https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/template_strings)}
  2033. */
  2034. function format(str, args) {
  2035. return Object.keys(args).reduce((res, k) => res.replace(new RegExp(`{${k}}`, 'g'), toString$1(args[k])), str);
  2036. }
  2037.  
  2038. /**
  2039. * 判断是否为小数的正则表达式
  2040. */
  2041. const FloatRule = /^(-?\d+)(.\d+)?$/;
  2042. /**
  2043. * 判断是否为整数的正则表达式
  2044. */
  2045. const IntegerRule = /^-?\d+$/;
  2046. /**
  2047. * 判断是否为邮箱的正则表达式
  2048. */
  2049. const EmailRule = /^\w+((-\w+)|(\.\w+))*@[A-Za-z0-9]+((\.|-)[A-Za-z0-9]+)*\.[A-Za-z]+$/;
  2050. /**
  2051. * 判断是否为 ipv4 地址的正则表达式
  2052. */
  2053. const Ipv4Rule = /^((25[0-5]|2[0-4]\d|1?\d?\d)\.){3}(25[0-5]|2[0-4]\d|1?\d?\d)$/;
  2054. /**
  2055. * 判断是否为固定电话的正则表达式
  2056. */
  2057. const TelephoneRule = /^0[1-9][0-9]{1,2}-[2-8][0-9]{6,7}$/;
  2058. /**
  2059. * 判断是否为移动电话的正则表达式
  2060. */
  2061. const MobileRule = /^(((13[0-9]{1})|15[0-9]{1}|18[0-9]{1}|)+\d{8})$/;
  2062. /**
  2063. * 判断是否为域名的正则表达式
  2064. */
  2065. const DomainRule = /^(?=^.{3,255}$)[a-zA-Z0-9][-a-zA-Z0-9]{0,62}(\.[a-zA-Z0-9][-a-zA-Z0-9]{0,62})+$/;
  2066. /**
  2067. * 判断是否为邮政编码的正则表达式
  2068. */
  2069. const PostcodeRule = /^\d{6}$/;
  2070. /**
  2071. * 字符串校验
  2072. * TODO 使用 any 可能是个严重的错误。。。
  2073. */
  2074. class StringValidator {
  2075. /**
  2076. * 判断一个字符串是否为空字符串
  2077. * @param str 字符串
  2078. * @returns 是否为空字符串
  2079. */
  2080. static isEmpty(str) {
  2081. return isNullOrUndefined(str) || str === '';
  2082. }
  2083. /**
  2084. * 判断一个字符串是否为空白的字符串
  2085. * @param str 字符串
  2086. * @returns 是否为空字符串
  2087. */
  2088. static isBlank(str) {
  2089. return StringValidator.isEmpty(str) || str.trim() === '';
  2090. }
  2091. /**
  2092. * 判断字符串是否位小数
  2093. * @param str 需要进行判断的字符串
  2094. * @returns 是否为小数
  2095. */
  2096. static isFloat(str) {
  2097. if (isNullOrUndefined(str)) {
  2098. return false;
  2099. }
  2100. return FloatRule.test(str);
  2101. }
  2102. /**
  2103. * 判断字符串是否位整数
  2104. * @param str 需要进行判断的字符串
  2105. * @returns 是否为小数
  2106. */
  2107. static isInteger(str) {
  2108. return !isNullOrUndefined(str) && IntegerRule.test(str);
  2109. }
  2110. /**
  2111. * 判断邮箱的格式是否正确
  2112. * @param str 邮箱字符串
  2113. * @returns 是否是邮箱
  2114. */
  2115. static isEmail(str) {
  2116. return !isNullOrUndefined(str) && EmailRule.test(str);
  2117. }
  2118. /**
  2119. * 判断 ipv4 地址的格式是否正确
  2120. * @param str ipv4 字符串
  2121. * @returns 是否是 ipv4 地址
  2122. */
  2123. static isIpv4(str) {
  2124. return !isNullOrUndefined(str) && Ipv4Rule.test(str);
  2125. }
  2126. /**
  2127. * 判断字符串是否为正确的端口号
  2128. * 正确的端口号是 1-65535
  2129. * @param str 字符串
  2130. * @returns 是否为端口号
  2131. */
  2132. static isPort(str) {
  2133. // tslint:disable-next-line:radix
  2134. return StringValidator.isInteger(str) && isRange(parseInt(str), 1, 65535);
  2135. }
  2136. /**
  2137. * 判断是否为固定电话
  2138. * @param str 字符串
  2139. * @returns 是否为固定电话
  2140. */
  2141. static isTelephone(str) {
  2142. return !isNullOrUndefined(str) && TelephoneRule.test(str);
  2143. }
  2144. /**
  2145. * 判断是否为移动电话
  2146. * @param str 字符串
  2147. * @returns 是否为移动电话
  2148. */
  2149. static isMobile(str) {
  2150. return !isNullOrUndefined(str) && MobileRule.test(str);
  2151. }
  2152. /**
  2153. * 判断是否为域名
  2154. * @param str 字符串
  2155. * @returns 是否为域名
  2156. */
  2157. static isDomain(str) {
  2158. return !isNullOrUndefined(str) && DomainRule.test(str);
  2159. }
  2160. /**
  2161. * 判断是否为邮政编码
  2162. * @param str 字符串
  2163. * @returns 是否为邮政编码
  2164. */
  2165. static isPostcode(str) {
  2166. return !isNullOrUndefined(str) && PostcodeRule.test(str);
  2167. }
  2168. }
  2169. /**
  2170. * 导出一个字符串校验的对象
  2171. * @deprecated 已废弃,请直接使用类的静态函数
  2172. */
  2173. const stringValidator = StringValidator;
  2174.  
  2175. /**
  2176. * 判断字符串是否位小数
  2177. * @param str 需要进行判断的字符串
  2178. * @returns 是否为小数
  2179. * @deprecated 已废弃,请使用 {@link stringValidator#isFloat}
  2180. */
  2181. function isFloat(str) {
  2182. return stringValidator.isFloat(str);
  2183. }
  2184.  
  2185. /**
  2186. * 判断字符串是否位整数
  2187. * @param str 需要进行判断的字符串
  2188. * @returns 是否为小数
  2189. * @deprecated 已废弃,请使用 {@link stringValidator#isInteger}
  2190. */
  2191. function isNumber(str) {
  2192. return stringValidator.isInteger(str);
  2193. }
  2194.  
  2195. /**
  2196. * 字符串安全的转换为大写
  2197. * @param str 字符串
  2198. * @returns 转换后得到的全大写字符串
  2199. */
  2200. function toUpperCase(str) {
  2201. if (isNullOrUndefined(str) || typeof str !== 'string') {
  2202. return str;
  2203. }
  2204. return str.toUpperCase();
  2205. }
  2206.  
  2207. /**
  2208. * 将空白字符串转换为 null
  2209. *
  2210. * @param str 将空字符串转换为 {@code null}
  2211. * @returns 可能为 {@code null}
  2212. */
  2213. function blankToNull(str) {
  2214. return StringValidator.isBlank(str) ? null : str;
  2215. }
  2216.  
  2217. /**
  2218. * 置空对象所有空白的属性
  2219. * @param obj 对象
  2220. * @returns 将所有的空白属性全部转换为 null 的新对象
  2221. */
  2222. function blankToNullField(obj) {
  2223. return Object.keys(obj).reduce((res, k) => {
  2224. const v = Reflect.get(obj, k);
  2225. Reflect.set(res, k, typeof v === 'string' ? blankToNull(v) : v);
  2226. return res;
  2227. }, {});
  2228. }
  2229.  
  2230. /**
  2231. * 将对象的所有属性置空
  2232. * @param obj 需要置空属性的对象
  2233. * @returns 返回一个新的对象
  2234. */
  2235. function emptyAllField(obj) {
  2236. return Object.keys(obj).reduce((res, k) => {
  2237. Reflect.set(res, k, null);
  2238. return res;
  2239. }, {});
  2240. }
  2241.  
  2242. /**
  2243. * 排除对象中的指定字段
  2244. * 注: 此处将获得一个浅拷贝对象
  2245. * @param obj 排除对象
  2246. * @param fields 要排除的多个字段
  2247. * @returns 排除完指定字段得到的新的对象
  2248. */
  2249. function excludeFields(obj, ...fields) {
  2250. const set = new Set(fields);
  2251. return Object.keys(obj).reduce((res, k) => {
  2252. if (!set.has(k)) {
  2253. Reflect.set(res, k, Reflect.get(obj, k));
  2254. }
  2255. return res;
  2256. }, {});
  2257. }
  2258.  
  2259. /**
  2260. * 将 Map 转换为 Object 对象
  2261. * @param map Map 键值表
  2262. * @returns 转换得到的 Object 对象
  2263. */
  2264. function mapToObject(map) {
  2265. const res = {};
  2266. for (const [k, v] of map) {
  2267. Reflect.set(res, k, v);
  2268. }
  2269. return res;
  2270. }
  2271.  
  2272. /**
  2273. * 生成一个随机的数字
  2274. * 如果没有参数,则会抛出异常
  2275. * @param num1 第一个参数,如果只有一个参数,则认为是最大值,最小值为 0
  2276. * @param num2 第二个参数,如果该参数存在,则认为第二个是最大值,忽略剩余的参数
  2277. * @returns 生成的随机整数
  2278. */
  2279. function randomInt(num1, num2) {
  2280. const min = num2 ? num1 : 0;
  2281. const max = num2 ? num2 : num1;
  2282. return min + Math.floor(Math.random() * (max - min));
  2283. }
  2284.  
  2285. /**
  2286. * 计算月有多少天
  2287. * @param date 日期
  2288. * @returns 月的总天数
  2289. */
  2290. function calcMonEndDay(date) {
  2291. const monthToDay = [
  2292. [new Set([1, 3, 5, 7, 8, 10, 12]), 30],
  2293. [new Set([4, 6, 9, 11]), 30],
  2294. [new Set([2]), 28],
  2295. ];
  2296. const year = date.getFullYear();
  2297. const month = date.getMonth() + 1;
  2298. const days = monthToDay.find(([monthSet]) => monthSet.has(month))[1];
  2299. return days + (month === 2 && year % 4 === 0 ? 1 : 0);
  2300. }
  2301. /**
  2302. * 日期固定时间点
  2303. */
  2304. class DateConstants {
  2305. /**
  2306. * 获取指定日期一天的开始时间
  2307. * @param date 指定的时间,默认为当前日期
  2308. * @returns 一天的开始时间
  2309. */
  2310. static dayStart(date = new Date()) {
  2311. return new Date(`${dateFormat(date, 'yyyy-MM-dd')}T00:00:00.000`);
  2312. }
  2313. /**
  2314. * 获取指定日期一天的结束时间
  2315. * @param date 指定的时间,默认为当前日期
  2316. * @returns 一天的结束时间
  2317. */
  2318. static dayEnd(date = new Date()) {
  2319. return new Date(`${dateFormat(date, 'yyyy-MM-dd')}T23:59:59.999`);
  2320. }
  2321. /**
  2322. * 获取指定日期所在月的开始时间
  2323. * @param date 指定的时间,默认为当前日期
  2324. * @returns 月的开始时间
  2325. */
  2326. static monthStart(date = new Date()) {
  2327. return new Date(`${dateFormat(date, 'yyyy-MM')}-01T00:00:00.000`);
  2328. }
  2329. /**
  2330. * 获取指定日期所在月的结束时间
  2331. * @param date 指定的时间,默认为当前日期
  2332. * @returns 月的结束时间
  2333. */
  2334. static monthEnd(date = new Date()) {
  2335. return new Date(`${dateFormat(date, 'yyyy-MM')}-${calcMonEndDay(date)}T23:59:59.999`);
  2336. }
  2337. /**
  2338. * 获取指定日期所在年份的新年开始时间
  2339. * @param date 指定的时间,默认为当前日期
  2340. * @returns 新年开始时间
  2341. */
  2342. static yearStart(date = new Date()) {
  2343. return new Date(`${date.getFullYear()}-01-01T00:00:00.000`);
  2344. }
  2345. /**
  2346. * 获取指定日期所在年份的旧年结束时间
  2347. * @param date 指定的时间,默认为当前日期
  2348. * @returns 旧年结束时间
  2349. */
  2350. static yearEnd(date = new Date()) {
  2351. return new Date(`${date.getFullYear()}-12-31T23:59:59.999`);
  2352. }
  2353. }
  2354. /**
  2355. * 导出一个日期固定时间点的对象
  2356. * @deprecated 已废弃,请直接使用类的静态函数
  2357. */
  2358. const dateConstants = DateConstants;
  2359.  
  2360. /**
  2361. * 一天标准的毫秒数
  2362. */
  2363. const DAY_UNIT_TIME = 1000 * 60 * 60 * 24;
  2364. /**
  2365. * 日期增强
  2366. */
  2367. class DateEnhance {
  2368. /**
  2369. * 构造函数
  2370. * @param date 要增强的日期
  2371. */
  2372. constructor(date) {
  2373. this.date = date;
  2374. }
  2375. /**
  2376. * 获取到年份
  2377. * @returns
  2378. */
  2379. year() {
  2380. return this.date.getFullYear();
  2381. }
  2382. /**
  2383. * 获取月份
  2384. * @returns
  2385. * @deprecated 已废弃,请使用 {@link this#monthOfYear} 函数
  2386. */
  2387. month() {
  2388. return this.date.getMonth();
  2389. }
  2390. /**
  2391. * 获取今年的第几个月份
  2392. * 和 {@link this#month} 不同的是不再从 0 计算月份
  2393. */
  2394. monthOfYear() {
  2395. return this.date.getMonth() + 1;
  2396. }
  2397. /**
  2398. * 获取一年内的第多少天
  2399. * 注: 这个天数指定的在第几天而非过去了多少天,例如 2018-01-10 的结果会是 10
  2400. * @returns
  2401. */
  2402. dayOfYear() {
  2403. return Math.ceil((this.date.getTime() - dateConstants.yearStart(this.date).getTime()) /
  2404. DAY_UNIT_TIME);
  2405. }
  2406. /**
  2407. * 获取一个月内的第多少天
  2408. * 注: 这个天数指的是在第几天而非过去了多少天,例如 2018-01-10 的结果会是 10
  2409. * @returns
  2410. */
  2411. dayOfMonth() {
  2412. return this.date.getDate();
  2413. }
  2414. /**
  2415. * 获取一个星期内的第多少天
  2416. * @returns
  2417. */
  2418. dayOfWeek() {
  2419. return this.date.getDay();
  2420. }
  2421. /**
  2422. * 获取一年内的第多少星期
  2423. * 注: 这个星期指定的在第几天而非过去了多少天,例如 2018-01-10 的结果会是 10
  2424. * @returns
  2425. */
  2426. weekOfYear() {
  2427. return Math.ceil(this.dayOfYear() / 7);
  2428. }
  2429. /**
  2430. * 获取一个月内的第多少星期
  2431. * @returns
  2432. */
  2433. weekOfMonth() {
  2434. return Math.ceil(this.dayOfMonth() / 7);
  2435. }
  2436. /**
  2437. * 获取季度
  2438. * @returns
  2439. */
  2440. quarter() {
  2441. const month = this.month();
  2442. if (isRange(month, 0, 3)) {
  2443. return 1;
  2444. }
  2445. else if (isRange(month, 3, 6)) {
  2446. return 2;
  2447. }
  2448. else if (isRange(month, 6, 9)) {
  2449. return 3;
  2450. }
  2451. else {
  2452. return 4;
  2453. }
  2454. }
  2455. /**
  2456. * 获取小时
  2457. * @returns
  2458. */
  2459. hour() {
  2460. return this.date.getHours();
  2461. }
  2462. /**
  2463. * 获取分钟
  2464. * @returns
  2465. */
  2466. minute() {
  2467. return this.date.getMinutes();
  2468. }
  2469. /**
  2470. * 获取秒
  2471. * @returns
  2472. */
  2473. second() {
  2474. return this.date.getSeconds();
  2475. }
  2476. /**
  2477. * 获取毫秒
  2478. * @returns
  2479. */
  2480. milliSecond() {
  2481. return this.date.getMilliseconds();
  2482. }
  2483. }
  2484. /**
  2485. * 获取一个增强的日期
  2486. * @param date 要增强的日期
  2487. * @returns 增强日期
  2488. */
  2489. function dateEnhance(date) {
  2490. return new DateEnhance(date);
  2491. }
  2492.  
  2493. /**
  2494. * 获取一年内的第多少星期
  2495. * @param date 日期
  2496. * @returns 这个日期第多少个星期
  2497. * @deprecated 不推荐使用,请使用 {@see dateEnhance} 代替
  2498. */
  2499. function getYearWeek(date) {
  2500. return dateEnhance(date).weekOfYear();
  2501. }
  2502.  
  2503. /**
  2504. * 时间日期间隔
  2505. */
  2506. class DateBetween {
  2507. /**
  2508. * 构造函数
  2509. * @param start 开始时间
  2510. * @param end 结束时间
  2511. */
  2512. constructor(start, end) {
  2513. this.start = start;
  2514. this.end = end;
  2515. }
  2516. /**
  2517. * 获取毫秒差值
  2518. * @returns 毫秒差值
  2519. */
  2520. milliSecond() {
  2521. return this.end.getTime() - this.start.getTime();
  2522. }
  2523. /**
  2524. * 获取秒差值
  2525. * @returns 秒差值
  2526. */
  2527. second() {
  2528. return Math.floor(this.milliSecond() / 1000);
  2529. }
  2530. /**
  2531. * 获取分钟差值
  2532. * @returns 分钟差值
  2533. */
  2534. minute() {
  2535. return Math.floor(this.second() / 60);
  2536. }
  2537. /**
  2538. * 获取小时差值
  2539. * @returns 小时差值
  2540. */
  2541. hour() {
  2542. return Math.floor(this.minute() / 60);
  2543. }
  2544. /**
  2545. * 获取天数差值
  2546. * @returns 天数差值
  2547. */
  2548. day() {
  2549. return Math.floor(this.hour() / 24);
  2550. }
  2551. /**
  2552. * 获取月份差值
  2553. * 注: 此处获取的差值是按月计算的,即 2018-12-31 => 2019-01-01 也被认为相差一个月
  2554. * @returns 月份差值
  2555. */
  2556. month() {
  2557. const year = this.year();
  2558. const month = this.end.getMonth() - this.start.getMonth();
  2559. return year * 12 + month;
  2560. }
  2561. /**
  2562. * 获取年份差值
  2563. * 注: 此处获取的差值是按年计算的,即 2018-12-31 => 2019-01-01 也被认为相差一年
  2564. * @returns 年份差值
  2565. */
  2566. year() {
  2567. return this.end.getFullYear() - this.start.getFullYear();
  2568. }
  2569. }
  2570. /**
  2571. * 获取两个时间的差值
  2572. * @param start 开始时间
  2573. * @param end 结束时间
  2574. * @returns 差值对象
  2575. */
  2576. function dateBetween(start, end) {
  2577. return new DateBetween(start, end);
  2578. }
  2579.  
  2580. /**
  2581. * 返回合理参数本身的函数
  2582. * 1. 如果没有参数则返回 undefined
  2583. * 2. 如果只有一个参数则返回参数本身
  2584. * 3. 如果有两个以上的参数则返回参数列表
  2585. * @param args 任何对象
  2586. * @returns 传入的参数
  2587. */
  2588. function returnReasonableItself(...args) {
  2589. const len = args.length;
  2590. if (len === 0) {
  2591. return null;
  2592. }
  2593. if (len === 1) {
  2594. return args[0];
  2595. }
  2596. return args;
  2597. }
  2598.  
  2599. /**
  2600. * 从数组中移除指定的元素
  2601. * 注: 时间复杂度为 1~3On
  2602. * @param arr 需要被过滤的数组
  2603. * @param deleteItems 要过滤的元素数组
  2604. * @param k 每个元素的唯一键函数
  2605. */
  2606. function filterItems(arr, deleteItems, k = returnItself) {
  2607. const kFn = getKFn(k);
  2608. const kSet = new Set(deleteItems.map(kFn));
  2609. return arr.filter((v, i, arr) => !kSet.has(kFn(v, i, arr)));
  2610. }
  2611.  
  2612. /**
  2613. * 比较两个数组的差异
  2614. * @param left 第一个数组
  2615. * @param right 第二个数组
  2616. * @param k 每个元素的唯一标识产生函数
  2617. * @returns 比较的差异结果
  2618. */
  2619. function diffBy(left, right, k = returnItself) {
  2620. const kFn = getKFn(k);
  2621. // 首先得到两个 kSet 集合用于过滤
  2622. const kThanSet = new Set(left.map(kFn));
  2623. const kThatSet = new Set(right.map(kFn));
  2624. const leftUnique = left.filter((v, ...args) => !kThatSet.has(kFn(v, ...args)));
  2625. const rightUnique = right.filter((v, ...args) => !kThanSet.has(kFn(v, ...args)));
  2626. const kLeftSet = new Set(leftUnique.map(kFn));
  2627. const common = left.filter((v, ...args) => !kLeftSet.has(kFn(v, ...args)));
  2628. return { left: leftUnique, right: rightUnique, common };
  2629. }
  2630.  
  2631. /**
  2632. * 比较两个数组的差异
  2633. * @deprecated 已废弃,请使用更简洁的 {@link diffBy}
  2634. */
  2635. const arrayDiffBy = diffBy;
  2636.  
  2637. /**
  2638. * 使用 Generator 实现一个从 0 开始的无限自增序列
  2639. */
  2640. function* autoIncrementGenerator() {
  2641. for (let i = 0;; i++) {
  2642. /**
  2643. * @returns 每次获取都返回循环中的当前迭代变量,然后暂停于此处
  2644. */
  2645. yield i;
  2646. }
  2647. }
  2648. /**
  2649. * 生成器对象
  2650. */
  2651. const generator = autoIncrementGenerator();
  2652. /**
  2653. * 获取自增长序列的最新值
  2654. * @returns 最新值
  2655. */
  2656. function autoIncrement() {
  2657. return generator.next().value;
  2658. }
  2659.  
  2660. /**
  2661. * 转换接口
  2662. * @interface
  2663. */
  2664. class IConverter {
  2665. /**
  2666. * 将字符串解析为字符串列表
  2667. *
  2668. * @param str 字符串
  2669. * @return {Array.<String>} 字符串列表
  2670. * @abstract
  2671. */
  2672. from(str) {
  2673. throw new Error('子类必须重写 from 函数');
  2674. }
  2675. /**
  2676. * 将字符串列表构造为字符串
  2677. *
  2678. * @param list 字符串列表
  2679. * @return {String} 字符串
  2680. * @abstract
  2681. */
  2682. to(list) {
  2683. throw new Error('子类必须重写 to 函数');
  2684. }
  2685. }
  2686.  
  2687. /**
  2688. * 驼峰风格解析
  2689. */
  2690. class CamelOrPascalFrom extends IConverter {
  2691. /**
  2692. * 将字符串解析为字符串列表
  2693. *
  2694. * @param str 字符串
  2695. * @return {Array.<String>} 字符串列表
  2696. * @override
  2697. */
  2698. from(str) {
  2699. const result = [];
  2700. const len = str.length;
  2701. let old = 0;
  2702. for (let i = 0; i < len; i++) {
  2703. const c = str.charAt(i);
  2704. if (c >= 'A' && c <= 'Z') {
  2705. if (i !== 0) {
  2706. result.push(str.substring(old, i));
  2707. }
  2708. old = i;
  2709. }
  2710. }
  2711. if (old !== str.length) {
  2712. result.push(str.substring(old, str.length));
  2713. }
  2714. return result;
  2715. }
  2716. }
  2717.  
  2718. /**
  2719. * 小写开头的驼峰转换器
  2720. *
  2721. */
  2722. class CamelConverter extends CamelOrPascalFrom {
  2723. /**
  2724. * 将字符串列表构造为字符串
  2725. *
  2726. * @param list 字符串列表
  2727. * @return {String} 字符串
  2728. * @override
  2729. */
  2730. to(list) {
  2731. return list.reduce((res, s, i) => {
  2732. const str = toLowerCase(s);
  2733. return (res +=
  2734. (i === 0 ? toLowerCase : toUpperCase)(str.substring(0, 1)) +
  2735. str.substring(1));
  2736. }, '');
  2737. }
  2738. }
  2739.  
  2740. /**
  2741. * 大写开头的驼峰转换器
  2742. */
  2743. class PascalConverter extends CamelOrPascalFrom {
  2744. /**
  2745. * 将字符串列表构造为字符串
  2746. *
  2747. * @param list 字符串列表
  2748. * @return {String} 字符串
  2749. * @override
  2750. */
  2751. to(list) {
  2752. return list.reduce((res, s) => {
  2753. const str = toLowerCase(s);
  2754. return (res += toUpperCase(str.substring(0, 1)) + str.substring(1));
  2755. }, '');
  2756. }
  2757. }
  2758.  
  2759. /**
  2760. * 下划线风格解析
  2761. */
  2762. class SnakeOrScreamingSnakeFrom extends IConverter {
  2763. /**
  2764. * 将字符串解析为字符串列表
  2765. *
  2766. * @param str 字符串
  2767. * @return {Array.<String>} 字符串列表
  2768. * @override
  2769. */
  2770. from(str) {
  2771. return str.split('_');
  2772. }
  2773. }
  2774.  
  2775. /**
  2776. * 小写下划线的转换器
  2777. */
  2778. class SnakeConverter extends SnakeOrScreamingSnakeFrom {
  2779. /**
  2780. * 将字符串列表构造为字符串
  2781. *
  2782. * @param list 字符串列表
  2783. * @return {String} 字符串
  2784. * @override
  2785. */
  2786. to(list) {
  2787. return list.map(toLowerCase).join('_');
  2788. }
  2789. }
  2790.  
  2791. /**
  2792. * 大写下划线的转换器
  2793. */
  2794. class ScreamingSnakeConverter extends SnakeOrScreamingSnakeFrom {
  2795. /**
  2796. * 将字符串列表构造为字符串
  2797. *
  2798. * @param list 字符串列表
  2799. * @return {String} 字符串
  2800. * @override
  2801. */
  2802. to(list) {
  2803. return list.map(toUpperCase).join('_');
  2804. }
  2805. }
  2806.  
  2807. /**
  2808. * @enum {Symbol} 字符串风格常量对象
  2809. */
  2810. (function (StringStyleType) {
  2811. /**
  2812. * 小写驼峰
  2813. */
  2814. StringStyleType[StringStyleType["Camel"] = 1] = "Camel";
  2815. /**
  2816. * 大写驼峰
  2817. */
  2818. StringStyleType[StringStyleType["Pascal"] = 2] = "Pascal";
  2819. /**
  2820. * 小写下划线
  2821. */
  2822. StringStyleType[StringStyleType["Snake"] = 3] = "Snake";
  2823. /**
  2824. * 大写下划线
  2825. */
  2826. StringStyleType[StringStyleType["ScreamingSnake"] = 4] = "ScreamingSnake";
  2827. })(exports.StringStyleType || (exports.StringStyleType = {}));
  2828.  
  2829. /**
  2830. * 转换器工厂
  2831. */
  2832. class ConverterFactory {
  2833. /**
  2834. * 获取一个转换器实例
  2835. *
  2836. * @param styleType 转换风格,使用了 {@link stringStyleType} 定义的常量对象
  2837. * @return {IConverter} 转换器对象
  2838. * @throws 如果获取未定义过的转换器,则会抛出异常
  2839. */
  2840. static getInstance(styleType) {
  2841. switch (styleType) {
  2842. case exports.StringStyleType.Camel:
  2843. return new CamelConverter();
  2844. case exports.StringStyleType.Pascal:
  2845. return new PascalConverter();
  2846. case exports.StringStyleType.Snake:
  2847. return new SnakeConverter();
  2848. case exports.StringStyleType.ScreamingSnake:
  2849. return new ScreamingSnakeConverter();
  2850. default:
  2851. throw new Error('No corresponding converter found');
  2852. }
  2853. }
  2854. }
  2855.  
  2856. /**
  2857. * 字符串风格转换器
  2858. * 请不要直接使用构造函数创建,而是用 {@link StringStyleUtil.getConverter} 来获得一个转换器
  2859. * @private
  2860. */
  2861. class StringStyleConverter {
  2862. /**
  2863. * 构造一个字符串任意风格转换器
  2864. * @param from 转换字符串的风格
  2865. * @param to 需要转换的风格
  2866. * @private
  2867. */
  2868. constructor(from, to) {
  2869. /**
  2870. * @field 解析字符串风格的转换器
  2871. * @type {IConverter}
  2872. * @private
  2873. */
  2874. this.fromConverter = ConverterFactory.getInstance(from);
  2875. /**
  2876. * @field 构造字符串风格的转换器
  2877. * @type {IConverter}
  2878. * @private
  2879. */
  2880. this.toConverter = ConverterFactory.getInstance(to);
  2881. }
  2882. /**
  2883. * 转换字符串的风格
  2884. *
  2885. * @param str 要转换的字符串
  2886. * @return {String} 转换得到的字符串
  2887. */
  2888. convert(str) {
  2889. if (stringValidator.isEmpty(str)) {
  2890. return str;
  2891. }
  2892. return this.toConverter.to(this.fromConverter.from(str));
  2893. }
  2894. }
  2895.  
  2896. /**
  2897. * 包装一个函数为指定参数只执行一次的函数
  2898. * @param fn 需要包装的函数
  2899. * @param identity 参数转换的函数,参数为需要包装函数的参数
  2900. * @returns 需要被包装的函数
  2901. */
  2902. function onceOfSameParam(fn, identity = (args) => `onceOfSameParam-${fn.toString()}-${JSON.stringify(args)}`) {
  2903. const cacheMap = new Map();
  2904. const res = new Proxy(fn, {
  2905. apply(_, _this, args) {
  2906. const key = identity(args);
  2907. const old = cacheMap.get(key);
  2908. if (old !== undefined) {
  2909. return old;
  2910. }
  2911. const res = Reflect.apply(_, _this, args);
  2912. return compatibleAsync(res, res => {
  2913. cacheMap.set(key, res);
  2914. return res;
  2915. });
  2916. },
  2917. });
  2918. return Object.assign(res, {
  2919. origin: fn,
  2920. clear(...keys) {
  2921. if (keys.length) {
  2922. cacheMap.clear();
  2923. }
  2924. else {
  2925. keys.forEach(key => cacheMap.delete(key));
  2926. }
  2927. },
  2928. });
  2929. }
  2930.  
  2931. /**
  2932. * 包装获取字符串风格转换器
  2933. * 此处采用了单例模式,每种转换器只会有一个
  2934. *
  2935. * @param from 解析风格
  2936. * @param to 转换风格
  2937. * @return {StringStyleConverter} 转换器的实例
  2938. */
  2939. const _getConverter = onceOfSameParam(
  2940. /**
  2941. * @param from 解析风格
  2942. * @param to 转换风格
  2943. * @return {StringStyleConverter} 转换器的实例
  2944. */
  2945. (from, to) => new StringStyleConverter(from, to));
  2946. /**
  2947. * 字符串风格转换工具类
  2948. */
  2949. class StringStyleUtil {
  2950. /**
  2951. * 获取一个转换器的实例
  2952. * 该函数获取的转换器可以任意复用,请优先使用函数
  2953. * @param from 解析风格
  2954. * @param to 转换风格
  2955. * @return {StringStyleConverter} 转换器的实例
  2956. */
  2957. static getConverter(from, to) {
  2958. return _getConverter(from, to);
  2959. }
  2960. /**
  2961. * 直接转换字符串的风格
  2962. * 请优先使用可以复用的 {@link StringStyleUtil.getConverter} 函数
  2963. * @param from 解析风格
  2964. * @param to 转换风格
  2965. * @param str 要转换的字符串
  2966. * @return {String} 转换得到的字符串
  2967. */
  2968. static convert(from, to, str) {
  2969. return StringStyleUtil.getConverter(from, to).convert(str);
  2970. }
  2971. }
  2972.  
  2973. /**
  2974. * 递归使对象不可变
  2975. * @param obj 任何非空对象
  2976. * @returns 新的不可变对象
  2977. */
  2978. function deepFreeze(obj) {
  2979. const freeze = (v) => {
  2980. if (TypeValidator.isObject(v)) {
  2981. deepFreeze(v);
  2982. }
  2983. };
  2984. // 数组和对象分别处理
  2985. if (TypeValidator.isArray(obj)) {
  2986. obj.forEach(freeze);
  2987. }
  2988. else if (TypeValidator.isObject(obj)) {
  2989. Object.keys(obj)
  2990. .map(k => Reflect.get(obj, k))
  2991. .forEach(freeze);
  2992. }
  2993. return Object.freeze(obj);
  2994. }
  2995.  
  2996. /**
  2997. * 包装对象,使其成为可以任意深度调用而不会出现 undefined 调用的问题
  2998. * 注意: 该函数不能进行递归调用({@link JSON.stringfy}),一定会造成堆栈溢出的问题(RangeError: Maximum call stack size exceeded)
  2999. * @param obj 任意一个 Object 对象
  3000. * @param [defaultValue] 默认值,默认为 {}
  3001. * @returns 包装后的对象
  3002. */
  3003. function deepProxy(obj, defaultValue = {}) {
  3004. const handler = {
  3005. get(target, k) {
  3006. let v = Reflect.get(target, k);
  3007. if (isNullOrUndefined(v)) {
  3008. v = defaultValue;
  3009. }
  3010. if (typeof v !== 'object') {
  3011. return v;
  3012. }
  3013. return new Proxy(v, handler);
  3014. },
  3015. };
  3016. return new Proxy(obj, handler);
  3017. }
  3018.  
  3019. /**
  3020. * 将函数包装为柯里化函数
  3021. * 注: 该函数模仿了 Lodash 的 curry 函数
  3022. * @param fn 需要包装的函数
  3023. * @param {...any} args 应用的部分参数
  3024. * @returns 包装后的函数
  3025. * @deprecated 由于之前的理解错误,该函数在下个大版本将会被废弃,请使用命名更合适的 {@link partial}
  3026. */
  3027. function curry(fn, ...args) {
  3028. const realArgs = args.filter(arg => arg !== curry._);
  3029. // 如果函数参数足够则调用传入的函数
  3030. if (realArgs.length >= fn.length) {
  3031. return fn(...realArgs);
  3032. }
  3033. /**
  3034. * 最终返回的函数
  3035. * @param otherArgs 接受任意参数
  3036. * @returns 返回一个函数,或者函数调用完成返回结果
  3037. */
  3038. function innerFn(...otherArgs) {
  3039. // 记录需要移除补到前面的参数
  3040. const removeIndexSet = new Set();
  3041. let i = 0;
  3042. const newArgs = args.map(arg => {
  3043. if (arg !== curry._ ||
  3044. otherArgs[i] === undefined ||
  3045. otherArgs[i] === curry._) {
  3046. return arg;
  3047. }
  3048. removeIndexSet.add(i);
  3049. // 每次补偿前面的 curry._ 参数计数器 +1
  3050. return otherArgs[i++];
  3051. });
  3052. const newOtherArgs = otherArgs.filter((_v, i) => !removeIndexSet.has(i));
  3053. return curry(fn, ...newArgs, ...newOtherArgs);
  3054. }
  3055. // 定义柯里化函数的剩余参数长度,便于在其他地方进行部分参数应用
  3056. // 注: 不使用 length 属性的原因是 length 属性
  3057. innerFn._length = fn.length - args.filter(arg => arg !== curry._).length;
  3058. // 自定义 toString 函数便于调试
  3059. innerFn.toString = () => `name: ${fn.name}, args: [${args.map(o => o.toString()).join(', ')}]`;
  3060. innerFn._curry = true;
  3061. return innerFn;
  3062. }
  3063. /**
  3064. * 柯里化的占位符,需要应用后面的参数时使用
  3065. * 例如 {@link curry(fn)(curry._, 1)} 意味着函数 fn 的第二个参数将被确定为 1
  3066. */
  3067. curry._ = Symbol('_');
  3068.  
  3069. /**
  3070. * 快速根据指定函数对数组进行排序
  3071. * 注: 使用递归实现,对于超大数组(其实前端的数组不可能特别大吧?#笑)可能造成堆栈溢出
  3072. * @param arr 需要排序的数组
  3073. * @param k 对数组中每个元素都产生可比较的值的函数,默认返回自身进行比较
  3074. * @returns 排序后的新数组
  3075. */
  3076. function sortBy(arr, k = returnItself) {
  3077. const kFn = getKFn(k);
  3078. // 此处为了让 typedoc 能生成文档而不得不加上类型
  3079. const newArr = arr.map((v, i) => [v, i]);
  3080. function _sort(arr, fn) {
  3081. // 边界条件,如果传入数组的值
  3082. if (arr.length <= 1) {
  3083. return arr;
  3084. }
  3085. // 根据中间值对数组分治为两个数组
  3086. const medianIndex = Math.floor(arr.length / 2);
  3087. const medianValue = arr[medianIndex];
  3088. const left = [];
  3089. const right = [];
  3090. for (let i = 0, len = arr.length; i < len; i++) {
  3091. if (i === medianIndex) {
  3092. continue;
  3093. }
  3094. const v = arr[i];
  3095. if (fn(v, medianValue) <= 0) {
  3096. left.push(v);
  3097. }
  3098. else {
  3099. right.push(v);
  3100. }
  3101. }
  3102. return _sort(left, fn)
  3103. .concat([medianValue])
  3104. .concat(_sort(right, fn));
  3105. }
  3106. return _sort(newArr, ([t1, i1], [t2, i2]) => {
  3107. const k1 = kFn(t1, i1, arr);
  3108. const k2 = kFn(t2, i2, arr);
  3109. if (k1 === k2) {
  3110. return 0;
  3111. }
  3112. else if (k1 < k2) {
  3113. return -1;
  3114. }
  3115. else {
  3116. return 1;
  3117. }
  3118. }).map(([_v, i]) => arr[i]);
  3119. }
  3120.  
  3121. /**
  3122. * 日期格式化器
  3123. * 包含格式化为字符串和解析字符串为日期的函数
  3124. */
  3125. class DateFormatter {
  3126. /**
  3127. * 构造函数
  3128. * @param fmt 日期时间格式
  3129. */
  3130. constructor(fmt) {
  3131. this.fmt = fmt;
  3132. }
  3133. /**
  3134. * 格式化
  3135. * @param date 需要格式化的日期
  3136. * @returns 格式化的字符串
  3137. */
  3138. format(date) {
  3139. if (isNullOrUndefined(date)) {
  3140. return '';
  3141. }
  3142. return dateFormat(date, this.fmt);
  3143. }
  3144. /**
  3145. * 解析字符串为日期对象
  3146. * @param str 字符串
  3147. * @returns 解析得到的日期
  3148. */
  3149. parse(str) {
  3150. if (stringValidator.isEmpty(str)) {
  3151. return null;
  3152. }
  3153. return dateParse(str, this.fmt);
  3154. }
  3155. /**
  3156. * 将日期时间字符串转换为前端指定格式的字符串
  3157. * 主要适用场景是前端接收到后端的日期时间一般是一个字符串,然而需要自定义格式的时候还必须先创建 {@link Date} 对象才能格式化,略微繁琐,故使用该函数
  3158. * @param str 字符串
  3159. * @param parseFmt 解析的日期时间格式。默认直接使用 {@link new Date()} 创建
  3160. * @returns 转换后得到的字符串
  3161. */
  3162. strFormat(str, parseFmt) {
  3163. if (stringValidator.isEmpty(str)) {
  3164. return '';
  3165. }
  3166. const date = parseFmt ? dateParse(str, parseFmt) : new Date(str);
  3167. return dateFormat(date, this.fmt);
  3168. }
  3169. }
  3170. /**
  3171. * 日期格式化器
  3172. */
  3173. DateFormatter.dateFormatter = new DateFormatter('yyyy-MM-dd');
  3174. /**
  3175. * 时间格式化器
  3176. */
  3177. DateFormatter.timeFormatter = new DateFormatter('hh:mm:ss');
  3178. /**
  3179. * 日期时间格式化器
  3180. */
  3181. DateFormatter.dateTimeFormatter = new DateFormatter('yyyy-MM-dd hh:mm:ss');
  3182.  
  3183. /**
  3184. * 查询符合条件的元素的下标
  3185. * @param arr 查询的数组
  3186. * @param fn 谓词
  3187. * @param num 查询的第几个符合条件的元素,默认为 1,和默认的 findIndex 行为保持一致
  3188. * @returns 符合条件的元素的下标,如果没有则返回 -1
  3189. */
  3190. function findIndex(arr, fn, num = 1) {
  3191. let k = 0;
  3192. for (let i = 0, len = arr.length; i < len; i++) {
  3193. if (fn.call(arr, arr[i], i, arr) && ++k >= num) {
  3194. return i;
  3195. }
  3196. }
  3197. return -1;
  3198. }
  3199.  
  3200. /**
  3201. * 连接两个函数并自动柯里化
  3202. * 注: 该函数依赖于 length,所以不支持默认参数以及不定参数
  3203. * @param fn1 第一个函数
  3204. * @param fn2 第二个函数
  3205. * @returns 连接后的函数
  3206. */
  3207. const _compose = (fn1, fn2) => {
  3208. return function (...args) {
  3209. const i = findIndex(args, v => v !== curry._, fn1._length || fn1.length);
  3210. const res = curry(fn1, ...args);
  3211. // 如果这个函数的参数不足,则返回它
  3212. if (i === -1) {
  3213. return _compose(res, fn2);
  3214. }
  3215. // 否则将结果以及多余的参数应用到下一个函数上
  3216. return curry(fn2, res, ...args.slice(i + 1));
  3217. };
  3218. };
  3219. /**
  3220. * 将多个函数组合起来
  3221. * 前面函数的返回值将变成后面函数的第一个参数,如果到了最后一个函数执行完成,则直接返回
  3222. * 注: 该函数是自动柯里化,将对所有传入的函数进行柯里化处理
  3223. * 注: 该函数支持一次调用传入全部函数的参数
  3224. * @param fns 多个需要连接函数
  3225. * @returns 连接后的柯里化函数
  3226. * TODO 这里需要进行类型优化
  3227. */
  3228. function compose(...fns) {
  3229. return fns.reduceRight((fn1, fn2) => _compose(fn2, fn1));
  3230. }
  3231.  
  3232. /**
  3233. * 递归排除对象中的指定字段
  3234. * @param obj 需要排除的对象
  3235. * @param {...obj} fields 需要排除的字段
  3236. */
  3237. function deepExcludeFields(obj, ...fields) {
  3238. if (TypeValidator.isArray(obj)) {
  3239. return obj.map(o => deepExcludeFields(o, ...fields));
  3240. }
  3241. else if (TypeValidator.isDate(obj)) {
  3242. return obj;
  3243. }
  3244. else if (TypeValidator.isObject(obj)) {
  3245. const temp = excludeFields(obj, ...fields);
  3246. return Object.keys(temp).reduce((res, k) => {
  3247. const v = Reflect.get(res, k);
  3248. Reflect.set(res, k, deepExcludeFields(v, ...fields));
  3249. return res;
  3250. }, temp);
  3251. }
  3252. else {
  3253. return obj;
  3254. }
  3255. }
  3256.  
  3257. /**
  3258. * 递归排除对象中的指定字段
  3259. * @param obj 需要排除的对象
  3260. * @param {...obj} fields 需要排除的字段
  3261. * @deprecated 已废弃,请使用统一使用 `deep` 开头的 {@link deepExcludeFields} 函数
  3262. */
  3263. function excludeFieldsDeep(obj, ...fields) {
  3264. return deepExcludeFields(obj, ...fields);
  3265. }
  3266.  
  3267. /**
  3268. * 缓存的值
  3269. */
  3270. class CacheVal {
  3271. /**
  3272. * 构造函数
  3273. * @param options 缓存值对象
  3274. * @param options.key 缓存的键原始值
  3275. * @param options.val 缓存的值
  3276. * @param options.cacheOption 缓存的选项
  3277. */
  3278. constructor(options = {}) {
  3279. Object.assign(this, options);
  3280. }
  3281. }
  3282.  
  3283. /**
  3284. * 无限的超时时间
  3285. * TODO 此处暂时使用字符串作为一种折衷方法,因为 Symbol 无法被序列化为 JSON,反向序列化也是不可能的
  3286. */
  3287. const TimeoutInfinite = 'TimeoutInfinite';
  3288.  
  3289. /**
  3290. * 使用 LocalStorage 实现的缓存
  3291. * 1. get: 根据 key 获取
  3292. * 2. set: 根据 key value 设置,会覆盖
  3293. * 3. touch: 获取并刷新超时时间
  3294. * 4. add: 根据 key value 添加,不会覆盖
  3295. * 5. del: 根据 key 删除
  3296. * 6. clearExpired: 清除所有过期的缓存
  3297. */
  3298. class LocalStorageCache {
  3299. /**
  3300. * 构造函数
  3301. * @param cacheOption 全局缓存选项
  3302. */
  3303. constructor({ timeout = TimeoutInfinite, serialize = JSON.stringify, deserialize = JSON.parse, } = {}) {
  3304. // 这里必须强制转换,因为 timeStart 在全局选项中是不可能存在的
  3305. this.cacheOption = {
  3306. timeout,
  3307. serialize,
  3308. deserialize,
  3309. };
  3310. /**
  3311. * 缓存对象,默认使用 localStorage
  3312. */
  3313. this.localStorage = window.localStorage;
  3314. // 创建后将异步清空所有过期的缓存
  3315. this.clearExpired();
  3316. }
  3317. /**
  3318. * 清空所有过期的 key
  3319. * 注: 该函数是异步执行的
  3320. */
  3321. clearExpired() {
  3322. return __awaiter(this, void 0, void 0, function* () {
  3323. const local = this.localStorage;
  3324. const getKeys = () => {
  3325. const len = local.length;
  3326. const res = [];
  3327. for (let i = 0; i < len; i++) {
  3328. res.push(local.key(i));
  3329. }
  3330. return res;
  3331. };
  3332. getKeys()
  3333. .filter(not(isNullOrUndefined))
  3334. .map(key => safeExec(() => JSON.parse(local.getItem(key))))
  3335. .filter(cacheVal => !isNullOrUndefined(cacheVal) &&
  3336. isNullOrUndefined(cacheVal.cacheOption))
  3337. // TODO 这里暂时加个补丁,过滤掉 timeStart,timeout 为 undefined 的缓存
  3338. .filter(({ cacheOption = {} }) => {
  3339. const { timeStart, timeout } = cacheOption;
  3340. if (isNullOrUndefined(timeStart) || isNullOrUndefined(timeout)) {
  3341. return false;
  3342. }
  3343. return timeout !== TimeoutInfinite && Date.now() - timeStart > timeout;
  3344. })
  3345. .forEach(({ key }) => local.removeItem(key));
  3346. });
  3347. }
  3348. /**
  3349. * 根据 key + value 添加
  3350. * 如果不存在则添加,否则忽略
  3351. * @param key 缓存的 key
  3352. * @param val 缓存的 value
  3353. * @param cacheOption 缓存的选项,默认为无限时间
  3354. * @override
  3355. */
  3356. add(key, val, timeout) {
  3357. const result = this.get(key);
  3358. if (result !== null) {
  3359. return;
  3360. }
  3361. this.set(key, val, timeout);
  3362. }
  3363. /**
  3364. * 根据指定的 key 删除
  3365. * 如果存在则删除,否则忽略
  3366. * @param key 删除的 key
  3367. * @override
  3368. */
  3369. del(key) {
  3370. this.localStorage.removeItem(key);
  3371. }
  3372. /**
  3373. * 根据指定的 key 修改
  3374. * 不管是否存在都会设置
  3375. * @param key 修改的 key
  3376. * @param val 修改的 value
  3377. * @param timeout 修改的选项
  3378. * @override
  3379. */
  3380. set(key, val, timeout) {
  3381. this.localStorage.setItem(key, JSON.stringify(new CacheVal({
  3382. key,
  3383. val: this.cacheOption.serialize(val),
  3384. // 我们不需要缓存序列化/反序列化策略(实际上也无法缓存)
  3385. cacheOption: {
  3386. timeStart: Date.now(),
  3387. timeout: timeout || this.cacheOption.timeout,
  3388. },
  3389. })));
  3390. }
  3391. /**
  3392. * 根据 key 获取
  3393. * 如果存在则获取,否则忽略
  3394. * @param key 指定的 key
  3395. * @param timeout 获取的选项
  3396. * @returns 获取到的缓存值
  3397. * @override
  3398. */
  3399. get(key) {
  3400. const str = this.localStorage.getItem(key);
  3401. const cacheVal = safeExec(() => JSON.parse(str));
  3402. if (isNullOrUndefined(cacheVal) ||
  3403. isNullOrUndefined(cacheVal.cacheOption)) {
  3404. return null;
  3405. }
  3406. const [timeStart, timeout, deserialize] = [
  3407. cacheVal.cacheOption.timeStart,
  3408. cacheVal.cacheOption.timeout,
  3409. this.cacheOption.deserialize,
  3410. ];
  3411. // 如果超时则删除并返回 null
  3412. if (timeout !== TimeoutInfinite && Date.now() - timeStart > timeout) {
  3413. this.del(key);
  3414. return null;
  3415. }
  3416. try {
  3417. return deserialize(cacheVal.val);
  3418. }
  3419. catch (e) {
  3420. this.del(key);
  3421. return null;
  3422. }
  3423. }
  3424. /**
  3425. * 根据 key 获取并刷新超时时间
  3426. * @param key 指定的 key
  3427. * @param cacheOption 获取的选项
  3428. * @returns 获取到的缓存值
  3429. * @override
  3430. */
  3431. touch(key) {
  3432. const str = this.localStorage.getItem(key);
  3433. /**
  3434. * @type {CacheVal}
  3435. */
  3436. const cacheVal = safeExec(() => JSON.parse(str));
  3437. if (isNullOrUndefined(cacheVal) ||
  3438. isNullOrUndefined(cacheVal.cacheOption)) {
  3439. return null;
  3440. }
  3441. const [timeStart, timeout, deserialize] = [
  3442. cacheVal.cacheOption.timeStart,
  3443. cacheVal.cacheOption.timeout,
  3444. this.cacheOption.deserialize,
  3445. ];
  3446. // 如果超时则删除并返回 null
  3447. if (timeout !== TimeoutInfinite && Date.now() - timeStart > timeout) {
  3448. this.del(key);
  3449. return null;
  3450. }
  3451. try {
  3452. const result = deserialize(cacheVal.val);
  3453. this.set(key, result, { timeStart: Date.now(), timeout });
  3454. return result;
  3455. }
  3456. catch (e) {
  3457. this.del(key);
  3458. return null;
  3459. }
  3460. }
  3461. }
  3462.  
  3463. /**
  3464. * 默认使用的 {@link ICache} 接口的缓存实现
  3465. */
  3466. const cache = new LocalStorageCache();
  3467. /**
  3468. * 缓存工具类
  3469. * 主要实现缓存高阶函数的封装
  3470. */
  3471. class CacheUtil {
  3472. /**
  3473. * 将指定函数包装为只调用一次为缓存函数
  3474. * @param fn 需要包装的函数
  3475. * @param options 缓存选项对象。可选项
  3476. * @param options.identity 缓存标识。默认为函数 {@link toString},但有时候不太可行(继承自基类的函数)
  3477. * @param options.timeout 缓存时间。默认为无限
  3478. * @returns 包装后的函数
  3479. */
  3480. static once(fn, { identity = fn.toString(), timeout } = {}) {
  3481. const generateKey = () => `CacheUtil.onceOfSameParam-${identity}`;
  3482. const innerFn = new Proxy(fn, {
  3483. apply(_, _this, args) {
  3484. const key = generateKey();
  3485. const val = cache.get(key);
  3486. if (val !== null) {
  3487. return val;
  3488. }
  3489. return compatibleAsync(Reflect.apply(_, _this, args), res => {
  3490. cache.set(key, res, timeout);
  3491. return res;
  3492. });
  3493. },
  3494. });
  3495. return Object.assign(innerFn, {
  3496. origin: fn,
  3497. clear() {
  3498. cache.del(generateKey());
  3499. },
  3500. });
  3501. }
  3502. /**
  3503. * 包裹函数为缓存函数
  3504. * @param fn 一个接受一些参数并返回结果的函数
  3505. * @param options 缓存选项对象。可选项
  3506. * @param options.identity 缓存标识。默认为函数 {@link toString},但有时候不太可行(继承自基类的函数)
  3507. * @param options.timeout 缓存时间。默认为无限
  3508. * @returns 带有缓存功能的函数
  3509. */
  3510. static onceOfSameParam(fn, { identity = fn.toString(), timeout } = {}) {
  3511. const generateKey = (args) => `CacheUtil.onceOfSameParam-${identity}-${JSON.stringify(args)}`;
  3512. const innerFn = new Proxy(fn, {
  3513. apply(_, _this, args) {
  3514. const key = generateKey(args);
  3515. const val = cache.get(key);
  3516. if (val !== null) {
  3517. return val;
  3518. }
  3519. return compatibleAsync(Reflect.apply(_, _this, args), res => {
  3520. cache.set(key, res, timeout);
  3521. return res;
  3522. });
  3523. },
  3524. });
  3525. return Object.assign(innerFn, {
  3526. origin: fn,
  3527. clear(...args) {
  3528. cache.del(generateKey(args));
  3529. },
  3530. });
  3531. }
  3532. }
  3533. /**
  3534. * 导出一个默认的缓存工具对象
  3535. * @deprecated 已废弃,请直接使用类的静态函数
  3536. */
  3537. const cacheUtil = CacheUtil;
  3538.  
  3539. /**
  3540. * 一个空的函数
  3541. * @param args 接受任何参数
  3542. */
  3543. function emptyFunc(...args) { }
  3544.  
  3545. /**
  3546. * 禁止他人调试网站相关方法的集合对象
  3547. */
  3548. class AntiDebug {
  3549. /**
  3550. * 不停循环 debugger 防止有人调试代码
  3551. * @returns 取消函数
  3552. */
  3553. static cyclingDebugger() {
  3554. const res = setInterval(() => {
  3555. debugger;
  3556. }, 100);
  3557. return () => clearInterval(res);
  3558. }
  3559. /**
  3560. * 检查是否正在 debugger 并调用回调函数
  3561. * @param fn 回调函数,默认为重载页面
  3562. * @returns 取消函数
  3563. */
  3564. static checkDebug(fn = () => window.location.reload()) {
  3565. const res = setInterval(() => {
  3566. const diff = timing(() => {
  3567. for (let i = 0; i < 1000; i++) {
  3568. console.log(i);
  3569. console.clear();
  3570. }
  3571. });
  3572. if (diff > 500) {
  3573. console.log(diff);
  3574. fn();
  3575. }
  3576. }, 1000);
  3577. return () => clearInterval(res);
  3578. }
  3579. /**
  3580. * 禁用控制台调试输出
  3581. * @returns 取消函数
  3582. */
  3583. static disableConsoleOutput() {
  3584. if (!window.console) {
  3585. return emptyFunc;
  3586. }
  3587. const map = arrayToMap(Object.keys(console), returnItself, k => {
  3588. // @ts-ignore
  3589. const temp = console[k];
  3590. // @ts-ignore
  3591. console[k] = emptyFunc;
  3592. return temp;
  3593. });
  3594. return () => {
  3595. for (const [k, v] of map) {
  3596. // @ts-ignore
  3597. console[k] = v;
  3598. }
  3599. };
  3600. }
  3601. }
  3602. /**
  3603. * 禁止他人调试网站相关方法的集合对象
  3604. * @deprecated 已废弃,请直接使用类的静态函数
  3605. */
  3606. const antiDebug = AntiDebug;
  3607.  
  3608. /**
  3609. * 判断一个字符串是否为空白的字符串
  3610. * @param str 字符串
  3611. * @returns 是否为空字符串
  3612. * @deprecated 已废弃,请使用 {@link stringValidator#isBlank}
  3613. */
  3614. function isBlank(str) {
  3615. return stringValidator.isBlank(str);
  3616. }
  3617.  
  3618. /**
  3619. * 判断一个字符串是否为空字符串
  3620. * @param str 字符串
  3621. * @returns 是否为空字符串
  3622. * @deprecated 已废弃,请使用 {@link stringValidator#isEmpty}
  3623. */
  3624. function isEmpty(str) {
  3625. return stringValidator.isEmpty(str);
  3626. }
  3627.  
  3628. /**
  3629. * 加载一个远程脚本文件
  3630. * @param src 远程脚本路径
  3631. * @returns 等待异步加载脚本完成
  3632. */
  3633. function loadScript(src) {
  3634. return new Promise((resolve, reject) => {
  3635. const script = document.createElement('script');
  3636. script.src = src;
  3637. script.addEventListener('load', () => resolve());
  3638. script.addEventListener('error', reject);
  3639. document.body.appendChild(script);
  3640. });
  3641. }
  3642.  
  3643. /**
  3644. * 将一个谓词函数取反
  3645. * 如果是同步函数,则返回的函数也是同步的,否则返回的是取反后的异步函数
  3646. * @param fn 要取反的函数
  3647. * @returns 取反得到的函数
  3648. * @deprecated 已废弃,请使用 {@link CombinedPredicate.not} 进行为此取反
  3649. */
  3650. function deny(fn) {
  3651. return CombinedPredicate.not(fn);
  3652. }
  3653.  
  3654. /**
  3655. * 数组校验器
  3656. */
  3657. class ArrayValidator {
  3658. /**
  3659. * 是否为空数组
  3660. * @param array 空数组
  3661. * @returns 是否为空数组
  3662. */
  3663. static isEmpty(array) {
  3664. return (isNullOrUndefined(array) ||
  3665. !(array instanceof Array) ||
  3666. array.length === 0);
  3667. }
  3668. }
  3669. /**
  3670. * 导出一个默认的数组校验对象
  3671. * @deprecated 已废弃,请直接使用类的静态函数
  3672. */
  3673. const arrayValidator = ArrayValidator;
  3674.  
  3675. /**
  3676. * 路径工具
  3677. */
  3678. class PathUtil {
  3679. /**
  3680. * 拼接多个路径
  3681. *
  3682. * @param paths 路径数组
  3683. * @return {String} 拼接完成的路径
  3684. */
  3685. static join(...paths) {
  3686. return paths.reduce(PathUtil._join);
  3687. }
  3688. /**
  3689. * 拼接两个路径
  3690. *
  3691. * @param pathStart 开始路径
  3692. * @param pathEnd 结束路径
  3693. * @return {String} 拼接完成的两个路径
  3694. */
  3695. static _join(pathStart, pathEnd) {
  3696. if (pathStart.endsWith(PathUtil.Separator)) {
  3697. return (pathStart + pathEnd).replace(PathUtil.Separator + PathUtil.Separator, PathUtil.Separator);
  3698. }
  3699. if (pathEnd.startsWith(PathUtil.Separator)) {
  3700. return pathStart + pathEnd;
  3701. }
  3702. return pathStart + PathUtil.Separator + pathEnd;
  3703. }
  3704. }
  3705. /**
  3706. * 路径分隔符
  3707. */
  3708. PathUtil.Separator = '/';
  3709. /**
  3710. * 导出一个路径工具类
  3711. * @deprecated 已废弃,请直接使用类的静态函数
  3712. */
  3713. const pathUtil = PathUtil;
  3714.  
  3715. /**
  3716. * 自定义的日志类
  3717. * 主要便于在开发环境下正常显示调试信息,在生产环境则默认关闭它
  3718. */
  3719. class Logger {
  3720. /**
  3721. * 构造函数
  3722. * @param options 可选项
  3723. * @param options.enable 是否开启日志
  3724. */
  3725. constructor({ enable = true } = {}) {
  3726. this.debug = console.debug;
  3727. this.error = console.error;
  3728. this.info = console.info;
  3729. this.log = console.log;
  3730. this.warn = console.warn;
  3731. this.dir = console.dir;
  3732. this.dirxml = console.dirxml;
  3733. this.table = console.table;
  3734. this.trace = console.trace;
  3735. this.group = console.group;
  3736. this.groupCollapsed = console.groupCollapsed;
  3737. this.groupEnd = console.groupEnd;
  3738. this.clear = console.clear;
  3739. this.count = console.count;
  3740. this.assert = console.assert;
  3741. this.profile = console.profile;
  3742. this.profileEnd = console.profileEnd;
  3743. this.time = console.time;
  3744. this.timeEnd = console.timeEnd;
  3745. this.timeStamp = console.timeStamp;
  3746. this._enable = enable;
  3747. }
  3748. /**
  3749. * 设置 enable 的 setter 属性,在改变时合并对应的子类对象实现
  3750. * @param enable 是否开启
  3751. */
  3752. set enable(enable) {
  3753. /**
  3754. * @field 是否开启全局控制台,该属性只写
  3755. */
  3756. this._enable = enable;
  3757. Object.keys(console).forEach(
  3758. // @ts-ignore
  3759. k => (this[k] = enable ? console[k] : emptyFunc));
  3760. }
  3761. }
  3762. /**
  3763. * 导出一个全局可用的 Logger 对象
  3764. * 使用 enable 属性控制是否开启日志输出,默认为 true
  3765. */
  3766. const logger = new Logger();
  3767.  
  3768. /**
  3769. * 将 Object 对象 转换为 Map
  3770. * @param obj Object 对象
  3771. * @returns 转换得到的 Map 键值表
  3772. */
  3773. function objectToMap(obj) {
  3774. return Object.keys(obj).reduce((map, k) => map.set(k, Reflect.get(obj, k)), new Map());
  3775. }
  3776.  
  3777. /**
  3778. * 将列表转换为树节点
  3779. * 注: 该函数默认树的根节点只有一个,如果有多个,则返回一个数组
  3780. * @param list 树节点列表
  3781. * @param options 其他选项
  3782. * @returns 树节点,或是树节点列表
  3783. */
  3784. function listToTree(list, { bridge = returnItself, isRoot = node => !node.parentId, } = {}) {
  3785. const arr = [];
  3786. const res = list.reduce((root, _sub) => {
  3787. const sub = bridge(_sub);
  3788. if (isRoot(sub)) {
  3789. root.push(sub);
  3790. return root;
  3791. }
  3792. for (const _parent of list) {
  3793. const parent = bridge(_parent);
  3794. if (sub.parentId === parent.id) {
  3795. parent.child = parent.child || [];
  3796. parent.child.push(sub);
  3797. return root;
  3798. }
  3799. }
  3800. return root;
  3801. }, arr);
  3802. // 根据顶级节点的数量决定如何返回
  3803. const len = res.length;
  3804. if (len === 0)
  3805. return {};
  3806. if (len === 1)
  3807. return res[0];
  3808. return res;
  3809. }
  3810.  
  3811. /**
  3812. * 桥接对象不存在的字段
  3813. * @param map 代理的字段映射 Map
  3814. * @returns 转换一个对象为代理对象
  3815. */
  3816. function bridge(map) {
  3817. /**
  3818. * 为对象添加代理的函数
  3819. * @param obj 任何对象
  3820. * @returns 代理后的对象
  3821. */
  3822. return function (obj) {
  3823. return new Proxy(obj, {
  3824. get(_, k) {
  3825. if (Reflect.has(map, k)) {
  3826. return Reflect.get(_, Reflect.get(map, k));
  3827. }
  3828. return Reflect.get(_, k);
  3829. },
  3830. set(_, k, v) {
  3831. if (Reflect.has(map, k)) {
  3832. Reflect.set(_, Reflect.get(map, k), v);
  3833. return true;
  3834. }
  3835. Reflect.set(_, k, v);
  3836. return true;
  3837. },
  3838. });
  3839. };
  3840. }
  3841.  
  3842. /**
  3843. * 遍历并映射一棵树的每个节点
  3844. * @param root 树节点
  3845. * @param options 其他选项
  3846. * @returns 递归遍历后的树节点
  3847. */
  3848. function treeMapping(root, { before = returnItself, after = returnItself, paramFn = (node, ...args) => [], } = {}) {
  3849. /**
  3850. * 遍历一颗完整的树
  3851. * @param node 要遍历的树节点
  3852. * @param args 每次递归遍历时的参数
  3853. */
  3854. function _treeMapping(node, ...args) {
  3855. // 之前的操作
  3856. const _node = before(node, ...args);
  3857. const _child = _node.child;
  3858. if (!arrayValidator.isEmpty(_child)) {
  3859. _node.child = _child.map(v =>
  3860. // 产生一个参数
  3861. _treeMapping(v, ...paramFn(_node, ...args)));
  3862. }
  3863. // 之后的操作
  3864. return after(_node, ...args);
  3865. }
  3866. return _treeMapping(root);
  3867. }
  3868.  
  3869. /**
  3870. * 将树节点转为树节点列表
  3871. * 这里使用了循环进行遍历,而非传统的递归方式
  3872. * @param root 树节点
  3873. * @param options 其他选项
  3874. * @returns 树节点列表
  3875. */
  3876. function treeToList(root, { calcPath = false, bridge = returnItself, } = {}) {
  3877. const res = [];
  3878. const temp = bridge(root);
  3879. if (calcPath) {
  3880. temp.path = temp.id + '';
  3881. }
  3882. // 利用队列缓存所有未处理的节点
  3883. const queue = [temp];
  3884. // 使用 Set 防止可能的重复引用
  3885. const filterSet = new Set();
  3886. // 使用 lastIdMap 避免重复添加
  3887. const lastIdMap = new Map();
  3888. for (let value; queue.length > 0;) {
  3889. const first = queue.shift();
  3890. value = bridge(first);
  3891. // 判断重复
  3892. if (value === undefined || filterSet.has(first)) {
  3893. continue;
  3894. }
  3895. filterSet.add(first);
  3896. res.push(value);
  3897. const child = value.child;
  3898. if (ArrayValidator.isEmpty(child)) {
  3899. continue;
  3900. }
  3901. const childNonIllegal = child.filter(v => !isNullOrUndefined(v) || filterSet.has(v));
  3902. // TODO 这里和上面的代码明显重复,待优化。。。
  3903. queue.push(...(calcPath
  3904. ? childNonIllegal.map(v => {
  3905. const _v = bridge(v);
  3906. // 如果最后一个的 id 等于自身,说明已经被添加过了
  3907. if (lastIdMap.get(_v.id) === _v.id) {
  3908. return _v;
  3909. }
  3910. // 使用父节点绝对路径 + 当前 id
  3911. _v.path = value.path + ',' + _v.id;
  3912. lastIdMap.set(_v.id, _v.id);
  3913. return _v;
  3914. })
  3915. : childNonIllegal));
  3916. }
  3917. return res;
  3918. }
  3919.  
  3920. /**
  3921. * 树节点桥接工具类
  3922. * 主要实现了桥接 {@field bridge} {@field bridgeTree} 和 {@field bridgeList} 三个函数,事实上桥接之后再转换相当于做了两遍,但就目前而言暂且只能如此了
  3923. */
  3924. class NodeBridgeUtil {
  3925. /**
  3926. * 桥接对象为标准的树结构
  3927. * @param nodeBridge 桥接对象
  3928. * @returns 代理函数
  3929. */
  3930. static bridge({ id = 'id', parentId = 'parentId', child = 'child', path = 'path', } = {}) {
  3931. return bridge({
  3932. id,
  3933. parentId,
  3934. child,
  3935. path,
  3936. });
  3937. }
  3938. /**
  3939. * 桥接一棵完整的树
  3940. * @param tree 树节点
  3941. * @param nodeBridge 桥接对象
  3942. * @returns 代理后的树对象
  3943. */
  3944. static bridgeTree(tree, nodeBridge) {
  3945. return treeMapping(tree, {
  3946. before: this.bridge(nodeBridge),
  3947. });
  3948. }
  3949. /**
  3950. * 桥接一个树节点列表
  3951. * @param list 树节点列表
  3952. * @param nodeBridge 桥接对象
  3953. * @returns 代理后的树节点列表
  3954. */
  3955. static bridgeList(list, nodeBridge) {
  3956. return list.map(this.bridge(nodeBridge));
  3957. }
  3958. }
  3959. /**
  3960. * 导出一个 NodeBridgeUtil 的实例
  3961. * @deprecated 已废弃,请直接使用类的静态函数
  3962. */
  3963. const nodeBridgeUtil = NodeBridgeUtil;
  3964.  
  3965. /**
  3966. * 获取对象中所有的属性及对应的值,包括 ES6 新增的 Symbol 类型的属性
  3967. * @param obj 任何对象
  3968. * @returns 属性及其对应值的二维数组
  3969. * @deprecated 该函数将要被废弃,实质上应用场景很窄
  3970. */
  3971. function getObjectEntries(obj) {
  3972. const mFn = k => [
  3973. k,
  3974. Reflect.get(obj, k),
  3975. ];
  3976. return Reflect.ownKeys(obj).map(mFn);
  3977. }
  3978.  
  3979. /**
  3980. * 获取对象中所有的属性,包括 ES6 新增的 Symbol 类型的属性
  3981. * @param obj 任何对象
  3982. * @returns 属性数组
  3983. * @deprecated 已废弃,请使用 ES6 {@see Reflect.ownKeys} 代替
  3984. * 具体参考 {@url(https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Reflect/ownKeys)}
  3985. */
  3986. function getObjectKeys(obj) {
  3987. if (isNullOrUndefined(obj)) {
  3988. return [];
  3989. }
  3990. return Reflect.ownKeys(obj);
  3991. }
  3992.  
  3993. /**
  3994. * 比较两个浮点数是否相等
  3995. * 具体实现采用差值取绝对值并与 {@link Number.EPSILON} 比较的方式,如果小于浮点数最小差值,则认为它们是 [相等] 的
  3996. * @param num1 第一个浮点数
  3997. * @param num2 第二个浮点数
  3998. * @returns 两数是否相等
  3999. */
  4000. function floatEquals(num1, num2) {
  4001. return Math.abs(num1 - num2) < Number.EPSILON;
  4002. }
  4003.  
  4004. //TODO 暂时绕过类型错误,之后有时间再修
  4005. // export function assign<T, A>(target: T, a: A): T & A
  4006. // export function assign<T, A, B>(target: T, a: A, b: B): T & A & B
  4007. // export function assign<T, A, B, C>(target: T, a: A, b: B, c: C): T & A & B & C
  4008. // export function assign<T, A, B, C, D>(
  4009. // target: T,
  4010. // a: A,
  4011. // b: B,
  4012. // c: C,
  4013. // d: D,
  4014. // ): T & A & B & C & D
  4015. /**
  4016. * 合并多个对象的属性
  4017. * 1. 该合并的方式为浅层合并,只会合并一层的对象
  4018. * 2. 默认忽略值为 undefined/null 的属性
  4019. * @param target 覆盖的对象上
  4020. * @param {...Object} sources 任意数量的对象
  4021. * @returns 合并后的对象
  4022. */
  4023. function assign(target, ...sources) {
  4024. return [target, ...sources].reduce((res, source) => {
  4025. if (isNullOrUndefined(source)) {
  4026. return res;
  4027. }
  4028. return Object.keys(source).reduce((res, k) => {
  4029. const v = Reflect.get(source, k);
  4030. if (isNullOrUndefined(v)) {
  4031. return res;
  4032. }
  4033. Reflect.set(res, k, v);
  4034. return res;
  4035. }, res);
  4036. }, {});
  4037. }
  4038.  
  4039. /**
  4040. * 根据不同的源对象获取不同的正则匹配,代表不需要拷贝的属性
  4041. * @param source 源对象
  4042. * @returns 匹配内部属性的正则表达式
  4043. */
  4044. function getInnerFieldRule(source) {
  4045. if (source instanceof Function) {
  4046. return /^(?:constructor|prototype|arguments|caller|name|bind|call|apply|toString|length)$/;
  4047. }
  4048. else {
  4049. return /^(?:toString|length)$/;
  4050. }
  4051. }
  4052. /**
  4053. * 拷贝对象的属性到目标对象上
  4054. * @param target 目标对象
  4055. * @param source 源对象
  4056. * @returns 返回 {@param target} 目标对象
  4057. */
  4058. function _copyProps(target, source) {
  4059. const innerField = getInnerFieldRule(source);
  4060. Reflect.ownKeys(source).forEach(prop => {
  4061. if (typeof prop === 'string' && innerField.test(prop)) {
  4062. return;
  4063. }
  4064. Reflect.set(target, prop, Reflect.get(source, prop));
  4065. });
  4066. return target;
  4067. }
  4068. /**
  4069. * 混合多个类
  4070. * @param {...Class} mixins 需要混合的多个类及其构造函数参数映射函数的 Map 集合
  4071. * @returns 返回一个混合后的类,构造函数将的参数
  4072. */
  4073. function aggregation(mixins) {
  4074. const arr = Array.from(mixins);
  4075. class Aggregate {
  4076. /**
  4077. * @param args 任意数量的参数
  4078. */
  4079. constructor(...args) {
  4080. arr.forEach(([Mixin, fn = returnItself]) => _copyProps(this, new Mixin(...fn(args))));
  4081. }
  4082. }
  4083. arr.forEach(([Mixin]) => {
  4084. _copyProps(Aggregate.prototype, Mixin.prototype);
  4085. _copyProps(Aggregate, Mixin);
  4086. });
  4087. return Aggregate;
  4088. }
  4089.  
  4090. /**
  4091. * 包装一个异步函数为有限制并发功能的函数
  4092. * @param fn 异步函数
  4093. * @param options 可选参数
  4094. * @param options.limit 并发限制数量,默认为 1
  4095. * @returns 返回被包装后的限制并发功能的函数
  4096. */
  4097. function asyncLimiting(fn, { limit = 1 } = {}) {
  4098. // 当前正在执行异步的数量
  4099. let execCount = 0;
  4100. // waitArr 等待的队列
  4101. const takeQueue = [];
  4102. // 是否增加了 execCount 的标记
  4103. let flag = false;
  4104. return new Proxy(fn, {
  4105. apply(_, _this, args) {
  4106. return __awaiter(this, void 0, void 0, function* () {
  4107. const _takeRun = () => __awaiter(this, void 0, void 0, function* () {
  4108. if (!flag) {
  4109. execCount++;
  4110. flag = false;
  4111. }
  4112. const tempArgs = takeQueue.shift();
  4113. try {
  4114. return yield Reflect.apply(_, _this, tempArgs);
  4115. }
  4116. finally {
  4117. execCount--;
  4118. }
  4119. });
  4120. takeQueue.push(args);
  4121. yield wait(() => {
  4122. const result = execCount < limit;
  4123. // 如果等待结束则必须立刻增加 execCount,避免微任务与宏任务调度可能产生错误
  4124. if (result) {
  4125. flag = true;
  4126. execCount++;
  4127. }
  4128. return result;
  4129. });
  4130. return _takeRun();
  4131. });
  4132. },
  4133. });
  4134. }
  4135.  
  4136. /**
  4137. * 默认的超时时间,可以认为是无限
  4138. */
  4139. const TimeoutInfinity = () => false;
  4140. /**
  4141. * 创建一个 Lock 对象,用于锁住当前的当前的异步流程
  4142. */
  4143. class Locker {
  4144. /**
  4145. * @param options 可选项
  4146. * @param options.limit 限制并发数量,默认为 1
  4147. * @param options.timeout 超时时间,默认为无限
  4148. */
  4149. constructor({ limit = 1, timeout } = {}) {
  4150. this.limit = limit;
  4151. this.timeout = timeout || TimeoutInfinity;
  4152. }
  4153. /**
  4154. * 当前是否锁住了
  4155. * @returns 是否锁住了
  4156. */
  4157. isLocked() {
  4158. return this.limit <= 0;
  4159. }
  4160. /**
  4161. * 添加异步锁
  4162. * @param timeout 超时时间,默认为全局 timeout
  4163. * @returns 进行等待
  4164. */
  4165. lock(timeout = this.timeout) {
  4166. return __awaiter(this, void 0, void 0, function* () {
  4167. if (this.isLocked()) {
  4168. /**
  4169. * @type {Number|Function}
  4170. */
  4171. yield Promise.race([
  4172. wait(() => {
  4173. const result = !this.isLocked();
  4174. if (result) {
  4175. this.limit--;
  4176. }
  4177. return result;
  4178. }),
  4179. wait(timeout),
  4180. ]);
  4181. }
  4182. else {
  4183. this.limit--;
  4184. }
  4185. });
  4186. }
  4187. /**
  4188. * 删除异步锁
  4189. */
  4190. unlock() {
  4191. this.limit++;
  4192. }
  4193. }
  4194.  
  4195. /**
  4196. * 包装一个函数为有错误重试功能的函数
  4197. * 注: 如果发生错误,最终将抛出最后一次调用的异常
  4198. * @param fn 需要被包装的函数
  4199. * @param num 调用的次数。默认为 1
  4200. * @param errorCheck 检查返回结果是否需要重试的函数。默认只要 resolve() 就返回 true
  4201. * @returns 包装后的有错误重试功能的函数
  4202. */
  4203. function trySometime(fn, num = 1, errorCheck = res => true) {
  4204. return new Proxy(fn, {
  4205. apply(target, thisArg, args) {
  4206. return __awaiter(this, void 0, void 0, function* () {
  4207. let err;
  4208. for (let i = 0; i < num; i++) {
  4209. try {
  4210. // 等待结果出来
  4211. const res = yield Reflect.apply(target, thisArg, args);
  4212. // 如果没问题就直接返回了
  4213. if (errorCheck(res)) {
  4214. return res;
  4215. }
  4216. // 否则抛出异常以进行下一次重试
  4217. throw res;
  4218. }
  4219. catch (error) {
  4220. err = error;
  4221. }
  4222. }
  4223. throw err;
  4224. });
  4225. },
  4226. });
  4227. }
  4228.  
  4229. /**
  4230. * 包装一个函数为有错误重试功能的函数
  4231. * 注意: 该函数是并行运行,所以一旦调用,就会同时调用 n 次,不管之前有没有失败。。。此函数不适合包装有副作用的操作,例如修改用户信息,请使用 {@link trySometime} 替代
  4232. * @param fn 需要被包装的函数
  4233. * @param num 调用的次数。默认为 1
  4234. * @param errorCheck 检查返回结果是否需要重试的函数。默认只要 resolve() 就返回 true
  4235. * @returns 包装后的有错误重试功能的函数
  4236. */
  4237. function trySometimeParallel(fn, num = 1, errorCheck = res => true) {
  4238. return new Proxy(fn, {
  4239. apply(target, thisArg, args) {
  4240. return __awaiter(this, void 0, void 0, function* () {
  4241. return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () {
  4242. let err;
  4243. try {
  4244. yield Promise.all(range(0, num).map(() => __awaiter(this, void 0, void 0, function* () {
  4245. try {
  4246. const res = yield Reflect.apply(target, thisArg, args);
  4247. if (errorCheck(res) === true) {
  4248. resolve(res);
  4249. }
  4250. throw res;
  4251. }
  4252. catch (error) {
  4253. err = error;
  4254. }
  4255. })));
  4256. }
  4257. catch (error) {
  4258. console.log(error);
  4259. }
  4260. reject(err);
  4261. }));
  4262. });
  4263. },
  4264. });
  4265. }
  4266.  
  4267. /**
  4268. * 深度比较两个对象是否相等
  4269. * @param x 任何对象
  4270. * @param y 任何对象
  4271. * @returns 是否相等
  4272. */
  4273. function compare(x, y) {
  4274. if ((typeof x === 'number' || x instanceof Number) &&
  4275. (typeof y === 'number' || y instanceof Number)) {
  4276. const _x = +x;
  4277. const _y = +y;
  4278. // 如果都是 NaN 则直接返回 true
  4279. if (isNaN(_x) && isNaN(_y)) {
  4280. return true;
  4281. }
  4282. // 如果是 -0/+0 则返回 false
  4283. if (_x === _y) {
  4284. return 1 / _x === 1 / _y;
  4285. }
  4286. // 如果均为数字且两数之差的绝对值小于浮点数的最小精度(此举主要是为了避免浮点数的精度丢失)
  4287. if (Math.abs(_x - _y) < Number.EPSILON) {
  4288. return true;
  4289. }
  4290. }
  4291. // 如果恒等表达式成立则直接返回 true
  4292. if (x === y) {
  4293. return true;
  4294. }
  4295. // 比较正则和字符串
  4296. if ((x instanceof RegExp && y instanceof RegExp) ||
  4297. ((typeof x === 'string' || x instanceof String) &&
  4298. (typeof y === 'string' || y instanceof String))) {
  4299. return x.toString() === y.toString();
  4300. }
  4301. // 如果都是时间则比较它们的时间戳
  4302. if (x instanceof Date && y instanceof Date) {
  4303. return x.getTime() === y.getTime();
  4304. }
  4305. // 如果两者有一个不是 Object 类型的话则直接返回 false
  4306. // 注: 此处直接返回 false 是因为特殊原生类型的都在上面处理过了
  4307. // 注: Array 可以按照 Object 的逻辑进行处理
  4308. if (!(x instanceof Object && y instanceof Object)) {
  4309. return false;
  4310. }
  4311. // 比较它们的原型
  4312. if (x.prototype !== y.prototype) {
  4313. return false;
  4314. }
  4315. // 比较构造函数
  4316. if (x.constructor !== y.constructor) {
  4317. return false;
  4318. }
  4319. // 比较 y 中的属性是否全部都在 x 中
  4320. for (const p of Object.keys(y)) {
  4321. if (!Reflect.has(x, p)) {
  4322. return false;
  4323. }
  4324. }
  4325. // 比较 x 中的属性是否全部都在 y 中
  4326. for (const p of Object.keys(x)) {
  4327. if (!Reflect.has(y, p)) {
  4328. return false;
  4329. }
  4330. // 比较每个元素的类型,如果不同则直接返回 false
  4331. if (typeof y[p] !== typeof x[p]) {
  4332. return false;
  4333. }
  4334. // 递归比较每个元素
  4335. if (!compare(x[p], y[p])) {
  4336. return false;
  4337. }
  4338. }
  4339. // 全部比较完成仍然没有结果就返回 true
  4340. return true;
  4341. }
  4342.  
  4343. /**
  4344. * 阻塞主线程指定时间
  4345. * 注: 和 {@see wait} 不同,该函数会真的阻塞住主线程,即这段时间内其他的代码都无法执行,例如用户的点击事件!
  4346. * @param time 阻塞毫秒数
  4347. */
  4348. function sleep(time) {
  4349. const end = performance.now() + time;
  4350. while (performance.now() <= end) { }
  4351. }
  4352.  
  4353. /**
  4354. * 包装一个函数为异步函数
  4355. * 如果是一个异步函数,则直接返回,否则返回一部函数
  4356. * @param fn 任意一个函数
  4357. * @returns 返回的异步结果 Promise 对象
  4358. * @typeparam R 原函数函数返回值类型
  4359. */
  4360. function async(fn) {
  4361. return new Proxy(fn, {
  4362. apply(_, _this, args) {
  4363. return __awaiter(this, void 0, void 0, function* () {
  4364. return yield Reflect.apply(_, _this, args);
  4365. });
  4366. },
  4367. });
  4368. }
  4369.  
  4370. /**
  4371. * 将一个异步函数包装为具有时序的异步函数
  4372. * 注: 该函数会按照调用顺序依次返回结果,后面的调用的结果需要等待前面的,所以如果不关心过时的结果,请使用 {@link switchMap} 函数
  4373. * @param fn 一个普通的异步函数
  4374. * @returns 包装后的函数
  4375. */
  4376. function mergeMap(fn) {
  4377. // 当前执行的异步操作 id
  4378. let id = 0;
  4379. // 所执行的异步操作 id 列表
  4380. const ids = new Set();
  4381. return new Proxy(fn, {
  4382. apply(_, _this, args) {
  4383. return __awaiter(this, void 0, void 0, function* () {
  4384. const prom = Reflect.apply(_, _this, args);
  4385. const temp = id;
  4386. ids.add(temp);
  4387. id++;
  4388. yield wait(() => !ids.has(temp - 1));
  4389. ids.delete(temp);
  4390. return yield prom;
  4391. });
  4392. },
  4393. });
  4394. }
  4395.  
  4396. /**
  4397. * 将一个异步函数包装为具有时序的异步函数
  4398. * 注: 该函数会丢弃过期的异步操作结果,这样的话性能会稍稍提高(主要是响应比较快的结果会立刻生效而不必等待前面的响应结果)
  4399. * @param fn 一个普通的异步函数
  4400. * @returns 包装后的函数
  4401. */
  4402. function switchMap(fn) {
  4403. // 当前执行的异步操作 id
  4404. let id = 0;
  4405. // 最后一次异步操作的 id,小于这个的操作结果会被丢弃
  4406. let last = 0;
  4407. // 缓存最后一次异步操作的结果
  4408. let cache;
  4409. return new Proxy(fn, {
  4410. apply(_, _this, args) {
  4411. return __awaiter(this, void 0, void 0, function* () {
  4412. const temp = id;
  4413. id++;
  4414. const res = yield Reflect.apply(_, _this, args);
  4415. if (temp < last) {
  4416. return cache;
  4417. }
  4418. cache = res;
  4419. last = temp;
  4420. return res;
  4421. });
  4422. },
  4423. });
  4424. }
  4425.  
  4426. /**
  4427. * 将指定函数包装为只调用一次,其他的调用返回旧值
  4428. * 主要适用场景是只允许调用一次的地方,例如 Tab 的初始化
  4429. * * 示意图:
  4430. * a => b => c => d => e =>
  4431. * a ==|====|====|====|====>
  4432. * |b |c |d |e (die)
  4433. *
  4434. * @param fn 需要包装的函数
  4435. * @returns 包装后的函数
  4436. */
  4437. function once(fn) {
  4438. let flag = true;
  4439. let cache;
  4440. const res = new Proxy(fn, {
  4441. apply(target, thisArg, args) {
  4442. if (!flag) {
  4443. return cache;
  4444. }
  4445. flag = false;
  4446. // 如果是异步函数则返回异步的结果
  4447. return compatibleAsync(Reflect.apply(target, thisArg, args), res => {
  4448. cache = res;
  4449. return cache;
  4450. });
  4451. },
  4452. });
  4453. return Object.assign(res, {
  4454. origin: fn,
  4455. clear() {
  4456. cache = null;
  4457. },
  4458. });
  4459. }
  4460.  
  4461. /**
  4462. * 将一个异步函数包装为具有时序的异步函数
  4463. * 注: 该函数会按照调用顺序依次返回结果,后面的执行的调用(不是调用结果)需要等待前面的,此函数适用于异步函数的内里执行也必须保证顺序时使用,否则请使用 {@link mergeMap} 函数
  4464. * 注: 该函数其实相当于调用 {@code asyncLimiting(fn, {limit: 1})} 函数
  4465. * 例如即时保存文档到服务器,当然要等待上一次的请求结束才能请求下一次,不然数据库保存的数据就存在谬误了
  4466. * @param fn 一个普通的异步函数
  4467. * @returns 包装后的函数
  4468. */
  4469. function concatMap(fn) {
  4470. // 当前执行的异步操作 id
  4471. let id = 0;
  4472. // 所执行的异步操作 id 列表
  4473. const ids = new Set();
  4474. return new Proxy(fn, {
  4475. apply(_, _this, args) {
  4476. return __awaiter(this, void 0, void 0, function* () {
  4477. const temp = id;
  4478. ids.add(temp);
  4479. id++;
  4480. yield wait(() => !ids.has(temp - 1));
  4481. const prom = Reflect.apply(_, _this, args);
  4482. ids.delete(temp);
  4483. return yield prom;
  4484. });
  4485. },
  4486. });
  4487. }
  4488.  
  4489. /**
  4490. * 重复执行指定的函数
  4491. * @param num 重复的次数
  4492. * @param fn 执行的函数,如果是异步函数,则返回 Array.<Promise>
  4493. * @param {...Object} args 参数
  4494. * @returns 执行返回结果
  4495. */
  4496. function repeatedCall(num, fn, ...args) {
  4497. return range(0, num).map(() => fn(...args));
  4498. }
  4499.  
  4500. /**
  4501. * 发布订阅模式
  4502. * @typeparam T 订阅主题的类型,虽然可以为 any,但这里是刻意进行限制以避免 “全局” 的发布订阅中心对象
  4503. */
  4504. class PubSubMachine {
  4505. constructor() {
  4506. /**
  4507. * 订阅者集合
  4508. */
  4509. this.subMap = new Map();
  4510. }
  4511. /**
  4512. * 发布一个主题
  4513. * @param topic 发布的主题
  4514. * @param [args] 主题订阅所需要的参数
  4515. */
  4516. pub(topic, ...args) {
  4517. const fns = this.subMap.get(topic);
  4518. if (fns === undefined) {
  4519. return;
  4520. }
  4521. fns.forEach(fn => fn(...args));
  4522. }
  4523. /**
  4524. * 订阅一个主题
  4525. * @param topic 订阅的主题
  4526. * @param fn 回调的函数
  4527. */
  4528. sub(topic, fn) {
  4529. if (!this.subMap.has(topic)) {
  4530. this.subMap.set(topic, []);
  4531. }
  4532. this.subMap.get(topic).push(fn);
  4533. }
  4534. /**
  4535. * 取消订阅
  4536. * @param topic 订阅的主题
  4537. * @param fn 订阅的函数,没有则删除这个主题下所有的函数
  4538. */
  4539. unsub(topic, fn) {
  4540. if (fn === undefined) {
  4541. this.subMap.delete(topic);
  4542. return;
  4543. }
  4544. if (!this.subMap.has(topic)) {
  4545. return;
  4546. }
  4547. const fns = this.subMap.get(topic);
  4548. fns.splice(fns.indexOf(fn), 1);
  4549. }
  4550. }
  4551.  
  4552. /**
  4553. * 提取对象数组为 Map
  4554. * @param arr 对象数组
  4555. * @param fields 提取的字段
  4556. * @returns 提取字段名对应其字段值数组的 Map
  4557. * @typeparam T 数组元素的类型,必须为可提取字段的对象
  4558. */
  4559. function extractFieldMap(arr, fields) {
  4560. return arr.reduce((res, v) => {
  4561. for (const [k, _arr] of res) {
  4562. _arr.push(Reflect.get(v, k));
  4563. }
  4564. return res;
  4565. }, arrayToMap(fields, k => k, () => new Array()));
  4566. }
  4567.  
  4568. /**
  4569. * 数组按照指定长度进行分段为二维数组
  4570. * 注: num 必须要大于 1
  4571. * @param arr 要进行分段的数组
  4572. * @param num 每段的长度
  4573. * @returns 分段后的二维数组
  4574. */
  4575. function segmentation(arr, num) {
  4576. return arr.reduce((res, v, i) => {
  4577. const index = (i + 1) % num;
  4578. if (index === 1) {
  4579. res.push([]);
  4580. }
  4581. res[res.length - 1].push(v);
  4582. return res;
  4583. }, new Array());
  4584. }
  4585.  
  4586. /**
  4587. * 切换 DOM 元素的 class
  4588. * @param {Element} el DOM 元素
  4589. * @param {Object} obj 切换的状态/class 键值对象
  4590. * @return 根据状态切换 class 的函数
  4591. */
  4592. function toggleClass(el, obj) {
  4593. const arr = Array.from(Object.values(obj));
  4594. /**
  4595. * 返回切换 class 的函数
  4596. * @param state 切换的状态
  4597. */
  4598. return function toggle(state) {
  4599. arr.forEach(v => el.classList.remove(v));
  4600. el.classList.add(obj[state]);
  4601. };
  4602. }
  4603.  
  4604. /**
  4605. * 将函数包装为偏函数
  4606. * 注: 该函数模仿了 underscore 的 partial 函数
  4607. * @param fn 需要包装的函数
  4608. * @param {...any} args 应用的部分参数
  4609. * @returns 包装后的函数
  4610. */
  4611. function partial(fn, ...args) {
  4612. const realArgs = args.filter(arg => arg !== partial._);
  4613. // 如果函数参数足够则调用传入的函数
  4614. if (realArgs.length >= fn.length) {
  4615. return fn(...realArgs);
  4616. }
  4617. /**
  4618. * 最终返回的函数
  4619. * @param otherArgs 接受任意参数
  4620. * @returns 返回一个函数,或者函数调用完成返回结果
  4621. */
  4622. function innerFn(...otherArgs) {
  4623. // 记录需要移除补到前面的参数
  4624. const removeIndexSet = new Set();
  4625. let i = 0;
  4626. const newArgs = args.map(arg => {
  4627. if (arg !== partial._ ||
  4628. otherArgs[i] === undefined ||
  4629. otherArgs[i] === partial._) {
  4630. return arg;
  4631. }
  4632. removeIndexSet.add(i);
  4633. // 每次补偿前面的 partial._ 参数计数器 +1
  4634. return otherArgs[i++];
  4635. });
  4636. const newOtherArgs = otherArgs.filter((_v, i) => !removeIndexSet.has(i));
  4637. return partial(fn, ...newArgs, ...newOtherArgs);
  4638. }
  4639. // 定义偏函数的剩余参数长度,便于在其他地方进行部分参数应用
  4640. // 注: 不使用 length 属性的原因是 length 属性
  4641. innerFn._length = fn.length - args.filter(arg => arg !== partial._).length;
  4642. // 自定义 toString 函数便于调试
  4643. innerFn.toString = () => `name: ${fn.name}, args: [${args.map(o => o.toString()).join(', ')}]`;
  4644. innerFn._partial = true;
  4645. return innerFn;
  4646. }
  4647. /**
  4648. * 偏的占位符,需要应用后面的参数时使用
  4649. * 例如 {@link partial(fn)(partial._, 1)} 意味着函数 fn 的第二个参数将被确定为 1
  4650. */
  4651. partial._ = Symbol('_');
  4652.  
  4653. /**
  4654. * 事件工具类
  4655. */
  4656. class EventUtil {
  4657. static add(dom, type, listener, options) {
  4658. if (!EventUtil.listenerMap.has(dom)) {
  4659. EventUtil.listenerMap.set(dom, []);
  4660. }
  4661. EventUtil.listenerMap.get(dom).push({
  4662. type,
  4663. listener,
  4664. options,
  4665. });
  4666. dom.addEventListener(type, listener, options);
  4667. }
  4668. static remove(dom, type, listener, options) {
  4669. dom.removeEventListener(type, listener, options);
  4670. EventUtil.listenerMap.set(dom, (EventUtil.listenerMap.get(dom) || []).filter(cacheListener => cacheListener.type !== type ||
  4671. cacheListener.listener !== listener ||
  4672. cacheListener.options !== options));
  4673. }
  4674. static removeByType(dom, type, options) {
  4675. const listenerList = EventUtil.listenerMap.get(dom);
  4676. if (listenerList === undefined) {
  4677. return;
  4678. }
  4679. const map = groupBy(listenerList, cacheListener => type === cacheListener.type && options === cacheListener.options);
  4680. const removeCacheListenerList = map.get(true) || [];
  4681. const retainCacheListenerList = map.get(true) || [];
  4682. EventUtil.listenerMap.set(dom, retainCacheListenerList);
  4683. removeCacheListenerList.forEach(cacheListener => dom.removeEventListener(cacheListener.type, cacheListener.listener, cacheListener.options));
  4684. }
  4685. }
  4686. /**
  4687. * 缓存的事件监听对象映射表
  4688. */
  4689. EventUtil.listenerMap = new Map();
  4690.  
  4691. exports.AntiDebug = AntiDebug;
  4692. exports.ArrayValidator = ArrayValidator;
  4693. exports.AsyncArray = AsyncArray;
  4694. exports.CacheUtil = CacheUtil;
  4695. exports.CombinedPredicate = CombinedPredicate;
  4696. exports.DateConstants = DateConstants;
  4697. exports.DateFormatter = DateFormatter;
  4698. exports.EventUtil = EventUtil;
  4699. exports.FetchLimiting = FetchLimiting;
  4700. exports.LocalStorageCache = LocalStorageCache;
  4701. exports.Locker = Locker;
  4702. exports.Logger = Logger;
  4703. exports.NodeBridgeUtil = NodeBridgeUtil;
  4704. exports.PathUtil = PathUtil;
  4705. exports.PubSubMachine = PubSubMachine;
  4706. exports.StateMachine = StateMachine;
  4707. exports.StringStyleUtil = StringStyleUtil;
  4708. exports.StringValidator = StringValidator;
  4709. exports.TypeValidator = TypeValidator;
  4710. exports.aggregation = aggregation;
  4711. exports.and = and;
  4712. exports.antiDebug = antiDebug;
  4713. exports.appends = appends;
  4714. exports.arrayDiffBy = arrayDiffBy;
  4715. exports.arrayToMap = arrayToMap;
  4716. exports.arrayValidator = arrayValidator;
  4717. exports.asIterator = asIterator;
  4718. exports.assign = assign;
  4719. exports.async = async;
  4720. exports.asyncFlatMap = asyncFlatMap;
  4721. exports.asyncLimiting = asyncLimiting;
  4722. exports.autoIncrement = autoIncrement;
  4723. exports.blankToNull = blankToNull;
  4724. exports.blankToNullField = blankToNullField;
  4725. exports.bridge = bridge;
  4726. exports.cacheUtil = cacheUtil;
  4727. exports.compare = compare;
  4728. exports.compatibleAsync = compatibleAsync;
  4729. exports.compose = compose;
  4730. exports.concatMap = concatMap;
  4731. exports.copyText = copyText;
  4732. exports.createElByString = createElByString;
  4733. exports.curry = curry;
  4734. exports.dateBetween = dateBetween;
  4735. exports.dateConstants = dateConstants;
  4736. exports.dateEnhance = dateEnhance;
  4737. exports.dateFormat = dateFormat;
  4738. exports.dateParse = dateParse;
  4739. exports.debounce = debounce;
  4740. exports.deepExcludeFields = deepExcludeFields;
  4741. exports.deepFreeze = deepFreeze;
  4742. exports.deepProxy = deepProxy;
  4743. exports.deletes = deletes;
  4744. exports.deny = deny;
  4745. exports.diffBy = diffBy;
  4746. exports.download = download;
  4747. exports.downloadString = downloadString;
  4748. exports.downloadUrl = downloadUrl;
  4749. exports.emptyAllField = emptyAllField;
  4750. exports.emptyFunc = emptyFunc;
  4751. exports.excludeFields = excludeFields;
  4752. exports.excludeFieldsDeep = excludeFieldsDeep;
  4753. exports.extractFieldMap = extractFieldMap;
  4754. exports.fetchTimeout = fetchTimeout;
  4755. exports.fill = fill;
  4756. exports.filterItems = filterItems;
  4757. exports.findIndex = findIndex;
  4758. exports.flatMap = flatMap;
  4759. exports.floatEquals = floatEquals;
  4760. exports.formDataToArray = formDataToArray;
  4761. exports.format = format;
  4762. exports.getCookies = getCookies;
  4763. exports.getCursorPosition = getCursorPosition;
  4764. exports.getCusorPostion = getCusorPostion;
  4765. exports.getObjectEntries = getObjectEntries;
  4766. exports.getObjectKeys = getObjectKeys;
  4767. exports.getYearWeek = getYearWeek;
  4768. exports.groupBy = groupBy;
  4769. exports.insertText = insertText;
  4770. exports.isBlank = isBlank;
  4771. exports.isEditable = isEditable;
  4772. exports.isEmpty = isEmpty;
  4773. exports.isFloat = isFloat;
  4774. exports.isNullOrUndefined = isNullOrUndefined;
  4775. exports.isNumber = isNumber;
  4776. exports.isRange = isRange;
  4777. exports.lastFocus = lastFocus;
  4778. exports.listToTree = listToTree;
  4779. exports.loadResource = loadResource;
  4780. exports.loadScript = loadScript;
  4781. exports.logger = logger;
  4782. exports.mapToObject = mapToObject;
  4783. exports.mergeMap = mergeMap;
  4784. exports.nodeBridgeUtil = nodeBridgeUtil;
  4785. exports.not = not;
  4786. exports.objToFormData = objToFormData;
  4787. exports.objectToMap = objectToMap;
  4788. exports.once = once;
  4789. exports.onceOfSameParam = onceOfSameParam;
  4790. exports.or = or;
  4791. exports.parseUrl = parseUrl;
  4792. exports.partial = partial;
  4793. exports.pathUtil = pathUtil;
  4794. exports.randomInt = randomInt;
  4795. exports.range = range;
  4796. exports.readLocal = readLocal;
  4797. exports.removeEl = removeEl;
  4798. exports.removeText = removeText;
  4799. exports.repeatedCall = repeatedCall;
  4800. exports.returnItself = returnItself;
  4801. exports.returnReasonableItself = returnReasonableItself;
  4802. exports.safeExec = safeExec;
  4803. exports.segmentation = segmentation;
  4804. exports.setCursorPosition = setCursorPosition;
  4805. exports.setCusorPostion = setCusorPostion;
  4806. exports.sets = sets;
  4807. exports.singleModel = singleModel;
  4808. exports.sleep = sleep;
  4809. exports.sortBy = sortBy;
  4810. exports.spliceParams = spliceParams;
  4811. exports.strToArrayBuffer = strToArrayBuffer;
  4812. exports.strToDate = strToDate;
  4813. exports.stringValidator = stringValidator;
  4814. exports.switchMap = switchMap;
  4815. exports.throttle = throttle;
  4816. exports.timing = timing;
  4817. exports.toLowerCase = toLowerCase;
  4818. exports.toObject = toObject;
  4819. exports.toString = toString$1;
  4820. exports.toUpperCase = toUpperCase;
  4821. exports.toggleClass = toggleClass;
  4822. exports.treeMapping = treeMapping;
  4823. exports.treeToList = treeToList;
  4824. exports.trySometime = trySometime;
  4825. exports.trySometimeParallel = trySometimeParallel;
  4826. exports.uniqueBy = uniqueBy;
  4827. exports.wait = wait;
  4828. exports.waitResource = waitResource;
  4829. exports.watch = watch;
  4830. exports.watchEventListener = watchEventListener;
  4831. exports.watchObject = watchObject;
  4832.  
  4833. Object.defineProperty(exports, '__esModule', { value: true });
  4834.  
  4835. }));
  4836. //# sourceMappingURL=rx-util.js.map