vue2-client 1.15.84 → 1.15.86

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (377) hide show
  1. package/.claude/settings.local.json +12 -0
  2. package/.editorconfig +9 -9
  3. package/.env +20 -20
  4. package/.env.development +1 -1
  5. package/.env.iot +19 -19
  6. package/.env.message +19 -19
  7. package/.env.revenue +19 -19
  8. package/.env.runtime +19 -19
  9. package/.eslintrc.json +5 -5
  10. package/CLAUDE.md +89 -89
  11. package/README.md +65 -65
  12. package/babel.config.js +21 -21
  13. package/docs/Logic/345/207/275/346/225/260/344/275/277/347/224/250/347/233/270/345/205/263.md +46 -46
  14. package/docs/notice.md +22 -22
  15. package/docs//345/207/275/346/225/260/344/275/277/347/224/250/347/233/270/345/205/263.md +179 -179
  16. package/jest.config.js +22 -22
  17. package/package.json +111 -111
  18. package/src/App.vue +196 -196
  19. package/src/ReportView.js +13 -13
  20. package/src/base-client/components/common/AddressSearchCombobox/AddressSearchCombobox.vue +532 -532
  21. package/src/base-client/components/common/AddressSearchCombobox/index.js +3 -3
  22. package/src/base-client/components/common/AmapMarker/index.js +3 -3
  23. package/src/base-client/components/common/CitySelect/CitySelect.vue +376 -376
  24. package/src/base-client/components/common/CitySelect/demo.vue +43 -43
  25. package/src/base-client/components/common/ColorPickerCombobox/ColorPickerCombobox.vue +99 -99
  26. package/src/base-client/components/common/ColorPickerCombobox/demo.vue +34 -34
  27. package/src/base-client/components/common/FormGroupEdit/FormGroupEdit.vue +146 -146
  28. package/src/base-client/components/common/JSONToTree/index.js +3 -3
  29. package/src/base-client/components/common/Upload/Upload.vue +323 -323
  30. package/src/base-client/components/common/XAddForm/index.js +3 -3
  31. package/src/base-client/components/common/XAddForm/index.md +61 -61
  32. package/src/base-client/components/common/XAddNativeForm/XAddNativeForm.vue +1169 -1169
  33. package/src/base-client/components/common/XAddNativeForm/demo.vue +54 -40
  34. package/src/base-client/components/common/XAddNativeForm/index.js +3 -3
  35. package/src/base-client/components/common/XAddReport/XAddReport.vue +212 -212
  36. package/src/base-client/components/common/XBadge/index.js +3 -3
  37. package/src/base-client/components/common/XBadge/index.md +39 -39
  38. package/src/base-client/components/common/XButtons/XButtons.vue +284 -284
  39. package/src/base-client/components/common/XCalendar/XCalendar.vue +369 -369
  40. package/src/base-client/components/common/XCalendar/index.md +284 -284
  41. package/src/base-client/components/common/XCard/index.js +3 -3
  42. package/src/base-client/components/common/XCard/index.md +43 -43
  43. package/src/base-client/components/common/XCardSet/XCardSet.vue +300 -300
  44. package/src/base-client/components/common/XCollapse/XCollapse.vue +354 -354
  45. package/src/base-client/components/common/XConversation/XConversation.vue +576 -576
  46. package/src/base-client/components/common/XConversation/XConversationDemo.vue +28 -28
  47. package/src/base-client/components/common/XDataCard/XDataCard.vue +629 -629
  48. package/src/base-client/components/common/XDatePicker/index.vue +276 -276
  49. package/src/base-client/components/common/XDescriptions/XDescriptions.vue +174 -174
  50. package/src/base-client/components/common/XDescriptions/XDescriptionsGroup.vue +314 -314
  51. package/src/base-client/components/common/XDescriptions/demo.vue +51 -51
  52. package/src/base-client/components/common/XForm/XForm.vue +420 -420
  53. package/src/base-client/components/common/XForm/XFormItem.vue +1474 -1390
  54. package/src/base-client/components/common/XForm/XTreeSelect.vue +276 -276
  55. package/src/base-client/components/common/XForm/demo.vue +105 -105
  56. package/src/base-client/components/common/XForm/index.js +3 -3
  57. package/src/base-client/components/common/XFormCol/XFormCol.vue +157 -157
  58. package/src/base-client/components/common/XFormCol/index.js +3 -3
  59. package/src/base-client/components/common/XFormCol/index.md +35 -35
  60. package/src/base-client/components/common/XFormGroup/XFormGroup.vue +301 -301
  61. package/src/base-client/components/common/XFormGroup/demo.vue +41 -41
  62. package/src/base-client/components/common/XFormTable/XFormTable.vue +938 -938
  63. package/src/base-client/components/common/XFormTable/demo.vue +87 -87
  64. package/src/base-client/components/common/XFormTable/index.js +3 -3
  65. package/src/base-client/components/common/XImportExcel/XImportExcel.vue +174 -174
  66. package/src/base-client/components/common/XImportExcel/index.js +3 -3
  67. package/src/base-client/components/common/XImportExcel/index.md +38 -38
  68. package/src/base-client/components/common/XInput/XInput.vue +128 -128
  69. package/src/base-client/components/common/XInput/index.js +3 -3
  70. package/src/base-client/components/common/XInput/index.md +97 -97
  71. package/src/base-client/components/common/XIntervalPicker/XIntervalPicker.vue +121 -121
  72. package/src/base-client/components/common/XLicensePlate/index.js +3 -3
  73. package/src/base-client/components/common/XLicensePlate/index.md +38 -38
  74. package/src/base-client/components/common/XPrint/Demo.vue +41 -41
  75. package/src/base-client/components/common/XPrint/PrintBill.vue +308 -308
  76. package/src/base-client/components/common/XRate/demo.vue +102 -102
  77. package/src/base-client/components/common/XRate/index.vue +149 -149
  78. package/src/base-client/components/common/XReport/XReport.vue +963 -963
  79. package/src/base-client/components/common/XReport/XReportDemo.vue +70 -70
  80. package/src/base-client/components/common/XReport/XReportTrGroup.vue +1005 -1005
  81. package/src/base-client/components/common/XReport/index.md +103 -103
  82. package/src/base-client/components/common/XReportDrawer/XReportDrawer.vue +201 -201
  83. package/src/base-client/components/common/XReportGrid/XReport.vue +1075 -1075
  84. package/src/base-client/components/common/XReportGrid/XReportDemo.vue +44 -44
  85. package/src/base-client/components/common/XReportGrid/XReportDesign.vue +620 -620
  86. package/src/base-client/components/common/XReportGrid/XReportTrGroup.vue +723 -723
  87. package/src/base-client/components/common/XReportGrid/print.js +184 -184
  88. package/src/base-client/components/common/XTab/XTab.vue +299 -299
  89. package/src/base-client/components/common/XTable/ExportExcel.vue +283 -283
  90. package/src/base-client/components/common/XTable/XTable.vue +1599 -1599
  91. package/src/base-client/components/common/XTable/XTableWrapper.vue +597 -597
  92. package/src/base-client/components/common/XTable/index.js +3 -3
  93. package/src/base-client/components/common/XTimeline/XTimeline.vue +358 -358
  94. package/src/base-client/components/common/XTimeline/index.md +191 -191
  95. package/src/base-client/components/common/XTree/XTreePro.vue +452 -452
  96. package/src/base-client/components/common/XTreeOne/index.js +3 -3
  97. package/src/base-client/components/common/XUploadFilesView/index.vue +485 -485
  98. package/src/base-client/components/his/XCharge/XCharge.vue +238 -238
  99. package/src/base-client/components/his/XCheckbox/XCheckbox.vue +105 -105
  100. package/src/base-client/components/his/XCheckbox/index.md +253 -253
  101. package/src/base-client/components/his/XHDescriptions/XHDescriptions.vue +430 -430
  102. package/src/base-client/components/his/XHDescriptions/index.md +217 -217
  103. package/src/base-client/components/his/XHisEditor/XHisEditor.vue +629 -629
  104. package/src/base-client/components/his/XHisEditor/diagnosisAutocomplete.js +263 -263
  105. package/src/base-client/components/his/XList/XList.vue +495 -495
  106. package/src/base-client/components/his/XQuestionnaire/XQuestionnaire.json +3 -3
  107. package/src/base-client/components/his/XQuestionnaire/XQuestionnaire.vue +106 -106
  108. package/src/base-client/components/his/XQuestionnaire/XQuestionnaireDemo.vue +51 -51
  109. package/src/base-client/components/his/XQuestionnaire/XQuestionnaireItem.vue +269 -269
  110. package/src/base-client/components/his/XRadio/XRadio.vue +125 -125
  111. package/src/base-client/components/his/XRadio/index.md +234 -234
  112. package/src/base-client/components/his/XSelect/XSelect.vue +72 -72
  113. package/src/base-client/components/his/XShiftSchedule/XShiftSchedule.vue +234 -234
  114. package/src/base-client/components/his/XShiftSchedule/dome.vue +29 -29
  115. package/src/base-client/components/his/XSidebar/XSidebar.vue +240 -240
  116. package/src/base-client/components/his/XTextCard/XTextCard.vue +207 -207
  117. package/src/base-client/components/his/XTimeSelect/XTimeSelect.vue +162 -162
  118. package/src/base-client/components/his/XTimeSelect/XTimeSelectDemo.vue +23 -23
  119. package/src/base-client/components/his/XTitle/README.md +113 -113
  120. package/src/base-client/components/his/XTitle/XTitle.vue +123 -123
  121. package/src/base-client/components/his/XTreeRows/TreeNode.vue +107 -107
  122. package/src/base-client/components/his/XTreeRows/XTreeRows.vue +307 -307
  123. package/src/base-client/components/his/threeTestOrders/dome.vue +68 -68
  124. package/src/base-client/components/his/threeTestOrders/editor.vue +111 -111
  125. package/src/base-client/components/his/threeTestOrders/textBox.vue +457 -457
  126. package/src/base-client/components/his/threeTestOrders/threeTestOrders.vue +475 -475
  127. package/src/base-client/components/layout/XPageView/RenderRow.vue +88 -88
  128. package/src/base-client/components/layout/XPageView/XErrorView.vue +22 -22
  129. package/src/base-client/components/layout/XPageView/XPageRowTemplate.vue +37 -37
  130. package/src/base-client/components/layout/XPageView/XPageView.vue +223 -223
  131. package/src/base-client/components/layout/XPageView/XTab/XTab.vue +96 -96
  132. package/src/base-client/components/layout/XPageView/XTab/index.js +3 -3
  133. package/src/base-client/components/layout/XPageView/componentTypes.js +22 -22
  134. package/src/base-client/components/layout/XPageView/index.js +2 -2
  135. package/src/base-client/components/layout/XPageView/index.md +96 -96
  136. package/src/base-client/components/system/DictionaryDetailsView/index.js +3 -3
  137. package/src/base-client/components/system/DictionaryDetailsView/index.md +41 -41
  138. package/src/base-client/components/system/LogDetailsView/LogDetailsView.vue +376 -376
  139. package/src/base-client/components/system/LogDetailsView/index.js +3 -3
  140. package/src/base-client/components/system/LogDetailsView/index.md +41 -41
  141. package/src/base-client/components/system/QueryParamsDetailsView/index.js +3 -3
  142. package/src/base-client/components/ticket/TicketDetailsView/TicketDetailsView.vue +807 -807
  143. package/src/base-client/components/ticket/TicketDetailsView/index.js +3 -3
  144. package/src/base-client/components/ticket/TicketDetailsView/index.md +29 -29
  145. package/src/base-client/components/ticket/TicketDetailsView/part/TicketDetailsFlow.vue +260 -260
  146. package/src/base-client/components/ticket/TicketDetailsView/part/index.js +3 -3
  147. package/src/base-client/components/ticket/TicketSubmitSuccessView/TicketSubmitSuccessView.vue +532 -532
  148. package/src/base-client/components/ticket/TicketSubmitSuccessView/index.js +3 -3
  149. package/src/base-client/components/ticket/TicketSubmitSuccessView/index.md +29 -29
  150. package/src/base-client/plugins/AppData.js +126 -126
  151. package/src/base-client/plugins/PagedList.js +177 -177
  152. package/src/base-client/plugins/__tests__/selectValueTypeHelper.test.js +154 -0
  153. package/src/base-client/plugins/authority-plugin.js +167 -167
  154. package/src/base-client/plugins/compatible/LoginServiceOA.js +20 -20
  155. package/src/base-client/plugins/i18n-extend.js +32 -32
  156. package/src/base-client/plugins/moment.js +8 -8
  157. package/src/base-client/plugins/selectValueTypeHelper.js +281 -0
  158. package/src/bootstrap.js +51 -51
  159. package/src/components/Ellipsis/Ellipsis.vue +65 -65
  160. package/src/components/Ellipsis/index.js +3 -3
  161. package/src/components/Ellipsis/index.md +38 -38
  162. package/src/components/FileImageItem/FileItem.vue +320 -320
  163. package/src/components/FileImageItem/FileItemGroup.vue +197 -197
  164. package/src/components/FileImageItem/ImageItem.vue +107 -107
  165. package/src/components/FileImageItem/index.js +4 -4
  166. package/src/components/FilePreview/FilePreview.vue +181 -181
  167. package/src/components/FilePreview/FilePreviewDemo.vue +30 -30
  168. package/src/components/FilePreview/index.js +3 -3
  169. package/src/components/HeightScanner/index.vue +615 -615
  170. package/src/components/STable/README.md +341 -341
  171. package/src/components/STable/index.js +558 -558
  172. package/src/components/TableSetting/TableSetting.vue +143 -143
  173. package/src/components/TableSetting/index.js +3 -3
  174. package/src/components/Trend/Trend.vue +41 -41
  175. package/src/components/Trend/index.js +3 -3
  176. package/src/components/Trend/index.less +41 -41
  177. package/src/components/Trend/index.md +45 -45
  178. package/src/components/_util/util.js +46 -46
  179. package/src/components/cache/AKeepAlive.js +179 -179
  180. package/src/components/exception/typeConfig.js +19 -19
  181. package/src/components/form/FormRow.vue +52 -52
  182. package/src/components/index.less +5 -5
  183. package/src/components/menu/Contextmenu.vue +84 -84
  184. package/src/components/menu/index.less +38 -38
  185. package/src/components/page/header/PageHeader.vue +64 -64
  186. package/src/components/page/header/index.less +40 -40
  187. package/src/components/result/Result.vue +77 -77
  188. package/src/components/setting/SettingItem.vue +26 -26
  189. package/src/components/setting/i18n.js +117 -117
  190. package/src/components/table/StandardTable.vue +141 -141
  191. package/src/components/table/advance/ActionColumns.vue +158 -158
  192. package/src/components/table/advance/ActionSize.vue +45 -45
  193. package/src/components/table/advance/AdvanceTable.vue +275 -275
  194. package/src/components/table/advance/SearchArea.vue +355 -355
  195. package/src/components/table/advance/index.js +2 -2
  196. package/src/components/table/api/ApiTable.vue +50 -50
  197. package/src/components/task/TaskGroup.vue +80 -80
  198. package/src/components/task/TaskItem.vue +26 -26
  199. package/src/components/tool/AvatarList.vue +68 -68
  200. package/src/components/tool/DetailList.vue +157 -157
  201. package/src/components/tool/Drawer.vue +142 -142
  202. package/src/components/tool/FooterToolBar.vue +30 -30
  203. package/src/components/tool/HeadInfo.vue +35 -35
  204. package/src/components/tool/TagSelect.vue +83 -83
  205. package/src/components/tool/TagSelectOption.vue +33 -33
  206. package/src/components/transition/PageToggleTransition.vue +97 -97
  207. package/src/config/default/admin.config.js +18 -18
  208. package/src/config/default/animate.config.js +21 -21
  209. package/src/config/default/index.js +6 -6
  210. package/src/config/index.js +3 -3
  211. package/src/config/replacer/index.js +10 -10
  212. package/src/config/replacer/resolve.config.js +67 -67
  213. package/src/expression/ExpressionRunner.js +26 -26
  214. package/src/expression/TestExpression.js +509 -509
  215. package/src/expression/core/Delegate.js +115 -115
  216. package/src/expression/core/Expression.js +1358 -1358
  217. package/src/expression/core/Program.js +932 -932
  218. package/src/expression/core/Token.js +27 -27
  219. package/src/expression/enums/ExpressionType.js +81 -81
  220. package/src/expression/enums/TokenType.js +11 -11
  221. package/src/expression/exception/BreakWayException.js +2 -2
  222. package/src/expression/exception/ContinueWayException.js +2 -2
  223. package/src/expression/exception/ExpressionException.js +28 -28
  224. package/src/expression/exception/ReturnWayException.js +14 -14
  225. package/src/expression/exception/ServiceException.js +22 -22
  226. package/src/expression/instances/LogicConsole.js +44 -44
  227. package/src/expression/ts/ExpressionRunner.ts +28 -28
  228. package/src/expression/ts/TestExpression.ts +509 -509
  229. package/src/expression/ts/core/Delegate.ts +114 -114
  230. package/src/expression/ts/core/Expression.ts +1309 -1309
  231. package/src/expression/ts/core/Program.ts +950 -950
  232. package/src/expression/ts/core/Token.ts +29 -29
  233. package/src/expression/ts/enums/ExpressionType.ts +81 -81
  234. package/src/expression/ts/enums/TokenType.ts +13 -13
  235. package/src/expression/ts/exception/BreakWayException.ts +2 -2
  236. package/src/expression/ts/exception/ContinueWayException.ts +2 -2
  237. package/src/expression/ts/exception/ExpressionException.ts +28 -28
  238. package/src/expression/ts/exception/ReturnWayException.ts +14 -14
  239. package/src/expression/ts/exception/ServiceException.ts +22 -22
  240. package/src/expression/ts/instances/JSONArray.ts +48 -48
  241. package/src/expression/ts/instances/JSONObject.ts +109 -109
  242. package/src/expression/ts/instances/LogicConsole.ts +32 -32
  243. package/src/layouts/AdminLayout.vue +176 -176
  244. package/src/layouts/BlankView.vue +79 -79
  245. package/src/layouts/ComponentLayoutOne.vue +47 -47
  246. package/src/layouts/GridView.vue +43 -43
  247. package/src/layouts/PageView.vue +55 -55
  248. package/src/layouts/footer/PageFooter.vue +49 -49
  249. package/src/layouts/header/HeaderAvatar.vue +64 -64
  250. package/src/layouts/header/HeaderSearch.vue +67 -67
  251. package/src/layouts/header/index.less +92 -92
  252. package/src/layouts/tabs/TabsView.vue +383 -383
  253. package/src/layouts/tabs/i18n.js +25 -25
  254. package/src/layouts/tabs/index.js +2 -2
  255. package/src/logic/LogicRunner.js +62 -62
  256. package/src/logic/TestLogic.js +13 -13
  257. package/src/logic/plugins/common/DateTools.js +35 -35
  258. package/src/logic/plugins/common/VueTools.js +30 -30
  259. package/src/logic/plugins/index.js +7 -7
  260. package/src/logic/ts/LogicRunner.ts +67 -67
  261. package/src/logic/ts/TestLogic.ts +13 -13
  262. package/src/main.js +33 -33
  263. package/src/mixins/formValidationMixin.js +46 -46
  264. package/src/mock/common/activityData.js +32 -32
  265. package/src/mock/common/index.js +89 -89
  266. package/src/mock/common/reportData.js +20 -20
  267. package/src/mock/common/tableData.js +118 -118
  268. package/src/mock/index.js +12 -12
  269. package/src/mock/project/index.js +17 -17
  270. package/src/mock/user/current.js +13 -13
  271. package/src/mock/user/login.js +39 -39
  272. package/src/mock/user/routes.js +61 -61
  273. package/src/mock/workplace/index.js +15 -15
  274. package/src/pages/LogicCallExample/index.vue +46 -46
  275. package/src/pages/ReportGrid/index.vue +76 -76
  276. package/src/pages/ReportView.vue +50 -50
  277. package/src/pages/WorkflowDetail/WorkFlowDemo.vue +47 -47
  278. package/src/pages/WorkflowDetail/WorkFlowDemo2.vue +204 -204
  279. package/src/pages/WorkflowDetail/WorkFlowDemo3.vue +203 -203
  280. package/src/pages/WorkflowDetail/WorkflowDetail.vue +391 -391
  281. package/src/pages/WorkflowDetail/WorkflowPageDetail/LeaveMessage.vue +388 -388
  282. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowBaseInformation.vue +415 -415
  283. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowHandle.vue +1766 -1765
  284. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowHandleReso.vue +975 -975
  285. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowPreview.vue +109 -109
  286. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowTimeline.vue +929 -929
  287. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkOrderParentDetails.vue +222 -222
  288. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkflowDetailResso.vue +243 -243
  289. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkflowLog.vue +188 -188
  290. package/src/pages/WorkflowDetail/WorkflowPageDetail/components/WorkflowPersonSelector.vue +109 -109
  291. package/src/pages/WorkflowDetail/WorkflowPageDetail/worklog.vue +97 -97
  292. package/src/pages/XPageViewExample/index.vue +149 -149
  293. package/src/pages/addressSelect/addressDemo.vue +24 -24
  294. package/src/pages/addressSelect/index.vue +270 -270
  295. package/src/pages/dashboard/workplace/i18n.js +40 -40
  296. package/src/pages/dashboard/workplace/index.js +2 -2
  297. package/src/pages/dashboard/workplace/index.less +59 -59
  298. package/src/pages/exception/403.vue +21 -21
  299. package/src/pages/exception/404.vue +24 -24
  300. package/src/pages/exception/500.vue +21 -21
  301. package/src/pages/login/index.js +2 -2
  302. package/src/pages/report/ReportTableHome.vue +28 -28
  303. package/src/pages/resourceManage/depListManage.vue +23 -23
  304. package/src/pages/resourceManage/funListManage.vue +23 -23
  305. package/src/pages/resourceManage/index.js +15 -15
  306. package/src/pages/resourceManage/resourceManageMain.vue +57 -57
  307. package/src/pages/resourceManage/roleListManage.vue +23 -23
  308. package/src/pages/resourceManage/staffListManage.vue +23 -23
  309. package/src/pages/system/file/Info.vue +56 -56
  310. package/src/pages/system/file/index.vue +317 -317
  311. package/src/pages/system/settings/index.vue +126 -126
  312. package/src/pages/userInfoDetailManage/FillCardRecordQuery/index.vue +77 -77
  313. package/src/pages/userInfoDetailManage/FillGasRecordQuery/index.vue +75 -75
  314. package/src/pages/userInfoDetailManage/InsuranceDetailQuery/index.vue +64 -64
  315. package/src/pages/userInfoDetailManage/MachineRecordQuery/index.vue +75 -75
  316. package/src/pages/userInfoDetailManage/OtherChargeRecordQuery/index.vue +75 -75
  317. package/src/pages/userInfoDetailManage/PriceAdjustments/index.vue +64 -64
  318. package/src/pages/userInfoDetailManage/UserChargeRecordQuery/index.vue +94 -94
  319. package/src/pages/userInfoDetailManage/UserException/index.vue +64 -64
  320. package/src/pages/userInfoDetailManage/UserHandRecordQuery/index.vue +87 -87
  321. package/src/pages/userInfoDetailManage/UserRecordQuery/index.vue +74 -74
  322. package/src/pages/userInfoDetailManage/index.vue +290 -290
  323. package/src/pages/userInfoDetailManage/uploadFilesHistory/ImagePreview.vue +101 -101
  324. package/src/pages/userInfoDetailManage/uploadFilesHistory/index.vue +129 -129
  325. package/src/pages/userInfoDetailManage/userInfoDetailQueryTabs.vue +144 -144
  326. package/src/plugins/HiPrintPlugin.js +164 -164
  327. package/src/router/async/router.map.js +126 -126
  328. package/src/router/guards.js +262 -262
  329. package/src/router/i18n.js +57 -57
  330. package/src/router.js +17 -17
  331. package/src/services/api/DictionaryDetailsViewApi.js +6 -6
  332. package/src/services/api/LogDetailsViewApi.js +10 -10
  333. package/src/services/api/QueryParamsDetailsViewApi.js +6 -6
  334. package/src/services/api/logininfor/index.js +6 -6
  335. package/src/services/api/manage.js +8 -8
  336. package/src/services/api/restTools.js +215 -215
  337. package/src/services/api/workFlow.js +57 -57
  338. package/src/services/apiService.js +16 -16
  339. package/src/services/dataSource.js +12 -12
  340. package/src/services/index.js +7 -7
  341. package/src/services/user.js +92 -92
  342. package/src/services/v3Api.js +116 -116
  343. package/src/store/index.js +5 -5
  344. package/src/store/mutation-types.js +4 -4
  345. package/src/theme/antd/ant-menu.less +2 -2
  346. package/src/theme/antd/ant-message.less +3 -3
  347. package/src/theme/antd/ant-table.less +22 -22
  348. package/src/theme/antd/ant-time-picker.less +3 -3
  349. package/src/theme/antd/index.less +3 -3
  350. package/src/theme/default/color.less +43 -43
  351. package/src/theme/default/index.less +3 -3
  352. package/src/theme/default/nprogress.less +76 -76
  353. package/src/theme/global.less +279 -279
  354. package/src/theme/index.less +5 -5
  355. package/src/theme/reportTable.less +58 -58
  356. package/src/theme/theme.less +1 -1
  357. package/src/utils/EncryptUtil.js +162 -162
  358. package/src/utils/Objects.js +25 -25
  359. package/src/utils/axios-interceptors.js +100 -100
  360. package/src/utils/colors.js +107 -107
  361. package/src/utils/common.js +10 -10
  362. package/src/utils/excel/Blob.js +180 -180
  363. package/src/utils/excel/Export2Excel.js +141 -141
  364. package/src/utils/filter.js +21 -21
  365. package/src/utils/i18n.js +80 -80
  366. package/src/utils/indexedDB.js +549 -549
  367. package/src/utils/microAppUtils.js +49 -49
  368. package/src/utils/request.js +395 -395
  369. package/src/utils/routerUtil.js +553 -553
  370. package/src/utils/themeUtil.js +100 -100
  371. package/test/Tree.spec.js +168 -168
  372. package/test/myDialog.spec.js +47 -47
  373. package/test/request.test.js +17 -17
  374. package/test/util.test.js +53 -53
  375. package/test/v3Api.test.js +1984 -1984
  376. package/tests/unit/ReportTable.spec.js +16 -16
  377. package/vue.config.js +222 -222
@@ -1,963 +1,963 @@
1
- <template>
2
- <div>
3
- <!-- 骨架屏 -->
4
- <a-card v-if="showSkeleton">
5
- <a-skeleton active/>
6
- </a-card>
7
- <template v-if="noPadding">
8
- <!-- 主体表格 -->
9
- <XReportDesign
10
- @updateImg="updateImg"
11
- v-if="scanFinish"
12
- :show-img-in-cell="showImgInCell"
13
- :img-prefix="imgPrefix"
14
- :use-oss-for-img="useOssForImg"
15
- :display-only="displayOnly"
16
- :config="type === 'display' ? originalConfig : activeConfig"
17
- :slot-config-name="type === 'display' ? undefined : activatedSlotName"
18
- :for-display="type === 'display'"
19
- ref="XReportDesign"
20
- id="printReady"
21
- :server-name="serverName"
22
- :show-title="showTitle"
23
- :no-padding="noPadding"
24
- :no-top-border="noTopBorder"
25
- :show-images="hasImages"
26
- :image-list="imageList">
27
- </XReportDesign>
28
- </template>
29
- <template v-else>
30
- <!-- 用以包裹整个页面 -->
31
- <a-card v-if="!showSkeleton">
32
- <!-- 切换菜单 -->
33
- <a-radio-group
34
- v-model="type"
35
- default-value="a"
36
- button-style="solid"
37
- @change="tabChanged"
38
- v-show="!onlyDisplay && editMode">
39
- <a-radio-button value="design" v-if="!onlyDisplay">
40
- 设计
41
- </a-radio-button>
42
- <a-radio-button value="display" style="border-radius: 0">
43
- 预览
44
- </a-radio-button>
45
- </a-radio-group>
46
- <a-radio-button @click="saveConfig" style="border-radius: 0 4px 4px 0" v-if="showSaveButton">
47
- 保存
48
- </a-radio-button>
49
- <!-- 主体表格 -->
50
- <XReportDesign
51
- v-if="scanFinish"
52
- @updateImg="updateImg"
53
- :show-img-in-cell="showImgInCell"
54
- :img-prefix="imgPrefix"
55
- :use-oss-for-img="useOssForImg"
56
- :display-only="displayOnly"
57
- :config="type === 'display' ? originalConfig : activeConfig"
58
- :slot-config-name="type === 'display' ? undefined : activatedSlotName"
59
- :for-display="type === 'display'"
60
- :no-padding="noPadding"
61
- :no-top-border="noTopBorder"
62
- :show-title="showTitle"
63
- ref="XReportDesign"
64
- id="printReady"
65
- :server-name="serverName"
66
- :show-images="hasImages"
67
- :image-list="imageList">
68
- </XReportDesign>
69
- <!-- 导出按钮 -->
70
- <template v-if="type === 'display'">
71
- <div class="tools">
72
- <div class="toolsItem">
73
- <a-space>
74
- <a-button type="primary" @click="printDocument">
75
- 打印
76
- </a-button>
77
- <a-button @click="exportPDF">
78
- 导出PDF
79
- </a-button>
80
- </a-space>
81
- </div>
82
- </div>
83
- </template>
84
- </a-card>
85
- </template>
86
- </div>
87
- </template>
88
-
89
- <script>
90
- // 转PDF用
91
- import HtmlToPdf from '@vue2-client/utils/htmlToPDF'
92
- import { getConfigByName } from '@vue2-client/services/api/common'
93
- import XReportDesign from '@vue2-client/base-client/components/common/XReport/XReportDesign.vue'
94
- import { printElement } from './print'
95
-
96
- export default {
97
- name: 'XReport',
98
- props: {
99
- files: {
100
- type: Array,
101
- default: () => {
102
- return []
103
- }
104
- },
105
- // 控制用户权限,user和admin
106
- authority: {
107
- type: String,
108
- default: 'user'
109
- },
110
- // 是否为编辑模式
111
- editMode: {
112
- type: Boolean,
113
- default: true
114
- },
115
- // 配置名
116
- configName: {
117
- type: String,
118
- required: true
119
- },
120
- // 插槽名
121
- activatedSlotName: {
122
- type: String,
123
- default: undefined
124
- },
125
- // 本地配置,调试用
126
- localConfig: {
127
- type: Object,
128
- default: undefined
129
- },
130
- // 兼容老版本配置
131
- dontFormat: {
132
- type: Boolean,
133
- default: true
134
- },
135
- showImgInCell: {
136
- type: Boolean,
137
- default: false
138
- },
139
- // 数据
140
- configData: {
141
- type: Object,
142
- default: undefined
143
- },
144
- // 命名空间
145
- serverName: {
146
- type: String,
147
- default: process.env.VUE_APP_SYSTEM_NAME
148
- },
149
- // 只做展示
150
- displayOnly: {
151
- type: Boolean,
152
- default: false
153
- },
154
- // 表格没有边距
155
- noPadding: {
156
- type: Boolean,
157
- default: false
158
- },
159
- // 表格没有上边框,与noPadding搭配可以实现连续表格
160
- noTopBorder: {
161
- type: Boolean,
162
- default: false
163
- },
164
- // 是否展示标题
165
- showTitle: {
166
- type: Boolean,
167
- default: true
168
- },
169
- // 是否展示保存按钮
170
- showSaveButton: {
171
- type: Boolean,
172
- default: true
173
- },
174
- // 是否将组件注册到外层提供的容器中,方便外侧统一保存
175
- registerMap: {
176
- type: Array,
177
- default: undefined
178
- },
179
- // 图片是否使用OSS来保存
180
- useOssForImg: {
181
- type: Boolean,
182
- default: true
183
- },
184
- // 图片上传后添加前缀
185
- imgPrefix: {
186
- type: String,
187
- default: undefined
188
- }
189
- },
190
- components: {
191
- XReportDesign
192
- },
193
- data () {
194
- return {
195
- // 控制骨架屏显隐
196
- showSkeleton: true,
197
- // 配置
198
- config: undefined,
199
- // 当前显示模式,编辑模式,预览模式
200
- type: 'design',
201
- // 仅供展示,不可编辑
202
- onlyDisplay: false,
203
- // 每行最大列数,非必要请勿更改,现在的设计器完全是基于每行12列来设计的
204
- maxColSpan: 12,
205
- // 定义是否完成配置的扫描,未完成不要渲染子组件
206
- scanFinish: false,
207
- // 当前激活的配置文件
208
- activeConfig: null,
209
- // 原始配置文件
210
- // 用于展示。某些情况下“设计页”中的内容仅为“预览页”表格其中的一部分
211
- originalConfig: null,
212
- // 扫描到的配置
213
- configFromWeb: {},
214
- // 用于获取配置的锁
215
- timer: undefined,
216
- // 是否包含图片
217
- hasImages: false,
218
- // 图片列表
219
- imageList: [],
220
- // 保存最原始的数据,用于判断哪些数据被更改了
221
- dataCache: undefined,
222
- // 判断哪些数据被更改了,存储对应的key
223
- diff: []
224
- }
225
- },
226
- watch: {
227
- // 如果配置名更改了,重新获取配置
228
- configName (val) {
229
- if (val) {
230
- getConfigByName(this.configName, undefined, res => {
231
- this.config = res
232
- this.configInit()
233
- })
234
- }
235
- },
236
- // 如果本地配置更改了,重新初始化
237
- localConfig (val) {
238
- if (val) {
239
- this.config = val
240
- this.configInit()
241
- }
242
- }
243
- },
244
- methods: {
245
- // 向外暴露图片修改后的数据,某些外部需要自己管理图片的保存与修改
246
- updateImg (data) {
247
- this.$emit('updateImg', data)
248
- },
249
- // 导出数据,某些外部需要统一控制数据的变动
250
- exportData (skipValidation = false) {
251
- // 如果不跳过校验,先进行必填字段校验
252
- if (!skipValidation) {
253
- const validation = this.validateRequiredFields()
254
- if (!validation.isValid) {
255
- const errorMessages = validation.errors.map(error => error.message).join(';')
256
- console.warn(`数据导出警告:存在必填字段未填写 - ${errorMessages}`)
257
- // 在导出时只警告,不阻止导出
258
- }
259
- }
260
-
261
- // 获取当前修改后的数据
262
- let tempData
263
- if (this.activeConfig === undefined || this.activeConfig === null) {
264
- tempData = this.originalConfig.data
265
- } else {
266
- const tempDataKeys = Object.keys(this.activeConfig.tempData)
267
- tempDataKeys.forEach(key => {
268
- this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
269
- })
270
- tempData = this.activeConfig.data
271
- }
272
- // 对比数据的差异
273
- this.diff = []
274
- this.compareProps(tempData, this.dataCache)
275
- this.diff.forEach(eachDiff => {
276
- const arr = eachDiff.split('.')
277
- let targetData = tempData[arr[0]]
278
- if (arr.length !== 1) {
279
- for (let i = 1; i < arr.length - 1; i++) {
280
- const path = arr[i]
281
- targetData = targetData[path]
282
- }
283
- }
284
- // 将修改的数据,添加update = true属性
285
- targetData.update = true
286
- })
287
- return tempData
288
- },
289
- // 对比两个obj有哪里不同
290
- compareProps (obj1, obj2, path = '') {
291
- for (const key in obj1) {
292
- // 如果一个是undefined
293
- if (typeof obj2[key] === 'undefined') {
294
- this.diff.push(path + key)
295
- // 如果是数组长度不一样
296
- } else if (Array.isArray(obj1) && Array.isArray(obj2)) {
297
- if (obj1[key].length !== obj2[key].length) {
298
- this.diff.push(path + key)
299
- }
300
- // 如果都是对象,并且存在同样的key,递归子key
301
- } else if (typeof obj1[key] === 'object' && typeof obj2[key] === 'object') {
302
- this.compareProps(obj1[key], obj2[key], path + key + '.')
303
- // 如果不是obj,对比其数据
304
- } else if (obj1[key] !== obj2[key]) {
305
- this.diff.push(path + key)
306
- }
307
- }
308
- },
309
- // 验证必填字段
310
- validateRequiredFields () {
311
- const errors = []
312
- const config = this.type === 'display' ? this.originalConfig : this.activeConfig
313
-
314
- if (!config || !config.columns) {
315
- return { isValid: true, errors: [] }
316
- }
317
-
318
- // 遍历所有配置项检查必填字段
319
- config.columns.forEach((row, rowIndex) => {
320
- row.forEach((cell, cellIndex) => {
321
- if (cell.required && cell.dataIndex) {
322
- let value
323
- if (cell.dataIndex.indexOf('@@@') !== -1) {
324
- // 处理深层嵌套数据
325
- // const arr = cell.dataIndex.split('@@@')
326
- value = config.tempData && config.tempData[cell.dataIndex]
327
- } else {
328
- value = config.data[cell.dataIndex]
329
- }
330
-
331
- // 检查值是否为空
332
- if (!value || (typeof value === 'string' && value.trim() === '')) {
333
- const message = cell.requiredMessage || this.getDefaultRequiredMessage(cell.type)
334
- errors.push({
335
- dataIndex: cell.dataIndex,
336
- message: message,
337
- rowIndex: rowIndex,
338
- cellIndex: cellIndex,
339
- type: cell.type
340
- })
341
- }
342
- }
343
- })
344
- })
345
-
346
- return {
347
- isValid: errors.length === 0,
348
- errors: errors
349
- }
350
- },
351
- // 获取默认的必填提示信息
352
- getDefaultRequiredMessage (fieldType) {
353
- const defaultMessages = {
354
- datePicker: '请选择日期',
355
- timePicker: '请选择日期时间',
356
- input: '请填写此字段',
357
- inputs: '请填写此字段'
358
- }
359
- return defaultMessages[fieldType] || '此字段为必填项'
360
- },
361
- // 正常的保存方法,当前修改内容会直接全部导出到外部
362
- saveConfig () {
363
- // 先进行必填字段校验
364
- const validation = this.validateRequiredFields()
365
- if (!validation.isValid) {
366
- const errorMessages = validation.errors.map(error => error.message).join(';')
367
- this.$message.error(`保存失败:${errorMessages}`)
368
- return false
369
- }
370
-
371
- if (this.activeConfig === undefined || this.activeConfig === null) {
372
- return this.originalConfig.data
373
- } else {
374
- const tempDataKeys = Object.keys(this.activeConfig.tempData)
375
- tempDataKeys.forEach(key => {
376
- this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
377
- })
378
- this.$emit('saveConfig', this.$refs.XReportDesign.activatedConfig)
379
- return true
380
- }
381
- },
382
- // 通过@@@分割临时变量,找到对应的key,并修改它的值
383
- changeDeepObject (obj, strPath, newVal) {
384
- const arr = strPath.split('@@@')
385
- if (obj[arr[0]] === undefined) {
386
- obj = obj.images
387
- }
388
- if (arr.length === 1) {
389
- obj[arr[0]] = newVal
390
- } else {
391
- let result = obj[arr[0]]
392
- arr.shift()
393
- while (arr.length > 1) {
394
- result = result[arr[0]]
395
- arr.shift()
396
- }
397
- if (result) {
398
- result[arr[0]] = newVal
399
- }
400
- }
401
- },
402
- // 检查slot是否在配置文件中包含,如果没有包含,则视为非法获取
403
- checkSlotDefine (config) {
404
- const slotsDeclare = config.slotsDeclare
405
- const total = slotsDeclare.length
406
- let count = 0
407
- slotsDeclare.forEach(declare => {
408
- config.columns.forEach(row => {
409
- row.forEach(cell => {
410
- if (cell.slotConfig === declare) {
411
- count++
412
- }
413
- })
414
- })
415
- })
416
-
417
- return count === total
418
- },
419
- // 切换了标签页
420
- tabChanged (key) {
421
- this.scanFinish = false
422
- this.originalConfig.data = { ...this.originalConfig.data, ...this.config.data }
423
- this.config.data = this.originalConfig.data
424
- if (this.type === 'display') {
425
- const tempDataKeys = Object.keys(this.activeConfig.tempData)
426
- tempDataKeys.forEach(key => {
427
- this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
428
- })
429
- let count = 0
430
- this.imageList = []
431
- const keys = Object.keys(this.config.data.images)
432
- keys.forEach(key => {
433
- if (this.config.data.images[key].length > 0) {
434
- this.imageList = [...this.imageList, ...this.config.data.images[key]]
435
- count++
436
- }
437
- })
438
- this.hasImages = count > 0
439
- } else {
440
- this.hasImages = false
441
- }
442
- this.$nextTick(() => {
443
- this.scanFinish = true
444
- })
445
- },
446
- // 获取当前日期,为保存文件命名用
447
- getDate () {
448
- const currentDate = new Date()
449
-
450
- const year = currentDate.getFullYear()
451
- const month = String(currentDate.getMonth() + 1).padStart(2, '0')
452
- const day = String(currentDate.getDate()).padStart(2, '0')
453
-
454
- const formattedDate = `${year}_${month}_${day}`
455
-
456
- return formattedDate
457
- },
458
- // 打印
459
- printDocument () {
460
- // x-report
461
- const printContent = window.rawDocument.getElementById('printReady')
462
- printElement(printContent)
463
- this.$message.success('操作成功!')
464
- },
465
- // 导出PDF
466
- exportPDF () {
467
- const date = this.getDate()
468
- let title = this.config.title
469
- title = title.replace(/<[^>]+>/g, '')
470
- const fileName = date + '' + title
471
- HtmlToPdf.getPdf(fileName, '#printReady')
472
- },
473
- // 用于分割配置中的colums,将需要处理的数组提取出来
474
- formatConfigRow () {
475
- for (let i = 0; i < this.config.columns.length; i++) {
476
- // 对原始数组进行递归,依次将该位置拆分为三个部分,当前处理位置之前的,当前处理位置,当前处理位置之后的
477
- const before = this.config.columns.slice(0, i)
478
- const after = this.config.columns.slice(i + 1, this.config.columns.length)
479
-
480
- // 将当前处理的数组交给处理的方法
481
- const x = this.checkRow(this.config.columns[i])
482
-
483
- const newArr = []
484
-
485
- // 拼接之前的数组
486
- if (before.length > 0) {
487
- if (before.length >= 1) {
488
- before.forEach(item => {
489
- newArr.push(item)
490
- })
491
- } else {
492
- newArr.push(before)
493
- }
494
- }
495
-
496
- // 拼接不需要更改当前节点处理完成的数组
497
- newArr.push(x.old)
498
-
499
- // 如果处理了新加的数据,拼接
500
- if (x.add.length > 0) {
501
- for (let j = 0; j < x.add.length; j++) {
502
- if (x.add[j]) {
503
- newArr.push(x.add[j])
504
- i++
505
- }
506
- }
507
- }
508
-
509
- // 拼接之后的数组
510
- if (after.length > 0) {
511
- if (after.length >= 1) {
512
- after.forEach(item => {
513
- newArr.push(item)
514
- })
515
- } else {
516
- newArr.push(after)
517
- }
518
- }
519
-
520
- this.config.columns = newArr
521
- }
522
- },
523
- // 路径中含有@@@的key,将其解析,并返回其数据
524
- getDeepObject (obj, strPath) {
525
- const arr = strPath.split('@@@')
526
- let result = obj[arr[0]]
527
- arr.shift()
528
- try {
529
- while (arr.length > 0) {
530
- result = result[arr[0]]
531
- arr.shift()
532
- }
533
- } catch (e) {
534
- result = undefined
535
- }
536
- return result
537
- },
538
- // 处理colums数组,为声明了rowspan的单元格,自动匹配格式
539
- checkRow (rowArr) {
540
- // 不需要更改的数据
541
- const original = []
542
- // 需要更改新加的数据
543
- const addArr = []
544
- // 统计rowspan出现的总和
545
- let count = 0
546
- // 统计声明列需要的rowspan总数
547
- let total = 0
548
- // 是否为声明行
549
- let titleCellFlag = false
550
- // 是否为声明行后第一行
551
- let firstSubLine = false
552
- // 需要处理的行,新的index值
553
- let subRowIndex = 0
554
- // 新生成的行,临时存储
555
- const waitForAddArr = []
556
- // 用于记录声明行的colspan避免格式错误
557
- let preColSpan = 0
558
- // 用于统计循环次数,判断是否是最后一次
559
- let forEachCount = 0
560
-
561
- // 标记所有数据
562
- rowArr.forEach(cell => {
563
- forEachCount++
564
- // 如果该行没有rowspan则默认其为1,不要影响统计结果
565
- if (!cell.rowSpan) {
566
- cell.rowSpan = 0
567
- }
568
-
569
- if (cell.text && total !== 0) { // 如果遇到了下一个声明行,证明rowspan少了一行,需要补充一个占位格
570
- const nullObj = {
571
- type: 'placeHolderColumn',
572
- order: subRowIndex,
573
- noBoarder: true,
574
- needSplit: true,
575
- colSpan: preColSpan,
576
- dontShowRow: true
577
- }
578
- subRowIndex++
579
- waitForAddArr.push(nullObj)
580
- total = 0
581
- count = 0
582
- titleCellFlag = false
583
- firstSubLine = false
584
- } else if ((total !== count + cell.rowSpan) && forEachCount === rowArr.length) {
585
- // 如果没有遇到了下一个声明行,但已经是当前行最后一个数据,也证明rowspan少了一行,需要补充一个占位格
586
- const nullObj = {
587
- type: 'placeHolderColumn',
588
- order: subRowIndex,
589
- noBoarder: true,
590
- needSplit: true,
591
- colSpan: preColSpan,
592
- dontShowRow: true
593
- }
594
- subRowIndex++
595
- waitForAddArr.push(nullObj)
596
- total = 0
597
- count = 0
598
- titleCellFlag = false
599
- firstSubLine = false
600
- }
601
-
602
- // 判断是否为声明行
603
- if (cell.text && total === 0) {
604
- // 将声明行声明的rowspan作为总数,判断下方rowspan相加是否等于声明行声明的数量
605
- total = cell.rowSpan
606
- titleCellFlag = false
607
- firstSubLine = true
608
- subRowIndex = 1
609
- cell.show = true
610
- cell.showRowSpan = total
611
- } else if (cell.rowSpan > 0 && !titleCellFlag && firstSubLine) { // 判断是否为声明行后首行,因为首行不需要移动
612
- count += cell.rowSpan
613
- firstSubLine = false
614
- cell.noBoarder = true
615
- cell.show = true
616
- cell.showRowSpan = total
617
- } else if (cell.rowSpan > 0 && !titleCellFlag && !firstSubLine) { // 既非声明行,也非首行,需要移动
618
- count += cell.rowSpan
619
- // cell.type = 'notShow'
620
- cell.needSplit = true
621
- cell.order = subRowIndex
622
- cell.dontShowRow = true
623
- subRowIndex++
624
-
625
- // 如果之前添加过空行补充位置,刚好最后一位还有内容,将其互换
626
- if (forEachCount === rowArr.length && !waitForAddArr[waitForAddArr.length - 1].dataIndex) {
627
- waitForAddArr[waitForAddArr.length - 1].order += 1
628
- cell.order -= 1
629
- waitForAddArr.push(cell)
630
- } else {
631
- waitForAddArr.push(cell)
632
- }
633
- }
634
-
635
- // 如果count和total相等了,证明已经处理完成。将计数器还原
636
- if (count === total) {
637
- total = 0
638
- count = 0
639
- titleCellFlag = false
640
- firstSubLine = false
641
- }
642
- // 保存上一个的colspan,保持生成的格子与原格式一致
643
- preColSpan = cell.colSpan
644
- })
645
-
646
- // 将所有不需要移动的放入original
647
- rowArr.forEach(cell => {
648
- if (cell.needSplit !== true) {
649
- original.push(cell)
650
- }
651
- })
652
-
653
- // 增加新的数组
654
- waitForAddArr.forEach(cell => {
655
- const target = cell.order
656
- // if (cell.type === 'notShow') {
657
- // cell.type = 'inputs'
658
- // }
659
- cell.noBoarder = true
660
- if (addArr[target] === undefined) {
661
- const temp = []
662
- temp.push(cell)
663
- addArr[target] = temp
664
- } else if (addArr[target].length > 0) {
665
- addArr[target].push(cell)
666
- }
667
- })
668
-
669
- // 如果没有新增,将单元格边框设置为显示
670
- if (addArr.length < 1) {
671
- original.forEach(cell => {
672
- if (cell.type === 'input' || cell.type === 'inputs') {
673
- cell.noBoarder = false
674
- }
675
- })
676
- }
677
-
678
- return {
679
- old: original,
680
- add: addArr
681
- }
682
- },
683
- // 扫描配置,如果有插槽则拼接插槽
684
- scanConfigSlot (config) {
685
- const columnsArr = config.columns
686
- for (let i = 0; i < columnsArr.length; i++) {
687
- for (let j = 0; j < columnsArr[i].length; j++) {
688
- // 如果发现type为slot,开始匹配对应的slot配置文件
689
- if (columnsArr[i][j].type === 'slot') {
690
- const targetName = columnsArr[i][j].slotConfig
691
- // 找不到目标插槽配置
692
- if (!this.configFromWeb[targetName] || !this.configFromWeb[targetName].columns) {
693
- console.error('无法找到目标插槽的配置!')
694
- return
695
- }
696
-
697
- // 替换columns,合并data
698
- config.columns[i] = []
699
- const before = config.columns.slice(0, i)
700
- let after = config.columns.slice(i + 1, config.columns.length)
701
-
702
- const addArr = []
703
- for (let k = 0; k < this.configFromWeb[targetName].columns.length; k++) {
704
- const temp = []
705
- this.configFromWeb[targetName].columns[k].forEach(cell => {
706
- temp.push(cell)
707
- })
708
- addArr.push(temp)
709
- }
710
-
711
- const newArr = []
712
- // 拼接之前的数组
713
- if (before.length > 0) {
714
- if (before.length >= 1) {
715
- before.forEach(item => {
716
- newArr.push(item)
717
- })
718
- } else {
719
- newArr.push(before)
720
- }
721
- }
722
-
723
- addArr.forEach(arr => {
724
- newArr.push(arr)
725
- })
726
-
727
- // 拼接之后的数组
728
- if (after.length === 1) {
729
- if (after[0].type === 'slot' || after[0][0].type === 'slot') {
730
- after = []
731
- }
732
- }
733
- if (after.length > 0) {
734
- if (after.length >= 1) {
735
- after.forEach(item => {
736
- newArr.push(item)
737
- })
738
- } else {
739
- newArr.push(after)
740
- }
741
- }
742
-
743
- config.columns = newArr
744
- if (this.configFromWeb[targetName].slotsDeclare) {
745
- config.slotsDeclare = this.configFromWeb[targetName].slotsDeclare
746
- } else {
747
- config.slotsDeclare = []
748
- }
749
-
750
- if (config.data.images && this.configFromWeb[targetName].data.images) {
751
- config.data.images = { ...config.data.images, ...this.configFromWeb[targetName].data.images }
752
- delete this.configFromWeb[targetName].data.images
753
- }
754
- config.data = { ...config.data, ...this.configFromWeb[targetName].data }
755
- }
756
- }
757
- }
758
- this.config = config
759
- },
760
- // 扫描所有插槽名
761
- scanConfigName (config, resut) {
762
- if (config.slotsDeclare) {
763
- config.slotsDeclare.forEach(name => {
764
- resut.push(name)
765
- })
766
- }
767
- },
768
- // 获取插槽
769
- getConfigAndJoin (config, outerLock) {
770
- // 检查主配置插槽声明是否合法
771
- const check = this.checkSlotDefine(config)
772
- const waitForDownloadSlotName = []
773
- if (check) {
774
- // 扫描主配置中声明的插槽名
775
- this.scanConfigName(config, waitForDownloadSlotName)
776
-
777
- const total = waitForDownloadSlotName.length
778
- let count = 0
779
-
780
- // 挨个获取插槽
781
- waitForDownloadSlotName.forEach(configName => {
782
- getConfigByName(configName, this.serverName, res => {
783
- this.configFromWeb[configName] = res
784
- count++
785
- })
786
- })
787
-
788
- // 使用定时器循环判断锁状态,用于多个插槽,要等待统一获取完成之后,再进行下一步初始化
789
- const timer = setInterval(() => {
790
- console.log('插槽下载进度,当前:' + count + '/' + total)
791
- if (count >= total) {
792
- clearInterval(timer)
793
- this.scanConfigSlot(config)
794
- if (config.slotsDeclare.length > 0) {
795
- const lock = { status: true }
796
- this.getConfigAndJoin(config, lock)
797
- const innerTimer = setInterval(() => {
798
- if (!lock.status) {
799
- clearInterval(innerTimer)
800
- outerLock.status = false
801
- }
802
- }, 100)
803
- } else {
804
- outerLock.status = false
805
- }
806
- }
807
- }, 100)
808
- } else {
809
- console.error('插槽配置有误!')
810
- outerLock.status = false
811
- }
812
- },
813
- // 获取配置之后的初始化
814
- configInit () {
815
- // 上锁,等待所有配置获取完成
816
- const lock = { status: true }
817
-
818
- // 获取插槽
819
- this.getConfigAndJoin(this.config, lock)
820
-
821
- // 用定时器循环查看锁状态
822
- this.timer = setInterval(() => {
823
- if (!lock.status) {
824
- clearInterval(this.timer)
825
- console.log('拼接完成', this.config)
826
- // 将初始化好的配置拷贝一份留存
827
- this.originalConfig = Object.assign({}, this.config)
828
- if (!this.dontFormat) {
829
- // 扫描配置文件中有没有rowSpan,进行格式化调整
830
- this.formatConfigRow(this.config)
831
- }
832
- this.activeConfig = this.config
833
- this.showSkeleton = false
834
- // 判断是否有动态Index
835
- this.activeConfig.columns.forEach(row => {
836
- row.forEach(cell => {
837
- if (cell.dynamicDataIndex === true) {
838
- // 如果有动态index,取其函数,运行函数得到真实index保存
839
- // eslint-disable-next-line no-eval
840
- const func = eval('(' + cell.customFunctionForDynamicDataIndex + ')')
841
- cell.dataIndex = func(this.config)
842
- }
843
- })
844
- })
845
- // 将数据复制到临时数据中,带有@@@的数据,我们将其整体作为一个key保存,当编辑完成后,再将其解析,回填到需要的数据中
846
- this.activeConfig.tempData = {}
847
- // 是否有@@@深层引用
848
- this.activeConfig.columns.forEach(row => {
849
- row.forEach(cell => {
850
- // 将@@@解析
851
- if (cell.dataIndex !== undefined && cell.dataIndex.indexOf('@@@') !== -1) {
852
- this.activeConfig.tempData[cell.dataIndex] = this.getDeepObject(this.activeConfig.data, cell.dataIndex)
853
- }
854
- })
855
- })
856
- this.$nextTick(() => {
857
- this.scanFinish = true
858
- })
859
- }
860
- }, 100)
861
- },
862
- // 初始化JSON配置
863
- jsonConfigInit () {
864
- if (this.configData === undefined) {
865
- console.error('未找到数据!')
866
- } else {
867
- this.originalConfig = Object.assign({}, this.config)
868
- this.originalConfig.data = Object.assign(this.originalConfig.data, JSON.parse(JSON.stringify(this.configData)))
869
- this.type = 'display'
870
- // this.onlyDisplay = true
871
- this.showSkeleton = false
872
- this.$nextTick(() => {
873
- this.scanFinish = true
874
- })
875
- }
876
- },
877
- initByConfigName () {
878
- getConfigByName(this.configName, this.serverName, res => {
879
- this.config = res
880
- if (this.config.designMode === 'json') {
881
- if (this.configData !== undefined) {
882
- this.config.data = Object.assign({}, this.config.data, this.configData)
883
- }
884
- this.jsonConfigInit()
885
- } else {
886
- if (this.configData !== undefined) {
887
- this.config.data = Object.assign(this.config.data, this.configData)
888
- }
889
- if (this.config.data.images === undefined) {
890
- this.config.data.images = {}
891
- }
892
- this.configInit()
893
- }
894
- })
895
- },
896
- // 提供主动让外部调用的初始化方法
897
- async init ({ configName, configData }) {
898
- this.configName = configName
899
- this.configData = configData
900
- await this.initByConfigName()
901
- },
902
- // 提供主动赋值方法
903
- setData (data) {
904
- this.config.data = data
905
- this.configInit()
906
- }
907
- },
908
- beforeMount () {
909
- // 如果只是展示
910
- if (this.displayOnly) {
911
- this.onlyDisplay = true
912
- this.type = 'display'
913
- }
914
- // 如果有本地配置,优先使用本地配置
915
- if (this.localConfig) {
916
- // 如果配置是json渲染器
917
- if (this.localConfig.designMode === 'json') {
918
- this.config = this.localConfig
919
- if (this.configData !== undefined) {
920
- this.config.data = Object.assign(this.config.data, this.configData)
921
- }
922
- this.jsonConfigInit()
923
- } else {
924
- // 如果配置是普通渲染器
925
- this.config = this.localConfig
926
- if (this.configData !== undefined) {
927
- this.config.data = Object.assign(this.config.data, this.configData)
928
- }
929
- if (this.config.data.images === undefined) {
930
- this.config.data.images = {}
931
- }
932
- this.configInit()
933
- }
934
- } else if (this.configName) {
935
- // 如果本地配置没有值,则从琉璃中获取
936
- this.initByConfigName()
937
- }
938
- },
939
- mounted () {
940
- // 如果外界传来了registerMap,我们将本VM对象注册到map中
941
- if (this.registerMap !== undefined) {
942
- this.registerMap.push(this)
943
- }
944
- // 将原始数据备份保存
945
- if (this.configData) {
946
- this.dataCache = JSON.parse(JSON.stringify(this.configData))
947
- } else if (this.config) {
948
- this.dataCache = JSON.parse(JSON.stringify(this.config.data))
949
- }
950
- }
951
- }
952
- </script>
953
-
954
- <style lang="less" scoped>
955
- .tools {
956
- text-align: center;
957
- cursor: pointer;
958
-
959
- .toolsItem {
960
- display: inline-block;
961
- }
962
- }
963
- </style>
1
+ <template>
2
+ <div>
3
+ <!-- 骨架屏 -->
4
+ <a-card v-if="showSkeleton">
5
+ <a-skeleton active/>
6
+ </a-card>
7
+ <template v-if="noPadding">
8
+ <!-- 主体表格 -->
9
+ <XReportDesign
10
+ @updateImg="updateImg"
11
+ v-if="scanFinish"
12
+ :show-img-in-cell="showImgInCell"
13
+ :img-prefix="imgPrefix"
14
+ :use-oss-for-img="useOssForImg"
15
+ :display-only="displayOnly"
16
+ :config="type === 'display' ? originalConfig : activeConfig"
17
+ :slot-config-name="type === 'display' ? undefined : activatedSlotName"
18
+ :for-display="type === 'display'"
19
+ ref="XReportDesign"
20
+ id="printReady"
21
+ :server-name="serverName"
22
+ :show-title="showTitle"
23
+ :no-padding="noPadding"
24
+ :no-top-border="noTopBorder"
25
+ :show-images="hasImages"
26
+ :image-list="imageList">
27
+ </XReportDesign>
28
+ </template>
29
+ <template v-else>
30
+ <!-- 用以包裹整个页面 -->
31
+ <a-card v-if="!showSkeleton">
32
+ <!-- 切换菜单 -->
33
+ <a-radio-group
34
+ v-model="type"
35
+ default-value="a"
36
+ button-style="solid"
37
+ @change="tabChanged"
38
+ v-show="!onlyDisplay && editMode">
39
+ <a-radio-button value="design" v-if="!onlyDisplay">
40
+ 设计
41
+ </a-radio-button>
42
+ <a-radio-button value="display" style="border-radius: 0">
43
+ 预览
44
+ </a-radio-button>
45
+ </a-radio-group>
46
+ <a-radio-button @click="saveConfig" style="border-radius: 0 4px 4px 0" v-if="showSaveButton">
47
+ 保存
48
+ </a-radio-button>
49
+ <!-- 主体表格 -->
50
+ <XReportDesign
51
+ v-if="scanFinish"
52
+ @updateImg="updateImg"
53
+ :show-img-in-cell="showImgInCell"
54
+ :img-prefix="imgPrefix"
55
+ :use-oss-for-img="useOssForImg"
56
+ :display-only="displayOnly"
57
+ :config="type === 'display' ? originalConfig : activeConfig"
58
+ :slot-config-name="type === 'display' ? undefined : activatedSlotName"
59
+ :for-display="type === 'display'"
60
+ :no-padding="noPadding"
61
+ :no-top-border="noTopBorder"
62
+ :show-title="showTitle"
63
+ ref="XReportDesign"
64
+ id="printReady"
65
+ :server-name="serverName"
66
+ :show-images="hasImages"
67
+ :image-list="imageList">
68
+ </XReportDesign>
69
+ <!-- 导出按钮 -->
70
+ <template v-if="type === 'display'">
71
+ <div class="tools">
72
+ <div class="toolsItem">
73
+ <a-space>
74
+ <a-button type="primary" @click="printDocument">
75
+ 打印
76
+ </a-button>
77
+ <a-button @click="exportPDF">
78
+ 导出PDF
79
+ </a-button>
80
+ </a-space>
81
+ </div>
82
+ </div>
83
+ </template>
84
+ </a-card>
85
+ </template>
86
+ </div>
87
+ </template>
88
+
89
+ <script>
90
+ // 转PDF用
91
+ import HtmlToPdf from '@vue2-client/utils/htmlToPDF'
92
+ import { getConfigByName } from '@vue2-client/services/api/common'
93
+ import XReportDesign from '@vue2-client/base-client/components/common/XReport/XReportDesign.vue'
94
+ import { printElement } from './print'
95
+
96
+ export default {
97
+ name: 'XReport',
98
+ props: {
99
+ files: {
100
+ type: Array,
101
+ default: () => {
102
+ return []
103
+ }
104
+ },
105
+ // 控制用户权限,user和admin
106
+ authority: {
107
+ type: String,
108
+ default: 'user'
109
+ },
110
+ // 是否为编辑模式
111
+ editMode: {
112
+ type: Boolean,
113
+ default: true
114
+ },
115
+ // 配置名
116
+ configName: {
117
+ type: String,
118
+ required: true
119
+ },
120
+ // 插槽名
121
+ activatedSlotName: {
122
+ type: String,
123
+ default: undefined
124
+ },
125
+ // 本地配置,调试用
126
+ localConfig: {
127
+ type: Object,
128
+ default: undefined
129
+ },
130
+ // 兼容老版本配置
131
+ dontFormat: {
132
+ type: Boolean,
133
+ default: true
134
+ },
135
+ showImgInCell: {
136
+ type: Boolean,
137
+ default: false
138
+ },
139
+ // 数据
140
+ configData: {
141
+ type: Object,
142
+ default: undefined
143
+ },
144
+ // 命名空间
145
+ serverName: {
146
+ type: String,
147
+ default: process.env.VUE_APP_SYSTEM_NAME
148
+ },
149
+ // 只做展示
150
+ displayOnly: {
151
+ type: Boolean,
152
+ default: false
153
+ },
154
+ // 表格没有边距
155
+ noPadding: {
156
+ type: Boolean,
157
+ default: false
158
+ },
159
+ // 表格没有上边框,与noPadding搭配可以实现连续表格
160
+ noTopBorder: {
161
+ type: Boolean,
162
+ default: false
163
+ },
164
+ // 是否展示标题
165
+ showTitle: {
166
+ type: Boolean,
167
+ default: true
168
+ },
169
+ // 是否展示保存按钮
170
+ showSaveButton: {
171
+ type: Boolean,
172
+ default: true
173
+ },
174
+ // 是否将组件注册到外层提供的容器中,方便外侧统一保存
175
+ registerMap: {
176
+ type: Array,
177
+ default: undefined
178
+ },
179
+ // 图片是否使用OSS来保存
180
+ useOssForImg: {
181
+ type: Boolean,
182
+ default: true
183
+ },
184
+ // 图片上传后添加前缀
185
+ imgPrefix: {
186
+ type: String,
187
+ default: undefined
188
+ }
189
+ },
190
+ components: {
191
+ XReportDesign
192
+ },
193
+ data () {
194
+ return {
195
+ // 控制骨架屏显隐
196
+ showSkeleton: true,
197
+ // 配置
198
+ config: undefined,
199
+ // 当前显示模式,编辑模式,预览模式
200
+ type: 'design',
201
+ // 仅供展示,不可编辑
202
+ onlyDisplay: false,
203
+ // 每行最大列数,非必要请勿更改,现在的设计器完全是基于每行12列来设计的
204
+ maxColSpan: 12,
205
+ // 定义是否完成配置的扫描,未完成不要渲染子组件
206
+ scanFinish: false,
207
+ // 当前激活的配置文件
208
+ activeConfig: null,
209
+ // 原始配置文件
210
+ // 用于展示。某些情况下“设计页”中的内容仅为“预览页”表格其中的一部分
211
+ originalConfig: null,
212
+ // 扫描到的配置
213
+ configFromWeb: {},
214
+ // 用于获取配置的锁
215
+ timer: undefined,
216
+ // 是否包含图片
217
+ hasImages: false,
218
+ // 图片列表
219
+ imageList: [],
220
+ // 保存最原始的数据,用于判断哪些数据被更改了
221
+ dataCache: undefined,
222
+ // 判断哪些数据被更改了,存储对应的key
223
+ diff: []
224
+ }
225
+ },
226
+ watch: {
227
+ // 如果配置名更改了,重新获取配置
228
+ configName (val) {
229
+ if (val) {
230
+ getConfigByName(this.configName, undefined, res => {
231
+ this.config = res
232
+ this.configInit()
233
+ })
234
+ }
235
+ },
236
+ // 如果本地配置更改了,重新初始化
237
+ localConfig (val) {
238
+ if (val) {
239
+ this.config = val
240
+ this.configInit()
241
+ }
242
+ }
243
+ },
244
+ methods: {
245
+ // 向外暴露图片修改后的数据,某些外部需要自己管理图片的保存与修改
246
+ updateImg (data) {
247
+ this.$emit('updateImg', data)
248
+ },
249
+ // 导出数据,某些外部需要统一控制数据的变动
250
+ exportData (skipValidation = false) {
251
+ // 如果不跳过校验,先进行必填字段校验
252
+ if (!skipValidation) {
253
+ const validation = this.validateRequiredFields()
254
+ if (!validation.isValid) {
255
+ const errorMessages = validation.errors.map(error => error.message).join(';')
256
+ console.warn(`数据导出警告:存在必填字段未填写 - ${errorMessages}`)
257
+ // 在导出时只警告,不阻止导出
258
+ }
259
+ }
260
+
261
+ // 获取当前修改后的数据
262
+ let tempData
263
+ if (this.activeConfig === undefined || this.activeConfig === null) {
264
+ tempData = this.originalConfig.data
265
+ } else {
266
+ const tempDataKeys = Object.keys(this.activeConfig.tempData)
267
+ tempDataKeys.forEach(key => {
268
+ this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
269
+ })
270
+ tempData = this.activeConfig.data
271
+ }
272
+ // 对比数据的差异
273
+ this.diff = []
274
+ this.compareProps(tempData, this.dataCache)
275
+ this.diff.forEach(eachDiff => {
276
+ const arr = eachDiff.split('.')
277
+ let targetData = tempData[arr[0]]
278
+ if (arr.length !== 1) {
279
+ for (let i = 1; i < arr.length - 1; i++) {
280
+ const path = arr[i]
281
+ targetData = targetData[path]
282
+ }
283
+ }
284
+ // 将修改的数据,添加update = true属性
285
+ targetData.update = true
286
+ })
287
+ return tempData
288
+ },
289
+ // 对比两个obj有哪里不同
290
+ compareProps (obj1, obj2, path = '') {
291
+ for (const key in obj1) {
292
+ // 如果一个是undefined
293
+ if (typeof obj2[key] === 'undefined') {
294
+ this.diff.push(path + key)
295
+ // 如果是数组长度不一样
296
+ } else if (Array.isArray(obj1) && Array.isArray(obj2)) {
297
+ if (obj1[key].length !== obj2[key].length) {
298
+ this.diff.push(path + key)
299
+ }
300
+ // 如果都是对象,并且存在同样的key,递归子key
301
+ } else if (typeof obj1[key] === 'object' && typeof obj2[key] === 'object') {
302
+ this.compareProps(obj1[key], obj2[key], path + key + '.')
303
+ // 如果不是obj,对比其数据
304
+ } else if (obj1[key] !== obj2[key]) {
305
+ this.diff.push(path + key)
306
+ }
307
+ }
308
+ },
309
+ // 验证必填字段
310
+ validateRequiredFields () {
311
+ const errors = []
312
+ const config = this.type === 'display' ? this.originalConfig : this.activeConfig
313
+
314
+ if (!config || !config.columns) {
315
+ return { isValid: true, errors: [] }
316
+ }
317
+
318
+ // 遍历所有配置项检查必填字段
319
+ config.columns.forEach((row, rowIndex) => {
320
+ row.forEach((cell, cellIndex) => {
321
+ if (cell.required && cell.dataIndex) {
322
+ let value
323
+ if (cell.dataIndex.indexOf('@@@') !== -1) {
324
+ // 处理深层嵌套数据
325
+ // const arr = cell.dataIndex.split('@@@')
326
+ value = config.tempData && config.tempData[cell.dataIndex]
327
+ } else {
328
+ value = config.data[cell.dataIndex]
329
+ }
330
+
331
+ // 检查值是否为空
332
+ if (!value || (typeof value === 'string' && value.trim() === '')) {
333
+ const message = cell.requiredMessage || this.getDefaultRequiredMessage(cell.type)
334
+ errors.push({
335
+ dataIndex: cell.dataIndex,
336
+ message: message,
337
+ rowIndex: rowIndex,
338
+ cellIndex: cellIndex,
339
+ type: cell.type
340
+ })
341
+ }
342
+ }
343
+ })
344
+ })
345
+
346
+ return {
347
+ isValid: errors.length === 0,
348
+ errors: errors
349
+ }
350
+ },
351
+ // 获取默认的必填提示信息
352
+ getDefaultRequiredMessage (fieldType) {
353
+ const defaultMessages = {
354
+ datePicker: '请选择日期',
355
+ timePicker: '请选择日期时间',
356
+ input: '请填写此字段',
357
+ inputs: '请填写此字段'
358
+ }
359
+ return defaultMessages[fieldType] || '此字段为必填项'
360
+ },
361
+ // 正常的保存方法,当前修改内容会直接全部导出到外部
362
+ saveConfig () {
363
+ // 先进行必填字段校验
364
+ const validation = this.validateRequiredFields()
365
+ if (!validation.isValid) {
366
+ const errorMessages = validation.errors.map(error => error.message).join(';')
367
+ this.$message.error(`保存失败:${errorMessages}`)
368
+ return false
369
+ }
370
+
371
+ if (this.activeConfig === undefined || this.activeConfig === null) {
372
+ return this.originalConfig.data
373
+ } else {
374
+ const tempDataKeys = Object.keys(this.activeConfig.tempData)
375
+ tempDataKeys.forEach(key => {
376
+ this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
377
+ })
378
+ this.$emit('saveConfig', this.$refs.XReportDesign.activatedConfig)
379
+ return true
380
+ }
381
+ },
382
+ // 通过@@@分割临时变量,找到对应的key,并修改它的值
383
+ changeDeepObject (obj, strPath, newVal) {
384
+ const arr = strPath.split('@@@')
385
+ if (obj[arr[0]] === undefined) {
386
+ obj = obj.images
387
+ }
388
+ if (arr.length === 1) {
389
+ obj[arr[0]] = newVal
390
+ } else {
391
+ let result = obj[arr[0]]
392
+ arr.shift()
393
+ while (arr.length > 1) {
394
+ result = result[arr[0]]
395
+ arr.shift()
396
+ }
397
+ if (result) {
398
+ result[arr[0]] = newVal
399
+ }
400
+ }
401
+ },
402
+ // 检查slot是否在配置文件中包含,如果没有包含,则视为非法获取
403
+ checkSlotDefine (config) {
404
+ const slotsDeclare = config.slotsDeclare
405
+ const total = slotsDeclare.length
406
+ let count = 0
407
+ slotsDeclare.forEach(declare => {
408
+ config.columns.forEach(row => {
409
+ row.forEach(cell => {
410
+ if (cell.slotConfig === declare) {
411
+ count++
412
+ }
413
+ })
414
+ })
415
+ })
416
+
417
+ return count === total
418
+ },
419
+ // 切换了标签页
420
+ tabChanged (key) {
421
+ this.scanFinish = false
422
+ this.originalConfig.data = { ...this.originalConfig.data, ...this.config.data }
423
+ this.config.data = this.originalConfig.data
424
+ if (this.type === 'display') {
425
+ const tempDataKeys = Object.keys(this.activeConfig.tempData)
426
+ tempDataKeys.forEach(key => {
427
+ this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
428
+ })
429
+ let count = 0
430
+ this.imageList = []
431
+ const keys = Object.keys(this.config.data.images)
432
+ keys.forEach(key => {
433
+ if (this.config.data.images[key].length > 0) {
434
+ this.imageList = [...this.imageList, ...this.config.data.images[key]]
435
+ count++
436
+ }
437
+ })
438
+ this.hasImages = count > 0
439
+ } else {
440
+ this.hasImages = false
441
+ }
442
+ this.$nextTick(() => {
443
+ this.scanFinish = true
444
+ })
445
+ },
446
+ // 获取当前日期,为保存文件命名用
447
+ getDate () {
448
+ const currentDate = new Date()
449
+
450
+ const year = currentDate.getFullYear()
451
+ const month = String(currentDate.getMonth() + 1).padStart(2, '0')
452
+ const day = String(currentDate.getDate()).padStart(2, '0')
453
+
454
+ const formattedDate = `${year}_${month}_${day}`
455
+
456
+ return formattedDate
457
+ },
458
+ // 打印
459
+ printDocument () {
460
+ // x-report
461
+ const printContent = window.rawDocument.getElementById('printReady')
462
+ printElement(printContent)
463
+ this.$message.success('操作成功!')
464
+ },
465
+ // 导出PDF
466
+ exportPDF () {
467
+ const date = this.getDate()
468
+ let title = this.config.title
469
+ title = title.replace(/<[^>]+>/g, '')
470
+ const fileName = date + '' + title
471
+ HtmlToPdf.getPdf(fileName, '#printReady')
472
+ },
473
+ // 用于分割配置中的colums,将需要处理的数组提取出来
474
+ formatConfigRow () {
475
+ for (let i = 0; i < this.config.columns.length; i++) {
476
+ // 对原始数组进行递归,依次将该位置拆分为三个部分,当前处理位置之前的,当前处理位置,当前处理位置之后的
477
+ const before = this.config.columns.slice(0, i)
478
+ const after = this.config.columns.slice(i + 1, this.config.columns.length)
479
+
480
+ // 将当前处理的数组交给处理的方法
481
+ const x = this.checkRow(this.config.columns[i])
482
+
483
+ const newArr = []
484
+
485
+ // 拼接之前的数组
486
+ if (before.length > 0) {
487
+ if (before.length >= 1) {
488
+ before.forEach(item => {
489
+ newArr.push(item)
490
+ })
491
+ } else {
492
+ newArr.push(before)
493
+ }
494
+ }
495
+
496
+ // 拼接不需要更改当前节点处理完成的数组
497
+ newArr.push(x.old)
498
+
499
+ // 如果处理了新加的数据,拼接
500
+ if (x.add.length > 0) {
501
+ for (let j = 0; j < x.add.length; j++) {
502
+ if (x.add[j]) {
503
+ newArr.push(x.add[j])
504
+ i++
505
+ }
506
+ }
507
+ }
508
+
509
+ // 拼接之后的数组
510
+ if (after.length > 0) {
511
+ if (after.length >= 1) {
512
+ after.forEach(item => {
513
+ newArr.push(item)
514
+ })
515
+ } else {
516
+ newArr.push(after)
517
+ }
518
+ }
519
+
520
+ this.config.columns = newArr
521
+ }
522
+ },
523
+ // 路径中含有@@@的key,将其解析,并返回其数据
524
+ getDeepObject (obj, strPath) {
525
+ const arr = strPath.split('@@@')
526
+ let result = obj[arr[0]]
527
+ arr.shift()
528
+ try {
529
+ while (arr.length > 0) {
530
+ result = result[arr[0]]
531
+ arr.shift()
532
+ }
533
+ } catch (e) {
534
+ result = undefined
535
+ }
536
+ return result
537
+ },
538
+ // 处理colums数组,为声明了rowspan的单元格,自动匹配格式
539
+ checkRow (rowArr) {
540
+ // 不需要更改的数据
541
+ const original = []
542
+ // 需要更改新加的数据
543
+ const addArr = []
544
+ // 统计rowspan出现的总和
545
+ let count = 0
546
+ // 统计声明列需要的rowspan总数
547
+ let total = 0
548
+ // 是否为声明行
549
+ let titleCellFlag = false
550
+ // 是否为声明行后第一行
551
+ let firstSubLine = false
552
+ // 需要处理的行,新的index值
553
+ let subRowIndex = 0
554
+ // 新生成的行,临时存储
555
+ const waitForAddArr = []
556
+ // 用于记录声明行的colspan避免格式错误
557
+ let preColSpan = 0
558
+ // 用于统计循环次数,判断是否是最后一次
559
+ let forEachCount = 0
560
+
561
+ // 标记所有数据
562
+ rowArr.forEach(cell => {
563
+ forEachCount++
564
+ // 如果该行没有rowspan则默认其为1,不要影响统计结果
565
+ if (!cell.rowSpan) {
566
+ cell.rowSpan = 0
567
+ }
568
+
569
+ if (cell.text && total !== 0) { // 如果遇到了下一个声明行,证明rowspan少了一行,需要补充一个占位格
570
+ const nullObj = {
571
+ type: 'placeHolderColumn',
572
+ order: subRowIndex,
573
+ noBoarder: true,
574
+ needSplit: true,
575
+ colSpan: preColSpan,
576
+ dontShowRow: true
577
+ }
578
+ subRowIndex++
579
+ waitForAddArr.push(nullObj)
580
+ total = 0
581
+ count = 0
582
+ titleCellFlag = false
583
+ firstSubLine = false
584
+ } else if ((total !== count + cell.rowSpan) && forEachCount === rowArr.length) {
585
+ // 如果没有遇到了下一个声明行,但已经是当前行最后一个数据,也证明rowspan少了一行,需要补充一个占位格
586
+ const nullObj = {
587
+ type: 'placeHolderColumn',
588
+ order: subRowIndex,
589
+ noBoarder: true,
590
+ needSplit: true,
591
+ colSpan: preColSpan,
592
+ dontShowRow: true
593
+ }
594
+ subRowIndex++
595
+ waitForAddArr.push(nullObj)
596
+ total = 0
597
+ count = 0
598
+ titleCellFlag = false
599
+ firstSubLine = false
600
+ }
601
+
602
+ // 判断是否为声明行
603
+ if (cell.text && total === 0) {
604
+ // 将声明行声明的rowspan作为总数,判断下方rowspan相加是否等于声明行声明的数量
605
+ total = cell.rowSpan
606
+ titleCellFlag = false
607
+ firstSubLine = true
608
+ subRowIndex = 1
609
+ cell.show = true
610
+ cell.showRowSpan = total
611
+ } else if (cell.rowSpan > 0 && !titleCellFlag && firstSubLine) { // 判断是否为声明行后首行,因为首行不需要移动
612
+ count += cell.rowSpan
613
+ firstSubLine = false
614
+ cell.noBoarder = true
615
+ cell.show = true
616
+ cell.showRowSpan = total
617
+ } else if (cell.rowSpan > 0 && !titleCellFlag && !firstSubLine) { // 既非声明行,也非首行,需要移动
618
+ count += cell.rowSpan
619
+ // cell.type = 'notShow'
620
+ cell.needSplit = true
621
+ cell.order = subRowIndex
622
+ cell.dontShowRow = true
623
+ subRowIndex++
624
+
625
+ // 如果之前添加过空行补充位置,刚好最后一位还有内容,将其互换
626
+ if (forEachCount === rowArr.length && !waitForAddArr[waitForAddArr.length - 1].dataIndex) {
627
+ waitForAddArr[waitForAddArr.length - 1].order += 1
628
+ cell.order -= 1
629
+ waitForAddArr.push(cell)
630
+ } else {
631
+ waitForAddArr.push(cell)
632
+ }
633
+ }
634
+
635
+ // 如果count和total相等了,证明已经处理完成。将计数器还原
636
+ if (count === total) {
637
+ total = 0
638
+ count = 0
639
+ titleCellFlag = false
640
+ firstSubLine = false
641
+ }
642
+ // 保存上一个的colspan,保持生成的格子与原格式一致
643
+ preColSpan = cell.colSpan
644
+ })
645
+
646
+ // 将所有不需要移动的放入original
647
+ rowArr.forEach(cell => {
648
+ if (cell.needSplit !== true) {
649
+ original.push(cell)
650
+ }
651
+ })
652
+
653
+ // 增加新的数组
654
+ waitForAddArr.forEach(cell => {
655
+ const target = cell.order
656
+ // if (cell.type === 'notShow') {
657
+ // cell.type = 'inputs'
658
+ // }
659
+ cell.noBoarder = true
660
+ if (addArr[target] === undefined) {
661
+ const temp = []
662
+ temp.push(cell)
663
+ addArr[target] = temp
664
+ } else if (addArr[target].length > 0) {
665
+ addArr[target].push(cell)
666
+ }
667
+ })
668
+
669
+ // 如果没有新增,将单元格边框设置为显示
670
+ if (addArr.length < 1) {
671
+ original.forEach(cell => {
672
+ if (cell.type === 'input' || cell.type === 'inputs') {
673
+ cell.noBoarder = false
674
+ }
675
+ })
676
+ }
677
+
678
+ return {
679
+ old: original,
680
+ add: addArr
681
+ }
682
+ },
683
+ // 扫描配置,如果有插槽则拼接插槽
684
+ scanConfigSlot (config) {
685
+ const columnsArr = config.columns
686
+ for (let i = 0; i < columnsArr.length; i++) {
687
+ for (let j = 0; j < columnsArr[i].length; j++) {
688
+ // 如果发现type为slot,开始匹配对应的slot配置文件
689
+ if (columnsArr[i][j].type === 'slot') {
690
+ const targetName = columnsArr[i][j].slotConfig
691
+ // 找不到目标插槽配置
692
+ if (!this.configFromWeb[targetName] || !this.configFromWeb[targetName].columns) {
693
+ console.error('无法找到目标插槽的配置!')
694
+ return
695
+ }
696
+
697
+ // 替换columns,合并data
698
+ config.columns[i] = []
699
+ const before = config.columns.slice(0, i)
700
+ let after = config.columns.slice(i + 1, config.columns.length)
701
+
702
+ const addArr = []
703
+ for (let k = 0; k < this.configFromWeb[targetName].columns.length; k++) {
704
+ const temp = []
705
+ this.configFromWeb[targetName].columns[k].forEach(cell => {
706
+ temp.push(cell)
707
+ })
708
+ addArr.push(temp)
709
+ }
710
+
711
+ const newArr = []
712
+ // 拼接之前的数组
713
+ if (before.length > 0) {
714
+ if (before.length >= 1) {
715
+ before.forEach(item => {
716
+ newArr.push(item)
717
+ })
718
+ } else {
719
+ newArr.push(before)
720
+ }
721
+ }
722
+
723
+ addArr.forEach(arr => {
724
+ newArr.push(arr)
725
+ })
726
+
727
+ // 拼接之后的数组
728
+ if (after.length === 1) {
729
+ if (after[0].type === 'slot' || after[0][0].type === 'slot') {
730
+ after = []
731
+ }
732
+ }
733
+ if (after.length > 0) {
734
+ if (after.length >= 1) {
735
+ after.forEach(item => {
736
+ newArr.push(item)
737
+ })
738
+ } else {
739
+ newArr.push(after)
740
+ }
741
+ }
742
+
743
+ config.columns = newArr
744
+ if (this.configFromWeb[targetName].slotsDeclare) {
745
+ config.slotsDeclare = this.configFromWeb[targetName].slotsDeclare
746
+ } else {
747
+ config.slotsDeclare = []
748
+ }
749
+
750
+ if (config.data.images && this.configFromWeb[targetName].data.images) {
751
+ config.data.images = { ...config.data.images, ...this.configFromWeb[targetName].data.images }
752
+ delete this.configFromWeb[targetName].data.images
753
+ }
754
+ config.data = { ...config.data, ...this.configFromWeb[targetName].data }
755
+ }
756
+ }
757
+ }
758
+ this.config = config
759
+ },
760
+ // 扫描所有插槽名
761
+ scanConfigName (config, resut) {
762
+ if (config.slotsDeclare) {
763
+ config.slotsDeclare.forEach(name => {
764
+ resut.push(name)
765
+ })
766
+ }
767
+ },
768
+ // 获取插槽
769
+ getConfigAndJoin (config, outerLock) {
770
+ // 检查主配置插槽声明是否合法
771
+ const check = this.checkSlotDefine(config)
772
+ const waitForDownloadSlotName = []
773
+ if (check) {
774
+ // 扫描主配置中声明的插槽名
775
+ this.scanConfigName(config, waitForDownloadSlotName)
776
+
777
+ const total = waitForDownloadSlotName.length
778
+ let count = 0
779
+
780
+ // 挨个获取插槽
781
+ waitForDownloadSlotName.forEach(configName => {
782
+ getConfigByName(configName, this.serverName, res => {
783
+ this.configFromWeb[configName] = res
784
+ count++
785
+ })
786
+ })
787
+
788
+ // 使用定时器循环判断锁状态,用于多个插槽,要等待统一获取完成之后,再进行下一步初始化
789
+ const timer = setInterval(() => {
790
+ console.log('插槽下载进度,当前:' + count + '/' + total)
791
+ if (count >= total) {
792
+ clearInterval(timer)
793
+ this.scanConfigSlot(config)
794
+ if (config.slotsDeclare.length > 0) {
795
+ const lock = { status: true }
796
+ this.getConfigAndJoin(config, lock)
797
+ const innerTimer = setInterval(() => {
798
+ if (!lock.status) {
799
+ clearInterval(innerTimer)
800
+ outerLock.status = false
801
+ }
802
+ }, 100)
803
+ } else {
804
+ outerLock.status = false
805
+ }
806
+ }
807
+ }, 100)
808
+ } else {
809
+ console.error('插槽配置有误!')
810
+ outerLock.status = false
811
+ }
812
+ },
813
+ // 获取配置之后的初始化
814
+ configInit () {
815
+ // 上锁,等待所有配置获取完成
816
+ const lock = { status: true }
817
+
818
+ // 获取插槽
819
+ this.getConfigAndJoin(this.config, lock)
820
+
821
+ // 用定时器循环查看锁状态
822
+ this.timer = setInterval(() => {
823
+ if (!lock.status) {
824
+ clearInterval(this.timer)
825
+ console.log('拼接完成', this.config)
826
+ // 将初始化好的配置拷贝一份留存
827
+ this.originalConfig = Object.assign({}, this.config)
828
+ if (!this.dontFormat) {
829
+ // 扫描配置文件中有没有rowSpan,进行格式化调整
830
+ this.formatConfigRow(this.config)
831
+ }
832
+ this.activeConfig = this.config
833
+ this.showSkeleton = false
834
+ // 判断是否有动态Index
835
+ this.activeConfig.columns.forEach(row => {
836
+ row.forEach(cell => {
837
+ if (cell.dynamicDataIndex === true) {
838
+ // 如果有动态index,取其函数,运行函数得到真实index保存
839
+ // eslint-disable-next-line no-eval
840
+ const func = eval('(' + cell.customFunctionForDynamicDataIndex + ')')
841
+ cell.dataIndex = func(this.config)
842
+ }
843
+ })
844
+ })
845
+ // 将数据复制到临时数据中,带有@@@的数据,我们将其整体作为一个key保存,当编辑完成后,再将其解析,回填到需要的数据中
846
+ this.activeConfig.tempData = {}
847
+ // 是否有@@@深层引用
848
+ this.activeConfig.columns.forEach(row => {
849
+ row.forEach(cell => {
850
+ // 将@@@解析
851
+ if (cell.dataIndex !== undefined && cell.dataIndex.indexOf('@@@') !== -1) {
852
+ this.activeConfig.tempData[cell.dataIndex] = this.getDeepObject(this.activeConfig.data, cell.dataIndex)
853
+ }
854
+ })
855
+ })
856
+ this.$nextTick(() => {
857
+ this.scanFinish = true
858
+ })
859
+ }
860
+ }, 100)
861
+ },
862
+ // 初始化JSON配置
863
+ jsonConfigInit () {
864
+ if (this.configData === undefined) {
865
+ console.error('未找到数据!')
866
+ } else {
867
+ this.originalConfig = Object.assign({}, this.config)
868
+ this.originalConfig.data = Object.assign(this.originalConfig.data, JSON.parse(JSON.stringify(this.configData)))
869
+ this.type = 'display'
870
+ // this.onlyDisplay = true
871
+ this.showSkeleton = false
872
+ this.$nextTick(() => {
873
+ this.scanFinish = true
874
+ })
875
+ }
876
+ },
877
+ initByConfigName () {
878
+ getConfigByName(this.configName, this.serverName, res => {
879
+ this.config = res
880
+ if (this.config.designMode === 'json') {
881
+ if (this.configData !== undefined) {
882
+ this.config.data = Object.assign({}, this.config.data, this.configData)
883
+ }
884
+ this.jsonConfigInit()
885
+ } else {
886
+ if (this.configData !== undefined) {
887
+ this.config.data = Object.assign(this.config.data, this.configData)
888
+ }
889
+ if (this.config.data.images === undefined) {
890
+ this.config.data.images = {}
891
+ }
892
+ this.configInit()
893
+ }
894
+ })
895
+ },
896
+ // 提供主动让外部调用的初始化方法
897
+ async init ({ configName, configData }) {
898
+ this.configName = configName
899
+ this.configData = configData
900
+ await this.initByConfigName()
901
+ },
902
+ // 提供主动赋值方法
903
+ setData (data) {
904
+ this.config.data = data
905
+ this.configInit()
906
+ }
907
+ },
908
+ beforeMount () {
909
+ // 如果只是展示
910
+ if (this.displayOnly) {
911
+ this.onlyDisplay = true
912
+ this.type = 'display'
913
+ }
914
+ // 如果有本地配置,优先使用本地配置
915
+ if (this.localConfig) {
916
+ // 如果配置是json渲染器
917
+ if (this.localConfig.designMode === 'json') {
918
+ this.config = this.localConfig
919
+ if (this.configData !== undefined) {
920
+ this.config.data = Object.assign(this.config.data, this.configData)
921
+ }
922
+ this.jsonConfigInit()
923
+ } else {
924
+ // 如果配置是普通渲染器
925
+ this.config = this.localConfig
926
+ if (this.configData !== undefined) {
927
+ this.config.data = Object.assign(this.config.data, this.configData)
928
+ }
929
+ if (this.config.data.images === undefined) {
930
+ this.config.data.images = {}
931
+ }
932
+ this.configInit()
933
+ }
934
+ } else if (this.configName) {
935
+ // 如果本地配置没有值,则从琉璃中获取
936
+ this.initByConfigName()
937
+ }
938
+ },
939
+ mounted () {
940
+ // 如果外界传来了registerMap,我们将本VM对象注册到map中
941
+ if (this.registerMap !== undefined) {
942
+ this.registerMap.push(this)
943
+ }
944
+ // 将原始数据备份保存
945
+ if (this.configData) {
946
+ this.dataCache = JSON.parse(JSON.stringify(this.configData))
947
+ } else if (this.config) {
948
+ this.dataCache = JSON.parse(JSON.stringify(this.config.data))
949
+ }
950
+ }
951
+ }
952
+ </script>
953
+
954
+ <style lang="less" scoped>
955
+ .tools {
956
+ text-align: center;
957
+ cursor: pointer;
958
+
959
+ .toolsItem {
960
+ display: inline-block;
961
+ }
962
+ }
963
+ </style>