vue2-client 1.9.13 → 1.9.14

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 (324) hide show
  1. package/.env +19 -19
  2. package/.env.gaslink +19 -19
  3. package/.env.revenue +19 -19
  4. package/.eslintrc.js +90 -90
  5. package/CHANGELOG.md +824 -824
  6. package/Components.md +60 -60
  7. package/babel.config.js +21 -21
  8. package/docs/Logic/345/207/275/346/225/260/344/275/277/347/224/250/347/233/270/345/205/263.md +45 -45
  9. package/docs/LowCode/lowcode.md +155 -155
  10. package/docs/LowCode/lowcodeForDeveloper.md +230 -230
  11. package/docs/index.md +30 -30
  12. package/docs//345/207/275/346/225/260/344/275/277/347/224/250/347/233/270/345/205/263.md +121 -121
  13. package/index.js +31 -31
  14. package/jest-transform-stub.js +8 -8
  15. package/jest.config.js +21 -21
  16. package/jest.setup.js +7 -7
  17. package/package.json +101 -101
  18. package/public/his/editor/editor.html +51 -51
  19. package/public/his/editor/mock/bind_data.html +779 -779
  20. package/public/his/editor/mock/data_table.html +40 -40
  21. package/public/his/editor/mock/sign.html +75 -75
  22. package/public/his/editor/vender/JsBarcode.all.js +3669 -3669
  23. package/public/his/editor/vender/date97/My97DatePicker.htm +65 -65
  24. package/public/his/editor/vender/date97/WdatePicker.js +677 -677
  25. package/public/his/editor/vender/date97/calendar.js +4 -4
  26. package/public/his/editor/vender/date97/lang/en.js +13 -13
  27. package/public/his/editor/vender/date97/lang/zh-cn.js +13 -13
  28. package/public/his/editor/vender/date97/lang/zh-tw.js +13 -13
  29. package/public/his/editor/vender/date97/skin/WdatePicker.css +10 -10
  30. package/public/his/editor/vender/date97/skin/default/datepicker.css +328 -328
  31. package/public/his/editor/vender/date97/skin/ext/datepicker.css +308 -308
  32. package/public/his/editor/vender/date97/skin/whyGreen/datepicker.css +255 -255
  33. package/public/his/editor/vender/diff.js +1627 -1627
  34. package/public/his/editor/vender/editor.js +1 -1
  35. package/public/his/editor/vender/fabric.js +31187 -31187
  36. package/public/his/editor/vender/jquery/jquery.base64.js +190 -190
  37. package/public/his/editor/vender/jquery/jquery.js +10872 -10872
  38. package/public/his/editor/vender/jquery/jquery.print.js +255 -255
  39. package/public/his/editor/vender/jquery/zTreeStyle/zTreeStyle.css +96 -96
  40. package/public/his/editor/vender/mui/mui.min.css +4 -4
  41. package/public/his/editor/vender/mui/mui.min.js +5 -5
  42. package/public/his/editor/vender/mui/mui.picker.min.css +6 -6
  43. package/public/his/editor/vender/mui/mui.picker.min.js +6 -6
  44. package/public/his/editor/vender/qrcode.js +7 -7
  45. package/public/his/editor/vender/requirejs/require.js +2145 -2145
  46. package/public/his/editor/vender/signature/jSignature.CompressorSVG.js +518 -518
  47. package/public/his/editor/vender/signature/jSignature.UndoButton.js +164 -164
  48. package/public/his/editor/vender/signature/jSignature.js +1486 -1486
  49. package/public/his/editor/vender/validator.js +5094 -5094
  50. package/public/his/editor/vender/weui/weui.css +5659 -5659
  51. package/public/his/editor/vender/weui/weui.min.css +4 -4
  52. package/public/his/editor/vender/weui/weui.min.js +11 -11
  53. package/public/index.html +27 -27
  54. package/src/App.vue +188 -188
  55. package/src/ReportView.js +19 -19
  56. package/src/assets/img/querySlotDemo.svg +15 -15
  57. package/src/assets/svg/badtwo.svg +1 -1
  58. package/src/assets/svg/goodtwo.svg +1 -1
  59. package/src/base-client/components/common/AMisRender/index.js +3 -3
  60. package/src/base-client/components/common/AMisRender/index.vue +263 -263
  61. package/src/base-client/components/common/AddressSearchCombobox/AddressSearchCombobox.vue +470 -470
  62. package/src/base-client/components/common/AddressSearchCombobox/IcMapIcon.vue +16 -16
  63. package/src/base-client/components/common/AddressSearchCombobox/demo.vue +36 -36
  64. package/src/base-client/components/common/AddressSearchCombobox/ic_map.svg +6 -6
  65. package/src/base-client/components/common/AmapMarker/AmapPointRendering.vue +120 -120
  66. package/src/base-client/components/common/CitySelect/CitySelect.vue +342 -342
  67. package/src/base-client/components/common/CitySelect/index.js +3 -3
  68. package/src/base-client/components/common/CitySelect/index.md +109 -109
  69. package/src/base-client/components/common/CreateQuery/CreateQuery.vue +669 -669
  70. package/src/base-client/components/common/CreateQuery/CreateQueryItem.vue +1014 -1014
  71. package/src/base-client/components/common/CreateQuery/index.js +3 -3
  72. package/src/base-client/components/common/CreateQuery/index.md +42 -42
  73. package/src/base-client/components/common/CreateSimpleFormQuery/CreateSimpleFormQuery.vue +452 -452
  74. package/src/base-client/components/common/CreateSimpleFormQuery/CreateSimpleFormQueryItem.vue +511 -511
  75. package/src/base-client/components/common/CreateSimpleFormQuery/index.js +3 -3
  76. package/src/base-client/components/common/CreateSimpleFormQuery/index.md +42 -42
  77. package/src/base-client/components/common/FormGroupEdit/FormGroupEdit.vue +149 -149
  78. package/src/base-client/components/common/FormGroupEdit/index.js +3 -3
  79. package/src/base-client/components/common/FormGroupEdit/index.md +43 -43
  80. package/src/base-client/components/common/FormGroupQuery/FormGroupQuery.vue +166 -166
  81. package/src/base-client/components/common/FormGroupQuery/index.js +3 -3
  82. package/src/base-client/components/common/FormGroupQuery/index.md +43 -43
  83. package/src/base-client/components/common/JSONToTree/jsontotree.vue +271 -271
  84. package/src/base-client/components/common/LowCodeComponent/LowCodeEditorModal.vue +108 -108
  85. package/src/base-client/components/common/LowCodeComponent/LowCodeEditorPanel.vue +413 -413
  86. package/src/base-client/components/common/LowCodeComponent/LowCodePageOrganization.vue +502 -502
  87. package/src/base-client/components/common/LowCodeComponent/LowCodeRender.vue +728 -728
  88. package/src/base-client/components/common/LowCodeComponent/LowCodeRenderEnter.vue +29 -29
  89. package/src/base-client/components/common/LowCodeComponent/LowCodeUIStore.vue +219 -219
  90. package/src/base-client/components/common/LowCodeComponent/modal/lowCodeAddPageModal.vue +117 -117
  91. package/src/base-client/components/common/LowCodeComponent/modal/lowCodeCustomJSModal.vue +80 -80
  92. package/src/base-client/components/common/LowCodeComponent/modal/lowCodeEventEditorModal.vue +398 -398
  93. package/src/base-client/components/common/LowCodeComponent/modal/lowCodeLifeCycleModal.vue +65 -65
  94. package/src/base-client/components/common/LowCodeComponent/modal/lowCodeLogicCallbackModal.vue +64 -64
  95. package/src/base-client/components/common/LowCodeComponent/modal/lowCodeLogicParamModal.vue +73 -73
  96. package/src/base-client/components/common/LowCodeComponent/modal/lowCodeRunFunctionParamModal.vue +76 -76
  97. package/src/base-client/components/common/PersonSetting/PersonSetting.vue +208 -208
  98. package/src/base-client/components/common/PersonSetting/index.js +3 -3
  99. package/src/base-client/components/common/Tree/Tree.vue +149 -149
  100. package/src/base-client/components/common/Tree/index.js +2 -2
  101. package/src/base-client/components/common/Upload/Upload.vue +239 -239
  102. package/src/base-client/components/common/Upload/index.js +3 -3
  103. package/src/base-client/components/common/XAddForm/XAddForm.vue +105 -105
  104. package/src/base-client/components/common/XAddNativeForm/XAddNativeForm.vue +803 -803
  105. package/src/base-client/components/common/XAddNativeForm/index.md +146 -146
  106. package/src/base-client/components/common/XAddNativeFormOA/XAddNativeFormOA.vue +303 -303
  107. package/src/base-client/components/common/XAddNativeFormOA/index.js +3 -3
  108. package/src/base-client/components/common/XAddNativeFormOA/index.md +146 -146
  109. package/src/base-client/components/common/XAddReport/XAddReport.vue +180 -180
  110. package/src/base-client/components/common/XAddReport/index.js +3 -3
  111. package/src/base-client/components/common/XAddReport/index.md +56 -56
  112. package/src/base-client/components/common/XBadge/XBadge.vue +86 -86
  113. package/src/base-client/components/common/XCard/XCard.vue +64 -64
  114. package/src/base-client/components/common/XDataCard/XDataCard.vue +364 -364
  115. package/src/base-client/components/common/XDataCard/index.js +3 -3
  116. package/src/base-client/components/common/XDataCard/index.md +1 -1
  117. package/src/base-client/components/common/XDataDrawer/XDataDrawer.vue +180 -180
  118. package/src/base-client/components/common/XDataDrawer/index.js +3 -3
  119. package/src/base-client/components/common/XDataDrawer/index.md +41 -41
  120. package/src/base-client/components/common/XDescriptions/XDescriptions.vue +169 -169
  121. package/src/base-client/components/common/XDescriptions/XDescriptionsGroup.vue +304 -304
  122. package/src/base-client/components/common/XDescriptions/demo.vue +50 -50
  123. package/src/base-client/components/common/XDescriptions/index.js +3 -3
  124. package/src/base-client/components/common/XDescriptions/index.md +83 -83
  125. package/src/base-client/components/common/XDetailsView/XDetailsView.vue +238 -238
  126. package/src/base-client/components/common/XDetailsView/index.js +3 -3
  127. package/src/base-client/components/common/XForm/XForm.vue +316 -316
  128. package/src/base-client/components/common/XForm/XFormItem.vue +1152 -1152
  129. package/src/base-client/components/common/XForm/XTreeSelect.vue +207 -207
  130. package/src/base-client/components/common/XForm/index.md +178 -178
  131. package/src/base-client/components/common/XFormCol/XFormCol.vue +36 -36
  132. package/src/base-client/components/common/XFormGroup/XFormGroup.vue +250 -250
  133. package/src/base-client/components/common/XFormGroup/demo.vue +40 -40
  134. package/src/base-client/components/common/XFormGroup/index.js +3 -3
  135. package/src/base-client/components/common/XFormGroup/index.md +38 -38
  136. package/src/base-client/components/common/XFormGroupDetails/XFormGroupDetails.vue +72 -72
  137. package/src/base-client/components/common/XFormGroupDetails/index.js +3 -3
  138. package/src/base-client/components/common/XFormTable/XFormTable.vue +638 -638
  139. package/src/base-client/components/common/XFormTable/demo.vue +75 -75
  140. package/src/base-client/components/common/XFormTable/index.md +98 -98
  141. package/src/base-client/components/common/XImportExcel/XImportExcel.vue +162 -162
  142. package/src/base-client/components/common/XLicensePlate/XLicensePlate.vue +193 -193
  143. package/src/base-client/components/common/XLicensePlate/XLicensePlateDemo.vue +48 -48
  144. package/src/base-client/components/common/XReport/XReport.vue +873 -873
  145. package/src/base-client/components/common/XReport/XReportDemo.vue +304 -304
  146. package/src/base-client/components/common/XReport/XReportDesign.vue +508 -508
  147. package/src/base-client/components/common/XReport/XReportJsonRender.vue +381 -381
  148. package/src/base-client/components/common/XReport/XReportTrGroup.vue +802 -802
  149. package/src/base-client/components/common/XReport/index.js +3 -3
  150. package/src/base-client/components/common/XReport/index.md +44 -44
  151. package/src/base-client/components/common/XReport/print.js +186 -186
  152. package/src/base-client/components/common/XReportGrid/XReport.vue +936 -936
  153. package/src/base-client/components/common/XReportGrid/XReportDemo.vue +47 -47
  154. package/src/base-client/components/common/XReportGrid/XReportDesign.vue +556 -556
  155. package/src/base-client/components/common/XReportGrid/XReportJsonRender.vue +381 -381
  156. package/src/base-client/components/common/XReportGrid/XReportTrGroup.vue +948 -948
  157. package/src/base-client/components/common/XReportGrid/index.js +3 -3
  158. package/src/base-client/components/common/XReportGrid/index.md +44 -44
  159. package/src/base-client/components/common/XReportGrid/print.js +186 -186
  160. package/src/base-client/components/common/XReportSlot/XReportSlot.vue +110 -110
  161. package/src/base-client/components/common/XReportSlot/index.js +3 -3
  162. package/src/base-client/components/common/XReportSlot/index.md +48 -48
  163. package/src/base-client/components/common/XSimpleDescriptions/XSimpleDescriptions.vue +166 -166
  164. package/src/base-client/components/common/XSimpleDescriptions/index.js +3 -3
  165. package/src/base-client/components/common/XSimpleDescriptions/index.md +7 -7
  166. package/src/base-client/components/common/XStepView/XStepView.vue +252 -252
  167. package/src/base-client/components/common/XStepView/index.js +3 -3
  168. package/src/base-client/components/common/XStepView/index.md +31 -31
  169. package/src/base-client/components/common/XTab/XTab.vue +149 -149
  170. package/src/base-client/components/common/XTab/XTabDemo.vue +22 -22
  171. package/src/base-client/components/common/XTab/index.js +3 -3
  172. package/src/base-client/components/common/XTable/XTable.vue +1121 -1121
  173. package/src/base-client/components/common/XTable/index.md +255 -255
  174. package/src/base-client/components/common/XTree/XTree.vue +423 -423
  175. package/src/base-client/components/common/XTree/XTreePro.vue +418 -418
  176. package/src/base-client/components/common/XTree/index.js +3 -3
  177. package/src/base-client/components/common/XTree/index.md +36 -36
  178. package/src/base-client/components/common/XTreeOne/XTreeOne.vue +113 -113
  179. package/src/base-client/components/common/XTreeOne/XTreeOnePro.vue +128 -128
  180. package/src/base-client/components/common/richTextModal/index.vue +56 -56
  181. package/src/base-client/components/common/richTextModal/richDemo.vue +48 -48
  182. package/src/base-client/components/his/XHisEditor/XHisEditor.vue +196 -196
  183. package/src/base-client/components/his/XHisEditor/index.js +3 -3
  184. package/src/base-client/components/index.js +51 -51
  185. package/src/base-client/components/layout/XPageView/RenderRow.vue +63 -63
  186. package/src/base-client/components/layout/XPageView/XErrorView.vue +11 -11
  187. package/src/base-client/components/layout/XPageView/XPageView.vue +155 -155
  188. package/src/base-client/components/layout/XPageView/index.js +3 -3
  189. package/src/base-client/components/layout/XPageView/index.md +38 -38
  190. package/src/base-client/components/layout/XTreeView/XTreeView.vue +130 -130
  191. package/src/base-client/components/layout/XTreeView/index.js +3 -3
  192. package/src/base-client/components/layout/XTreeView/index.md +46 -46
  193. package/src/base-client/components/system/DictionaryDetailsView/DictionaryDetailsView.vue +232 -232
  194. package/src/base-client/components/system/QueryParamsDetailsView/QueryParamsDetailsView.vue +281 -281
  195. package/src/base-client/plugins/AppData.js +121 -121
  196. package/src/base-client/plugins/Config.js +19 -19
  197. package/src/base-client/plugins/GetLoginInfoService.js +183 -183
  198. package/src/base-client/plugins/tabs-page-plugin.js +39 -39
  199. package/src/bootstrap.js +39 -39
  200. package/src/components/CodeMirror/inedx.vue +118 -118
  201. package/src/components/CodeMirror/setting.js +40 -40
  202. package/src/components/FilePreview/FilePreview.vue +166 -166
  203. package/src/components/NumberInfo/NumberInfo.vue +54 -54
  204. package/src/components/STable/index.js +380 -380
  205. package/src/components/checkbox/ColorCheckbox.vue +157 -157
  206. package/src/components/checkbox/ImgCheckbox.vue +163 -163
  207. package/src/components/exception/ExceptionPage.vue +70 -70
  208. package/src/components/menu/SideMenu.vue +75 -75
  209. package/src/components/menu/menu.js +273 -273
  210. package/src/components/tool/AStepItem.vue +60 -60
  211. package/src/config/CreateQueryConfig.js +322 -322
  212. package/src/config/default/antd.config.js +89 -89
  213. package/src/config/default/setting.config.js +55 -55
  214. package/src/font-style/font.css +4 -4
  215. package/src/layouts/CommonLayout.vue +56 -56
  216. package/src/layouts/GridView.vue +45 -45
  217. package/src/layouts/PageLayout.vue +151 -151
  218. package/src/layouts/SinglePageView.vue +136 -136
  219. package/src/layouts/header/AdminHeader.vue +132 -132
  220. package/src/layouts/header/HeaderNotice.vue +177 -177
  221. package/src/layouts/tabs/TabsHead.vue +189 -189
  222. package/src/layouts/tabs/TabsView.vue +389 -389
  223. package/src/lib.js +1 -1
  224. package/src/main.js +30 -30
  225. package/src/mock/extend/index.js +84 -84
  226. package/src/mock/goods/index.js +108 -108
  227. package/src/pages/AMisDemo/AMisDemo.vue +325 -325
  228. package/src/pages/AMisDemo/AMisDemo2.vue +74 -74
  229. package/src/pages/DefaultExample/index.vue +77 -77
  230. package/src/pages/DynamicStatistics/ChartSelector.vue +331 -331
  231. package/src/pages/DynamicStatistics/DataTabs.vue +83 -83
  232. package/src/pages/DynamicStatistics/DynamicTable.vue +128 -128
  233. package/src/pages/DynamicStatistics/EvaluationArea.vue +69 -69
  234. package/src/pages/DynamicStatistics/FavoriteList.vue +51 -51
  235. package/src/pages/DynamicStatistics/QuestionHistoryAndFavorites.vue +591 -591
  236. package/src/pages/DynamicStatistics/SearchBar.vue +192 -192
  237. package/src/pages/DynamicStatistics/index.vue +282 -282
  238. package/src/pages/Example/childIndex.vue +15 -15
  239. package/src/pages/Example/index.vue +30 -30
  240. package/src/pages/NewDynamicStatistics/ChartSelector.vue +331 -331
  241. package/src/pages/NewDynamicStatistics/DataTabs.vue +122 -122
  242. package/src/pages/NewDynamicStatistics/DynamicTable.vue +128 -128
  243. package/src/pages/NewDynamicStatistics/EvaluationArea.vue +69 -69
  244. package/src/pages/NewDynamicStatistics/FavoriteList.vue +51 -51
  245. package/src/pages/NewDynamicStatistics/QuestionHistoryAndFavorites.vue +289 -289
  246. package/src/pages/NewDynamicStatistics/SearchBar.vue +193 -193
  247. package/src/pages/NewDynamicStatistics/index.vue +258 -258
  248. package/src/pages/ReportGrid/index.vue +76 -76
  249. package/src/pages/ServiceReview/index.vue +284 -284
  250. package/src/pages/SubExample/index.vue +26 -26
  251. package/src/pages/WorkflowDetail/WorkFlowDemo.vue +32 -32
  252. package/src/pages/WorkflowDetail/WorkflowDetail.vue +230 -230
  253. package/src/pages/WorkflowDetail/WorkflowPageDetail/LeaveMessage.vue +131 -131
  254. package/src/pages/WorkflowDetail/WorkflowPageDetail/TrimTextTail.vue +23 -23
  255. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowBaseInformation.vue +302 -302
  256. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowBaseInformationDetails.vue +276 -276
  257. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowHandle.vue +863 -863
  258. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowHandleReso.vue +997 -997
  259. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkFlowTimeline.vue +222 -222
  260. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkOrderParentDetails.vue +233 -233
  261. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkflowDetailResso.vue +261 -261
  262. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkflowListResolution.vue +248 -248
  263. package/src/pages/WorkflowDetail/WorkflowPageDetail/WorkflowLog.vue +73 -73
  264. package/src/pages/XPageViewExample/index.vue +38 -38
  265. package/src/pages/XReportView/index.vue +64 -64
  266. package/src/pages/XTreeOneProExample/index.vue +67 -67
  267. package/src/pages/login/Login.vue +378 -378
  268. package/src/pages/login/LoginV3.vue +389 -389
  269. package/src/pages/lowCode/lowCodeEditor.vue +1219 -1219
  270. package/src/pages/lowCode/lowCodeRenderPage.vue +43 -43
  271. package/src/pages/resourceManage/orgListManage.vue +98 -98
  272. package/src/pages/system/dictionary/index.vue +44 -44
  273. package/src/pages/system/monitor/loginInfor/index.vue +37 -37
  274. package/src/pages/system/monitor/operLog/index.vue +37 -37
  275. package/src/pages/system/settings/modifyPassword.vue +117 -117
  276. package/src/pages/system/ticket/index.vue +480 -480
  277. package/src/pages/system/ticket/submitTicketSuccess.vue +484 -484
  278. package/src/plugins/EventLogPlugin.js +33 -33
  279. package/src/plugins/FindParentsData.js +17 -17
  280. package/src/router/async/config.async.js +34 -34
  281. package/src/router/async/router.map.js +179 -179
  282. package/src/router/guards.js +264 -264
  283. package/src/router/index.js +27 -27
  284. package/src/router.js +19 -19
  285. package/src/services/api/TicketDetailsViewApi.js +46 -46
  286. package/src/services/api/cas.js +79 -79
  287. package/src/services/api/common.js +329 -329
  288. package/src/services/api/entity.js +18 -18
  289. package/src/services/api/index.js +17 -17
  290. package/src/services/api/restTools.js +67 -67
  291. package/src/services/api/workFlow.js +63 -63
  292. package/src/services/apiService.js +15 -15
  293. package/src/services/user.js +90 -90
  294. package/src/services/v3Api.js +81 -81
  295. package/src/store/modules/account.js +115 -115
  296. package/src/store/modules/index.js +5 -5
  297. package/src/store/modules/lowCode.js +33 -33
  298. package/src/store/modules/setting.js +119 -119
  299. package/src/theme/default/style.less +58 -58
  300. package/src/theme/global.less +159 -159
  301. package/src/utils/authority-utils.js +85 -85
  302. package/src/utils/errorCode.js +6 -6
  303. package/src/utils/formatter.js +80 -80
  304. package/src/utils/htmlToPDF.js +108 -108
  305. package/src/utils/htmlToPDFApi.js +5 -5
  306. package/src/utils/indexedDB.js +258 -258
  307. package/src/utils/login.js +188 -188
  308. package/src/utils/lowcode/lowcodeComponentMixin.js +120 -120
  309. package/src/utils/lowcode/lowcodeLog.js +29 -29
  310. package/src/utils/lowcode/lowcodeUtils.js +373 -373
  311. package/src/utils/lowcode/registerComponentForEditor.js +1 -1
  312. package/src/utils/lowcode/registerComponentForRender.js +11 -11
  313. package/src/utils/map-utils.js +47 -47
  314. package/src/utils/reg.js +95 -95
  315. package/src/utils/request.js +362 -362
  316. package/src/utils/routerUtil.js +450 -450
  317. package/src/utils/runEvalFunction.js +14 -14
  318. package/src/utils/util.js +281 -281
  319. package/src/utils/waterMark.js +31 -31
  320. package/test/Amis.spec.js +163 -163
  321. package/test/Tree.spec.js +167 -167
  322. package/test/myDialog.spec.js +46 -46
  323. package/vue.config.js +196 -196
  324. package//350/277/201/347/247/273/346/227/245/345/277/227.md +15 -15
@@ -1,936 +1,936 @@
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
- @selectRow="selectRow"
12
- v-if="scanFinish"
13
- :show-img-in-cell="showImgInCell"
14
- :img-prefix="imgPrefix"
15
- :use-oss-for-img="useOssForImg"
16
- :display-only="displayOnly"
17
- :config="type === 'display' ? originalConfig : activeConfig"
18
- :slot-config-name="type === 'display' ? undefined : activatedSlotName"
19
- :for-display="type === 'display'"
20
- ref="XReportDesign"
21
- id="printReady"
22
- :server-name="serverName"
23
- :env="env"
24
- :show-title="showTitle"
25
- :no-padding="noPadding"
26
- :no-top-border="noTopBorder"
27
- :show-images="hasImages"
28
- :image-list="imageList">
29
- </XReportDesign>
30
- </template>
31
- <template v-else>
32
- <!-- 用以包裹整个页面 -->
33
- <a-card v-if="!showSkeleton">
34
- <!-- 切换菜单 -->
35
- <a-radio-group
36
- v-model="type"
37
- default-value="a"
38
- button-style="solid"
39
- @change="tabChanged"
40
- v-show="!onlyDisplay && editMode">
41
- <a-radio-button value="design" v-if="!onlyDisplay">
42
- 设计
43
- </a-radio-button>
44
- <a-radio-button value="display" style="border-radius: 0">
45
- 预览
46
- </a-radio-button>
47
- </a-radio-group>
48
- <a-radio-button @click="saveConfig" style="border-radius: 0 4px 4px 0" v-if="showSaveButton">
49
- 保存
50
- </a-radio-button>
51
- <!-- 主体表格 -->
52
- <XReportDesign
53
- v-if="scanFinish"
54
- @updateImg="updateImg"
55
- @selectRow="selectRow"
56
- :show-img-in-cell="showImgInCell"
57
- :img-prefix="imgPrefix"
58
- :use-oss-for-img="useOssForImg"
59
- :display-only="displayOnly"
60
- :config="type === 'display' ? originalConfig : activeConfig"
61
- :slot-config-name="type === 'display' ? undefined : activatedSlotName"
62
- :for-display="type === 'display'"
63
- :no-padding="noPadding"
64
- :no-top-border="noTopBorder"
65
- :show-title="showTitle"
66
- ref="XReportDesign"
67
- id="printReady"
68
- :server-name="serverName"
69
- :env="env"
70
- :show-images="hasImages"
71
- :image-list="imageList">
72
- </XReportDesign>
73
- </a-card>
74
- </template>
75
- <!-- 弹出框 -->
76
- <x-add-report
77
- :env="env"
78
- ref="xAddReport"
79
- />
80
- </div>
81
- </template>
82
-
83
- <script>
84
- // 转PDF用
85
- import HtmlToPdf from '@vue2-client/utils/htmlToPDF'
86
- import { getConfigByName, runLogic } from '@vue2-client/services/api/common'
87
- import XReportDesign from './XReportDesign.vue'
88
- import { printElement } from './print'
89
- // import XAddReport from '@vue2-client/base-client/components/common/XAddReport'
90
-
91
- export default {
92
- name: 'XReport',
93
- props: {
94
- files: {
95
- type: Array,
96
- default: () => {
97
- return []
98
- }
99
- },
100
- // 控制用户权限,user和admin
101
- authority: {
102
- type: String,
103
- default: 'user'
104
- },
105
- // 是否为编辑模式
106
- editMode: {
107
- type: Boolean,
108
- default: true
109
- },
110
- // 配置名
111
- configName: {
112
- type: String,
113
- required: true
114
- },
115
- // 插槽名
116
- activatedSlotName: {
117
- type: String,
118
- default: undefined
119
- },
120
- // 本地配置,调试用
121
- localConfig: {
122
- type: Object,
123
- default: undefined
124
- },
125
- // 兼容老版本配置
126
- dontFormat: {
127
- type: Boolean,
128
- default: true
129
- },
130
- showImgInCell: {
131
- type: Boolean,
132
- default: false
133
- },
134
- // 数据
135
- configData: {
136
- type: Object,
137
- default: undefined
138
- },
139
- // 命名空间
140
- serverName: {
141
- type: String,
142
- default: process.env.VUE_APP_SYSTEM_NAME
143
- },
144
- // 环境
145
- env: {
146
- type: String,
147
- default: 'prod'
148
- },
149
- // 只做展示
150
- displayOnly: {
151
- type: Boolean,
152
- default: true
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: false
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
- XAddReport: () => import('@vue2-client/base-client/components/common/XAddReport'),
192
- XReportDesign
193
- },
194
- data () {
195
- return {
196
- // 控制骨架屏显隐
197
- showSkeleton: true,
198
- // 配置
199
- config: undefined,
200
- // 当前显示模式,编辑模式,预览模式
201
- type: 'design',
202
- // 仅供展示,不可编辑
203
- onlyDisplay: false,
204
- // 每行最大列数,非必要请勿更改,现在的设计器完全是基于每行12列来设计的
205
- maxColSpan: 12,
206
- // 定义是否完成配置的扫描,未完成不要渲染子组件
207
- scanFinish: false,
208
- // 当前激活的配置文件
209
- activeConfig: null,
210
- // 原始配置文件
211
- // 用于展示。某些情况下“设计页”中的内容仅为“预览页”表格其中的一部分
212
- originalConfig: null,
213
- // 扫描到的配置
214
- configFromWeb: {},
215
- // 用于获取配置的锁
216
- timer: undefined,
217
- // 是否包含图片
218
- hasImages: false,
219
- // 图片列表
220
- imageList: [],
221
- // 保存最原始的数据,用于判断哪些数据被更改了
222
- dataCache: undefined,
223
- // 判断哪些数据被更改了,存储对应的key
224
- diff: [],
225
- }
226
- },
227
- beforeDestroy () {
228
- clearInterval(this.timer)
229
- },
230
- watch: {
231
- // 如果配置名更改了,重新获取配置
232
- configName (val) {
233
- if (val) {
234
- getConfigByName(this.configName, undefined, res => {
235
- this.config = res
236
- this.configInit()
237
- }, this.env === 'dev')
238
- }
239
- },
240
- // 如果本地配置更改了,重新初始化
241
- localConfig: {
242
- deep: true,
243
- immediate: true,
244
- handler (val) {
245
- if (val) {
246
- this.config = val
247
- this.configInit()
248
- }
249
- }
250
- },
251
- },
252
- provide () {
253
- return {
254
- runLogic: runLogic,
255
- openDialog: this.openDialog,
256
- registerComponent: this.registerComponent,
257
- getComponentByName: this.getComponentByName,
258
- getParentComponentByName: this.getComponentByName,
259
- getConfigByName: getConfigByName
260
- }
261
- },
262
- methods: {
263
- // 把组件注册到refs中,方便调用
264
- registerComponent (componentName, component) {
265
- console.log('内部注册', this.$options.name, componentName)
266
- this.$refs[componentName] = component
267
- console.log('内部注册完成', this.$refs)
268
- },
269
- // 根据名字从注册到组件中获取组件
270
- getComponentByName (componentName) {
271
- console.log('内部取组件', this.$options.name, componentName)
272
- console.log('内部组件内容', this.$refs)
273
- return this.$refs[componentName]
274
- },
275
- /**
276
- * @param configName 栅格配置名称
277
- * @param selectedId 选中得id
278
- * @param mixinData 需要混入得数据
279
- * @param env 其他传递给打开窗口的数据
280
- */
281
- openDialog (configName, selectedId, mixinData, outEnv) {
282
- console.log('openDialog', configName, selectedId)
283
- this.$refs.xAddReport.init({
284
- configName: configName,
285
- selectedId: selectedId,
286
- mixinData: mixinData,
287
- outEnv: outEnv
288
- })
289
- },
290
- // 向外暴露图片修改后的数据,某些外部需要自己管理图片的保存与修改
291
- updateImg (data) {
292
- this.$emit('updateImg', data)
293
- },
294
- // 导出数据,某些外部需要统一控制数据的变动
295
- exportData () {
296
- // 获取当前修改后的数据
297
- let tempData
298
- if (this.activeConfig === undefined || this.activeConfig === null) {
299
- tempData = this.originalConfig.data
300
- } else {
301
- const tempDataKeys = Object.keys(this.activeConfig.tempData)
302
- tempDataKeys.forEach(key => {
303
- this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
304
- })
305
- tempData = this.activeConfig.data
306
- }
307
- // 对比数据的差异
308
- this.diff = []
309
- this.compareProps(tempData, this.dataCache)
310
- this.diff.forEach(eachDiff => {
311
- const arr = eachDiff.split('.')
312
- let targetData = tempData[arr[0]]
313
- if (arr.length !== 1) {
314
- for (let i = 1; i < arr.length - 1; i++) {
315
- const path = arr[i]
316
- targetData = targetData[path]
317
- }
318
- }
319
- // 将修改的数据,添加update = true属性
320
- targetData.update = true
321
- })
322
- return tempData
323
- },
324
- // 对比两个obj有哪里不同
325
- compareProps (obj1, obj2, path = '') {
326
- for (const key in obj1) {
327
- // 如果一个是undefined
328
- if (typeof obj2[key] === 'undefined') {
329
- this.diff.push(path + key)
330
- // 如果是数组长度不一样
331
- } else if (Array.isArray(obj1) && Array.isArray(obj2)) {
332
- if (obj1[key].length !== obj2[key].length) {
333
- this.diff.push(path + key)
334
- }
335
- // 如果都是对象,并且存在同样的key,递归子key
336
- } else if (typeof obj1[key] === 'object' && typeof obj2[key] === 'object') {
337
- this.compareProps(obj1[key], obj2[key], path + key + '.')
338
- // 如果不是obj,对比其数据
339
- } else if (obj1[key] !== obj2[key]) {
340
- this.diff.push(path + key)
341
- }
342
- }
343
- },
344
- selectRow (selectedRowKeys, selectedRows) {
345
- this.table_selectedRowKeys = selectedRowKeys
346
- this.table_selectedRows = selectedRows
347
- console.log('')
348
- this.$emit('selectRow', selectedRowKeys, selectedRows)
349
- },
350
- // 注册组件到$refs中
351
- registerComponentToRefs (componentName, component) {
352
- this.$refs[componentName] = component
353
- },
354
-
355
- // 正常的保存方法,当前修改内容会直接全部导出到外部
356
- saveConfig () {
357
- if (this.activeConfig === undefined || this.activeConfig === null) {
358
- return this.originalConfig.data
359
- } else {
360
- const tempDataKeys = Object.keys(this.activeConfig.tempData)
361
- tempDataKeys.forEach(key => {
362
- this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
363
- })
364
- this.$emit('saveConfig', this.$refs.XReportDesign.activatedConfig)
365
- }
366
- },
367
- // 通过@@@分割临时变量,找到对应的key,并修改它的值
368
- changeDeepObject (obj, strPath, newVal) {
369
- const arr = strPath.split('@@@')
370
- if (obj[arr[0]] === undefined) {
371
- obj = obj.images
372
- }
373
- if (arr.length === 1) {
374
- obj[arr[0]] = newVal
375
- } else {
376
- let result = obj[arr[0]]
377
- arr.shift()
378
- while (arr.length > 1) {
379
- result = result[arr[0]]
380
- arr.shift()
381
- }
382
- if (result) {
383
- result[arr[0]] = newVal
384
- }
385
- }
386
- },
387
- // 检查slot是否在配置文件中包含,如果没有包含,则视为非法获取
388
- checkSlotDefine (config) {
389
- const slotsDeclare = config.slotsDeclare
390
- const total = slotsDeclare.length
391
- let count = 0
392
- slotsDeclare.forEach(declare => {
393
- config.columns.forEach(row => {
394
- row.forEach(cell => {
395
- if (cell.slotConfig === declare) {
396
- count++
397
- }
398
- })
399
- })
400
- })
401
-
402
- return count === total
403
- },
404
- // 切换了标签页
405
- tabChanged (key) {
406
- this.scanFinish = false
407
- this.originalConfig.data = { ...this.originalConfig.data, ...this.config.data }
408
- this.config.data = this.originalConfig.data
409
- if (this.type === 'display') {
410
- const tempDataKeys = Object.keys(this.activeConfig.tempData)
411
- tempDataKeys.forEach(key => {
412
- this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
413
- })
414
- let count = 0
415
- this.imageList = []
416
- const keys = Object.keys(this.config.data.images)
417
- keys.forEach(key => {
418
- if (this.config.data.images[key].length > 0) {
419
- this.imageList = [...this.imageList, ...this.config.data.images[key]]
420
- count++
421
- }
422
- })
423
- this.hasImages = count > 0
424
- } else {
425
- this.hasImages = false
426
- }
427
- this.$nextTick(() => {
428
- this.scanFinish = true
429
- })
430
- },
431
- // 获取当前日期,为保存文件命名用
432
- getDate () {
433
- const currentDate = new Date()
434
-
435
- const year = currentDate.getFullYear()
436
- const month = String(currentDate.getMonth() + 1).padStart(2, '0')
437
- const day = String(currentDate.getDate()).padStart(2, '0')
438
-
439
- const formattedDate = `${year}_${month}_${day}`
440
-
441
- return formattedDate
442
- },
443
- // 打印
444
- printDocument () {
445
- // x-report
446
- const printContent = window.rawDocument.getElementById('printReady')
447
- printElement(printContent)
448
- this.$message.success('操作成功!')
449
- },
450
- // 导出PDF
451
- exportPDF () {
452
- const date = this.getDate()
453
- let title = this.config.title
454
- title = title.replace(/<[^>]+>/g, '')
455
- const fileName = date + '' + title
456
- HtmlToPdf.getPdf(fileName, '#printReady')
457
- },
458
- // 用于分割配置中的colums,将需要处理的数组提取出来
459
- formatConfigRow () {
460
- for (let i = 0; i < this.config.columns.length; i++) {
461
- // 对原始数组进行递归,依次将该位置拆分为三个部分,当前处理位置之前的,当前处理位置,当前处理位置之后的
462
- const before = this.config.columns.slice(0, i)
463
- const after = this.config.columns.slice(i + 1, this.config.columns.length)
464
-
465
- // 将当前处理的数组交给处理的方法
466
- const x = this.checkRow(this.config.columns[i])
467
-
468
- const newArr = []
469
-
470
- // 拼接之前的数组
471
- if (before.length > 0) {
472
- if (before.length >= 1) {
473
- before.forEach(item => {
474
- newArr.push(item)
475
- })
476
- } else {
477
- newArr.push(before)
478
- }
479
- }
480
-
481
- // 拼接不需要更改当前节点处理完成的数组
482
- newArr.push(x.old)
483
-
484
- // 如果处理了新加的数据,拼接
485
- if (x.add.length > 0) {
486
- for (let j = 0; j < x.add.length; j++) {
487
- if (x.add[j]) {
488
- newArr.push(x.add[j])
489
- i++
490
- }
491
- }
492
- }
493
-
494
- // 拼接之后的数组
495
- if (after.length > 0) {
496
- if (after.length >= 1) {
497
- after.forEach(item => {
498
- newArr.push(item)
499
- })
500
- } else {
501
- newArr.push(after)
502
- }
503
- }
504
-
505
- this.config.columns = newArr
506
- }
507
- },
508
- // 路径中含有@@@的key,将其解析,并返回其数据
509
- getDeepObject (obj, strPath) {
510
- const arr = strPath.split('@@@')
511
- let result = obj[arr[0]]
512
- arr.shift()
513
- try {
514
- while (arr.length > 0) {
515
- result = result[arr[0]]
516
- arr.shift()
517
- }
518
- } catch (e) {
519
- result = undefined
520
- }
521
- return result
522
- },
523
- // 处理colums数组,为声明了rowspan的单元格,自动匹配格式
524
- checkRow (rowArr) {
525
- // 不需要更改的数据
526
- const original = []
527
- // 需要更改新加的数据
528
- const addArr = []
529
- // 统计rowspan出现的总和
530
- let count = 0
531
- // 统计声明列需要的rowspan总数
532
- let total = 0
533
- // 是否为声明行
534
- let titleCellFlag = false
535
- // 是否为声明行后第一行
536
- let firstSubLine = false
537
- // 需要处理的行,新的index值
538
- let subRowIndex = 0
539
- // 新生成的行,临时存储
540
- const waitForAddArr = []
541
- // 用于记录声明行的colspan避免格式错误
542
- let preColSpan = 0
543
- // 用于统计循环次数,判断是否是最后一次
544
- let forEachCount = 0
545
-
546
- // 标记所有数据
547
- rowArr.forEach(cell => {
548
- forEachCount++
549
- // 如果该行没有rowspan则默认其为1,不要影响统计结果
550
- if (!cell.rowSpan) {
551
- cell.rowSpan = 0
552
- }
553
-
554
- if (cell.text && total !== 0) { // 如果遇到了下一个声明行,证明rowspan少了一行,需要补充一个占位格
555
- const nullObj = {
556
- type: 'placeHolderColumn',
557
- order: subRowIndex,
558
- noBoarder: true,
559
- needSplit: true,
560
- colSpan: preColSpan,
561
- dontShowRow: true
562
- }
563
- subRowIndex++
564
- waitForAddArr.push(nullObj)
565
- total = 0
566
- count = 0
567
- titleCellFlag = false
568
- firstSubLine = false
569
- } else if ((total !== count + cell.rowSpan) && forEachCount === rowArr.length) {
570
- // 如果没有遇到了下一个声明行,但已经是当前行最后一个数据,也证明rowspan少了一行,需要补充一个占位格
571
- const nullObj = {
572
- type: 'placeHolderColumn',
573
- order: subRowIndex,
574
- noBoarder: true,
575
- needSplit: true,
576
- colSpan: preColSpan,
577
- dontShowRow: true
578
- }
579
- subRowIndex++
580
- waitForAddArr.push(nullObj)
581
- total = 0
582
- count = 0
583
- titleCellFlag = false
584
- firstSubLine = false
585
- }
586
-
587
- // 判断是否为声明行
588
- if (cell.text && total === 0) {
589
- // 将声明行声明的rowspan作为总数,判断下方rowspan相加是否等于声明行声明的数量
590
- total = cell.rowSpan
591
- titleCellFlag = false
592
- firstSubLine = true
593
- subRowIndex = 1
594
- cell.show = true
595
- cell.showRowSpan = total
596
- } else if (cell.rowSpan > 0 && !titleCellFlag && firstSubLine) { // 判断是否为声明行后首行,因为首行不需要移动
597
- count += cell.rowSpan
598
- firstSubLine = false
599
- cell.noBoarder = true
600
- cell.show = true
601
- cell.showRowSpan = total
602
- } else if (cell.rowSpan > 0 && !titleCellFlag && !firstSubLine) { // 既非声明行,也非首行,需要移动
603
- count += cell.rowSpan
604
- // cell.type = 'notShow'
605
- cell.needSplit = true
606
- cell.order = subRowIndex
607
- cell.dontShowRow = true
608
- subRowIndex++
609
-
610
- // 如果之前添加过空行补充位置,刚好最后一位还有内容,将其互换
611
- if (forEachCount === rowArr.length && !waitForAddArr[waitForAddArr.length - 1].dataIndex) {
612
- waitForAddArr[waitForAddArr.length - 1].order += 1
613
- cell.order -= 1
614
- waitForAddArr.push(cell)
615
- } else {
616
- waitForAddArr.push(cell)
617
- }
618
- }
619
-
620
- // 如果count和total相等了,证明已经处理完成。将计数器还原
621
- if (count === total) {
622
- total = 0
623
- count = 0
624
- titleCellFlag = false
625
- firstSubLine = false
626
- }
627
- // 保存上一个的colspan,保持生成的格子与原格式一致
628
- preColSpan = cell.colSpan
629
- })
630
-
631
- // 将所有不需要移动的放入original
632
- rowArr.forEach(cell => {
633
- if (cell.needSplit !== true) {
634
- original.push(cell)
635
- }
636
- })
637
-
638
- // 增加新的数组
639
- waitForAddArr.forEach(cell => {
640
- const target = cell.order
641
- // if (cell.type === 'notShow') {
642
- // cell.type = 'inputs'
643
- // }
644
- cell.noBoarder = true
645
- if (addArr[target] === undefined) {
646
- const temp = []
647
- temp.push(cell)
648
- addArr[target] = temp
649
- } else if (addArr[target].length > 0) {
650
- addArr[target].push(cell)
651
- }
652
- })
653
-
654
- // 如果没有新增,将单元格边框设置为显示
655
- if (addArr.length < 1) {
656
- original.forEach(cell => {
657
- if (cell.type === 'input' || cell.type === 'inputs') {
658
- cell.noBoarder = false
659
- }
660
- })
661
- }
662
-
663
- return {
664
- old: original,
665
- add: addArr
666
- }
667
- },
668
- // 扫描配置,如果有插槽则拼接插槽
669
- scanConfigSlot (config) {
670
- const columnsArr = config.columns
671
- for (let i = 0; i < columnsArr.length; i++) {
672
- for (let j = 0; j < columnsArr[i].length; j++) {
673
- // 如果发现type为slot,开始匹配对应的slot配置文件
674
- if (columnsArr[i][j].type === 'slot') {
675
- const targetName = columnsArr[i][j].slotConfig
676
- // 找不到目标插槽配置
677
- if (!this.configFromWeb[targetName] || !this.configFromWeb[targetName].columns) {
678
- console.error('无法找到目标插槽的配置!')
679
- return
680
- }
681
-
682
- // 替换columns,合并data
683
- config.columns[i] = []
684
- const before = config.columns.slice(0, i)
685
- let after = config.columns.slice(i + 1, config.columns.length)
686
-
687
- const addArr = []
688
- for (let k = 0; k < this.configFromWeb[targetName].columns.length; k++) {
689
- const temp = []
690
- this.configFromWeb[targetName].columns[k].forEach(cell => {
691
- temp.push(cell)
692
- })
693
- addArr.push(temp)
694
- }
695
-
696
- const newArr = []
697
- // 拼接之前的数组
698
- if (before.length > 0) {
699
- if (before.length >= 1) {
700
- before.forEach(item => {
701
- newArr.push(item)
702
- })
703
- } else {
704
- newArr.push(before)
705
- }
706
- }
707
-
708
- addArr.forEach(arr => {
709
- newArr.push(arr)
710
- })
711
-
712
- // 拼接之后的数组
713
- if (after.length === 1) {
714
- if (after[0].type === 'slot' || after[0][0].type === 'slot') {
715
- after = []
716
- }
717
- }
718
- if (after.length > 0) {
719
- if (after.length >= 1) {
720
- after.forEach(item => {
721
- newArr.push(item)
722
- })
723
- } else {
724
- newArr.push(after)
725
- }
726
- }
727
-
728
- config.columns = newArr
729
- if (this.configFromWeb[targetName].slotsDeclare) {
730
- config.slotsDeclare = this.configFromWeb[targetName].slotsDeclare
731
- } else {
732
- config.slotsDeclare = []
733
- }
734
-
735
- if (config.data.images && this.configFromWeb[targetName].data.images) {
736
- config.data.images = { ...config.data.images, ...this.configFromWeb[targetName].data.images }
737
- delete this.configFromWeb[targetName].data.images
738
- }
739
- config.data = { ...config.data, ...this.configFromWeb[targetName].data }
740
- }
741
- }
742
- }
743
- this.config = config
744
- },
745
- // 扫描所有插槽名
746
- scanConfigName (config, resut) {
747
- if (config.slotsDeclare) {
748
- config.slotsDeclare.forEach(name => {
749
- resut.push(name)
750
- })
751
- }
752
- },
753
- // 获取插槽
754
- getConfigAndJoin (config, outerLock) {
755
- // 检查主配置插槽声明是否合法
756
- const check = this.checkSlotDefine(config)
757
- const waitForDownloadSlotName = []
758
- if (check) {
759
- // 扫描主配置中声明的插槽名
760
- this.scanConfigName(config, waitForDownloadSlotName)
761
-
762
- const total = waitForDownloadSlotName.length
763
- let count = 0
764
-
765
- // 挨个获取插槽
766
- waitForDownloadSlotName.forEach(configName => {
767
- getConfigByName(configName, this.serverName, res => {
768
- this.configFromWeb[configName] = res
769
- count++
770
- }, this.env === 'dev')
771
- })
772
-
773
- // 使用定时器循环判断锁状态,用于多个插槽,要等待统一获取完成之后,再进行下一步初始化
774
- const timer = setInterval(() => {
775
- console.log('插槽下载进度,当前:' + count + '/' + total)
776
- if (count >= total) {
777
- clearInterval(timer)
778
- this.scanConfigSlot(config)
779
- if (config.slotsDeclare.length > 0) {
780
- const lock = { status: true }
781
- this.getConfigAndJoin(config, lock)
782
- const innerTimer = setInterval(() => {
783
- if (!lock.status) {
784
- clearInterval(innerTimer)
785
- outerLock.status = false
786
- }
787
- }, 100)
788
- } else {
789
- outerLock.status = false
790
- }
791
- }
792
- }, 100)
793
- } else {
794
- console.error('插槽配置有误!')
795
- outerLock.status = false
796
- }
797
- },
798
- // 获取配置之后的初始化
799
- configInit () {
800
- console.log('拼接完成', this.config)
801
- // 将初始化好的配置拷贝一份留存
802
- this.originalConfig = Object.assign({}, this.config)
803
- if (!this.dontFormat) {
804
- // 扫描配置文件中有没有rowSpan,进行格式化调整
805
- this.formatConfigRow(this.config)
806
- }
807
- this.activeConfig = this.config
808
- this.showSkeleton = false
809
- // 判断是否有动态Index
810
- this.activeConfig.columns.forEach(row => {
811
- row.forEach(cell => {
812
- if (cell.dynamicDataIndex === true) {
813
- // 如果有动态index,取其函数,运行函数得到真实index保存
814
- // eslint-disable-next-line no-eval
815
- const func = eval('(' + cell.customFunctionForDynamicDataIndex + ')')
816
- cell.dataIndex = func(this.config)
817
- }
818
- // 处理 自定义函数的旧逻辑
819
- if (['action', 'click'].includes(cell.eventType) && cell.customFunction && !cell.events) {
820
- cell.events = []
821
- cell.events.push({
822
- type: cell.eventType,
823
- customFunction: cell.customFunction
824
- })
825
- }
826
- })
827
- })
828
- // 将数据复制到临时数据中,带有@@@的数据,我们将其整体作为一个key保存,当编辑完成后,再将其解析,回填到需要的数据中
829
- this.activeConfig.tempData = {}
830
- // 是否有@@@深层引用
831
- this.activeConfig.columns.forEach(row => {
832
- row.forEach(cell => {
833
- // 将@@@解析
834
- if (cell.dataIndex !== undefined && cell.dataIndex.indexOf('@@@') !== -1) {
835
- this.activeConfig.tempData[cell.dataIndex] = this.getDeepObject(this.activeConfig.data, cell.dataIndex)
836
- }
837
- })
838
- })
839
- this.$nextTick(() => {
840
- this.scanFinish = true
841
- })
842
- },
843
- // 初始化JSON配置
844
- jsonConfigInit () {
845
- if (this.configData === undefined) {
846
- console.error('未找到数据!')
847
- } else {
848
- this.originalConfig = Object.assign({}, this.config)
849
- this.originalConfig.data = JSON.parse(JSON.stringify(this.configData))
850
- this.type = 'display'
851
- // this.onlyDisplay = true
852
- this.showSkeleton = false
853
- this.$nextTick(() => {
854
- this.scanFinish = true
855
- })
856
- }
857
- },
858
- onSubmit () {
859
- console.log('this.table_selectedRowKeys', this.table_selectedRowKeys)
860
- console.log('this.table_selectedRows', this.table_selectedRows)
861
- },
862
- },
863
- beforeMount () {
864
- // 如果只是展示
865
- if (this.displayOnly) {
866
- this.onlyDisplay = true
867
- this.type = 'display'
868
- }
869
- // 如果有本地配置,优先使用本地配置
870
- if (this.localConfig) {
871
- // 如果配置是json渲染器
872
- if (this.localConfig.designMode === 'json') {
873
- this.config = this.localConfig
874
- if (this.configData !== undefined) {
875
- this.config.data = this.configData
876
- }
877
- this.jsonConfigInit()
878
- } else {
879
- // 如果配置是普通渲染器
880
- this.config = this.localConfig
881
- if (this.configData !== undefined) {
882
- this.config.data = this.configData
883
- }
884
- if (this.config.data.images === undefined) {
885
- this.config.data.images = {}
886
- }
887
- this.configInit()
888
- }
889
- } else {
890
- // 如果本地配置没有值,则从琉璃中获取
891
- getConfigByName(this.configName, this.serverName, res => {
892
- this.config = res
893
- if (this.config.designMode === 'json') {
894
- if (this.configData !== undefined) {
895
- this.config.data = this.configData
896
- }
897
- this.jsonConfigInit()
898
- } else {
899
- if (this.configData !== undefined) {
900
- this.config.data = this.configData
901
- }
902
- if (this.config.data.images === undefined) {
903
- this.config.data.images = {}
904
- }
905
- this.configInit()
906
- }
907
- }, this.env === 'dev')
908
- }
909
- },
910
- mounted () {
911
- // 如果外界传来了registerMap,我们将本VM对象注册到map中
912
- if (this.registerMap !== undefined) {
913
- this.registerMap.push(this)
914
- }
915
- // 将原始数据备份保存
916
- if (this.configData) {
917
- this.dataCache = JSON.parse(JSON.stringify(this.configData))
918
- } else {
919
- if (this.config?.data) {
920
- this.dataCache = JSON.parse(JSON.stringify(this.config.data))
921
- }
922
- }
923
- }
924
- }
925
- </script>
926
-
927
- <style lang="less" scoped>
928
- .tools {
929
- text-align: center;
930
- cursor: pointer;
931
-
932
- .toolsItem {
933
- display: inline-block;
934
- }
935
- }
936
- </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
+ @selectRow="selectRow"
12
+ v-if="scanFinish"
13
+ :show-img-in-cell="showImgInCell"
14
+ :img-prefix="imgPrefix"
15
+ :use-oss-for-img="useOssForImg"
16
+ :display-only="displayOnly"
17
+ :config="type === 'display' ? originalConfig : activeConfig"
18
+ :slot-config-name="type === 'display' ? undefined : activatedSlotName"
19
+ :for-display="type === 'display'"
20
+ ref="XReportDesign"
21
+ id="printReady"
22
+ :server-name="serverName"
23
+ :env="env"
24
+ :show-title="showTitle"
25
+ :no-padding="noPadding"
26
+ :no-top-border="noTopBorder"
27
+ :show-images="hasImages"
28
+ :image-list="imageList">
29
+ </XReportDesign>
30
+ </template>
31
+ <template v-else>
32
+ <!-- 用以包裹整个页面 -->
33
+ <a-card v-if="!showSkeleton">
34
+ <!-- 切换菜单 -->
35
+ <a-radio-group
36
+ v-model="type"
37
+ default-value="a"
38
+ button-style="solid"
39
+ @change="tabChanged"
40
+ v-show="!onlyDisplay && editMode">
41
+ <a-radio-button value="design" v-if="!onlyDisplay">
42
+ 设计
43
+ </a-radio-button>
44
+ <a-radio-button value="display" style="border-radius: 0">
45
+ 预览
46
+ </a-radio-button>
47
+ </a-radio-group>
48
+ <a-radio-button @click="saveConfig" style="border-radius: 0 4px 4px 0" v-if="showSaveButton">
49
+ 保存
50
+ </a-radio-button>
51
+ <!-- 主体表格 -->
52
+ <XReportDesign
53
+ v-if="scanFinish"
54
+ @updateImg="updateImg"
55
+ @selectRow="selectRow"
56
+ :show-img-in-cell="showImgInCell"
57
+ :img-prefix="imgPrefix"
58
+ :use-oss-for-img="useOssForImg"
59
+ :display-only="displayOnly"
60
+ :config="type === 'display' ? originalConfig : activeConfig"
61
+ :slot-config-name="type === 'display' ? undefined : activatedSlotName"
62
+ :for-display="type === 'display'"
63
+ :no-padding="noPadding"
64
+ :no-top-border="noTopBorder"
65
+ :show-title="showTitle"
66
+ ref="XReportDesign"
67
+ id="printReady"
68
+ :server-name="serverName"
69
+ :env="env"
70
+ :show-images="hasImages"
71
+ :image-list="imageList">
72
+ </XReportDesign>
73
+ </a-card>
74
+ </template>
75
+ <!-- 弹出框 -->
76
+ <x-add-report
77
+ :env="env"
78
+ ref="xAddReport"
79
+ />
80
+ </div>
81
+ </template>
82
+
83
+ <script>
84
+ // 转PDF用
85
+ import HtmlToPdf from '@vue2-client/utils/htmlToPDF'
86
+ import { getConfigByName, runLogic } from '@vue2-client/services/api/common'
87
+ import XReportDesign from './XReportDesign.vue'
88
+ import { printElement } from './print'
89
+ // import XAddReport from '@vue2-client/base-client/components/common/XAddReport'
90
+
91
+ export default {
92
+ name: 'XReport',
93
+ props: {
94
+ files: {
95
+ type: Array,
96
+ default: () => {
97
+ return []
98
+ }
99
+ },
100
+ // 控制用户权限,user和admin
101
+ authority: {
102
+ type: String,
103
+ default: 'user'
104
+ },
105
+ // 是否为编辑模式
106
+ editMode: {
107
+ type: Boolean,
108
+ default: true
109
+ },
110
+ // 配置名
111
+ configName: {
112
+ type: String,
113
+ required: true
114
+ },
115
+ // 插槽名
116
+ activatedSlotName: {
117
+ type: String,
118
+ default: undefined
119
+ },
120
+ // 本地配置,调试用
121
+ localConfig: {
122
+ type: Object,
123
+ default: undefined
124
+ },
125
+ // 兼容老版本配置
126
+ dontFormat: {
127
+ type: Boolean,
128
+ default: true
129
+ },
130
+ showImgInCell: {
131
+ type: Boolean,
132
+ default: false
133
+ },
134
+ // 数据
135
+ configData: {
136
+ type: Object,
137
+ default: undefined
138
+ },
139
+ // 命名空间
140
+ serverName: {
141
+ type: String,
142
+ default: process.env.VUE_APP_SYSTEM_NAME
143
+ },
144
+ // 环境
145
+ env: {
146
+ type: String,
147
+ default: 'prod'
148
+ },
149
+ // 只做展示
150
+ displayOnly: {
151
+ type: Boolean,
152
+ default: true
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: false
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
+ XAddReport: () => import('@vue2-client/base-client/components/common/XAddReport'),
192
+ XReportDesign
193
+ },
194
+ data () {
195
+ return {
196
+ // 控制骨架屏显隐
197
+ showSkeleton: true,
198
+ // 配置
199
+ config: undefined,
200
+ // 当前显示模式,编辑模式,预览模式
201
+ type: 'design',
202
+ // 仅供展示,不可编辑
203
+ onlyDisplay: false,
204
+ // 每行最大列数,非必要请勿更改,现在的设计器完全是基于每行12列来设计的
205
+ maxColSpan: 12,
206
+ // 定义是否完成配置的扫描,未完成不要渲染子组件
207
+ scanFinish: false,
208
+ // 当前激活的配置文件
209
+ activeConfig: null,
210
+ // 原始配置文件
211
+ // 用于展示。某些情况下“设计页”中的内容仅为“预览页”表格其中的一部分
212
+ originalConfig: null,
213
+ // 扫描到的配置
214
+ configFromWeb: {},
215
+ // 用于获取配置的锁
216
+ timer: undefined,
217
+ // 是否包含图片
218
+ hasImages: false,
219
+ // 图片列表
220
+ imageList: [],
221
+ // 保存最原始的数据,用于判断哪些数据被更改了
222
+ dataCache: undefined,
223
+ // 判断哪些数据被更改了,存储对应的key
224
+ diff: [],
225
+ }
226
+ },
227
+ beforeDestroy () {
228
+ clearInterval(this.timer)
229
+ },
230
+ watch: {
231
+ // 如果配置名更改了,重新获取配置
232
+ configName (val) {
233
+ if (val) {
234
+ getConfigByName(this.configName, undefined, res => {
235
+ this.config = res
236
+ this.configInit()
237
+ }, this.env === 'dev')
238
+ }
239
+ },
240
+ // 如果本地配置更改了,重新初始化
241
+ localConfig: {
242
+ deep: true,
243
+ immediate: true,
244
+ handler (val) {
245
+ if (val) {
246
+ this.config = val
247
+ this.configInit()
248
+ }
249
+ }
250
+ },
251
+ },
252
+ provide () {
253
+ return {
254
+ runLogic: runLogic,
255
+ openDialog: this.openDialog,
256
+ registerComponent: this.registerComponent,
257
+ getComponentByName: this.getComponentByName,
258
+ getParentComponentByName: this.getComponentByName,
259
+ getConfigByName: getConfigByName
260
+ }
261
+ },
262
+ methods: {
263
+ // 把组件注册到refs中,方便调用
264
+ registerComponent (componentName, component) {
265
+ console.log('内部注册', this.$options.name, componentName)
266
+ this.$refs[componentName] = component
267
+ console.log('内部注册完成', this.$refs)
268
+ },
269
+ // 根据名字从注册到组件中获取组件
270
+ getComponentByName (componentName) {
271
+ console.log('内部取组件', this.$options.name, componentName)
272
+ console.log('内部组件内容', this.$refs)
273
+ return this.$refs[componentName]
274
+ },
275
+ /**
276
+ * @param configName 栅格配置名称
277
+ * @param selectedId 选中得id
278
+ * @param mixinData 需要混入得数据
279
+ * @param env 其他传递给打开窗口的数据
280
+ */
281
+ openDialog (configName, selectedId, mixinData, outEnv) {
282
+ console.log('openDialog', configName, selectedId)
283
+ this.$refs.xAddReport.init({
284
+ configName: configName,
285
+ selectedId: selectedId,
286
+ mixinData: mixinData,
287
+ outEnv: outEnv
288
+ })
289
+ },
290
+ // 向外暴露图片修改后的数据,某些外部需要自己管理图片的保存与修改
291
+ updateImg (data) {
292
+ this.$emit('updateImg', data)
293
+ },
294
+ // 导出数据,某些外部需要统一控制数据的变动
295
+ exportData () {
296
+ // 获取当前修改后的数据
297
+ let tempData
298
+ if (this.activeConfig === undefined || this.activeConfig === null) {
299
+ tempData = this.originalConfig.data
300
+ } else {
301
+ const tempDataKeys = Object.keys(this.activeConfig.tempData)
302
+ tempDataKeys.forEach(key => {
303
+ this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
304
+ })
305
+ tempData = this.activeConfig.data
306
+ }
307
+ // 对比数据的差异
308
+ this.diff = []
309
+ this.compareProps(tempData, this.dataCache)
310
+ this.diff.forEach(eachDiff => {
311
+ const arr = eachDiff.split('.')
312
+ let targetData = tempData[arr[0]]
313
+ if (arr.length !== 1) {
314
+ for (let i = 1; i < arr.length - 1; i++) {
315
+ const path = arr[i]
316
+ targetData = targetData[path]
317
+ }
318
+ }
319
+ // 将修改的数据,添加update = true属性
320
+ targetData.update = true
321
+ })
322
+ return tempData
323
+ },
324
+ // 对比两个obj有哪里不同
325
+ compareProps (obj1, obj2, path = '') {
326
+ for (const key in obj1) {
327
+ // 如果一个是undefined
328
+ if (typeof obj2[key] === 'undefined') {
329
+ this.diff.push(path + key)
330
+ // 如果是数组长度不一样
331
+ } else if (Array.isArray(obj1) && Array.isArray(obj2)) {
332
+ if (obj1[key].length !== obj2[key].length) {
333
+ this.diff.push(path + key)
334
+ }
335
+ // 如果都是对象,并且存在同样的key,递归子key
336
+ } else if (typeof obj1[key] === 'object' && typeof obj2[key] === 'object') {
337
+ this.compareProps(obj1[key], obj2[key], path + key + '.')
338
+ // 如果不是obj,对比其数据
339
+ } else if (obj1[key] !== obj2[key]) {
340
+ this.diff.push(path + key)
341
+ }
342
+ }
343
+ },
344
+ selectRow (selectedRowKeys, selectedRows) {
345
+ this.table_selectedRowKeys = selectedRowKeys
346
+ this.table_selectedRows = selectedRows
347
+ console.log('')
348
+ this.$emit('selectRow', selectedRowKeys, selectedRows)
349
+ },
350
+ // 注册组件到$refs中
351
+ registerComponentToRefs (componentName, component) {
352
+ this.$refs[componentName] = component
353
+ },
354
+
355
+ // 正常的保存方法,当前修改内容会直接全部导出到外部
356
+ saveConfig () {
357
+ if (this.activeConfig === undefined || this.activeConfig === null) {
358
+ return this.originalConfig.data
359
+ } else {
360
+ const tempDataKeys = Object.keys(this.activeConfig.tempData)
361
+ tempDataKeys.forEach(key => {
362
+ this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
363
+ })
364
+ this.$emit('saveConfig', this.$refs.XReportDesign.activatedConfig)
365
+ }
366
+ },
367
+ // 通过@@@分割临时变量,找到对应的key,并修改它的值
368
+ changeDeepObject (obj, strPath, newVal) {
369
+ const arr = strPath.split('@@@')
370
+ if (obj[arr[0]] === undefined) {
371
+ obj = obj.images
372
+ }
373
+ if (arr.length === 1) {
374
+ obj[arr[0]] = newVal
375
+ } else {
376
+ let result = obj[arr[0]]
377
+ arr.shift()
378
+ while (arr.length > 1) {
379
+ result = result[arr[0]]
380
+ arr.shift()
381
+ }
382
+ if (result) {
383
+ result[arr[0]] = newVal
384
+ }
385
+ }
386
+ },
387
+ // 检查slot是否在配置文件中包含,如果没有包含,则视为非法获取
388
+ checkSlotDefine (config) {
389
+ const slotsDeclare = config.slotsDeclare
390
+ const total = slotsDeclare.length
391
+ let count = 0
392
+ slotsDeclare.forEach(declare => {
393
+ config.columns.forEach(row => {
394
+ row.forEach(cell => {
395
+ if (cell.slotConfig === declare) {
396
+ count++
397
+ }
398
+ })
399
+ })
400
+ })
401
+
402
+ return count === total
403
+ },
404
+ // 切换了标签页
405
+ tabChanged (key) {
406
+ this.scanFinish = false
407
+ this.originalConfig.data = { ...this.originalConfig.data, ...this.config.data }
408
+ this.config.data = this.originalConfig.data
409
+ if (this.type === 'display') {
410
+ const tempDataKeys = Object.keys(this.activeConfig.tempData)
411
+ tempDataKeys.forEach(key => {
412
+ this.changeDeepObject(this.activeConfig.data, key, this.activeConfig.tempData[key])
413
+ })
414
+ let count = 0
415
+ this.imageList = []
416
+ const keys = Object.keys(this.config.data.images)
417
+ keys.forEach(key => {
418
+ if (this.config.data.images[key].length > 0) {
419
+ this.imageList = [...this.imageList, ...this.config.data.images[key]]
420
+ count++
421
+ }
422
+ })
423
+ this.hasImages = count > 0
424
+ } else {
425
+ this.hasImages = false
426
+ }
427
+ this.$nextTick(() => {
428
+ this.scanFinish = true
429
+ })
430
+ },
431
+ // 获取当前日期,为保存文件命名用
432
+ getDate () {
433
+ const currentDate = new Date()
434
+
435
+ const year = currentDate.getFullYear()
436
+ const month = String(currentDate.getMonth() + 1).padStart(2, '0')
437
+ const day = String(currentDate.getDate()).padStart(2, '0')
438
+
439
+ const formattedDate = `${year}_${month}_${day}`
440
+
441
+ return formattedDate
442
+ },
443
+ // 打印
444
+ printDocument () {
445
+ // x-report
446
+ const printContent = window.rawDocument.getElementById('printReady')
447
+ printElement(printContent)
448
+ this.$message.success('操作成功!')
449
+ },
450
+ // 导出PDF
451
+ exportPDF () {
452
+ const date = this.getDate()
453
+ let title = this.config.title
454
+ title = title.replace(/<[^>]+>/g, '')
455
+ const fileName = date + '' + title
456
+ HtmlToPdf.getPdf(fileName, '#printReady')
457
+ },
458
+ // 用于分割配置中的colums,将需要处理的数组提取出来
459
+ formatConfigRow () {
460
+ for (let i = 0; i < this.config.columns.length; i++) {
461
+ // 对原始数组进行递归,依次将该位置拆分为三个部分,当前处理位置之前的,当前处理位置,当前处理位置之后的
462
+ const before = this.config.columns.slice(0, i)
463
+ const after = this.config.columns.slice(i + 1, this.config.columns.length)
464
+
465
+ // 将当前处理的数组交给处理的方法
466
+ const x = this.checkRow(this.config.columns[i])
467
+
468
+ const newArr = []
469
+
470
+ // 拼接之前的数组
471
+ if (before.length > 0) {
472
+ if (before.length >= 1) {
473
+ before.forEach(item => {
474
+ newArr.push(item)
475
+ })
476
+ } else {
477
+ newArr.push(before)
478
+ }
479
+ }
480
+
481
+ // 拼接不需要更改当前节点处理完成的数组
482
+ newArr.push(x.old)
483
+
484
+ // 如果处理了新加的数据,拼接
485
+ if (x.add.length > 0) {
486
+ for (let j = 0; j < x.add.length; j++) {
487
+ if (x.add[j]) {
488
+ newArr.push(x.add[j])
489
+ i++
490
+ }
491
+ }
492
+ }
493
+
494
+ // 拼接之后的数组
495
+ if (after.length > 0) {
496
+ if (after.length >= 1) {
497
+ after.forEach(item => {
498
+ newArr.push(item)
499
+ })
500
+ } else {
501
+ newArr.push(after)
502
+ }
503
+ }
504
+
505
+ this.config.columns = newArr
506
+ }
507
+ },
508
+ // 路径中含有@@@的key,将其解析,并返回其数据
509
+ getDeepObject (obj, strPath) {
510
+ const arr = strPath.split('@@@')
511
+ let result = obj[arr[0]]
512
+ arr.shift()
513
+ try {
514
+ while (arr.length > 0) {
515
+ result = result[arr[0]]
516
+ arr.shift()
517
+ }
518
+ } catch (e) {
519
+ result = undefined
520
+ }
521
+ return result
522
+ },
523
+ // 处理colums数组,为声明了rowspan的单元格,自动匹配格式
524
+ checkRow (rowArr) {
525
+ // 不需要更改的数据
526
+ const original = []
527
+ // 需要更改新加的数据
528
+ const addArr = []
529
+ // 统计rowspan出现的总和
530
+ let count = 0
531
+ // 统计声明列需要的rowspan总数
532
+ let total = 0
533
+ // 是否为声明行
534
+ let titleCellFlag = false
535
+ // 是否为声明行后第一行
536
+ let firstSubLine = false
537
+ // 需要处理的行,新的index值
538
+ let subRowIndex = 0
539
+ // 新生成的行,临时存储
540
+ const waitForAddArr = []
541
+ // 用于记录声明行的colspan避免格式错误
542
+ let preColSpan = 0
543
+ // 用于统计循环次数,判断是否是最后一次
544
+ let forEachCount = 0
545
+
546
+ // 标记所有数据
547
+ rowArr.forEach(cell => {
548
+ forEachCount++
549
+ // 如果该行没有rowspan则默认其为1,不要影响统计结果
550
+ if (!cell.rowSpan) {
551
+ cell.rowSpan = 0
552
+ }
553
+
554
+ if (cell.text && total !== 0) { // 如果遇到了下一个声明行,证明rowspan少了一行,需要补充一个占位格
555
+ const nullObj = {
556
+ type: 'placeHolderColumn',
557
+ order: subRowIndex,
558
+ noBoarder: true,
559
+ needSplit: true,
560
+ colSpan: preColSpan,
561
+ dontShowRow: true
562
+ }
563
+ subRowIndex++
564
+ waitForAddArr.push(nullObj)
565
+ total = 0
566
+ count = 0
567
+ titleCellFlag = false
568
+ firstSubLine = false
569
+ } else if ((total !== count + cell.rowSpan) && forEachCount === rowArr.length) {
570
+ // 如果没有遇到了下一个声明行,但已经是当前行最后一个数据,也证明rowspan少了一行,需要补充一个占位格
571
+ const nullObj = {
572
+ type: 'placeHolderColumn',
573
+ order: subRowIndex,
574
+ noBoarder: true,
575
+ needSplit: true,
576
+ colSpan: preColSpan,
577
+ dontShowRow: true
578
+ }
579
+ subRowIndex++
580
+ waitForAddArr.push(nullObj)
581
+ total = 0
582
+ count = 0
583
+ titleCellFlag = false
584
+ firstSubLine = false
585
+ }
586
+
587
+ // 判断是否为声明行
588
+ if (cell.text && total === 0) {
589
+ // 将声明行声明的rowspan作为总数,判断下方rowspan相加是否等于声明行声明的数量
590
+ total = cell.rowSpan
591
+ titleCellFlag = false
592
+ firstSubLine = true
593
+ subRowIndex = 1
594
+ cell.show = true
595
+ cell.showRowSpan = total
596
+ } else if (cell.rowSpan > 0 && !titleCellFlag && firstSubLine) { // 判断是否为声明行后首行,因为首行不需要移动
597
+ count += cell.rowSpan
598
+ firstSubLine = false
599
+ cell.noBoarder = true
600
+ cell.show = true
601
+ cell.showRowSpan = total
602
+ } else if (cell.rowSpan > 0 && !titleCellFlag && !firstSubLine) { // 既非声明行,也非首行,需要移动
603
+ count += cell.rowSpan
604
+ // cell.type = 'notShow'
605
+ cell.needSplit = true
606
+ cell.order = subRowIndex
607
+ cell.dontShowRow = true
608
+ subRowIndex++
609
+
610
+ // 如果之前添加过空行补充位置,刚好最后一位还有内容,将其互换
611
+ if (forEachCount === rowArr.length && !waitForAddArr[waitForAddArr.length - 1].dataIndex) {
612
+ waitForAddArr[waitForAddArr.length - 1].order += 1
613
+ cell.order -= 1
614
+ waitForAddArr.push(cell)
615
+ } else {
616
+ waitForAddArr.push(cell)
617
+ }
618
+ }
619
+
620
+ // 如果count和total相等了,证明已经处理完成。将计数器还原
621
+ if (count === total) {
622
+ total = 0
623
+ count = 0
624
+ titleCellFlag = false
625
+ firstSubLine = false
626
+ }
627
+ // 保存上一个的colspan,保持生成的格子与原格式一致
628
+ preColSpan = cell.colSpan
629
+ })
630
+
631
+ // 将所有不需要移动的放入original
632
+ rowArr.forEach(cell => {
633
+ if (cell.needSplit !== true) {
634
+ original.push(cell)
635
+ }
636
+ })
637
+
638
+ // 增加新的数组
639
+ waitForAddArr.forEach(cell => {
640
+ const target = cell.order
641
+ // if (cell.type === 'notShow') {
642
+ // cell.type = 'inputs'
643
+ // }
644
+ cell.noBoarder = true
645
+ if (addArr[target] === undefined) {
646
+ const temp = []
647
+ temp.push(cell)
648
+ addArr[target] = temp
649
+ } else if (addArr[target].length > 0) {
650
+ addArr[target].push(cell)
651
+ }
652
+ })
653
+
654
+ // 如果没有新增,将单元格边框设置为显示
655
+ if (addArr.length < 1) {
656
+ original.forEach(cell => {
657
+ if (cell.type === 'input' || cell.type === 'inputs') {
658
+ cell.noBoarder = false
659
+ }
660
+ })
661
+ }
662
+
663
+ return {
664
+ old: original,
665
+ add: addArr
666
+ }
667
+ },
668
+ // 扫描配置,如果有插槽则拼接插槽
669
+ scanConfigSlot (config) {
670
+ const columnsArr = config.columns
671
+ for (let i = 0; i < columnsArr.length; i++) {
672
+ for (let j = 0; j < columnsArr[i].length; j++) {
673
+ // 如果发现type为slot,开始匹配对应的slot配置文件
674
+ if (columnsArr[i][j].type === 'slot') {
675
+ const targetName = columnsArr[i][j].slotConfig
676
+ // 找不到目标插槽配置
677
+ if (!this.configFromWeb[targetName] || !this.configFromWeb[targetName].columns) {
678
+ console.error('无法找到目标插槽的配置!')
679
+ return
680
+ }
681
+
682
+ // 替换columns,合并data
683
+ config.columns[i] = []
684
+ const before = config.columns.slice(0, i)
685
+ let after = config.columns.slice(i + 1, config.columns.length)
686
+
687
+ const addArr = []
688
+ for (let k = 0; k < this.configFromWeb[targetName].columns.length; k++) {
689
+ const temp = []
690
+ this.configFromWeb[targetName].columns[k].forEach(cell => {
691
+ temp.push(cell)
692
+ })
693
+ addArr.push(temp)
694
+ }
695
+
696
+ const newArr = []
697
+ // 拼接之前的数组
698
+ if (before.length > 0) {
699
+ if (before.length >= 1) {
700
+ before.forEach(item => {
701
+ newArr.push(item)
702
+ })
703
+ } else {
704
+ newArr.push(before)
705
+ }
706
+ }
707
+
708
+ addArr.forEach(arr => {
709
+ newArr.push(arr)
710
+ })
711
+
712
+ // 拼接之后的数组
713
+ if (after.length === 1) {
714
+ if (after[0].type === 'slot' || after[0][0].type === 'slot') {
715
+ after = []
716
+ }
717
+ }
718
+ if (after.length > 0) {
719
+ if (after.length >= 1) {
720
+ after.forEach(item => {
721
+ newArr.push(item)
722
+ })
723
+ } else {
724
+ newArr.push(after)
725
+ }
726
+ }
727
+
728
+ config.columns = newArr
729
+ if (this.configFromWeb[targetName].slotsDeclare) {
730
+ config.slotsDeclare = this.configFromWeb[targetName].slotsDeclare
731
+ } else {
732
+ config.slotsDeclare = []
733
+ }
734
+
735
+ if (config.data.images && this.configFromWeb[targetName].data.images) {
736
+ config.data.images = { ...config.data.images, ...this.configFromWeb[targetName].data.images }
737
+ delete this.configFromWeb[targetName].data.images
738
+ }
739
+ config.data = { ...config.data, ...this.configFromWeb[targetName].data }
740
+ }
741
+ }
742
+ }
743
+ this.config = config
744
+ },
745
+ // 扫描所有插槽名
746
+ scanConfigName (config, resut) {
747
+ if (config.slotsDeclare) {
748
+ config.slotsDeclare.forEach(name => {
749
+ resut.push(name)
750
+ })
751
+ }
752
+ },
753
+ // 获取插槽
754
+ getConfigAndJoin (config, outerLock) {
755
+ // 检查主配置插槽声明是否合法
756
+ const check = this.checkSlotDefine(config)
757
+ const waitForDownloadSlotName = []
758
+ if (check) {
759
+ // 扫描主配置中声明的插槽名
760
+ this.scanConfigName(config, waitForDownloadSlotName)
761
+
762
+ const total = waitForDownloadSlotName.length
763
+ let count = 0
764
+
765
+ // 挨个获取插槽
766
+ waitForDownloadSlotName.forEach(configName => {
767
+ getConfigByName(configName, this.serverName, res => {
768
+ this.configFromWeb[configName] = res
769
+ count++
770
+ }, this.env === 'dev')
771
+ })
772
+
773
+ // 使用定时器循环判断锁状态,用于多个插槽,要等待统一获取完成之后,再进行下一步初始化
774
+ const timer = setInterval(() => {
775
+ console.log('插槽下载进度,当前:' + count + '/' + total)
776
+ if (count >= total) {
777
+ clearInterval(timer)
778
+ this.scanConfigSlot(config)
779
+ if (config.slotsDeclare.length > 0) {
780
+ const lock = { status: true }
781
+ this.getConfigAndJoin(config, lock)
782
+ const innerTimer = setInterval(() => {
783
+ if (!lock.status) {
784
+ clearInterval(innerTimer)
785
+ outerLock.status = false
786
+ }
787
+ }, 100)
788
+ } else {
789
+ outerLock.status = false
790
+ }
791
+ }
792
+ }, 100)
793
+ } else {
794
+ console.error('插槽配置有误!')
795
+ outerLock.status = false
796
+ }
797
+ },
798
+ // 获取配置之后的初始化
799
+ configInit () {
800
+ console.log('拼接完成', this.config)
801
+ // 将初始化好的配置拷贝一份留存
802
+ this.originalConfig = Object.assign({}, this.config)
803
+ if (!this.dontFormat) {
804
+ // 扫描配置文件中有没有rowSpan,进行格式化调整
805
+ this.formatConfigRow(this.config)
806
+ }
807
+ this.activeConfig = this.config
808
+ this.showSkeleton = false
809
+ // 判断是否有动态Index
810
+ this.activeConfig.columns.forEach(row => {
811
+ row.forEach(cell => {
812
+ if (cell.dynamicDataIndex === true) {
813
+ // 如果有动态index,取其函数,运行函数得到真实index保存
814
+ // eslint-disable-next-line no-eval
815
+ const func = eval('(' + cell.customFunctionForDynamicDataIndex + ')')
816
+ cell.dataIndex = func(this.config)
817
+ }
818
+ // 处理 自定义函数的旧逻辑
819
+ if (['action', 'click'].includes(cell.eventType) && cell.customFunction && !cell.events) {
820
+ cell.events = []
821
+ cell.events.push({
822
+ type: cell.eventType,
823
+ customFunction: cell.customFunction
824
+ })
825
+ }
826
+ })
827
+ })
828
+ // 将数据复制到临时数据中,带有@@@的数据,我们将其整体作为一个key保存,当编辑完成后,再将其解析,回填到需要的数据中
829
+ this.activeConfig.tempData = {}
830
+ // 是否有@@@深层引用
831
+ this.activeConfig.columns.forEach(row => {
832
+ row.forEach(cell => {
833
+ // 将@@@解析
834
+ if (cell.dataIndex !== undefined && cell.dataIndex.indexOf('@@@') !== -1) {
835
+ this.activeConfig.tempData[cell.dataIndex] = this.getDeepObject(this.activeConfig.data, cell.dataIndex)
836
+ }
837
+ })
838
+ })
839
+ this.$nextTick(() => {
840
+ this.scanFinish = true
841
+ })
842
+ },
843
+ // 初始化JSON配置
844
+ jsonConfigInit () {
845
+ if (this.configData === undefined) {
846
+ console.error('未找到数据!')
847
+ } else {
848
+ this.originalConfig = Object.assign({}, this.config)
849
+ this.originalConfig.data = JSON.parse(JSON.stringify(this.configData))
850
+ this.type = 'display'
851
+ // this.onlyDisplay = true
852
+ this.showSkeleton = false
853
+ this.$nextTick(() => {
854
+ this.scanFinish = true
855
+ })
856
+ }
857
+ },
858
+ onSubmit () {
859
+ console.log('this.table_selectedRowKeys', this.table_selectedRowKeys)
860
+ console.log('this.table_selectedRows', this.table_selectedRows)
861
+ },
862
+ },
863
+ beforeMount () {
864
+ // 如果只是展示
865
+ if (this.displayOnly) {
866
+ this.onlyDisplay = true
867
+ this.type = 'display'
868
+ }
869
+ // 如果有本地配置,优先使用本地配置
870
+ if (this.localConfig) {
871
+ // 如果配置是json渲染器
872
+ if (this.localConfig.designMode === 'json') {
873
+ this.config = this.localConfig
874
+ if (this.configData !== undefined) {
875
+ this.config.data = this.configData
876
+ }
877
+ this.jsonConfigInit()
878
+ } else {
879
+ // 如果配置是普通渲染器
880
+ this.config = this.localConfig
881
+ if (this.configData !== undefined) {
882
+ this.config.data = this.configData
883
+ }
884
+ if (this.config.data.images === undefined) {
885
+ this.config.data.images = {}
886
+ }
887
+ this.configInit()
888
+ }
889
+ } else {
890
+ // 如果本地配置没有值,则从琉璃中获取
891
+ getConfigByName(this.configName, this.serverName, res => {
892
+ this.config = res
893
+ if (this.config.designMode === 'json') {
894
+ if (this.configData !== undefined) {
895
+ this.config.data = this.configData
896
+ }
897
+ this.jsonConfigInit()
898
+ } else {
899
+ if (this.configData !== undefined) {
900
+ this.config.data = this.configData
901
+ }
902
+ if (this.config.data.images === undefined) {
903
+ this.config.data.images = {}
904
+ }
905
+ this.configInit()
906
+ }
907
+ }, this.env === 'dev')
908
+ }
909
+ },
910
+ mounted () {
911
+ // 如果外界传来了registerMap,我们将本VM对象注册到map中
912
+ if (this.registerMap !== undefined) {
913
+ this.registerMap.push(this)
914
+ }
915
+ // 将原始数据备份保存
916
+ if (this.configData) {
917
+ this.dataCache = JSON.parse(JSON.stringify(this.configData))
918
+ } else {
919
+ if (this.config?.data) {
920
+ this.dataCache = JSON.parse(JSON.stringify(this.config.data))
921
+ }
922
+ }
923
+ }
924
+ }
925
+ </script>
926
+
927
+ <style lang="less" scoped>
928
+ .tools {
929
+ text-align: center;
930
+ cursor: pointer;
931
+
932
+ .toolsItem {
933
+ display: inline-block;
934
+ }
935
+ }
936
+ </style>