@netang/quasar 0.1.21 → 0.1.23

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 (219) hide show
  1. package/.editorconfig +12 -12
  2. package/_docs/docs/components/field-table.md +58 -58
  3. package/_docs/docs/components/field-tree.md +21 -21
  4. package/_docs/docs/components/table.md +24 -24
  5. package/_docs/docs/utils/table.md +196 -196
  6. package/components/editor-code/index.vue +325 -325
  7. package/components/field-table/index.vue +1227 -1227
  8. package/components/field-tree/index.vue +757 -759
  9. package/components/private/components/move-to-tree/index.vue +2 -2
  10. package/components/private/edit-power-data/index.vue +846 -846
  11. package/components/private/table-visible-columns-button/index.vue +109 -109
  12. package/components/render/index.vue +127 -127
  13. package/components/search-item/index.vue +212 -212
  14. package/components/select/index.vue +177 -177
  15. package/components/table/index.vue +471 -471
  16. package/components/table-splitter/index.vue +377 -377
  17. package/components/table-summary/index.vue +107 -107
  18. package/components/tree/index.vue +1636 -0
  19. package/docs/404.html +33 -33
  20. package/docs/assets/404.html-60b35caa.js +1 -1
  21. package/docs/assets/404.html-d1e63d77.js +1 -1
  22. package/docs/assets/alert.html-b2a2a72f.js +5 -5
  23. package/docs/assets/alert.html-ba46d137.js +1 -1
  24. package/docs/assets/app-9f30aa4b.js +6 -6
  25. package/docs/assets/area.html-01b9b58d.js +42 -42
  26. package/docs/assets/area.html-9a4fce6a.js +1 -1
  27. package/docs/assets/arr.html-145d27e7.js +1 -1
  28. package/docs/assets/arr.html-674e65ab.js +11 -11
  29. package/docs/assets/auth.html-579fa830.js +1 -1
  30. package/docs/assets/auth.html-8544ed95.js +8 -8
  31. package/docs/assets/bus.html-c71254aa.js +1 -1
  32. package/docs/assets/bus.html-dc7d3d19.js +6 -6
  33. package/docs/assets/column-title.html-c735cb5a.js +3 -3
  34. package/docs/assets/column-title.html-e9316762.js +1 -1
  35. package/docs/assets/confirm.html-ddfdc27f.js +10 -10
  36. package/docs/assets/confirm.html-ef3e2bef.js +1 -1
  37. package/docs/assets/copy.html-d20345b6.js +1 -1
  38. package/docs/assets/copy.html-ef8c8571.js +13 -13
  39. package/docs/assets/data.html-6432175d.js +30 -30
  40. package/docs/assets/data.html-a3b05d5b.js +1 -1
  41. package/docs/assets/dialog.html-1f698e5a.js +1 -1
  42. package/docs/assets/dialog.html-62902b83.js +68 -68
  43. package/docs/assets/dialog.html-baea77c9.js +1 -1
  44. package/docs/assets/dialog.html-bb082fc4.js +1 -1
  45. package/docs/assets/dict.html-1311da3d.js +23 -23
  46. package/docs/assets/dict.html-b96fbf0c.js +1 -1
  47. package/docs/assets/dictOptions.html-7c4f40a5.js +1 -1
  48. package/docs/assets/dictOptions.html-fb99d175.js +5 -5
  49. package/docs/assets/dragger.html-668d3efa.js +1 -1
  50. package/docs/assets/dragger.html-749d585a.js +1 -1
  51. package/docs/assets/editor-code.html-6ab26ea9.js +1 -1
  52. package/docs/assets/editor-code.html-d196205d.js +1 -1
  53. package/docs/assets/empty.html-1c139131.js +1 -1
  54. package/docs/assets/empty.html-1e9c441d.js +1 -1
  55. package/docs/assets/field-date.html-069fdb13.js +1 -1
  56. package/docs/assets/field-date.html-ad204aa9.js +1 -1
  57. package/docs/assets/field-table.html-ce480f03.js +1 -1
  58. package/docs/assets/field-table.html-d9236160.js +1 -1
  59. package/docs/assets/field-text.html-7277c62f.js +1 -1
  60. package/docs/assets/field-text.html-ccb4cecf.js +1 -1
  61. package/docs/assets/field-tree.html-519bfb45.js +1 -1
  62. package/docs/assets/field-tree.html-fdc748d6.js +1 -1
  63. package/docs/assets/form.html-2b562c37.js +2 -2
  64. package/docs/assets/form.html-75104cd5.js +1 -1
  65. package/docs/assets/framework-204010b2.js +5 -5
  66. package/docs/assets/getData.html-990e3787.js +1 -1
  67. package/docs/assets/getData.html-bb72025f.js +34 -34
  68. package/docs/assets/getFile.html-42368004.js +1 -1
  69. package/docs/assets/getFile.html-99abd054.js +3 -3
  70. package/docs/assets/getImage.html-3429c5a1.js +1 -1
  71. package/docs/assets/getImage.html-4d886d83.js +3 -3
  72. package/docs/assets/getTime.html-7435f922.js +1 -1
  73. package/docs/assets/getTime.html-b37f49eb.js +20 -20
  74. package/docs/assets/img.html-7d1da657.js +1 -1
  75. package/docs/assets/img.html-fbea1105.js +1 -1
  76. package/docs/assets/index.html-1695dd7c.js +1 -1
  77. package/docs/assets/index.html-65a4aa67.js +1 -1
  78. package/docs/assets/index.html-7b98d5bd.js +1 -1
  79. package/docs/assets/index.html-c01f2648.js +1 -1
  80. package/docs/assets/input-number.html-0b250d2a.js +1 -1
  81. package/docs/assets/input-number.html-a8eb0378.js +1 -1
  82. package/docs/assets/list-menu-item.html-7f1b4611.js +1 -1
  83. package/docs/assets/list-menu-item.html-84ed5ab8.js +1 -1
  84. package/docs/assets/list-menu.html-28b4163f.js +1 -1
  85. package/docs/assets/list-menu.html-cb6ba95b.js +1 -1
  86. package/docs/assets/loading.html-dae9e39d.js +6 -6
  87. package/docs/assets/loading.html-dc74c9e6.js +1 -1
  88. package/docs/assets/notify.html-e6c4c514.js +1 -1
  89. package/docs/assets/notify.html-f2c4d914.js +8 -8
  90. package/docs/assets/power-page.html-32e02f82.js +1 -1
  91. package/docs/assets/power-page.html-485e77da.js +1 -1
  92. package/docs/assets/power.html-d258cc19.js +93 -93
  93. package/docs/assets/power.html-e490bd32.js +1 -1
  94. package/docs/assets/previewImage.html-6a6b4245.js +1 -1
  95. package/docs/assets/previewImage.html-c5b7e945.js +2 -2
  96. package/docs/assets/price.html-1882c548.js +19 -19
  97. package/docs/assets/price.html-94d3f5be.js +1 -1
  98. package/docs/assets/price.html-d213df0f.js +1 -1
  99. package/docs/assets/price.html-deaf880f.js +1 -1
  100. package/docs/assets/render.html-8efcbdd4.js +1 -1
  101. package/docs/assets/render.html-df228e38.js +1 -1
  102. package/docs/assets/rule.html-2cd57fc2.js +13 -13
  103. package/docs/assets/rule.html-61662001.js +1 -1
  104. package/docs/assets/ruleValid.html-04fe2552.js +1 -1
  105. package/docs/assets/ruleValid.html-e0a776af.js +14 -14
  106. package/docs/assets/search-0782d0d1.svg +1 -1
  107. package/docs/assets/search-item.html-3f75394c.js +1 -1
  108. package/docs/assets/search-item.html-4e942ecd.js +1 -1
  109. package/docs/assets/search.html-2807043e.js +1 -1
  110. package/docs/assets/search.html-c24f8806.js +1 -1
  111. package/docs/assets/select.html-00d0607c.js +1 -1
  112. package/docs/assets/select.html-de7731f5.js +1 -1
  113. package/docs/assets/splitter.html-56f51a70.js +1 -1
  114. package/docs/assets/splitter.html-f5c836d7.js +1 -1
  115. package/docs/assets/style-161e43ab.css +1 -1
  116. package/docs/assets/symbols.html-a6aea4bf.js +1 -1
  117. package/docs/assets/symbols.html-b1f65bad.js +21 -21
  118. package/docs/assets/table-column-fixed.html-3a69e7b2.js +1 -1
  119. package/docs/assets/table-column-fixed.html-e763c38b.js +1 -1
  120. package/docs/assets/table-pagination.html-236934d3.js +1 -1
  121. package/docs/assets/table-pagination.html-c37ee2ac.js +1 -1
  122. package/docs/assets/table-splitter.html-07eab15c.js +1 -1
  123. package/docs/assets/table-splitter.html-7670ee65.js +1 -1
  124. package/docs/assets/table-summary.html-04db434f.js +1 -1
  125. package/docs/assets/table-summary.html-943c65a0.js +1 -1
  126. package/docs/assets/table.html-36253ad7.js +1 -1
  127. package/docs/assets/table.html-7f9c5d1b.js +38 -38
  128. package/docs/assets/table.html-93d53dc8.js +1 -1
  129. package/docs/assets/table.html-ac99b9cb.js +1 -1
  130. package/docs/assets/thumbnail.html-bab1976b.js +1 -1
  131. package/docs/assets/thumbnail.html-eb64e5e8.js +1 -1
  132. package/docs/assets/timestamp.html-4e54f79b.js +13 -13
  133. package/docs/assets/timestamp.html-d0e1b88a.js +1 -1
  134. package/docs/assets/toast.html-58ecbe21.js +1 -1
  135. package/docs/assets/toast.html-c9b9d36b.js +6 -6
  136. package/docs/assets/toolbar.html-83d9f97c.js +1 -1
  137. package/docs/assets/toolbar.html-ff7b8c92.js +1 -1
  138. package/docs/assets/tree.html-d07cbe79.js +23 -23
  139. package/docs/assets/tree.html-ea04193e.js +1 -1
  140. package/docs/assets/uploader-query.html-05590718.js +1 -1
  141. package/docs/assets/uploader-query.html-3175bac5.js +1 -1
  142. package/docs/assets/uploader.html-36da4394.js +2 -2
  143. package/docs/assets/uploader.html-6b5f3079.js +1 -1
  144. package/docs/assets/uploader.html-b9340b57.js +1 -1
  145. package/docs/assets/uploader.html-bc1c22e3.js +1 -1
  146. package/docs/assets/value-format.html-8ae3d47d.js +1 -1
  147. package/docs/assets/value-format.html-afa99b3d.js +1 -1
  148. package/docs/components/column-title.html +35 -35
  149. package/docs/components/data.html +62 -62
  150. package/docs/components/dialog.html +33 -33
  151. package/docs/components/dragger.html +33 -33
  152. package/docs/components/editor-code.html +33 -33
  153. package/docs/components/empty.html +33 -33
  154. package/docs/components/field-date.html +33 -33
  155. package/docs/components/field-table.html +33 -33
  156. package/docs/components/field-text.html +33 -33
  157. package/docs/components/field-tree.html +33 -33
  158. package/docs/components/img.html +33 -33
  159. package/docs/components/input-number.html +33 -33
  160. package/docs/components/list-menu-item.html +33 -33
  161. package/docs/components/list-menu.html +33 -33
  162. package/docs/components/power-page.html +33 -33
  163. package/docs/components/price.html +33 -33
  164. package/docs/components/render.html +33 -33
  165. package/docs/components/search-item.html +33 -33
  166. package/docs/components/search.html +33 -33
  167. package/docs/components/select.html +33 -33
  168. package/docs/components/splitter.html +33 -33
  169. package/docs/components/table-column-fixed.html +33 -33
  170. package/docs/components/table-pagination.html +33 -33
  171. package/docs/components/table-splitter.html +33 -33
  172. package/docs/components/table-summary.html +33 -33
  173. package/docs/components/table.html +33 -33
  174. package/docs/components/thumbnail.html +33 -33
  175. package/docs/components/toolbar.html +33 -33
  176. package/docs/components/uploader-query.html +33 -33
  177. package/docs/components/uploader.html +33 -33
  178. package/docs/components/value-format.html +33 -33
  179. package/docs/index.html +33 -33
  180. package/docs/utils/alert.html +37 -37
  181. package/docs/utils/area.html +74 -74
  182. package/docs/utils/arr.html +43 -43
  183. package/docs/utils/auth.html +40 -40
  184. package/docs/utils/bus.html +38 -38
  185. package/docs/utils/confirm.html +42 -42
  186. package/docs/utils/copy.html +45 -45
  187. package/docs/utils/dialog.html +100 -100
  188. package/docs/utils/dict.html +55 -55
  189. package/docs/utils/dictOptions.html +37 -37
  190. package/docs/utils/form.html +34 -34
  191. package/docs/utils/getData.html +66 -66
  192. package/docs/utils/getFile.html +35 -35
  193. package/docs/utils/getImage.html +35 -35
  194. package/docs/utils/getTime.html +52 -52
  195. package/docs/utils/index.html +33 -33
  196. package/docs/utils/loading.html +38 -38
  197. package/docs/utils/notify.html +40 -40
  198. package/docs/utils/power.html +125 -125
  199. package/docs/utils/previewImage.html +34 -34
  200. package/docs/utils/price.html +51 -51
  201. package/docs/utils/rule.html +45 -45
  202. package/docs/utils/ruleValid.html +46 -46
  203. package/docs/utils/symbols.html +53 -53
  204. package/docs/utils/table.html +70 -70
  205. package/docs/utils/timestamp.html +45 -45
  206. package/docs/utils/toast.html +38 -38
  207. package/docs/utils/tree.html +55 -55
  208. package/docs/utils/uploader.html +34 -34
  209. package/package.json +1 -1
  210. package/sass/common.scss +179 -179
  211. package/sass/variables.scss +138 -138
  212. package/utils/$form.js +68 -68
  213. package/utils/$power.js +1225 -1225
  214. package/utils/$render.js +75 -75
  215. package/utils/$table.js +1171 -1171
  216. package/utils/$tree.js +655 -712
  217. package/utils/index.js +62 -62
  218. package/utils/timestamp.js +18 -18
  219. package/utils/useSearch.js +496 -496
package/utils/$power.js CHANGED
@@ -1,1225 +1,1225 @@
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 = $n_has(options, '$dialog') ? options.$dialog : inject(NDialogKey)
94
- const hasDialog = !! $dialog
95
-
96
- // 获取渲染注入
97
- const $render = $n_has(options, '$render') ? options.$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, 'getRoute')) {
151
-
152
- // 设为渲染组件的路由
153
- $route = $render.getRoute()
154
-
155
- // 否则获取当前路由
156
- } else {
157
-
158
- $route = $currentRoute
159
- }
160
-
161
- // quasar 对象
162
- const $q = useQuasar()
163
-
164
- // 表格已选数据
165
- const tableSelected = ref([])
166
-
167
- // 是否显示左边侧滑菜单
168
- const leftDrawerModelValue = ref(null)
169
-
170
- // 是否显示右边侧滑菜单
171
- const rightDrawerModelValue = ref(null)
172
-
173
- /**
174
- * 检查是否上传中
175
- */
176
- function checkUploading() {
177
- for (const uploader of data.uploader) {
178
- if (uploader.checkUploading()) {
179
- return true
180
- }
181
- }
182
- return false
183
- }
184
-
185
- const _data = {
186
- // 表格实例
187
- $table: null,
188
- // 表单实例
189
- $form: null,
190
- }
191
-
192
- // 注入数据
193
- const data = {
194
- // 页面加载
195
- pageLoading: ref(o.pageLoading),
196
- // 页面状态
197
- pageStatus: ref(o.pageStatus),
198
- // 空状态描述
199
- emptyDescription: ref(o.emptyDescription),
200
-
201
- // 当前路由全路径
202
- routeFullPath: $route.fullPath,
203
- // 当前路由路径
204
- routePath: $route.path,
205
- // 当前路由参数
206
- routeQuery: $route.query,
207
- // 获取当前路由
208
- getRoute() {
209
- return $route
210
- },
211
- // 格式化权限按钮
212
- formatPowerBtns: o.formatPowerBtns,
213
-
214
- // 左边侧滑菜单数据
215
- leftDrawer: {
216
- // 图标
217
- icon: o.leftDrawerIcon,
218
- // 是否显示
219
- modelValue: leftDrawerModelValue,
220
- // 是否显示切换按钮
221
- showButton() {
222
- return leftDrawerModelValue.value !== null
223
- },
224
- // 切换
225
- toggle() {
226
- if (leftDrawerModelValue.value !== null) {
227
- leftDrawerModelValue.value = ! leftDrawerModelValue.value
228
- }
229
- },
230
- },
231
- // 右边侧滑菜单数据
232
- rightDrawer: {
233
- // 图标
234
- icon: o.rightDrawerIcon,
235
- // 是否显示
236
- modelValue: rightDrawerModelValue,
237
- // 是否显示切换按钮
238
- showButton() {
239
- return rightDrawerModelValue.value !== null
240
- },
241
- // 切换
242
- toggle() {
243
- if (rightDrawerModelValue.value !== null) {
244
- rightDrawerModelValue.value = ! rightDrawerModelValue.value
245
- }
246
- },
247
- },
248
- // 表格已选数据
249
- tableSelected,
250
- // 上传器
251
- uploader: [],
252
- // 检查是否上传中
253
- checkUploading,
254
- }
255
-
256
- // 如果是权限页面
257
- if (o.power) {
258
-
259
- // 获取当前页面角色权限
260
- const { status, data: res } = getPageData($route)
261
- if (! status) {
262
- o.pageStatus = false
263
- o.emptyDescription = res.msg
264
- o.power = false
265
-
266
- } else {
267
- // 当前页面权限
268
- data.powerPage = res.page
269
- // 当前页面权限按钮
270
- data.powerBtns = ref(o.showPowerBtns ? res.btns : [])
271
- // 当前页面工具栏权限按钮
272
- data.toolbarPowerBtns = computed(function() {
273
-
274
- if (
275
- // 如果显示工具栏权限按钮
276
- o.showToolbarPowerBtns
277
- // 有权限按钮数据
278
- && $n_isValidArray(data.powerBtns.value)
279
- ) {
280
- const lists = []
281
-
282
- // 格式化权限按钮列表
283
- $n_forEach($n_filter(formatBtns(data.powerBtns.value), e => e.type > 2), function(item) {
284
-
285
- if (! item.hidden) {
286
-
287
- // 如果是单条数据显示
288
- const isSingle = item.show === 'single'
289
-
290
- // 如果是单条 || 多条显示
291
- if (isSingle || item.show === 'multiple') {
292
-
293
- // 初始为不显示
294
- item.show = false
295
-
296
- // 如果有表格选中数据
297
- if ($n_isValidArray(data.tableSelected.value)) {
298
- // 如果是单个显示
299
- if (isSingle) {
300
- item.show = data.tableSelected.value.length === 1
301
-
302
- // 否则是多个显示
303
- } else {
304
- item.show = data.tableSelected.value.length >= 1
305
- }
306
- }
307
- }
308
-
309
- // 如果是手机模式
310
- if ($q.platform.is.mobile) {
311
- item.icon = undefined
312
- }
313
- }
314
-
315
- if (
316
- // 如果有格式化权限按钮方法
317
- $n_isFunction(o.formatPowerBtns)
318
- && o.formatPowerBtns(item, false, tableSelected.value) === false
319
- ) {
320
- return
321
- }
322
-
323
- lists.push(item)
324
- })
325
-
326
- return lists
327
- }
328
-
329
- return []
330
- })
331
-
332
- // 权限按钮点击
333
- data.powerBtnClick = async function (powerBtn, tableSelected) {
334
-
335
- // 权限请求
336
- await request({
337
- // power
338
- $power: data,
339
- // 按钮数据
340
- powerBtn,
341
- // 权限路由参数
342
- $route,
343
- // 当前路由参数
344
- $currentRoute,
345
- // 表格选中数据
346
- tableSelected,
347
- // 表格实例
348
- $table: _data.$table,
349
- // 表单实例
350
- $form: _data.$form,
351
- // 检查是否正在上传文件
352
- checkUploading,
353
-
354
- // 请求前执行
355
- requestBefore: o.requestBefore,
356
- // 请求成功执行
357
- requestSuccess: o.requestSuccess,
358
- // 请求失败执行
359
- requestFail: o.requestFail,
360
- // 请求后执行
361
- requestAfter: o.requestAfter,
362
- })
363
- }
364
- }
365
- }
366
-
367
- // 如果没有开启权限
368
- if (! o.power) {
369
- // 当前页面权限
370
- data.powerPage = {}
371
- // 当前页面权限按钮
372
- data.powerBtns = ref([])
373
- // 当前页面工具栏权限按钮
374
- data.toolbarPowerBtns = ref([])
375
- // 权限按钮点击
376
- data.powerBtnClick = ()=>{}
377
- }
378
-
379
- // 更新数据
380
- data.update = function(cb) {
381
- cb(data, _data)
382
- }
383
-
384
- // 提供可以被后代组件注入的值
385
- provide(NPowerKey, data)
386
-
387
- return data
388
- }
389
-
390
- /**
391
- * 设置权限数据
392
- */
393
- function setData(data) {
394
-
395
- // 如果没有角色数据
396
- if (! $n_isValidObject(data)) {
397
- return
398
- }
399
-
400
- const {
401
- rows,
402
- v,
403
- } = data
404
-
405
- if (! $n_isValidArray(rows) || ! v) {
406
- return
407
- }
408
-
409
- // all id
410
- const all = {}
411
- // 页面
412
- const urls = {}
413
- // 按钮
414
- const btns = {}
415
- // 菜单
416
- const menus = []
417
-
418
- for (const item of rows) {
419
-
420
- // 【格式化 start】
421
- // --------------------------------------------------
422
- if (item.data) {
423
- item.data = $n_json.parse(item.data)
424
- }
425
- item.data = $n_isValidObject(item.data) ? $n_numberDeep(item.data) : {}
426
-
427
- // 设置数据类型
428
- item.data.type = item.data_type
429
- delete item.data_type
430
-
431
- // 标识
432
- item.name = ''
433
-
434
- // 如果有 url
435
- if ($n_isValidString(item.url)) {
436
-
437
- // url 首位加上反斜杠
438
- item.url = $n_slash($n_toLower($n_trimString(item.url)), 'start', true)
439
- if (item.url) {
440
-
441
- item.data.url = item.url
442
-
443
- // 截取最后一个反斜杠
444
- const lastIndex = item.data.url.lastIndexOf('/')
445
- if (lastIndex > -1) {
446
- item.name = item.data.url.substring(lastIndex + 1)
447
- }
448
-
449
- } else {
450
- item.url = ''
451
- item.data.url = ''
452
- }
453
-
454
- } else {
455
- item.url = ''
456
- item.data.url = ''
457
- }
458
-
459
- // 菜单
460
- if (item.type === 1) {
461
- menus.push(item)
462
-
463
- // 按钮
464
- } else if (item.type > 2) {
465
-
466
- // 按钮类型
467
- switch (item.type) {
468
-
469
- // 默认按钮
470
- case 11:
471
- item.color = 'default'
472
- break;
473
-
474
- // 主要按钮
475
- case 12:
476
- item.color = 'primary'
477
- break;
478
-
479
- // 成功按钮
480
- case 13:
481
- item.color = 'secondary'
482
- break;
483
-
484
- // 信息按钮
485
- case 14:
486
- item.color = 'info'
487
- break;
488
-
489
- // 警告按钮
490
- case 15:
491
- item.color = 'warning'
492
- break;
493
-
494
- // 危险按钮
495
- case 16:
496
- item.color = 'negative'
497
- break;
498
-
499
- // 隐藏按钮
500
- case 20:
501
- item.hidden = true
502
- break;
503
- }
504
- }
505
- // 【格式化 end】
506
- // --------------------------------------------------
507
-
508
- // url
509
- if (item.url) {
510
- // 添加至 all
511
- all[item.id] = item
512
- // 添加至页面
513
- urls[item.url] = item
514
- }
515
- }
516
-
517
- for (const item of rows) {
518
-
519
- // 如果有跳转页面
520
- if ($n_has(item.data, 'toPage')) {
521
- // 设置跳转页面地址
522
- item.data.toPage = $n_has(all, item.data.toPage) ? all[item.data.toPage].data.url : null
523
- }
524
-
525
- // 如果有请求成功执行类型
526
- // else if ($n_has(item.data, 'requestSuccess.type')) {
527
- // // 如果请求成功执行类型是关闭窗口、跳转并刷新页面
528
- // if (item.data.requestSuccess.type === 'closePushRefresh') {
529
- // // 设置刷新页面地址
530
- // item.data.requestSuccess.params =
531
- // (
532
- // // 如果有刷新页面的参数 id
533
- // $n_has(item.data.requestSuccess, 'params')
534
- // // 如果有页面数据
535
- // && $n_has(all, item.data.requestSuccess.params)
536
- // ) ? all[item.data.requestSuccess.params].data.url : null
537
- // }
538
- // }
539
-
540
- if (
541
- // 数据/按钮
542
- item.type > 1
543
- // 有父级数据
544
- && $n_has(all, item.pid)
545
- ) {
546
- const pItem = all[item.pid]
547
- if ($n_has(btns, pItem.url)) {
548
- btns[pItem.url].push(item)
549
- } else {
550
- btns[pItem.url] = [item]
551
- }
552
- }
553
- }
554
-
555
- // 保存至权限状态中
556
- statePower.value = {
557
- // 角色权限版本
558
- v,
559
- // all id
560
- all,
561
- // 页面
562
- urls,
563
- // 按钮
564
- btns,
565
- // 菜单
566
- menus,
567
- }
568
- }
569
-
570
- /**
571
- * 获取权限
572
- */
573
- async function getData() {
574
-
575
- // 如果没有权限数据
576
- if (! statePower.value.v) {
577
-
578
- // 获取权限数据
579
- const res = await $n_getData($n_config('apiDataPowerName'))
580
- if (res === false) {
581
- statePower.value = {
582
- // 权限版本
583
- v: null,
584
- // all id
585
- all: {},
586
- // 页面
587
- urls: {},
588
- // 按钮
589
- btns: {},
590
- // 菜单
591
- menus: [],
592
- }
593
- return $n_fail()
594
- }
595
- }
596
-
597
- // 如果有权限状态数据, 则直接返回
598
- if (statePower.value.v) {
599
- return $n_success($n_cloneDeep(statePower.value))
600
- }
601
-
602
- return $n_fail()
603
- }
604
-
605
- /**
606
- * 解析传参
607
- */
608
- function parseQuery(data, settings) {
609
-
610
- // 如果配置是字符串
611
- if ($n_isValidString(settings)) {
612
-
613
- // 如果返回所有传参
614
- if (settings === 'all') {
615
- return $n_isValidObject(data) ? data : {}
616
- }
617
-
618
- // 将字符串放到数组中
619
- settings = [settings]
620
-
621
- // 如果配置是对象
622
- } else if ($n_isValidObject(settings)) {
623
- settings = [settings]
624
- }
625
-
626
- const query = {}
627
-
628
- // 如果配置是数组
629
- if ($n_isValidArray(settings)) {
630
-
631
- // 别名
632
- const alias = {}
633
-
634
- for (let item of settings) {
635
- // 如果是需要的字段
636
- if ($n_isValidString(item)) {
637
-
638
- // 将字段转小写
639
- item = $n_toLower($n_trimString(item))
640
-
641
- // 判断字段是否有 as 别名
642
- const arr = $n_split(item, ' as ')
643
-
644
- // 如果有别名
645
- if (arr.length === 2) {
646
- alias[$n_trimString(arr[0])] = $n_trimString(arr[1])
647
-
648
- // 否则别名就是当前字段本身
649
- } else {
650
- alias[item] = item
651
- }
652
-
653
- // 否则如果是自定义传参
654
- } else if ($n_isValidObject(item)) {
655
- Object.assign(query, item)
656
- }
657
- }
658
-
659
- if (
660
- // 如果有参数数据
661
- $n_isValidObject(data)
662
- // 如果有定义别名
663
- && $n_isValidObject(alias)
664
- ) {
665
- $n_forIn(data, function(value, key) {
666
-
667
- // 如果当前字段在别名中
668
- if ($n_has(alias, key)) {
669
- query[alias[key]] = value
670
- }
671
- })
672
- }
673
- }
674
-
675
- return query
676
- }
677
-
678
- /**
679
- * 格式化权限按钮
680
- */
681
- function formatBtns(powerBtns, filterBtns, toObject = false) {
682
-
683
- const newLists = []
684
-
685
- $n_forEach(powerBtns, function(item) {
686
-
687
- item = $n_cloneDeep(item)
688
-
689
- const {
690
- name,
691
- icon,
692
- } = item
693
-
694
- Object.assign(item, {
695
- // 图标
696
- icon: icon || undefined,
697
- // 隐藏按钮
698
- hidden: $n_get(item, 'hidden') === true,
699
- // 显示按钮类型
700
- show: $n_has(item, 'data.show') ? item.data.show : true,
701
- })
702
-
703
- // 是否固定按钮
704
- item.fixed =
705
- // 非隐藏按钮
706
- ! item.hidden
707
- // 固定列
708
- && $n_get(item, 'data.fixed') === true
709
- // 单个按钮
710
- && item.show === 'single'
711
- // 按钮有图标
712
- && !! item.icon
713
-
714
- // 如果是对象
715
- if ($n_isValidObject(filterBtns)) {
716
- if ($n_has(filterBtns, name)) {
717
- newLists.push($n_merge(item, filterBtns[name]))
718
- }
719
-
720
- // 如果是数组
721
- } else if ($n_isValidArray(filterBtns)) {
722
- if ($n_indexOf(filterBtns, name) > -1) {
723
- newLists.push($n_merge(item, filterBtns[name]))
724
- }
725
-
726
- } else {
727
- newLists.push(item)
728
- }
729
- })
730
-
731
- // 转数组
732
- if (toObject) {
733
- const obj = {}
734
- for (const item of newLists) {
735
- obj[item.name] = item
736
- }
737
- return obj
738
- }
739
-
740
- return newLists
741
- }
742
-
743
- /**
744
- * 获取请求传参
745
- */
746
- function getRequestQuery(o) {
747
-
748
- // 传参
749
- const query = {}
750
-
751
- // 如果有请求传参的传参设置
752
- if ($n_has(o.powerBtn.data, 'requestQuery.query')) {
753
- const resQuery = parseQuery(o.query, o.powerBtn.data.requestQuery.query)
754
- if ($n_isValidObject(resQuery)) {
755
- Object.assign(query, resQuery)
756
- }
757
- }
758
-
759
- // 获取列表数据
760
- if (
761
- // 如果按钮参数有显示类型
762
- $n_has(o.powerBtn.data, 'show')
763
- // 按钮参数的显示类型必须是单选或多选
764
- && $n_indexOf(['single', 'multiple'], o.powerBtn.data.show) > -1
765
- // 如果有请求传参的列表设置
766
- && $n_has(o.powerBtn.data, 'requestQuery.list')
767
- // 如果有表格数据
768
- && $n_isValidArray(o.tableSelected)
769
- ) {
770
- let newQuery = {}
771
-
772
- // 如果是单选
773
- if (o.powerBtn.data.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, o.powerBtn.data.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
- let query = getRequestQuery(o)
926
-
927
- // 如果是打开新窗口
928
- // --------------------------------------------------
929
- if (o.powerBtn.data.type === dicts.POWER_DATA_TYPE__OPEN) {
930
-
931
- query = formatQuery(query, true)
932
-
933
- // 如果按钮有标题
934
- const pageTitle = $n_trimString(o.powerBtn.title)
935
- if (pageTitle) {
936
- query.n_page_title = pageTitle
937
- }
938
-
939
- // 如果不是禁止添加来源页面参数
940
- if ($n_get(o.powerBtn.data, 'noFromPageQuery') !== true) {
941
- // 来源页面是当前路由的完整路径
942
- query.n_from_page = encodeURIComponent($currentRoute.fullPath)
943
- }
944
-
945
- // 请求前执行
946
- const resBefore = await $n_runAsync(o.requestBefore)({
947
- options: o,
948
- requestData: query,
949
- })
950
- if (resBefore !== void 0) {
951
- if (resBefore === false) {
952
- return
953
- }
954
- query = resBefore
955
- }
956
-
957
- $n_router.push({
958
- path: o.powerBtn.data.url,
959
- query,
960
- })
961
- return
962
- }
963
-
964
- // 请求数据
965
- let requestData = {}
966
-
967
- // 如果是提交表单
968
- // --------------------------------------------------
969
- if (o.powerBtn.data.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(o.powerBtn.data, '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
- if ($n_isValidObject(o.$form.requestRawFormData.value)) {
1010
- Object.assign(requestData, {
1011
- n__raw: o.$form.requestRawFormData.value
1012
- })
1013
- }
1014
-
1015
- // 如果是请求数据
1016
- // --------------------------------------------------
1017
- } else {
1018
- // 获取表格注入
1019
- o.$table = $n_has(options, '$table') ? options.$table : inject(NTableKey)
1020
-
1021
- // 获取请求数据
1022
- requestData = formatQuery(query, false)
1023
- }
1024
-
1025
- // 判断是否有确认框
1026
- const isConfirm = $n_get(o.powerBtn.data, 'confirm')
1027
- if (
1028
- // 如果有确认框
1029
- isConfirm
1030
- // 如果有密码确认框
1031
- || $n_get(o.powerBtn.data, 'confirmPassword')
1032
- ) {
1033
- // 如果需要先弹出确认框
1034
- if (isConfirm) {
1035
-
1036
- // 确认框
1037
- $n_confirm({
1038
- // 重要操作,请输入登录密码并确认后操作
1039
- message: $n_isValidString(isConfirm) ? isConfirm : '确认要执行该操作吗?',
1040
- })
1041
- // 点击确认执行
1042
- .onOk(onRequest)
1043
-
1044
- return
1045
- }
1046
- }
1047
-
1048
- // 否则执行请求
1049
- await onRequest()
1050
-
1051
- /**
1052
- * 请求事件
1053
- */
1054
- async function onRequest() {
1055
-
1056
- // 请求前执行
1057
- const resBefore = await $n_runAsync(o.requestBefore)({
1058
- options: o,
1059
- requestData,
1060
- })
1061
- if (resBefore !== void 0) {
1062
- if (resBefore === false) {
1063
- return
1064
- }
1065
- requestData = resBefore
1066
- }
1067
-
1068
- // 请求
1069
- const res = await $n_http({
1070
- // 请求地址
1071
- url: o.powerBtn.data.url,
1072
- // 请求数据
1073
- data: requestData,
1074
- })
1075
-
1076
- // 返回结果数据
1077
- const resultData = Object.assign({
1078
- // 参数
1079
- options: o,
1080
- // 请求数据
1081
- requestData,
1082
- }, res)
1083
-
1084
- // 请求后执行
1085
- if (await $n_runAsync(o.requestAfter)(resultData) === false) {
1086
- return
1087
- }
1088
-
1089
- // 如果请求成功
1090
- if (res.status) {
1091
-
1092
- // 下一步
1093
- function next(isNotify = true) {
1094
-
1095
- // 轻提示
1096
- if (isNotify) {
1097
- $n_toast({
1098
- type: 'positive',
1099
- message: '恭喜您,操作成功',
1100
- })
1101
- }
1102
-
1103
- // 判断是否有请求成功后的操作动作
1104
- if ($n_has(o.powerBtn.data, 'requestSuccess.type')) {
1105
- switch (o.powerBtn.data.requestSuccess.type) {
1106
-
1107
- // 关闭当前页面
1108
- case 'close':
1109
- // 关闭窗口并跳转页面
1110
- case 'closePush':
1111
- // 关闭窗口、跳转并刷新页面
1112
- case 'closePushRefresh':
1113
-
1114
- // 如果是渲染页面
1115
- // 说明该页面在 <table-splitter> 组件内部被渲染, 则不需要关闭当前窗口
1116
- if ($n_has($route.query, 'n_renderpage') && $route.query.n_renderpage === 1) {
1117
- // 则无任何操作
1118
- return
1119
- }
1120
-
1121
- const opts = {
1122
- type: 'closeCurrentTab',
1123
- }
1124
-
1125
- if (
1126
- // 如果不是关闭当前页面, 则为关闭窗口并跳转页面
1127
- o.powerBtn.data.requestSuccess.type !== 'close'
1128
- // 如果有来源页面
1129
- && $n_has($route.query, 'n_from_page')
1130
- && $n_isValidString($route.query.n_from_page)
1131
- ) {
1132
- Object.assign(opts, {
1133
- // 跳转页面地址
1134
- pushPage: decodeURIComponent($route.query.n_from_page),
1135
- // 是否跳转并刷新页面
1136
- isPushRefresh: o.powerBtn.data.requestSuccess.type === 'closePushRefresh',
1137
- })
1138
-
1139
- // 否则如果定义了跳转页面
1140
- // else if ($n_has(o.powerBtn.data, 'requestSuccess.params') && $n_isValidString(o.powerBtn.data.requestSuccess.params)) {
1141
- // pushPage = o.powerBtn.data.requestSuccess.params
1142
- // }
1143
- }
1144
-
1145
- // 关闭当前标签页
1146
- $n_bus.emit('main', opts)
1147
- break
1148
-
1149
- // 重置表单
1150
- case 'resetForm':
1151
- $n_run(o.$form?.resetForm)()
1152
- break
1153
-
1154
- // 刷新列表
1155
- case 'refreshList':
1156
- $n_run(o.$table?.tableRefresh)()
1157
- break
1158
- }
1159
- }
1160
- }
1161
-
1162
- // 请求成功执行
1163
- if (await $n_runAsync(o.requestSuccess)(Object.assign({ next }, resultData)) === false) {
1164
- return
1165
- }
1166
-
1167
- // 下一步
1168
- next()
1169
-
1170
- } else {
1171
- // 请求失败执行
1172
- $n_run(o.requestFail)(resultData)
1173
- }
1174
- }
1175
- }
1176
-
1177
- /**
1178
- * 获取路由页面的角色权限
1179
- */
1180
- function getPageData($route) {
1181
-
1182
- if (! $route) {
1183
- $route = $n_router.getRoute()
1184
- }
1185
-
1186
- const path = $n_get($route, 'path')
1187
- if (! path) {
1188
- return $n_fail('路由参数错误')
1189
- }
1190
-
1191
- if (! statePower.value.v) {
1192
- return $n_fail('没有获取到权限数据')
1193
- }
1194
-
1195
- // 获取角色数据
1196
- const { urls, btns } = $n_cloneDeep(statePower.value)
1197
- if (! $n_has(urls, path)) {
1198
- return $n_fail('该页面没有权限')
1199
- }
1200
-
1201
- return $n_success({
1202
- page: urls[path],
1203
- btns: $n_has(btns, path) ? btns[path] : [],
1204
- })
1205
- }
1206
-
1207
- /**
1208
- * 权限业务
1209
- */
1210
- const $power = {
1211
- // 创建权限实例
1212
- create,
1213
- // 设置权限数据
1214
- setData,
1215
- // 获取权限数据
1216
- getData,
1217
- // 获取路由页面的角色权限
1218
- getPageData,
1219
- // 格式化权限按钮
1220
- formatBtns,
1221
- // 请求
1222
- request,
1223
- }
1224
-
1225
- 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 = $n_has(options, '$dialog') ? options.$dialog : inject(NDialogKey)
94
+ const hasDialog = !! $dialog
95
+
96
+ // 获取渲染注入
97
+ const $render = $n_has(options, '$render') ? options.$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, 'getRoute')) {
151
+
152
+ // 设为渲染组件的路由
153
+ $route = $render.getRoute()
154
+
155
+ // 否则获取当前路由
156
+ } else {
157
+
158
+ $route = $currentRoute
159
+ }
160
+
161
+ // quasar 对象
162
+ const $q = useQuasar()
163
+
164
+ // 表格已选数据
165
+ const tableSelected = ref([])
166
+
167
+ // 是否显示左边侧滑菜单
168
+ const leftDrawerModelValue = ref(null)
169
+
170
+ // 是否显示右边侧滑菜单
171
+ const rightDrawerModelValue = ref(null)
172
+
173
+ /**
174
+ * 检查是否上传中
175
+ */
176
+ function checkUploading() {
177
+ for (const uploader of data.uploader) {
178
+ if (uploader.checkUploading()) {
179
+ return true
180
+ }
181
+ }
182
+ return false
183
+ }
184
+
185
+ const _data = {
186
+ // 表格实例
187
+ $table: null,
188
+ // 表单实例
189
+ $form: null,
190
+ }
191
+
192
+ // 注入数据
193
+ const data = {
194
+ // 页面加载
195
+ pageLoading: ref(o.pageLoading),
196
+ // 页面状态
197
+ pageStatus: ref(o.pageStatus),
198
+ // 空状态描述
199
+ emptyDescription: ref(o.emptyDescription),
200
+
201
+ // 当前路由全路径
202
+ routeFullPath: $route.fullPath,
203
+ // 当前路由路径
204
+ routePath: $route.path,
205
+ // 当前路由参数
206
+ routeQuery: $route.query,
207
+ // 获取当前路由
208
+ getRoute() {
209
+ return $route
210
+ },
211
+ // 格式化权限按钮
212
+ formatPowerBtns: o.formatPowerBtns,
213
+
214
+ // 左边侧滑菜单数据
215
+ leftDrawer: {
216
+ // 图标
217
+ icon: o.leftDrawerIcon,
218
+ // 是否显示
219
+ modelValue: leftDrawerModelValue,
220
+ // 是否显示切换按钮
221
+ showButton() {
222
+ return leftDrawerModelValue.value !== null
223
+ },
224
+ // 切换
225
+ toggle() {
226
+ if (leftDrawerModelValue.value !== null) {
227
+ leftDrawerModelValue.value = ! leftDrawerModelValue.value
228
+ }
229
+ },
230
+ },
231
+ // 右边侧滑菜单数据
232
+ rightDrawer: {
233
+ // 图标
234
+ icon: o.rightDrawerIcon,
235
+ // 是否显示
236
+ modelValue: rightDrawerModelValue,
237
+ // 是否显示切换按钮
238
+ showButton() {
239
+ return rightDrawerModelValue.value !== null
240
+ },
241
+ // 切换
242
+ toggle() {
243
+ if (rightDrawerModelValue.value !== null) {
244
+ rightDrawerModelValue.value = ! rightDrawerModelValue.value
245
+ }
246
+ },
247
+ },
248
+ // 表格已选数据
249
+ tableSelected,
250
+ // 上传器
251
+ uploader: [],
252
+ // 检查是否上传中
253
+ checkUploading,
254
+ }
255
+
256
+ // 如果是权限页面
257
+ if (o.power) {
258
+
259
+ // 获取当前页面角色权限
260
+ const { status, data: res } = getPageData($route)
261
+ if (! status) {
262
+ o.pageStatus = false
263
+ o.emptyDescription = res.msg
264
+ o.power = false
265
+
266
+ } else {
267
+ // 当前页面权限
268
+ data.powerPage = res.page
269
+ // 当前页面权限按钮
270
+ data.powerBtns = ref(o.showPowerBtns ? res.btns : [])
271
+ // 当前页面工具栏权限按钮
272
+ data.toolbarPowerBtns = computed(function() {
273
+
274
+ if (
275
+ // 如果显示工具栏权限按钮
276
+ o.showToolbarPowerBtns
277
+ // 有权限按钮数据
278
+ && $n_isValidArray(data.powerBtns.value)
279
+ ) {
280
+ const lists = []
281
+
282
+ // 格式化权限按钮列表
283
+ $n_forEach($n_filter(formatBtns(data.powerBtns.value), e => e.type > 2), function(item) {
284
+
285
+ if (! item.hidden) {
286
+
287
+ // 如果是单条数据显示
288
+ const isSingle = item.show === 'single'
289
+
290
+ // 如果是单条 || 多条显示
291
+ if (isSingle || item.show === 'multiple') {
292
+
293
+ // 初始为不显示
294
+ item.show = false
295
+
296
+ // 如果有表格选中数据
297
+ if ($n_isValidArray(data.tableSelected.value)) {
298
+ // 如果是单个显示
299
+ if (isSingle) {
300
+ item.show = data.tableSelected.value.length === 1
301
+
302
+ // 否则是多个显示
303
+ } else {
304
+ item.show = data.tableSelected.value.length >= 1
305
+ }
306
+ }
307
+ }
308
+
309
+ // 如果是手机模式
310
+ if ($q.platform.is.mobile) {
311
+ item.icon = undefined
312
+ }
313
+ }
314
+
315
+ if (
316
+ // 如果有格式化权限按钮方法
317
+ $n_isFunction(o.formatPowerBtns)
318
+ && o.formatPowerBtns(item, false, tableSelected.value) === false
319
+ ) {
320
+ return
321
+ }
322
+
323
+ lists.push(item)
324
+ })
325
+
326
+ return lists
327
+ }
328
+
329
+ return []
330
+ })
331
+
332
+ // 权限按钮点击
333
+ data.powerBtnClick = async function (powerBtn, tableSelected) {
334
+
335
+ // 权限请求
336
+ await request({
337
+ // power
338
+ $power: data,
339
+ // 按钮数据
340
+ powerBtn,
341
+ // 权限路由参数
342
+ $route,
343
+ // 当前路由参数
344
+ $currentRoute,
345
+ // 表格选中数据
346
+ tableSelected,
347
+ // 表格实例
348
+ $table: _data.$table,
349
+ // 表单实例
350
+ $form: _data.$form,
351
+ // 检查是否正在上传文件
352
+ checkUploading,
353
+
354
+ // 请求前执行
355
+ requestBefore: o.requestBefore,
356
+ // 请求成功执行
357
+ requestSuccess: o.requestSuccess,
358
+ // 请求失败执行
359
+ requestFail: o.requestFail,
360
+ // 请求后执行
361
+ requestAfter: o.requestAfter,
362
+ })
363
+ }
364
+ }
365
+ }
366
+
367
+ // 如果没有开启权限
368
+ if (! o.power) {
369
+ // 当前页面权限
370
+ data.powerPage = {}
371
+ // 当前页面权限按钮
372
+ data.powerBtns = ref([])
373
+ // 当前页面工具栏权限按钮
374
+ data.toolbarPowerBtns = ref([])
375
+ // 权限按钮点击
376
+ data.powerBtnClick = ()=>{}
377
+ }
378
+
379
+ // 更新数据
380
+ data.update = function(cb) {
381
+ cb(data, _data)
382
+ }
383
+
384
+ // 提供可以被后代组件注入的值
385
+ provide(NPowerKey, data)
386
+
387
+ return data
388
+ }
389
+
390
+ /**
391
+ * 设置权限数据
392
+ */
393
+ function setData(data) {
394
+
395
+ // 如果没有角色数据
396
+ if (! $n_isValidObject(data)) {
397
+ return
398
+ }
399
+
400
+ const {
401
+ rows,
402
+ v,
403
+ } = data
404
+
405
+ if (! $n_isValidArray(rows) || ! v) {
406
+ return
407
+ }
408
+
409
+ // all id
410
+ const all = {}
411
+ // 页面
412
+ const urls = {}
413
+ // 按钮
414
+ const btns = {}
415
+ // 菜单
416
+ const menus = []
417
+
418
+ for (const item of rows) {
419
+
420
+ // 【格式化 start】
421
+ // --------------------------------------------------
422
+ if (item.data) {
423
+ item.data = $n_json.parse(item.data)
424
+ }
425
+ item.data = $n_isValidObject(item.data) ? $n_numberDeep(item.data) : {}
426
+
427
+ // 设置数据类型
428
+ item.data.type = item.data_type
429
+ delete item.data_type
430
+
431
+ // 标识
432
+ item.name = ''
433
+
434
+ // 如果有 url
435
+ if ($n_isValidString(item.url)) {
436
+
437
+ // url 首位加上反斜杠
438
+ item.url = $n_slash($n_toLower($n_trimString(item.url)), 'start', true)
439
+ if (item.url) {
440
+
441
+ item.data.url = item.url
442
+
443
+ // 截取最后一个反斜杠
444
+ const lastIndex = item.data.url.lastIndexOf('/')
445
+ if (lastIndex > -1) {
446
+ item.name = item.data.url.substring(lastIndex + 1)
447
+ }
448
+
449
+ } else {
450
+ item.url = ''
451
+ item.data.url = ''
452
+ }
453
+
454
+ } else {
455
+ item.url = ''
456
+ item.data.url = ''
457
+ }
458
+
459
+ // 菜单
460
+ if (item.type === 1) {
461
+ menus.push(item)
462
+
463
+ // 按钮
464
+ } else if (item.type > 2) {
465
+
466
+ // 按钮类型
467
+ switch (item.type) {
468
+
469
+ // 默认按钮
470
+ case 11:
471
+ item.color = 'default'
472
+ break;
473
+
474
+ // 主要按钮
475
+ case 12:
476
+ item.color = 'primary'
477
+ break;
478
+
479
+ // 成功按钮
480
+ case 13:
481
+ item.color = 'secondary'
482
+ break;
483
+
484
+ // 信息按钮
485
+ case 14:
486
+ item.color = 'info'
487
+ break;
488
+
489
+ // 警告按钮
490
+ case 15:
491
+ item.color = 'warning'
492
+ break;
493
+
494
+ // 危险按钮
495
+ case 16:
496
+ item.color = 'negative'
497
+ break;
498
+
499
+ // 隐藏按钮
500
+ case 20:
501
+ item.hidden = true
502
+ break;
503
+ }
504
+ }
505
+ // 【格式化 end】
506
+ // --------------------------------------------------
507
+
508
+ // url
509
+ if (item.url) {
510
+ // 添加至 all
511
+ all[item.id] = item
512
+ // 添加至页面
513
+ urls[item.url] = item
514
+ }
515
+ }
516
+
517
+ for (const item of rows) {
518
+
519
+ // 如果有跳转页面
520
+ if ($n_has(item.data, 'toPage')) {
521
+ // 设置跳转页面地址
522
+ item.data.toPage = $n_has(all, item.data.toPage) ? all[item.data.toPage].data.url : null
523
+ }
524
+
525
+ // 如果有请求成功执行类型
526
+ // else if ($n_has(item.data, 'requestSuccess.type')) {
527
+ // // 如果请求成功执行类型是关闭窗口、跳转并刷新页面
528
+ // if (item.data.requestSuccess.type === 'closePushRefresh') {
529
+ // // 设置刷新页面地址
530
+ // item.data.requestSuccess.params =
531
+ // (
532
+ // // 如果有刷新页面的参数 id
533
+ // $n_has(item.data.requestSuccess, 'params')
534
+ // // 如果有页面数据
535
+ // && $n_has(all, item.data.requestSuccess.params)
536
+ // ) ? all[item.data.requestSuccess.params].data.url : null
537
+ // }
538
+ // }
539
+
540
+ if (
541
+ // 数据/按钮
542
+ item.type > 1
543
+ // 有父级数据
544
+ && $n_has(all, item.pid)
545
+ ) {
546
+ const pItem = all[item.pid]
547
+ if ($n_has(btns, pItem.url)) {
548
+ btns[pItem.url].push(item)
549
+ } else {
550
+ btns[pItem.url] = [item]
551
+ }
552
+ }
553
+ }
554
+
555
+ // 保存至权限状态中
556
+ statePower.value = {
557
+ // 角色权限版本
558
+ v,
559
+ // all id
560
+ all,
561
+ // 页面
562
+ urls,
563
+ // 按钮
564
+ btns,
565
+ // 菜单
566
+ menus,
567
+ }
568
+ }
569
+
570
+ /**
571
+ * 获取权限
572
+ */
573
+ async function getData() {
574
+
575
+ // 如果没有权限数据
576
+ if (! statePower.value.v) {
577
+
578
+ // 获取权限数据
579
+ const res = await $n_getData($n_config('apiDataPowerName'))
580
+ if (res === false) {
581
+ statePower.value = {
582
+ // 权限版本
583
+ v: null,
584
+ // all id
585
+ all: {},
586
+ // 页面
587
+ urls: {},
588
+ // 按钮
589
+ btns: {},
590
+ // 菜单
591
+ menus: [],
592
+ }
593
+ return $n_fail()
594
+ }
595
+ }
596
+
597
+ // 如果有权限状态数据, 则直接返回
598
+ if (statePower.value.v) {
599
+ return $n_success($n_cloneDeep(statePower.value))
600
+ }
601
+
602
+ return $n_fail()
603
+ }
604
+
605
+ /**
606
+ * 解析传参
607
+ */
608
+ function parseQuery(data, settings) {
609
+
610
+ // 如果配置是字符串
611
+ if ($n_isValidString(settings)) {
612
+
613
+ // 如果返回所有传参
614
+ if (settings === 'all') {
615
+ return $n_isValidObject(data) ? data : {}
616
+ }
617
+
618
+ // 将字符串放到数组中
619
+ settings = [settings]
620
+
621
+ // 如果配置是对象
622
+ } else if ($n_isValidObject(settings)) {
623
+ settings = [settings]
624
+ }
625
+
626
+ const query = {}
627
+
628
+ // 如果配置是数组
629
+ if ($n_isValidArray(settings)) {
630
+
631
+ // 别名
632
+ const alias = {}
633
+
634
+ for (let item of settings) {
635
+ // 如果是需要的字段
636
+ if ($n_isValidString(item)) {
637
+
638
+ // 将字段转小写
639
+ item = $n_toLower($n_trimString(item))
640
+
641
+ // 判断字段是否有 as 别名
642
+ const arr = $n_split(item, ' as ')
643
+
644
+ // 如果有别名
645
+ if (arr.length === 2) {
646
+ alias[$n_trimString(arr[0])] = $n_trimString(arr[1])
647
+
648
+ // 否则别名就是当前字段本身
649
+ } else {
650
+ alias[item] = item
651
+ }
652
+
653
+ // 否则如果是自定义传参
654
+ } else if ($n_isValidObject(item)) {
655
+ Object.assign(query, item)
656
+ }
657
+ }
658
+
659
+ if (
660
+ // 如果有参数数据
661
+ $n_isValidObject(data)
662
+ // 如果有定义别名
663
+ && $n_isValidObject(alias)
664
+ ) {
665
+ $n_forIn(data, function(value, key) {
666
+
667
+ // 如果当前字段在别名中
668
+ if ($n_has(alias, key)) {
669
+ query[alias[key]] = value
670
+ }
671
+ })
672
+ }
673
+ }
674
+
675
+ return query
676
+ }
677
+
678
+ /**
679
+ * 格式化权限按钮
680
+ */
681
+ function formatBtns(powerBtns, filterBtns, toObject = false) {
682
+
683
+ const newLists = []
684
+
685
+ $n_forEach(powerBtns, function(item) {
686
+
687
+ item = $n_cloneDeep(item)
688
+
689
+ const {
690
+ name,
691
+ icon,
692
+ } = item
693
+
694
+ Object.assign(item, {
695
+ // 图标
696
+ icon: icon || undefined,
697
+ // 隐藏按钮
698
+ hidden: $n_get(item, 'hidden') === true,
699
+ // 显示按钮类型
700
+ show: $n_has(item, 'data.show') ? item.data.show : true,
701
+ })
702
+
703
+ // 是否固定按钮
704
+ item.fixed =
705
+ // 非隐藏按钮
706
+ ! item.hidden
707
+ // 固定列
708
+ && $n_get(item, 'data.fixed') === true
709
+ // 单个按钮
710
+ && item.show === 'single'
711
+ // 按钮有图标
712
+ && !! item.icon
713
+
714
+ // 如果是对象
715
+ if ($n_isValidObject(filterBtns)) {
716
+ if ($n_has(filterBtns, name)) {
717
+ newLists.push($n_merge(item, filterBtns[name]))
718
+ }
719
+
720
+ // 如果是数组
721
+ } else if ($n_isValidArray(filterBtns)) {
722
+ if ($n_indexOf(filterBtns, name) > -1) {
723
+ newLists.push($n_merge(item, filterBtns[name]))
724
+ }
725
+
726
+ } else {
727
+ newLists.push(item)
728
+ }
729
+ })
730
+
731
+ // 转数组
732
+ if (toObject) {
733
+ const obj = {}
734
+ for (const item of newLists) {
735
+ obj[item.name] = item
736
+ }
737
+ return obj
738
+ }
739
+
740
+ return newLists
741
+ }
742
+
743
+ /**
744
+ * 获取请求传参
745
+ */
746
+ function getRequestQuery(o) {
747
+
748
+ // 传参
749
+ const query = {}
750
+
751
+ // 如果有请求传参的传参设置
752
+ if ($n_has(o.powerBtn.data, 'requestQuery.query')) {
753
+ const resQuery = parseQuery(o.query, o.powerBtn.data.requestQuery.query)
754
+ if ($n_isValidObject(resQuery)) {
755
+ Object.assign(query, resQuery)
756
+ }
757
+ }
758
+
759
+ // 获取列表数据
760
+ if (
761
+ // 如果按钮参数有显示类型
762
+ $n_has(o.powerBtn.data, 'show')
763
+ // 按钮参数的显示类型必须是单选或多选
764
+ && $n_indexOf(['single', 'multiple'], o.powerBtn.data.show) > -1
765
+ // 如果有请求传参的列表设置
766
+ && $n_has(o.powerBtn.data, 'requestQuery.list')
767
+ // 如果有表格数据
768
+ && $n_isValidArray(o.tableSelected)
769
+ ) {
770
+ let newQuery = {}
771
+
772
+ // 如果是单选
773
+ if (o.powerBtn.data.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, o.powerBtn.data.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
+ let query = getRequestQuery(o)
926
+
927
+ // 如果是打开新窗口
928
+ // --------------------------------------------------
929
+ if (o.powerBtn.data.type === dicts.POWER_DATA_TYPE__OPEN) {
930
+
931
+ query = formatQuery(query, true)
932
+
933
+ // 如果按钮有标题
934
+ const pageTitle = $n_trimString(o.powerBtn.title)
935
+ if (pageTitle) {
936
+ query.n_page_title = pageTitle
937
+ }
938
+
939
+ // 如果不是禁止添加来源页面参数
940
+ if ($n_get(o.powerBtn.data, 'noFromPageQuery') !== true) {
941
+ // 来源页面是当前路由的完整路径
942
+ query.n_from_page = encodeURIComponent($currentRoute.fullPath)
943
+ }
944
+
945
+ // 请求前执行
946
+ const resBefore = await $n_runAsync(o.requestBefore)({
947
+ options: o,
948
+ requestData: query,
949
+ })
950
+ if (resBefore !== void 0) {
951
+ if (resBefore === false) {
952
+ return
953
+ }
954
+ query = resBefore
955
+ }
956
+
957
+ $n_router.push({
958
+ path: o.powerBtn.data.url,
959
+ query,
960
+ })
961
+ return
962
+ }
963
+
964
+ // 请求数据
965
+ let requestData = {}
966
+
967
+ // 如果是提交表单
968
+ // --------------------------------------------------
969
+ if (o.powerBtn.data.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(o.powerBtn.data, '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
+ if ($n_isValidObject(o.$form.requestRawFormData.value)) {
1010
+ Object.assign(requestData, {
1011
+ n__raw: o.$form.requestRawFormData.value
1012
+ })
1013
+ }
1014
+
1015
+ // 如果是请求数据
1016
+ // --------------------------------------------------
1017
+ } else {
1018
+ // 获取表格注入
1019
+ o.$table = $n_has(options, '$table') ? options.$table : inject(NTableKey)
1020
+
1021
+ // 获取请求数据
1022
+ requestData = formatQuery(query, false)
1023
+ }
1024
+
1025
+ // 判断是否有确认框
1026
+ const isConfirm = $n_get(o.powerBtn.data, 'confirm')
1027
+ if (
1028
+ // 如果有确认框
1029
+ isConfirm
1030
+ // 如果有密码确认框
1031
+ || $n_get(o.powerBtn.data, 'confirmPassword')
1032
+ ) {
1033
+ // 如果需要先弹出确认框
1034
+ if (isConfirm) {
1035
+
1036
+ // 确认框
1037
+ $n_confirm({
1038
+ // 重要操作,请输入登录密码并确认后操作
1039
+ message: $n_isValidString(isConfirm) ? isConfirm : '确认要执行该操作吗?',
1040
+ })
1041
+ // 点击确认执行
1042
+ .onOk(onRequest)
1043
+
1044
+ return
1045
+ }
1046
+ }
1047
+
1048
+ // 否则执行请求
1049
+ await onRequest()
1050
+
1051
+ /**
1052
+ * 请求事件
1053
+ */
1054
+ async function onRequest() {
1055
+
1056
+ // 请求前执行
1057
+ const resBefore = await $n_runAsync(o.requestBefore)({
1058
+ options: o,
1059
+ requestData,
1060
+ })
1061
+ if (resBefore !== void 0) {
1062
+ if (resBefore === false) {
1063
+ return
1064
+ }
1065
+ requestData = resBefore
1066
+ }
1067
+
1068
+ // 请求
1069
+ const res = await $n_http({
1070
+ // 请求地址
1071
+ url: o.powerBtn.data.url,
1072
+ // 请求数据
1073
+ data: requestData,
1074
+ })
1075
+
1076
+ // 返回结果数据
1077
+ const resultData = Object.assign({
1078
+ // 参数
1079
+ options: o,
1080
+ // 请求数据
1081
+ requestData,
1082
+ }, res)
1083
+
1084
+ // 请求后执行
1085
+ if (await $n_runAsync(o.requestAfter)(resultData) === false) {
1086
+ return
1087
+ }
1088
+
1089
+ // 如果请求成功
1090
+ if (res.status) {
1091
+
1092
+ // 下一步
1093
+ function next(isNotify = true) {
1094
+
1095
+ // 轻提示
1096
+ if (isNotify) {
1097
+ $n_toast({
1098
+ type: 'positive',
1099
+ message: '恭喜您,操作成功',
1100
+ })
1101
+ }
1102
+
1103
+ // 判断是否有请求成功后的操作动作
1104
+ if ($n_has(o.powerBtn.data, 'requestSuccess.type')) {
1105
+ switch (o.powerBtn.data.requestSuccess.type) {
1106
+
1107
+ // 关闭当前页面
1108
+ case 'close':
1109
+ // 关闭窗口并跳转页面
1110
+ case 'closePush':
1111
+ // 关闭窗口、跳转并刷新页面
1112
+ case 'closePushRefresh':
1113
+
1114
+ // 如果是渲染页面
1115
+ // 说明该页面在 <table-splitter> 组件内部被渲染, 则不需要关闭当前窗口
1116
+ if ($n_has($route.query, 'n_renderpage') && $route.query.n_renderpage === 1) {
1117
+ // 则无任何操作
1118
+ return
1119
+ }
1120
+
1121
+ const opts = {
1122
+ type: 'closeCurrentTab',
1123
+ }
1124
+
1125
+ if (
1126
+ // 如果不是关闭当前页面, 则为关闭窗口并跳转页面
1127
+ o.powerBtn.data.requestSuccess.type !== 'close'
1128
+ // 如果有来源页面
1129
+ && $n_has($route.query, 'n_from_page')
1130
+ && $n_isValidString($route.query.n_from_page)
1131
+ ) {
1132
+ Object.assign(opts, {
1133
+ // 跳转页面地址
1134
+ pushPage: decodeURIComponent($route.query.n_from_page),
1135
+ // 是否跳转并刷新页面
1136
+ isPushRefresh: o.powerBtn.data.requestSuccess.type === 'closePushRefresh',
1137
+ })
1138
+
1139
+ // 否则如果定义了跳转页面
1140
+ // else if ($n_has(o.powerBtn.data, 'requestSuccess.params') && $n_isValidString(o.powerBtn.data.requestSuccess.params)) {
1141
+ // pushPage = o.powerBtn.data.requestSuccess.params
1142
+ // }
1143
+ }
1144
+
1145
+ // 关闭当前标签页
1146
+ $n_bus.emit('main', opts)
1147
+ break
1148
+
1149
+ // 重置表单
1150
+ case 'resetForm':
1151
+ $n_run(o.$form?.resetForm)()
1152
+ break
1153
+
1154
+ // 刷新列表
1155
+ case 'refreshList':
1156
+ $n_run(o.$table?.tableRefresh)()
1157
+ break
1158
+ }
1159
+ }
1160
+ }
1161
+
1162
+ // 请求成功执行
1163
+ if (await $n_runAsync(o.requestSuccess)(Object.assign({ next }, resultData)) === false) {
1164
+ return
1165
+ }
1166
+
1167
+ // 下一步
1168
+ next()
1169
+
1170
+ } else {
1171
+ // 请求失败执行
1172
+ $n_run(o.requestFail)(resultData)
1173
+ }
1174
+ }
1175
+ }
1176
+
1177
+ /**
1178
+ * 获取路由页面的角色权限
1179
+ */
1180
+ function getPageData($route) {
1181
+
1182
+ if (! $route) {
1183
+ $route = $n_router.getRoute()
1184
+ }
1185
+
1186
+ const path = $n_get($route, 'path')
1187
+ if (! path) {
1188
+ return $n_fail('路由参数错误')
1189
+ }
1190
+
1191
+ if (! statePower.value.v) {
1192
+ return $n_fail('没有获取到权限数据')
1193
+ }
1194
+
1195
+ // 获取角色数据
1196
+ const { urls, btns } = $n_cloneDeep(statePower.value)
1197
+ if (! $n_has(urls, path)) {
1198
+ return $n_fail('该页面没有权限')
1199
+ }
1200
+
1201
+ return $n_success({
1202
+ page: urls[path],
1203
+ btns: $n_has(btns, path) ? btns[path] : [],
1204
+ })
1205
+ }
1206
+
1207
+ /**
1208
+ * 权限业务
1209
+ */
1210
+ const $power = {
1211
+ // 创建权限实例
1212
+ create,
1213
+ // 设置权限数据
1214
+ setData,
1215
+ // 获取权限数据
1216
+ getData,
1217
+ // 获取路由页面的角色权限
1218
+ getPageData,
1219
+ // 格式化权限按钮
1220
+ formatBtns,
1221
+ // 请求
1222
+ request,
1223
+ }
1224
+
1225
+ export default $power