af-mobile-client-vue3 1.3.12 → 1.3.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (271) hide show
  1. package/.claude/settings.local.json +10 -0
  2. package/.cursorrules +60 -60
  3. package/.editorconfig +9 -9
  4. package/.env +10 -10
  5. package/.env.development +1 -1
  6. package/.env.production +1 -1
  7. package/.node-version +1 -1
  8. package/.vscode/extensions.json +12 -12
  9. package/.vscode/settings.json +66 -66
  10. package/CLAUDE.md +189 -184
  11. package/README.md +182 -181
  12. package/af-example-mobile-vue-web.iml +9 -9
  13. package/build/vite/index.ts +98 -98
  14. package/build/vite/optimize.ts +34 -34
  15. package/build/vite/vconsole.ts +47 -47
  16. package/commitlint.config.ts +32 -32
  17. package/compress.js +36 -36
  18. package/eslint.config.ts +30 -30
  19. package/index.html +23 -23
  20. package/mock/data.ts +20 -20
  21. package/mock/index.ts +7 -7
  22. package/mock/modules/prose.mock.ts +13 -13
  23. package/mock/modules/user.mock.ts +152 -152
  24. package/mock/util.ts +19 -19
  25. package/netlify.toml +12 -12
  26. package/package.json +114 -114
  27. package/postcss.config.ts +27 -27
  28. package/public/favicon.svg +4 -4
  29. package/public/safari-pinned-tab.svg +4 -4
  30. package/scripts/verifyCommit.js +19 -19
  31. package/src/App.vue +79 -79
  32. package/src/api/mock/index.ts +30 -30
  33. package/src/api/user/index.ts +40 -40
  34. package/src/assets/img/user/login/background-shadow-1.svg +20 -20
  35. package/src/assets/img/user/login/logo-background.svg +20 -20
  36. package/src/bootstrap.ts +26 -26
  37. package/src/components/core/BeautifulLoading/index.vue +52 -52
  38. package/src/components/core/ImageUploader/index.vue +244 -244
  39. package/src/components/core/NavBar/index.vue +53 -53
  40. package/src/components/core/Tabbar/index.vue +32 -32
  41. package/src/components/core/Uploader/index.vue +124 -124
  42. package/src/components/core/XGridDropOption/index.vue +154 -156
  43. package/src/components/core/XMultiSelect/index.vue +183 -183
  44. package/src/components/core/XSelect/index.vue +149 -149
  45. package/src/components/data/InfoDisplay/index.vue +132 -0
  46. package/src/components/data/UserDetail/api.ts +24 -0
  47. package/src/components/data/UserDetail/index.vue +620 -0
  48. package/src/components/data/UserDetail/recordEntries.ts +159 -0
  49. package/src/components/data/UserDetail/types.ts +26 -0
  50. package/src/components/data/XBadge/index.vue +82 -82
  51. package/src/components/data/XCellDetail/index.vue +105 -105
  52. package/src/components/data/XCellList/XCellList.md +313 -313
  53. package/src/components/data/XCellList/index.vue +1075 -1075
  54. package/src/components/data/XCellListFilter/QrScanner/index.vue +207 -207
  55. package/src/components/data/XCellListFilter/QrScanner/startScanAnimation.ts +53 -53
  56. package/src/components/data/XCellListFilter/VpnRecognition/index.vue +119 -119
  57. package/src/components/data/XCellListFilter/index.vue +705 -705
  58. package/src/components/data/XForm/index.vue +659 -659
  59. package/src/components/data/XFormGroup/doc/DeviceForm.vue +122 -122
  60. package/src/components/data/XFormGroup/doc/FormGroupDemo.vue +56 -56
  61. package/src/components/data/XFormGroup/doc/README.md +286 -273
  62. package/src/components/data/XFormGroup/doc/UserForm.vue +102 -102
  63. package/src/components/data/XFormGroup/index.vue +240 -240
  64. package/src/components/data/XFormItem/index.vue +1310 -1310
  65. package/src/components/data/XOlMap/README.md +227 -227
  66. package/src/components/data/XOlMap/XLocationPicker/index.vue +226 -225
  67. package/src/components/data/XOlMap/index.vue +1490 -1490
  68. package/src/components/data/XOlMap/types.ts +149 -149
  69. package/src/components/data/XOlMap/utils/wgs84ToGcj02.js +154 -154
  70. package/src/components/data/XReportForm/DateTimeSecondsPicker.vue +208 -208
  71. package/src/components/data/XReportForm/XReportFormJsonRender.vue +220 -220
  72. package/src/components/data/XReportForm/index.vue +1393 -1393
  73. package/src/components/data/XReportGrid/XAddReport/XAddReport.vue +198 -198
  74. package/src/components/data/XReportGrid/XAddReport/index.js +3 -3
  75. package/src/components/data/XReportGrid/XAddReport/index.md +53 -52
  76. package/src/components/data/XReportGrid/XAddReport/index.ts +10 -10
  77. package/src/components/data/XReportGrid/XReport.vue +960 -960
  78. package/src/components/data/XReportGrid/XReportDemo.vue +33 -33
  79. package/src/components/data/XReportGrid/XReportDesign.vue +597 -597
  80. package/src/components/data/XReportGrid/XReportDrawer/XReportDrawer.vue +148 -148
  81. package/src/components/data/XReportGrid/XReportDrawer/index.js +3 -3
  82. package/src/components/data/XReportGrid/XReportDrawer/index.ts +10 -10
  83. package/src/components/data/XReportGrid/XReportJsonRender.vue +399 -399
  84. package/src/components/data/XReportGrid/XReportTrGroup.vue +592 -592
  85. package/src/components/data/XReportGrid/index.md +46 -42
  86. package/src/components/data/XReportGrid/print.js +184 -184
  87. package/src/components/data/XSignature/index.vue +284 -285
  88. package/src/components/data/XTag/index.vue +10 -10
  89. package/src/components/layout/NormalDataLayout/index.vue +69 -69
  90. package/src/components/layout/TabBarLayout/index.vue +40 -40
  91. package/src/composables/dark.ts +5 -5
  92. package/src/config/routes.ts +9 -9
  93. package/src/constants/index.ts +2 -2
  94. package/src/enums/requestEnum.ts +25 -25
  95. package/src/expression/ExpressionRunner.ts +28 -28
  96. package/src/expression/TestExpression.ts +510 -510
  97. package/src/expression/core/Delegate.ts +116 -116
  98. package/src/expression/core/Expression.ts +1359 -1359
  99. package/src/expression/core/Program.ts +985 -985
  100. package/src/expression/core/Token.ts +29 -29
  101. package/src/expression/enums/ExpressionType.ts +81 -81
  102. package/src/expression/enums/TokenType.ts +11 -11
  103. package/src/expression/exception/BreakWayException.ts +2 -2
  104. package/src/expression/exception/ContinueWayException.ts +2 -2
  105. package/src/expression/exception/ExpressionException.ts +29 -29
  106. package/src/expression/exception/ReturnWayException.ts +14 -14
  107. package/src/expression/exception/ServiceException.ts +22 -22
  108. package/src/expression/instances/JSONArray.ts +52 -52
  109. package/src/expression/instances/JSONObject.ts +118 -118
  110. package/src/expression/instances/LogicConsole.ts +31 -31
  111. package/src/font-style/font.css +4 -4
  112. package/src/hooks/useBoolean.ts +26 -0
  113. package/src/hooks/useCommon.ts +9 -9
  114. package/src/hooks/useLoading.ts +16 -0
  115. package/src/hooks/useLogin.ts +97 -97
  116. package/src/icons/svg/check-in.svg +32 -32
  117. package/src/icons/svg/dark.svg +4 -4
  118. package/src/icons/svg/github.svg +4 -4
  119. package/src/icons/svg/light.svg +4 -4
  120. package/src/icons/svg/link.svg +4 -4
  121. package/src/icons/svgo.yml +22 -22
  122. package/src/layout/GridView/index.vue +16 -16
  123. package/src/layout/PageLayout.vue +9 -9
  124. package/src/layout/SingleLayout.vue +9 -9
  125. package/src/locales/en-US.json +128 -128
  126. package/src/locales/zh-CN.json +128 -128
  127. package/src/logic/LogicRunner.ts +67 -67
  128. package/src/logic/TestLogic.ts +13 -13
  129. package/src/logic/plugins/common/DateTools.ts +35 -35
  130. package/src/logic/plugins/common/VueTools.ts +30 -30
  131. package/src/logic/plugins/index.ts +7 -7
  132. package/src/main.ts +44 -44
  133. package/src/plugins/AppData.ts +38 -38
  134. package/src/plugins/GetLoginInfoService.ts +10 -10
  135. package/src/plugins/collectIcons.ts +10 -0
  136. package/src/plugins/index.ts +11 -11
  137. package/src/router/README.md +8 -8
  138. package/src/router/guards.ts +59 -59
  139. package/src/router/index.ts +35 -35
  140. package/src/router/invoiceRoutes.ts +33 -33
  141. package/src/router/routes.ts +341 -177
  142. package/src/router/types.ts +7 -7
  143. package/src/services/api/Login.ts +6 -6
  144. package/src/services/api/common.ts +109 -109
  145. package/src/services/api/index.ts +7 -7
  146. package/src/services/api/manage.ts +8 -8
  147. package/src/services/api/search.ts +16 -16
  148. package/src/services/api/user.ts +17 -17
  149. package/src/services/restTools.ts +56 -56
  150. package/src/services/v3Api.ts +147 -147
  151. package/src/stores/index.ts +11 -11
  152. package/src/stores/modules/counter.ts +19 -19
  153. package/src/stores/modules/routeCache.ts +23 -23
  154. package/src/stores/modules/setting.ts +76 -76
  155. package/src/stores/modules/user.ts +235 -235
  156. package/src/stores/mutation-type.ts +7 -7
  157. package/src/styles/app.less +36 -36
  158. package/src/styles/login.less +109 -109
  159. package/src/styles/var.less +16 -16
  160. package/src/types/env.d.ts +16 -16
  161. package/src/types/settings.ts +1 -1
  162. package/src/types/vue-router.d.ts +9 -9
  163. package/src/utils/Storage.ts +124 -124
  164. package/src/utils/authority-utils.ts +84 -84
  165. package/src/utils/common.ts +41 -41
  166. package/src/utils/crypto.ts +39 -39
  167. package/src/utils/dataUtil.ts +42 -42
  168. package/src/utils/dictUtil.ts +52 -52
  169. package/src/utils/http/index.ts +199 -199
  170. package/src/utils/i18n.ts +72 -72
  171. package/src/utils/indexedDB.ts +195 -195
  172. package/src/utils/inline-px-to-vw.ts +28 -28
  173. package/src/utils/mobileUtil.ts +34 -34
  174. package/src/utils/progress.ts +19 -19
  175. package/src/utils/routerUtil.ts +271 -271
  176. package/src/utils/runEvalFunction.ts +13 -13
  177. package/src/utils/secureStorage.ts +71 -71
  178. package/src/utils/set-page-title.ts +5 -5
  179. package/src/utils/validate.ts +6 -6
  180. package/src/utils/wechatUtil.ts +9 -9
  181. package/src/views/chat/index.vue +153 -153
  182. package/src/views/common/LoadError.vue +63 -63
  183. package/src/views/common/NotFound.vue +67 -67
  184. package/src/views/component/EvaluateRecordView/index.vue +40 -40
  185. package/src/views/component/IconifyView/index.vue +504 -507
  186. package/src/views/component/UserDetailView/UserDetailPage.vue +77 -0
  187. package/src/views/component/UserDetailView/index.vue +234 -0
  188. package/src/views/component/XCellDetailView/index.vue +217 -217
  189. package/src/views/component/XCellListView/index.vue +108 -157
  190. package/src/views/component/XFormAppraiseView/index.vue +174 -174
  191. package/src/views/component/XFormGroupView/index.vue +78 -82
  192. package/src/views/component/XFormView/index.vue +27 -27
  193. package/src/views/component/XOlMapView/XLocationPicker/index.vue +118 -118
  194. package/src/views/component/XOlMapView/index.vue +434 -434
  195. package/src/views/component/XOlMapView/testData.ts +64 -64
  196. package/src/views/component/XReportFormIframeView/index.vue +47 -47
  197. package/src/views/component/XReportFormView/index.vue +13 -13
  198. package/src/views/component/XReportGridView/index.vue +17 -17
  199. package/src/views/component/XRequestView/index.vue +234 -234
  200. package/src/views/component/XSignatureView/index.vue +50 -50
  201. package/src/views/component/index.vue +181 -177
  202. package/src/views/component/menu.vue +117 -117
  203. package/src/views/component/notice.vue +46 -46
  204. package/src/views/component/topNav.vue +36 -36
  205. package/src/views/invoiceShow/index.vue +61 -61
  206. package/src/views/user/login/ForgetPasswordForm.vue +94 -94
  207. package/src/views/user/login/LoginForm.vue +346 -346
  208. package/src/views/user/login/LoginTitle.vue +76 -76
  209. package/src/views/user/login/LoginWave.vue +109 -109
  210. package/src/views/user/login/index.vue +22 -22
  211. package/src/views/user/my/comm/ModifyPassword.vue +346 -346
  212. package/src/views/user/my/index.vue +340 -340
  213. package/src/views/userRecords/AbnormalAlarmRecords.vue +21 -0
  214. package/src/views/userRecords/CardReplacementRecords.vue +21 -0
  215. package/src/views/userRecords/ChangeRecords.vue +19 -0
  216. package/src/views/userRecords/CommandViewRecords.vue +20 -0
  217. package/src/views/userRecords/GasCompensationRecords.vue +20 -0
  218. package/src/views/userRecords/InstrumentCollectionRecords.vue +21 -0
  219. package/src/views/userRecords/MeterRecords.vue +20 -0
  220. package/src/views/userRecords/OperateRecords.vue +51 -0
  221. package/src/views/userRecords/OtherChargeRecords.vue +19 -0
  222. package/src/views/userRecords/PaymentRecords.vue +28 -0
  223. package/src/views/userRecords/PriceAdjustmentRecords.vue +19 -0
  224. package/src/views/userRecords/ReplacementRecords.vue +19 -0
  225. package/src/views/userRecords/SafetyRecords.vue +19 -0
  226. package/src/views/userRecords/TransactionRecords.vue +21 -0
  227. package/src/views/userRecords/TransferRecords.vue +19 -0
  228. package/src/views/userRecords/operateRecordDetail/index.vue +316 -0
  229. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/AddUserDetail.vue +124 -0
  230. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/AdvanceDeliveryDetail.vue +88 -0
  231. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/AutoAccountsCancelDetail.vue +205 -0
  232. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/AutoAccountsDetail.vue +192 -0
  233. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/BankDkDetail.vue +192 -0
  234. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/BankPayDetail.vue +192 -0
  235. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/BlacklistDetail.vue +153 -0
  236. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/CancellationDetail.vue +101 -0
  237. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/CardMeterCenterCancelDetail.vue +127 -0
  238. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/CardMeterCenterDetail.vue +153 -0
  239. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/CardOverUserDetail.vue +153 -0
  240. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/ChangeMeterCancelDetail.vue +166 -0
  241. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/ChangeMeterDetail.vue +205 -0
  242. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/DisableManageDetail.vue +127 -0
  243. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/EnableManageDetail.vue +114 -0
  244. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/FaZheChangeDetail.vue +124 -0
  245. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/FeeDeductionDetail.vue +153 -0
  246. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/GasPriceChangeDetail.vue +126 -0
  247. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/InputtorChangeDetail.vue +126 -0
  248. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/IotMeterCenterCancelDetail.vue +114 -0
  249. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/IotMeterCenterDetail.vue +127 -0
  250. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/IotOpenDetail.vue +88 -0
  251. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/MachineCardDetail.vue +101 -0
  252. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/MachineMeterCenterCancelDetail.vue +218 -0
  253. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/MachineMeterCenterDetail.vue +153 -0
  254. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/OffGasAddGasDetail.vue +140 -0
  255. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/OtherChargeCancelDetail.vue +127 -0
  256. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/OtherChargeDetail.vue +114 -0
  257. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/OverUserChangeDetail.vue +127 -0
  258. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/ReBillDetail.vue +127 -0
  259. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/RefundDetail.vue +114 -0
  260. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/ReplaceCardManageCancelDetail.vue +127 -0
  261. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/ReplaceCardManageDetail.vue +114 -0
  262. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/SaleCardGasDetail.vue +140 -0
  263. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/TransferManageCancelDetail.vue +152 -0
  264. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/TransferManageDetail.vue +178 -0
  265. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/UserChangeDetail.vue +123 -0
  266. package/src/views/userRecords/operateRecordDetail/operateRecordDetails/WechatPayDetail.vue +192 -0
  267. package/src/views/userRecords/types.ts +66 -0
  268. package/tsconfig.json +39 -39
  269. package/uno.config.ts +82 -78
  270. package/vite.config.ts +118 -118
  271. package/src/views/component/XFormGroupView/xformgroup222.vue +0 -97
@@ -1,244 +1,244 @@
1
- <script setup lang="ts">
2
- import cameraIcon from '@af-mobile-client-vue3/assets/img/component/camera.png'
3
- import { deleteFile } from '@af-mobile-client-vue3/services/api/common'
4
- import { mobileUtil } from '@af-mobile-client-vue3/utils/mobileUtil'
5
- import {
6
- ActionSheet,
7
- Uploader as vanUploader,
8
- } from 'vant'
9
- import { ref } from 'vue'
10
-
11
- const props = defineProps({
12
- imageList: Array<any>,
13
- outerIndex: { default: undefined },
14
- authority: { default: 'user' },
15
- uploadMode: { default: 'server' },
16
- attr: { type: Object as () => { addOrEdit?: string, acceptCount?: number, uploadImage?: boolean }, default: () => ({}) },
17
- mode: { default: '新增' }, // 预览
18
- })
19
- const emit = defineEmits(['updateFileList'])
20
-
21
- const imageList = ref<Array<any>>(props.imageList ?? [])
22
-
23
- // 触发拍照
24
- function triggerCamera() {
25
- mobileUtil.execute({
26
- funcName: 'takePicture',
27
- param: {},
28
- callbackFunc: (result: any) => {
29
- if (result.status === 'success') {
30
- handlePhotoUpload(result.data)
31
- }
32
- },
33
- })
34
- }
35
-
36
- // 处理拍照后的上传
37
- function getImageMimeType(fileName: string): string {
38
- const ext = fileName.split('.').pop()?.toLowerCase()
39
- if (ext === 'jpg' || ext === 'jpeg')
40
- return 'image/jpeg'
41
- if (ext === 'png')
42
- return 'image/png'
43
- if (ext === 'gif')
44
- return 'image/gif'
45
- return 'image/png' // 默认
46
- }
47
-
48
- function handlePhotoUpload(photoData: any) {
49
- // 添加临时预览
50
- const mimeType = getImageMimeType(photoData.filePath)
51
- const tempFile = {
52
- uid: Date.now() + Math.random().toString(36).substr(2, 5),
53
- name: photoData.filePath.split('/').pop(),
54
- status: 'uploading',
55
- message: '上传中...',
56
- url: `data:${mimeType};base64,${photoData.content}`,
57
- }
58
-
59
- if (!imageList.value) {
60
- imageList.value = [tempFile]
61
- }
62
- else {
63
- imageList.value.push(tempFile)
64
- }
65
-
66
- const param = {
67
- resUploadMode: props.uploadMode,
68
- pathKey: 'Default',
69
- formType: 'image',
70
- useType: 'Default',
71
- resUploadStock: '1',
72
- filename: photoData.name,
73
- filesize: photoData.size,
74
- f_operator: 'server',
75
- imgPath: photoData.filePath,
76
- urlPath: `/api/${import.meta.env.VITE_APP_SYSTEM_NAME}/resource/upload`,
77
- }
78
- // 上传到服务器
79
- mobileUtil.execute({
80
- funcName: 'uploadResource',
81
- param,
82
- callbackFunc: (result: any) => {
83
- const index = imageList.value.findIndex(item => item.uid === tempFile.uid)
84
- if (result.status === 'success') {
85
- if (index !== -1) {
86
- imageList.value[index].uid = result.data.id
87
- imageList.value[index].id = result.data.id
88
- delete imageList.value[index].message
89
- imageList.value[index].status = 'done'
90
- imageList.value[index].url = result.data.f_downloadpath
91
- }
92
- }
93
- else {
94
- if (index !== -1) {
95
- imageList.value[index].status = 'failed'
96
- imageList.value[index].message = '上传失败'
97
- }
98
- }
99
-
100
- const doneIds = Object.values(imageList.value)
101
- .filter(item => item.status === 'done')
102
- .map(item => item.id)
103
-
104
- if (props.outerIndex !== undefined)
105
- emit('updateFileList', doneIds, props.outerIndex)
106
- else
107
- emit('updateFileList', doneIds)
108
- },
109
- })
110
- }
111
-
112
- // 删除图片
113
- function deleteFileFunction(file: any) {
114
- if (file.id) {
115
- deleteFile({ ids: [file.id], f_state: '删除' }).then((res: any) => {
116
- if (res.msg !== undefined) {
117
- const targetIndex = imageList.value.findIndex(item => item.id === file.id)
118
- if (targetIndex !== -1) {
119
- imageList.value.splice(targetIndex, 1)
120
- const doneIds = Object.values(imageList.value)
121
- .filter(item => item.status === 'done')
122
- .map(item => item.id)
123
- if (props.outerIndex !== undefined)
124
- emit('updateFileList', doneIds, props.outerIndex)
125
- else
126
- emit('updateFileList', doneIds)
127
- }
128
- }
129
- })
130
- }
131
- }
132
-
133
- const showActionSheet = ref(false)
134
- const uploaderRef = ref()
135
-
136
- const actionOptions = [
137
- { name: '拍照', key: 'camera' },
138
- { name: '上传', key: 'file' },
139
- ]
140
-
141
- function handleUploadAreaClick() {
142
- if (props.attr?.uploadImage) {
143
- showActionSheet.value = true
144
- }
145
- else {
146
- triggerCamera()
147
- }
148
- }
149
-
150
- function handleActionSelect(option: any) {
151
- showActionSheet.value = false
152
- if (option.key === 'camera') {
153
- triggerCamera()
154
- }
155
- else if (option.key === 'file') {
156
- mobileUtil.execute({
157
- funcName: 'photoAlbum',
158
- param: {},
159
- callbackFunc: (result: any) => {
160
- if (result.status === 'success') {
161
- result.data?.photos.forEach((photo: any) => {
162
- handlePhotoUpload(photo)
163
- })
164
- }
165
- },
166
- })
167
- }
168
- }
169
- </script>
170
-
171
- <template>
172
- <div class="uploader-container">
173
- <div
174
- v-if="props.mode !== '预览' && (imageList.length < props.attr?.acceptCount && props.attr?.addOrEdit !== 'readonly')"
175
- class="custom-upload-area"
176
- @click="handleUploadAreaClick"
177
- >
178
- <img :src="cameraIcon" alt="camera" class="upload-camera-img">
179
- <div class="upload-title">
180
- 上传附件
181
- </div>
182
- <div v-if="props.attr?.acceptCount" class="upload-desc">
183
- 已上传 {{ imageList.length }} 个,最多可上传 {{ props.attr?.acceptCount }} 个
184
- </div>
185
- </div>
186
- <van-uploader
187
- ref="uploaderRef"
188
- v-model="imageList"
189
- :show-upload="false"
190
- :deletable="(props.attr?.addOrEdit !== 'readonly' && props.authority === 'admin') && props.mode !== '预览'"
191
- :multiple="props.authority === 'admin'"
192
- :preview-image="true"
193
- :before-delete="props.attr?.addOrEdit !== 'readonly' && props.authority === 'admin' ? deleteFileFunction : undefined"
194
- />
195
- <ActionSheet
196
- v-model:show="showActionSheet"
197
- :actions="actionOptions"
198
- cancel-text="取消"
199
- @select="handleActionSelect"
200
- />
201
- </div>
202
- </template>
203
-
204
- <style scoped lang="less">
205
- .uploader-container {
206
- display: flex;
207
- flex-direction: column;
208
- gap: 16px;
209
- }
210
- .custom-upload-area {
211
- width: 100%;
212
- min-width: 220px;
213
- min-height: 100px;
214
- margin: 0 auto 12px auto;
215
- box-sizing: border-box;
216
- border: 1px dashed #d9d9d9;
217
- border-radius: 10px;
218
- background: #fff;
219
- display: flex;
220
- flex-direction: column;
221
- align-items: center;
222
- justify-content: center;
223
- cursor: pointer;
224
- padding: 12px 0 10px 0;
225
- .upload-camera-img {
226
- width: 28px;
227
- height: 28px;
228
- margin-bottom: 6px;
229
- object-fit: contain;
230
- display: block;
231
- }
232
- .upload-title {
233
- font-size: 14px;
234
- color: #bfc1c6;
235
- font-weight: 400;
236
- margin-bottom: 2px;
237
- }
238
- .upload-desc {
239
- font-size: 12px;
240
- color: #bfc1c6;
241
- margin-top: 0;
242
- }
243
- }
244
- </style>
1
+ <script setup lang="ts">
2
+ import cameraIcon from '@af-mobile-client-vue3/assets/img/component/camera.png'
3
+ import { deleteFile } from '@af-mobile-client-vue3/services/api/common'
4
+ import { mobileUtil } from '@af-mobile-client-vue3/utils/mobileUtil'
5
+ import {
6
+ ActionSheet,
7
+ Uploader as vanUploader,
8
+ } from 'vant'
9
+ import { ref } from 'vue'
10
+
11
+ const props = defineProps({
12
+ imageList: Array<any>,
13
+ outerIndex: { default: undefined },
14
+ authority: { default: 'user' },
15
+ uploadMode: { default: 'server' },
16
+ attr: { type: Object as () => { addOrEdit?: string, acceptCount?: number, uploadImage?: boolean }, default: () => ({}) },
17
+ mode: { default: '新增' }, // 预览
18
+ })
19
+ const emit = defineEmits(['updateFileList'])
20
+
21
+ const imageList = ref<Array<any>>(props.imageList ?? [])
22
+
23
+ // 触发拍照
24
+ function triggerCamera() {
25
+ mobileUtil.execute({
26
+ funcName: 'takePicture',
27
+ param: {},
28
+ callbackFunc: (result: any) => {
29
+ if (result.status === 'success') {
30
+ handlePhotoUpload(result.data)
31
+ }
32
+ },
33
+ })
34
+ }
35
+
36
+ // 处理拍照后的上传
37
+ function getImageMimeType(fileName: string): string {
38
+ const ext = fileName.split('.').pop()?.toLowerCase()
39
+ if (ext === 'jpg' || ext === 'jpeg')
40
+ return 'image/jpeg'
41
+ if (ext === 'png')
42
+ return 'image/png'
43
+ if (ext === 'gif')
44
+ return 'image/gif'
45
+ return 'image/png' // 默认
46
+ }
47
+
48
+ function handlePhotoUpload(photoData: any) {
49
+ // 添加临时预览
50
+ const mimeType = getImageMimeType(photoData.filePath)
51
+ const tempFile = {
52
+ uid: Date.now() + Math.random().toString(36).substr(2, 5),
53
+ name: photoData.filePath.split('/').pop(),
54
+ status: 'uploading',
55
+ message: '上传中...',
56
+ url: `data:${mimeType};base64,${photoData.content}`,
57
+ }
58
+
59
+ if (!imageList.value) {
60
+ imageList.value = [tempFile]
61
+ }
62
+ else {
63
+ imageList.value.push(tempFile)
64
+ }
65
+
66
+ const param = {
67
+ resUploadMode: props.uploadMode,
68
+ pathKey: 'Default',
69
+ formType: 'image',
70
+ useType: 'Default',
71
+ resUploadStock: '1',
72
+ filename: photoData.name,
73
+ filesize: photoData.size,
74
+ f_operator: 'server',
75
+ imgPath: photoData.filePath,
76
+ urlPath: `/api/${import.meta.env.VITE_APP_SYSTEM_NAME}/resource/upload`,
77
+ }
78
+ // 上传到服务器
79
+ mobileUtil.execute({
80
+ funcName: 'uploadResource',
81
+ param,
82
+ callbackFunc: (result: any) => {
83
+ const index = imageList.value.findIndex(item => item.uid === tempFile.uid)
84
+ if (result.status === 'success') {
85
+ if (index !== -1) {
86
+ imageList.value[index].uid = result.data.id
87
+ imageList.value[index].id = result.data.id
88
+ delete imageList.value[index].message
89
+ imageList.value[index].status = 'done'
90
+ imageList.value[index].url = result.data.f_downloadpath
91
+ }
92
+ }
93
+ else {
94
+ if (index !== -1) {
95
+ imageList.value[index].status = 'failed'
96
+ imageList.value[index].message = '上传失败'
97
+ }
98
+ }
99
+
100
+ const doneIds = Object.values(imageList.value)
101
+ .filter(item => item.status === 'done')
102
+ .map(item => item.id)
103
+
104
+ if (props.outerIndex !== undefined)
105
+ emit('updateFileList', doneIds, props.outerIndex)
106
+ else
107
+ emit('updateFileList', doneIds)
108
+ },
109
+ })
110
+ }
111
+
112
+ // 删除图片
113
+ function deleteFileFunction(file: any) {
114
+ if (file.id) {
115
+ deleteFile({ ids: [file.id], f_state: '删除' }).then((res: any) => {
116
+ if (res.msg !== undefined) {
117
+ const targetIndex = imageList.value.findIndex(item => item.id === file.id)
118
+ if (targetIndex !== -1) {
119
+ imageList.value.splice(targetIndex, 1)
120
+ const doneIds = Object.values(imageList.value)
121
+ .filter(item => item.status === 'done')
122
+ .map(item => item.id)
123
+ if (props.outerIndex !== undefined)
124
+ emit('updateFileList', doneIds, props.outerIndex)
125
+ else
126
+ emit('updateFileList', doneIds)
127
+ }
128
+ }
129
+ })
130
+ }
131
+ }
132
+
133
+ const showActionSheet = ref(false)
134
+ const uploaderRef = ref()
135
+
136
+ const actionOptions = [
137
+ { name: '拍照', key: 'camera' },
138
+ { name: '上传', key: 'file' },
139
+ ]
140
+
141
+ function handleUploadAreaClick() {
142
+ if (props.attr?.uploadImage) {
143
+ showActionSheet.value = true
144
+ }
145
+ else {
146
+ triggerCamera()
147
+ }
148
+ }
149
+
150
+ function handleActionSelect(option: any) {
151
+ showActionSheet.value = false
152
+ if (option.key === 'camera') {
153
+ triggerCamera()
154
+ }
155
+ else if (option.key === 'file') {
156
+ mobileUtil.execute({
157
+ funcName: 'photoAlbum',
158
+ param: {},
159
+ callbackFunc: (result: any) => {
160
+ if (result.status === 'success') {
161
+ result.data?.photos.forEach((photo: any) => {
162
+ handlePhotoUpload(photo)
163
+ })
164
+ }
165
+ },
166
+ })
167
+ }
168
+ }
169
+ </script>
170
+
171
+ <template>
172
+ <div class="uploader-container">
173
+ <div
174
+ v-if="props.mode !== '预览' && (imageList.length < props.attr?.acceptCount && props.attr?.addOrEdit !== 'readonly')"
175
+ class="custom-upload-area"
176
+ @click="handleUploadAreaClick"
177
+ >
178
+ <img :src="cameraIcon" alt="camera" class="upload-camera-img">
179
+ <div class="upload-title">
180
+ 上传附件
181
+ </div>
182
+ <div v-if="props.attr?.acceptCount" class="upload-desc">
183
+ 已上传 {{ imageList.length }} 个,最多可上传 {{ props.attr?.acceptCount }} 个
184
+ </div>
185
+ </div>
186
+ <van-uploader
187
+ ref="uploaderRef"
188
+ v-model="imageList"
189
+ :show-upload="false"
190
+ :deletable="(props.attr?.addOrEdit !== 'readonly' && props.authority === 'admin') && props.mode !== '预览'"
191
+ :multiple="props.authority === 'admin'"
192
+ :preview-image="true"
193
+ :before-delete="props.attr?.addOrEdit !== 'readonly' && props.authority === 'admin' ? deleteFileFunction : undefined"
194
+ />
195
+ <ActionSheet
196
+ v-model:show="showActionSheet"
197
+ :actions="actionOptions"
198
+ cancel-text="取消"
199
+ @select="handleActionSelect"
200
+ />
201
+ </div>
202
+ </template>
203
+
204
+ <style scoped lang="less">
205
+ .uploader-container {
206
+ display: flex;
207
+ flex-direction: column;
208
+ gap: 16px;
209
+ }
210
+ .custom-upload-area {
211
+ width: 100%;
212
+ min-width: 220px;
213
+ min-height: 100px;
214
+ margin: 0 auto 12px auto;
215
+ box-sizing: border-box;
216
+ border: 1px dashed #d9d9d9;
217
+ border-radius: 10px;
218
+ background: #fff;
219
+ display: flex;
220
+ flex-direction: column;
221
+ align-items: center;
222
+ justify-content: center;
223
+ cursor: pointer;
224
+ padding: 12px 0 10px 0;
225
+ .upload-camera-img {
226
+ width: 28px;
227
+ height: 28px;
228
+ margin-bottom: 6px;
229
+ object-fit: contain;
230
+ display: block;
231
+ }
232
+ .upload-title {
233
+ font-size: 14px;
234
+ color: #bfc1c6;
235
+ font-weight: 400;
236
+ margin-bottom: 2px;
237
+ }
238
+ .upload-desc {
239
+ font-size: 12px;
240
+ color: #bfc1c6;
241
+ margin-top: 0;
242
+ }
243
+ }
244
+ </style>
@@ -1,53 +1,53 @@
1
- <script setup lang="ts">
2
- import { rootRouteList } from '@af-mobile-client-vue3/config/routes'
3
- import { NavBar as VanNavBar } from 'vant/es'
4
- import { computed } from 'vue'
5
- import { useI18n } from 'vue-i18n'
6
- import { useRoute, useRouter } from 'vue-router'
7
-
8
- const route = useRoute()
9
- const router = useRouter()
10
- const { t } = useI18n()
11
-
12
- const showNavBar = computed(() => route.meta.navBar)
13
-
14
- /**
15
- * Get page title
16
- * Located in src/locales/json
17
- */
18
- const title = computed(() => {
19
- if (!route.meta)
20
- return ''
21
-
22
- return route.meta.i18n ? t(route.meta.i18n as string) : (route.meta.title || '')
23
- })
24
-
25
- /**
26
- * Show the left arrow
27
- * If route name is in rootRouteList, hide left arrow
28
- */
29
- const showLeftArrow = computed(() => {
30
- return !(route.name && rootRouteList.includes(route.name as string))
31
- })
32
-
33
- // back
34
- function onBack() {
35
- if (window.history.state.back)
36
- history.back()
37
- else
38
- router.replace('/')
39
- }
40
- </script>
41
-
42
- <template>
43
- <VanNavBar
44
- v-show="showNavBar"
45
- :title="title as string"
46
- :fixed="true"
47
- clickable placeholder
48
- :left-arrow="showLeftArrow"
49
- @click-left="onBack"
50
- />
51
- </template>
52
-
53
- <style scoped></style>
1
+ <script setup lang="ts">
2
+ import { rootRouteList } from '@af-mobile-client-vue3/config/routes'
3
+ import { NavBar as VanNavBar } from 'vant/es'
4
+ import { computed } from 'vue'
5
+ import { useI18n } from 'vue-i18n'
6
+ import { useRoute, useRouter } from 'vue-router'
7
+
8
+ const route = useRoute()
9
+ const router = useRouter()
10
+ const { t } = useI18n()
11
+
12
+ const showNavBar = computed(() => route.meta.navBar)
13
+
14
+ /**
15
+ * Get page title
16
+ * Located in src/locales/json
17
+ */
18
+ const title = computed(() => {
19
+ if (!route.meta)
20
+ return ''
21
+
22
+ return route.meta.i18n ? t(route.meta.i18n as string) : (route.meta.title || '')
23
+ })
24
+
25
+ /**
26
+ * Show the left arrow
27
+ * If route name is in rootRouteList, hide left arrow
28
+ */
29
+ const showLeftArrow = computed(() => {
30
+ return !(route.name && rootRouteList.includes(route.name as string))
31
+ })
32
+
33
+ // back
34
+ function onBack() {
35
+ if (window.history.state.back)
36
+ history.back()
37
+ else
38
+ router.replace('/')
39
+ }
40
+ </script>
41
+
42
+ <template>
43
+ <VanNavBar
44
+ v-show="showNavBar"
45
+ :title="title as string"
46
+ :fixed="true"
47
+ clickable placeholder
48
+ :left-arrow="showLeftArrow"
49
+ @click-left="onBack"
50
+ />
51
+ </template>
52
+
53
+ <style scoped></style>
@@ -1,32 +1,32 @@
1
- <script setup lang="ts">
2
- import { rootRouteList } from '@af-mobile-client-vue3/config/routes'
3
- import { Tabbar as VanTabbar, TabbarItem as VanTabbarItem } from 'vant'
4
- import { computed, ref } from 'vue'
5
- import { useRoute } from 'vue-router'
6
- import 'vant/lib/tabbar-item/index.css'
7
- import 'vant/lib/tabbar/index.css'
8
-
9
- const active = ref(0)
10
- const route = useRoute()
11
-
12
- const show = computed(() => {
13
- return route.name && rootRouteList.includes(route.name as string)
14
- })
15
- </script>
16
-
17
- <template>
18
- <VanTabbar v-if="show" v-model="active" placeholder route>
19
- <VanTabbarItem replace to="/Component/main">
20
- {{ '首页' }}
21
- <template #icon>
22
- <div class="i-carbon:home" />
23
- </template>
24
- </VanTabbarItem>
25
- <VanTabbarItem replace to="/user/my">
26
- {{ '我的' }}
27
- <template #icon>
28
- <div class="i-carbon:user" />
29
- </template>
30
- </VanTabbarItem>
31
- </VanTabbar>
32
- </template>
1
+ <script setup lang="ts">
2
+ import { rootRouteList } from '@af-mobile-client-vue3/config/routes'
3
+ import { Tabbar as VanTabbar, TabbarItem as VanTabbarItem } from 'vant'
4
+ import { computed, ref } from 'vue'
5
+ import { useRoute } from 'vue-router'
6
+ import 'vant/lib/tabbar-item/index.css'
7
+ import 'vant/lib/tabbar/index.css'
8
+
9
+ const active = ref(0)
10
+ const route = useRoute()
11
+
12
+ const show = computed(() => {
13
+ return route.name && rootRouteList.includes(route.name as string)
14
+ })
15
+ </script>
16
+
17
+ <template>
18
+ <VanTabbar v-if="show" v-model="active" placeholder route>
19
+ <VanTabbarItem replace to="/Component/main">
20
+ {{ '首页' }}
21
+ <template #icon>
22
+ <div class="i-carbon:home" />
23
+ </template>
24
+ </VanTabbarItem>
25
+ <VanTabbarItem replace to="/user/my">
26
+ {{ '我的' }}
27
+ <template #icon>
28
+ <div class="i-carbon:user" />
29
+ </template>
30
+ </VanTabbarItem>
31
+ </VanTabbar>
32
+ </template>