@netang/quasar 0.0.102 → 0.0.104

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 (162) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +11 -11
  3. package/_docs/docs/.vuepress/client.js +8 -8
  4. package/_docs/docs/.vuepress/config.js +40 -40
  5. package/_docs/docs/.vuepress/configs/index.js +2 -2
  6. package/_docs/docs/.vuepress/configs/navbar/index.js +1 -1
  7. package/_docs/docs/.vuepress/configs/navbar/zh.js +16 -16
  8. package/_docs/docs/.vuepress/configs/sidebar/index.js +1 -1
  9. package/_docs/docs/.vuepress/configs/sidebar/zh.js +75 -75
  10. package/_docs/docs/.vuepress/public/css/index.css +3 -3
  11. package/_docs/docs/.vuepress/styles/index.scss +3 -3
  12. package/_docs/docs/components/column-title.md +25 -25
  13. package/_docs/docs/components/data.md +66 -66
  14. package/_docs/docs/components/dialog.md +59 -59
  15. package/_docs/docs/components/dragger.md +26 -26
  16. package/_docs/docs/components/editor-code.md +16 -16
  17. package/_docs/docs/components/empty.md +13 -13
  18. package/_docs/docs/components/field-date.md +16 -16
  19. package/_docs/docs/components/field-text.md +57 -57
  20. package/_docs/docs/components/field-tree.md +14 -14
  21. package/_docs/docs/components/img.md +25 -25
  22. package/_docs/docs/components/input-number.md +21 -21
  23. package/_docs/docs/components/list-menu-item.md +21 -21
  24. package/_docs/docs/components/list-menu.md +21 -21
  25. package/_docs/docs/components/power-page.md +21 -21
  26. package/_docs/docs/components/price.md +21 -21
  27. package/_docs/docs/components/render.md +12 -12
  28. package/_docs/docs/components/search-item.md +10 -10
  29. package/_docs/docs/components/search.md +12 -12
  30. package/_docs/docs/components/select.md +11 -11
  31. package/_docs/docs/components/splitter.md +15 -15
  32. package/_docs/docs/components/table-column-fixed.md +20 -20
  33. package/_docs/docs/components/table-pagination.md +20 -20
  34. package/_docs/docs/components/table-splitter.md +20 -20
  35. package/_docs/docs/components/table-summary.md +20 -20
  36. package/_docs/docs/components/table.md +25 -25
  37. package/_docs/docs/components/thumbnail.md +18 -18
  38. package/_docs/docs/components/toolbar.md +9 -9
  39. package/_docs/docs/components/uploader-query.md +19 -19
  40. package/_docs/docs/components/uploader.md +16 -16
  41. package/_docs/docs/components/value-format.md +26 -26
  42. package/_docs/docs/index.md +1 -1
  43. package/_docs/docs/utils/alert.md +26 -26
  44. package/_docs/docs/utils/area.md +112 -112
  45. package/_docs/docs/utils/arr.md +80 -80
  46. package/_docs/docs/utils/auth.md +101 -101
  47. package/_docs/docs/utils/bus.md +18 -18
  48. package/_docs/docs/utils/confirm.md +31 -31
  49. package/_docs/docs/utils/copy.md +22 -22
  50. package/_docs/docs/utils/dialog.md +98 -98
  51. package/_docs/docs/utils/dict.md +50 -50
  52. package/_docs/docs/utils/dictOptions.md +27 -27
  53. package/_docs/docs/utils/form.md +33 -33
  54. package/_docs/docs/utils/getData.md +60 -60
  55. package/_docs/docs/utils/getFile.md +21 -21
  56. package/_docs/docs/utils/getImage.md +33 -33
  57. package/_docs/docs/utils/getTime.md +51 -51
  58. package/_docs/docs/utils/index.md +1 -1
  59. package/_docs/docs/utils/loading.md +18 -18
  60. package/_docs/docs/utils/notify.md +29 -29
  61. package/_docs/docs/utils/power.md +353 -353
  62. package/_docs/docs/utils/previewImage.md +11 -11
  63. package/_docs/docs/utils/price.md +45 -45
  64. package/_docs/docs/utils/rule.md +30 -30
  65. package/_docs/docs/utils/ruleValid.md +31 -31
  66. package/_docs/docs/utils/symbols.md +30 -30
  67. package/_docs/docs/utils/table.md +194 -194
  68. package/_docs/docs/utils/timestamp.md +27 -27
  69. package/_docs/docs/utils/toast.md +27 -27
  70. package/_docs/docs/utils/tree.md +174 -174
  71. package/_docs/docs/utils/uploader.md +29 -29
  72. package/_docs/package.json +11 -11
  73. package/components/column-title/index.vue +37 -37
  74. package/components/data/index.vue +20 -20
  75. package/components/dialog/index.vue +372 -372
  76. package/components/dragger/index.vue +203 -203
  77. package/components/drawer/index.vue +303 -303
  78. package/components/editor-code/index.vue +289 -289
  79. package/components/empty/index.vue +71 -71
  80. package/components/field-date/index.vue +850 -850
  81. package/components/field-date/methods.js +100 -100
  82. package/components/field-table/index.vue +1222 -1222
  83. package/components/field-text/index.vue +165 -165
  84. package/components/field-tree/index.vue +103 -81
  85. package/components/img/index.vue +202 -202
  86. package/components/input-number/index.vue +546 -546
  87. package/components/list-menu/index.vue +149 -149
  88. package/components/list-menu-item/index.vue +79 -79
  89. package/components/power-page/index.vue +92 -92
  90. package/components/price/index.vue +188 -188
  91. package/components/private/components/index.js +11 -11
  92. package/components/private/components/move-to-tree/index.vue +154 -154
  93. package/components/private/edit-power-data/index.vue +816 -816
  94. package/components/private/table-visible-columns-button/index.vue +109 -109
  95. package/components/render/index.vue +150 -150
  96. package/components/search/index.vue +4 -2
  97. package/components/search-item/index.vue +4 -2
  98. package/components/splitter/index.vue +415 -415
  99. package/components/table/index.vue +456 -456
  100. package/components/table-column-fixed/index.vue +112 -112
  101. package/components/table-pagination/index.vue +192 -192
  102. package/components/table-splitter/index.vue +360 -360
  103. package/components/table-summary/index.vue +110 -110
  104. package/components/thumbnail/index.vue +72 -72
  105. package/components/toolbar/container.vue +31 -31
  106. package/components/toolbar/index.vue +136 -136
  107. package/components/uploader/index.vue +158 -158
  108. package/components/uploader-query/index.vue +758 -758
  109. package/components/value-format/index.vue +274 -274
  110. package/configs/area3.js +1 -1
  111. package/docs/css/index.css +3 -3
  112. package/package.json +1 -1
  113. package/sass/common.scss +5 -0
  114. package/sass/index.scss +14 -14
  115. package/sass/line.scss +39 -39
  116. package/sass/quasar/btn.scss +46 -46
  117. package/sass/quasar/common.scss +3 -3
  118. package/sass/quasar/dialog.scss +7 -7
  119. package/sass/quasar/drawer.scss +6 -6
  120. package/sass/quasar/field.scss +243 -243
  121. package/sass/quasar/loading.scss +6 -6
  122. package/sass/quasar/menu.scss +8 -8
  123. package/sass/quasar/table.scss +150 -150
  124. package/sass/quasar/toolbar.scss +22 -22
  125. package/store/index.js +29 -29
  126. package/utils/$auth.js +127 -127
  127. package/utils/$form.js +56 -56
  128. package/utils/$power.js +1215 -1215
  129. package/utils/$rule.js +13 -13
  130. package/utils/$ruleValid.js +10 -10
  131. package/utils/$table.js +999 -999
  132. package/utils/$tree.js +713 -713
  133. package/utils/alert.js +12 -12
  134. package/utils/area.js +400 -400
  135. package/utils/arr.js +51 -51
  136. package/utils/bus.js +6 -6
  137. package/utils/config.js +52 -52
  138. package/utils/confirm.js +11 -11
  139. package/utils/copy.js +30 -30
  140. package/utils/dialog.js +36 -36
  141. package/utils/dict.js +21 -21
  142. package/utils/dictOptions.js +28 -28
  143. package/utils/getData.js +73 -73
  144. package/utils/getFile.js +40 -40
  145. package/utils/getImage.js +153 -153
  146. package/utils/getTime.js +106 -106
  147. package/utils/index.js +61 -61
  148. package/utils/loading.js +15 -15
  149. package/utils/notify.js +13 -13
  150. package/utils/previewImage.js +10 -10
  151. package/utils/price.js +18 -18
  152. package/utils/symbols.js +18 -18
  153. package/utils/timestamp.js +18 -18
  154. package/utils/toast.js +13 -13
  155. package/utils/uploader/aliyun.js +6 -6
  156. package/utils/uploader/local.js +8 -8
  157. package/utils/uploader/qiniu.js +321 -321
  158. package/utils/uploader.js +1059 -1059
  159. package/utils/useAuth.js +30 -30
  160. package/utils/useRouter.js +47 -47
  161. package/utils/useSearch.js +0 -6
  162. package/utils/useUploader.js +53 -53
package/utils/$power.js CHANGED
@@ -1,1215 +1,1215 @@
1
- import { provide, inject, ref, computed } from 'vue'
2
- import { useQuasar } from 'quasar'
3
-
4
- import $n_has from 'lodash/has'
5
- import $n_get from 'lodash/get'
6
- import $n_merge from 'lodash/merge'
7
- import $n_filter from 'lodash/filter'
8
- import $n_toLower from 'lodash/toLower'
9
- import $n_isNumber from 'lodash/isNumber'
10
- import $n_cloneDeep from 'lodash/cloneDeep'
11
- import $n_isFunction from 'lodash/isFunction'
12
- import $n_pick from 'lodash/pick'
13
-
14
- import $n_router from '@netang/utils/vue/router'
15
-
16
- import $n_isValidArray from '@netang/utils/isValidArray'
17
- import $n_isValidObject from '@netang/utils/isValidObject'
18
- import $n_isValidString from '@netang/utils/isValidString'
19
- import $n_isRequired from '@netang/utils/isRequired'
20
- import $n_isNumeric from '@netang/utils/isNumeric'
21
- import $n_forEach from '@netang/utils/forEach'
22
- import $n_forIn from '@netang/utils/forIn'
23
- import $n_slash from '@netang/utils/slash'
24
- import $n_json from '@netang/utils/json'
25
- import $n_join from '@netang/utils/join'
26
- import $n_fail from '@netang/utils/fail'
27
- import $n_success from '@netang/utils/success'
28
- import $n_split from '@netang/utils/split'
29
- import $n_trimString from '@netang/utils/trimString'
30
- import $n_numberDeep from '@netang/utils/numberDeep'
31
- import $n_indexOf from '@netang/utils/indexOf'
32
- import $n_runAsync from '@netang/utils/runAsync'
33
- import $n_run from '@netang/utils/run'
34
- import $n_http from '@netang/utils/http'
35
-
36
- import { statePower } from '../store'
37
- import { NRenderKey, NPowerKey, NDialogKey, NFormKey, NTableKey } from './symbols'
38
-
39
- import $n_getData from './getData'
40
- import $n_toast from './toast'
41
- import $n_confirm from './confirm'
42
- import $n_bus from './bus'
43
-
44
- import $n_config from './config'
45
-
46
- import { configs } from './config'
47
-
48
- const {
49
- // 字典常量
50
- dicts,
51
- } = configs
52
-
53
- /**
54
- * 创建权限实例
55
- */
56
- function create(options) {
57
-
58
- // 获取参数
59
- const o = Object.assign({
60
- // 路由路径
61
- path: '',
62
- // 路由参数
63
- query: {},
64
- // 是否加载页面
65
- pageLoading: false,
66
- // 页面状态
67
- pageStatus: null,
68
- // 空状态描述
69
- emptyDescription: '',
70
- // 是否开启权限
71
- power: true,
72
- // 是否显示权限按钮
73
- showPowerBtns: true,
74
- // 是否显示工具栏权限按钮
75
- showToolbarPowerBtns: true,
76
- // 格式化权限按钮
77
- formatPowerBtns: null,
78
- // 左边侧滑菜单图标
79
- leftDrawerIcon: 'format_list_bulleted',
80
- // 右边侧滑菜单图标
81
- rightDrawerIcon: 'search',
82
- // 请求前执行
83
- requestBefore: null,
84
- // 请求成功执行
85
- requestSuccess: null,
86
- // 请求失败执行
87
- requestFail: null,
88
- // 请求后执行
89
- requestAfter: null,
90
- }, options)
91
-
92
- // 获取对话框渲染注入
93
- const $dialog = inject(NDialogKey)
94
- const hasDialog = !! $dialog
95
-
96
- // 获取渲染注入
97
- const $render = inject(NRenderKey)
98
- const hasRender = !! $render
99
-
100
- // 如果有对话框注入
101
- if (hasDialog) {
102
- const {
103
- dialogProps,
104
- } = $dialog
105
-
106
- // 合并权限参数
107
- Object.assign(o, $n_pick(dialogProps, [ 'path', 'query' ]))
108
-
109
- // 合并权限参数
110
- if ($n_has($dialog, 'props.powerProps') && $n_isValidObject($dialog.props.powerProps)) {
111
- $n_merge(o, $dialog.props.powerProps)
112
- }
113
- }
114
-
115
- // 如果有渲染注入
116
- if (hasRender) {
117
- // 如果有权限传参, 则合并参数
118
- const powerProps = $n_get($render, 'props.powerProps')
119
- if ($n_isValidObject(powerProps)) {
120
- $n_merge(o, powerProps)
121
- }
122
- }
123
-
124
- // 获取当前路由
125
- const $currentRoute = $n_router.getRoute()
126
-
127
- // 权限路由
128
- let $route
129
-
130
- // 如果没有路由
131
- if (o.path === false) {
132
-
133
- // 设为空路由
134
- $route = {
135
- fullPath: '',
136
- path: '',
137
- query: $n_isValidObject(o.query) ? o.query : {},
138
- }
139
-
140
- // 如果有自定义路径
141
- } else if ($n_isValidString(o.path)) {
142
-
143
- // 获取自定义路由
144
- $route = $n_router.resolve({
145
- path: o.path,
146
- query: $n_isValidObject(o.query) ? o.query : {},
147
- })
148
-
149
- // 如果在渲染组件内 && 该渲染组件有自定义路由
150
- } else if (hasRender && $n_has($render, '$route')) {
151
-
152
- // 设为渲染组件的路由
153
- $route = $render.$route
154
-
155
- // 否则获取当前路由
156
- } else {
157
- $route = $currentRoute
158
- }
159
-
160
- // quasar 对象
161
- const $q = useQuasar()
162
-
163
- // 表格已选数据
164
- const tableSelected = ref([])
165
-
166
- // 是否显示左边侧滑菜单
167
- const leftDrawerModelValue = ref(null)
168
-
169
- // 是否显示右边侧滑菜单
170
- const rightDrawerModelValue = ref(null)
171
-
172
- /**
173
- * 检查是否上传中
174
- */
175
- function checkUploading() {
176
- for (const uploader of data.uploader) {
177
- if (uploader.checkUploading()) {
178
- return true
179
- }
180
- }
181
- return false
182
- }
183
-
184
- const _data = {
185
- // 表格实例
186
- $table: null,
187
- // 表单实例
188
- $form: null,
189
- }
190
-
191
- // 注入数据
192
- const data = {
193
- // 页面加载
194
- pageLoading: ref(o.pageLoading),
195
- // 页面状态
196
- pageStatus: ref(o.pageStatus),
197
- // 空状态描述
198
- emptyDescription: ref(o.emptyDescription),
199
-
200
- // 当前路由全路径
201
- routeFullPath: $route.fullPath,
202
- // 当前路由路径
203
- routePath: $route.path,
204
- // 当前路由参数
205
- routeQuery: $route.query,
206
- // 获取当前路由
207
- getRoute() {
208
- return $route
209
- },
210
- // 格式化权限按钮
211
- formatPowerBtns: o.formatPowerBtns,
212
-
213
- // 左边侧滑菜单数据
214
- leftDrawer: {
215
- // 图标
216
- icon: o.leftDrawerIcon,
217
- // 是否显示
218
- modelValue: leftDrawerModelValue,
219
- // 是否显示切换按钮
220
- showButton() {
221
- return leftDrawerModelValue.value !== null
222
- },
223
- // 切换
224
- toggle() {
225
- if (leftDrawerModelValue.value !== null) {
226
- leftDrawerModelValue.value = ! leftDrawerModelValue.value
227
- }
228
- },
229
- },
230
- // 右边侧滑菜单数据
231
- rightDrawer: {
232
- // 图标
233
- icon: o.rightDrawerIcon,
234
- // 是否显示
235
- modelValue: rightDrawerModelValue,
236
- // 是否显示切换按钮
237
- showButton() {
238
- return rightDrawerModelValue.value !== null
239
- },
240
- // 切换
241
- toggle() {
242
- if (rightDrawerModelValue.value !== null) {
243
- rightDrawerModelValue.value = ! rightDrawerModelValue.value
244
- }
245
- },
246
- },
247
- // 表格已选数据
248
- tableSelected,
249
- // 上传器
250
- uploader: [],
251
- // 检查是否上传中
252
- checkUploading,
253
- }
254
-
255
- // 如果是权限页面
256
- if (o.power) {
257
-
258
- // 获取当前页面角色权限
259
- const { status, data: res } = getPageData($route)
260
- if (! status) {
261
- o.pageStatus = false
262
- o.emptyDescription = res.msg
263
- o.power = false
264
-
265
- } else {
266
- // 当前页面权限
267
- data.powerPage = res.page
268
- // 当前页面权限按钮
269
- data.powerBtns = ref(o.showPowerBtns ? res.btns : [])
270
- // 当前页面工具栏权限按钮
271
- data.toolbarPowerBtns = computed(function() {
272
-
273
- if (
274
- // 如果显示工具栏权限按钮
275
- o.showToolbarPowerBtns
276
- // 有权限按钮数据
277
- && $n_isValidArray(data.powerBtns.value)
278
- ) {
279
- const lists = []
280
-
281
- // 格式化权限按钮列表
282
- $n_forEach($n_filter(formatBtns(data.powerBtns.value), e => e.type > 2), function(item) {
283
-
284
- if (! item.hidden) {
285
-
286
- // 如果是单条数据显示
287
- const isSingle = item.show === 'single'
288
-
289
- // 如果是单条 || 多条显示
290
- if (isSingle || item.show === 'multiple') {
291
-
292
- // 初始为不显示
293
- item.show = false
294
-
295
- // 如果有表格选中数据
296
- if ($n_isValidArray(data.tableSelected.value)) {
297
- // 如果是单个显示
298
- if (isSingle) {
299
- item.show = data.tableSelected.value.length === 1
300
-
301
- // 否则是多个显示
302
- } else {
303
- item.show = data.tableSelected.value.length >= 1
304
- }
305
- }
306
- }
307
-
308
- // 如果是手机模式
309
- if ($q.platform.is.mobile) {
310
- item.icon = undefined
311
- }
312
- }
313
-
314
- if (
315
- // 如果有格式化权限按钮方法
316
- $n_isFunction(o.formatPowerBtns)
317
- && o.formatPowerBtns(item, false, tableSelected.value) === false
318
- ) {
319
- return
320
- }
321
-
322
- lists.push(item)
323
- })
324
-
325
- return lists
326
- }
327
-
328
- return []
329
- })
330
-
331
- // 权限按钮点击
332
- data.powerBtnClick = async function (powerBtn, tableSelected) {
333
-
334
- // 权限请求
335
- await request({
336
- // 按钮数据
337
- powerBtn,
338
- // 权限路由参数
339
- $route,
340
- // 当前路由参数
341
- $currentRoute,
342
- // 表格选中数据
343
- tableSelected,
344
- // 表格实例
345
- $table: _data.$table,
346
- // 表单实例
347
- $form: _data.$form,
348
- // 检查是否正在上传文件
349
- checkUploading,
350
-
351
- // 请求前执行
352
- requestBefore: o.requestBefore,
353
- // 请求成功执行
354
- requestSuccess: o.requestSuccess,
355
- // 请求失败执行
356
- requestFail: o.requestFail,
357
- // 请求后执行
358
- requestAfter: o.requestAfter,
359
- })
360
- }
361
- }
362
- }
363
-
364
- // 如果没有开启权限
365
- if (! o.power) {
366
- // 当前页面权限
367
- data.powerPage = {}
368
- // 当前页面权限按钮
369
- data.powerBtns = ref([])
370
- // 当前页面工具栏权限按钮
371
- data.toolbarPowerBtns = ref([])
372
- // 权限按钮点击
373
- data.powerBtnClick = ()=>{}
374
- }
375
-
376
- // 更新数据
377
- data.update = function(cb) {
378
- cb(data, _data)
379
- }
380
-
381
- // 提供可以被后代组件注入的值
382
- provide(NPowerKey, data)
383
-
384
- return data
385
- }
386
-
387
- /**
388
- * 设置权限数据
389
- */
390
- function setData(data) {
391
-
392
- // 如果没有角色数据
393
- if (! $n_isValidObject(data)) {
394
- return
395
- }
396
-
397
- const {
398
- rows,
399
- v,
400
- } = data
401
-
402
- if (! $n_isValidArray(rows) || ! v) {
403
- return
404
- }
405
-
406
- // all id
407
- const all = {}
408
- // 页面
409
- const urls = {}
410
- // 按钮
411
- const btns = {}
412
- // 菜单
413
- const menus = []
414
-
415
- for (const item of rows) {
416
-
417
- // 【格式化 start】
418
- // --------------------------------------------------
419
- if (item.data) {
420
- item.data = $n_json.parse(item.data)
421
- }
422
- item.data = $n_isValidObject(item.data) ? $n_numberDeep(item.data) : {}
423
-
424
- // 设置数据类型
425
- item.data.type = item.data_type
426
- delete item.data_type
427
-
428
- // 标识
429
- item.name = ''
430
-
431
- // 如果有 url
432
- if ($n_isValidString(item.url)) {
433
-
434
- // url 首位加上反斜杠
435
- item.url = $n_slash($n_toLower($n_trimString(item.url)), 'start', true)
436
- if (item.url) {
437
-
438
- item.data.url = item.url
439
-
440
- // 截取最后一个反斜杠
441
- const lastIndex = item.data.url.lastIndexOf('/')
442
- if (lastIndex > -1) {
443
- item.name = item.data.url.substring(lastIndex + 1)
444
- }
445
-
446
- } else {
447
- item.url = ''
448
- item.data.url = ''
449
- }
450
-
451
- } else {
452
- item.url = ''
453
- item.data.url = ''
454
- }
455
-
456
- // 菜单
457
- if (item.type === 1) {
458
- menus.push(item)
459
-
460
- // 按钮
461
- } else if (item.type > 2) {
462
-
463
- // 按钮类型
464
- switch (item.type) {
465
-
466
- // 默认按钮
467
- case 11:
468
- item.color = 'default'
469
- break;
470
-
471
- // 主要按钮
472
- case 12:
473
- item.color = 'primary'
474
- break;
475
-
476
- // 成功按钮
477
- case 13:
478
- item.color = 'secondary'
479
- break;
480
-
481
- // 信息按钮
482
- case 14:
483
- item.color = 'info'
484
- break;
485
-
486
- // 警告按钮
487
- case 15:
488
- item.color = 'warning'
489
- break;
490
-
491
- // 危险按钮
492
- case 16:
493
- item.color = 'negative'
494
- break;
495
-
496
- // 隐藏按钮
497
- case 20:
498
- item.hidden = true
499
- break;
500
- }
501
- }
502
- // 【格式化 end】
503
- // --------------------------------------------------
504
-
505
- // url
506
- if (item.url) {
507
- // 添加至 all
508
- all[item.id] = item
509
- // 添加至页面
510
- urls[item.url] = item
511
- }
512
- }
513
-
514
- for (const item of rows) {
515
-
516
- // 如果有跳转页面
517
- if ($n_has(item.data, 'toPage')) {
518
- // 设置跳转页面地址
519
- item.data.toPage = $n_has(all, item.data.toPage) ? all[item.data.toPage].data.url : null
520
- }
521
-
522
- // 如果有请求成功执行类型
523
- // else if ($n_has(item.data, 'requestSuccess.type')) {
524
- // // 如果请求成功执行类型是关闭窗口、跳转并刷新页面
525
- // if (item.data.requestSuccess.type === 'closePushRefresh') {
526
- // // 设置刷新页面地址
527
- // item.data.requestSuccess.params =
528
- // (
529
- // // 如果有刷新页面的参数 id
530
- // $n_has(item.data.requestSuccess, 'params')
531
- // // 如果有页面数据
532
- // && $n_has(all, item.data.requestSuccess.params)
533
- // ) ? all[item.data.requestSuccess.params].data.url : null
534
- // }
535
- // }
536
-
537
- if (
538
- // 数据/按钮
539
- item.type > 1
540
- // 有父级数据
541
- && $n_has(all, item.pid)
542
- ) {
543
- const pItem = all[item.pid]
544
- if ($n_has(btns, pItem.url)) {
545
- btns[pItem.url].push(item)
546
- } else {
547
- btns[pItem.url] = [item]
548
- }
549
- }
550
- }
551
-
552
- // 保存至权限状态中
553
- statePower.value = {
554
- // 角色权限版本
555
- v,
556
- // all id
557
- all,
558
- // 页面
559
- urls,
560
- // 按钮
561
- btns,
562
- // 菜单
563
- menus,
564
- }
565
- }
566
-
567
- /**
568
- * 获取权限
569
- */
570
- async function getData() {
571
-
572
- // 如果没有权限数据
573
- if (! statePower.value.v) {
574
-
575
- // 获取权限数据
576
- const res = await $n_getData($n_config('apiDataPowerName'))
577
- if (res === false) {
578
- statePower.value = {
579
- // 权限版本
580
- v: null,
581
- // all id
582
- all: {},
583
- // 页面
584
- urls: {},
585
- // 按钮
586
- btns: {},
587
- // 菜单
588
- menus: [],
589
- }
590
- return $n_fail()
591
- }
592
- }
593
-
594
- // 如果有权限状态数据, 则直接返回
595
- if (statePower.value.v) {
596
- return $n_success($n_cloneDeep(statePower.value))
597
- }
598
-
599
- return $n_fail()
600
- }
601
-
602
- /**
603
- * 解析传参
604
- */
605
- function parseQuery(data, settings) {
606
-
607
- // 如果配置是字符串
608
- if ($n_isValidString(settings)) {
609
-
610
- // 如果返回所有传参
611
- if (settings === 'all') {
612
- return $n_isValidObject(data) ? data : {}
613
- }
614
-
615
- // 将字符串放到数组中
616
- settings = [settings]
617
-
618
- // 如果配置是对象
619
- } else if ($n_isValidObject(settings)) {
620
- settings = [settings]
621
- }
622
-
623
- const query = {}
624
-
625
- // 如果配置是数组
626
- if ($n_isValidArray(settings)) {
627
-
628
- // 别名
629
- const alias = {}
630
-
631
- for (let item of settings) {
632
- // 如果是需要的字段
633
- if ($n_isValidString(item)) {
634
-
635
- // 将字段转小写
636
- item = $n_toLower($n_trimString(item))
637
-
638
- // 判断字段是否有 as 别名
639
- const arr = $n_split(item, ' as ')
640
-
641
- // 如果有别名
642
- if (arr.length === 2) {
643
- alias[$n_trimString(arr[0])] = $n_trimString(arr[1])
644
-
645
- // 否则别名就是当前字段本身
646
- } else {
647
- alias[item] = item
648
- }
649
-
650
- // 否则如果是自定义传参
651
- } else if ($n_isValidObject(item)) {
652
- Object.assign(query, item)
653
- }
654
- }
655
-
656
- if (
657
- // 如果有参数数据
658
- $n_isValidObject(data)
659
- // 如果有定义别名
660
- && $n_isValidObject(alias)
661
- ) {
662
- $n_forIn(data, function(value, key) {
663
-
664
- // 如果当前字段在别名中
665
- if ($n_has(alias, key)) {
666
- query[alias[key]] = value
667
- }
668
- })
669
- }
670
- }
671
-
672
- return query
673
- }
674
-
675
- /**
676
- * 格式化权限按钮
677
- */
678
- function formatBtns(powerBtns, filterBtns, toObject = false) {
679
-
680
- const newLists = []
681
-
682
- $n_forEach(powerBtns, function(item) {
683
-
684
- item = $n_cloneDeep(item)
685
-
686
- const {
687
- name,
688
- icon,
689
- } = item
690
-
691
- Object.assign(item, {
692
- // 图标
693
- icon: icon || undefined,
694
- // 隐藏按钮
695
- hidden: $n_get(item, 'hidden') === true,
696
- // 显示按钮类型
697
- show: $n_has(item, 'data.show') ? item.data.show : true,
698
- })
699
-
700
- // 是否固定按钮
701
- item.fixed =
702
- // 非隐藏按钮
703
- ! item.hidden
704
- // 固定列
705
- && $n_get(item, 'data.fixed') === true
706
- // 单个按钮
707
- && item.show === 'single'
708
- // 按钮有图标
709
- && !! item.icon
710
-
711
- // 如果是对象
712
- if ($n_isValidObject(filterBtns)) {
713
- if ($n_has(filterBtns, name)) {
714
- newLists.push($n_merge(item, filterBtns[name]))
715
- }
716
-
717
- // 如果是数组
718
- } else if ($n_isValidArray(filterBtns)) {
719
- if ($n_indexOf(filterBtns, name) > -1) {
720
- newLists.push($n_merge(item, filterBtns[name]))
721
- }
722
-
723
- } else {
724
- newLists.push(item)
725
- }
726
- })
727
-
728
- // 转数组
729
- if (toObject) {
730
- const obj = {}
731
- for (const item of newLists) {
732
- obj[item.name] = item
733
- }
734
- return obj
735
- }
736
-
737
- return newLists
738
- }
739
-
740
- /**
741
- * 获取请求传参
742
- */
743
- function getRequestQuery(o) {
744
-
745
- // 获取按钮数据
746
- const btnData = o.powerBtn.data
747
-
748
- // 传参
749
- const query = {}
750
-
751
- // 如果有请求传参的传参设置
752
- if ($n_has(btnData, 'requestQuery.query')) {
753
- const resQuery = parseQuery(o.query, btnData.requestQuery.query)
754
- if ($n_isValidObject(resQuery)) {
755
- Object.assign(query, resQuery)
756
- }
757
- }
758
-
759
- // 获取列表数据
760
- if (
761
- // 如果按钮参数有显示类型
762
- $n_has(btnData, 'show')
763
- // 按钮参数的显示类型必须是单选或多选
764
- && $n_indexOf(['single', 'multiple'], btnData.show) > -1
765
- // 如果有请求传参的列表设置
766
- && $n_has(btnData, 'requestQuery.list')
767
- // 如果有表格数据
768
- && $n_isValidArray(o.tableSelected)
769
- ) {
770
- let newQuery = {}
771
-
772
- // 如果是单选
773
- if (btnData.show === 'single') {
774
- // 取表格选中第一条数据
775
- newQuery = o.tableSelected[0]
776
-
777
- // 否则是多选
778
- } else {
779
- // 合并表格选中的每一条数据
780
- for (const item of o.tableSelected) {
781
- $n_forIn(item, function(value, key) {
782
- if ($n_has(newQuery, key)) {
783
- newQuery[key].push(value)
784
- } else {
785
- newQuery[key] = [value]
786
- }
787
- })
788
- }
789
- }
790
-
791
- const resTable = parseQuery(newQuery, btnData.requestQuery.list)
792
- if ($n_isValidObject(resTable)) {
793
- Object.assign(query, resTable)
794
- }
795
- }
796
-
797
- return $n_cloneDeep($n_numberDeep(query))
798
- }
799
-
800
- /**
801
- * 格式化参数
802
- */
803
- function formatQuery(query, isJoinArr) {
804
-
805
- const newQuery = {}
806
-
807
- // 格式化参数
808
- $n_forIn(query, function(value, key) {
809
-
810
- // 如果是数字
811
- if ($n_isNumeric(value)) {
812
- newQuery[key] = $n_isNumber(value) ? value : Number(value)
813
-
814
- // 如果是字符串
815
- } else if ($n_isValidString(value)) {
816
- newQuery[key] = $n_trimString(value)
817
-
818
- // 如果是数组
819
- } else if ($n_isValidArray(value)) {
820
-
821
- const arr = []
822
- for (const val of value) {
823
-
824
- // 如果为有效值
825
- if ($n_isRequired(val)) {
826
-
827
- // 如果是数字
828
- if ($n_isNumeric(val)) {
829
- arr.push($n_isNumber(val) ? val : Number(val))
830
-
831
- // 如果是字符串
832
- } else if ($n_isValidString(val)) {
833
- arr.push($n_trimString(val))
834
-
835
- // 否则为数组或对象
836
- } else {
837
- arr.push(val)
838
- }
839
- }
840
- }
841
- if (arr.length) {
842
- newQuery[key] = isJoinArr ? $n_join(arr, ',') : arr
843
- }
844
- }
845
- })
846
-
847
- return newQuery
848
- }
849
-
850
- /**
851
- * 角色请求
852
- */
853
- async function request(options) {
854
-
855
- // 参数
856
- const o = Object.assign({
857
- // 权限按钮数据
858
- powerBtn: {},
859
- // 表格选中数据
860
- tableSelected: [],
861
- // 检查是否正在上传文件
862
- checkUploading: null,
863
- // 请求前执行
864
- requestBefore: null,
865
- // 请求成功执行
866
- requestSuccess: null,
867
- // 请求失败执行
868
- requestFail: null,
869
- // 请求后执行
870
- requestAfter: null,
871
- }, options)
872
-
873
- const {
874
- // 权限路由参数
875
- $route,
876
- // 当前路由参数
877
- $currentRoute,
878
- } = options
879
-
880
- o.query = $route.query
881
-
882
- // 判断类型
883
- if (! $n_get(o.powerBtn, 'data.type')) {
884
-
885
- // 【调试模式】
886
- // --------------------------------------------------
887
- // #ifdef IS_DEBUG
888
- console.log('没有定义数据类型')
889
- // #endif
890
- // --------------------------------------------------
891
-
892
- return
893
- }
894
-
895
- // 克隆按钮
896
- o.powerBtn = $n_cloneDeep(o.powerBtn)
897
-
898
- // 判断 url
899
- o.powerBtn.data.url = $n_toLower($n_trimString(o.powerBtn.data.url))
900
- if (! o.powerBtn.data.url) {
901
-
902
- if (
903
- // 如果没有跳转页面地址
904
- ! $n_has(o.powerBtn.data, 'toPage')
905
- // 或跳转页面地址为空
906
- || ! $n_isValidString(o.powerBtn.data.toPage)
907
- ) {
908
- // 【调试模式】
909
- // --------------------------------------------------
910
- // #ifdef IS_DEBUG
911
- console.log('没有定义 url')
912
- // #endif
913
- // --------------------------------------------------
914
-
915
- return
916
- }
917
-
918
- // 用跳转页面地址替换 toPage
919
- o.powerBtn.data = Object.assign({}, o.powerBtn.data, {
920
- url: o.powerBtn.data.toPage,
921
- })
922
- }
923
-
924
- // 获取按钮数据
925
- const btnData = o.powerBtn.data
926
-
927
- // 获取请求参数
928
- let query = getRequestQuery(o)
929
-
930
- // 如果是打开新窗口
931
- // --------------------------------------------------
932
- if (btnData.type === dicts.POWER_DATA_TYPE__OPEN) {
933
-
934
- query = formatQuery(query, true)
935
-
936
- // 如果按钮有标题
937
- const pageTitle = $n_trimString(o.powerBtn.title)
938
- if (pageTitle) {
939
- query.n_page_title = pageTitle
940
- }
941
-
942
- // 如果不是禁止添加来源页面参数
943
- if ($n_get(btnData, 'noFromPageQuery') !== true) {
944
- // 来源页面是当前路由的完整路径
945
- query.n_from_page = encodeURIComponent($currentRoute.fullPath)
946
- }
947
-
948
- // 请求前执行
949
- const resBefore = await $n_runAsync(o.requestBefore)({ options: o, requestData: query })
950
- if (resBefore !== void 0) {
951
- if (resBefore === false) {
952
- return
953
- }
954
- query = resBefore
955
- }
956
-
957
- $n_router.push({
958
- path: btnData.url,
959
- query,
960
- })
961
- return
962
- }
963
-
964
- // 请求数据
965
- let requestData = {}
966
-
967
- // 如果是提交表单
968
- // --------------------------------------------------
969
- if (btnData.type === dicts.POWER_DATA_TYPE__FORM) {
970
-
971
- // 获取表单注入
972
- o.$form = $n_has(options, '$form') ? options.$form : inject(NFormKey)
973
-
974
- if (! o.$form) {
975
- throw new Error('没有创建表单实例')
976
- }
977
-
978
- // 如果验证表单
979
- if ($n_get(btnData, 'validate') !== false) {
980
-
981
- if (! o.$form.formRef) {
982
- throw new Error('没有绑定 fromRef')
983
- }
984
-
985
- // 验证表单
986
- if (! await o.$form.formRef.value.validate()) {
987
- return
988
- }
989
- }
990
-
991
- // 验证表单数据
992
- if (! $n_isValidObject(o.$form.formData.value)) {
993
- throw new Error('没有获取到表单数据')
994
- }
995
-
996
- // 检查是否正在上传文件
997
- if ($n_isFunction(o.checkUploading) && o.checkUploading()) {
998
- // 轻提示
999
- $n_toast({
1000
- message: '文件上传中,请耐心等待',
1001
- })
1002
- return
1003
- }
1004
-
1005
- // 获取请求数据
1006
- requestData = $n_merge({}, formatQuery(query, false), o.$form.formData.value)
1007
-
1008
- // 如果是请求数据
1009
- // --------------------------------------------------
1010
- } else {
1011
- // 获取表格注入
1012
- o.$table = $n_has(options, '$table') ? options.$table : inject(NTableKey)
1013
-
1014
- // 获取请求数据
1015
- requestData = formatQuery(query, false)
1016
- }
1017
-
1018
- // 判断是否有确认框
1019
- const isConfirm = $n_get(btnData, 'confirm')
1020
- if (
1021
- // 如果有确认框
1022
- isConfirm
1023
- // 如果有密码确认框
1024
- || $n_get(btnData, 'confirmPassword')
1025
- ) {
1026
- // 如果需要先弹出确认框
1027
- if (isConfirm) {
1028
-
1029
- // 确认框
1030
- $n_confirm({
1031
- // 重要操作,请输入登录密码并确认后操作
1032
- message: $n_isValidString(isConfirm) ? isConfirm : '确认要执行该操作吗?',
1033
- })
1034
- // 点击确认执行
1035
- .onOk(onRequest)
1036
-
1037
- return
1038
- }
1039
- }
1040
-
1041
- // 否则执行请求
1042
- await onRequest()
1043
-
1044
- /**
1045
- * 请求事件
1046
- */
1047
- async function onRequest() {
1048
-
1049
- // 请求前执行
1050
- const resBefore = await $n_runAsync(o.requestBefore)({ options: o, requestData })
1051
- if (resBefore !== void 0) {
1052
- if (resBefore === false) {
1053
- return
1054
- }
1055
- requestData = resBefore
1056
- }
1057
-
1058
- // 请求
1059
- const res = await $n_http({
1060
- // 请求地址
1061
- url: btnData.url,
1062
- // 请求数据
1063
- data: requestData,
1064
- })
1065
-
1066
- // 返回结果数据
1067
- const resultData = Object.assign({
1068
- // 参数
1069
- options: o,
1070
- // 请求数据
1071
- requestData,
1072
- }, res)
1073
-
1074
- // 请求后执行
1075
- if (await $n_runAsync(o.requestAfter)(resultData) === false) {
1076
- return
1077
- }
1078
-
1079
- // 如果请求成功
1080
- if (res.status) {
1081
-
1082
- // 下一步
1083
- function next(isNotify = true) {
1084
-
1085
- // 轻提示
1086
- if (isNotify) {
1087
- $n_toast({
1088
- type: 'positive',
1089
- message: '恭喜您,操作成功',
1090
- })
1091
- }
1092
-
1093
- // 判断是否有请求成功后的操作动作
1094
- if ($n_has(btnData, 'requestSuccess.type')) {
1095
- switch (btnData.requestSuccess.type) {
1096
-
1097
- // 关闭当前页面
1098
- case 'close':
1099
- // 关闭窗口并跳转页面
1100
- case 'closePush':
1101
- // 关闭窗口、跳转并刷新页面
1102
- case 'closePushRefresh':
1103
-
1104
- // 如果是渲染页面
1105
- // 说明该页面在 <table-splitter> 组件内部被渲染, 则不需要关闭当前窗口
1106
- if ($n_has($route.query, 'n_renderpage') && $route.query.n_renderpage === 1) {
1107
- // 则无任何操作
1108
- return
1109
- }
1110
-
1111
- const opts = {
1112
- type: 'closeCurrentTab',
1113
- }
1114
-
1115
- if (
1116
- // 如果不是关闭当前页面, 则为关闭窗口并跳转页面
1117
- btnData.requestSuccess.type !== 'close'
1118
- // 如果有来源页面
1119
- && $n_has($route.query, 'n_from_page')
1120
- && $n_isValidString($route.query.n_from_page)
1121
- ) {
1122
- Object.assign(opts, {
1123
- // 跳转页面地址
1124
- pushPage: decodeURIComponent($route.query.n_from_page),
1125
- // 是否跳转并刷新页面
1126
- isPushRefresh: btnData.requestSuccess.type === 'closePushRefresh',
1127
- })
1128
-
1129
- // 否则如果定义了跳转页面
1130
- // else if ($n_has(btnData, 'requestSuccess.params') && $n_isValidString(btnData.requestSuccess.params)) {
1131
- // pushPage = btnData.requestSuccess.params
1132
- // }
1133
- }
1134
-
1135
- // 关闭当前标签页
1136
- $n_bus.emit('main', opts)
1137
- break
1138
-
1139
- // 重置表单
1140
- case 'resetForm':
1141
- $n_run(o.$form?.resetForm)()
1142
- break
1143
-
1144
- // 刷新列表
1145
- case 'refreshList':
1146
- $n_run(o.$table?.tableRefresh)()
1147
- break
1148
- }
1149
- }
1150
- }
1151
-
1152
- // 请求成功执行
1153
- if (await $n_runAsync(o.requestSuccess)(Object.assign({ next }, resultData)) === false) {
1154
- return
1155
- }
1156
-
1157
- // 下一步
1158
- next()
1159
-
1160
- } else {
1161
- // 请求失败执行
1162
- $n_run(o.requestFail)(resultData)
1163
- }
1164
- }
1165
- }
1166
-
1167
- /**
1168
- * 获取路由页面的角色权限
1169
- */
1170
- function getPageData($route) {
1171
-
1172
- if (! $route) {
1173
- $route = $n_router.getRoute()
1174
- }
1175
-
1176
- const path = $n_get($route, 'path')
1177
- if (! path) {
1178
- return $n_fail('路由参数错误')
1179
- }
1180
-
1181
- if (! statePower.value.v) {
1182
- return $n_fail('没有获取到权限数据')
1183
- }
1184
-
1185
- // 获取角色数据
1186
- const { urls, btns } = $n_cloneDeep(statePower.value)
1187
- if (! $n_has(urls, path)) {
1188
- return $n_fail('该页面没有权限')
1189
- }
1190
-
1191
- return $n_success({
1192
- page: urls[path],
1193
- btns: $n_has(btns, path) ? btns[path] : [],
1194
- })
1195
- }
1196
-
1197
- /**
1198
- * 权限业务
1199
- */
1200
- const $power = {
1201
- // 创建权限实例
1202
- create,
1203
- // 设置权限数据
1204
- setData,
1205
- // 获取权限数据
1206
- getData,
1207
- // 获取路由页面的角色权限
1208
- getPageData,
1209
- // 格式化权限按钮
1210
- formatBtns,
1211
- // 请求
1212
- request,
1213
- }
1214
-
1215
- export default $power
1
+ import { provide, inject, ref, computed } from 'vue'
2
+ import { useQuasar } from 'quasar'
3
+
4
+ import $n_has from 'lodash/has'
5
+ import $n_get from 'lodash/get'
6
+ import $n_merge from 'lodash/merge'
7
+ import $n_filter from 'lodash/filter'
8
+ import $n_toLower from 'lodash/toLower'
9
+ import $n_isNumber from 'lodash/isNumber'
10
+ import $n_cloneDeep from 'lodash/cloneDeep'
11
+ import $n_isFunction from 'lodash/isFunction'
12
+ import $n_pick from 'lodash/pick'
13
+
14
+ import $n_router from '@netang/utils/vue/router'
15
+
16
+ import $n_isValidArray from '@netang/utils/isValidArray'
17
+ import $n_isValidObject from '@netang/utils/isValidObject'
18
+ import $n_isValidString from '@netang/utils/isValidString'
19
+ import $n_isRequired from '@netang/utils/isRequired'
20
+ import $n_isNumeric from '@netang/utils/isNumeric'
21
+ import $n_forEach from '@netang/utils/forEach'
22
+ import $n_forIn from '@netang/utils/forIn'
23
+ import $n_slash from '@netang/utils/slash'
24
+ import $n_json from '@netang/utils/json'
25
+ import $n_join from '@netang/utils/join'
26
+ import $n_fail from '@netang/utils/fail'
27
+ import $n_success from '@netang/utils/success'
28
+ import $n_split from '@netang/utils/split'
29
+ import $n_trimString from '@netang/utils/trimString'
30
+ import $n_numberDeep from '@netang/utils/numberDeep'
31
+ import $n_indexOf from '@netang/utils/indexOf'
32
+ import $n_runAsync from '@netang/utils/runAsync'
33
+ import $n_run from '@netang/utils/run'
34
+ import $n_http from '@netang/utils/http'
35
+
36
+ import { statePower } from '../store'
37
+ import { NRenderKey, NPowerKey, NDialogKey, NFormKey, NTableKey } from './symbols'
38
+
39
+ import $n_getData from './getData'
40
+ import $n_toast from './toast'
41
+ import $n_confirm from './confirm'
42
+ import $n_bus from './bus'
43
+
44
+ import $n_config from './config'
45
+
46
+ import { configs } from './config'
47
+
48
+ const {
49
+ // 字典常量
50
+ dicts,
51
+ } = configs
52
+
53
+ /**
54
+ * 创建权限实例
55
+ */
56
+ function create(options) {
57
+
58
+ // 获取参数
59
+ const o = Object.assign({
60
+ // 路由路径
61
+ path: '',
62
+ // 路由参数
63
+ query: {},
64
+ // 是否加载页面
65
+ pageLoading: false,
66
+ // 页面状态
67
+ pageStatus: null,
68
+ // 空状态描述
69
+ emptyDescription: '',
70
+ // 是否开启权限
71
+ power: true,
72
+ // 是否显示权限按钮
73
+ showPowerBtns: true,
74
+ // 是否显示工具栏权限按钮
75
+ showToolbarPowerBtns: true,
76
+ // 格式化权限按钮
77
+ formatPowerBtns: null,
78
+ // 左边侧滑菜单图标
79
+ leftDrawerIcon: 'format_list_bulleted',
80
+ // 右边侧滑菜单图标
81
+ rightDrawerIcon: 'search',
82
+ // 请求前执行
83
+ requestBefore: null,
84
+ // 请求成功执行
85
+ requestSuccess: null,
86
+ // 请求失败执行
87
+ requestFail: null,
88
+ // 请求后执行
89
+ requestAfter: null,
90
+ }, options)
91
+
92
+ // 获取对话框渲染注入
93
+ const $dialog = inject(NDialogKey)
94
+ const hasDialog = !! $dialog
95
+
96
+ // 获取渲染注入
97
+ const $render = inject(NRenderKey)
98
+ const hasRender = !! $render
99
+
100
+ // 如果有对话框注入
101
+ if (hasDialog) {
102
+ const {
103
+ dialogProps,
104
+ } = $dialog
105
+
106
+ // 合并权限参数
107
+ Object.assign(o, $n_pick(dialogProps, [ 'path', 'query' ]))
108
+
109
+ // 合并权限参数
110
+ if ($n_has($dialog, 'props.powerProps') && $n_isValidObject($dialog.props.powerProps)) {
111
+ $n_merge(o, $dialog.props.powerProps)
112
+ }
113
+ }
114
+
115
+ // 如果有渲染注入
116
+ if (hasRender) {
117
+ // 如果有权限传参, 则合并参数
118
+ const powerProps = $n_get($render, 'props.powerProps')
119
+ if ($n_isValidObject(powerProps)) {
120
+ $n_merge(o, powerProps)
121
+ }
122
+ }
123
+
124
+ // 获取当前路由
125
+ const $currentRoute = $n_router.getRoute()
126
+
127
+ // 权限路由
128
+ let $route
129
+
130
+ // 如果没有路由
131
+ if (o.path === false) {
132
+
133
+ // 设为空路由
134
+ $route = {
135
+ fullPath: '',
136
+ path: '',
137
+ query: $n_isValidObject(o.query) ? o.query : {},
138
+ }
139
+
140
+ // 如果有自定义路径
141
+ } else if ($n_isValidString(o.path)) {
142
+
143
+ // 获取自定义路由
144
+ $route = $n_router.resolve({
145
+ path: o.path,
146
+ query: $n_isValidObject(o.query) ? o.query : {},
147
+ })
148
+
149
+ // 如果在渲染组件内 && 该渲染组件有自定义路由
150
+ } else if (hasRender && $n_has($render, '$route')) {
151
+
152
+ // 设为渲染组件的路由
153
+ $route = $render.$route
154
+
155
+ // 否则获取当前路由
156
+ } else {
157
+ $route = $currentRoute
158
+ }
159
+
160
+ // quasar 对象
161
+ const $q = useQuasar()
162
+
163
+ // 表格已选数据
164
+ const tableSelected = ref([])
165
+
166
+ // 是否显示左边侧滑菜单
167
+ const leftDrawerModelValue = ref(null)
168
+
169
+ // 是否显示右边侧滑菜单
170
+ const rightDrawerModelValue = ref(null)
171
+
172
+ /**
173
+ * 检查是否上传中
174
+ */
175
+ function checkUploading() {
176
+ for (const uploader of data.uploader) {
177
+ if (uploader.checkUploading()) {
178
+ return true
179
+ }
180
+ }
181
+ return false
182
+ }
183
+
184
+ const _data = {
185
+ // 表格实例
186
+ $table: null,
187
+ // 表单实例
188
+ $form: null,
189
+ }
190
+
191
+ // 注入数据
192
+ const data = {
193
+ // 页面加载
194
+ pageLoading: ref(o.pageLoading),
195
+ // 页面状态
196
+ pageStatus: ref(o.pageStatus),
197
+ // 空状态描述
198
+ emptyDescription: ref(o.emptyDescription),
199
+
200
+ // 当前路由全路径
201
+ routeFullPath: $route.fullPath,
202
+ // 当前路由路径
203
+ routePath: $route.path,
204
+ // 当前路由参数
205
+ routeQuery: $route.query,
206
+ // 获取当前路由
207
+ getRoute() {
208
+ return $route
209
+ },
210
+ // 格式化权限按钮
211
+ formatPowerBtns: o.formatPowerBtns,
212
+
213
+ // 左边侧滑菜单数据
214
+ leftDrawer: {
215
+ // 图标
216
+ icon: o.leftDrawerIcon,
217
+ // 是否显示
218
+ modelValue: leftDrawerModelValue,
219
+ // 是否显示切换按钮
220
+ showButton() {
221
+ return leftDrawerModelValue.value !== null
222
+ },
223
+ // 切换
224
+ toggle() {
225
+ if (leftDrawerModelValue.value !== null) {
226
+ leftDrawerModelValue.value = ! leftDrawerModelValue.value
227
+ }
228
+ },
229
+ },
230
+ // 右边侧滑菜单数据
231
+ rightDrawer: {
232
+ // 图标
233
+ icon: o.rightDrawerIcon,
234
+ // 是否显示
235
+ modelValue: rightDrawerModelValue,
236
+ // 是否显示切换按钮
237
+ showButton() {
238
+ return rightDrawerModelValue.value !== null
239
+ },
240
+ // 切换
241
+ toggle() {
242
+ if (rightDrawerModelValue.value !== null) {
243
+ rightDrawerModelValue.value = ! rightDrawerModelValue.value
244
+ }
245
+ },
246
+ },
247
+ // 表格已选数据
248
+ tableSelected,
249
+ // 上传器
250
+ uploader: [],
251
+ // 检查是否上传中
252
+ checkUploading,
253
+ }
254
+
255
+ // 如果是权限页面
256
+ if (o.power) {
257
+
258
+ // 获取当前页面角色权限
259
+ const { status, data: res } = getPageData($route)
260
+ if (! status) {
261
+ o.pageStatus = false
262
+ o.emptyDescription = res.msg
263
+ o.power = false
264
+
265
+ } else {
266
+ // 当前页面权限
267
+ data.powerPage = res.page
268
+ // 当前页面权限按钮
269
+ data.powerBtns = ref(o.showPowerBtns ? res.btns : [])
270
+ // 当前页面工具栏权限按钮
271
+ data.toolbarPowerBtns = computed(function() {
272
+
273
+ if (
274
+ // 如果显示工具栏权限按钮
275
+ o.showToolbarPowerBtns
276
+ // 有权限按钮数据
277
+ && $n_isValidArray(data.powerBtns.value)
278
+ ) {
279
+ const lists = []
280
+
281
+ // 格式化权限按钮列表
282
+ $n_forEach($n_filter(formatBtns(data.powerBtns.value), e => e.type > 2), function(item) {
283
+
284
+ if (! item.hidden) {
285
+
286
+ // 如果是单条数据显示
287
+ const isSingle = item.show === 'single'
288
+
289
+ // 如果是单条 || 多条显示
290
+ if (isSingle || item.show === 'multiple') {
291
+
292
+ // 初始为不显示
293
+ item.show = false
294
+
295
+ // 如果有表格选中数据
296
+ if ($n_isValidArray(data.tableSelected.value)) {
297
+ // 如果是单个显示
298
+ if (isSingle) {
299
+ item.show = data.tableSelected.value.length === 1
300
+
301
+ // 否则是多个显示
302
+ } else {
303
+ item.show = data.tableSelected.value.length >= 1
304
+ }
305
+ }
306
+ }
307
+
308
+ // 如果是手机模式
309
+ if ($q.platform.is.mobile) {
310
+ item.icon = undefined
311
+ }
312
+ }
313
+
314
+ if (
315
+ // 如果有格式化权限按钮方法
316
+ $n_isFunction(o.formatPowerBtns)
317
+ && o.formatPowerBtns(item, false, tableSelected.value) === false
318
+ ) {
319
+ return
320
+ }
321
+
322
+ lists.push(item)
323
+ })
324
+
325
+ return lists
326
+ }
327
+
328
+ return []
329
+ })
330
+
331
+ // 权限按钮点击
332
+ data.powerBtnClick = async function (powerBtn, tableSelected) {
333
+
334
+ // 权限请求
335
+ await request({
336
+ // 按钮数据
337
+ powerBtn,
338
+ // 权限路由参数
339
+ $route,
340
+ // 当前路由参数
341
+ $currentRoute,
342
+ // 表格选中数据
343
+ tableSelected,
344
+ // 表格实例
345
+ $table: _data.$table,
346
+ // 表单实例
347
+ $form: _data.$form,
348
+ // 检查是否正在上传文件
349
+ checkUploading,
350
+
351
+ // 请求前执行
352
+ requestBefore: o.requestBefore,
353
+ // 请求成功执行
354
+ requestSuccess: o.requestSuccess,
355
+ // 请求失败执行
356
+ requestFail: o.requestFail,
357
+ // 请求后执行
358
+ requestAfter: o.requestAfter,
359
+ })
360
+ }
361
+ }
362
+ }
363
+
364
+ // 如果没有开启权限
365
+ if (! o.power) {
366
+ // 当前页面权限
367
+ data.powerPage = {}
368
+ // 当前页面权限按钮
369
+ data.powerBtns = ref([])
370
+ // 当前页面工具栏权限按钮
371
+ data.toolbarPowerBtns = ref([])
372
+ // 权限按钮点击
373
+ data.powerBtnClick = ()=>{}
374
+ }
375
+
376
+ // 更新数据
377
+ data.update = function(cb) {
378
+ cb(data, _data)
379
+ }
380
+
381
+ // 提供可以被后代组件注入的值
382
+ provide(NPowerKey, data)
383
+
384
+ return data
385
+ }
386
+
387
+ /**
388
+ * 设置权限数据
389
+ */
390
+ function setData(data) {
391
+
392
+ // 如果没有角色数据
393
+ if (! $n_isValidObject(data)) {
394
+ return
395
+ }
396
+
397
+ const {
398
+ rows,
399
+ v,
400
+ } = data
401
+
402
+ if (! $n_isValidArray(rows) || ! v) {
403
+ return
404
+ }
405
+
406
+ // all id
407
+ const all = {}
408
+ // 页面
409
+ const urls = {}
410
+ // 按钮
411
+ const btns = {}
412
+ // 菜单
413
+ const menus = []
414
+
415
+ for (const item of rows) {
416
+
417
+ // 【格式化 start】
418
+ // --------------------------------------------------
419
+ if (item.data) {
420
+ item.data = $n_json.parse(item.data)
421
+ }
422
+ item.data = $n_isValidObject(item.data) ? $n_numberDeep(item.data) : {}
423
+
424
+ // 设置数据类型
425
+ item.data.type = item.data_type
426
+ delete item.data_type
427
+
428
+ // 标识
429
+ item.name = ''
430
+
431
+ // 如果有 url
432
+ if ($n_isValidString(item.url)) {
433
+
434
+ // url 首位加上反斜杠
435
+ item.url = $n_slash($n_toLower($n_trimString(item.url)), 'start', true)
436
+ if (item.url) {
437
+
438
+ item.data.url = item.url
439
+
440
+ // 截取最后一个反斜杠
441
+ const lastIndex = item.data.url.lastIndexOf('/')
442
+ if (lastIndex > -1) {
443
+ item.name = item.data.url.substring(lastIndex + 1)
444
+ }
445
+
446
+ } else {
447
+ item.url = ''
448
+ item.data.url = ''
449
+ }
450
+
451
+ } else {
452
+ item.url = ''
453
+ item.data.url = ''
454
+ }
455
+
456
+ // 菜单
457
+ if (item.type === 1) {
458
+ menus.push(item)
459
+
460
+ // 按钮
461
+ } else if (item.type > 2) {
462
+
463
+ // 按钮类型
464
+ switch (item.type) {
465
+
466
+ // 默认按钮
467
+ case 11:
468
+ item.color = 'default'
469
+ break;
470
+
471
+ // 主要按钮
472
+ case 12:
473
+ item.color = 'primary'
474
+ break;
475
+
476
+ // 成功按钮
477
+ case 13:
478
+ item.color = 'secondary'
479
+ break;
480
+
481
+ // 信息按钮
482
+ case 14:
483
+ item.color = 'info'
484
+ break;
485
+
486
+ // 警告按钮
487
+ case 15:
488
+ item.color = 'warning'
489
+ break;
490
+
491
+ // 危险按钮
492
+ case 16:
493
+ item.color = 'negative'
494
+ break;
495
+
496
+ // 隐藏按钮
497
+ case 20:
498
+ item.hidden = true
499
+ break;
500
+ }
501
+ }
502
+ // 【格式化 end】
503
+ // --------------------------------------------------
504
+
505
+ // url
506
+ if (item.url) {
507
+ // 添加至 all
508
+ all[item.id] = item
509
+ // 添加至页面
510
+ urls[item.url] = item
511
+ }
512
+ }
513
+
514
+ for (const item of rows) {
515
+
516
+ // 如果有跳转页面
517
+ if ($n_has(item.data, 'toPage')) {
518
+ // 设置跳转页面地址
519
+ item.data.toPage = $n_has(all, item.data.toPage) ? all[item.data.toPage].data.url : null
520
+ }
521
+
522
+ // 如果有请求成功执行类型
523
+ // else if ($n_has(item.data, 'requestSuccess.type')) {
524
+ // // 如果请求成功执行类型是关闭窗口、跳转并刷新页面
525
+ // if (item.data.requestSuccess.type === 'closePushRefresh') {
526
+ // // 设置刷新页面地址
527
+ // item.data.requestSuccess.params =
528
+ // (
529
+ // // 如果有刷新页面的参数 id
530
+ // $n_has(item.data.requestSuccess, 'params')
531
+ // // 如果有页面数据
532
+ // && $n_has(all, item.data.requestSuccess.params)
533
+ // ) ? all[item.data.requestSuccess.params].data.url : null
534
+ // }
535
+ // }
536
+
537
+ if (
538
+ // 数据/按钮
539
+ item.type > 1
540
+ // 有父级数据
541
+ && $n_has(all, item.pid)
542
+ ) {
543
+ const pItem = all[item.pid]
544
+ if ($n_has(btns, pItem.url)) {
545
+ btns[pItem.url].push(item)
546
+ } else {
547
+ btns[pItem.url] = [item]
548
+ }
549
+ }
550
+ }
551
+
552
+ // 保存至权限状态中
553
+ statePower.value = {
554
+ // 角色权限版本
555
+ v,
556
+ // all id
557
+ all,
558
+ // 页面
559
+ urls,
560
+ // 按钮
561
+ btns,
562
+ // 菜单
563
+ menus,
564
+ }
565
+ }
566
+
567
+ /**
568
+ * 获取权限
569
+ */
570
+ async function getData() {
571
+
572
+ // 如果没有权限数据
573
+ if (! statePower.value.v) {
574
+
575
+ // 获取权限数据
576
+ const res = await $n_getData($n_config('apiDataPowerName'))
577
+ if (res === false) {
578
+ statePower.value = {
579
+ // 权限版本
580
+ v: null,
581
+ // all id
582
+ all: {},
583
+ // 页面
584
+ urls: {},
585
+ // 按钮
586
+ btns: {},
587
+ // 菜单
588
+ menus: [],
589
+ }
590
+ return $n_fail()
591
+ }
592
+ }
593
+
594
+ // 如果有权限状态数据, 则直接返回
595
+ if (statePower.value.v) {
596
+ return $n_success($n_cloneDeep(statePower.value))
597
+ }
598
+
599
+ return $n_fail()
600
+ }
601
+
602
+ /**
603
+ * 解析传参
604
+ */
605
+ function parseQuery(data, settings) {
606
+
607
+ // 如果配置是字符串
608
+ if ($n_isValidString(settings)) {
609
+
610
+ // 如果返回所有传参
611
+ if (settings === 'all') {
612
+ return $n_isValidObject(data) ? data : {}
613
+ }
614
+
615
+ // 将字符串放到数组中
616
+ settings = [settings]
617
+
618
+ // 如果配置是对象
619
+ } else if ($n_isValidObject(settings)) {
620
+ settings = [settings]
621
+ }
622
+
623
+ const query = {}
624
+
625
+ // 如果配置是数组
626
+ if ($n_isValidArray(settings)) {
627
+
628
+ // 别名
629
+ const alias = {}
630
+
631
+ for (let item of settings) {
632
+ // 如果是需要的字段
633
+ if ($n_isValidString(item)) {
634
+
635
+ // 将字段转小写
636
+ item = $n_toLower($n_trimString(item))
637
+
638
+ // 判断字段是否有 as 别名
639
+ const arr = $n_split(item, ' as ')
640
+
641
+ // 如果有别名
642
+ if (arr.length === 2) {
643
+ alias[$n_trimString(arr[0])] = $n_trimString(arr[1])
644
+
645
+ // 否则别名就是当前字段本身
646
+ } else {
647
+ alias[item] = item
648
+ }
649
+
650
+ // 否则如果是自定义传参
651
+ } else if ($n_isValidObject(item)) {
652
+ Object.assign(query, item)
653
+ }
654
+ }
655
+
656
+ if (
657
+ // 如果有参数数据
658
+ $n_isValidObject(data)
659
+ // 如果有定义别名
660
+ && $n_isValidObject(alias)
661
+ ) {
662
+ $n_forIn(data, function(value, key) {
663
+
664
+ // 如果当前字段在别名中
665
+ if ($n_has(alias, key)) {
666
+ query[alias[key]] = value
667
+ }
668
+ })
669
+ }
670
+ }
671
+
672
+ return query
673
+ }
674
+
675
+ /**
676
+ * 格式化权限按钮
677
+ */
678
+ function formatBtns(powerBtns, filterBtns, toObject = false) {
679
+
680
+ const newLists = []
681
+
682
+ $n_forEach(powerBtns, function(item) {
683
+
684
+ item = $n_cloneDeep(item)
685
+
686
+ const {
687
+ name,
688
+ icon,
689
+ } = item
690
+
691
+ Object.assign(item, {
692
+ // 图标
693
+ icon: icon || undefined,
694
+ // 隐藏按钮
695
+ hidden: $n_get(item, 'hidden') === true,
696
+ // 显示按钮类型
697
+ show: $n_has(item, 'data.show') ? item.data.show : true,
698
+ })
699
+
700
+ // 是否固定按钮
701
+ item.fixed =
702
+ // 非隐藏按钮
703
+ ! item.hidden
704
+ // 固定列
705
+ && $n_get(item, 'data.fixed') === true
706
+ // 单个按钮
707
+ && item.show === 'single'
708
+ // 按钮有图标
709
+ && !! item.icon
710
+
711
+ // 如果是对象
712
+ if ($n_isValidObject(filterBtns)) {
713
+ if ($n_has(filterBtns, name)) {
714
+ newLists.push($n_merge(item, filterBtns[name]))
715
+ }
716
+
717
+ // 如果是数组
718
+ } else if ($n_isValidArray(filterBtns)) {
719
+ if ($n_indexOf(filterBtns, name) > -1) {
720
+ newLists.push($n_merge(item, filterBtns[name]))
721
+ }
722
+
723
+ } else {
724
+ newLists.push(item)
725
+ }
726
+ })
727
+
728
+ // 转数组
729
+ if (toObject) {
730
+ const obj = {}
731
+ for (const item of newLists) {
732
+ obj[item.name] = item
733
+ }
734
+ return obj
735
+ }
736
+
737
+ return newLists
738
+ }
739
+
740
+ /**
741
+ * 获取请求传参
742
+ */
743
+ function getRequestQuery(o) {
744
+
745
+ // 获取按钮数据
746
+ const btnData = o.powerBtn.data
747
+
748
+ // 传参
749
+ const query = {}
750
+
751
+ // 如果有请求传参的传参设置
752
+ if ($n_has(btnData, 'requestQuery.query')) {
753
+ const resQuery = parseQuery(o.query, btnData.requestQuery.query)
754
+ if ($n_isValidObject(resQuery)) {
755
+ Object.assign(query, resQuery)
756
+ }
757
+ }
758
+
759
+ // 获取列表数据
760
+ if (
761
+ // 如果按钮参数有显示类型
762
+ $n_has(btnData, 'show')
763
+ // 按钮参数的显示类型必须是单选或多选
764
+ && $n_indexOf(['single', 'multiple'], btnData.show) > -1
765
+ // 如果有请求传参的列表设置
766
+ && $n_has(btnData, 'requestQuery.list')
767
+ // 如果有表格数据
768
+ && $n_isValidArray(o.tableSelected)
769
+ ) {
770
+ let newQuery = {}
771
+
772
+ // 如果是单选
773
+ if (btnData.show === 'single') {
774
+ // 取表格选中第一条数据
775
+ newQuery = o.tableSelected[0]
776
+
777
+ // 否则是多选
778
+ } else {
779
+ // 合并表格选中的每一条数据
780
+ for (const item of o.tableSelected) {
781
+ $n_forIn(item, function(value, key) {
782
+ if ($n_has(newQuery, key)) {
783
+ newQuery[key].push(value)
784
+ } else {
785
+ newQuery[key] = [value]
786
+ }
787
+ })
788
+ }
789
+ }
790
+
791
+ const resTable = parseQuery(newQuery, btnData.requestQuery.list)
792
+ if ($n_isValidObject(resTable)) {
793
+ Object.assign(query, resTable)
794
+ }
795
+ }
796
+
797
+ return $n_cloneDeep($n_numberDeep(query))
798
+ }
799
+
800
+ /**
801
+ * 格式化参数
802
+ */
803
+ function formatQuery(query, isJoinArr) {
804
+
805
+ const newQuery = {}
806
+
807
+ // 格式化参数
808
+ $n_forIn(query, function(value, key) {
809
+
810
+ // 如果是数字
811
+ if ($n_isNumeric(value)) {
812
+ newQuery[key] = $n_isNumber(value) ? value : Number(value)
813
+
814
+ // 如果是字符串
815
+ } else if ($n_isValidString(value)) {
816
+ newQuery[key] = $n_trimString(value)
817
+
818
+ // 如果是数组
819
+ } else if ($n_isValidArray(value)) {
820
+
821
+ const arr = []
822
+ for (const val of value) {
823
+
824
+ // 如果为有效值
825
+ if ($n_isRequired(val)) {
826
+
827
+ // 如果是数字
828
+ if ($n_isNumeric(val)) {
829
+ arr.push($n_isNumber(val) ? val : Number(val))
830
+
831
+ // 如果是字符串
832
+ } else if ($n_isValidString(val)) {
833
+ arr.push($n_trimString(val))
834
+
835
+ // 否则为数组或对象
836
+ } else {
837
+ arr.push(val)
838
+ }
839
+ }
840
+ }
841
+ if (arr.length) {
842
+ newQuery[key] = isJoinArr ? $n_join(arr, ',') : arr
843
+ }
844
+ }
845
+ })
846
+
847
+ return newQuery
848
+ }
849
+
850
+ /**
851
+ * 角色请求
852
+ */
853
+ async function request(options) {
854
+
855
+ // 参数
856
+ const o = Object.assign({
857
+ // 权限按钮数据
858
+ powerBtn: {},
859
+ // 表格选中数据
860
+ tableSelected: [],
861
+ // 检查是否正在上传文件
862
+ checkUploading: null,
863
+ // 请求前执行
864
+ requestBefore: null,
865
+ // 请求成功执行
866
+ requestSuccess: null,
867
+ // 请求失败执行
868
+ requestFail: null,
869
+ // 请求后执行
870
+ requestAfter: null,
871
+ }, options)
872
+
873
+ const {
874
+ // 权限路由参数
875
+ $route,
876
+ // 当前路由参数
877
+ $currentRoute,
878
+ } = options
879
+
880
+ o.query = $route.query
881
+
882
+ // 判断类型
883
+ if (! $n_get(o.powerBtn, 'data.type')) {
884
+
885
+ // 【调试模式】
886
+ // --------------------------------------------------
887
+ // #ifdef IS_DEBUG
888
+ console.log('没有定义数据类型')
889
+ // #endif
890
+ // --------------------------------------------------
891
+
892
+ return
893
+ }
894
+
895
+ // 克隆按钮
896
+ o.powerBtn = $n_cloneDeep(o.powerBtn)
897
+
898
+ // 判断 url
899
+ o.powerBtn.data.url = $n_toLower($n_trimString(o.powerBtn.data.url))
900
+ if (! o.powerBtn.data.url) {
901
+
902
+ if (
903
+ // 如果没有跳转页面地址
904
+ ! $n_has(o.powerBtn.data, 'toPage')
905
+ // 或跳转页面地址为空
906
+ || ! $n_isValidString(o.powerBtn.data.toPage)
907
+ ) {
908
+ // 【调试模式】
909
+ // --------------------------------------------------
910
+ // #ifdef IS_DEBUG
911
+ console.log('没有定义 url')
912
+ // #endif
913
+ // --------------------------------------------------
914
+
915
+ return
916
+ }
917
+
918
+ // 用跳转页面地址替换 toPage
919
+ o.powerBtn.data = Object.assign({}, o.powerBtn.data, {
920
+ url: o.powerBtn.data.toPage,
921
+ })
922
+ }
923
+
924
+ // 获取按钮数据
925
+ const btnData = o.powerBtn.data
926
+
927
+ // 获取请求参数
928
+ let query = getRequestQuery(o)
929
+
930
+ // 如果是打开新窗口
931
+ // --------------------------------------------------
932
+ if (btnData.type === dicts.POWER_DATA_TYPE__OPEN) {
933
+
934
+ query = formatQuery(query, true)
935
+
936
+ // 如果按钮有标题
937
+ const pageTitle = $n_trimString(o.powerBtn.title)
938
+ if (pageTitle) {
939
+ query.n_page_title = pageTitle
940
+ }
941
+
942
+ // 如果不是禁止添加来源页面参数
943
+ if ($n_get(btnData, 'noFromPageQuery') !== true) {
944
+ // 来源页面是当前路由的完整路径
945
+ query.n_from_page = encodeURIComponent($currentRoute.fullPath)
946
+ }
947
+
948
+ // 请求前执行
949
+ const resBefore = await $n_runAsync(o.requestBefore)({ options: o, requestData: query })
950
+ if (resBefore !== void 0) {
951
+ if (resBefore === false) {
952
+ return
953
+ }
954
+ query = resBefore
955
+ }
956
+
957
+ $n_router.push({
958
+ path: btnData.url,
959
+ query,
960
+ })
961
+ return
962
+ }
963
+
964
+ // 请求数据
965
+ let requestData = {}
966
+
967
+ // 如果是提交表单
968
+ // --------------------------------------------------
969
+ if (btnData.type === dicts.POWER_DATA_TYPE__FORM) {
970
+
971
+ // 获取表单注入
972
+ o.$form = $n_has(options, '$form') ? options.$form : inject(NFormKey)
973
+
974
+ if (! o.$form) {
975
+ throw new Error('没有创建表单实例')
976
+ }
977
+
978
+ // 如果验证表单
979
+ if ($n_get(btnData, 'validate') !== false) {
980
+
981
+ if (! o.$form.formRef) {
982
+ throw new Error('没有绑定 fromRef')
983
+ }
984
+
985
+ // 验证表单
986
+ if (! await o.$form.formRef.value.validate()) {
987
+ return
988
+ }
989
+ }
990
+
991
+ // 验证表单数据
992
+ if (! $n_isValidObject(o.$form.formData.value)) {
993
+ throw new Error('没有获取到表单数据')
994
+ }
995
+
996
+ // 检查是否正在上传文件
997
+ if ($n_isFunction(o.checkUploading) && o.checkUploading()) {
998
+ // 轻提示
999
+ $n_toast({
1000
+ message: '文件上传中,请耐心等待',
1001
+ })
1002
+ return
1003
+ }
1004
+
1005
+ // 获取请求数据
1006
+ requestData = $n_merge({}, formatQuery(query, false), o.$form.formData.value)
1007
+
1008
+ // 如果是请求数据
1009
+ // --------------------------------------------------
1010
+ } else {
1011
+ // 获取表格注入
1012
+ o.$table = $n_has(options, '$table') ? options.$table : inject(NTableKey)
1013
+
1014
+ // 获取请求数据
1015
+ requestData = formatQuery(query, false)
1016
+ }
1017
+
1018
+ // 判断是否有确认框
1019
+ const isConfirm = $n_get(btnData, 'confirm')
1020
+ if (
1021
+ // 如果有确认框
1022
+ isConfirm
1023
+ // 如果有密码确认框
1024
+ || $n_get(btnData, 'confirmPassword')
1025
+ ) {
1026
+ // 如果需要先弹出确认框
1027
+ if (isConfirm) {
1028
+
1029
+ // 确认框
1030
+ $n_confirm({
1031
+ // 重要操作,请输入登录密码并确认后操作
1032
+ message: $n_isValidString(isConfirm) ? isConfirm : '确认要执行该操作吗?',
1033
+ })
1034
+ // 点击确认执行
1035
+ .onOk(onRequest)
1036
+
1037
+ return
1038
+ }
1039
+ }
1040
+
1041
+ // 否则执行请求
1042
+ await onRequest()
1043
+
1044
+ /**
1045
+ * 请求事件
1046
+ */
1047
+ async function onRequest() {
1048
+
1049
+ // 请求前执行
1050
+ const resBefore = await $n_runAsync(o.requestBefore)({ options: o, requestData })
1051
+ if (resBefore !== void 0) {
1052
+ if (resBefore === false) {
1053
+ return
1054
+ }
1055
+ requestData = resBefore
1056
+ }
1057
+
1058
+ // 请求
1059
+ const res = await $n_http({
1060
+ // 请求地址
1061
+ url: btnData.url,
1062
+ // 请求数据
1063
+ data: requestData,
1064
+ })
1065
+
1066
+ // 返回结果数据
1067
+ const resultData = Object.assign({
1068
+ // 参数
1069
+ options: o,
1070
+ // 请求数据
1071
+ requestData,
1072
+ }, res)
1073
+
1074
+ // 请求后执行
1075
+ if (await $n_runAsync(o.requestAfter)(resultData) === false) {
1076
+ return
1077
+ }
1078
+
1079
+ // 如果请求成功
1080
+ if (res.status) {
1081
+
1082
+ // 下一步
1083
+ function next(isNotify = true) {
1084
+
1085
+ // 轻提示
1086
+ if (isNotify) {
1087
+ $n_toast({
1088
+ type: 'positive',
1089
+ message: '恭喜您,操作成功',
1090
+ })
1091
+ }
1092
+
1093
+ // 判断是否有请求成功后的操作动作
1094
+ if ($n_has(btnData, 'requestSuccess.type')) {
1095
+ switch (btnData.requestSuccess.type) {
1096
+
1097
+ // 关闭当前页面
1098
+ case 'close':
1099
+ // 关闭窗口并跳转页面
1100
+ case 'closePush':
1101
+ // 关闭窗口、跳转并刷新页面
1102
+ case 'closePushRefresh':
1103
+
1104
+ // 如果是渲染页面
1105
+ // 说明该页面在 <table-splitter> 组件内部被渲染, 则不需要关闭当前窗口
1106
+ if ($n_has($route.query, 'n_renderpage') && $route.query.n_renderpage === 1) {
1107
+ // 则无任何操作
1108
+ return
1109
+ }
1110
+
1111
+ const opts = {
1112
+ type: 'closeCurrentTab',
1113
+ }
1114
+
1115
+ if (
1116
+ // 如果不是关闭当前页面, 则为关闭窗口并跳转页面
1117
+ btnData.requestSuccess.type !== 'close'
1118
+ // 如果有来源页面
1119
+ && $n_has($route.query, 'n_from_page')
1120
+ && $n_isValidString($route.query.n_from_page)
1121
+ ) {
1122
+ Object.assign(opts, {
1123
+ // 跳转页面地址
1124
+ pushPage: decodeURIComponent($route.query.n_from_page),
1125
+ // 是否跳转并刷新页面
1126
+ isPushRefresh: btnData.requestSuccess.type === 'closePushRefresh',
1127
+ })
1128
+
1129
+ // 否则如果定义了跳转页面
1130
+ // else if ($n_has(btnData, 'requestSuccess.params') && $n_isValidString(btnData.requestSuccess.params)) {
1131
+ // pushPage = btnData.requestSuccess.params
1132
+ // }
1133
+ }
1134
+
1135
+ // 关闭当前标签页
1136
+ $n_bus.emit('main', opts)
1137
+ break
1138
+
1139
+ // 重置表单
1140
+ case 'resetForm':
1141
+ $n_run(o.$form?.resetForm)()
1142
+ break
1143
+
1144
+ // 刷新列表
1145
+ case 'refreshList':
1146
+ $n_run(o.$table?.tableRefresh)()
1147
+ break
1148
+ }
1149
+ }
1150
+ }
1151
+
1152
+ // 请求成功执行
1153
+ if (await $n_runAsync(o.requestSuccess)(Object.assign({ next }, resultData)) === false) {
1154
+ return
1155
+ }
1156
+
1157
+ // 下一步
1158
+ next()
1159
+
1160
+ } else {
1161
+ // 请求失败执行
1162
+ $n_run(o.requestFail)(resultData)
1163
+ }
1164
+ }
1165
+ }
1166
+
1167
+ /**
1168
+ * 获取路由页面的角色权限
1169
+ */
1170
+ function getPageData($route) {
1171
+
1172
+ if (! $route) {
1173
+ $route = $n_router.getRoute()
1174
+ }
1175
+
1176
+ const path = $n_get($route, 'path')
1177
+ if (! path) {
1178
+ return $n_fail('路由参数错误')
1179
+ }
1180
+
1181
+ if (! statePower.value.v) {
1182
+ return $n_fail('没有获取到权限数据')
1183
+ }
1184
+
1185
+ // 获取角色数据
1186
+ const { urls, btns } = $n_cloneDeep(statePower.value)
1187
+ if (! $n_has(urls, path)) {
1188
+ return $n_fail('该页面没有权限')
1189
+ }
1190
+
1191
+ return $n_success({
1192
+ page: urls[path],
1193
+ btns: $n_has(btns, path) ? btns[path] : [],
1194
+ })
1195
+ }
1196
+
1197
+ /**
1198
+ * 权限业务
1199
+ */
1200
+ const $power = {
1201
+ // 创建权限实例
1202
+ create,
1203
+ // 设置权限数据
1204
+ setData,
1205
+ // 获取权限数据
1206
+ getData,
1207
+ // 获取路由页面的角色权限
1208
+ getPageData,
1209
+ // 格式化权限按钮
1210
+ formatBtns,
1211
+ // 请求
1212
+ request,
1213
+ }
1214
+
1215
+ export default $power