util.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. import { INFORMATION_ICON, SCHEME_ANDRIOD, SCHEME_IOS } from "@/common/config";
  2. import * as FingerprintJS from "@fingerprintjs/fingerprintjs";
  3. import CallApp from "callapp-lib";
  4. import CryptoJS from "crypto-js";
  5. // 验证密码等级
  6. export const checkPassLevel = (pass) => {
  7. let level = 0;
  8. if (pass.length < 8 || pass.length > 12) return false;
  9. if (pass.match(/([a-z])+/)) level++;
  10. if (pass.match(/([A-Z])+/)) level++;
  11. if (pass.match(/([0-9])+/)) level++;
  12. return level >= 2;
  13. };
  14. // 验证支付密码
  15. export const checkPayPass = (pass) => pass.length === 6 && /^\d+$/.test(pass);
  16. /**
  17. * 参数校验器
  18. *
  19. * @param {*} data
  20. * @param {*} validatorList
  21. */
  22. export const validatorFun = (data, validatorList = []) => {
  23. // 默认校验器列表
  24. const validatorDefault = {
  25. // 是否不为空
  26. ["notNull"]: (val) =>
  27. !(typeof val == "undefined" || val == null || val === ""),
  28. // 是否是手机号
  29. ["isMobile"]: (val) => /^1[3456789]\d{9}$/.test(val),
  30. ["checkPassLevel"]: checkPassLevel,
  31. ["checkPayPass"]: checkPayPass,
  32. };
  33. const errList = [];
  34. validatorList.forEach(([key, ...list]) => {
  35. // 字段值
  36. const val = data[key];
  37. list.some(([validator, errMsg]) => {
  38. const validatorType = typeof validator;
  39. // 如果校验器类型为string,查找默认验证器,默认验证器不存在则不校验
  40. const stringResult =
  41. validatorType === "string" &&
  42. !(validatorDefault[validator]
  43. ? validatorDefault[validator](val)
  44. : true);
  45. // 如果校验器类型为function,调用该方法
  46. const funResult = validatorType === "function" && !validator(val, data);
  47. // 判断校验结果,true = 不通过,false = 通过,不通过则收集错误信息
  48. if (stringResult || funResult) {
  49. errList.push({
  50. key,
  51. val,
  52. errMsg,
  53. validator,
  54. });
  55. }
  56. return stringResult || funResult;
  57. });
  58. });
  59. return errList;
  60. };
  61. /**
  62. * 数字转中文数字
  63. * @param {Object} num
  64. */
  65. export function numToChinese(num) {
  66. if (!/^\d*(\.\d*)?$/.test(num)) {
  67. alert("Number is wrong!");
  68. return "Number is wrong!";
  69. }
  70. var AA = new Array(
  71. "零",
  72. "一",
  73. "二",
  74. "三",
  75. "四",
  76. "五",
  77. "六",
  78. "七",
  79. "八",
  80. "九"
  81. );
  82. var BB = new Array("", "十", "百", "千", "万", "亿", "点", "");
  83. var a = ("" + num).replace(/(^0*)/g, "").split("."),
  84. k = 0,
  85. re = "";
  86. for (var i = a[0].length - 1; i >= 0; i--) {
  87. switch (k) {
  88. case 0:
  89. re = BB[7] + re;
  90. break;
  91. case 4:
  92. if (!new RegExp("0{4}\\d{" + (a[0].length - i - 1) + "}$").test(a[0]))
  93. re = BB[4] + re;
  94. break;
  95. case 8:
  96. re = BB[5] + re;
  97. BB[7] = BB[5];
  98. k = 0;
  99. break;
  100. }
  101. if (k % 4 == 2 && a[0].charAt(i + 2) != 0 && a[0].charAt(i + 1) == 0)
  102. re = AA[0] + re;
  103. if (a[0].charAt(i) != 0) re = AA[a[0].charAt(i)] + BB[k % 4] + re;
  104. k++;
  105. }
  106. if (a.length > 1) {
  107. //加上小数部分(如果有小数部分)
  108. re += BB[6];
  109. for (var i = 0; i < a[1].length; i++) re += AA[a[1].charAt(i)];
  110. }
  111. return re;
  112. }
  113. /**
  114. * 监听滚动到底部
  115. * @param {() => {}} callback 回调方法
  116. * @param {number} bottom 触发距离,当前默认:350px
  117. */
  118. let lstbOption = {
  119. callback: () => {},
  120. isListenerScrollToBottom: false,
  121. lastScrollTop: 0,
  122. isCallBack: false,
  123. bottom: 100,
  124. listener: false,
  125. };
  126. export const listenerScrollToBottom = (callbackTemp, bottomTemp = 250) => {
  127. lstbOption.callback = callbackTemp;
  128. lstbOption.bottom = bottomTemp;
  129. lstbOption.isListenerScrollToBottom = true;
  130. window.addEventListener("scroll", function () {
  131. const docEl = document.documentElement;
  132. const scrollTop = docEl.scrollTop || document.body.scrollTop;
  133. const scrollHeight = docEl.scrollHeight;
  134. const clientHeight = docEl.clientHeight;
  135. const newScrollTop = scrollTop + clientHeight;
  136. // 判断条件 --> 正向滚动 且 达到触发距离
  137. const flatCallbackTrue =
  138. newScrollTop > lstbOption.lastScrollTop &&
  139. newScrollTop >= scrollHeight - lstbOption.bottom;
  140. // 判断条件 --> 逆向滚动 且 达到触发距离加50px的距离
  141. const flatCallbackFalse1 =
  142. newScrollTop < lstbOption.lastScrollTop &&
  143. newScrollTop < scrollHeight - (lstbOption.bottom + 50);
  144. // 判断条件 --> 正向滚动 且 没有达到触发距离
  145. const flatCallbackFalse2 =
  146. newScrollTop > lstbOption.lastScrollTop &&
  147. newScrollTop < scrollHeight - lstbOption.bottom;
  148. // console.log('监听滚动到底部:', lastScrollTop, newScrollTop, scrollHeight, scrollHeight - bottom);
  149. if (flatCallbackTrue) {
  150. // 判断达到触发距离后是否触发过,触发过则不执行回调
  151. if (!lstbOption.isCallBack) {
  152. lstbOption.isCallBack = true;
  153. lstbOption.callback && lstbOption.callback();
  154. }
  155. } else if (flatCallbackFalse1 || flatCallbackFalse2) {
  156. lstbOption.isCallBack = false;
  157. }
  158. // 更新最后一次滚动
  159. lstbOption.lastScrollTop = newScrollTop;
  160. });
  161. };
  162. /**
  163. * 清除滚动条监听后执行的方法
  164. */
  165. export const removeListenerScrollToBottom = () => {
  166. lstbOption = {
  167. callback: () => {},
  168. isListenerScrollToBottom: false,
  169. lastScrollTop: 0,
  170. isCallBack: false,
  171. bottom: 100,
  172. listener: false,
  173. };
  174. };
  175. /**
  176. * 判断是否是微信中打开
  177. * @returns {boolean}
  178. */
  179. export const isWeChatOpen = () => {
  180. const ua = navigator.userAgent.toLowerCase();
  181. return ua.indexOf("micromessenger") > -1;
  182. };
  183. /**
  184. * 复制到粘贴板
  185. * @param text
  186. * @param callback
  187. */
  188. export const copy = (text, callback) => {
  189. // 禁止复制
  190. document.oncopy = function () {
  191. return true;
  192. };
  193. const tag = document.createElement("textarea");
  194. tag.setAttribute("id", "cp_hgz_input");
  195. tag.value = text;
  196. document.getElementsByTagName("body")[0].appendChild(tag);
  197. document.getElementById("cp_hgz_input").select();
  198. document.execCommand("copy");
  199. document.getElementById("cp_hgz_input").remove();
  200. if (callback) {
  201. callback(text);
  202. }
  203. // 禁止复制
  204. document.oncopy = function () {
  205. return false;
  206. };
  207. };
  208. /**
  209. * 获取文件类型
  210. * @param filename
  211. * @returns {string}
  212. */
  213. export const getFileType = (filename) =>
  214. filename.substring(filename.lastIndexOf(".") + 1).toLocaleLowerCase();
  215. /**
  216. * 获取文件图标
  217. * @param fileType
  218. * @returns {*}
  219. */
  220. export const getFileIcon = (fileType) => INFORMATION_ICON[fileType];
  221. /**
  222. * 格式化富文本,防止图片过大出现横向滚动条
  223. * @param str
  224. * @returns {*}
  225. */
  226. export const formatRichText = (str) => {
  227. return str.replace(/<[^>]+>/g, "");
  228. };
  229. //获取设备唯一标识
  230. export const getUniqueCode = () => {
  231. return FingerprintJS.load().then((fp) => {
  232. return fp.get().then((result) => result.visitorId);
  233. });
  234. };
  235. // 设备类型 1 安卓手机设备 2 ios手机设备 3 安卓ipad 4 iosipad 5 pc电脑设备号
  236. export const getFacilityType = () => {
  237. var ua = navigator.userAgent, //获取浏览器UA
  238. isWindowsPhone = /(?:Windows Phone)/.test(ua),
  239. isSymbian = /(?:SymbianOS)/.test(ua) || isWindowsPhone,
  240. isAndroid = /(?:Android)/.test(ua) && /(?:Mobile)/.test(ua),
  241. isFireFox = /(?:Firefox)/.test(ua),
  242. isChrome = /(?:Chrome|CriOS)/.test(ua),
  243. isIosTablet = /(?:iPad|PlayBook)/.test(ua),
  244. isAndroidTablet = /(?:Android)/.test(ua) && !/(?:Mobile)/.test(ua),
  245. isTablet =
  246. isIosTablet || isAndroidTablet || (isFireFox && /(?:Tablet)/.test(ua)),
  247. isiPhone = /(?:iPhone)/.test(ua) && !isTablet,
  248. isPc = !isiPhone && !isAndroid && !isSymbian && !isTablet;
  249. if (isPc) return 5; //判断是否PC
  250. else if (isiPhone) return 2; //判断是否iPhone
  251. else if (isAndroid) return 1; //判断是否Android
  252. else if (isIosTablet) return 4; //判断是否ios平板
  253. else if (isAndroidTablet) return 3; //判断是否Android平板
  254. };
  255. // 是否是Safari浏览器
  256. export const isSafari = () =>
  257. /Safari/.test(navigator.userAgent) && !/Chrome/.test(navigator.userAgent);
  258. // 判断是否是苹果设备 /iphone|ipod|ipad|macintosh/i
  259. export const isAppleMobileDevice = () =>
  260. isSafari() || /iphone|ipod|ipad/i.test(navigator.userAgent.toLowerCase());
  261. // 打开APP初始化
  262. const callLib = new CallApp({
  263. scheme: {
  264. protocol: isAppleMobileDevice() ? SCHEME_IOS : SCHEME_ANDRIOD,
  265. },
  266. });
  267. export const openApp = (path) => {
  268. setTimeout(() => loading.clear(), 5000);
  269. callLib.open({
  270. path: "",
  271. callback: () => {
  272. loading.clear();
  273. if (isAppleMobileDevice()) {
  274. location.href = "https://gaoyixia.hdlkeji.com/apidoc/?appKey=V1.0";
  275. } else {
  276. location.href = `myinfo://gyx.net:8888${path}`;
  277. }
  278. },
  279. });
  280. };
  281. /**
  282. * 设置是否可以复制
  283. * @param isNoCopy
  284. */
  285. export const setCopyFlag = (isNoCopy = false) => {
  286. // 禁止右键菜单
  287. document.oncontextmenu = function () {
  288. return isNoCopy;
  289. };
  290. // // 禁止文字选择
  291. // document.onselectstart = function () {
  292. // return isNoCopy;
  293. // };
  294. // 禁止复制
  295. // document.oncopy = function () {
  296. // return isNoCopy;
  297. // };
  298. // // 禁止剪切
  299. // document.oncut = function () {
  300. // return isNoCopy;
  301. // };
  302. };
  303. // 获取富文本的纯文字内容
  304. export const getPlainText = (richCont) => {
  305. const str = richCont;
  306. let value = richCont;
  307. if (richCont) {
  308. // 方法一:
  309. value = value.replace(/\s*/g, ""); //去掉空格
  310. value = value.replace(/<[^>]+>/g, ""); //去掉所有的html标记
  311. value = value.replace(/↵/g, ""); //去掉所有的↵符号
  312. value = value.replace(/[\r\n]/g, ""); //去掉回车换行
  313. value = value.replace(/&nbsp;/g, ""); //去掉空格
  314. value = convertIdeogramToNormalCharacter(value);
  315. return value;
  316. // 方法二:
  317. // value = value.replace(/(\n)/g, "");
  318. // value = value.replace(/(\t)/g, "");
  319. // value = value.replace(/(\r)/g, "");
  320. // value = value.replace(/<\/?[^>]*>/g, "");
  321. // value = value.replace(/\s*/g, "");
  322. // value = convertIdeogramToNormalCharacter(value);
  323. // return value;
  324. } else {
  325. return null;
  326. }
  327. };
  328. /**
  329. * TripleDES - 3DES解密
  330. * @param base64Str
  331. * @param key
  332. * @returns {string}
  333. * @constructor
  334. */
  335. export const TripleDESDecrypt = (base64Str, key = "123456") => {
  336. const keyHex = CryptoJS.enc.Utf8.parse(key);
  337. const decryted = CryptoJS.TripleDES.decrypt(
  338. {
  339. ciphertext: CryptoJS.enc.Base64.parse(base64Str),
  340. },
  341. keyHex,
  342. {
  343. mode: CryptoJS.mode.ECB,
  344. padding: CryptoJS.pad.Pkcs7,
  345. // 偏移量
  346. // iv: CryptoJS.enc.Utf8.parse(ivstr)
  347. }
  348. );
  349. return decryted.toString(CryptoJS.enc.Utf8);
  350. };
  351. /**
  352. * 清除页面缓存
  353. * @param fn
  354. * @returns {{beforeRouteLeave(*, *, *): void}}
  355. */
  356. export const beforeRouteLeaveFn = (fn = () => 0) => ({
  357. // 清除页面缓存
  358. beforeRouteLeave(to, from, next) {
  359. fn(this);
  360. if (from.meta.toPagesKeep.includes(to.path)) {
  361. from.meta.keepAlive = true;
  362. } else {
  363. let vnode = this.$vnode;
  364. let parentVnode = vnode && vnode.parent;
  365. if (
  366. parentVnode &&
  367. parentVnode.componentInstance &&
  368. parentVnode.componentInstance.cache
  369. ) {
  370. let key =
  371. vnode.key == null
  372. ? node.componentOptions.Ctor.cid +
  373. (vnode.componentOptions.tag
  374. ? `::${vnode.componentOptions.tag}`
  375. : "")
  376. : vnode.key;
  377. let cache = parentVnode.componentInstance.cache;
  378. let keys = parentVnode.componentInstance.keys;
  379. if (cache[key]) {
  380. this.$destroy();
  381. // remove key
  382. if (keys.length) {
  383. let index = keys.indexOf(key);
  384. if (index > -1) {
  385. keys.splice(index, 1);
  386. }
  387. }
  388. cache[key] = null;
  389. }
  390. }
  391. }
  392. next();
  393. },
  394. });
  395. /**
  396. * 获取字符串长度,两个英文字符算1个
  397. * @param str
  398. * @returns {number}
  399. */
  400. export const getStringSize = (str = "") => {
  401. const letterSize = (str.match(/[a-z0-9,;.!@#-+/\\$%^*()<>?:"'{}~]/gi) || "")
  402. .length;
  403. const chineseSize = str.length - letterSize;
  404. return Math.ceil(letterSize / 2) + chineseSize;
  405. };
  406. /**
  407. * 根据字符串长度截取字符串,两个英文字符算1个
  408. * @param str
  409. * @param maxSize
  410. * @returns {number}
  411. */
  412. export const getStringByMaxSize = (str, maxSize) => {
  413. const strArr = str.split("");
  414. let size = 0;
  415. let newStr = "";
  416. for (let i = 0; i < strArr.length; i++) {
  417. if (/[a-z]|[0-9]|[,;.!@#-+/\\$%^*()<>?:"'{}~]/i.test(strArr[i])) {
  418. size = parseFloat((size + 0.5).toFixed(1));
  419. } else {
  420. size += 1;
  421. }
  422. newStr += strArr[i];
  423. if (size >= maxSize) {
  424. break;
  425. }
  426. }
  427. return newStr;
  428. };