@uxda/appkit 4.3.3 → 4.3.5

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 (135) hide show
  1. package/.eslintrc.mjs +7 -7
  2. package/COMPONENT_USAGE.md +1523 -1523
  3. package/PROJECT_DOCS.md +142 -142
  4. package/README.md +187 -187
  5. package/babel.config.js +12 -12
  6. package/dist/appkit.css +63 -15
  7. package/dist/index.js +766 -254
  8. package/package.json +79 -79
  9. package/project.config.json +15 -15
  10. package/project.tt.json +13 -13
  11. package/rollup.config.mjs +78 -78
  12. package/src/Appkit.ts +72 -72
  13. package/src/balance/api/endpoints.ts +133 -133
  14. package/src/balance/api/index.ts +118 -118
  15. package/src/balance/components/AccountView.vue +770 -770
  16. package/src/balance/components/BalanceCard.vue +210 -210
  17. package/src/balance/components/BalanceReminder.vue +84 -84
  18. package/src/balance/components/ConsumptionFilter.vue +218 -218
  19. package/src/balance/components/ConsumptionRules.vue +68 -68
  20. package/src/balance/components/DateFilter.vue +259 -259
  21. package/src/balance/components/DateRange.vue +111 -111
  22. package/src/balance/components/ListFilter.vue +62 -62
  23. package/src/balance/components/ListFilterPicker.vue +191 -191
  24. package/src/balance/components/PromoterCard.vue +307 -307
  25. package/src/balance/components/SecondBalance.vue +77 -77
  26. package/src/balance/components/Tip.vue +45 -45
  27. package/src/balance/components/index.ts +8 -8
  28. package/src/balance/types.ts +99 -99
  29. package/src/components/bt-cropper/index.vue +730 -730
  30. package/src/components/bt-cropper/utils/calcCropper.js +42 -42
  31. package/src/components/bt-cropper/utils/calcImagePosition.js +23 -23
  32. package/src/components/bt-cropper/utils/calcImageSize.js +37 -37
  33. package/src/components/bt-cropper/utils/calcPointDistance.js +12 -12
  34. package/src/components/bt-cropper/utils/calcRightAndBottom.js +7 -7
  35. package/src/components/bt-cropper/utils/ratio.js +3 -3
  36. package/src/components/bt-cropper/utils/tools.js +25 -25
  37. package/src/components/dd-area/index.vue +225 -225
  38. package/src/components/dd-icon/doc.md +21 -21
  39. package/src/components/dd-icon/index.vue +23 -23
  40. package/src/components/dd-notice-bar/index.vue +78 -78
  41. package/src/components/dd-search/doc.md +34 -34
  42. package/src/components/dd-search/index.vue +168 -168
  43. package/src/components/dd-selector/index.vue +124 -124
  44. package/src/components/dd-skeleton/doc.md +19 -19
  45. package/src/components/dd-skeleton/index.vue +36 -36
  46. package/src/global.ts +6 -6
  47. package/src/index.ts +101 -101
  48. package/src/main.scss +1 -1
  49. package/src/notice/api/endpoints.ts +54 -54
  50. package/src/notice/api/index.ts +121 -121
  51. package/src/notice/components/NoticeBanner.vue +247 -247
  52. package/src/notice/components/NoticeEntry.vue +99 -99
  53. package/src/notice/components/NoticeList.vue +311 -311
  54. package/src/notice/components/NoticeList2.vue +399 -398
  55. package/src/notice/components/NoticePopup.vue +163 -163
  56. package/src/notice/components/index.ts +6 -6
  57. package/src/notice/components/useCommonList.ts +87 -86
  58. package/src/notice/components/useNotice.ts +35 -35
  59. package/src/notice/index.ts +1 -1
  60. package/src/notice/types.ts +25 -25
  61. package/src/payment/api/config.ts +7 -7
  62. package/src/payment/api/endpoints.ts +98 -96
  63. package/src/payment/api/index.ts +108 -107
  64. package/src/payment/components/AmountPicker.vue +90 -90
  65. package/src/payment/components/RechargeResult.vue +68 -69
  66. package/src/payment/components/RechargeView.vue +191 -191
  67. package/src/payment/components/RightsPicker.vue +105 -105
  68. package/src/payment/components/TradeView.vue +523 -363
  69. package/src/payment/components/UserAgreement.vue +234 -234
  70. package/src/payment/components/index.ts +22 -22
  71. package/src/payment/index.ts +5 -5
  72. package/src/payment/services/index.ts +16 -16
  73. package/src/payment/services/invoke-recharge.ts +25 -25
  74. package/src/payment/services/request-payment.ts +132 -130
  75. package/src/payment/types.ts +34 -33
  76. package/src/register/components/SelfRegistration.vue +233 -233
  77. package/src/register/components/index.ts +2 -2
  78. package/src/scenarios/components/SharePoster.vue +364 -364
  79. package/src/scenarios/components/index.ts +2 -2
  80. package/src/scenarios/components/poster-paste.vue +93 -93
  81. package/src/scenarios/components/share-poster.md +273 -273
  82. package/src/shared/components/AppDrawer.vue +53 -53
  83. package/src/shared/components/AppVerify.vue +137 -128
  84. package/src/shared/components/DeviceVersion.vue +78 -78
  85. package/src/shared/components/EmptyView.vue +33 -33
  86. package/src/shared/components/OcrBank.vue +202 -0
  87. package/src/shared/components/OcrBusinessLicense.vue +120 -137
  88. package/src/shared/components/OcrIcon.vue +267 -229
  89. package/src/shared/components/OcrInvoice.vue +218 -0
  90. package/src/shared/components/PageHeader.vue +84 -84
  91. package/src/shared/components/index.ts +10 -8
  92. package/src/shared/composables/index.ts +10 -9
  93. package/src/shared/composables/useAmount.ts +46 -46
  94. package/src/shared/composables/useCompress.ts +64 -0
  95. package/src/shared/composables/useCountdown.ts +46 -46
  96. package/src/shared/composables/useCrypto.ts +76 -76
  97. package/src/shared/composables/useDeviceEnv.ts +26 -26
  98. package/src/shared/composables/useDragBox.ts +97 -97
  99. package/src/shared/composables/useEncode.ts +43 -43
  100. package/src/shared/composables/useLogger.ts +144 -144
  101. package/src/shared/composables/useSafeArea.ts +46 -46
  102. package/src/shared/composables/useTabbar.ts +24 -24
  103. package/src/shared/composables/useUpload.ts +106 -61
  104. package/src/shared/composables/useValidator.ts +32 -32
  105. package/src/shared/composables/useWxAuth.ts +48 -48
  106. package/src/shared/http/Http.ts +149 -148
  107. package/src/shared/http/index.ts +1 -1
  108. package/src/shared/http/types.ts +163 -163
  109. package/src/shared/index.ts +9 -9
  110. package/src/shared/tracking/directives/index.ts +40 -40
  111. package/src/shared/tracking/examples/page-tracking-template.vue +27 -27
  112. package/src/shared/tracking/tracking-sdk.ts +0 -1
  113. package/src/shared/weixin/index.ts +9 -9
  114. package/src/shared/weixin/jssdk.ts +103 -103
  115. package/src/shared/weixin/payment.ts +38 -38
  116. package/src/styles/vars.scss +3 -3
  117. package/src/user/api/endpoints.ts +17 -17
  118. package/src/user/api/index.ts +123 -123
  119. package/src/user/components/LoginSetting.vue +114 -114
  120. package/src/user/components/UserAuth.vue +218 -218
  121. package/src/user/components/UserBinding.vue +277 -277
  122. package/src/user/components/UserBindingSuccess.vue +80 -80
  123. package/src/user/components/UserEntry.vue +139 -139
  124. package/src/user/components/UserFeedback.vue +428 -427
  125. package/src/user/components/UserFeedbackEntry.vue +175 -175
  126. package/src/user/components/UserHeadCrop.vue +65 -65
  127. package/src/user/components/UserInfo.vue +710 -709
  128. package/src/user/components/UserResourceEmpty.vue +75 -75
  129. package/src/user/components/index.ts +23 -23
  130. package/src/user/index.ts +1 -1
  131. package/src/utils/utils.ts +33 -33
  132. package/tsconfig.json +30 -30
  133. package/types/global.d.ts +24 -22
  134. package/types/vue.d.ts +10 -10
  135. package/dist/assets/asset-DcH8Kg-2 +0 -1
@@ -0,0 +1,218 @@
1
+ <template>
2
+ <div class="ocr-invoice" :class="[disabled ? 'disabled' : '', className]" v-track-click="'发票识别-点击'" @click="!customClick ? onUpload() : null">
3
+ <slot name="icon">
4
+ <ns-icon name="https://cdn.ddjf.com/static/images/beidouxing/ocr-icon.png"/>
5
+ </slot>
6
+ </div>
7
+
8
+ <nut-action-sheet v-model:visible="activeSheetVisible" :menu-items="actionSheetMenus" @choose="chooseImages"
9
+ cancel-txt="取消" />
10
+ </template>
11
+
12
+ <script lang="ts" setup>
13
+ import Taro, { showToast, showLoading, hideLoading, chooseMedia, chooseMessageFile, uploadFile } from '@tarojs/taro'
14
+ import { NsIcon, useNutshell } from '@uxda/nutshell/taro'
15
+ import { useAppKitOptions } from '../../Appkit'
16
+ import { ref } from 'vue';
17
+ import { useHttp } from '../../balance/api'
18
+ import { compressImage, getCompressQuality } from '../composables/useUpload'
19
+
20
+ const appKitOptions = useAppKitOptions()
21
+ const $http = useHttp(),
22
+ $n = useNutshell()
23
+
24
+ const emits = defineEmits(['complete'])
25
+
26
+ type OcrProps = {
27
+ disabled?: boolean,
28
+ side?: 'face' | 'back',
29
+ className?: string
30
+ customUpload?: Function
31
+ uploadUrl?: string
32
+ customClick?: boolean
33
+ }
34
+ type FileType = {
35
+ "downloadUrl": string,
36
+ "fileId": string,
37
+ "fileName": string,
38
+ "fileSize": number,
39
+ "fileSuffix": string,
40
+ "fileType": string,
41
+ "originalUrl": string
42
+ }
43
+
44
+ const props = withDefaults(defineProps<OcrProps>(), {
45
+ disabled: false,
46
+ side: 'face',
47
+ className: '',
48
+ uploadUrl: '/hkbase/file/uploadFile',
49
+ customClick: false
50
+ })
51
+
52
+ async function onUploadFile(csRes: any) {
53
+ try {
54
+ let { path, size, tempFilePath } = csRes.tempFiles[0]
55
+ const compressImg: any = (await compressImage(path || tempFilePath, getCompressQuality(size))) || {}
56
+ const filePath = compressImg.tempFilePath || path
57
+
58
+ if (props.customUpload) {
59
+ props.customUpload(filePath)
60
+ return
61
+ }
62
+
63
+ showLoading({ title: '发票识别中..', mask: true })
64
+
65
+ const session = appKitOptions.token()
66
+ const baseUrl = appKitOptions.baseUrl()
67
+ const upRes: any = await uploadFile({
68
+ url: `${baseUrl}${props.uploadUrl}`,
69
+ filePath,
70
+ name: 'file',
71
+ formData: {
72
+ objectNo: `min${Date.now()}`,
73
+ appCode: appKitOptions.app(),
74
+ },
75
+ header: {
76
+ token: session || '',
77
+ },
78
+ })
79
+
80
+ const res = JSON.parse(upRes.data)
81
+ if (res.code === '200') {
82
+ await getOcrInfo(res.result)
83
+ } else {
84
+ hideLoading()
85
+ showToast({
86
+ title: res.msg,
87
+ icon: 'error',
88
+ })
89
+ }
90
+ } catch (err) {
91
+ hideLoading()
92
+ console.log(err)
93
+ }
94
+ }
95
+
96
+ // 根据证件路径获取证件信息
97
+ async function getOcrInfo(file: string | FileType) {
98
+ try {
99
+ const res: any = await $http.get('/hkbase/common/vatInvoice', {
100
+ fileUrl: typeof file === 'string' ? file : file.originalUrl,
101
+ fileType: 'img'
102
+ })
103
+ hideLoading()
104
+
105
+ if (!res?.purchaserRegisterNum) {
106
+ $n.dialog({
107
+ title: '识别失败',
108
+ message: `您上传的图片可能不够清晰或与当前功能不符,请重新上传一张清晰、完整的图片。谢谢!`,
109
+ okText: '我知道了',
110
+ cancelText: '',
111
+ })
112
+ return
113
+ }
114
+
115
+ emits('complete', {
116
+ invoiceDate: res?.invoiceDate,
117
+ invoiceNum: res?.invoiceNum,
118
+ invoiceNumConfirm: res?.invoiceNumConfirm,
119
+ invoiceType: res?.invoiceType,
120
+ noteDrawer: res?.noteDrawer,
121
+ purchaserBank: res?.purchaserBank,
122
+ purchaserName: res?.purchaserName,
123
+ purchaserRegisterNum: res?.purchaserRegisterNum,
124
+ remarks: res?.remarks,
125
+ sellerName: res?.sellerName,
126
+ sellerRegisterNum: res?.sellerRegisterNum,
127
+ serviceType: res?.serviceType,
128
+ totalAmount: res?.totalAmount,
129
+ totalTax: res?.totalTax,
130
+ fileUrl: typeof file === 'string' ? file : file.originalUrl,
131
+ fileKey: typeof file === 'string' ? file : file.fileId,
132
+ })
133
+ } catch (err) {
134
+ hideLoading()
135
+ }
136
+ }
137
+
138
+ // 上传图片操作面板
139
+ const activeSheetVisible = ref(false)
140
+ const actionSheetMenus = [
141
+ {
142
+ name: '拍摄',
143
+ type: 'camera',
144
+ },
145
+ {
146
+ name: '从相册选择',
147
+ type: 'album',
148
+ },
149
+ {
150
+ name: '从聊天会话选择',
151
+ type: 'message',
152
+ },
153
+ ]
154
+ if (Taro.getEnv() === 'WEB') {
155
+ actionSheetMenus.pop()
156
+ }
157
+
158
+ // 选择图像上传
159
+ async function chooseImages(item: any) {
160
+ if (['camera', 'album'].includes(item.type)) {
161
+ const csRes = await chooseMedia({
162
+ count: 1,
163
+ sourceType: [item.type], // "camera" | "album"
164
+ maxDuration: 60, // 使用duration属性判断是图片还是视频,图片没有该属性
165
+ })
166
+
167
+ onUploadFile(csRes)
168
+ } else {
169
+ const csRes = await chooseMessageFile({
170
+ count: 1,
171
+ type: 'image',
172
+ })
173
+
174
+ onUploadFile(csRes)
175
+ }
176
+ }
177
+
178
+ async function onUpload() {
179
+ if (props.disabled) return
180
+
181
+ if (Taro.getEnv() === 'WEB') {
182
+ const csRes = await chooseMedia({
183
+ count: 1,
184
+ sourceType: ['album'], // "camera" | "album"
185
+ maxDuration: 60, // 使用duration属性判断是图片还是视频,图片没有该属性
186
+ })
187
+
188
+ onUploadFile(csRes)
189
+ return
190
+ }
191
+
192
+ activeSheetVisible.value = true
193
+ }
194
+
195
+ defineExpose({
196
+ onUpload
197
+ })
198
+ </script>
199
+
200
+ <style lang="scss">
201
+ .ocr-invoice {
202
+ width: 24px;
203
+ height: 24px;
204
+ display: inline-flex;
205
+ align-items: center;
206
+
207
+ .ns-icon {
208
+ width: 24px;
209
+ height: 24px;
210
+ }
211
+
212
+ &.disabled {
213
+ .ns-icon {
214
+ filter: brightness(1.5) grayscale(1);
215
+ }
216
+ }
217
+ }
218
+ </style>
@@ -1,85 +1,85 @@
1
- <template>
2
- <div :class="[
3
- 'page-header',
4
- `color-mode-${colorMode}`
5
- ]" :style="cssVars">
6
- <h1 class="page-title" v-if="title">{{ title }}</h1>
7
- <view class="back-button" v-track-click="'页面头部-返回'" @click="onBackButtonClick"></view>
8
- <slot></slot>
9
- </div>
10
- </template>
11
-
12
- <script lang="ts" setup>
13
- import { useSafeArea } from '../composables'
14
-
15
-
16
- const safeArea = useSafeArea()
17
-
18
- const cssVars = {
19
- '--top': `${safeArea.status}px`,
20
- '--height': `${safeArea.nav}px`
21
- }
22
-
23
- export type PageHeaderColorMode = 'light' | 'dark'
24
-
25
- export interface PageHeaderProps {
26
- title?: string,
27
- colorMode?: PageHeaderColorMode
28
- }
29
-
30
- const emit = defineEmits<{
31
- (event: 'close'): void
32
- }>()
33
-
34
- withDefaults(
35
- defineProps<PageHeaderProps>(), {
36
- title: '',
37
- colorMode: 'light'
38
- }
39
- )
40
-
41
- const onBackButtonClick = () => {
42
- emit('close')
43
- }
44
- </script>
45
-
46
- <style lang="scss">
47
- .page-header {
48
- display: flex;
49
- position: relative;
50
- align-items: center;
51
- padding-top: var(--top);
52
- min-height: var(--height);
53
-
54
- .page-title {
55
- position: absolute;
56
- font-size: 18px;
57
- height: var(--height);
58
- line-height: var(--height);
59
- top: var(--top);
60
- width: 100%;
61
- text-align: center;
62
- margin: 0;
63
- }
64
-
65
- .back-button {
66
- position: absolute;
67
- top: var(--top);
68
- left: 10px;
69
- width: var(--height);
70
- height: var(--height);
71
- background-image: url("data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDIiIGhlaWdodD0iNDIiIHZpZXdCb3g9IjAgMCA0MiA0MiIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPHBhdGggb3BhY2l0eT0iMC4wMSIgZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0wLjk5OTg3OCA0MUw0MC45OTk5IDQxTDQwLjk5OTkgMUwwLjk5OTg3OCAxTDAuOTk5ODc4IDQxWiIgZmlsbD0iI0Q4RDhEOCIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPHBhdGggZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0xMi45OTMzIDE4Ljk0NzdDMTEuODYzMyAyMC4wMTQ1IDExLjgxMiAyMS43OTU0IDEyLjg3ODggMjIuOTI1NUwyNy45MTA1IDM3LjEzNDVDMjguNjkyNSAzNy44NzM0IDI5LjkyNTUgMzcuODM4NCAzMC42NjQ0IDM3LjA1NjRDMzEuMzY4MSAzNi4zMTE2IDMxLjM2OTkgMzUuMTU3OSAzMC42OTI5IDM0LjQxMTRMMTYuNTAxMiAyMC45OTM5TDMwLjU4NTcgNy42OTc4NUMzMS4zMzA4IDYuOTk0NDkgMzEuMjk4MiA1LjcyNzk1IDMwLjY2NDkgNC45NDQwMkMyOS45NjE2IDQuMTk4OTUgMjguODA5OCA0LjEzMTI5IDI4LjAyNTkgNC43NjQ1MkwxMi45OTMzIDE4Ljk0NzdaIiBmaWxsPSJibGFjayIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPC9zdmc+Cg==");
72
- background-size: 50%;
73
- background-repeat: no-repeat;
74
- background-position: center center;
75
- }
76
-
77
- &.color-mode-dark {
78
- color: var(--text-color-dark-mode, #fff);
79
-
80
- .back-button {
81
- background-image: url("data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDIiIGhlaWdodD0iNDIiIHZpZXdCb3g9IjAgMCA0MiA0MiIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPHBhdGggb3BhY2l0eT0iMC4wMSIgZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0wLjk5OTg0NyA0MUw0MC45OTk4IDQxTDQwLjk5OTggMUwwLjk5OTg0NyAxTDAuOTk5ODQ3IDQxWiIgZmlsbD0iI0Q4RDhEOCIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPHBhdGggZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0xMi45OTM0IDE4Ljk0NzVDMTEuODYzNCAyMC4wMTQzIDExLjgxMjEgMjEuNzk1MiAxMi44Nzg5IDIyLjkyNTJMMjcuOTEwNiAzNy4xMzQzQzI4LjY5MjcgMzcuODczMSAyOS45MjU2IDM3LjgzODIgMzAuNjY0NSAzNy4wNTYyQzMxLjM2ODIgMzYuMzExNCAzMS4zNyAzNS4xNTc2IDMwLjY5MyAzNC40MTExTDE2LjUwMTMgMjAuOTkzNkwzMC41ODU4IDcuNjk3NjFDMzEuMzMwOSA2Ljk5NDI0IDMxLjI5ODMgNS43Mjc3MSAzMC42NjUxIDQuOTQzNzdDMjkuOTYxNyA0LjE5ODcxIDI4LjgwOTkgNC4xMzEwNCAyOC4wMjYgNC43NjQyN0wxMi45OTM0IDE4Ljk0NzVaIiBmaWxsPSJ3aGl0ZSIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPC9zdmc+Cg==");
82
- }
83
- }
84
- }
1
+ <template>
2
+ <div :class="[
3
+ 'page-header',
4
+ `color-mode-${colorMode}`
5
+ ]" :style="cssVars">
6
+ <h1 class="page-title" v-if="title">{{ title }}</h1>
7
+ <view class="back-button" v-track-click="'页面头部-返回'" @click="onBackButtonClick"></view>
8
+ <slot></slot>
9
+ </div>
10
+ </template>
11
+
12
+ <script lang="ts" setup>
13
+ import { useSafeArea } from '../composables'
14
+
15
+
16
+ const safeArea = useSafeArea()
17
+
18
+ const cssVars = {
19
+ '--top': `${safeArea.status}px`,
20
+ '--height': `${safeArea.nav}px`
21
+ }
22
+
23
+ export type PageHeaderColorMode = 'light' | 'dark'
24
+
25
+ export interface PageHeaderProps {
26
+ title?: string,
27
+ colorMode?: PageHeaderColorMode
28
+ }
29
+
30
+ const emit = defineEmits<{
31
+ (event: 'close'): void
32
+ }>()
33
+
34
+ withDefaults(
35
+ defineProps<PageHeaderProps>(), {
36
+ title: '',
37
+ colorMode: 'light'
38
+ }
39
+ )
40
+
41
+ const onBackButtonClick = () => {
42
+ emit('close')
43
+ }
44
+ </script>
45
+
46
+ <style lang="scss">
47
+ .page-header {
48
+ display: flex;
49
+ position: relative;
50
+ align-items: center;
51
+ padding-top: var(--top);
52
+ min-height: var(--height);
53
+
54
+ .page-title {
55
+ position: absolute;
56
+ font-size: 18px;
57
+ height: var(--height);
58
+ line-height: var(--height);
59
+ top: var(--top);
60
+ width: 100%;
61
+ text-align: center;
62
+ margin: 0;
63
+ }
64
+
65
+ .back-button {
66
+ position: absolute;
67
+ top: var(--top);
68
+ left: 10px;
69
+ width: var(--height);
70
+ height: var(--height);
71
+ background-image: url("data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDIiIGhlaWdodD0iNDIiIHZpZXdCb3g9IjAgMCA0MiA0MiIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPHBhdGggb3BhY2l0eT0iMC4wMSIgZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0wLjk5OTg3OCA0MUw0MC45OTk5IDQxTDQwLjk5OTkgMUwwLjk5OTg3OCAxTDAuOTk5ODc4IDQxWiIgZmlsbD0iI0Q4RDhEOCIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPHBhdGggZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0xMi45OTMzIDE4Ljk0NzdDMTEuODYzMyAyMC4wMTQ1IDExLjgxMiAyMS43OTU0IDEyLjg3ODggMjIuOTI1NUwyNy45MTA1IDM3LjEzNDVDMjguNjkyNSAzNy44NzM0IDI5LjkyNTUgMzcuODM4NCAzMC42NjQ0IDM3LjA1NjRDMzEuMzY4MSAzNi4zMTE2IDMxLjM2OTkgMzUuMTU3OSAzMC42OTI5IDM0LjQxMTRMMTYuNTAxMiAyMC45OTM5TDMwLjU4NTcgNy42OTc4NUMzMS4zMzA4IDYuOTk0NDkgMzEuMjk4MiA1LjcyNzk1IDMwLjY2NDkgNC45NDQwMkMyOS45NjE2IDQuMTk4OTUgMjguODA5OCA0LjEzMTI5IDI4LjAyNTkgNC43NjQ1MkwxMi45OTMzIDE4Ljk0NzdaIiBmaWxsPSJibGFjayIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPC9zdmc+Cg==");
72
+ background-size: 50%;
73
+ background-repeat: no-repeat;
74
+ background-position: center center;
75
+ }
76
+
77
+ &.color-mode-dark {
78
+ color: var(--text-color-dark-mode, #fff);
79
+
80
+ .back-button {
81
+ background-image: url("data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDIiIGhlaWdodD0iNDIiIHZpZXdCb3g9IjAgMCA0MiA0MiIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KPHBhdGggb3BhY2l0eT0iMC4wMSIgZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0wLjk5OTg0NyA0MUw0MC45OTk4IDQxTDQwLjk5OTggMUwwLjk5OTg0NyAxTDAuOTk5ODQ3IDQxWiIgZmlsbD0iI0Q4RDhEOCIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPHBhdGggZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0xMi45OTM0IDE4Ljk0NzVDMTEuODYzNCAyMC4wMTQzIDExLjgxMjEgMjEuNzk1MiAxMi44Nzg5IDIyLjkyNTJMMjcuOTEwNiAzNy4xMzQzQzI4LjY5MjcgMzcuODczMSAyOS45MjU2IDM3LjgzODIgMzAuNjY0NSAzNy4wNTYyQzMxLjM2ODIgMzYuMzExNCAzMS4zNyAzNS4xNTc2IDMwLjY5MyAzNC40MTExTDE2LjUwMTMgMjAuOTkzNkwzMC41ODU4IDcuNjk3NjFDMzEuMzMwOSA2Ljk5NDI0IDMxLjI5ODMgNS43Mjc3MSAzMC42NjUxIDQuOTQzNzdDMjkuOTYxNyA0LjE5ODcxIDI4LjgwOTkgNC4xMzEwNCAyOC4wMjYgNC43NjQyN0wxMi45OTM0IDE4Ljk0NzVaIiBmaWxsPSJ3aGl0ZSIgc3Ryb2tlPSJibGFjayIgc3Ryb2tlLW9wYWNpdHk9IjAuMDEiLz4KPC9zdmc+Cg==");
82
+ }
83
+ }
84
+ }
85
85
  </style>
@@ -1,8 +1,10 @@
1
- import AppDrawer from './AppDrawer.vue'
2
- import PageHeader from './PageHeader.vue'
3
- import AppVerify from './AppVerify.vue'
4
- import DeviceVersion from './DeviceVersion.vue'
5
- import OcrIcon from './OcrIcon.vue'
6
- import OcrBusinessLicense from './OcrBusinessLicense.vue'
7
-
8
- export { AppDrawer, PageHeader, DeviceVersion, AppVerify, OcrIcon, OcrBusinessLicense }
1
+ import AppDrawer from './AppDrawer.vue'
2
+ import PageHeader from './PageHeader.vue'
3
+ import AppVerify from './AppVerify.vue'
4
+ import DeviceVersion from './DeviceVersion.vue'
5
+ import OcrIcon from './OcrIcon.vue'
6
+ import OcrBank from './OcrBank.vue'
7
+ import OcrBusinessLicense from './OcrBusinessLicense.vue'
8
+ import OcrInvoice from './OcrInvoice.vue'
9
+
10
+ export { AppDrawer, PageHeader, DeviceVersion, AppVerify, OcrIcon, OcrBank,OcrBusinessLicense, OcrInvoice }
@@ -1,9 +1,10 @@
1
- export * from './useSafeArea'
2
- export * from './useTabbar'
3
- export * from './useCountdown'
4
- export * from './useValidator'
5
- export * from './useEncode'
6
- export * from './useUpload'
7
- export * from './useCrypto'
8
- export * from './useLogger'
9
- export * from './useWxAuth'
1
+ export * from './useSafeArea'
2
+ export * from './useTabbar'
3
+ export * from './useCountdown'
4
+ export * from './useValidator'
5
+ export * from './useEncode'
6
+ export * from './useUpload'
7
+ export * from './useCrypto'
8
+ export * from './useLogger'
9
+ export * from './useWxAuth'
10
+ export * from './useCompress'
@@ -1,46 +1,46 @@
1
- /**
2
- * 金额通用方法
3
- */
4
-
5
- export function useAmount() {
6
- /**
7
- * 限制数字小数位
8
- */
9
- function limitDecimal(value, precision = 1) {
10
- if (!value) return value
11
-
12
- value = String(value)
13
-
14
- // 限制最多precision位小数
15
- if (value.includes('.')) {
16
- value = value.slice(0, value.indexOf('.') + (precision + 1))
17
- }
18
-
19
- // 移除非数字字符
20
- value = value.replace(/[^0-9.]/g, '')
21
-
22
- // 防止出现多个小数点
23
- value = value.replace(/\.{2,}/g, '.')
24
-
25
- return value
26
- }
27
-
28
- /**
29
- * 格式化金额
30
- */
31
- function formatAmount(value: string | number, digits = 2) {
32
- if (!value) {
33
- return '0'
34
- }
35
-
36
- return value.toLocaleString('en-US', {
37
- minimumFractionDigits: 0,
38
- maximumFractionDigits: digits,
39
- })
40
- }
41
-
42
- return {
43
- limitDecimal,
44
- formatAmount,
45
- }
46
- }
1
+ /**
2
+ * 金额通用方法
3
+ */
4
+
5
+ export function useAmount() {
6
+ /**
7
+ * 限制数字小数位
8
+ */
9
+ function limitDecimal(value, precision = 1) {
10
+ if (!value) return value
11
+
12
+ value = String(value)
13
+
14
+ // 限制最多precision位小数
15
+ if (value.includes('.')) {
16
+ value = value.slice(0, value.indexOf('.') + (precision + 1))
17
+ }
18
+
19
+ // 移除非数字字符
20
+ value = value.replace(/[^0-9.]/g, '')
21
+
22
+ // 防止出现多个小数点
23
+ value = value.replace(/\.{2,}/g, '.')
24
+
25
+ return value
26
+ }
27
+
28
+ /**
29
+ * 格式化金额
30
+ */
31
+ function formatAmount(value: string | number, digits = 2) {
32
+ if (!value) {
33
+ return '0'
34
+ }
35
+
36
+ return value.toLocaleString('en-US', {
37
+ minimumFractionDigits: 0,
38
+ maximumFractionDigits: digits,
39
+ })
40
+ }
41
+
42
+ return {
43
+ limitDecimal,
44
+ formatAmount,
45
+ }
46
+ }
@@ -0,0 +1,64 @@
1
+ import Taro, { uploadFile } from '@tarojs/taro'
2
+ import { Media } from '@uxda/nutshell/taro'
3
+ import { useAppKitOptions } from '../../Appkit'
4
+
5
+ export type UploadConfig = {
6
+ baseUrl: string
7
+ name?: string
8
+ headers?: Record<string, string>
9
+ }
10
+
11
+ const mappings: { [x: string]: keyof Media } = {
12
+ downloadUrl: 'thrumb',
13
+ fileId: 'id',
14
+ fileName: 'name',
15
+ fileSize: 'size',
16
+ fileSuffix: 'ext',
17
+ fileType: 'type',
18
+ originalUrl: 'url',
19
+ }
20
+
21
+ const transformFields = (row: any) => {
22
+ return Object.fromEntries(Object.entries(row).map(([k, v]) => [mappings[k] || k, v]))
23
+ }
24
+
25
+ type UploadFunction = (url: string, file: Media) => Promise<Media>
26
+
27
+ export const useUpload = (config: UploadConfig) => {
28
+ const appkitOptions = useAppKitOptions()
29
+
30
+ // 上传文件
31
+ const upload: UploadFunction = (url: string, file: Media) => {
32
+ return new Promise<Media>((resolve, reject) => {
33
+ uploadFile({
34
+ url: config.baseUrl + url,
35
+ filePath: file.path!,
36
+ name: 'file',
37
+ formData: {
38
+ objectNo: `min${Date.now()}`,
39
+ appCode: config.headers?.appcode || appkitOptions.app(),
40
+ },
41
+ header: {
42
+ ...config.headers,
43
+ token: appkitOptions.tempToken() || appkitOptions.token(),
44
+ },
45
+ success: (rsp: any) => {
46
+ const { data } = rsp
47
+ try {
48
+ const response = JSON.parse(data)
49
+ console.log('===response', response)
50
+ resolve(transformFields(response.result))
51
+ } catch (e) {
52
+ reject({
53
+ message: '文件上传异常',
54
+ })
55
+ }
56
+ },
57
+ })
58
+ })
59
+ }
60
+
61
+ return {
62
+ upload,
63
+ }
64
+ }
@@ -1,46 +1,46 @@
1
- /**
2
- * @description: 倒计时
3
- * @param {number} time
4
- * @return {*}
5
- */
6
- import { ref, onUnmounted, computed } from 'vue'
7
-
8
- export function useCountdown() {
9
- let timer: any = null
10
- const countdown = ref(60)
11
- const countdownActive = computed(() => countdown.value > 0 && countdown.value < 60)
12
-
13
- const startCountdown = () => {
14
- countdown.value = 60
15
- timer = setInterval(() => {
16
- countdown.value--
17
- if (countdown.value <= 0) {
18
- stopCountdown()
19
- }
20
- }, 1000)
21
- }
22
-
23
- const stopCountdown = () => {
24
- clearInterval(timer)
25
- countdown.value = 60 // 重置倒计时
26
- }
27
-
28
- onUnmounted(() => {
29
- // stopCountdown() // 组件卸载时确保清除定时器
30
- })
31
-
32
- const countdownText = computed(() => {
33
- if (countdown.value === 60) {
34
- return '获取验证码'
35
- } else {
36
- return `重新发送(${countdown.value}s)`
37
- }
38
- })
39
-
40
- return {
41
- countdownActive,
42
- countdownText,
43
- startCountdown,
44
- stopCountdown,
45
- }
46
- }
1
+ /**
2
+ * @description: 倒计时
3
+ * @param {number} time
4
+ * @return {*}
5
+ */
6
+ import { ref, onUnmounted, computed } from 'vue'
7
+
8
+ export function useCountdown() {
9
+ let timer: any = null
10
+ const countdown = ref(60)
11
+ const countdownActive = computed(() => countdown.value > 0 && countdown.value < 60)
12
+
13
+ const startCountdown = () => {
14
+ countdown.value = 60
15
+ timer = setInterval(() => {
16
+ countdown.value--
17
+ if (countdown.value <= 0) {
18
+ stopCountdown()
19
+ }
20
+ }, 1000)
21
+ }
22
+
23
+ const stopCountdown = () => {
24
+ clearInterval(timer)
25
+ countdown.value = 60 // 重置倒计时
26
+ }
27
+
28
+ onUnmounted(() => {
29
+ // stopCountdown() // 组件卸载时确保清除定时器
30
+ })
31
+
32
+ const countdownText = computed(() => {
33
+ if (countdown.value === 60) {
34
+ return '获取验证码'
35
+ } else {
36
+ return `重新发送(${countdown.value}s)`
37
+ }
38
+ })
39
+
40
+ return {
41
+ countdownActive,
42
+ countdownText,
43
+ startCountdown,
44
+ stopCountdown,
45
+ }
46
+ }