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
@@ -0,0 +1,620 @@
1
+ <script setup lang="ts">
2
+ import type { RecordEntry } from './recordEntries'
3
+ import type { BaseUser, ConfigItem } from './types'
4
+ import useLoading from '@af-mobile-client-vue3/hooks/useLoading'
5
+ import { mobileUtil } from '@af-mobile-client-vue3/utils/mobileUtil'
6
+ import { Button as VanButton, Empty as VanEmpty, Icon as VanIcon, Loading as VanLoading } from 'vant'
7
+ import { computed, ref, watch } from 'vue'
8
+ import { useRouter } from 'vue-router'
9
+ import InfoDisplay from '../InfoDisplay/index.vue'
10
+ import { getCacheUserDetail, getRecentBusinessTime } from './api'
11
+ import { defaultRecordEntries } from './recordEntries'
12
+
13
+ interface Props {
14
+ userInfoId: string // 用户ID(必传)
15
+ showRecentTime?: boolean // 是否展示历史时间,默认false
16
+ recordEntries?: RecordEntry[] // 记录入口配置(可选)
17
+ businessButtonText?: string // 业务办理按钮文本,默认"业务办理"
18
+ showBottomButtons?: boolean // 是否显示底部按钮区域,默认false
19
+ showHeader?: boolean // 是否显示头部导航栏,默认false
20
+ headerTitle?: string // 头部标题,默认"用户档案详情"
21
+ getUserDetailApi?: (id: string) => Promise<BaseUser> // 自定义获取用户详情接口
22
+ getRecentTimeApi?: (id: string) => Promise<Record<string, string>> // 自定义获取历史时间接口
23
+ }
24
+
25
+ interface Emits {
26
+ (e: 'close'): void
27
+ (e: 'recordClick', entry: RecordEntry, user: BaseUser): void
28
+ (e: 'businessClick', user: BaseUser): void
29
+ (e: 'print', user: BaseUser): void
30
+ }
31
+
32
+ const props = withDefaults(defineProps<Props>(), {
33
+ showRecentTime: false,
34
+ businessButtonText: '业务办理',
35
+ showBottomButtons: false,
36
+ showHeader: false,
37
+ headerTitle: '用户档案详情',
38
+ recordEntries: () => defaultRecordEntries,
39
+ })
40
+
41
+ const emit = defineEmits<Emits>()
42
+ const router = useRouter()
43
+
44
+ // 用户数据
45
+ const user = ref<BaseUser | null>(null)
46
+ const { loading: userLoading, startLoading: startUserLoading, endLoading: endUserLoading } = useLoading(false)
47
+ const userError = ref<string | null>(null)
48
+
49
+ // 控制用户信息展开收起
50
+ const showUserInfo = ref(false)
51
+
52
+ // 最近记录
53
+ const recentRecord = ref<Record<string, string>>({})
54
+ const { loading: recentRecordLoading, startLoading: startRecentRecordLoading, endLoading: endRecentRecordLoading } = useLoading(false)
55
+
56
+ // 状态类名
57
+ const statusClass = computed(() => {
58
+ if (!user.value?.f_user_state)
59
+ return ''
60
+
61
+ switch (user.value.f_user_state) {
62
+ case '正常':
63
+ return 'status-badge--normal'
64
+ case '欠费':
65
+ return 'status-badge--overdue'
66
+ case '暂停':
67
+ return 'status-badge--suspended'
68
+ default:
69
+ return ''
70
+ }
71
+ })
72
+
73
+ // 用户详细信息配置
74
+ const userDetailConfig = computed<ConfigItem[]>(() => {
75
+ if (!user.value)
76
+ return []
77
+
78
+ return [
79
+ { label: '联系电话', field: 'f_user_phone' },
80
+ { label: '用户类型', field: 'f_user_type' },
81
+ { label: '用户地址', field: 'f_address', full: true },
82
+ { label: '表具类型', field: 'f_meter_type' },
83
+ { label: '表具编号', field: 'f_meternumber' },
84
+ {
85
+ label: '卡号',
86
+ field: 'f_card_id',
87
+ condition: data => data.f_meter_type?.includes('卡表') || data.f_hascard === '是',
88
+ },
89
+ {
90
+ label: '账户余额',
91
+ field: 'f_balance',
92
+ format: value => `¥ ${value}`,
93
+ condition: data => !data.f_meter_type?.includes('物联网表') || (data.f_meter_type?.includes('物联网表') && data.f_user_balance > 0),
94
+ },
95
+ {
96
+ label: '表上余额',
97
+ field: 'f_balance_amount',
98
+ condition: data => data.f_meter_type?.includes('物联网表'),
99
+ },
100
+ { label: '购气次数', field: 'f_times' },
101
+ {
102
+ label: '累计购气(m³)',
103
+ field: 'f_total_gas',
104
+ format: value => `${value}m³`,
105
+ },
106
+ {
107
+ label: '阀控状态',
108
+ field: 'f_valve_state',
109
+ condition: data => data.f_meter_type?.includes('物联网表'),
110
+ },
111
+ ].filter(item => !item.condition || item.condition(user.value))
112
+ })
113
+
114
+ // 筛选符合当前用户表具类型的记录入口
115
+ const filteredRecordEntries = computed(() => {
116
+ if (!user.value?.f_meter_type)
117
+ return []
118
+
119
+ return props.recordEntries.filter(entry =>
120
+ entry.forMeterTypes.includes(user.value!.f_meter_type),
121
+ )
122
+ })
123
+
124
+ // 获取用户详情
125
+ async function fetchUserDetail() {
126
+ if (!props.userInfoId)
127
+ return
128
+
129
+ try {
130
+ startUserLoading()
131
+ userError.value = null
132
+ const api = props.getUserDetailApi || getCacheUserDetail
133
+ user.value = await api(props.userInfoId)
134
+ }
135
+ catch (error) {
136
+ console.error('获取用户详情失败:', error)
137
+ userError.value = error instanceof Error ? error.message : '获取用户详情失败'
138
+ user.value = null
139
+ }
140
+ finally {
141
+ endUserLoading()
142
+ }
143
+ }
144
+
145
+ // 重试获取用户详情
146
+ function retryFetchUserDetail() {
147
+ fetchUserDetail()
148
+ }
149
+
150
+ // 获取最近记录
151
+ async function fetchRecentRecord() {
152
+ if (!props.showRecentTime || !props.userInfoId)
153
+ return
154
+
155
+ try {
156
+ startRecentRecordLoading()
157
+ const api = props.getRecentTimeApi || getRecentBusinessTime
158
+ recentRecord.value = await api(props.userInfoId)
159
+ }
160
+ catch (error) {
161
+ console.error('获取最近记录失败:', error)
162
+ recentRecord.value = {}
163
+ }
164
+ finally {
165
+ endRecentRecordLoading()
166
+ }
167
+ }
168
+
169
+ // 查看记录详情
170
+ function viewRecordDetail(entry: RecordEntry) {
171
+ if (!user.value)
172
+ return
173
+ router.push({
174
+ name: entry.route,
175
+ query: {
176
+ userName: user.value.f_user_name,
177
+ userId: user.value.f_userinfo_id,
178
+ userfilesId: user.value.f_userfiles_id,
179
+ },
180
+ })
181
+ }
182
+
183
+ // 打开业务办理
184
+ function openBusinessHandler() {
185
+ if (!user.value)
186
+ return
187
+ emit('businessClick', user.value)
188
+ }
189
+
190
+ // 打印档案
191
+ function printProfile() {
192
+ if (!user.value)
193
+ return
194
+
195
+ // 默认打印实现
196
+ const printContent = [
197
+ { type: 3, text: '自 助 购 气 凭 证', fontsize: 3, isbold: true, align: 'center' },
198
+ { type: 3, text: '----------------------------', fontsize: 3, isbold: true, align: 'center' },
199
+ { type: 3, text: `用户编号:${user.value.f_userinfo_code}`, fontsize: 2, isbold: true, align: 'left' },
200
+ { type: 3, text: `用户姓名:${user.value.f_user_name}`, fontsize: 2, isbold: true, align: 'left' },
201
+ { type: 3, text: `用户地址:${user.value.f_address}`, fontsize: 2, isbold: true, align: 'left' },
202
+ { type: 3, text: '', fontsize: 3, isbold: true, align: 'center' },
203
+ { type: 4, value: 5, unit: 'pixel' },
204
+ ]
205
+
206
+ mobileUtil.execute({
207
+ funcName: 'print',
208
+ param: { data: printContent },
209
+ callbackFunc: (result) => {
210
+ console.log('打印结果:', result)
211
+ },
212
+ })
213
+
214
+ // 同时触发自定义打印事件
215
+ emit('print', user.value)
216
+ }
217
+
218
+ // 监听用户ID变化
219
+ watch(() => props.userInfoId, async (newId) => {
220
+ if (newId) {
221
+ await fetchUserDetail()
222
+ if (props.showRecentTime) {
223
+ await fetchRecentRecord()
224
+ }
225
+ }
226
+ }, { immediate: true })
227
+ </script>
228
+
229
+ <template>
230
+ <div class="user-detail" :class="{ 'has-header': props.showHeader }">
231
+ <!-- 头部导航栏 -->
232
+ <div v-if="props.showHeader" class="user-detail__header">
233
+ <div class="header-left">
234
+ <button type="button" class="back-button" @click="emit('close')">
235
+ <VanIcon name="arrow-left" />
236
+ </button>
237
+ <h3 class="header-title">
238
+ {{ props.headerTitle }}
239
+ </h3>
240
+ </div>
241
+ </div>
242
+
243
+ <!-- 加载状态 -->
244
+ <div v-if="userLoading" class="loading-container">
245
+ <VanLoading size="24px" vertical>
246
+ 加载中...
247
+ </VanLoading>
248
+ </div>
249
+
250
+ <!-- 错误状态 -->
251
+ <div v-else-if="userError" class="error-container">
252
+ <VanEmpty
253
+ image="error"
254
+ :description="userError"
255
+ >
256
+ <VanButton type="primary" @click="retryFetchUserDetail">
257
+ 重试
258
+ </VanButton>
259
+ </VanEmpty>
260
+ </div>
261
+
262
+ <!-- 用户详情内容 -->
263
+ <div v-else-if="user" class="user-detail__content">
264
+ <!-- 用户基本信息卡片 -->
265
+ <div class="user-info-card">
266
+ <div
267
+ class="user-info-header"
268
+ @click="showUserInfo = !showUserInfo"
269
+ >
270
+ <div class="user-info-header__left">
271
+ <div class="user-avatar bg-blue-100">
272
+ <i class="i-fa6-solid-user text-blue-500" />
273
+ </div>
274
+ <div class="user-base-info">
275
+ <h3 class="user-name">
276
+ {{ user.f_user_name }}
277
+ </h3>
278
+ <p class="user-id">
279
+ {{ user.f_userinfo_code }}
280
+ </p>
281
+ </div>
282
+ </div>
283
+ <div class="user-info-header__right">
284
+ <span class="status-badge" :class="statusClass">{{ user.f_user_state }}</span>
285
+ <i :class="showUserInfo ? 'i-fa6-solid-chevron-up' : 'i-fa6-solid-chevron-down'" class="text-gray-400" />
286
+ </div>
287
+ </div>
288
+ <div v-show="showUserInfo" class="user-info-body">
289
+ <InfoDisplay :config="userDetailConfig" :data="user" />
290
+ </div>
291
+ </div>
292
+
293
+ <!-- 交易记录入口 -->
294
+ <div class="records-container">
295
+ <div
296
+ v-for="entry in filteredRecordEntries"
297
+ :key="entry.id"
298
+ class="record-card"
299
+ @click="viewRecordDetail(entry)"
300
+ >
301
+ <div class="record-card__icon" :class="[entry.bgColor]">
302
+ <i :class="[entry.icon, entry.textColor]" />
303
+ </div>
304
+ <div class="record-card__content">
305
+ <h3 class="record-title">
306
+ {{ entry.title }}
307
+ </h3>
308
+ <p class="record-date">
309
+ 最近{{ entry.dateLabel }}:
310
+ <template v-if="recentRecordLoading">
311
+ <i class="i-fa6-solid-spinner animate-spin" />
312
+ </template>
313
+ <template v-else>
314
+ {{ recentRecord[entry.dateLabel] || '无' }}
315
+ </template>
316
+ </p>
317
+ </div>
318
+ <i class="i-fa6-solid-chevron-right text-gray-400" />
319
+ </div>
320
+ </div>
321
+
322
+ <!-- 底部按钮 -->
323
+ <div v-if="props.showBottomButtons" class="fixed-bottom-buttons">
324
+ <!-- 自定义底部插槽 -->
325
+ <slot name="bottom" :user="user">
326
+ <!-- 默认底部按钮 -->
327
+ <VanButton
328
+ class="print-button"
329
+ plain
330
+ type="default"
331
+ @click="printProfile"
332
+ >
333
+ <i class="i-fa6-solid-print mr-1" />
334
+ 打印档案
335
+ </VanButton>
336
+ <VanButton
337
+ class="business-button"
338
+ type="primary"
339
+ @click="openBusinessHandler"
340
+ >
341
+ <i class="i-fa-solid-file-invoice" />
342
+ {{ businessButtonText }}
343
+ </VanButton>
344
+ </slot>
345
+ </div>
346
+ </div>
347
+ </div>
348
+ </template>
349
+
350
+ <style lang="less" scoped>
351
+ .user-detail {
352
+ position: relative;
353
+ background-color: #f5f7fa;
354
+ min-height: 100vh;
355
+ padding-bottom: 70px;
356
+
357
+ &__header {
358
+ position: fixed;
359
+ top: 0;
360
+ left: 0;
361
+ right: 0;
362
+ z-index: 10;
363
+ display: flex;
364
+ justify-content: space-between;
365
+ align-items: center;
366
+ padding: 12px 16px;
367
+ background-color: #fff;
368
+ box-shadow: 0 1px 2px rgba(0, 0, 0, 0.05);
369
+ }
370
+
371
+ &__content {
372
+ padding: 16px;
373
+ }
374
+
375
+ // 当有 header 时,为 content 腾出空间
376
+ &.has-header &__content {
377
+ padding-top: calc(16px + 46px); // 16px原有padding + 56px header高度
378
+ }
379
+ }
380
+
381
+ // 加载状态
382
+ .loading-container {
383
+ display: flex;
384
+ justify-content: center;
385
+ align-items: center;
386
+ height: 300px;
387
+ background-color: #fff;
388
+ margin: 16px;
389
+ border-radius: 8px;
390
+ }
391
+
392
+ // 错误状态
393
+ .error-container {
394
+ display: flex;
395
+ justify-content: center;
396
+ align-items: center;
397
+ min-height: 300px;
398
+ background-color: #fff;
399
+ margin: 16px;
400
+ border-radius: 8px;
401
+ padding: 32px 16px;
402
+ }
403
+
404
+ // 当有 header 时,调整加载和错误状态的位置
405
+ .has-header {
406
+ .loading-container,
407
+ .error-container {
408
+ margin-top: calc(16px + 56px); // header高度 + 原有margin
409
+ }
410
+ }
411
+
412
+ .header-left {
413
+ display: flex;
414
+ align-items: center;
415
+ }
416
+
417
+ .header-title {
418
+ font-size: 18px;
419
+ font-weight: 600;
420
+ color: #333;
421
+ margin: 0;
422
+ }
423
+
424
+ .back-button {
425
+ background: none;
426
+ border: none;
427
+ padding: 4px;
428
+ margin-right: 12px;
429
+ display: flex;
430
+ align-items: center;
431
+ justify-content: center;
432
+ color: #666;
433
+ cursor: pointer;
434
+
435
+ .van-icon {
436
+ font-size: 22px;
437
+ }
438
+ }
439
+
440
+ // 用户信息卡片
441
+ .user-info-card {
442
+ background-color: #fff;
443
+ border-radius: 8px;
444
+ box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
445
+ margin-bottom: 16px;
446
+ overflow: hidden;
447
+ }
448
+
449
+ .user-info-header {
450
+ display: flex;
451
+ justify-content: space-between;
452
+ align-items: center;
453
+ padding: 16px;
454
+ cursor: pointer;
455
+
456
+ &__left {
457
+ display: flex;
458
+ align-items: center;
459
+ }
460
+
461
+ &__right {
462
+ display: flex;
463
+ align-items: center;
464
+ gap: 12px;
465
+ }
466
+ }
467
+
468
+ .user-avatar {
469
+ width: 48px;
470
+ height: 48px;
471
+ border-radius: 50%;
472
+ background-color: #e6f7ff;
473
+ display: flex;
474
+ align-items: center;
475
+ justify-content: center;
476
+ margin-right: 12px;
477
+
478
+ .van-icon {
479
+ font-size: 24px;
480
+ color: var(--van-primary-color);
481
+ }
482
+ }
483
+
484
+ .user-base-info {
485
+ .user-name {
486
+ font-size: 16px;
487
+ font-weight: 600;
488
+ color: #333;
489
+ margin: 0 0 4px 0;
490
+ }
491
+
492
+ .user-id {
493
+ font-size: 13px;
494
+ color: #999;
495
+ margin: 0;
496
+ }
497
+ }
498
+
499
+ .status-badge {
500
+ display: inline-block;
501
+ padding: 2px 10px;
502
+ border-radius: 100px;
503
+ font-size: 12px;
504
+ font-weight: 500;
505
+
506
+ &--normal {
507
+ background-color: #e6ffed;
508
+ color: #52c41a;
509
+ }
510
+
511
+ &--overdue {
512
+ background-color: #fff1f0;
513
+ color: #f5222d;
514
+ }
515
+
516
+ &--suspended {
517
+ background-color: #fff7e6;
518
+ color: #fa8c16;
519
+ }
520
+ }
521
+
522
+ .user-info-body {
523
+ padding: 16px 16px;
524
+ border-top: 1px solid #f0f0f0;
525
+ }
526
+
527
+ // 记录入口卡片
528
+ .records-container {
529
+ display: flex;
530
+ flex-direction: column;
531
+ gap: 12px;
532
+ }
533
+
534
+ .record-card {
535
+ display: flex;
536
+ align-items: center;
537
+ background-color: #fff;
538
+ border-radius: 8px;
539
+ padding: 16px;
540
+ box-shadow: 0 1px 2px rgba(0, 0, 0, 0.05);
541
+ cursor: pointer;
542
+
543
+ &__icon {
544
+ width: 40px;
545
+ height: 40px;
546
+ border-radius: 50%;
547
+ display: flex;
548
+ align-items: center;
549
+ justify-content: center;
550
+ margin-right: 12px;
551
+ }
552
+
553
+ &__content {
554
+ flex: 1;
555
+ min-width: 0;
556
+ }
557
+ }
558
+
559
+ .record-title {
560
+ font-size: 15px;
561
+ font-weight: 500;
562
+ color: #333;
563
+ margin: 0 0 4px 0;
564
+ }
565
+
566
+ .record-date {
567
+ font-size: 12px;
568
+ color: #999;
569
+ margin: 0;
570
+ }
571
+
572
+ // 底部按钮
573
+ .fixed-bottom-buttons {
574
+ position: fixed;
575
+ bottom: 0;
576
+ left: 0;
577
+ right: 0;
578
+ display: flex;
579
+ justify-content: space-between;
580
+ padding: 12px 16px;
581
+ background-color: #fff;
582
+ box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.05);
583
+ z-index: 10;
584
+ gap: 12px;
585
+
586
+ .print-button,
587
+ .business-button {
588
+ flex: 1;
589
+ height: 44px;
590
+ font-size: 15px;
591
+ font-weight: 500;
592
+ border-radius: 8px;
593
+
594
+ .van-icon {
595
+ font-size: 18px;
596
+ margin-right: 4px;
597
+ vertical-align: -3px;
598
+ }
599
+ }
600
+
601
+ .business-button {
602
+ background-color: var(--van-primary-color);
603
+ border-color: var(--van-primary-color);
604
+ box-shadow: 0 2px 6px rgba(24, 144, 255, 0.3);
605
+ }
606
+ }
607
+
608
+ .animate-spin {
609
+ animation: spin 0.5s linear infinite;
610
+ }
611
+
612
+ @keyframes spin {
613
+ from {
614
+ transform: rotate(0deg);
615
+ }
616
+ to {
617
+ transform: rotate(360deg);
618
+ }
619
+ }
620
+ </style>