vue2-client 1.4.2 → 1.4.3

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