index.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  1. // 首页
  2. var homeTab = {
  3. id: 'home', // 唯一标识
  4. name: '首页',
  5. url: 'main.html', // 页面地址
  6. isNeedLoad: false, // 标注:是否需要此刻加载
  7. hideClose: true // 隐藏关闭键
  8. }
  9. // sa_admin对象
  10. var sa_admin = new Vue({
  11. components: {
  12. "nav-logo": httpVueLoader('sa-frame/nav/nav-logo.vue'), // logo
  13. "nav-menu-bar": httpVueLoader('sa-frame/nav/nav-menu-bar.vue'), // 菜单栏
  14. "nav-tool-bar": httpVueLoader('sa-frame/nav/nav-tool-bar.vue'), // 工具栏
  15. "nav-tab-bar": httpVueLoader('sa-frame/nav/nav-tab-bar.vue'), // tab栏
  16. "nav-view-vessel": httpVueLoader('sa-frame/nav/nav-view-vessel.vue'), // 视图容器
  17. "com-right-menu": httpVueLoader('sa-frame/nav/com-right-menu.vue'), // 右键菜单
  18. "com-add-tab": httpVueLoader('sa-frame/nav/com-add-tab.vue'), // 双击添加 tab 的弹窗
  19. },
  20. el: '.app',
  21. data: {
  22. // ------------------------------- 配置 -------------------------------
  23. title: '', // 页面标题 -- Sa-Admin
  24. logo: '', // logo地址 -- sa-frame/admin-logo.png
  25. icon: '', // icon地址 -- sa-frame/admin-logo.png
  26. version: 'v1.40.0', // 当前版本号
  27. updateTime: '2021-9-26', // 更新日期
  28. githubUrl: 'https://github.com/click33/sa-admin', // github地址
  29. isRemeOpen: true, // 是否记住上一次最后打开的窗口
  30. printInfo: true, // 是否在控制台打印信息
  31. homeTab: homeTab, // 主页首屏 Tab
  32. menuList: [], // 全部菜单集合
  33. showList: [], // 显示的菜单集合(id集合)
  34. plusVersion: 'v1.26.0', //sa-plus版本
  35. plusUpdateTime: '2021-10-24', //sa-plus版本
  36. plusGithubUrl: 'https://github.com/click33/sa-plus', // github地址
  37. // ------------------------------- 状态 -------------------------------
  38. themeV: localStorage.getItem('themeV') || '1', // 当前 / 默认的主题
  39. isOpen: true, // 当前是否展开菜单 (整体框架)
  40. isOpenRight: true, // 当前是否展开 (右边) (将右边盒子折叠与菜单折叠分开,这样可以减少动画的卡顿现象)
  41. activeMenuId: '0', // 正在高亮的菜单id
  42. isDrag: false, // 当前是否正在拖拽 tab
  43. dragTab: null, // 当前正在拖拽的 tab
  44. tabList: [homeTab], // 当前 Tab 集合
  45. viewList: [homeTab], // 当前 View 集合
  46. nativeTab: homeTab, // 当前正显示的Tab
  47. user: null,// user信息
  48. dropList: [], // 头像处下拉列表菜单
  49. },
  50. watch: {
  51. // 监听title改变时, 页面title也跟着切换
  52. title: function (newValue, oldValue) {
  53. document.querySelector('title').innerHTML = newValue;
  54. },
  55. // 监听 icon_url 网页图标
  56. icon: function (newValue, oldValue) {
  57. var icon = newValue;
  58. var iconTarget = document.querySelector('.admin-icon');
  59. if (iconTarget) {
  60. iconTarget.setAttribute('href', icon);
  61. }
  62. }
  63. },
  64. methods: {
  65. // ------------------- 初始化相关 --------------------
  66. // 初始化模板, 此方法必须且只能调用一次
  67. init: function (option) {
  68. // 打开上次最后的一个窗口
  69. this.showTabByHash();
  70. if (this.nativeTab.id == this.homeTab.id) {
  71. this.showHome();
  72. }
  73. // 打印版本等信息
  74. if (this.printInfo) {
  75. this.printVesion();
  76. }
  77. // 手动触发一下窗口变动监听
  78. window.onresize();
  79. },
  80. // 初始化菜单:
  81. // showList = 显示菜单id数组 —— (注意是id的数组),你填哪些id哪些菜单才会显示 ,为空时代表显示所有
  82. initMenu: function (showList) {
  83. this.setMenuList(window.menuList, showList);
  84. },
  85. // 写入菜单:
  86. // menuList = 全部菜单 —— 可以是已经渲染好的 tree 数组,也可以是一个尚未渲染的一维数组(你只要指定好 parent_id,Sa-Admin内部会自动渲染)
  87. // showList = 显示菜单id数组 —— (注意是id的数组),你填哪些id哪些菜单才会显示 ,为空时代表显示所有
  88. setMenuList: function (menuList, showList) {
  89. // 设置 全部菜单
  90. this.menuList = this.arrayToTree(menuList);
  91. // 设置 显示的菜单id
  92. showList = showList || this.getAllId(this.menuList);
  93. for (var i = 0; i < showList.length; i++) {
  94. showList[i] = showList[i] + '';
  95. }
  96. this.showList = showList;
  97. },
  98. // ------------------- Menu 相关操作 --------------------
  99. // 根据 id 查找 Menu
  100. getMenuById: function (id) {
  101. return this.findMenuById(this.menuList, id);
  102. },
  103. // 显示某个菜单,根据id
  104. showMenuById: function (id) {
  105. var menu = this.getMenuById(id);
  106. if (menu) {
  107. this.showTab(menu);
  108. }
  109. },
  110. // 显示homeTab
  111. showHome: function () {
  112. this.showTab(this.homeTab);
  113. },
  114. // 返回当前所有菜单的 一维数组 形式 (将树形菜单转化为一维数组并返回) 方便遍历
  115. getYwList: function () {
  116. var arr = [];
  117. function _dg(menuList) {
  118. menuList = menuList || [];
  119. for (var i = 0; i < menuList.length; i++) {
  120. var menu = menuList[i];
  121. arr.push(menu);
  122. // 如果有子菜单
  123. if (menu.childList) {
  124. _dg(menu.childList);
  125. }
  126. }
  127. }
  128. _dg(this.menuList);
  129. return arr;
  130. },
  131. // 获取菜单所有id
  132. getAllId: function () {
  133. var arr = [];
  134. this.getYwList().forEach(function (item) {
  135. arr.push(item.id);
  136. });
  137. return arr;
  138. },
  139. // ------------------- Tab 相关操作 --------------------
  140. // 刷新Tab
  141. f5Tab: function (tab) {
  142. var cs = '#iframe-' + tab.id;
  143. var iframe = document.querySelector(cs);
  144. if (iframe) {
  145. iframe.setAttribute('src', this.getTabUrl(tab));
  146. } else {
  147. tab.isNeedLoad = false;
  148. this.$nextTick(function () {
  149. tab.isNeedLoad = true;
  150. })
  151. }
  152. },
  153. // 获取 Tab,根据 id
  154. getTabById: function (id) {
  155. for (var i = 0; i < this.tabList.length; i++) {
  156. if (this.tabList[i].id + '' == id + '') {
  157. return this.tabList[i];
  158. }
  159. }
  160. return null;
  161. },
  162. // 添加一个Tab {id,name,url}
  163. addTab: function (tab) {
  164. // 如果没有提供id,则随机一个
  165. if (!tab.id) {
  166. tab.id = new Date().getTime() + '' + this.randomNum();
  167. }
  168. // 如果没有指定类型
  169. if (tab.view === undefined) {
  170. if (this.getUrlExt(tab.url).toLowerCase() == 'vue') {
  171. tab.view = httpVueLoader(tab.url);
  172. }
  173. }
  174. if (tab.isNeedLoad === undefined) {
  175. // tab.isNeedLoad = true;
  176. Vue.set(tab, 'isNeedLoad', true);
  177. }
  178. // console.log('添加之前:' + JSON.stringify(tab));
  179. this.tabList.push(tab);
  180. this.viewList.push(tab);
  181. // tab 超过 20 个,提示过多,如果用户无视继续添加则超过 30 个后不再提示
  182. if (this.tabList.length > 20 && this.tabList.length < 30) {
  183. sa_admin.$message({message: '选项卡过多会造成窗口卡顿,建议您关闭不使用的窗口', type: 'warning'});
  184. }
  185. },
  186. // 显示某个页面 (如果不存在, 则先添加)
  187. showTab: function (tab) {
  188. // 标注:需要此刻加载
  189. // tab.isNeedLoad = false;
  190. Vue.set(tab, 'isNeedLoad', true);
  191. // 如果是外部链接
  192. if (tab.is_blank) {
  193. return open(tab.url);
  194. }
  195. // 如果是当前正在显示的tab , 则直接返回,无需继续操作
  196. if (tab == this.nativeTab) {
  197. return;
  198. }
  199. // 如果是click函数
  200. if (tab.click) {
  201. if (tab.click() !== true) {
  202. return;
  203. }
  204. }
  205. // 如果这个 tab 还没有添加到 tabList 上
  206. if (this.getTabById(tab.id) == null) {
  207. this.addTab(tab);
  208. }
  209. // 然后开始显示这个 tab
  210. this.nativeTab = tab;
  211. // this.nativeTab.is_load = true; // 标注:已经加载过了
  212. this.activeMenuId = tab.id + ''; // 左边自动关联, 如果左边没有,则无效果
  213. // 刷新一下url中的锚链
  214. this.$nextTick(function () {
  215. this.f5HashByNativeTab();
  216. })
  217. // 调整一下滚动条
  218. this.$nextTick(function () {
  219. try {
  220. this.$refs['nav-tab-bar'].scrollToAuto();
  221. } catch (e) {
  222. }
  223. })
  224. },
  225. // 显示一个选项卡, 根据 id , 不存在则不显示
  226. showTabById: function (id) {
  227. var tab = this.getTabById(id);
  228. if (tab) {
  229. this.showTab(tab);
  230. }
  231. },
  232. // 关闭 tab (带动画)
  233. closeTab: function (tab, callFn) {
  234. // homeTab不能关闭
  235. if (tab == this.homeTab || tab.hideClose) {
  236. return;
  237. }
  238. // 执行关闭动画
  239. var div = document.querySelector('#tab-' + tab.id);
  240. div.style.width = div.offsetWidth + 'px';
  241. setTimeout(function () {
  242. div.style.width = '0px';
  243. }, 0);
  244. // 等待动画结束
  245. setTimeout(function () {
  246. // 如果 tab 为当前正在显示的 tab, 则切换为前一个 tab
  247. if (tab == this.nativeTab) {
  248. var index = this.tabList.indexOf(tab);
  249. var preTab = this.tabList[index - 1];
  250. if (preTab) {
  251. this.showTab(preTab);
  252. } else {
  253. var nextTab = this.tabList[index + 1];
  254. this.showTab(nextTab);
  255. }
  256. }
  257. // 从 tabList 中移除这个 tab
  258. sa_admin_code_util.arrayDelete(this.tabList, tab);
  259. sa_admin_code_util.arrayDelete(this.viewList, tab);
  260. // 如果有回调
  261. if (callFn) {
  262. this.$nextTick(function () {
  263. callFn();
  264. })
  265. }
  266. }.bind(this), 150);
  267. },
  268. // 关闭 tab, 根据 id
  269. closeTabById: function (id, callFn) {
  270. var tab = this.getTabById(id);
  271. if (tab) {
  272. this.closeTab(tab, callFn);
  273. }
  274. },
  275. // 悬浮打开 tab
  276. xfTab: function (tab) {
  277. console.log('悬浮');
  278. // layer打开
  279. var index = layer.open({
  280. type: 2,
  281. title: tab.name,
  282. moveOut: true, // 是否可拖动到外面
  283. maxmin: true, // 显示最大化按钮
  284. shadeClose: false,
  285. shade: 0,
  286. area: ['80%', '80%'],
  287. zIndex: layer.zIndex,
  288. content: this.getTabUrl(tab),
  289. // 解决拉伸或者最大化的时候,iframe高度不能自适应的问题
  290. resizing: function (layero) {
  291. sa_admin_code_util.solveLayerBug(index);
  292. },
  293. // 操作这个layer的时候置顶它
  294. success: function (layero) {
  295. layer.setTop(layero);
  296. }
  297. });
  298. // 解决拉伸或者最大化的时候,iframe高度不能自适应的问题
  299. document.querySelector('#layui-layer' + index + ' .layui-layer-max').onclick = function () {
  300. setTimeout(function () {
  301. sa_admin_code_util.solveLayerBug(index);
  302. }, 200)
  303. }
  304. },
  305. // 新窗口打开 tab
  306. newWinTab: function (tab) {
  307. open(this.getTabUrl(tab));
  308. // this.closeTab(tab);
  309. },
  310. // 获取指定 tab 所代表 iframe 的 url 地址 (同域下可获取最新地址, 跨域时只能获取初始化时的地址)
  311. getTabUrl: function (tab) {
  312. var cs = '#iframe-' + tab.id;
  313. var iframe = document.querySelector(cs);
  314. if (!iframe) {
  315. return tab.url;
  316. }
  317. try {
  318. return iframe.contentWindow.location.href;
  319. } catch (e) {
  320. return iframe.getAttribute('src');
  321. }
  322. },
  323. // ------------------- 框架整体相关操作 --------------------
  324. // 展开菜单
  325. startOpen: function () {
  326. this.isOpen = true;
  327. setTimeout(function () {
  328. this.isOpenRight = true;
  329. }.bind(this), 200);
  330. },
  331. // 折叠菜单
  332. endOpen: function () {
  333. this.isOpen = false;
  334. this.isOpenRight = false;
  335. },
  336. // ------------------- 锚链接路由相关 --------------------
  337. // 根据锚链接, 打开窗口
  338. showTabByHash: function () {
  339. // 如果非记住模式
  340. if (this.isRemeOpen == false) {
  341. return;
  342. }
  343. // 获取锚链接中的id
  344. var hash = location.hash;
  345. var id = hash.replace('#', '');
  346. if (id == '') {
  347. return;
  348. }
  349. // 如果已经存在与tabbar中
  350. var tab = this.getTabById(id);
  351. if (tab) {
  352. return this.showTab(tab);
  353. }
  354. // 否则从菜单中打开
  355. this.showMenuById(id);
  356. // 此时, 仍有一种tab打不开, 那就是自定义tab然后还已经关闭的,
  357. // 预设 解决方案: 在localStor里存储所有打开过的tab,
  358. // 以后如果有强需求这个功能时, 再实现
  359. },
  360. // 根据当前tab刷新一下锚链接
  361. f5HashByNativeTab: function () {
  362. // 如果非记住模式
  363. if (this.isRemeOpen == false) {
  364. return;
  365. }
  366. location.hash = this.nativeTab.id;
  367. },
  368. // ------------------- 工具方法 --------------------
  369. // 弹窗提示
  370. msg: function (msg) {
  371. layer.msg(msg)
  372. },
  373. // 返回随机数
  374. randomNum: function (min, max) {
  375. min = min || 1;
  376. max = max || 1000000000;
  377. return parseInt(Math.random() * (max - min + 1) + min, 10);
  378. },
  379. // 从 menuList 里查找指定 id 的 menu,支持多级递归
  380. findMenuById: function (menuList, id) {
  381. for (var i = 0; i < menuList.length; i++) {
  382. var menu = menuList[i];
  383. if (menu.id + '' == id + '') {
  384. return menu;
  385. }
  386. // 如果是二级或多级
  387. if (menu.childList) {
  388. var menu2 = this.findMenuById(menu.childList, id);
  389. if (menu2 != null) {
  390. return menu2;
  391. }
  392. }
  393. }
  394. return null;
  395. },
  396. // 获取文件后缀
  397. getUrlExt: function (url) {
  398. if (!url) {
  399. return "";
  400. }
  401. if (url.indexOf('?') > -1) {
  402. url = url.split('?')[0];
  403. }
  404. if (url.indexOf('#') > -1) {
  405. url = url.split('#')[0];
  406. }
  407. var index = url.lastIndexOf(".");
  408. if (index == -1) {
  409. return "";
  410. }
  411. var ext = url.substr(index + 1);
  412. return ext;
  413. },
  414. // 将一维平面数组转换为 Tree 菜单 (根据其指定的 parent_id 添加到其父菜单的childList)
  415. arrayToTree: function (menuList) {
  416. for (var i = 0; i < menuList.length; i++) {
  417. var menu = menuList[i];
  418. // 如果这个 Menu 指定了 parent_id 属性,则将其转移到其指定的父 Menu 的 childList 属性上
  419. if (menu.parent_id) {
  420. var parent_menu = this.findMenuById(menuList, menu.parent_id);
  421. if (parent_menu) {
  422. menu.parent_menu = parent_menu;
  423. parent_menu.childList = parent_menu.childList || [];
  424. parent_menu.childList.push(menu);
  425. menuList.splice(i, 1); // 从一维中删除
  426. i--;
  427. }
  428. }
  429. }
  430. return menuList;
  431. },
  432. // ------------------- 其它 --------------------
  433. // 获取指定 tab 栏的 window 对象, 用于多窗口通信
  434. getTabWindow: function (tabId) {
  435. var iframe = document.querySelector('#iframe-' + tabId);
  436. if (iframe != null) {
  437. return iframe.contentWindow;
  438. }
  439. return null;
  440. },
  441. // 打印版本
  442. // printVesion: function() {
  443. // console.log('欢迎使用Sa-Admin,当前版本:' + this.version + ",更新于:" + this.updateTime + ",GitHub地址:" + this.githubUrl);
  444. // console.log('如在使用中发现任何bug或者疑问,请加入QQ群交流:782974737,点击加入:' + 'https://jq.qq.com/?_wv=1027&k=5DHN5Ib');
  445. // },
  446. printVesion: function () {
  447. var str = ('欢迎使用sa-plus,当前版本:' + this.plusVersion + ",更新于:" + this.plusUpdateTime + ",GitHub地址:" + this.plusGithubUrl);
  448. // console.log('%c%s', 'color: green; font-size: 12px; font-weight: 400; margin-top: 4px; margin-bottom: 4px;', str);
  449. var str2 = ('如在使用中发现任何bug或者疑问,请加入QQ群交流:782974737,点击加入:' + 'https://jq.qq.com/?_wv=1027&k=5DHN5Ib');
  450. var s = str;// + ' \n' + str2;
  451. },
  452. },
  453. created: function () {
  454. }
  455. });
  456. var saAdmin = sa_admin;
  457. Vue.prototype.sa_admin = sa_admin;
  458. Vue.prototype.saAdmin = saAdmin;
  459. // 监听窗口大小变动
  460. window.onresize = function () {
  461. if (document.body.clientWidth < 800) {
  462. sa_admin.endOpen();
  463. } else {
  464. sa_admin.startOpen();
  465. }
  466. }
  467. // 监听锚链接变动
  468. window.onhashchange = function () {
  469. sa_admin.showTabByHash();
  470. }