增加图片自动压缩

pull/663/head
shixiaohe 2025-01-18 09:52:10 +08:00
parent ca32b6d102
commit d102e0a297
2 changed files with 130 additions and 44 deletions

View File

@ -71,25 +71,44 @@ type FileTypes =
//
const props = defineProps({
modelValue: propTypes.string.def(''),
drag: propTypes.bool.def(true),
disabled: propTypes.bool.def(false),
fileSize: propTypes.number.def(5),
fileType: propTypes.array.def(['image/jpeg', 'image/png', 'image/gif']),
height: propTypes.string.def('150px'),
width: propTypes.string.def('150px'),
borderradius: propTypes.string.def('8px'),
showDelete: propTypes.bool.def(true),
showBtnText: propTypes.bool.def(true),
//
drag: propTypes.bool.def(true), // ==> true
disabled: propTypes.bool.def(false), // ==> false
fileSize: propTypes.number.def(5), // ==> 5M
fileType: propTypes.array.def(['image/jpeg', 'image/png', 'image/gif']), // ==> ["image/jpeg", "image/png", "image/gif"]
height: propTypes.string.def('150px'), // ==> 150px
width: propTypes.string.def('150px'), // ==> 150px
borderradius: propTypes.string.def('8px'), // ==> 8px
showDelete: propTypes.bool.def(true), //
showBtnText: propTypes.bool.def(true), //
//
enableCompress: propTypes.bool.def(false), //
compressSize: propTypes.number.def(200), // KB
compressSize: propTypes.number.def(300), // KB
compressQuality: propTypes.number.def(0.8) // 0-1
})
const { t } = useI18n()
const message = useMessage()
const { t } = useI18n() //
const message = useMessage() //
// id
const uuid = ref('id-' + generateUUID())
//
const imagePreview = (imgUrl: string) => {
createImageViewer({
zIndex: 9999999,
urlList: [imgUrl]
})
}
const emit = defineEmits(['update:modelValue'])
const deleteImg = () => {
emit('update:modelValue', '')
}
const { uploadUrl, httpRequest } = useUpload()
const editImg = () => {
const dom = document.querySelector(`#${uuid.value} .el-upload__input`)
dom && dom.dispatchEvent(new MouseEvent('click'))
}
//
const compressImage = (file: File): Promise<Blob> => {
return new Promise((resolve, reject) => {
@ -138,27 +157,6 @@ const compressImage = (file: File): Promise<Blob> => {
reader.onerror = () => reject(new Error('Failed to read file'))
})
}
const imagePreview = (imgUrl: string) => {
createImageViewer({
zIndex: 9999999,
urlList: [imgUrl]
})
}
const emit = defineEmits(['update:modelValue'])
const deleteImg = () => {
emit('update:modelValue', '')
}
const { uploadUrl, httpRequest } = useUpload()
const editImg = () => {
const dom = document.querySelector(`#${uuid.value} .el-upload__input`)
dom && dom.dispatchEvent(new MouseEvent('click'))
}
//
const beforeUpload: UploadProps['beforeUpload'] = async (rawFile) => {
//
@ -180,7 +178,7 @@ const beforeUpload: UploadProps['beforeUpload'] = async (rawFile) => {
const compressedBlob = await compressImage(rawFile)
//
if (compressedBlob.size > props.compressSize * 1024) {
message.notifyWarning(`压缩后图片仍然超过 ${props.compressSize}KB请选择更小的图片`)
message.notifyWarning(`压缩后图片${compressedBlob.size/1024}KB仍然超过 ${props.compressSize}KB请选择更小的图片`)
return false
}
//
@ -195,16 +193,17 @@ const beforeUpload: UploadProps['beforeUpload'] = async (rawFile) => {
return true
}
//
const uploadSuccess: UploadProps['onSuccess'] = (res: any): void => {
message.success('上传成功')
emit('update:modelValue', res.data)
}
//
const uploadError = () => {
message.notifyError('图片上传失败,请您重新上传!')
}
</script>
<style lang="scss" scoped>
.is-error {
.upload {

View File

@ -81,7 +81,11 @@ const props = defineProps({
fileType: propTypes.array.def(['image/jpeg', 'image/png', 'image/gif']), // ==> ["image/jpeg", "image/png", "image/gif"]
height: propTypes.string.def('150px'), // ==> 150px
width: propTypes.string.def('150px'), // ==> 150px
borderradius: propTypes.string.def('8px') // ==> 8px
borderradius: propTypes.string.def('8px'), // ==> 8px
//
enableCompress: propTypes.bool.def(false), //
compressSize: propTypes.number.def(300), // KB
compressQuality: propTypes.number.def(0.8) // 0-1
})
const { uploadUrl, httpRequest } = useUpload()
@ -89,27 +93,110 @@ const { uploadUrl, httpRequest } = useUpload()
const fileList = ref<UploadUserFile[]>([])
const uploadNumber = ref<number>(0)
const uploadList = ref<UploadUserFile[]>([])
//
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'))
})
}
/**
* @description 文件上传之前判断
* @param rawFile 上传的文件
* */
const beforeUpload: UploadProps['beforeUpload'] = (rawFile) => {
const imgSize = rawFile.size / 1024 / 1024 < props.fileSize
const imgType = props.fileType
if (!imgType.includes(rawFile.type as FileTypes))
//
const beforeUpload: UploadProps['beforeUpload'] = async (rawFile) => {
//
if (!props.fileType.includes(rawFile.type as FileTypes)) {
ElNotification({
title: '温馨提示',
message: '上传图片不符合所需的格式!',
type: 'warning'
})
if (!imgSize)
return false
}
//
const originalSize = rawFile.size / 1024 / 1024
if (originalSize > props.fileSize) {
ElNotification({
title: '温馨提示',
message: `上传图片大小不能超过 ${props.fileSize}M`,
type: 'warning'
})
return false
}
//
if (props.enableCompress && rawFile.size > props.compressSize * 1024) {
try {
const compressedBlob = await compressImage(rawFile)
//
if (compressedBlob.size > props.compressSize * 1024) {
ElNotification({
title: '温馨提示',
message: `压缩后图片仍然超过 ${props.compressSize}KB请选择更小的图片`,
type: 'warning'
})
return false
}
//
const newFile = new File([compressedBlob], rawFile.name, { type: rawFile.type })
uploadNumber.value++
return newFile
} catch (error) {
ElNotification({
title: '温馨提示',
message: '图片压缩失败,请重试!',
type: 'error'
})
return false
}
}
uploadNumber.value++
return imgType.includes(rawFile.type as FileTypes) && imgSize
return true
}
//