router.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  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. // 厂商管理相关路由注册
  72. '/merchant/list': {
  73. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantList')),
  74. name: '商户列表',
  75. },
  76. '/merchant/create': {
  77. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantCreate')),
  78. name: '添加商户',
  79. },
  80. '/merchant/edit/:id': {
  81. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantCreate')),
  82. name: '编辑商户',
  83. },
  84. '/merchant/deposit/:id': {
  85. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Merchant/MerchantDeposit')),
  86. name: '余额充值',
  87. },
  88. // 校区管理相关路由注册
  89. '/campus/list': {
  90. component: dynamicWrapper(app, ['campus', 'merchant'], () => import('../routes/Campus/CampusList')),
  91. name: '校区列表',
  92. },
  93. '/campus/create': {
  94. component: dynamicWrapper(app, ['campus', 'merchant'], () => import('../routes/Campus/CampusCreate')),
  95. name: '添加校区',
  96. },
  97. '/campus/edit/:id': {
  98. component: dynamicWrapper(app, ['campus', 'merchant'], () => import('../routes/Campus/CampusCreate')),
  99. name: '编辑校区',
  100. },
  101. // 资源管理相关路由注册
  102. '/resource/picture': {
  103. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Picture/PictureList')),
  104. },
  105. '/resource/picture-upload': {
  106. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Picture/PictureUpload')),
  107. },
  108. '/resource/picture-upload/single': {
  109. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Picture/PictureSingleUpload')),
  110. name: '单图上传',
  111. },
  112. '/resource/picture-upload/result': {
  113. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Picture/PictureUploadResult')),
  114. name: '上传结果',
  115. },
  116. '/resource/picture-edit/:id': {
  117. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Picture/PictureEdit')),
  118. name: '编辑图片',
  119. },
  120. '/resource/video': {
  121. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Video/VideoList')),
  122. },
  123. '/resource/video-create': {
  124. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Video/VideoCreate')),
  125. name: '创建视频',
  126. },
  127. '/resource/video-edit/:id': {
  128. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/Video/VideoCreate')),
  129. name: '编辑视频',
  130. },
  131. '/resource/audiobook': {
  132. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/AudioBook/AudioBookList')),
  133. },
  134. '/resource/audiobook-create': {
  135. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/AudioBook/AudioBookCreate')),
  136. name: '创建有声读物',
  137. },
  138. '/resource/audiobook-edit/:id': {
  139. component: dynamicWrapper(app, ['resource'], () => import('../routes/Resource/AudioBook/AudioBookCreate')),
  140. name: '编辑有声读物',
  141. },
  142. // 系统管理相关路由注册
  143. '/system/cms-user': {
  144. component: dynamicWrapper(app, ['cmsUser'], () => import('../routes/System/CmsUser')),
  145. },
  146. '/system/cms-user/list': {
  147. component: dynamicWrapper(app, ['cmsUser'], () => import('../routes/System/CmsUser/CmsUserList')),
  148. },
  149. '/system/cms-user/create': {
  150. component: dynamicWrapper(app, ['cmsUser', 'merchant'], () => import('../routes/System/CmsUser/CmsUserCreate')),
  151. },
  152. '/system/cms-user/edit/:id': {
  153. component: dynamicWrapper(app, ['cmsUser'], () => import('../routes/System/CmsUser/CmsUserEdit')),
  154. },
  155. // 终端管理相关路由注册
  156. '/terminal/user': {
  157. component: dynamicWrapper(app, ['terminal', 'campus', 'merchant'], () => import('../routes/Terminal/User')),
  158. },
  159. '/terminal/user/list': {
  160. component: dynamicWrapper(app, ['terminal', 'campus', 'merchant'], () => import('../routes/Terminal/User/TerminalList')),
  161. },
  162. '/terminal/user/create': {
  163. component: dynamicWrapper(app, ['terminal', 'campus'], () => import('../routes/Terminal/User/TerminalCreate')),
  164. },
  165. '/terminal/user/edit/:id': {
  166. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Terminal/User/TerminalEdit')),
  167. },
  168. '/terminal/whitelist': {
  169. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Terminal/WhiteList')),
  170. },
  171. '/terminal/whitelist/list': {
  172. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Terminal/WhiteList/WhiteList')),
  173. },
  174. '/terminal/whitelist/create': {
  175. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Terminal/WhiteList/WhiteListCreate')),
  176. },
  177. '/terminal/whitelist/edit/:id': {
  178. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Terminal/WhiteList/WhiteListCreate')),
  179. },
  180. '/terminal/auth': {
  181. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Terminal/Auth')),
  182. },
  183. '/terminal/auth/list': {
  184. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Terminal/Auth/AuthList')),
  185. },
  186. // 产品管理相关路由注册
  187. '/product/courseware': {
  188. component: dynamicWrapper(app, [], () => import('../routes/Product/Courseware')),
  189. },
  190. '/product/courseware/list': {
  191. component: dynamicWrapper(app, ['courseware'], () => import('../routes/Product/Courseware/CoursewareList')),
  192. },
  193. '/product/courseware/create': {
  194. component: dynamicWrapper(app, ['courseware', 'resource'], () => import('../routes/Product/Courseware/CoursewareCreate')),
  195. },
  196. '/product/courseware/edit/:id': {
  197. component: dynamicWrapper(app, ['courseware', 'resource'], () => import('../routes/Product/Courseware/CoursewareCreate')),
  198. },
  199. '/product/lesson': {
  200. component: dynamicWrapper(app, ['lesson'], () => import('../routes/Product/Lesson')),
  201. },
  202. '/product/lesson/list': {
  203. component: dynamicWrapper(app, ['lesson'], () => import('../routes/Product/Lesson/LessonList')),
  204. },
  205. '/product/lesson/create': {
  206. component: dynamicWrapper(app, ['lesson', 'courseware'], () => import('../routes/Product/Lesson/LessonCreate')),
  207. },
  208. '/product/lesson/edit/:id': {
  209. component: dynamicWrapper(app, ['lesson', 'courseware'], () => import('../routes/Product/Lesson/LessonCreate')),
  210. },
  211. '/product/course': {
  212. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Course')),
  213. },
  214. '/product/course/list': {
  215. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Course/CourseList')),
  216. },
  217. '/product/course/create': {
  218. component: dynamicWrapper(app, ['lesson', 'resource', 'product', 'merchant'], () => import('../routes/Product/Course/CourseCreate')),
  219. },
  220. '/product/course/edit/:id': {
  221. component: dynamicWrapper(app, ['lesson', 'resource', 'product', 'merchant'], () => import('../routes/Product/Course/CourseCreate')),
  222. },
  223. '/product/support': {
  224. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Support')),
  225. },
  226. '/product/support/list': {
  227. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Support/SupportList')),
  228. },
  229. '/product/support/create': {
  230. component: dynamicWrapper(app, ['resource', 'product', 'merchant'], () => import('../routes/Product/Support/SupportCreate')),
  231. },
  232. '/product/support/edit/:id': {
  233. component: dynamicWrapper(app, ['resource', 'product', 'merchant'], () => import('../routes/Product/Support/SupportCreate')),
  234. },
  235. '/product/training': {
  236. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Training')),
  237. },
  238. '/product/training/list': {
  239. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Training/TrainingList')),
  240. },
  241. '/product/training/create': {
  242. component: dynamicWrapper(app, ['resource', 'product', 'merchant'], () => import('../routes/Product/Training/TrainingCreate')),
  243. },
  244. '/product/training/edit/:id': {
  245. component: dynamicWrapper(app, ['resource', 'product', 'merchant'], () => import('../routes/Product/Training/TrainingCreate')),
  246. },
  247. '/product/package': {
  248. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Package')),
  249. },
  250. '/product/package/list': {
  251. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Package/PackageList')),
  252. },
  253. '/product/package/create': {
  254. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Package/PackageCreate')),
  255. },
  256. '/product/package/edit/:id': {
  257. component: dynamicWrapper(app, ['product'], () => import('../routes/Product/Package/PackageCreate')),
  258. },
  259. // 产品出售相关路由注册
  260. '/shelves/course': {
  261. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves')),
  262. },
  263. '/shelves/course/list': {
  264. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesList')),
  265. },
  266. '/shelves/course/create': {
  267. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesCreate')),
  268. },
  269. '/shelves/course/edit': {
  270. component: dynamicWrapper(app, ['shelves', 'product', 'resource'], () => import('../routes/Shelves/ShelvesEdit')),
  271. },
  272. '/shelves/support': {
  273. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves')),
  274. },
  275. '/shelves/support/list': {
  276. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesList')),
  277. },
  278. '/shelves/support/create': {
  279. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesCreate')),
  280. },
  281. '/shelves/support/edit': {
  282. component: dynamicWrapper(app, ['shelves', 'resource'], () => import('../routes/Shelves/ShelvesEdit')),
  283. },
  284. '/shelves/training': {
  285. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves')),
  286. },
  287. '/shelves/training/list': {
  288. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesList')),
  289. },
  290. '/shelves/training/create': {
  291. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesCreate')),
  292. },
  293. '/shelves/training/edit': {
  294. component: dynamicWrapper(app, ['shelves', 'resource'], () => import('../routes/Shelves/ShelvesEdit')),
  295. },
  296. '/shelves/package': {
  297. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves')),
  298. },
  299. '/shelves/package/list': {
  300. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesList')),
  301. },
  302. '/shelves/package/create': {
  303. component: dynamicWrapper(app, ['shelves'], () => import('../routes/Shelves/ShelvesCreate')),
  304. },
  305. '/shelves/package/edit': {
  306. component: dynamicWrapper(app, ['shelves', 'resource'], () => import('../routes/Shelves/ShelvesEdit')),
  307. },
  308. // 前端管理相关路由注册
  309. '/frontend/tagType': {
  310. component: dynamicWrapper(app, ['tagType'], () => import('../routes/Frontend/TagType')),
  311. },
  312. '/frontend/tagType/list': {
  313. component: dynamicWrapper(app, ['tagType'], () => import('../routes/Frontend/TagType/TagTypeList')),
  314. },
  315. '/frontend/tagType/create': {
  316. component: dynamicWrapper(app, ['tagType', 'merchant'], () => import('../routes/Frontend/TagType/TagTypeCreate')),
  317. },
  318. '/frontend/tagType/edit/:id': {
  319. component: dynamicWrapper(app, ['tagType', 'merchant'], () => import('../routes/Frontend/TagType/TagTypeCreate')),
  320. },
  321. '/frontend/tag': {
  322. component: dynamicWrapper(app, ['tag'], () => import('../routes/Frontend/Tag')),
  323. },
  324. '/frontend/tag/list': {
  325. component: dynamicWrapper(app, ['tag'], () => import('../routes/Frontend/Tag/TagList')),
  326. },
  327. '/frontend/tag/create': {
  328. component: dynamicWrapper(app, ['tag', 'shelves'], () => import('../routes/Frontend/Tag/TagCreate')),
  329. },
  330. '/frontend/tag/edit/:id': {
  331. component: dynamicWrapper(app, ['tag', 'shelves'], () => import('../routes/Frontend/Tag/TagCreate')),
  332. },
  333. '/frontend/recommend': {
  334. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Frontend/Recommend')),
  335. },
  336. '/frontend/recommend/merchant-list': {
  337. component: dynamicWrapper(app, ['merchant'], () => import('../routes/Frontend/Recommend/RecommendList')),
  338. },
  339. '/frontend/recommend/course-edit/:id': {
  340. component: dynamicWrapper(app, ['merchant', 'shelves'], () => import('../routes/Frontend/Recommend/RecommendCourse')),
  341. },
  342. '/frontend/recommend/poster-edit/:id': {
  343. component: dynamicWrapper(app, ['merchant', 'shelves'], () => import('../routes/Frontend/Recommend/RecommendPoster')),
  344. },
  345. '/frontend/personalize': {
  346. component: dynamicWrapper(app, ['terminal'], () => import('../routes/Frontend/Personalize')),
  347. },
  348. '/frontend/personalize/list': {
  349. component: dynamicWrapper(app, ['terminal', 'campus', 'merchant'], () => import('../routes/Frontend/Personalize/PersonalizeList')),
  350. },
  351. '/frontend/personalize/edit/:id': {
  352. component: dynamicWrapper(app, ['terminal', 'tag', 'tagType', 'shelves'], () => import('../routes/Frontend/Personalize/PersonalizeEdit')),
  353. },
  354. '/trade/shopcart': {
  355. component: dynamicWrapper(app, [], () => import('../routes/Trade/ShopCart')),
  356. },
  357. '/trade/shopcart/list': {
  358. component: dynamicWrapper(app, ['terminal', 'campus', 'merchant'], () => import('../routes/Trade/ShopCart/ShopCartList')),
  359. },
  360. '/trade/shopcart/detail/:id': {
  361. component: dynamicWrapper(app, ['trade'], () => import('../routes/Trade/ShopCart/ShopCartDetail')),
  362. },
  363. '/trade/order': {
  364. component: dynamicWrapper(app, [], () => import('../routes/Trade/Order')),
  365. },
  366. '/trade/order/list': {
  367. component: dynamicWrapper(app, ['trade'], () => import('../routes/Trade/Order/OrderList')),
  368. },
  369. '/trade/order/create': {
  370. component: dynamicWrapper(app, ['trade', 'terminal', 'shelves'], () => import('../routes/Trade/Order/OrderCreate')),
  371. },
  372. '/trade/order/view/:id': {
  373. component: dynamicWrapper(app, ['trade'], () => import('../routes/Trade/Order/OrderDetail')),
  374. },
  375. '/trade/order/sub/:id': {
  376. component: dynamicWrapper(app, ['trade'], () => import('../routes/Trade/Order/SubOrderDetail')),
  377. },
  378. // 统计概览相关路由注册
  379. // '/dashboard/analysis': {
  380. // component: dynamicWrapper(app, ['chart'], () => import('../routes/Dashboard/Analysis')),
  381. // },
  382. '/dashboard/sold': {
  383. component: dynamicWrapper(app, ['trade'], () => import('../routes/Dashboard/SnapshotList')),
  384. },
  385. // 统计概览 贝尔安亲 需求增加
  386. '/dashboard/accounts': {
  387. component: dynamicWrapper(app, ['trade'], () => import('../routes/Dashboard/Accounts')),
  388. },
  389. // 总统计表
  390. '/dashboard/accounts/totalList': {
  391. component: dynamicWrapper(app, ['terminal', 'campus', 'merchant'], () => import('../routes/Dashboard/Accounts/AccountsTotalList')),
  392. },
  393. // 校区列表
  394. '/dashboard/accounts/campus': {
  395. component: dynamicWrapper(app, ['accounts'], () => import('../routes/Dashboard/Accounts/AccountsCampus')),
  396. },
  397. // 终端用户
  398. '/dashboard/accounts/terminals': {
  399. component: dynamicWrapper(app, ['accounts'], () => import('../routes/Dashboard/Accounts/AccountsTerminals')),
  400. },
  401. // 即将逾期
  402. '/dashboard/accounts/overdue': {
  403. component: dynamicWrapper(app, ['accounts'], () => import('../routes/Dashboard/Accounts/AccountsOverdue')),
  404. },
  405. // 异常相关路由注册
  406. '/exception/403': {
  407. component: dynamicWrapper(app, [], () => import('../routes/Exception/403')),
  408. },
  409. '/exception/404': {
  410. component: dynamicWrapper(app, [], () => import('../routes/Exception/404')),
  411. },
  412. '/exception/500': {
  413. component: dynamicWrapper(app, [], () => import('../routes/Exception/500')),
  414. },
  415. // 登录相关路由注册
  416. '/user': {
  417. component: dynamicWrapper(app, [], () => import('../layouts/UserLayout')),
  418. },
  419. '/user/login': {
  420. component: dynamicWrapper(app, ['login'], () => import('../routes/User/Login')),
  421. },
  422. // 帮助文档相关路由注册
  423. '/document/platform': {
  424. component: dynamicWrapper(app, [], () => import('../routes/Document/DocumentPlatform')),
  425. },
  426. };
  427. // Get name from ./menu.js or just set it in the router data.
  428. const menuData = getFlatMenuData(getMenuData());
  429. // Route configuration data
  430. // eg. {name,authority ...routerConfig }
  431. const routerData = {};
  432. // The route matches the menu
  433. Object.keys(routerConfig).forEach((path) => {
  434. // Regular match item name
  435. // eg. router /user/:id === /user/chen
  436. const pathRegexp = pathToRegexp(path);
  437. const menuKey = Object.keys(menuData).find(key => pathRegexp.test(`${key}`));
  438. let menuItem = {};
  439. // If menuKey is not empty
  440. if (menuKey) {
  441. menuItem = menuData[menuKey];
  442. }
  443. let router = routerConfig[path];
  444. // If you need to configure complex parameter routing,
  445. // 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
  446. // eg . /list/:type/user/info/:id
  447. router = {
  448. ...router,
  449. name: router.name || menuItem.name,
  450. authority: router.authority || menuItem.authority,
  451. };
  452. routerData[path] = router;
  453. });
  454. return routerData;
  455. };