如何使用FormData上传压缩裁剪后的图片Blob对象

简介: 在前端页面,我们通常会遇到需要用户上传图片的操作,可能还会在前端进行图片编辑的操作(比如头像的选区裁剪),然后如果图片过大的话,我们还会对图片进行压缩。这些功能我们通常通过Canvas来进行,最后使用Canvas API函数toDataURL来得到图片...

在前端页面,我们通常会遇到需要用户上传图片的操作,可能还会在前端进行图片编辑的操作(比如头像的选区裁剪),然后如果图片过大的话,我们还会对图片进行压缩。这些功能我们通常通过Canvas来进行,最后使用Canvas API函数toDataURL来得到图片的Base64字符串,然后当我们要上传到后台的时候,会面临2种选择:

  • 直接将图片的Base64字符串Post到后端进行处理和保存
  • 在前端将Base64字符串转换成二进制的Blob对象形式,再使用常规的文件上传形式(即FormData)来将其上传到后端

第一种方式对前端来说比较简单,主要的处理逻辑在后端。而第二种的话前端的工作就稍微复杂一些。考虑到后端采用接收二进制文件的方式来处理文件上传的情况比较多,所以我们来看一下前面所说的第二种情况在前端怎么来实现,以下是主要的示例代码:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>使用FormData上传压缩裁剪后的图片Blob对象</title>
</head>

<body>
    <input type="file" name="myfile" id="myfile" onchange="uploadHandler(event)">

    <script src="https://code.jquery.com/jquery-3.3.1.min.js"></script>

    <script>
        function uploadHandler(e) {
            var files = e.target.files || e.dataTransfer.files;

            if (files && files.length > 0) {
                var file = files[0];

                resizeImage(file).then(function (result) {
                    return typeof result === 'string' ? convertToBlob(result, file.type) : result;
                }).then(function (blob) {
                    // 构建FormData
                    var formData = new FormData();
                    //注意:此处第3个参数最好传入一个带后缀名的文件名,否则很有可能被后台认为不是有效的图片文件
                    formData.append("file", blob, file.name); 
                    // 上传文件
                    $.ajax({
                        url: '/api/upload',
                        method: "POST",
                        data: formData,
                        cache: false,
                        processData: false,
                        contentType: false
                    }).then(function (res) {
                        console.log(res);
                    }).catch(function (err) {
                        console.log(err);
                    })
                });
            }
        }

        /**
         * 压缩裁剪图片
         */
        function resizeImage(file) {
            return new Promise(function (resolve, reject) {
                var reader = new FileReader();

                reader.onload = function () {
                    var img = new Image();

                    img.onload = function () {
                        var w = this.naturalWidth;
                        var h = this.naturalHeight;
                        var maxW = 500;
                        var maxH = 500;

                        // 如果图片尺寸小于最大限制,则不压缩直接上传
                        if (w <= maxW && h <= maxH) {
                            resolve(file);
                            return;
                        }

                        var level = 0.6;
                        var multiple = Math.max(w / maxW, h / maxH);
                        var resizeW = w / multiple;
                        var resizeH = h / multiple;

                        var canvas = document.createElement("canvas");

                        canvas.width = resizeW;
                        canvas.height = resizeH;

                        var ctx = canvas.getContext("2d");

                        ctx.drawImage(img, 0, 0, resizeW, resizeH);

                        var base64Img = canvas.toDataURL(file.type, level);
                        var arr = base64Img.split(",");

                        resolve(arr[1]);
                    };

                    img.src = this.result;
                };

                reader.readAsDataURL(file);
            });
        }

        /**
         * 将图片的base64字符串转换为Blob对象
         */
        function convertToBlob(base64Str, fileType) {
            var base64 = window.atob(base64Str);
            var len = base64.length;
            var buff = new ArrayBuffer(len);
            var uarr = new Uint8Array(buff);

            for (var i = 0; i < len; i++) {
                uarr[i] = base64.charCodeAt(i);
            }

            var blob = null;

            try {
                blob = new Blob([buff], { type: fileType });
            } catch (e) {
                var BlobBuilder = window.BlobBuilder = (
                    window.BlobBuilder ||
                    window.WebKitBlobBuilder ||
                    window.MozBlobBuilder ||
                    window.MSBlobBuilder
                );

                if (e.name === "TypeError" && BlobBuilder) {
                    var builder = new BlobBuilder();
                    builder.append(buff);
                    blob = builder.getBlob(fileType);
                }
            }

            return blob;
        }
    </script>
</body>

</html>

代码中值得注意的一点是下面这行代码:

formData.append("file", blob, file.name); 

如果不传第三个参数的话,生成的表单数据中,上传文件对应的filename会被设置为blob

通常情况下这也是没问题的。但是可能由于后端使用的不同框架或自己的逻辑代码的原因,对上传的文件名做了强制的后缀名检查,会发生报错导致上传失败,遇到这种情况,请记得使用上面的方式加上第三个参数,这样问题应该就能迎刃而解了。

目录
相关文章
|
JavaScript
uniapp+vue3路由跳转传参
uniapp+vue3路由跳转传参
564 0
MAC使用CodeSign查看已签名的文件的数字签名情况
MAC使用CodeSign查看已签名的文件的数字签名情况
872 0
|
存储 IDE 程序员
揭秘 IPython 的 5 种最佳调试方法
一个好的集成开发环境(IDE)附带的调试器是开发人员能够拥有的最强大的工具之一,但并不是每个人都在使用一个带有很棒代码调试器的集成发环境(IDE)。
|
JavaScript 前端开发 Docker
全栈开发实战:结合Python、Vue和Docker进行部署
【4月更文挑战第10天】本文介绍了如何使用Python、Vue.js和Docker进行全栈开发和部署。Python搭配Flask创建后端API,Vue.js构建前端界面,Docker负责应用的容器化部署。通过编写Dockerfile,将Python应用构建成Docker镜像并运行,前端部分使用Vue CLI创建项目并与后端交互。最后,通过Nginx和另一个Dockerfile部署前端应用。这种组合提升了开发效率,保证了应用的可维护性和扩展性,适合不同规模的企业使用。
737 4
|
存储 Java
JAVA中String长度限制解决超出字符长度
JAVA中String长度限制解决超出字符长度
321 1
|
SQL Java 数据库
Sqoop【付诸实践 02】Sqoop1最新版 全库导入 + 数据过滤 + 字段类型支持 说明及举例代码(query参数及字段类型强制转换)
【2月更文挑战第10天】Sqoop【付诸实践 02】Sqoop1最新版 全库导入 + 数据过滤 + 字段类型支持 说明及举例代码(query参数及字段类型强制转换)
524 0
统信UOS系统开发笔记(三):从Qt源码编译安装之编译安装Qt5.12.8
上一篇,是使用Qt提供的安装包安装的,有些场景需要使用到自己编译的Qt,所以本篇如何在统信UOS系统上编译Qt5.12.8源码。
统信UOS系统开发笔记(三):从Qt源码编译安装之编译安装Qt5.12.8
|
弹性计算 Kubernetes 架构师
985毕业,工作3年,分享从阿里辞职到了国企的一路辛酸和经验
楼主本硕985,毕业的时候去了杭州某互联网大厂,后来又跳槽去了北京某互联网大厂。简单的谈一下互联网的感受吧,工作压力大,节奏快,但是从技术上确实得到了成长,尤其是当你维护与大促相关的系统的时候。记得在北京的时候,作为系统负责人,那个系统docker就800个,那个系统的并发量在全国来说肯定是top级别的。通过维护大促系统,排查跳点,不断地优化系统的框架,优化JVM,所带来的技术提升是质的飞跃。
|
SQL 存储 关系型数据库
通过cursor游标讲解,带你彻底搞懂python操作mysql数据库
通过cursor游标讲解,带你彻底搞懂python操作mysql数据库
通过cursor游标讲解,带你彻底搞懂python操作mysql数据库
使用 VSCode 调试 Electron 主进程代码
在开发 Electron 应用的时候,为了提高工作效率,我们需要借助于调试工具,及时发现并解决问题。
466 0
使用 VSCode 调试 Electron 主进程代码