Pre Merge pull request !662 from shixiaohe/master
commit
f84030a1f4
|
|
@ -71,21 +71,74 @@ type FileTypes =
|
||||||
// 接受父组件参数
|
// 接受父组件参数
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
modelValue: propTypes.string.def(''),
|
modelValue: propTypes.string.def(''),
|
||||||
drag: propTypes.bool.def(true), // 是否支持拖拽上传 ==> 非必传(默认为 true)
|
drag: propTypes.bool.def(true),
|
||||||
disabled: propTypes.bool.def(false), // 是否禁用上传组件 ==> 非必传(默认为 false)
|
disabled: propTypes.bool.def(false),
|
||||||
fileSize: propTypes.number.def(5), // 图片大小限制 ==> 非必传(默认为 5M)
|
fileSize: propTypes.number.def(5),
|
||||||
fileType: propTypes.array.def(['image/jpeg', 'image/png', 'image/gif']), // 图片类型限制 ==> 非必传(默认为 ["image/jpeg", "image/png", "image/gif"])
|
fileType: propTypes.array.def(['image/jpeg', 'image/png', 'image/gif']),
|
||||||
height: propTypes.string.def('150px'), // 组件高度 ==> 非必传(默认为 150px)
|
height: propTypes.string.def('150px'),
|
||||||
width: propTypes.string.def('150px'), // 组件宽度 ==> 非必传(默认为 150px)
|
width: propTypes.string.def('150px'),
|
||||||
borderradius: propTypes.string.def('8px'), // 组件边框圆角 ==> 非必传(默认为 8px)
|
borderradius: propTypes.string.def('8px'),
|
||||||
showDelete: propTypes.bool.def(true), // 是否显示删除按钮
|
showDelete: propTypes.bool.def(true),
|
||||||
showBtnText: propTypes.bool.def(true) // 是否显示按钮文字
|
showBtnText: propTypes.bool.def(true),
|
||||||
|
// 新增压缩相关属性
|
||||||
|
enableCompress: propTypes.bool.def(false), // 是否启用压缩
|
||||||
|
compressSize: propTypes.number.def(200), // 压缩后的大小限制,单位KB
|
||||||
|
compressQuality: propTypes.number.def(0.8) // 压缩质量,0-1之间
|
||||||
})
|
})
|
||||||
const { t } = useI18n() // 国际化
|
|
||||||
const message = useMessage() // 消息弹窗
|
const { t } = useI18n()
|
||||||
// 生成组件唯一id
|
const message = useMessage()
|
||||||
const uuid = ref('id-' + generateUUID())
|
const uuid = ref('id-' + generateUUID())
|
||||||
// 查看图片
|
|
||||||
|
// 压缩图片方法
|
||||||
|
const compressImage = (file: File): Promise<Blob> => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const reader = new FileReader()
|
||||||
|
reader.readAsDataURL(file)
|
||||||
|
reader.onload = (e) => {
|
||||||
|
const img = new Image()
|
||||||
|
img.src = e.target?.result as string
|
||||||
|
img.onload = () => {
|
||||||
|
const canvas = document.createElement('canvas')
|
||||||
|
const ctx = canvas.getContext('2d')
|
||||||
|
if (!ctx) {
|
||||||
|
reject(new Error('Failed to get canvas context'))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 计算压缩后的尺寸
|
||||||
|
let width = img.width
|
||||||
|
let height = img.height
|
||||||
|
const maxSize = Math.max(width, height)
|
||||||
|
if (maxSize > 1920) {
|
||||||
|
const ratio = 1920 / maxSize
|
||||||
|
width *= ratio
|
||||||
|
height *= ratio
|
||||||
|
}
|
||||||
|
|
||||||
|
canvas.width = width
|
||||||
|
canvas.height = height
|
||||||
|
ctx.drawImage(img, 0, 0, width, height)
|
||||||
|
|
||||||
|
// 转换为Blob
|
||||||
|
canvas.toBlob(
|
||||||
|
(blob) => {
|
||||||
|
if (blob) {
|
||||||
|
resolve(blob)
|
||||||
|
} else {
|
||||||
|
reject(new Error('Failed to compress image'))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
file.type,
|
||||||
|
props.compressQuality
|
||||||
|
)
|
||||||
|
}
|
||||||
|
img.onerror = () => reject(new Error('Failed to load image'))
|
||||||
|
}
|
||||||
|
reader.onerror = () => reject(new Error('Failed to read file'))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
const imagePreview = (imgUrl: string) => {
|
const imagePreview = (imgUrl: string) => {
|
||||||
createImageViewer({
|
createImageViewer({
|
||||||
zIndex: 9999999,
|
zIndex: 9999999,
|
||||||
|
|
@ -106,26 +159,52 @@ const editImg = () => {
|
||||||
dom && dom.dispatchEvent(new MouseEvent('click'))
|
dom && dom.dispatchEvent(new MouseEvent('click'))
|
||||||
}
|
}
|
||||||
|
|
||||||
const beforeUpload: UploadProps['beforeUpload'] = (rawFile) => {
|
// 修改上传前的处理方法
|
||||||
const imgSize = rawFile.size / 1024 / 1024 < props.fileSize
|
const beforeUpload: UploadProps['beforeUpload'] = async (rawFile) => {
|
||||||
const imgType = props.fileType
|
// 检查文件类型
|
||||||
if (!imgType.includes(rawFile.type as FileTypes))
|
if (!props.fileType.includes(rawFile.type as FileTypes)) {
|
||||||
message.notifyWarning('上传图片不符合所需的格式!')
|
message.notifyWarning('上传图片不符合所需的格式!')
|
||||||
if (!imgSize) message.notifyWarning(`上传图片大小不能超过 ${props.fileSize}M!`)
|
return false
|
||||||
return imgType.includes(rawFile.type as FileTypes) && imgSize
|
}
|
||||||
|
|
||||||
|
// 检查原始文件大小
|
||||||
|
const originalSize = rawFile.size / 1024 / 1024
|
||||||
|
if (originalSize > props.fileSize) {
|
||||||
|
message.notifyWarning(`上传图片大小不能超过 ${props.fileSize}M!`)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果启用压缩且文件大小超过压缩限制
|
||||||
|
if (props.enableCompress && rawFile.size > props.compressSize * 1024) {
|
||||||
|
try {
|
||||||
|
const compressedBlob = await compressImage(rawFile)
|
||||||
|
// 如果压缩后仍然超过限制
|
||||||
|
if (compressedBlob.size > props.compressSize * 1024) {
|
||||||
|
message.notifyWarning(`压缩后图片仍然超过 ${props.compressSize}KB,请选择更小的图片!`)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// 创建新的文件对象
|
||||||
|
const newFile = new File([compressedBlob], rawFile.name, { type: rawFile.type })
|
||||||
|
return newFile
|
||||||
|
} catch (error) {
|
||||||
|
message.notifyError('图片压缩失败,请重试!')
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// 图片上传成功提示
|
|
||||||
const uploadSuccess: UploadProps['onSuccess'] = (res: any): void => {
|
const uploadSuccess: UploadProps['onSuccess'] = (res: any): void => {
|
||||||
message.success('上传成功')
|
message.success('上传成功')
|
||||||
emit('update:modelValue', res.data)
|
emit('update:modelValue', res.data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 图片上传错误提示
|
|
||||||
const uploadError = () => {
|
const uploadError = () => {
|
||||||
message.notifyError('图片上传失败,请您重新上传!')
|
message.notifyError('图片上传失败,请您重新上传!')
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.is-error {
|
.is-error {
|
||||||
.upload {
|
.upload {
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue