router.js 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. import React, { createElement } from 'react';
  2. import { Spin } from 'antd';
  3. import pathToRegexp from 'path-to-regexp';
  4. import Loadable from 'react-loadable';
  5. import { getMenuData } from './menu';
  6. let routerDataCache;
  7. const getRouterDataCache = app => {
  8. if (!routerDataCache) {
  9. routerDataCache = getRouterData(app);
  10. }
  11. return routerDataCache;
  12. };
  13. const modelNotExisted = (app, model) =>
  14. // eslint-disable-next-line
  15. !app._models.some(({ namespace }) => {
  16. return namespace === model.substring(model.lastIndexOf('/') + 1);
  17. });
  18. // wrapper of dynamic
  19. const dynamicWrapper = (app, models, component) => {
  20. // register models
  21. models.forEach(model => {
  22. if (modelNotExisted(app, model)) {
  23. // eslint-disable-next-line
  24. app.model(require(`../models/${model}`).default);
  25. }
  26. });
  27. // () => require('module')
  28. // transformed by babel-plugin-dynamic-import-node-sync
  29. if (component.toString().indexOf('.then(') < 0) {
  30. return props => {
  31. return createElement(component().default, {
  32. ...props,
  33. routerData: getRouterDataCache(app),
  34. });
  35. };
  36. }
  37. // () => import('module')
  38. return Loadable({
  39. loader: () => {
  40. return component().then(raw => {
  41. const Component = raw.default || raw;
  42. return props =>
  43. createElement(Component, {
  44. ...props,
  45. routerData: getRouterDataCache(app),
  46. });
  47. });
  48. },
  49. loading: () => {
  50. return <Spin size="large" className="global-spin" />;
  51. },
  52. });
  53. };
  54. function getFlatMenuData(menus) {
  55. let keys = {};
  56. menus.forEach(item => {
  57. if (item.children) {
  58. keys[item.path] = { ...item };
  59. keys = { ...keys, ...getFlatMenuData(item.children) };
  60. } else {
  61. keys[item.path] = { ...item };
  62. }
  63. });
  64. return keys;
  65. }
  66. export const getRouterData = app => {
  67. const routerConfig = {
  68. '/': {
  69. component: dynamicWrapper(app, ['user', 'login'], () => import('../layouts/BasicLayout')),
  70. },
  71. '/dashboard/analysis': {
  72. component: dynamicWrapper(app, ['chart'], () => import('../routes/Dashboard/Analysis')),
  73. },
  74. '/dashboard/monitor': {
  75. component: dynamicWrapper(app, ['monitor'], () => import('../routes/Dashboard/Monitor')),
  76. },
  77. '/dashboard/workplace': {
  78. component: dynamicWrapper(app, ['project', 'activities', 'chart'], () =>
  79. import('../routes/Dashboard/Workplace')
  80. ),
  81. // hideInBreadcrumb: true,
  82. // name: '工作台',
  83. // authority: 'admin',
  84. },
  85. '/form/basic-form': {
  86. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/BasicForm')),
  87. },
  88. '/form/step-form': {
  89. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm')),
  90. },
  91. '/form/step-form/info': {
  92. name: '分步表单(填写转账信息)',
  93. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm/Step1')),
  94. },
  95. '/form/step-form/confirm': {
  96. name: '分步表单(确认转账信息)',
  97. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm/Step2')),
  98. },
  99. '/form/step-form/result': {
  100. name: '分步表单(完成)',
  101. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm/Step3')),
  102. },
  103. '/form/advanced-form': {
  104. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/AdvancedForm')),
  105. },
  106. '/list/table-list': {
  107. component: dynamicWrapper(app, ['rule'], () => import('../routes/List/TableList')),
  108. },
  109. '/list/basic-list': {
  110. component: dynamicWrapper(app, ['list'], () => import('../routes/List/BasicList')),
  111. },
  112. '/list/card-list': {
  113. component: dynamicWrapper(app, ['list'], () => import('../routes/List/CardList')),
  114. },
  115. '/list/search': {
  116. component: dynamicWrapper(app, ['list'], () => import('../routes/List/List')),
  117. },
  118. '/list/search/projects': {
  119. component: dynamicWrapper(app, ['list'], () => import('../routes/List/Projects')),
  120. },
  121. '/list/search/applications': {
  122. component: dynamicWrapper(app, ['list'], () => import('../routes/List/Applications')),
  123. },
  124. '/list/search/articles': {
  125. component: dynamicWrapper(app, ['list'], () => import('../routes/List/Articles')),
  126. },
  127. '/profile/basic': {
  128. component: dynamicWrapper(app, ['profile'], () => import('../routes/Profile/BasicProfile')),
  129. },
  130. '/profile/advanced': {
  131. component: dynamicWrapper(app, ['profile'], () =>
  132. import('../routes/Profile/AdvancedProfile')
  133. ),
  134. },
  135. '/result/success': {
  136. component: dynamicWrapper(app, [], () => import('../routes/Result/Success')),
  137. },
  138. '/result/fail': {
  139. component: dynamicWrapper(app, [], () => import('../routes/Result/Error')),
  140. },
  141. '/exception/403': {
  142. component: dynamicWrapper(app, [], () => import('../routes/Exception/403')),
  143. },
  144. '/exception/404': {
  145. component: dynamicWrapper(app, [], () => import('../routes/Exception/404')),
  146. },
  147. '/exception/500': {
  148. component: dynamicWrapper(app, [], () => import('../routes/Exception/500')),
  149. },
  150. '/exception/trigger': {
  151. component: dynamicWrapper(app, ['error'], () =>
  152. import('../routes/Exception/triggerException')
  153. ),
  154. },
  155. '/user': {
  156. component: dynamicWrapper(app, [], () => import('../layouts/UserLayout')),
  157. },
  158. '/user/login': {
  159. component: dynamicWrapper(app, ['login'], () => import('../routes/User/Login')),
  160. },
  161. '/user/register': {
  162. component: dynamicWrapper(app, ['register'], () => import('../routes/User/Register')),
  163. },
  164. '/user/register-result': {
  165. component: dynamicWrapper(app, [], () => import('../routes/User/RegisterResult')),
  166. },
  167. // '/user/:id': {
  168. // component: dynamicWrapper(app, [], () => import('../routes/User/SomeComponent')),
  169. // },
  170. };
  171. // Get name from ./menu.js or just set it in the router data.
  172. const menuData = getFlatMenuData(getMenuData());
  173. // Route configuration data
  174. // eg. {name,authority ...routerConfig }
  175. const routerData = {};
  176. // The route matches the menu
  177. Object.keys(routerConfig).forEach(path => {
  178. // Regular match item name
  179. // eg. router /user/:id === /user/chen
  180. const pathRegexp = pathToRegexp(path);
  181. const menuKey = Object.keys(menuData).find(key => pathRegexp.test(`${key}`));
  182. let menuItem = {};
  183. // If menuKey is not empty
  184. if (menuKey) {
  185. menuItem = menuData[menuKey];
  186. }
  187. let router = routerConfig[path];
  188. // If you need to configure complex parameter routing,
  189. // https://github.com/ant-design/ant-design-pro-site/blob/master/docs/router-and-nav.md#%E5%B8%A6%E5%8F%82%E6%95%B0%E7%9A%84%E8%B7%AF%E7%94%B1%E8%8F%9C%E5%8D%95
  190. // eg . /list/:type/user/info/:id
  191. router = {
  192. ...router,
  193. name: router.name || menuItem.name,
  194. authority: router.authority || menuItem.authority,
  195. hideInBreadcrumb: router.hideInBreadcrumb || menuItem.hideInBreadcrumb,
  196. };
  197. routerData[path] = router;
  198. });
  199. return routerData;
  200. };