ANTD upload组件上传大图片裁剪为两个小图片(方便图片加载)

思路说明

1、antd-img-crop插件

刚开始,看antd文档想用推荐的antd-img-crop插件来实现。优点:代码简单明了,没有神马难度,可以任意裁剪图片;效果与缺点:不符合目标,我只想自动把长图片裁切为几张小图片并上传,不需要用户自定义,自定义容易出现问题,gg。

2、blob数据流直接一分为2:

结果:切割后的数据流前半部分可以显示,后半部分无法正常显示,估计是暴力裁剪有的东西没有剪上,有大佬了解的可以说一下。

3、canvas:

效果:没啥问题,能实现,就是数据转换有一点点麻烦。

来吧,展示:

代码:

上传组件(照抄自定义上传):
html 复制代码
<Upload {...propsRule}>
                     <Button type="primary">选择文件</Button>
                   </Upload>
                   <div>
                     <Button
                       type="primary"
                       onClick={this.handleUpload.bind(this)}
                       disabled={fileList.length === 0}
                       loading={uploading}
                       style={{
                         marginTop: 16
                       }}
                     >
                    点击上传
                     </Button>
                   </div>
上传组件属性(propsRule):

dataURItoBlob 方法在百度搜到的,感谢原作者!需要用到的可以去搜一下,不同站的文章这里就不放链接了

js 复制代码
const propsRule = {
     accept: 'image/*',
     maxCount: 1,
     beforeUpload: async (file) => {
       if (file.size > 512 * 512) {
         file.status = 'error'
         file.response = '上传文件大小不能超过500kb'
         return false
       }
       try {
         const data = await this.handleFile(file)
         // 创建新的 Blob 对象
         const blob1 = this.dataURItoBlob(data[0])
         const blob2 = this.dataURItoBlob(data[1])
         // 你可以使用Blob对象进行进一步的处理,例如将其转换为File对象
         const file1 = new File([blob1], 'img1.png', { type: 'image/png' })
         const file2 = new File([blob2], 'img2.png', { type: 'image/png' })
         //  const url1 = URL.createObjectURL(file1)
         this.setState({ fileList: [file1, file2] })
         return false
       } catch (error) {
         console.error('Error:', error)
       }
     },
     fileList
   }
切割方法:
js 复制代码
 handleFile = async (file) => {
    return new Promise((resolve, reject) => {
      const imgBlob = new Image()
      imgBlob.src = URL.createObjectURL(file)
      imgBlob.onload = function() {
        const canvas = document.createElement('canvas')
        canvas.width = imgBlob.width
        canvas.height = imgBlob.height / 2
        const ctx = canvas.getContext('2d')
        // 切割图像
        ctx.drawImage(imgBlob, 0, 0, canvas.width, canvas.height, 0, 0, canvas.width, canvas.height)
        const imageData1 = ctx.getImageData(0, 0, canvas.width, canvas.height)
        const img1 = canvas.toDataURL(imageData1)
        ctx.clearRect(0, 0, canvas.width, canvas.height)

        // 切割图像
        ctx.drawImage(imgBlob, 0, imgBlob.height / 2, canvas.width, imgBlob.height / 2, 0, 0, canvas.width, canvas.height)
        const imageData2 = ctx.getImageData(0, canvas.height / 2, canvas.width, canvas.height / 2)
        const img2 = canvas.toDataURL(imageData2)

        // 数据处理完成后,调用 resolve 函数
        resolve([img1, img2])
      }
      imgBlob.onerror = function(error) {
        reject(error)
      }
    })
  }
上传方法:(略,随便一个post请求即可)
相关推荐
橘子星4 分钟前
基于 ES6 语法的 NLP 任务模块化开发实践
前端·javascript
月光刺眼7 分钟前
JS 底层执行机制探讨:执行上下文、变量提升与调用栈
前端·javascript
ZC跨境爬虫34 分钟前
跟着 MDN 学 JavaScript day_1:什么是 JavaScript?
开发语言·前端·javascript·ecmascript
xiaofeichaichai1 小时前
Vue 响应式原理
前端·javascript·vue.js
提子拌饭1331 小时前
模态窗鸿蒙PC Electron框架实现技术详解 - 饮料含糖量应用案例分析
前端·javascript·华为·electron·前端框架·开源·鸿蒙
光影少年2 小时前
react的Context 和 Redux 区别?
前端·javascript·react.js·前端框架
前端 贾公子2 小时前
uni-app工程化实战:基于vue-i18n和i18n-ally的国际化方案 (上)
前端·javascript·vue.js
半个落月2 小时前
面试必问的 JS 原型链,我用 16 个示例给你彻底讲明白
javascript
丷丩3 小时前
12. 渲染:MapLibre GL JS 集成与多源瓦片联动
javascript·矢量瓦片·maplibre gl js·地图服务器
橘子星3 小时前
别再懵圈!JS 执行机制的 “千层套路” 全揭秘
前端·javascript