vue2-client 1.4.28 → 1.4.29

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 (94) hide show
  1. package/CHANGELOG.md +413 -410
  2. package/index.js +30 -30
  3. package/package.json +78 -78
  4. package/src/base-client/components/common/AddressSearchCombobox/AddressSearchCombobox.vue +225 -225
  5. package/src/base-client/components/common/CreateQuery/CreateQuery.vue +595 -595
  6. package/src/base-client/components/common/CreateQuery/CreateQueryItem.vue +777 -777
  7. package/src/base-client/components/common/FormGroupQuery/FormGroupQuery.vue +165 -165
  8. package/src/base-client/components/common/JSONToTree/jsontotree.vue +275 -275
  9. package/src/base-client/components/common/XAddForm/XAddForm.vue +354 -354
  10. package/src/base-client/components/common/XAddNativeForm/XAddNativeForm.vue +327 -327
  11. package/src/base-client/components/common/XCard/XCard.vue +64 -64
  12. package/src/base-client/components/common/XForm/XForm.vue +274 -274
  13. package/src/base-client/components/common/XFormTable/XFormTable.vue +6 -3
  14. package/src/base-client/components/common/XFormTable/index.md +96 -96
  15. package/src/base-client/components/common/XTable/XTable.vue +1 -1
  16. package/src/base-client/components/system/QueryParamsDetailsView/QueryParamsDetailsView.vue +281 -281
  17. package/src/base-client/components/ticket/TicketDetailsView/TicketDetailsView.vue +807 -807
  18. package/src/base-client/components/ticket/TicketDetailsView/index.md +29 -29
  19. package/src/base-client/components/ticket/TicketDetailsView/part/TicketDetailsFlow.vue +260 -260
  20. package/src/base-client/components/ticket/TicketSubmitSuccessView/TicketSubmitSuccessView.vue +532 -532
  21. package/src/base-client/components/ticket/TicketSubmitSuccessView/index.md +29 -29
  22. package/src/base-client/plugins/AppData.js +73 -73
  23. package/src/base-client/plugins/GetLoginInfoService.js +179 -179
  24. package/src/base-client/plugins/PagedList.js +177 -177
  25. package/src/base-client/plugins/compatible/LoginServiceOA.js +20 -20
  26. package/src/base-client/plugins/i18n-extend.js +32 -32
  27. package/src/components/Ellipsis/index.md +38 -38
  28. package/src/components/NumberInfo/index.md +43 -43
  29. package/src/components/STable/README.md +341 -341
  30. package/src/components/STable/index.js +318 -318
  31. package/src/components/Trend/index.md +45 -45
  32. package/src/components/checkbox/ColorCheckbox.vue +157 -157
  33. package/src/components/checkbox/ImgCheckbox.vue +163 -163
  34. package/src/components/form/FormRow.vue +52 -52
  35. package/src/components/index.js +36 -36
  36. package/src/components/menu/SideMenu.vue +62 -62
  37. package/src/components/menu/menu.js +273 -273
  38. package/src/components/page/header/index.less +40 -40
  39. package/src/components/setting/Setting.vue +235 -235
  40. package/src/components/table/StandardTable.vue +141 -141
  41. package/src/components/table/advance/ActionColumns.vue +158 -158
  42. package/src/components/table/advance/SearchArea.vue +355 -355
  43. package/src/components/tool/AStepItem.vue +60 -60
  44. package/src/components/tool/AvatarList.vue +68 -68
  45. package/src/components/tool/Drawer.vue +142 -142
  46. package/src/components/tool/TagSelect.vue +83 -83
  47. package/src/components/transition/PageToggleTransition.vue +97 -97
  48. package/src/config/replacer/resolve.config.js +67 -67
  49. package/src/layouts/AdminLayout.vue +174 -174
  50. package/src/layouts/PageLayout.vue +151 -151
  51. package/src/layouts/SinglePageView.vue +93 -93
  52. package/src/layouts/header/AdminHeader.vue +104 -104
  53. package/src/layouts/header/HeaderNotice.vue +167 -167
  54. package/src/layouts/header/HeaderSearch.vue +67 -67
  55. package/src/layouts/header/InstitutionDetail.vue +181 -181
  56. package/src/layouts/header/index.less +92 -92
  57. package/src/layouts/tabs/TabsHead.vue +190 -190
  58. package/src/layouts/tabs/TabsView.vue +379 -379
  59. package/src/mock/goods/index.js +108 -108
  60. package/src/pages/CreateQueryPage.vue +65 -65
  61. package/src/pages/login/Login.vue +345 -345
  62. package/src/pages/report/ReportTable.js +124 -124
  63. package/src/pages/report/ReportTableHome.vue +28 -28
  64. package/src/pages/resourceManage/orgListManage.vue +98 -98
  65. package/src/pages/system/dictionary/index.vue +43 -43
  66. package/src/pages/system/file/index.vue +317 -317
  67. package/src/pages/system/monitor/loginInfor/index.vue +36 -36
  68. package/src/pages/system/monitor/operLog/index.vue +36 -36
  69. package/src/pages/system/queryParams/index.vue +43 -43
  70. package/src/router/async/config.async.js +27 -27
  71. package/src/router/async/router.map.js +65 -65
  72. package/src/router/index.js +27 -27
  73. package/src/services/api/DictionaryDetailsViewApi.js +6 -6
  74. package/src/services/api/LogDetailsViewApi.js +10 -10
  75. package/src/services/api/QueryParamsDetailsViewApi.js +6 -6
  76. package/src/services/api/TicketDetailsViewApi.js +34 -34
  77. package/src/services/api/cas.js +70 -70
  78. package/src/services/api/commonTempTable.js +10 -10
  79. package/src/services/api/index.js +17 -17
  80. package/src/services/api/manage.js +8 -8
  81. package/src/services/apiService.js +13 -13
  82. package/src/services/user.js +53 -53
  83. package/src/store/modules/index.js +4 -4
  84. package/src/theme/default/nprogress.less +76 -76
  85. package/src/theme/default/style.less +47 -47
  86. package/src/utils/colors.js +103 -103
  87. package/src/utils/excel/Blob.js +180 -180
  88. package/src/utils/excel/Export2Excel.js +141 -141
  89. package/src/utils/formatter.js +68 -68
  90. package/src/utils/i18n.js +80 -80
  91. package/src/utils/routerUtil.js +364 -364
  92. package/src/utils/theme-color-replacer-extend.js +91 -91
  93. package/src/utils/themeUtil.js +100 -100
  94. package/src/utils/util.js +230 -230
@@ -1,595 +1,595 @@
1
- <template>
2
- <a-drawer
3
- :visible="visible"
4
- :width="isMobile ? screenWidth : screenWidth * 0.85"
5
- placement="right"
6
- title="查询配置生成"
7
- @close="onClose"
8
- >
9
- <a-row :gutter="24">
10
- <a-col :lg="12" :md="12" :sm="24" :xl="14" :xs="24">
11
- <a-form-model
12
- ref="businessCreateForm"
13
- :label-col="labelCol"
14
- :model="form"
15
- :rules="rules"
16
- :wrapper-col="wrapperCol"
17
- >
18
- <a-form-model-item label="服务名称" prop="serviceName">
19
- <a-input v-model="form.serviceName" placeholder="查询接口所在的服务名称,默认af-system"/>
20
- </a-form-model-item>
21
- <a-form-model-item label="查询主表名" prop="tableName">
22
- <a-input v-model="form.tableName" placeholder="查询用的主表+别名,用空格隔开,如:t_userfiles u"/>
23
- </a-form-model-item>
24
- <a-form-model-item label="预设关联表" prop="joinArray">
25
- <a-popover placement="right" title="说明">
26
- <template slot="content">
27
- <p>配置你的查询中可能涉及到的所有关联表</p>
28
- <p>比如你的查询中涉及到了t_userinfo表,就需要加入t_userinfo表的关联,和你写SQL的关联一样</p>
29
- <p>请注意,你关联的<span style="color: #ff0000">除主表外的任何表</span>都需要配置</p>
30
- <a-input-group compact style="width: 400px;">
31
- <a-input placeholder="表别名" readOnly style="width: 20%"/>
32
- <a-input placeholder="关联条件" readOnly style="width: 80%"/>
33
- </a-input-group>
34
- <a-input-group compact style="width: 400px;">
35
- <a-input placeholder="表别名" readOnly style="width: 20%" value="ui"/>
36
- <a-input
37
- placeholder="关联条件"
38
- readOnly
39
- style="width: 80%"
40
- value="t_userinfo ui on i.f_userinfo_id = ui.f_userinfo_id"/>
41
- </a-input-group>
42
- </template>
43
- <a-button type="primary" @click="addJoinItem()">增加</a-button>
44
- </a-popover>
45
- <div v-for="(itemObj, index) in joinArray" :key="index">
46
- <a-input-group compact>
47
- <a-input
48
- v-model="itemObj.key"
49
- placeholder="表别名"
50
- style="width: 20%;position: relative;bottom: 1px;"
51
- @change="changeJoinArray()"/>
52
- <a-input v-model="itemObj.value" placeholder="关联条件" style="width: 80%" @change="changeJoinArray()">
53
- <a-icon slot="addonAfter" type="close" @click="removeJoinItem(index)"/>
54
- </a-input>
55
- </a-input-group>
56
- </div>
57
- </a-form-model-item>
58
- <a-form-model-item label="SQL查询表达式" prop="condition">
59
- <a-input v-model="form.condition.value" placeholder="用作SQL查询的固定条件表达式,如:gb.f_meter_type='物联网表',可选"/>
60
- <template v-if="form.condition.value && Object.keys(form.joinArray).length > 0">
61
- <a-alert message="提示:如果SQL查询表达式中用到了关联表,需要勾选用到的关联表别名" type="success"/>
62
- <a-checkbox-group v-model="form.condition.join" :options="conditionJoinArray"/>
63
- </template>
64
- </a-form-model-item>
65
- <a-form-model-item label="排序方式" prop="orderBy">
66
- <a-input v-model="form.orderBy" placeholder="排序字段,用别名+字段名+排序方式(可选)表示,如:u.id desc"/>
67
- </a-form-model-item>
68
- <a-form-model-item label="数据字段" prop="column">
69
- <a-button type="primary" @click="addColumnItem()">增加</a-button>
70
- <div
71
- v-for="(columnItem, index) in form.column"
72
- :key="index"
73
- class="column_item"
74
- draggable="true"
75
- @dragend="handleDragEnd($event, columnItem)"
76
- @dragenter="handleDragEnter($event, columnItem)"
77
- @dragstart="handleDragStart($event, columnItem)"
78
- @dragover.prevent="handleDragOver($event, columnItem)">
79
- <a-row v-if="ending && dragging && columnItem.key === ending.key && dragging.key !== ending.key" class="dragTipsWarp">
80
- <span class="dragTips">拖到此处放置</span>
81
- </a-row>
82
- <a-row :gutter="24">
83
- <a-col :span="20">
84
- <span style="font-weight: bold">{{ columnItem.title }}({{ columnItem.key }})</span>
85
- </a-col>
86
- <a-col :span="2">
87
- <a-icon type="edit" @click="editColumnItem(columnItem.key,index)"/>
88
- </a-col>
89
- <a-col :span="2">
90
- <a-icon type="close" @click="removeColumnItem(columnItem.key,index)"/>
91
- </a-col>
92
- </a-row>
93
- </div>
94
- </a-form-model-item>
95
- <a-form-model-item label="操作按钮配置" prop="buttonState">
96
- <a-checkbox-group v-model="buttonStateData" :options="buttonStateArray"/>
97
- </a-form-model-item>
98
- <a-form-model-item label="接口插槽" prop="apiSlot">
99
- <a-popover placement="right" title="说明">
100
- <template slot="content">
101
- <p>通过插槽,你可以轻松扩展查询配置的默认业务,配置的插槽会根据类型在默认业务的前后执行,类似于切面</p>
102
- <p><span style="color: #FF0000">注意:插槽不应改变默认的业务行为</span></p>
103
- <p>如果配置了业务执行前插槽,<br/>执行默认业务前会先执行设置的插槽logic,传给插槽的参数为data本身</p>
104
- <p>如果配置了业务执行后插槽,<br/>执行默认业务后会执行设置的插槽logic,传给插槽的参数为默认业务的返回值</p>
105
- <img src="@vue2-client/assets/img/querySlotDemo.svg" style="zoom:0.5">
106
- </template>
107
- <a-button type="primary" @click="addapiSlot()">增加</a-button>
108
- </a-popover>
109
- <div
110
- v-for="(columnItem, index) in form.apiSlotView"
111
- :key="index"
112
- >
113
- <a-input v-model="columnItem.slotName" placeholder="插槽logic名称">
114
- <a-select slot="addonBefore" v-model="columnItem.slotType" style="width: 10rem" placeholder="选择插槽类型">
115
- <a-select-option
116
- v-for="item in apiSlotData"
117
- :key="item.value"
118
- :value="item.value">
119
- {{ item.label }}
120
- </a-select-option>
121
- </a-select>
122
- </a-input>
123
- </div>
124
- </a-form-model-item>
125
- </a-form-model>
126
- <create-query-item ref="queryItem" @getColumn="getColumn" @itemHandle="itemHandle"/>
127
- <a-button type="primary" @click="view">操作</a-button>
128
- </a-col>
129
- <a-col :lg="12" :md="12" :sm="24" :xl="10" :xs="24">
130
- <a-card :bordered="false" size="small" style="overflow: auto" title="预览">
131
- <json-viewer
132
- :copyable="{copyText: '复制', copiedText: '已复制'}"
133
- :expand-depth="parseInt('100')"
134
- :value="result"
135
- style="overflow: auto;max-height: 440px"></json-viewer>
136
- </a-card>
137
- </a-col>
138
- </a-row>
139
- <a-modal
140
- :centered="true"
141
- :destroyOnClose="true"
142
- :visible="modelVisible"
143
- :width="isMobile ? screenWidth : screenWidth * 0.8"
144
- :zIndex="1001"
145
- title="效果预览"
146
- @cancel="onModelClose">
147
- <template slot="footer">
148
- <a-button key="close" @click="onModelClose">
149
- 返回
150
- </a-button>
151
- <a-button key="submit" type="primary" @click="submit">
152
- 保存
153
- </a-button>
154
- </template>
155
- <x-form-table
156
- :queryParamsJson="result"
157
- :view-mode="true">
158
- </x-form-table>
159
- </a-modal>
160
- </a-drawer>
161
- </template>
162
-
163
- <script>
164
- import XFormItem from '@vue2-client/base-client/components/common/XForm/XFormItem'
165
- import XFormTable from '@vue2-client/base-client/components/common/XFormTable/XFormTable'
166
- import JsonViewer from 'vue-json-viewer'
167
- import FileSaver from 'file-saver'
168
- import { mapState } from 'vuex'
169
- import CreateQueryItem from '@vue2-client/base-client/components/common/CreateQuery/CreateQueryItem'
170
-
171
- export default {
172
- name: 'CreateQuery',
173
- components: {
174
- CreateQueryItem,
175
- JsonViewer,
176
- XFormTable,
177
- XFormItem
178
- },
179
- data () {
180
- return {
181
- // 页面宽度
182
- screenWidth: document.documentElement.clientWidth,
183
- // 效果预览模态框是否展示
184
- modelVisible: false,
185
- // 操作按钮状态集合
186
- buttonStateArray: [
187
- { label: '新增', value: 'add' },
188
- { label: '修改', value: 'edit' },
189
- { label: '删除', value: 'delete' },
190
- { label: '导入', value: 'import' },
191
- { label: '导出', value: 'export' }
192
- ],
193
- // 操作按钮状态集合值
194
- buttonStateData: ['add', 'edit', 'delete', 'import', 'export'],
195
- labelCol: { span: 4 },
196
- wrapperCol: { span: 14 },
197
- form: {
198
- serviceName: '',
199
- tableName: '',
200
- joinArray: {},
201
- condition: {},
202
- orderBy: '',
203
- column: [],
204
- apiSlotView: [],
205
- apiSlot: {},
206
- },
207
- result: {},
208
- itemMap: {},
209
- selectIndex: null,
210
- joinArray: [],
211
- rules: {
212
- tableName: [{ required: true, message: '请输入查询表名', trigger: 'blur' }],
213
- orderBy: [{ required: true, message: '请输入排序方式', trigger: 'blur' }]
214
- },
215
- ending: null,
216
- dragging: null,
217
- // 操作按钮状态集合
218
- apiSlotData: [
219
- { label: '查询前插槽', value: 'queryBefore' },
220
- { label: '查询后插槽', value: 'queryAfter' },
221
- { label: '删除前插槽', value: 'delBefore' },
222
- { label: '删除后插槽', value: 'delAfter' },
223
- { label: '获取行信息前插槽', value: 'editQueryBefore' },
224
- { label: '获取行信息后插槽', value: 'editQueryAfter' },
225
- { label: '新增/修改保存前插槽', value: 'addOrEditSaveBefore' },
226
- { label: '新增/修改保存后插槽', value: 'addOrEditSaveAfter' },
227
- ],
228
- }
229
- },
230
- mounted () {
231
- this.initView()
232
- },
233
- computed: {
234
- ...mapState('setting', ['isMobile']),
235
- conditionJoinArray: function () {
236
- const result = []
237
- for (const item in this.form.joinArray) {
238
- if (item !== '') {
239
- result.push({
240
- label: item,
241
- value: item
242
- })
243
- }
244
- }
245
- if (result.length === 0) {
246
- result.push({
247
- label: '-',
248
- value: '-'
249
- })
250
- }
251
- return result
252
- },
253
- buttonState: function () {
254
- const result = {
255
- add: false,
256
- edit: false,
257
- delete: false,
258
- export: false
259
- }
260
- for (const item of this.buttonStateData) {
261
- result[item] = true
262
- }
263
- return result
264
- }
265
- },
266
- props: {
267
- visible: {
268
- type: Boolean,
269
- default: false
270
- },
271
- toEditJson: {
272
- type: Object,
273
- default: () => {}
274
- }
275
- },
276
- watch: {
277
- visible (rel) {
278
- if (rel) {
279
- this.initView()
280
- }
281
- if (rel && this.toEditJson) {
282
- // 处理预设关联表
283
- if (this.joinArray.length === 0) {
284
- for (const key in this.toEditJson.joinArray) {
285
- this.joinArray.push({
286
- key: key,
287
- value: this.toEditJson.joinArray[key]
288
- })
289
- }
290
- }
291
- // apiSlot obj to arr
292
- if (this.toEditJson.apiSlot) {
293
- this.toEditJson.apiSlotView = []
294
- for (const key of Object.keys(this.toEditJson.apiSlot)) {
295
- this.toEditJson.apiSlotView.push({ slotType: key, slotName: this.toEditJson.apiSlot[key] })
296
- }
297
- }
298
- // 处理具体表单项
299
- this.form = Object.assign(
300
- {
301
- tableName: '',
302
- joinArray: {},
303
- condition: {},
304
- apiSlotView: [],
305
- apiSlot: {},
306
- orderBy: '',
307
- column: []
308
- }, this.toEditJson
309
- )
310
- for (const columnItem of this.form.column) {
311
- // 数据模式兼容性处理
312
- if (!(columnItem.dataMode || columnItem.dataModeArray)) {
313
- columnItem.dataModeArray = ['queryForm', 'table', 'addOrEditForm', 'sqlQueryItem', 'sqlQueryCondition']
314
- } else if (columnItem.dataMode) {
315
- if (columnItem.dataMode === 'all') {
316
- columnItem.dataModeArray = ['queryForm', 'table', 'addOrEditForm', 'sqlQueryItem', 'sqlQueryCondition']
317
- } else if (columnItem.dataMode === 'form') {
318
- columnItem.dataModeArray = ['queryForm', 'addOrEditForm', 'sqlQueryCondition']
319
- } else if (columnItem.dataMode === 'table') {
320
- columnItem.dataModeArray = ['table', 'sqlQueryItem']
321
- } else if (columnItem.dataMode === 'table_form') {
322
- columnItem.dataModeArray = ['table', 'sqlQueryItem', 'sqlQueryCondition']
323
- } else if (columnItem.dataMode === 'only_form') {
324
- columnItem.dataModeArray = ['queryForm']
325
- } else if (columnItem.dataMode === 'only_table') {
326
- columnItem.dataModeArray = ['table']
327
- } else if (columnItem.dataMode === 'clear') {
328
- columnItem.dataModeArray = ['sqlQueryItem', 'sqlQueryCondition']
329
- } else if (columnItem.dataMode === 'only_add_modify') {
330
- columnItem.dataModeArray = ['addOrEditForm', 'sqlQueryItem']
331
- }
332
- }
333
- delete columnItem.dataMode
334
- // 插槽兼容处理
335
- if (columnItem.slot) {
336
- if (columnItem.slot.value && columnItem.slot.type === 'ellipsis') {
337
- columnItem.slotValue = columnItem.slot.value
338
- } else if (columnItem.slot.keyMap && columnItem.slot.type === 'badge') {
339
- columnItem.slotKeyMap = columnItem.slot.keyMap
340
- } else if (columnItem.slot.actionText && columnItem.slot.type === 'action') {
341
- columnItem.actionText = columnItem.slot.actionText
342
- }
343
- }
344
- // 必选项兼容处理
345
- if (columnItem.rule && columnItem.rule.required && columnItem.rule.required !== 'false') {
346
- columnItem.rule.required = columnItem.rule.required.toString()
347
- } else {
348
- if (!columnItem.rule) {
349
- columnItem.rule = {}
350
- }
351
- columnItem.rule.required = 'false'
352
- }
353
- // 数据源加载方式兼容处理
354
- if (columnItem.lazyLoad && columnItem.lazyLoad !== 'false') {
355
- columnItem.lazyLoad = columnItem.lazyLoad.toString()
356
- } else {
357
- columnItem.lazyLoad = 'false'
358
- }
359
- // 下拉框数据源兼容处理
360
- if ((columnItem.formType === 'select' || columnItem.formType === 'cascader') && columnItem.selectKey) {
361
- // 数据源为logic
362
- if (columnItem.selectKey.toString().startsWith('logic@')) {
363
- columnItem.selectType = 'logic'
364
- } else if (columnItem.selectKey instanceof Array || this.isJSON(columnItem.selectKey)) {
365
- // 数据源为固定json集合
366
- if (columnItem.selectKey instanceof Array) {
367
- columnItem.selectKey = JSON.stringify(columnItem.selectKey)
368
- }
369
- columnItem.selectType = 'fixArray'
370
- } else {
371
- columnItem.selectType = 'key'
372
- }
373
- }
374
- this.itemMap[columnItem.key] = Object.assign({
375
- key: '',
376
- title: '',
377
- slot: {},
378
- rule: {
379
- required: 'false'
380
- },
381
- dataModeArray: []
382
- }, columnItem)
383
- }
384
- // 处理操作按钮配置
385
- if (this.toEditJson.buttonState) {
386
- this.buttonStateData = []
387
- for (const buttonStateKey of Object.keys(this.toEditJson.buttonState)) {
388
- if (this.toEditJson.buttonState[buttonStateKey]) {
389
- this.buttonStateData.push(buttonStateKey)
390
- }
391
- }
392
- } else {
393
- this.buttonStateData = ['add', 'edit', 'delete', 'import', 'export']
394
- }
395
- }
396
- }
397
- },
398
- methods: {
399
- // 新增接口插槽
400
- addapiSlot () {
401
- if (this.form.apiSlotView.length < this.apiSlotData.length) {
402
- this.form.apiSlotView.push({})
403
- }
404
- },
405
- // 初始化组件
406
- initView () {
407
- this.joinArray = []
408
- this.result = {}
409
- },
410
- onClose () {
411
- this.$emit('update:visible', false)
412
- },
413
- onModelClose () {
414
- this.modelVisible = false
415
- },
416
- addJoinItem () {
417
- this.joinArray.push({
418
- key: '',
419
- value: ''
420
- })
421
- this.changeJoinArray()
422
- },
423
- removeJoinItem (index) {
424
- this.joinArray.splice(index, 1)
425
- this.changeJoinArray()
426
- },
427
- itemHandle (item, type) {
428
- this.itemMap[item.key] = item
429
- if (type === '新增') {
430
- this.form.column.push(item)
431
- } else {
432
- this.$set(this.form.column, this.selectIndex, item)
433
- }
434
- this.$message.success(`${type}成功`)
435
- this.$refs.queryItem.flashModal(false)
436
- },
437
- getColumn (callback) {
438
- callback(JSON.parse(JSON.stringify(this.form.column)))
439
- },
440
- addColumnItem () {
441
- this.type = '新增'
442
- this.$refs.queryItem.addColumnItemExecute()
443
- },
444
- editColumnItem (key, index) {
445
- if (this.itemMap[key]) {
446
- this.$refs.queryItem.editColumnItemExecute(this.itemMap[key])
447
- this.selectIndex = index
448
- } else {
449
- this.$message.warn('编辑失败')
450
- }
451
- },
452
- removeColumnItem (key, index) {
453
- const _this = this
454
- this.$confirm({
455
- title: '您确定要删除该数据项?',
456
- content: '删除的数据项无法恢复',
457
- okText: '确定',
458
- okType: 'danger',
459
- cancelText: '取消',
460
- onOk () {
461
- delete _this.itemMap[key]
462
- _this.form.column.splice(index, 1)
463
- }
464
- })
465
- },
466
- upColumnItem (key, index) {
467
- const newIndex = index - 1
468
- const itemA = this.form.column[newIndex]
469
- const itemB = this.form.column[index]
470
- this.form.column.splice(index, 1, itemA)
471
- this.form.column.splice(newIndex, 1, itemB)
472
- },
473
- downColumnItem (key, index) {
474
- const newIndex = index + 1
475
- const itemA = this.form.column[newIndex]
476
- const itemB = this.form.column[index]
477
- this.form.column.splice(index, 1, itemA)
478
- this.form.column.splice(newIndex, 1, itemB)
479
- },
480
- changeJoinArray () {
481
- const joinArrayObject = {}
482
- for (const item of this.joinArray) {
483
- joinArrayObject[item.key] = item.value
484
- }
485
- this.form.joinArray = joinArrayObject
486
- },
487
- exportJson () {
488
- const data = JSON.stringify(this.form, null, 2)
489
- const blob = new Blob([data], { type: 'application/json' })
490
- FileSaver.saveAs(blob, `Query.json`)
491
- this.$message.success('导出成功!')
492
- },
493
- viewHandle (then) {
494
- if (this.form.column.length === 0) {
495
- this.$message.error('你没有增加任何数据字段')
496
- return
497
- }
498
- this.result = JSON.parse(JSON.stringify(this.form))
499
- if (this.result.condition && (!this.result.condition.value || this.result.condition.value === '')) {
500
- delete this.result.condition
501
- }
502
- for (const item of this.result.column) {
503
- if (item.selectType === 'fixArray') {
504
- item.selectKey = JSON.parse(item.selectKey)
505
- }
506
- if (item.dataModeArray.length === 5) {
507
- delete item.dataModeArray
508
- }
509
- }
510
- // 查询插槽array to object
511
- delete this.result.apiSlotView
512
- for (let i = 0; i < this.form.apiSlotView.length; i++) {
513
- this.result.apiSlot[this.form.apiSlotView[i].slotType] = this.form.apiSlotView[i].slotName
514
- }
515
- this.result.buttonState = this.buttonState
516
- then()
517
- },
518
- view () {
519
- this.$refs.businessCreateForm.validate(valid => {
520
- if (valid) {
521
- this.viewHandle(() => {
522
- this.modelVisible = true
523
- })
524
- }
525
- })
526
- },
527
- submit () {
528
- this.onModelClose()
529
- this.$refs.businessCreateForm.validate(valid => {
530
- if (valid) {
531
- this.viewHandle(() => {
532
- // saveQueryParams
533
- this.$emit('saveQueryParams', this.result)
534
- })
535
- }
536
- })
537
- },
538
- // 判断是否为json字符串
539
- isJSON (str) {
540
- if (typeof str == 'string') {
541
- try {
542
- const obj = JSON.parse(str)
543
- return !!(typeof obj == 'object' && obj)
544
- } catch (e) {
545
- return false
546
- }
547
- }
548
- },
549
- // 拖拽事件
550
- handleDragStart (e, item) {
551
- this.dragging = item
552
- },
553
- handleDragEnd (e, item) {
554
- if (this.ending.key === this.dragging.key) {
555
- return
556
- }
557
- const newItems = [...this.form.column]
558
- const src = newItems.indexOf(this.dragging)
559
- const dst = newItems.indexOf(this.ending)
560
- // newItems.splice(src, 1, ...newItems.splice(dst, 1, newItems[src])) 替换位置
561
- // 挪移位置
562
- newItems.splice(src, 1)
563
- newItems.splice(dst, 0, this.dragging)
564
- this.form.column = newItems
565
- this.$nextTick(() => {
566
- this.dragging = null
567
- this.ending = null
568
- })
569
- },
570
- handleDragOver (e) {
571
- // 首先把div变成可以放置的元素,即重写dragenter/dragover
572
- // e.dataTransfer.dropEffect="move";//在dragenter中针对放置目标来设置!
573
- e.dataTransfer.dropEffect = 'move'
574
- },
575
- handleDragEnter (e, item) {
576
- // 为需要移动的元素设置dragstart事件
577
- e.dataTransfer.effectAllowed = 'move'
578
- this.ending = item
579
- }
580
- }
581
- }
582
- </script>
583
- <style lang="less" scoped>
584
- .column_item :hover {
585
- background-color:rgba(64, 169, 255,0.25);
586
- }
587
- .dragTips{
588
- display:block;
589
- text-align: center;
590
- border:1px dashed rgba(64, 169, 255,0.55);
591
- }
592
- .dragTipsWarp{
593
- padding: .5rem 2rem;
594
- }
595
- </style>
1
+ <template>
2
+ <a-drawer
3
+ :visible="visible"
4
+ :width="isMobile ? screenWidth : screenWidth * 0.85"
5
+ placement="right"
6
+ title="查询配置生成"
7
+ @close="onClose"
8
+ >
9
+ <a-row :gutter="24">
10
+ <a-col :lg="12" :md="12" :sm="24" :xl="14" :xs="24">
11
+ <a-form-model
12
+ ref="businessCreateForm"
13
+ :label-col="labelCol"
14
+ :model="form"
15
+ :rules="rules"
16
+ :wrapper-col="wrapperCol"
17
+ >
18
+ <a-form-model-item label="服务名称" prop="serviceName">
19
+ <a-input v-model="form.serviceName" placeholder="查询接口所在的服务名称,默认af-system"/>
20
+ </a-form-model-item>
21
+ <a-form-model-item label="查询主表名" prop="tableName">
22
+ <a-input v-model="form.tableName" placeholder="查询用的主表+别名,用空格隔开,如:t_userfiles u"/>
23
+ </a-form-model-item>
24
+ <a-form-model-item label="预设关联表" prop="joinArray">
25
+ <a-popover placement="right" title="说明">
26
+ <template slot="content">
27
+ <p>配置你的查询中可能涉及到的所有关联表</p>
28
+ <p>比如你的查询中涉及到了t_userinfo表,就需要加入t_userinfo表的关联,和你写SQL的关联一样</p>
29
+ <p>请注意,你关联的<span style="color: #ff0000">除主表外的任何表</span>都需要配置</p>
30
+ <a-input-group compact style="width: 400px;">
31
+ <a-input placeholder="表别名" readOnly style="width: 20%"/>
32
+ <a-input placeholder="关联条件" readOnly style="width: 80%"/>
33
+ </a-input-group>
34
+ <a-input-group compact style="width: 400px;">
35
+ <a-input placeholder="表别名" readOnly style="width: 20%" value="ui"/>
36
+ <a-input
37
+ placeholder="关联条件"
38
+ readOnly
39
+ style="width: 80%"
40
+ value="t_userinfo ui on i.f_userinfo_id = ui.f_userinfo_id"/>
41
+ </a-input-group>
42
+ </template>
43
+ <a-button type="primary" @click="addJoinItem()">增加</a-button>
44
+ </a-popover>
45
+ <div v-for="(itemObj, index) in joinArray" :key="index">
46
+ <a-input-group compact>
47
+ <a-input
48
+ v-model="itemObj.key"
49
+ placeholder="表别名"
50
+ style="width: 20%;position: relative;bottom: 1px;"
51
+ @change="changeJoinArray()"/>
52
+ <a-input v-model="itemObj.value" placeholder="关联条件" style="width: 80%" @change="changeJoinArray()">
53
+ <a-icon slot="addonAfter" type="close" @click="removeJoinItem(index)"/>
54
+ </a-input>
55
+ </a-input-group>
56
+ </div>
57
+ </a-form-model-item>
58
+ <a-form-model-item label="SQL查询表达式" prop="condition">
59
+ <a-input v-model="form.condition.value" placeholder="用作SQL查询的固定条件表达式,如:gb.f_meter_type='物联网表',可选"/>
60
+ <template v-if="form.condition.value && Object.keys(form.joinArray).length > 0">
61
+ <a-alert message="提示:如果SQL查询表达式中用到了关联表,需要勾选用到的关联表别名" type="success"/>
62
+ <a-checkbox-group v-model="form.condition.join" :options="conditionJoinArray"/>
63
+ </template>
64
+ </a-form-model-item>
65
+ <a-form-model-item label="排序方式" prop="orderBy">
66
+ <a-input v-model="form.orderBy" placeholder="排序字段,用别名+字段名+排序方式(可选)表示,如:u.id desc"/>
67
+ </a-form-model-item>
68
+ <a-form-model-item label="数据字段" prop="column">
69
+ <a-button type="primary" @click="addColumnItem()">增加</a-button>
70
+ <div
71
+ v-for="(columnItem, index) in form.column"
72
+ :key="index"
73
+ class="column_item"
74
+ draggable="true"
75
+ @dragend="handleDragEnd($event, columnItem)"
76
+ @dragenter="handleDragEnter($event, columnItem)"
77
+ @dragstart="handleDragStart($event, columnItem)"
78
+ @dragover.prevent="handleDragOver($event, columnItem)">
79
+ <a-row v-if="ending && dragging && columnItem.key === ending.key && dragging.key !== ending.key" class="dragTipsWarp">
80
+ <span class="dragTips">拖到此处放置</span>
81
+ </a-row>
82
+ <a-row :gutter="24">
83
+ <a-col :span="20">
84
+ <span style="font-weight: bold">{{ columnItem.title }}({{ columnItem.key }})</span>
85
+ </a-col>
86
+ <a-col :span="2">
87
+ <a-icon type="edit" @click="editColumnItem(columnItem.key,index)"/>
88
+ </a-col>
89
+ <a-col :span="2">
90
+ <a-icon type="close" @click="removeColumnItem(columnItem.key,index)"/>
91
+ </a-col>
92
+ </a-row>
93
+ </div>
94
+ </a-form-model-item>
95
+ <a-form-model-item label="操作按钮配置" prop="buttonState">
96
+ <a-checkbox-group v-model="buttonStateData" :options="buttonStateArray"/>
97
+ </a-form-model-item>
98
+ <a-form-model-item label="接口插槽" prop="apiSlot">
99
+ <a-popover placement="right" title="说明">
100
+ <template slot="content">
101
+ <p>通过插槽,你可以轻松扩展查询配置的默认业务,配置的插槽会根据类型在默认业务的前后执行,类似于切面</p>
102
+ <p><span style="color: #FF0000">注意:插槽不应改变默认的业务行为</span></p>
103
+ <p>如果配置了业务执行前插槽,<br/>执行默认业务前会先执行设置的插槽logic,传给插槽的参数为data本身</p>
104
+ <p>如果配置了业务执行后插槽,<br/>执行默认业务后会执行设置的插槽logic,传给插槽的参数为默认业务的返回值</p>
105
+ <img src="@vue2-client/assets/img/querySlotDemo.svg" style="zoom:0.5">
106
+ </template>
107
+ <a-button type="primary" @click="addapiSlot()">增加</a-button>
108
+ </a-popover>
109
+ <div
110
+ v-for="(columnItem, index) in form.apiSlotView"
111
+ :key="index"
112
+ >
113
+ <a-input v-model="columnItem.slotName" placeholder="插槽logic名称">
114
+ <a-select slot="addonBefore" v-model="columnItem.slotType" style="width: 10rem" placeholder="选择插槽类型">
115
+ <a-select-option
116
+ v-for="item in apiSlotData"
117
+ :key="item.value"
118
+ :value="item.value">
119
+ {{ item.label }}
120
+ </a-select-option>
121
+ </a-select>
122
+ </a-input>
123
+ </div>
124
+ </a-form-model-item>
125
+ </a-form-model>
126
+ <create-query-item ref="queryItem" @getColumn="getColumn" @itemHandle="itemHandle"/>
127
+ <a-button type="primary" @click="view">操作</a-button>
128
+ </a-col>
129
+ <a-col :lg="12" :md="12" :sm="24" :xl="10" :xs="24">
130
+ <a-card :bordered="false" size="small" style="overflow: auto" title="预览">
131
+ <json-viewer
132
+ :copyable="{copyText: '复制', copiedText: '已复制'}"
133
+ :expand-depth="parseInt('100')"
134
+ :value="result"
135
+ style="overflow: auto;max-height: 440px"></json-viewer>
136
+ </a-card>
137
+ </a-col>
138
+ </a-row>
139
+ <a-modal
140
+ :centered="true"
141
+ :destroyOnClose="true"
142
+ :visible="modelVisible"
143
+ :width="isMobile ? screenWidth : screenWidth * 0.8"
144
+ :zIndex="1001"
145
+ title="效果预览"
146
+ @cancel="onModelClose">
147
+ <template slot="footer">
148
+ <a-button key="close" @click="onModelClose">
149
+ 返回
150
+ </a-button>
151
+ <a-button key="submit" type="primary" @click="submit">
152
+ 保存
153
+ </a-button>
154
+ </template>
155
+ <x-form-table
156
+ :queryParamsJson="result"
157
+ :view-mode="true">
158
+ </x-form-table>
159
+ </a-modal>
160
+ </a-drawer>
161
+ </template>
162
+
163
+ <script>
164
+ import XFormItem from '@vue2-client/base-client/components/common/XForm/XFormItem'
165
+ import XFormTable from '@vue2-client/base-client/components/common/XFormTable/XFormTable'
166
+ import JsonViewer from 'vue-json-viewer'
167
+ import FileSaver from 'file-saver'
168
+ import { mapState } from 'vuex'
169
+ import CreateQueryItem from '@vue2-client/base-client/components/common/CreateQuery/CreateQueryItem'
170
+
171
+ export default {
172
+ name: 'CreateQuery',
173
+ components: {
174
+ CreateQueryItem,
175
+ JsonViewer,
176
+ XFormTable,
177
+ XFormItem
178
+ },
179
+ data () {
180
+ return {
181
+ // 页面宽度
182
+ screenWidth: document.documentElement.clientWidth,
183
+ // 效果预览模态框是否展示
184
+ modelVisible: false,
185
+ // 操作按钮状态集合
186
+ buttonStateArray: [
187
+ { label: '新增', value: 'add' },
188
+ { label: '修改', value: 'edit' },
189
+ { label: '删除', value: 'delete' },
190
+ { label: '导入', value: 'import' },
191
+ { label: '导出', value: 'export' }
192
+ ],
193
+ // 操作按钮状态集合值
194
+ buttonStateData: ['add', 'edit', 'delete', 'import', 'export'],
195
+ labelCol: { span: 4 },
196
+ wrapperCol: { span: 14 },
197
+ form: {
198
+ serviceName: '',
199
+ tableName: '',
200
+ joinArray: {},
201
+ condition: {},
202
+ orderBy: '',
203
+ column: [],
204
+ apiSlotView: [],
205
+ apiSlot: {},
206
+ },
207
+ result: {},
208
+ itemMap: {},
209
+ selectIndex: null,
210
+ joinArray: [],
211
+ rules: {
212
+ tableName: [{ required: true, message: '请输入查询表名', trigger: 'blur' }],
213
+ orderBy: [{ required: true, message: '请输入排序方式', trigger: 'blur' }]
214
+ },
215
+ ending: null,
216
+ dragging: null,
217
+ // 操作按钮状态集合
218
+ apiSlotData: [
219
+ { label: '查询前插槽', value: 'queryBefore' },
220
+ { label: '查询后插槽', value: 'queryAfter' },
221
+ { label: '删除前插槽', value: 'delBefore' },
222
+ { label: '删除后插槽', value: 'delAfter' },
223
+ { label: '获取行信息前插槽', value: 'editQueryBefore' },
224
+ { label: '获取行信息后插槽', value: 'editQueryAfter' },
225
+ { label: '新增/修改保存前插槽', value: 'addOrEditSaveBefore' },
226
+ { label: '新增/修改保存后插槽', value: 'addOrEditSaveAfter' },
227
+ ],
228
+ }
229
+ },
230
+ mounted () {
231
+ this.initView()
232
+ },
233
+ computed: {
234
+ ...mapState('setting', ['isMobile']),
235
+ conditionJoinArray: function () {
236
+ const result = []
237
+ for (const item in this.form.joinArray) {
238
+ if (item !== '') {
239
+ result.push({
240
+ label: item,
241
+ value: item
242
+ })
243
+ }
244
+ }
245
+ if (result.length === 0) {
246
+ result.push({
247
+ label: '-',
248
+ value: '-'
249
+ })
250
+ }
251
+ return result
252
+ },
253
+ buttonState: function () {
254
+ const result = {
255
+ add: false,
256
+ edit: false,
257
+ delete: false,
258
+ export: false
259
+ }
260
+ for (const item of this.buttonStateData) {
261
+ result[item] = true
262
+ }
263
+ return result
264
+ }
265
+ },
266
+ props: {
267
+ visible: {
268
+ type: Boolean,
269
+ default: false
270
+ },
271
+ toEditJson: {
272
+ type: Object,
273
+ default: () => {}
274
+ }
275
+ },
276
+ watch: {
277
+ visible (rel) {
278
+ if (rel) {
279
+ this.initView()
280
+ }
281
+ if (rel && this.toEditJson) {
282
+ // 处理预设关联表
283
+ if (this.joinArray.length === 0) {
284
+ for (const key in this.toEditJson.joinArray) {
285
+ this.joinArray.push({
286
+ key: key,
287
+ value: this.toEditJson.joinArray[key]
288
+ })
289
+ }
290
+ }
291
+ // apiSlot obj to arr
292
+ if (this.toEditJson.apiSlot) {
293
+ this.toEditJson.apiSlotView = []
294
+ for (const key of Object.keys(this.toEditJson.apiSlot)) {
295
+ this.toEditJson.apiSlotView.push({ slotType: key, slotName: this.toEditJson.apiSlot[key] })
296
+ }
297
+ }
298
+ // 处理具体表单项
299
+ this.form = Object.assign(
300
+ {
301
+ tableName: '',
302
+ joinArray: {},
303
+ condition: {},
304
+ apiSlotView: [],
305
+ apiSlot: {},
306
+ orderBy: '',
307
+ column: []
308
+ }, this.toEditJson
309
+ )
310
+ for (const columnItem of this.form.column) {
311
+ // 数据模式兼容性处理
312
+ if (!(columnItem.dataMode || columnItem.dataModeArray)) {
313
+ columnItem.dataModeArray = ['queryForm', 'table', 'addOrEditForm', 'sqlQueryItem', 'sqlQueryCondition']
314
+ } else if (columnItem.dataMode) {
315
+ if (columnItem.dataMode === 'all') {
316
+ columnItem.dataModeArray = ['queryForm', 'table', 'addOrEditForm', 'sqlQueryItem', 'sqlQueryCondition']
317
+ } else if (columnItem.dataMode === 'form') {
318
+ columnItem.dataModeArray = ['queryForm', 'addOrEditForm', 'sqlQueryCondition']
319
+ } else if (columnItem.dataMode === 'table') {
320
+ columnItem.dataModeArray = ['table', 'sqlQueryItem']
321
+ } else if (columnItem.dataMode === 'table_form') {
322
+ columnItem.dataModeArray = ['table', 'sqlQueryItem', 'sqlQueryCondition']
323
+ } else if (columnItem.dataMode === 'only_form') {
324
+ columnItem.dataModeArray = ['queryForm']
325
+ } else if (columnItem.dataMode === 'only_table') {
326
+ columnItem.dataModeArray = ['table']
327
+ } else if (columnItem.dataMode === 'clear') {
328
+ columnItem.dataModeArray = ['sqlQueryItem', 'sqlQueryCondition']
329
+ } else if (columnItem.dataMode === 'only_add_modify') {
330
+ columnItem.dataModeArray = ['addOrEditForm', 'sqlQueryItem']
331
+ }
332
+ }
333
+ delete columnItem.dataMode
334
+ // 插槽兼容处理
335
+ if (columnItem.slot) {
336
+ if (columnItem.slot.value && columnItem.slot.type === 'ellipsis') {
337
+ columnItem.slotValue = columnItem.slot.value
338
+ } else if (columnItem.slot.keyMap && columnItem.slot.type === 'badge') {
339
+ columnItem.slotKeyMap = columnItem.slot.keyMap
340
+ } else if (columnItem.slot.actionText && columnItem.slot.type === 'action') {
341
+ columnItem.actionText = columnItem.slot.actionText
342
+ }
343
+ }
344
+ // 必选项兼容处理
345
+ if (columnItem.rule && columnItem.rule.required && columnItem.rule.required !== 'false') {
346
+ columnItem.rule.required = columnItem.rule.required.toString()
347
+ } else {
348
+ if (!columnItem.rule) {
349
+ columnItem.rule = {}
350
+ }
351
+ columnItem.rule.required = 'false'
352
+ }
353
+ // 数据源加载方式兼容处理
354
+ if (columnItem.lazyLoad && columnItem.lazyLoad !== 'false') {
355
+ columnItem.lazyLoad = columnItem.lazyLoad.toString()
356
+ } else {
357
+ columnItem.lazyLoad = 'false'
358
+ }
359
+ // 下拉框数据源兼容处理
360
+ if ((columnItem.formType === 'select' || columnItem.formType === 'cascader') && columnItem.selectKey) {
361
+ // 数据源为logic
362
+ if (columnItem.selectKey.toString().startsWith('logic@')) {
363
+ columnItem.selectType = 'logic'
364
+ } else if (columnItem.selectKey instanceof Array || this.isJSON(columnItem.selectKey)) {
365
+ // 数据源为固定json集合
366
+ if (columnItem.selectKey instanceof Array) {
367
+ columnItem.selectKey = JSON.stringify(columnItem.selectKey)
368
+ }
369
+ columnItem.selectType = 'fixArray'
370
+ } else {
371
+ columnItem.selectType = 'key'
372
+ }
373
+ }
374
+ this.itemMap[columnItem.key] = Object.assign({
375
+ key: '',
376
+ title: '',
377
+ slot: {},
378
+ rule: {
379
+ required: 'false'
380
+ },
381
+ dataModeArray: []
382
+ }, columnItem)
383
+ }
384
+ // 处理操作按钮配置
385
+ if (this.toEditJson.buttonState) {
386
+ this.buttonStateData = []
387
+ for (const buttonStateKey of Object.keys(this.toEditJson.buttonState)) {
388
+ if (this.toEditJson.buttonState[buttonStateKey]) {
389
+ this.buttonStateData.push(buttonStateKey)
390
+ }
391
+ }
392
+ } else {
393
+ this.buttonStateData = ['add', 'edit', 'delete', 'import', 'export']
394
+ }
395
+ }
396
+ }
397
+ },
398
+ methods: {
399
+ // 新增接口插槽
400
+ addapiSlot () {
401
+ if (this.form.apiSlotView.length < this.apiSlotData.length) {
402
+ this.form.apiSlotView.push({})
403
+ }
404
+ },
405
+ // 初始化组件
406
+ initView () {
407
+ this.joinArray = []
408
+ this.result = {}
409
+ },
410
+ onClose () {
411
+ this.$emit('update:visible', false)
412
+ },
413
+ onModelClose () {
414
+ this.modelVisible = false
415
+ },
416
+ addJoinItem () {
417
+ this.joinArray.push({
418
+ key: '',
419
+ value: ''
420
+ })
421
+ this.changeJoinArray()
422
+ },
423
+ removeJoinItem (index) {
424
+ this.joinArray.splice(index, 1)
425
+ this.changeJoinArray()
426
+ },
427
+ itemHandle (item, type) {
428
+ this.itemMap[item.key] = item
429
+ if (type === '新增') {
430
+ this.form.column.push(item)
431
+ } else {
432
+ this.$set(this.form.column, this.selectIndex, item)
433
+ }
434
+ this.$message.success(`${type}成功`)
435
+ this.$refs.queryItem.flashModal(false)
436
+ },
437
+ getColumn (callback) {
438
+ callback(JSON.parse(JSON.stringify(this.form.column)))
439
+ },
440
+ addColumnItem () {
441
+ this.type = '新增'
442
+ this.$refs.queryItem.addColumnItemExecute()
443
+ },
444
+ editColumnItem (key, index) {
445
+ if (this.itemMap[key]) {
446
+ this.$refs.queryItem.editColumnItemExecute(this.itemMap[key])
447
+ this.selectIndex = index
448
+ } else {
449
+ this.$message.warn('编辑失败')
450
+ }
451
+ },
452
+ removeColumnItem (key, index) {
453
+ const _this = this
454
+ this.$confirm({
455
+ title: '您确定要删除该数据项?',
456
+ content: '删除的数据项无法恢复',
457
+ okText: '确定',
458
+ okType: 'danger',
459
+ cancelText: '取消',
460
+ onOk () {
461
+ delete _this.itemMap[key]
462
+ _this.form.column.splice(index, 1)
463
+ }
464
+ })
465
+ },
466
+ upColumnItem (key, index) {
467
+ const newIndex = index - 1
468
+ const itemA = this.form.column[newIndex]
469
+ const itemB = this.form.column[index]
470
+ this.form.column.splice(index, 1, itemA)
471
+ this.form.column.splice(newIndex, 1, itemB)
472
+ },
473
+ downColumnItem (key, index) {
474
+ const newIndex = index + 1
475
+ const itemA = this.form.column[newIndex]
476
+ const itemB = this.form.column[index]
477
+ this.form.column.splice(index, 1, itemA)
478
+ this.form.column.splice(newIndex, 1, itemB)
479
+ },
480
+ changeJoinArray () {
481
+ const joinArrayObject = {}
482
+ for (const item of this.joinArray) {
483
+ joinArrayObject[item.key] = item.value
484
+ }
485
+ this.form.joinArray = joinArrayObject
486
+ },
487
+ exportJson () {
488
+ const data = JSON.stringify(this.form, null, 2)
489
+ const blob = new Blob([data], { type: 'application/json' })
490
+ FileSaver.saveAs(blob, `Query.json`)
491
+ this.$message.success('导出成功!')
492
+ },
493
+ viewHandle (then) {
494
+ if (this.form.column.length === 0) {
495
+ this.$message.error('你没有增加任何数据字段')
496
+ return
497
+ }
498
+ this.result = JSON.parse(JSON.stringify(this.form))
499
+ if (this.result.condition && (!this.result.condition.value || this.result.condition.value === '')) {
500
+ delete this.result.condition
501
+ }
502
+ for (const item of this.result.column) {
503
+ if (item.selectType === 'fixArray') {
504
+ item.selectKey = JSON.parse(item.selectKey)
505
+ }
506
+ if (item.dataModeArray.length === 5) {
507
+ delete item.dataModeArray
508
+ }
509
+ }
510
+ // 查询插槽array to object
511
+ delete this.result.apiSlotView
512
+ for (let i = 0; i < this.form.apiSlotView.length; i++) {
513
+ this.result.apiSlot[this.form.apiSlotView[i].slotType] = this.form.apiSlotView[i].slotName
514
+ }
515
+ this.result.buttonState = this.buttonState
516
+ then()
517
+ },
518
+ view () {
519
+ this.$refs.businessCreateForm.validate(valid => {
520
+ if (valid) {
521
+ this.viewHandle(() => {
522
+ this.modelVisible = true
523
+ })
524
+ }
525
+ })
526
+ },
527
+ submit () {
528
+ this.onModelClose()
529
+ this.$refs.businessCreateForm.validate(valid => {
530
+ if (valid) {
531
+ this.viewHandle(() => {
532
+ // saveQueryParams
533
+ this.$emit('saveQueryParams', this.result)
534
+ })
535
+ }
536
+ })
537
+ },
538
+ // 判断是否为json字符串
539
+ isJSON (str) {
540
+ if (typeof str == 'string') {
541
+ try {
542
+ const obj = JSON.parse(str)
543
+ return !!(typeof obj == 'object' && obj)
544
+ } catch (e) {
545
+ return false
546
+ }
547
+ }
548
+ },
549
+ // 拖拽事件
550
+ handleDragStart (e, item) {
551
+ this.dragging = item
552
+ },
553
+ handleDragEnd (e, item) {
554
+ if (this.ending.key === this.dragging.key) {
555
+ return
556
+ }
557
+ const newItems = [...this.form.column]
558
+ const src = newItems.indexOf(this.dragging)
559
+ const dst = newItems.indexOf(this.ending)
560
+ // newItems.splice(src, 1, ...newItems.splice(dst, 1, newItems[src])) 替换位置
561
+ // 挪移位置
562
+ newItems.splice(src, 1)
563
+ newItems.splice(dst, 0, this.dragging)
564
+ this.form.column = newItems
565
+ this.$nextTick(() => {
566
+ this.dragging = null
567
+ this.ending = null
568
+ })
569
+ },
570
+ handleDragOver (e) {
571
+ // 首先把div变成可以放置的元素,即重写dragenter/dragover
572
+ // e.dataTransfer.dropEffect="move";//在dragenter中针对放置目标来设置!
573
+ e.dataTransfer.dropEffect = 'move'
574
+ },
575
+ handleDragEnter (e, item) {
576
+ // 为需要移动的元素设置dragstart事件
577
+ e.dataTransfer.effectAllowed = 'move'
578
+ this.ending = item
579
+ }
580
+ }
581
+ }
582
+ </script>
583
+ <style lang="less" scoped>
584
+ .column_item :hover {
585
+ background-color:rgba(64, 169, 255,0.25);
586
+ }
587
+ .dragTips{
588
+ display:block;
589
+ text-align: center;
590
+ border:1px dashed rgba(64, 169, 255,0.55);
591
+ }
592
+ .dragTipsWarp{
593
+ padding: .5rem 2rem;
594
+ }
595
+ </style>