utils.js 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733
  1. /**
  2. * utils.js
  3. * 各类js工具函数
  4. */
  5. /*
  6. * 时间格式化函数(将时间格式化为,2019年08月12日,2019-08-12,2019/08/12的形式)
  7. *
  8. *
  9. * pattern参数(想要什么格式的数据就传入什么格式的数据)
  10. * · 'yyyy-MM-dd' ---> 输出如2019-09-20
  11. * · 'yyyy-MM-dd hh:mm' ---> 输出如2019-09-20 08:20
  12. * · 'yyyy-MM-dd HH:mm:ss' ---> 输出如2019-09-20 18:20:23
  13. * · 'yyyy/MM/dd' ---> 输出如2019/09/20
  14. * · 'yyyy年MM月dd日' ---> 输出如2019年09月20日
  15. * · 'yyyy年MM月dd日 hh时mm分' ---> 输出如2019年09月20日 08时20分
  16. * · 'yyyy年MM月dd日 hh时mm分ss秒' ---> 输出如2019年09月20日 08时20分23秒
  17. * · 'yyyy年MM月dd日 hh时mm分ss秒 EE' ---> 输出如2019年09月20日 08时20分23秒 周二
  18. * · 'yyyy年MM月dd日 hh时mm分ss秒 EEE' ---> 输出如2019年09月20日 08时20分23秒 星期二
  19. *
  20. * 参考: https://www.cnblogs.com/mr-wuxiansheng/p/6296646.html
  21. */
  22. Date.prototype.pattern = function (fmt) {
  23. if (!fmt) return ''
  24. var o = {
  25. 'y+': this.getFullYear(),
  26. 'M+': this.getMonth() + 1, // 月份
  27. 'd+': this.getDate(), // 日
  28. // 12小时制
  29. 'h+': this.getHours() % 12 == 0 ? 12 : this.getHours() % 12, // 小时
  30. // 24小时制
  31. 'H+': this.getHours(), // 小时
  32. 'm+': this.getMinutes(), // 分
  33. 's+': this.getSeconds(), // 秒
  34. 'q+': Math.floor((this.getMonth() + 3) / 3), // 季度
  35. 'S': this.getMilliseconds(), // 毫秒
  36. 'E+': this.getDay(), // 周
  37. };
  38. var week = {
  39. '0': '日',
  40. '1': '一',
  41. '2': '二',
  42. '3': '三',
  43. '4': '四',
  44. '5': '五',
  45. '6': '六'
  46. };
  47. if (/(y+)/.test(fmt)) {
  48. fmt = fmt.replace(RegExp.$1, (this.getFullYear() + '').substr(4 - RegExp.$1.length));
  49. }
  50. if (/(E+)/.test(fmt)) {
  51. fmt = fmt.replace(RegExp.$1, ((RegExp.$1.length > 1) ? (RegExp.$1.length > 2 ? '星期' : '周') : '') + week[
  52. this.getDay() + '']);
  53. }
  54. for (var k in o) {
  55. if (new RegExp('(' + k + ')').test(fmt)) {
  56. fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (('00' + o[k]).substr(('' + o[k])
  57. .length)));
  58. }
  59. }
  60. return fmt;
  61. }
  62. /* 挂载到Number原型上,格式化小数用tofixed会造成位数不够补零的情况(处理不让自动补零)
  63. 栗子: 不是number类型的需要转一下
  64. Number('').fixed()
  65. Number('213412DDD').fixed()
  66. 1234.152354.fixed()
  67. 1234.152354.fixed(2)
  68. 1234.152354.fixed(10)
  69. */
  70. Number.prototype.fixed = function(len){
  71. len = isNaN(len)? 0 : len
  72. var num = Math.pow(10,len)
  73. return Math.round(this*num) / num
  74. }
  75. var versionChangeLog = {
  76. '0.0.1': 'init',
  77. '0.0.2': 'utils.dateFromNow 函数逻辑修改,具体修改内容查看函数注释'
  78. }
  79. var utils = {
  80. // utils版本号
  81. version: '0.0.2',
  82. isWeiXinBrowser: navigator.userAgent.toLowerCase().indexOf('micromessenger') !== -1,
  83. isAndroid: navigator.userAgent.toLowerCase().indexOf('android') !== -1,
  84. isIos: /iphone|ipod|ipad|ios/.test(navigator.userAgent.toLowerCase()),
  85. // 传入你要获取的参数的名字
  86. getParam: function (name) {
  87. var reg = new RegExp('(^|&)' + name + '=([^&]*)(&|$)', 'i');
  88. var r = window.location.search.substr(1).match(reg); //获取url中'?'符后的字符串并正则匹配
  89. var context = '';
  90. if (r != null) context = r[2];
  91. // 释放变量
  92. reg = null;
  93. r = null;
  94. return context == null || context == '' || context == 'undefined' ? '' : context;
  95. },
  96. // 移除url参数,但是不刷新页面
  97. delUrlParam: function (name){
  98. var href = window.location.href;
  99. const e = eval(`/&?${name}=[^&#]*/g`);
  100. href = href.replace(e, '');
  101. history.replaceState('','',href);// replaceState方法不会触发页面刷新,只是导致history对象发生变化,地址栏
  102. },
  103. // 取[m, n]随机数
  104. getRandomNumber: function (min, max) {
  105. return Math.floor(Math.random() * (max - min + 1) + min);
  106. },
  107. // 获取唯一的uuid
  108. // https://www.kancloud.cn/ifeng/js100/622666
  109. getRandomUuid: function () {
  110. var s = [];
  111. var hexDigits = "0123456789abcdef";
  112. for (var i = 0; i < 36; i++) {
  113. s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1);
  114. }
  115. s[14] = "4"; // bits 12-15 of the time_hi_and_version field to 0010
  116. s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1); // bits 6-7 of the clock_seq_hi_and_reserved to 01
  117. s[8] = s[13] = s[18] = s[23] = "-";
  118. var uuid = s.join("");
  119. return uuid;
  120. },
  121. // 获取随机字符串
  122. // 不传参数则获取长度不固定的字符串
  123. getRandomString: function (len) {
  124. var randomString = '';
  125. if (len) {
  126. /****默认去掉了容易混淆的字符oOLl,9gq,Vv,Uu,I1****/
  127. var $chars = 'ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678';
  128. var maxPos = $chars.length;
  129. for (i = 0; i < len; i++) {
  130. randomString += $chars.charAt(Math.floor(Math.random() * maxPos));
  131. }
  132. } else {
  133. // Math.random() 生成随机数字, eg: 0.123456
  134. // .toString(36) 转化成36进制 : "0.4fzyo82mvyr"
  135. // .substring(2) 去掉前面两位 : "yo82mvyr"
  136. // .slice(-8) 截取最后八位 : "yo82mvyr"
  137. randomString = Math.random().toString(36).substring(2)
  138. }
  139. return randomString;
  140. },
  141. // FROM: https://www.jianshu.com/p/90ed8b728975
  142. // 比较两个对象是否相等
  143. // 返回true为相等,返回false为不相等
  144. deepCompare: function (x, y) {
  145. var i, l, leftChain, rightChain;
  146. function compare2Objects(x, y) {
  147. var p;
  148. // remember that NaN === NaN returns false
  149. // and isNaN(undefined) returns true
  150. if (isNaN(x) && isNaN(y) && typeof x === 'number' && typeof y === 'number') {
  151. return true;
  152. }
  153. // Compare primitives and functions.
  154. // Check if both arguments link to the same object.
  155. // Especially useful on the step where we compare prototypes
  156. if (x === y) {
  157. return true;
  158. }
  159. // Works in case when functions are created in constructor.
  160. // Comparing dates is a common scenario. Another built-ins?
  161. // We can even handle functions passed across iframes
  162. if ((typeof x === 'function' && typeof y === 'function') ||
  163. (x instanceof Date && y instanceof Date) ||
  164. (x instanceof RegExp && y instanceof RegExp) ||
  165. (x instanceof String && y instanceof String) ||
  166. (x instanceof Number && y instanceof Number)) {
  167. return x.toString() === y.toString();
  168. }
  169. // At last checking prototypes as good as we can
  170. if (!(x instanceof Object && y instanceof Object)) {
  171. return false;
  172. }
  173. if (x.isPrototypeOf(y) || y.isPrototypeOf(x)) {
  174. return false;
  175. }
  176. if (x.constructor !== y.constructor) {
  177. return false;
  178. }
  179. if (x.prototype !== y.prototype) {
  180. return false;
  181. }
  182. // Check for infinitive linking loops
  183. if (leftChain.indexOf(x) > -1 || rightChain.indexOf(y) > -1) {
  184. return false;
  185. }
  186. // Quick checking of one object being a subset of another.
  187. // todo: cache the structure of arguments[0] for performance
  188. for (p in y) {
  189. if (y.hasOwnProperty(p) !== x.hasOwnProperty(p)) {
  190. return false;
  191. } else if (typeof y[p] !== typeof x[p]) {
  192. return false;
  193. }
  194. }
  195. for (p in x) {
  196. if (y.hasOwnProperty(p) !== x.hasOwnProperty(p)) {
  197. return false;
  198. } else if (typeof y[p] !== typeof x[p]) {
  199. return false;
  200. }
  201. switch (typeof (x[p])) {
  202. case 'object':
  203. case 'function':
  204. leftChain.push(x);
  205. rightChain.push(y);
  206. if (!compare2Objects(x[p], y[p])) {
  207. return false;
  208. }
  209. leftChain.pop();
  210. rightChain.pop();
  211. break;
  212. default:
  213. if (x[p] !== y[p]) {
  214. return false;
  215. }
  216. break;
  217. }
  218. }
  219. return true;
  220. }
  221. if (arguments.length < 1) {
  222. return true; //Die silently? Don't know how to handle such case, please help...
  223. // throw "Need two or more arguments to compare";
  224. }
  225. for (i = 1, l = arguments.length; i < l; i++) {
  226. leftChain = []; //Todo: this can be cached
  227. rightChain = [];
  228. if (!compare2Objects(arguments[0], arguments[i])) {
  229. return false;
  230. }
  231. }
  232. return true;
  233. },
  234. // 判断变量是否是数组
  235. isArray: function (o){
  236. return Object.prototype.toString.call(o) == '[object Array]';
  237. },
  238. // 数组对象根据某一个值进行从小到大冒泡排序
  239. // arr 数组
  240. // value 字符串
  241. bSort: function (arr, value) {
  242. var len = arr.length;
  243. for (var i = 0; i < len - 1; i++) {
  244. for (var j = 0; j < len - 1 - i; j++) {
  245. // 相邻元素两两对比,元素交换,大的元素交换到后面
  246. if (arr[j][value] > arr[j + 1][value]) {
  247. var temp = arr[j];
  248. arr[j] = arr[j + 1];
  249. arr[j + 1] = temp;
  250. }
  251. }
  252. }
  253. return arr;
  254. },
  255. // 数组去重
  256. unique: function (arr) {
  257. if (!this.isArray(arr)) {
  258. console.log('type error!')
  259. return
  260. }
  261. var array = [];
  262. for (var i = 0; i < arr.length; i++) {
  263. if (array .indexOf(arr[i]) === -1) {
  264. array .push(arr[i])
  265. }
  266. }
  267. return array
  268. },
  269. // 数组乱序
  270. shuffle: function (arr) {
  271. var targetArr = JSON.parse(JSON.stringify(arr || []))
  272. var hitArr = []
  273. var len = targetArr.length
  274. for (var i = 0; i < len; i++) {
  275. // 获取随机下标
  276. var index = Math.floor(Math.random() * targetArr.length)
  277. hitArr.push(targetArr[index])
  278. targetArr.splice(index, 1)
  279. }
  280. return hitArr.concat(targetArr)
  281. },
  282. // 解决ios返回不刷新页面的问题
  283. iosBackRefresh: function () {
  284. var isPageHide = false;
  285. window.addEventListener('pageshow', function () {
  286. if (isPageHide) {
  287. location.reload();
  288. }
  289. });
  290. window.addEventListener('pagehide', function () {
  291. isPageHide = true;
  292. });
  293. },
  294. // ios返回调用函数
  295. iosBackInvoke: function (callback) {
  296. var isPageHide = false;
  297. window.addEventListener('pageshow', function () {
  298. if (isPageHide) {
  299. callback && callback();
  300. }
  301. });
  302. window.addEventListener('pagehide', function () {
  303. isPageHide = true;
  304. });
  305. },
  306. /**
  307. * 格式化金钱的函数
  308. * @param {number} s 金额必传
  309. * @param {int:0-100} n 保留小数的位数(int:0-100)
  310. */
  311. formatMoney: function (s, n) {
  312. if (s==null||s=='null'){
  313. s = 0
  314. }
  315. if (n === undefined) {
  316. n = -1
  317. } else {
  318. n = n > 0 && n <= 20 ? n : 2;
  319. }
  320. var intS = parseInt(s)
  321. var point = '.'
  322. var left;
  323. var right;
  324. s = parseFloat((s + '').replace(/[^\d\.-]/g, ''))
  325. // 没传n,默认(如果为整数,则不保留小数。如果为浮点数,则保留两位小数)
  326. if (n === -1) {
  327. if (s === intS) {
  328. n = 0
  329. right = ''
  330. point = ''
  331. } else {
  332. n = 2
  333. s = s.toFixed(n);
  334. right = s.split('.')[1];
  335. }
  336. s = s + ''
  337. left = s.split('.')[0].split('').reverse();
  338. } else {
  339. s = parseFloat((s + '').replace(/[^\d\.-]/g, '')).toFixed(n) + '';
  340. left = s.split('.')[0].split('').reverse();
  341. right = s.split('.')[1];
  342. }
  343. t = '';
  344. for (i = 0; i < left.length; i++) {
  345. t += left[i] + ((i + 1) % 3 == 0 && (i + 1) != left.length ? ',' : '');
  346. }
  347. var money = t.split('').reverse().join('') + point + right;
  348. return money;
  349. },
  350. // 金额大写,链接:https://juejin.im/post/5a2a7a5051882535cd4abfce
  351. // upDigit(1682) result:"人民币壹仟陆佰捌拾贰元整"
  352. // upDigit(-1693) result:"欠壹仟陆佰玖拾叁元整"
  353. upPrice: function (n) {
  354. var fraction = ['角', '分', '厘']
  355. var digit = ['零', '壹', '贰', '叁', '肆', '伍', '陆', '柒', '捌', '玖']
  356. var unit = [
  357. ['元', '万', '亿'],
  358. ['', '拾', '佰', '仟']
  359. ]
  360. // var head = n < 0 ? '欠人民币' : '人民币'
  361. var head = ''
  362. n = Math.abs(n)
  363. var s = ''
  364. for (var i = 0; i < fraction.length; i++) {
  365. s += (digit[Math.floor(n * 10 * Math.pow(10, i)) % 10] + fraction[i]).replace(/零./, '')
  366. }
  367. s = s || '整'
  368. n = Math.floor(n)
  369. for (var i = 0; i < unit[0].length && n > 0; i++) {
  370. var p = ''
  371. for (var j = 0; j < unit[1].length && n > 0; j++) {
  372. p = digit[n % 10] + unit[1][j] + p
  373. n = Math.floor(n / 10)
  374. }
  375. s = p.replace(/(零.)*零$/, '').replace(/^$/, '零') + unit[0][i] + s
  376. // s = p + unit[0][i] + s;
  377. }
  378. return head + s.replace(/(零.)*零元/, '元').replace(/(零.)+/g, '零').replace(/^整$/, '零元整')
  379. },
  380. // 手机号中间4位加*
  381. addConfusionForTel: function (tel) {
  382. var reg = /^(\d{3})\d{4}(\d{4})$/
  383. return tel.replace(reg, '$1****$2')
  384. },
  385. // 手机号加空格
  386. addSpaceForTel: function (tel) {
  387. var regMap = {
  388. isConfuse: /^(\d{3})\*{4}(\d{4})$/,
  389. addSpace: /^(\d{3})(\d{4})(\d{4})$/
  390. }
  391. var confusion = regMap.isConfuse.test(tel)
  392. if (confusion) {
  393. return tel.replace(regMap.isConfuse, '$1 **** $2')
  394. } else {
  395. return tel.replace(regMap.addSpace, '$1 $2 $3')
  396. }
  397. },
  398. /**
  399. * 时间戳转换 多少秒、多少分、多少小时前 昨天以及昨天之前的显示年月日(今年的时间不显示年份)
  400. * @param {Date} originTime 要转换的时间对象(或者时间戳)
  401. */
  402. dateFromNow: function (originTime, useOld) {
  403. if (!originTime) return
  404. // 原始时间 - 传入的时间戳
  405. var originTimeStamp = +new Date(originTime)
  406. // 当前时间戳
  407. var nowTimeStamp = +new Date()
  408. // 时间戳相差多少
  409. var diffTimeStamp = nowTimeStamp - originTimeStamp
  410. var postfix = diffTimeStamp > 0 ? '前' : '后'
  411. // 求绝对值 ms(毫秒)
  412. var diffTimeStampAbsMs = Math.abs(diffTimeStamp)
  413. var diffTimeStampAbsS = Math.round(diffTimeStampAbsMs / 1000)
  414. // 10天的秒数
  415. var days11 = 11 * 24 * 60 * 60
  416. // var dataMap = {
  417. // zh: ['年', '个月', '星期', '天', '小时', '分钟', '秒'],
  418. // number: [365*24*60*60, 30*24*60*60, 7*24*60*60, 24*60*60, 60*60, 60, 1]
  419. // }
  420. var dataMap = {
  421. zh: ['天', '小时', '分钟', '秒'],
  422. number: [24 * 60 * 60, 60 * 60, 60, 1]
  423. }
  424. var timeString = ''
  425. // 10天前
  426. var tenDaysAgo = diffTimeStampAbsS > days11
  427. // 是否是当天
  428. var isCurrentDay = new Date(originTimeStamp).pattern('yyyy.MM.dd') === new Date().pattern('yyyy.MM.dd')
  429. var condition = !isCurrentDay
  430. if (useOld) {
  431. condition = tenDaysAgo
  432. }
  433. // 是否是当天
  434. if (condition) {
  435. // 大于10天,则使用正常日期显示
  436. var originDate = new Date(originTimeStamp)
  437. var nowDate = new Date()
  438. // 是否同年
  439. var sameYear = originDate.getFullYear() === nowDate.getFullYear()
  440. // 如果是当年,则不显示年
  441. var patternString = sameYear ? 'MM-dd' : 'yyyy-MM-dd'
  442. timeString = originDate.pattern(patternString)
  443. } else {
  444. for (var i = 0; i < dataMap.number.length; i++) {
  445. var inm = Math.floor(diffTimeStampAbsS / dataMap.number[i])
  446. if (inm != 0) {
  447. timeString = inm + dataMap.zh[i] + postfix
  448. break
  449. }
  450. }
  451. }
  452. return timeString
  453. },
  454. /**
  455. * 输入框获取焦点,隐藏底部固定按钮方法
  456. * @param {Array} inputs 传入输入框的DOM对象
  457. * @param {Object} footer 按键弹起要隐藏的footer的 DOM对象
  458. */
  459. inputFocusHideFooter: function (inputs, footer) {
  460. var isShowBtn = function (f) {
  461. if (f) {
  462. footer.style.display = ''
  463. } else {
  464. footer.style.display = 'none'
  465. }
  466. }
  467. // ios监听输入框的软键盘弹起和收起事件
  468. if (this.isIos) {
  469. inputs.forEach(function (item) {
  470. item.addEventListener('focus', function () {
  471. console.log('IOS 键盘弹出')
  472. // IOS 键盘弹起后操作
  473. isShowBtn(false)
  474. }, false)
  475. // IOS 键盘收起:IOS 点击输入框以外区域或点击收起按钮,输入框都会失去焦点,键盘会收起,
  476. item.addEventListener('blur', function () {
  477. console.log('IOS 键盘收起')
  478. // IOS 键盘收起后操作
  479. isShowBtn(true)
  480. })
  481. })
  482. }
  483. // Android 键盘收起:Android 键盘弹起或收起页面高度会发生变化,以此为依据获知键盘收起
  484. if (this.isAndroid) {
  485. var clientHeight = document.documentElement.clientHeight || document.body.clientHeight
  486. window.addEventListener('resize', function () {
  487. var nowClientHeight = document.documentElement.clientHeight || document.body.clientHeight
  488. if (clientHeight > nowClientHeight) {
  489. // 键盘弹出的事件处理
  490. console.log('Android 键盘弹出')
  491. isShowBtn(false)
  492. } else {
  493. console.log('Android 键盘收起')
  494. // 键盘收起的事件处理
  495. isShowBtn(true)
  496. }
  497. }, false)
  498. }
  499. },
  500. /**
  501. * 通用关键字高亮替换
  502. * @param {String} value 要高亮的字符串
  503. * @param {String|Array} oldChar 要被替换的字符串(或数组)
  504. * @param {String|Array} newChar 要替换成的字符串(或数组)
  505. *
  506. * 比如:要将 - `剑鱼标讯工具函数` 字符串中的 `工具` 高亮
  507. * 则此时 value -> `剑鱼标讯工具函数`
  508. * oldChar -> `工具`
  509. * newChar -> `<span class="highlight-text">工具</span>`
  510. *
  511. * 批量高亮-----
  512. * 比如:要将 - `剑鱼标讯工具函数` 字符串中的 `工具` `剑鱼` 高亮
  513. * 则此时 value -> `剑鱼标讯工具函数批量高亮`
  514. * oldChar -> ['工具', '剑鱼']
  515. * newChar -> ['<span class="highlight-text">', '</span>']
  516. *
  517. * 注意:此时newChar为一个长度为2的数组,数组中为高亮标签的起始标签和结束标签
  518. *
  519. */
  520. replaceKeyword: function (value, oldChar, newChar) {
  521. var _this = this
  522. if (!oldChar || !newChar) return value
  523. // oldChar的字符串数组,用来循环替换
  524. var oldCharArr = []
  525. if (this.isArray(oldChar)) {
  526. oldCharArr = oldChar.concat()
  527. } else {
  528. oldCharArr.push(oldChar)
  529. }
  530. // 数组去重
  531. oldCharArr = this.unique(oldCharArr)
  532. try {
  533. oldCharArr.forEach(function (item) {
  534. // 去空格之后为空字符串,则直接跳过当前替换
  535. if (item.replace(/\s+/g, '')) {
  536. var oc = item
  537. oc = oc.replace(/\$/g, '\\$')
  538. .replace(/\(/g, '\\(')
  539. .replace(/\)/g, '\\)')
  540. .replace(/\*/g, '\\*')
  541. .replace(/\+/g, '\\+')
  542. .replace(/\./g, '\\.')
  543. .replace(/\[/g, '\\[')
  544. .replace(/\]/g, '\\]')
  545. .replace(/\?/g, '\\?')
  546. .replace(/\\/g, '\\')
  547. .replace(/\//g, '\\/')
  548. .replace(/\^/g, '\\^')
  549. .replace(/\{/g, '\\{')
  550. .replace(/\}/g, '\\}')
  551. .replace(/\|/g, '\\|')
  552. if (_this.isArray(newChar)) {
  553. // 批量高亮
  554. var tempV = value
  555. value = value.replace(new RegExp('(' + oc + ')', 'gmi'), newChar[0] + oc + newChar[1])
  556. if (value === tempV && oc.indexOf('+') !== -1) {
  557. var splitReg = oc.split('\\+')
  558. splitReg.map(function (v) {
  559. value = value.replace(new RegExp('('+v+')', 'gmi'), newChar[0] + v + newChar[1])
  560. })
  561. }
  562. } else {
  563. // 普通单个高亮
  564. value = value.replace(new RegExp('(' + oc + ')', 'gmi'), newChar)
  565. }
  566. }
  567. })
  568. } catch (e) {
  569. console.log(e)
  570. return value
  571. }
  572. return value
  573. },
  574. /**
  575. * 批量删除sessionStorage中的数据
  576. * @param {RegExp} reg 传入一个要匹配的正则表达式
  577. */
  578. clearBatchSessionStorage: function (reg) {
  579. if (sessionStorage) {
  580. var sessKeys = Object.keys(sessionStorage)
  581. sessKeys.forEach(function (item) {
  582. if (reg.test(item)) {
  583. sessionStorage.removeItem(item)
  584. }
  585. })
  586. }
  587. },
  588. /**
  589. * 获取图片原始宽高
  590. * @param {Image/Object} img
  591. * @param {Function} callback
  592. */
  593. getImgNaturalDimensions: function (img, callback) {
  594. var naturalSize = {}
  595. if (img.naturalWidth && img.naturalHeight) { // 现代浏览器
  596. naturalSize.with = img.naturalWidth
  597. naturalSize.height = img.naturalHeight
  598. } else { // IE6/7/8
  599. var image = new Image()
  600. image.src = img.src
  601. image.onload = function() {
  602. naturalSize.with = image.width
  603. naturalSize.height = image.height
  604. callback && callback(naturalSize)
  605. }
  606. }
  607. return naturalSize
  608. },
  609. goToCustomerPage: function () {
  610. if (this.isWeiXinBrowser) {
  611. location.href = '/big/wx/page/customer'
  612. } else {
  613. location.href = '/jyapp/free/customer'
  614. }
  615. },
  616. moneyUnit: function (num, type, lv) {
  617. if (num === '' || num == null || num == undefined || isNaN(num)) return ''
  618. var levelArr = ['元', '万元', '亿元', '万亿元']
  619. if (type === void 0) {
  620. type = 'string';
  621. }
  622. if (lv === void 0) {
  623. lv = 0;
  624. }
  625. if (num === 0) {
  626. if (type === 'string') {
  627. return '0';
  628. }
  629. if (type === 'lv') {
  630. return levelArr[lv];
  631. }
  632. if (type === 'number') {
  633. return 0;
  634. }
  635. if (type === 'index') {
  636. return lv;
  637. }
  638. }
  639. var result = num / Math.pow(10000, lv);
  640. if (result > 10000 && lv < 2) {
  641. return this.moneyUnit(num, type, lv + 1)
  642. } else {
  643. if (type === 'string') {
  644. return String(Math.floor(result * 100) / 100).replace('.00', '') + levelArr[lv];
  645. }
  646. if (type === 'lv') {
  647. return levelArr[lv];
  648. }
  649. if (type === 'index') {
  650. return lv;
  651. }
  652. if (type === 'number') {
  653. return String(Math.floor(result * 100) / 100).replace('.00', '');
  654. }
  655. }
  656. },
  657. /**
  658. * 防抖
  659. * @param {Function} func 函数
  660. * @param {Number} delay 延时间隔 默认 200ms
  661. * @param {Boolean} immediate 是否首次执行
  662. */
  663. debounce: function (func, delay, immediate){
  664. var timer = null;
  665. delay = delay || 200
  666. return function() {
  667. var context = this;
  668. var args = arguments;
  669. if(timer) clearTimeout(timer);
  670. if(immediate){
  671. var doNow = !timer;
  672. timer = setTimeout(function(){
  673. timer = null;
  674. },delay);
  675. if(doNow){
  676. func.apply(context,args);
  677. }
  678. }else{
  679. timer = setTimeout(function(){
  680. func.apply(context,args);
  681. },delay);
  682. }
  683. }
  684. }
  685. }