router.js 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import { createElement } from 'react';
  2. import dynamic from 'dva/dynamic';
  3. import pathToRegexp from 'path-to-regexp';
  4. import { getMenuData } from './menu';
  5. let routerDataCache;
  6. const modelNotExisted = (app, model) => (
  7. // eslint-disable-next-line
  8. !app._models.some(({ namespace }) => {
  9. return namespace === model.substring(model.lastIndexOf('/') + 1);
  10. })
  11. );
  12. // wrapper of dynamic
  13. const dynamicWrapper = (app, models, component) => {
  14. // () => require('module')
  15. // transformed by babel-plugin-dynamic-import-node-sync
  16. if (component.toString().indexOf('.then(') < 0) {
  17. models.forEach((model) => {
  18. if (modelNotExisted(app, model)) {
  19. // eslint-disable-next-line
  20. app.model(require(`../models/${model}`).default);
  21. }
  22. });
  23. return (props) => {
  24. if (!routerDataCache) {
  25. routerDataCache = getRouterData(app);
  26. }
  27. return createElement(component().default, {
  28. ...props,
  29. routerData: routerDataCache,
  30. });
  31. };
  32. }
  33. // () => import('module')
  34. return dynamic({
  35. app,
  36. models: () => models.filter(
  37. model => modelNotExisted(app, model)).map(m => import(`../models/${m}.js`)
  38. ),
  39. // add routerData prop
  40. component: () => {
  41. if (!routerDataCache) {
  42. routerDataCache = getRouterData(app);
  43. }
  44. return component().then((raw) => {
  45. const Component = raw.default || raw;
  46. return props => createElement(Component, {
  47. ...props,
  48. routerData: routerDataCache,
  49. });
  50. });
  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. '/resource/image': {
  72. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/PictureList')),
  73. },
  74. '/resource/imageCreate': {
  75. component: dynamicWrapper(app, [], () => import('../routes/Resource/PictureCreate')),
  76. },
  77. '/resource/imageCreate/single': {
  78. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/PictureCreateSingle')),
  79. name: '单图上传',
  80. },
  81. '/resource/imageCreate/multiple': {
  82. component: dynamicWrapper(app, [], () => import('../routes/Resource/PictureCreateMultiple')),
  83. name: '多图上传',
  84. },
  85. '/resource/video': {
  86. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/VideoList')),
  87. },
  88. '/merchant': {
  89. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/Merchant')),
  90. },
  91. '/merchant/list': {
  92. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantList')),
  93. name: '商户列表',
  94. },
  95. '/merchant/create': {
  96. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantCreate')),
  97. name: '添加商户',
  98. },
  99. '/merchant/edit/:id': {
  100. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantCreate')),
  101. name: '编辑商户',
  102. },
  103. '/merchant/despoit/:id': {
  104. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantDespoit')),
  105. name: '余额充值',
  106. },
  107. '/campus': {
  108. component: dynamicWrapper(app, ['campus'], () => import('../routes/Campus/Campus')),
  109. },
  110. '/campus/list': {
  111. component: dynamicWrapper(app, ['campus'], () => import('../routes/Campus/CampusList')),
  112. name: '校区列表',
  113. },
  114. '/dashboard/analysis': {
  115. component: dynamicWrapper(app, ['chart'], () => import('../routes/Dashboard/Analysis')),
  116. },
  117. '/dashboard/monitor': {
  118. component: dynamicWrapper(app, ['monitor'], () => import('../routes/Dashboard/Monitor')),
  119. },
  120. '/dashboard/workplace': {
  121. component: dynamicWrapper(app, ['project', 'activities', 'chart'], () => import('../routes/Dashboard/Workplace')),
  122. // hideInBreadcrumb: true,
  123. // name: '工作台',
  124. // authority: 'admin',
  125. },
  126. '/form/basic-form': {
  127. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/BasicForm')),
  128. },
  129. '/form/step-form': {
  130. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm')),
  131. },
  132. '/form/step-form/info': {
  133. name: '分步表单(填写转账信息)',
  134. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm/Step1')),
  135. },
  136. '/form/step-form/confirm': {
  137. name: '分步表单(确认转账信息)',
  138. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm/Step2')),
  139. },
  140. '/form/step-form/result': {
  141. name: '分步表单(完成)',
  142. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/StepForm/Step3')),
  143. },
  144. '/form/advanced-form': {
  145. component: dynamicWrapper(app, ['form'], () => import('../routes/Forms/AdvancedForm')),
  146. },
  147. '/list/table-list': {
  148. component: dynamicWrapper(app, ['rule'], () => import('../routes/List/TableList')),
  149. },
  150. '/list/basic-list': {
  151. component: dynamicWrapper(app, ['list'], () => import('../routes/List/BasicList')),
  152. },
  153. '/list/card-list': {
  154. component: dynamicWrapper(app, ['list'], () => import('../routes/List/CardList')),
  155. },
  156. '/list/search': {
  157. component: dynamicWrapper(app, ['list'], () => import('../routes/List/List')),
  158. },
  159. '/list/search/projects': {
  160. component: dynamicWrapper(app, ['list'], () => import('../routes/List/Projects')),
  161. },
  162. '/list/search/applications': {
  163. component: dynamicWrapper(app, ['list'], () => import('../routes/List/Applications')),
  164. },
  165. '/list/search/articles': {
  166. component: dynamicWrapper(app, ['list'], () => import('../routes/List/Articles')),
  167. },
  168. '/profile/basic': {
  169. component: dynamicWrapper(app, ['profile'], () => import('../routes/Profile/BasicProfile')),
  170. },
  171. '/profile/advanced': {
  172. component: dynamicWrapper(app, ['profile'], () => import('../routes/Profile/AdvancedProfile')),
  173. },
  174. '/result/success': {
  175. component: dynamicWrapper(app, [], () => import('../routes/Result/Success')),
  176. },
  177. '/result/fail': {
  178. component: dynamicWrapper(app, [], () => import('../routes/Result/Error')),
  179. },
  180. '/exception/403': {
  181. component: dynamicWrapper(app, [], () => import('../routes/Exception/403')),
  182. },
  183. '/exception/404': {
  184. component: dynamicWrapper(app, [], () => import('../routes/Exception/404')),
  185. },
  186. '/exception/500': {
  187. component: dynamicWrapper(app, [], () => import('../routes/Exception/500')),
  188. },
  189. '/exception/trigger': {
  190. component: dynamicWrapper(app, ['error'], () => import('../routes/Exception/triggerException')),
  191. },
  192. '/user': {
  193. component: dynamicWrapper(app, [], () => import('../layouts/UserLayout')),
  194. },
  195. '/user/login': {
  196. component: dynamicWrapper(app, ['login'], () => import('../routes/User/Login')),
  197. },
  198. '/user/register': {
  199. component: dynamicWrapper(app, ['register'], () => import('../routes/User/Register')),
  200. },
  201. '/user/register-result': {
  202. component: dynamicWrapper(app, [], () => import('../routes/User/RegisterResult')),
  203. },
  204. // '/user/:id': {
  205. // component: dynamicWrapper(app, [], () => import('../routes/User/SomeComponent')),
  206. // },
  207. };
  208. // Get name from ./menu.js or just set it in the router data.
  209. const menuData = getFlatMenuData(getMenuData());
  210. // Route configuration data
  211. // eg. {name,authority ...routerConfig }
  212. const routerData = {};
  213. // The route matches the menu
  214. Object.keys(routerConfig).forEach((path) => {
  215. // Regular match item name
  216. // eg. router /user/:id === /user/chen
  217. const pathRegexp = pathToRegexp(path);
  218. const menuKey = Object.keys(menuData).find(key => pathRegexp.test(`${key}`));
  219. let menuItem = {};
  220. // If menuKey is not empty
  221. if (menuKey) {
  222. menuItem = menuData[menuKey];
  223. }
  224. let router = routerConfig[path];
  225. // If you need to configure complex parameter routing,
  226. // 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
  227. // eg . /list/:type/user/info/:id
  228. router = {
  229. ...router,
  230. name: router.name || menuItem.name,
  231. authority: router.authority || menuItem.authority,
  232. };
  233. routerData[path] = router;
  234. });
  235. return routerData;
  236. };