util.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. // +---------------------------------------------------------------------
  2. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  3. // +---------------------------------------------------------------------
  4. // | Copyright (c) 2016~2023 https://www.crmeb.com All rights reserved.
  5. // +---------------------------------------------------------------------
  6. // | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
  7. // +---------------------------------------------------------------------
  8. // | Author: CRMEB Team <admin@crmeb.com>
  9. // +---------------------------------------------------------------------
  10. import Cookies from 'js-cookie';
  11. // cookie保存的天数
  12. import config from '@/config';
  13. import { forEach, hasOneOf, objEqual } from '@/libs/tools';
  14. import { cloneDeep } from 'lodash';
  15. const { title, useI18n } = config;
  16. // 设置setCookies;
  17. // setToken
  18. export const setCookies = (key, val, cookieExpires) => {
  19. Cookies.set(key, val, { expires: cookieExpires || 1 });
  20. };
  21. // 获取getCookies;
  22. // getToken
  23. export const getCookies = (key) => {
  24. return Cookies.get(key);
  25. };
  26. export const removeCookies = (key) => {
  27. return Cookies.remove(key);
  28. };
  29. export const hasChild = (item) => {
  30. return item.children && item.children.length !== 0;
  31. };
  32. const showThisMenuEle = (item, access) => {
  33. if (item.meta && item.meta.access && item.meta.access.length) {
  34. if (hasOneOf(item.meta.access, access)) return true;
  35. else return false;
  36. } else return true;
  37. };
  38. /**
  39. * @param {Array} list 通过路由列表得到菜单列表
  40. * @returns {Array}
  41. */
  42. export const getMenuByRouter = (list, access) => {
  43. let res = [];
  44. forEach(list, (item) => {
  45. if (!item.meta || (item.meta && !item.meta.hideInMenu)) {
  46. let obj = {
  47. icon: (item.meta && item.meta.icon) || '',
  48. name: item.name,
  49. meta: item.meta,
  50. };
  51. if ((hasChild(item) || (item.meta && item.meta.showAlways)) && showThisMenuEle(item, access)) {
  52. obj.children = getMenuByRouter(item.children, access);
  53. }
  54. if (item.meta && item.meta.href) obj.href = item.meta.href;
  55. if (showThisMenuEle(item, access)) res.push(obj);
  56. }
  57. });
  58. return res;
  59. };
  60. /**
  61. * @param {Array} routeMetched 当前路由metched
  62. * @returns {Array}
  63. */
  64. export const getBreadCrumbList = (route, homeRoute) => {
  65. let homeItem = { ...homeRoute, icon: homeRoute.meta.icon };
  66. let routeMetched = route.matched;
  67. if (routeMetched.some((item) => item.name === homeRoute.name)) return [homeItem];
  68. let res = routeMetched
  69. .filter((item) => {
  70. return item.meta === undefined || !item.meta.hideInBread;
  71. })
  72. .map((item) => {
  73. let meta = { ...item.meta };
  74. if (meta.title && typeof meta.title === 'function') {
  75. meta.__titleIsFunction__ = true;
  76. meta.title = meta.title(route);
  77. }
  78. let obj = {
  79. icon: (item.meta && item.meta.icon) || '',
  80. name: item.name,
  81. meta: meta,
  82. };
  83. return obj;
  84. });
  85. res = res.filter((item) => {
  86. return !item.meta.hideInMenu;
  87. });
  88. return [{ ...homeItem, to: homeRoute.path }, ...res];
  89. };
  90. export const getRouteTitleHandled = (route) => {
  91. let router = { ...route };
  92. let meta = { ...route.meta };
  93. let title = '';
  94. if (meta.title) {
  95. if (typeof meta.title === 'function') {
  96. meta.__titleIsFunction__ = true;
  97. title = meta.title(router);
  98. } else title = meta.title;
  99. }
  100. meta.title = title;
  101. router.meta = meta;
  102. return router;
  103. };
  104. export const showTitle = (item, vm) => {
  105. let { title, __titleIsFunction__ } = item.meta;
  106. if (!title) return;
  107. if (useI18n) {
  108. if (title.includes('{{') && title.includes('}}') && useI18n)
  109. title = title.replace(/({{[\s\S]+?}})/, (m, str) => str.replace(/{{([\s\S]*)}}/, (m, _) => vm.$t(_.trim())));
  110. else if (__titleIsFunction__) title = item.meta.title;
  111. else title = vm.$t(item.name);
  112. } else title = (item.meta && item.meta.title) || item.name;
  113. return title;
  114. };
  115. /**
  116. * @description 本地存储和获取标签导航列表
  117. */
  118. export const setTagNavListInLocalstorage = (list) => {
  119. localStorage.tagNaveList = JSON.stringify(list);
  120. };
  121. /**
  122. * @returns {Array} 其中的每个元素只包含路由原信息中的name, path, meta三项
  123. */
  124. export const getTagNavListFromLocalstorage = () => {
  125. const list = localStorage.tagNaveList;
  126. return list ? JSON.parse(list) : [];
  127. };
  128. /**
  129. * @param {Array} routers 路由列表数组
  130. * @description 用于找到路由列表中name为home的对象
  131. */
  132. export const getHomeRoute = (routers, homeName = 'home') => {
  133. let i = -1;
  134. let len = routers.length;
  135. let homeRoute = {};
  136. while (++i < len) {
  137. let item = routers[i];
  138. if (item.children && item.children.length) {
  139. let res = getHomeRoute(item.children, homeName);
  140. if (res.name) return res;
  141. } else {
  142. if (item.name === homeName) homeRoute = item;
  143. }
  144. }
  145. return homeRoute;
  146. };
  147. /**
  148. * @param {*} list 现有标签导航列表
  149. * @param {*} newRoute 新添加的路由原信息对象
  150. * @description 如果该newRoute已经存在则不再添加
  151. */
  152. export const getNewTagList = (list, newRoute) => {
  153. const { name, path, meta } = newRoute;
  154. let newList = [...list];
  155. if (newList.findIndex((item) => item.path === path) >= 0) return newList;
  156. else newList.push({ name, path, meta });
  157. return newList;
  158. };
  159. /**
  160. * @param {*} access 用户权限数组,如 ['super_admin', 'admin']
  161. * @param {*} route 路由列表
  162. */
  163. const hasAccess = (access, route) => {
  164. if (route.meta && route.meta.access) return hasOneOf(access, route.meta.access);
  165. else return true;
  166. };
  167. /**
  168. * 权鉴
  169. * @param {*} name 即将跳转的路由name
  170. * @param {*} access 用户权限数组
  171. * @param {*} routes 路由列表
  172. * @description 用户是否可跳转到该页
  173. */
  174. export const canTurnTo = (name, access, routes) => {
  175. const routePermissionJudge = (list) => {
  176. return list.some((item) => {
  177. if (item.children && item.children.length) {
  178. return routePermissionJudge(item.children);
  179. } else if (item.name === name) {
  180. return hasAccess(access, item);
  181. }
  182. });
  183. };
  184. return routePermissionJudge(routes);
  185. };
  186. /**
  187. * @param {String} url
  188. * @description 从URL中解析参数
  189. */
  190. export const getParams = (url) => {
  191. const keyValueArr = url.split('?')[1].split('&');
  192. let paramObj = {};
  193. keyValueArr.forEach((item) => {
  194. const keyValue = item.split('=');
  195. paramObj[keyValue[0]] = keyValue[1];
  196. });
  197. return paramObj;
  198. };
  199. /**
  200. * @param {Array} list 标签列表
  201. * @param {String} name 当前关闭的标签的name
  202. */
  203. export const getNextRoute = (list, route) => {
  204. let res = {};
  205. if (list.length === 2) {
  206. res = getHomeRoute(list);
  207. } else {
  208. const index = list.findIndex((item) => routeEqual(item, route));
  209. if (index === list.length - 1) res = list[list.length - 2];
  210. else res = list[index + 1];
  211. }
  212. return res;
  213. };
  214. /**
  215. * @param {Number} times 回调函数需要执行的次数
  216. * @param {Function} callback 回调函数
  217. */
  218. export const doCustomTimes = (times, callback) => {
  219. let i = -1;
  220. while (++i < times) {
  221. callback(i);
  222. }
  223. };
  224. /**
  225. * @param {Object} file 从上传组件得到的文件对象
  226. * @returns {Promise} resolve参数是解析后的二维数组
  227. * @description 从Csv文件中解析出表格,解析成二维数组
  228. */
  229. export const getArrayFromFile = (file) => {
  230. let nameSplit = file.name.split('.');
  231. let format = nameSplit[nameSplit.length - 1];
  232. return new Promise((resolve, reject) => {
  233. let reader = new FileReader();
  234. reader.readAsText(file); // 以文本格式读取
  235. let arr = [];
  236. reader.onload = function (evt) {
  237. let data = evt.target.result; // 读到的数据
  238. let pasteData = data.trim();
  239. arr = pasteData
  240. .split(/[\n\u0085\u2028\u2029]|\r\n?/g)
  241. .map((row) => {
  242. return row.split('\t');
  243. })
  244. .map((item) => {
  245. return item[0].split(',');
  246. });
  247. if (format === 'csv') resolve(arr);
  248. else reject(new Error('[Format Error]:你上传的不是Csv文件'));
  249. };
  250. });
  251. };
  252. /**
  253. * @param {Array} array 表格数据二维数组
  254. * @returns {Object} { columns, tableData }
  255. * @description 从二维数组中获取表头和表格数据,将第一行作为表头,用于在iView的表格中展示数据
  256. */
  257. export const getTableDataFromArray = (array) => {
  258. let columns = [];
  259. let tableData = [];
  260. if (array.length > 1) {
  261. let titles = array.shift();
  262. columns = titles.map((item) => {
  263. return {
  264. title: item,
  265. key: item,
  266. };
  267. });
  268. tableData = array.map((item) => {
  269. let res = {};
  270. item.forEach((col, i) => {
  271. res[titles[i]] = col;
  272. });
  273. return res;
  274. });
  275. }
  276. return {
  277. columns,
  278. tableData,
  279. };
  280. };
  281. export const findNodeUpper = (ele, tag) => {
  282. if (ele.parentNode) {
  283. if (ele.parentNode.tagName === tag.toUpperCase()) {
  284. return ele.parentNode;
  285. } else {
  286. return findNodeUpper(ele.parentNode, tag);
  287. }
  288. }
  289. };
  290. export const findNodeUpperByClasses = (ele, classes) => {
  291. let parentNode = ele.parentNode;
  292. if (parentNode) {
  293. let classList = parentNode.classList;
  294. if (classList && classes.every((className) => classList.contains(className))) {
  295. return parentNode;
  296. } else {
  297. return findNodeUpperByClasses(parentNode, classes);
  298. }
  299. }
  300. };
  301. export const findNodeDownward = (ele, tag) => {
  302. const tagName = tag.toUpperCase();
  303. if (ele.childNodes.length) {
  304. let i = -1;
  305. let len = ele.childNodes.length;
  306. while (++i < len) {
  307. let child = ele.childNodes[i];
  308. if (child.tagName === tagName) return child;
  309. else return findNodeDownward(child, tag);
  310. }
  311. }
  312. };
  313. export const showByAccess = (access, canViewAccess) => {
  314. return hasOneOf(canViewAccess, access);
  315. };
  316. /**
  317. * @description 根据name/params/query判断两个路由对象是否相等
  318. * @param {*} route1 路由对象
  319. * @param {*} route2 路由对象
  320. */
  321. export const routeEqual = (route1, route2) => {
  322. const params1 = route1.params || {};
  323. const params2 = route2.params || {};
  324. const query1 = route1.query || {};
  325. const query2 = route2.query || {};
  326. return route1.name === route2.name && objEqual(params1, params2) && objEqual(query1, query2);
  327. };
  328. /**
  329. * 判断打开的标签列表里是否已存在这个新添加的路由对象
  330. */
  331. export const routeHasExist = (tagNavList, routeItem) => {
  332. let len = tagNavList.length;
  333. let res = false;
  334. doCustomTimes(len, (index) => {
  335. if (routeEqual(tagNavList[index], routeItem)) res = true;
  336. });
  337. return res;
  338. };
  339. export const localSave = (key, value) => {
  340. localStorage.setItem(key, value);
  341. };
  342. export const localRead = (key) => {
  343. return localStorage.getItem(key) || '';
  344. };
  345. // scrollTop animation
  346. export const scrollTop = (el, from = 0, to, duration = 500, endCallback) => {
  347. if (!window.requestAnimationFrame) {
  348. window.requestAnimationFrame =
  349. window.webkitRequestAnimationFrame ||
  350. window.mozRequestAnimationFrame ||
  351. window.msRequestAnimationFrame ||
  352. function (callback) {
  353. return window.setTimeout(callback, 1000 / 60);
  354. };
  355. }
  356. const difference = Math.abs(from - to);
  357. const step = Math.ceil((difference / duration) * 50);
  358. const scroll = (start, end, step) => {
  359. if (start === end) {
  360. endCallback && endCallback();
  361. return;
  362. }
  363. let d = start + step > end ? end : start + step;
  364. if (start > end) {
  365. d = start - step < end ? end : start - step;
  366. }
  367. if (el === window) {
  368. window.scrollTo(d, d);
  369. } else {
  370. el.scrollTop = d;
  371. }
  372. window.requestAnimationFrame(() => scroll(d, end, step));
  373. };
  374. scroll(from, to, step);
  375. };
  376. /**
  377. * @description 根据当前跳转的路由设置显示在浏览器标签的title
  378. * @param {Object} routeItem 路由对象
  379. * @param {Object} vm Vue实例
  380. */
  381. export const setTitle = (routeItem, vm) => {
  382. let winTitle = localStorage.getItem('ADMIN_TITLE') || title;
  383. const handledRoute = getRouteTitleHandled(routeItem);
  384. const pageTitle = showTitle(handledRoute, vm);
  385. const resTitle = pageTitle ? `${winTitle} - ${pageTitle}` : winTitle;
  386. window.document.title = resTitle;
  387. };
  388. export const R = (menuList, newOpenMenus) => {
  389. menuList.forEach((item) => {
  390. let newMenu = {};
  391. for (let i in item) {
  392. if (i !== 'children') newMenu[i] = cloneDeep(item[i]);
  393. }
  394. newOpenMenus.push(newMenu);
  395. item.children && R(item.children, newOpenMenus);
  396. });
  397. return newOpenMenus;
  398. };
  399. export function getMenuopen(to, menuList) {
  400. const allMenus = [];
  401. menuList.forEach((menu) => {
  402. const menus = transMenu(menu, []);
  403. allMenus.push({
  404. path: menu.path,
  405. openNames: [],
  406. });
  407. menus.forEach((item) => allMenus.push(item));
  408. });
  409. const currentMenu = allMenus.find((item) => item.path === to.path);
  410. return currentMenu ? currentMenu.openNames : [];
  411. }
  412. function transMenu(menu, openNames) {
  413. if (menu.children && menu.children.length) {
  414. const itemOpenNames = openNames.concat([menu.path]);
  415. return menu.children.reduce((all, item) => {
  416. all.push({
  417. path: item.path,
  418. openNames: itemOpenNames,
  419. });
  420. const foundChildren = transMenu(item, itemOpenNames);
  421. return all.concat(foundChildren);
  422. }, []);
  423. } else {
  424. return [menu].map((item) => {
  425. return {
  426. path: item.path,
  427. openNames: openNames,
  428. };
  429. });
  430. }
  431. }
  432. export function wss(wsSocketUrl) {
  433. let ishttps = document.location.protocol == 'https:';
  434. if (ishttps) {
  435. return wsSocketUrl.replace('ws:', 'wss:');
  436. } else {
  437. return wsSocketUrl.replace('wss:', 'ws:');
  438. }
  439. }