@uxda/appkit 4.3.11 → 4.3.13
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.
- package/dist/appkit.css +105 -12
- package/dist/index.js +1097 -407
- package/package.json +2 -5
- package/src/balance/api/index.ts +1 -1
- package/src/balance/components/PromoterCard.vue +1 -0
- package/src/notice/api/index.ts +1 -1
- package/src/notice/components/NoticeList2.vue +234 -73
- package/src/notice/components/useCommonList.ts +1 -0
- package/src/payment/api/endpoints.ts +2 -0
- package/src/payment/api/index.ts +2 -1
- package/src/payment/components/RechargeResult.vue +0 -1
- package/src/payment/components/TradeView.vue +329 -121
- package/src/payment/services/request-payment.ts +2 -0
- package/src/payment/types.ts +1 -0
- package/src/register/components/SelfRegistration.vue +1 -1
- package/src/shared/components/AppVerify.vue +15 -6
- package/src/shared/components/OcrBank.vue +202 -0
- package/src/shared/components/OcrBusinessLicense.vue +20 -37
- package/src/shared/components/OcrIcon.vue +105 -67
- package/src/shared/components/OcrInvoice.vue +322 -0
- package/src/shared/components/index.ts +3 -1
- package/src/shared/composables/index.ts +1 -0
- package/src/shared/composables/useCompress.ts +64 -0
- package/src/shared/composables/useUpload.ts +96 -51
- package/src/shared/http/Http.ts +1 -0
- package/src/shared/tracking/tracking-sdk.ts +0 -1
- package/src/shared/weixin/jssdk.ts +1 -0
- package/src/user/api/index.ts +1 -1
- package/src/user/components/UserAuth.vue +1 -1
- package/src/user/components/UserFeedback.vue +1 -0
- package/src/user/components/UserInfo.vue +2 -0
- package/types/global.d.ts +2 -0
|
@@ -0,0 +1,322 @@
|
|
|
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
|
+
/**
|
|
53
|
+
* 判断文件类型
|
|
54
|
+
*/
|
|
55
|
+
function getFileType(filePath: string, fileName?: string): 'img' | 'pdf' {
|
|
56
|
+
const lowerPath = filePath.toLowerCase()
|
|
57
|
+
const lowerName = (fileName || '').toLowerCase()
|
|
58
|
+
|
|
59
|
+
// 检查文件扩展名
|
|
60
|
+
if (lowerPath.endsWith('.pdf') || lowerName.endsWith('.pdf')) {
|
|
61
|
+
return 'pdf'
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
// 检查是否为图片格式
|
|
65
|
+
const imageExtensions = ['.jpg', '.jpeg', '.png', '.gif', '.bmp', '.webp']
|
|
66
|
+
if (imageExtensions.some(ext => lowerPath.endsWith(ext) || lowerName.endsWith(ext))) {
|
|
67
|
+
return 'img'
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// 默认为图片
|
|
71
|
+
return 'img'
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
async function onUploadFile(csRes: any, fileType?: 'img' | 'pdf') {
|
|
75
|
+
try {
|
|
76
|
+
let { path, size, tempFilePath, name } = csRes.tempFiles[0]
|
|
77
|
+
const originalPath = path || tempFilePath
|
|
78
|
+
|
|
79
|
+
// 判断文件类型
|
|
80
|
+
const detectedFileType = fileType || getFileType(originalPath, name)
|
|
81
|
+
|
|
82
|
+
// 只有图片才进行压缩
|
|
83
|
+
let filePath = originalPath
|
|
84
|
+
if (detectedFileType === 'img') {
|
|
85
|
+
const compressImg: any = (await compressImage(originalPath, getCompressQuality(size))) || {}
|
|
86
|
+
filePath = compressImg.tempFilePath || originalPath
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
if (props.customUpload) {
|
|
90
|
+
props.customUpload(filePath, detectedFileType)
|
|
91
|
+
return
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
showLoading({ title: '发票识别中..', mask: true })
|
|
95
|
+
|
|
96
|
+
const session = appKitOptions.token()
|
|
97
|
+
const baseUrl = appKitOptions.baseUrl()
|
|
98
|
+
const upRes: any = await uploadFile({
|
|
99
|
+
url: `${baseUrl}${props.uploadUrl}`,
|
|
100
|
+
filePath,
|
|
101
|
+
name: 'file',
|
|
102
|
+
formData: {
|
|
103
|
+
objectNo: `min${Date.now()}`,
|
|
104
|
+
appCode: appKitOptions.app(),
|
|
105
|
+
},
|
|
106
|
+
header: {
|
|
107
|
+
token: session || '',
|
|
108
|
+
},
|
|
109
|
+
})
|
|
110
|
+
|
|
111
|
+
const res = JSON.parse(upRes.data)
|
|
112
|
+
if (res.code === '200') {
|
|
113
|
+
await getOcrInfo(res.result, detectedFileType)
|
|
114
|
+
} else {
|
|
115
|
+
hideLoading()
|
|
116
|
+
showToast({
|
|
117
|
+
title: res.msg,
|
|
118
|
+
icon: 'error',
|
|
119
|
+
})
|
|
120
|
+
}
|
|
121
|
+
} catch (err) {
|
|
122
|
+
hideLoading()
|
|
123
|
+
console.log(err)
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// 根据证件路径获取证件信息
|
|
128
|
+
async function getOcrInfo(file: string | FileType, fileType: 'img' | 'pdf' = 'img') {
|
|
129
|
+
try {
|
|
130
|
+
const fileUrl = typeof file === 'string' ? file : file.originalUrl
|
|
131
|
+
const res: any = await $http.get('/hkbase/common/vatInvoice', {
|
|
132
|
+
fileUrl,
|
|
133
|
+
fileType: fileType
|
|
134
|
+
})
|
|
135
|
+
hideLoading()
|
|
136
|
+
|
|
137
|
+
if (!res?.purchaserRegisterNum) {
|
|
138
|
+
$n.dialog({
|
|
139
|
+
title: '识别失败',
|
|
140
|
+
message: `您上传的${fileType === 'pdf' ? '文件' : '图片'}可能不够清晰或与当前功能不符,请重新上传一张清晰、完整的${fileType === 'pdf' ? 'PDF文件' : '图片'}。谢谢!`,
|
|
141
|
+
okText: '我知道了',
|
|
142
|
+
cancelText: '',
|
|
143
|
+
})
|
|
144
|
+
return
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
emits('complete', {
|
|
148
|
+
invoiceDate: res?.invoiceDate,
|
|
149
|
+
invoiceNum: res?.invoiceNum,
|
|
150
|
+
invoiceNumConfirm: res?.invoiceNumConfirm,
|
|
151
|
+
invoiceType: res?.invoiceType,
|
|
152
|
+
noteDrawer: res?.noteDrawer,
|
|
153
|
+
purchaserBank: res?.purchaserBank,
|
|
154
|
+
purchaserName: res?.purchaserName,
|
|
155
|
+
purchaserRegisterNum: res?.purchaserRegisterNum,
|
|
156
|
+
remarks: res?.remarks,
|
|
157
|
+
sellerName: res?.sellerName,
|
|
158
|
+
sellerRegisterNum: res?.sellerRegisterNum,
|
|
159
|
+
serviceType: res?.serviceType,
|
|
160
|
+
totalAmount: res?.totalAmount,
|
|
161
|
+
totalTax: res?.totalTax,
|
|
162
|
+
fileUrl: fileUrl,
|
|
163
|
+
fileType: fileType,
|
|
164
|
+
fileKey: typeof file === 'string' ? file : file.fileId,
|
|
165
|
+
})
|
|
166
|
+
} catch (err) {
|
|
167
|
+
hideLoading()
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// 上传图片操作面板
|
|
172
|
+
const activeSheetVisible = ref(false)
|
|
173
|
+
const actionSheetMenus = [
|
|
174
|
+
{
|
|
175
|
+
name: '拍摄',
|
|
176
|
+
type: 'camera',
|
|
177
|
+
},
|
|
178
|
+
{
|
|
179
|
+
name: '从相册选择',
|
|
180
|
+
type: 'album',
|
|
181
|
+
},
|
|
182
|
+
{
|
|
183
|
+
name: '从聊天会话选择',
|
|
184
|
+
type: 'message',
|
|
185
|
+
},
|
|
186
|
+
{
|
|
187
|
+
name: '选择文件',
|
|
188
|
+
type: 'file',
|
|
189
|
+
},
|
|
190
|
+
]
|
|
191
|
+
if (Taro.getEnv() === 'WEB') {
|
|
192
|
+
// Web 端移除"从聊天会话选择",保留"选择文件"
|
|
193
|
+
actionSheetMenus.splice(2, 1)
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// 选择图像上传
|
|
197
|
+
async function chooseImages(item: any) {
|
|
198
|
+
if (['camera', 'album'].includes(item.type)) {
|
|
199
|
+
const csRes = await chooseMedia({
|
|
200
|
+
count: 1,
|
|
201
|
+
sourceType: [item.type], // "camera" | "album"
|
|
202
|
+
maxDuration: 60, // 使用duration属性判断是图片还是视频,图片没有该属性
|
|
203
|
+
})
|
|
204
|
+
|
|
205
|
+
onUploadFile(csRes)
|
|
206
|
+
} else if (item.type === 'message') {
|
|
207
|
+
const csRes = await chooseMessageFile({
|
|
208
|
+
count: 1,
|
|
209
|
+
type: 'image',
|
|
210
|
+
})
|
|
211
|
+
|
|
212
|
+
onUploadFile(csRes)
|
|
213
|
+
} else if (item.type === 'file') {
|
|
214
|
+
// 选择文件(包括PDF)
|
|
215
|
+
const csRes = await chooseMessageFile({
|
|
216
|
+
count: 1,
|
|
217
|
+
type: 'file',
|
|
218
|
+
})
|
|
219
|
+
|
|
220
|
+
// 不预先指定类型,让 onUploadFile 根据文件名自动判断
|
|
221
|
+
onUploadFile(csRes)
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
/**
|
|
226
|
+
* Web 端选择文件(支持图片和PDF)
|
|
227
|
+
*/
|
|
228
|
+
function chooseFileInWeb() {
|
|
229
|
+
return new Promise<void>((resolve, reject) => {
|
|
230
|
+
const input = document.createElement('input')
|
|
231
|
+
input.type = 'file'
|
|
232
|
+
input.accept = 'image/*,.pdf'
|
|
233
|
+
input.style.display = 'none'
|
|
234
|
+
|
|
235
|
+
input.onchange = async (e: any) => {
|
|
236
|
+
const file = e.target?.files?.[0]
|
|
237
|
+
if (!file) {
|
|
238
|
+
document.body.removeChild(input)
|
|
239
|
+
resolve()
|
|
240
|
+
return
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
// 根据文件 MIME 类型和扩展名判断文件类型
|
|
244
|
+
let fileType: 'img' | 'pdf' = 'img'
|
|
245
|
+
if (file.type === 'application/pdf' || file.name.toLowerCase().endsWith('.pdf')) {
|
|
246
|
+
fileType = 'pdf'
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
const filePath = URL.createObjectURL(file)
|
|
250
|
+
|
|
251
|
+
try {
|
|
252
|
+
// 构造类似 chooseMedia 返回的数据结构
|
|
253
|
+
const csRes = {
|
|
254
|
+
tempFiles: [{
|
|
255
|
+
path: filePath,
|
|
256
|
+
tempFilePath: filePath,
|
|
257
|
+
size: file.size,
|
|
258
|
+
name: file.name,
|
|
259
|
+
}]
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
await onUploadFile(csRes, fileType)
|
|
263
|
+
resolve()
|
|
264
|
+
} catch (error) {
|
|
265
|
+
reject(error)
|
|
266
|
+
} finally {
|
|
267
|
+
document.body.removeChild(input)
|
|
268
|
+
// 清理 blob URL
|
|
269
|
+
URL.revokeObjectURL(filePath)
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
input.oncancel = () => {
|
|
274
|
+
document.body.removeChild(input)
|
|
275
|
+
resolve()
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
document.body.appendChild(input)
|
|
279
|
+
input.click()
|
|
280
|
+
})
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
async function onUpload() {
|
|
284
|
+
if (props.disabled) return
|
|
285
|
+
|
|
286
|
+
if (Taro.getEnv() === 'WEB') {
|
|
287
|
+
// Web 端弹出选择:图片或PDF
|
|
288
|
+
try {
|
|
289
|
+
await chooseFileInWeb()
|
|
290
|
+
} catch (err) {
|
|
291
|
+
console.error('文件选择失败:', err)
|
|
292
|
+
}
|
|
293
|
+
return
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
activeSheetVisible.value = true
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
defineExpose({
|
|
300
|
+
onUpload
|
|
301
|
+
})
|
|
302
|
+
</script>
|
|
303
|
+
|
|
304
|
+
<style lang="scss">
|
|
305
|
+
.ocr-invoice {
|
|
306
|
+
width: 24px;
|
|
307
|
+
height: 24px;
|
|
308
|
+
display: inline-flex;
|
|
309
|
+
align-items: center;
|
|
310
|
+
|
|
311
|
+
.ns-icon {
|
|
312
|
+
width: 24px;
|
|
313
|
+
height: 24px;
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
&.disabled {
|
|
317
|
+
.ns-icon {
|
|
318
|
+
filter: brightness(1.5) grayscale(1);
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
</style>
|
|
@@ -3,6 +3,8 @@ import PageHeader from './PageHeader.vue'
|
|
|
3
3
|
import AppVerify from './AppVerify.vue'
|
|
4
4
|
import DeviceVersion from './DeviceVersion.vue'
|
|
5
5
|
import OcrIcon from './OcrIcon.vue'
|
|
6
|
+
import OcrBank from './OcrBank.vue'
|
|
6
7
|
import OcrBusinessLicense from './OcrBusinessLicense.vue'
|
|
8
|
+
import OcrInvoice from './OcrInvoice.vue'
|
|
7
9
|
|
|
8
|
-
export { AppDrawer, PageHeader, DeviceVersion, AppVerify, OcrIcon, OcrBusinessLicense }
|
|
10
|
+
export { AppDrawer, PageHeader, DeviceVersion, AppVerify, OcrIcon, OcrBank,OcrBusinessLicense, OcrInvoice }
|
|
@@ -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,61 +1,106 @@
|
|
|
1
|
-
import 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
|
-
}
|
|
1
|
+
import Taro from "@tarojs/taro";
|
|
10
2
|
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
3
|
+
/**
|
|
4
|
+
* 使用 canvas 压缩图片(Web 端)
|
|
5
|
+
*/
|
|
6
|
+
export const compressImageWithCanvas = (
|
|
7
|
+
src: string,
|
|
8
|
+
quality: number
|
|
9
|
+
): Promise<string> => {
|
|
10
|
+
return new Promise((resolve, reject) => {
|
|
11
|
+
const img = new Image();
|
|
12
|
+
img.crossOrigin = "anonymous";
|
|
20
13
|
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
14
|
+
img.onload = () => {
|
|
15
|
+
try {
|
|
16
|
+
const canvas = document.createElement("canvas");
|
|
17
|
+
const ctx = canvas.getContext("2d");
|
|
24
18
|
|
|
25
|
-
|
|
19
|
+
if (!ctx) {
|
|
20
|
+
reject(new Error("无法获取 canvas 上下文"));
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
26
23
|
|
|
27
|
-
|
|
28
|
-
|
|
24
|
+
// 设置画布尺寸(可以根据需要限制最大尺寸)
|
|
25
|
+
const maxWidth = 850;
|
|
26
|
+
const maxHeight = 850;
|
|
27
|
+
let width = img.width;
|
|
28
|
+
let height = img.height;
|
|
29
29
|
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
30
|
+
// 计算压缩后的尺寸
|
|
31
|
+
if (width > maxWidth || height > maxHeight) {
|
|
32
|
+
const ratio = Math.min(maxWidth / width, maxHeight / height);
|
|
33
|
+
width = width * ratio;
|
|
34
|
+
height = height * ratio;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
canvas.width = width;
|
|
38
|
+
canvas.height = height;
|
|
39
|
+
|
|
40
|
+
// 绘制图片到 canvas
|
|
41
|
+
ctx.drawImage(img, 0, 0, width, height);
|
|
42
|
+
|
|
43
|
+
// 转换为 blob
|
|
44
|
+
canvas.toBlob(
|
|
45
|
+
(blob) => {
|
|
46
|
+
if (!blob) {
|
|
47
|
+
reject(new Error("图片压缩失败"));
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
// 创建 blob URL
|
|
51
|
+
const compressedUrl = URL.createObjectURL(blob);
|
|
52
|
+
resolve(compressedUrl);
|
|
53
|
+
},
|
|
54
|
+
"image/jpeg",
|
|
55
|
+
quality / 100
|
|
56
|
+
);
|
|
57
|
+
} catch (error) {
|
|
58
|
+
reject(error);
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
img.onerror = () => {
|
|
63
|
+
reject(new Error("图片加载失败"));
|
|
64
|
+
};
|
|
65
|
+
|
|
66
|
+
img.src = src;
|
|
67
|
+
});
|
|
68
|
+
};
|
|
69
|
+
|
|
70
|
+
// 压缩图片
|
|
71
|
+
export async function compressImage(src: string, quality = 80) {
|
|
72
|
+
if (Taro.getEnv() === "WEB") {
|
|
73
|
+
try {
|
|
74
|
+
const compressedUrl = await compressImageWithCanvas(src, quality);
|
|
75
|
+
// 返回格式与小程序端一致,包含 tempFilePath 属性
|
|
76
|
+
return { tempFilePath: compressedUrl };
|
|
77
|
+
} catch (error) {
|
|
78
|
+
console.error("图片压缩失败:", error);
|
|
79
|
+
// 压缩失败时返回原图
|
|
80
|
+
return { tempFilePath: src };
|
|
81
|
+
}
|
|
82
|
+
} else {
|
|
83
|
+
return new Promise((resolve, reject) => {
|
|
84
|
+
Taro.compressImage({
|
|
85
|
+
src: src,
|
|
86
|
+
quality: quality,
|
|
87
|
+
success: (res) => {
|
|
88
|
+
resolve(res);
|
|
42
89
|
},
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
try {
|
|
46
|
-
const response = JSON.parse(data)
|
|
47
|
-
console.log('===response', response)
|
|
48
|
-
resolve(transformFields(response.result))
|
|
49
|
-
} catch (e) {
|
|
50
|
-
reject({
|
|
51
|
-
message: '文件上传异常',
|
|
52
|
-
})
|
|
53
|
-
}
|
|
90
|
+
fail: (res) => {
|
|
91
|
+
reject(res);
|
|
54
92
|
},
|
|
55
|
-
})
|
|
56
|
-
})
|
|
93
|
+
});
|
|
94
|
+
});
|
|
57
95
|
}
|
|
58
|
-
|
|
59
|
-
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// 获取压缩质量
|
|
99
|
+
export function getCompressQuality(size: number) {
|
|
100
|
+
let quality = 100;
|
|
101
|
+
const curSize = size / (1024 * 1024);
|
|
102
|
+
if (curSize > 6) {
|
|
103
|
+
quality = quality - ((curSize - 6) / curSize) * 100;
|
|
60
104
|
}
|
|
105
|
+
return quality;
|
|
61
106
|
}
|
package/src/shared/http/Http.ts
CHANGED
package/src/user/api/index.ts
CHANGED
|
@@ -65,7 +65,7 @@ function useHttp() {
|
|
|
65
65
|
const headers: any = {
|
|
66
66
|
Token: appkitOptions.tempToken() || appkitOptions.token(),
|
|
67
67
|
Appcode: appkitOptions.app(),
|
|
68
|
-
|
|
68
|
+
tid: appkitOptions.tenant(),
|
|
69
69
|
gray: appkitOptions.gray ? appkitOptions.gray() : '0',
|
|
70
70
|
}
|
|
71
71
|
|
|
@@ -6,7 +6,7 @@
|
|
|
6
6
|
<ns-input label="你的姓名" name="姓名" v-model="formData.姓名" placeholder="请输入或拍照识别" :maxlength="30" :disabled="已认证"
|
|
7
7
|
:rules="['required']">
|
|
8
8
|
<template #append>
|
|
9
|
-
<ocr-icon v-if="!已认证" @complete="onOcrComplete" />
|
|
9
|
+
<ocr-icon v-if="!已认证" :has-upload-vo="false" @complete="onOcrComplete" />
|
|
10
10
|
</template>
|
|
11
11
|
</ns-input>
|
|
12
12
|
<ns-input label="身份证号码" name="身份证号码" placeholder="请输入身份证号码" :maxlength="30" v-model="formData.身份证号码"
|
|
@@ -220,6 +220,7 @@ async function toUpload() {
|
|
|
220
220
|
async function updateImage(filePath: string) {
|
|
221
221
|
showLoading({
|
|
222
222
|
title: "上传中...",
|
|
223
|
+
mask: true
|
|
223
224
|
});
|
|
224
225
|
const appkitOptions = useAppKitOptions();
|
|
225
226
|
const $http = useHttp();
|
|
@@ -359,6 +360,7 @@ function toLogout() {
|
|
|
359
360
|
title: "提示",
|
|
360
361
|
content: "确定要退出登录吗?",
|
|
361
362
|
confirmText: "确定",
|
|
363
|
+
confirmColor: '#017fff',
|
|
362
364
|
success: async (e: any) => {
|
|
363
365
|
if (e.confirm) {
|
|
364
366
|
emits("logout");
|