资讯专栏INFORMATION COLUMN

前端文件上传(js/vue.js/axios/canvas图片压缩)

Lsnsh / 2453人阅读

摘要:哈哈主要还是我嫌麻烦四上传图片这里的页面样式,图片压缩和预览都和上面一样,这里我主要配置一下的,让接口能够成功上传。如果想让用户有更好的体验,可以对图片进行一下压缩和本地预览。

一、通过Form表单提交上传
HTML  enctype属性必不可少
    
上面一种方法通过表单自有属性进行提交,看似简单,但是也有其最大的缺点,那就是提交完毕之后直接进行了跳转,这对于当前的绝大多数的需求来说是不能满足的。那么我接下来介绍另一种直接利用xaj的post方法和FormData进行文件的上传。

二、通过jQuery post请求上传
HTML 多文件上传

JS
    (function(){
        $("#upload").change(function(e){
            let files = e.target.files;
            let params = new FormData();
            for(let i = 0; i < files.length; i++){
                //visit_file就是后台用来接受的字段,因为是一个数组,所以加一个[]
                params.append("visit_file[]", files[i], files[i].name);
            }
            
            $.ajax({
                type: "post",
                url: "http://192.168.1.101:8080/springbootdemo/file/upload",
                data: params,
                processData: false,//必不可少属性
                traditional: true,//必不可少属性
                contentType: false,//必不可少的属性
            }).success(function (data) {
                console.log(data);
            }).error(function () {
                console.log("上传失败");
        })
    })()
这种方法解决了上传文件之后页面跳转的问题,但是如果你传的图片很大的时候怎么办?接下来就是我们要说的上传较大文件时一个优化方案。如今,为了提升用户的体验,不仅让用户再浏览时给用户极致的体验,还要在上传时感受不到卡顿的现象,这对于后端来说就做不了了,只能交给前端。下面我们介绍一下这种优化方案。

三、本地预览,Canvas图片压缩,转blob二进制文件
HTML 多文件上传

JS
    (function(){
        $("#upload").change(function(e){
            let files = e.target.files;
            let params = new FormData();
            for(let i = 0; i < files.length; i++){
                //visit_file就是后台用来接受的字段,因为是一个数组,所以加一个[]
                params.append("visit_file[]", files[i], files[i].name);
            }
            //图片预览地址数组
            let previewArr = previewImage(files);   
            
            for(let i = 0 ; i < previewArr.length; i++){
                
            }   
        }
        
        //上传图片  压缩过的二进制文件只能单张上传处理,我试过多上同时上传失败了,你们也可以试试,也许可以找到方法
        function uploadImage(params){
                $.ajax({
                type: "post",
                url: "http://192.168.1.101:8080/springbootdemo/file/upload",
                data: params,
                processData: false,//必不可少属性
                traditional: true,//必不可少属性
                contentType: false,//必不可少的属性
                }).success(function (data) {
                    console.log(data);
                }).error(function () {
                    console.log("上传失败");
                })
         }
        //图片预览
        function previewImage(files){
                let previewsArr = [];
                for(let i = 0; i < files.length; i++){
                    let fileReader = new FileReader();
                    fileReader.readAsDataURL(files[i]);
                    fileReader.onloaded = () => {
                        //数组放入获取的base64本地图片地址
                        previewsArr.push(fileReader.result);
                    }
                }
                
                return previewsArr;
            
          }
         //图片压缩   
         function compressImage(base64URL){
                let img = new Image();
                let canvas = document.createElement("canvas");
                let context = canvas.getContext("2d");
                img.src = base64URL;
                
                img.onload = () => {
                    // 图片原始尺寸
                    var originWidth = img.width;
                    var originHeight = img.height;
                    // 最大尺寸限制
                    let maxWidth = 400,
                      maxHeight = 400;
                    // 目标尺寸
                    let targetWidth = originWidth,
                      targetHeight = originHeight;
                    // 图片尺寸超过400x400的限制
                    if (originWidth > maxWidth || originHeight > maxHeight) {
                      if (originWidth/originHeight > maxWidth/maxHeight) {
                        //更宽,按照宽度限定尺寸
                        targetWidth = maxWidth;
                        targetHeight = Math.round(maxWidth * (originHeight / originWidth));
                      }else{
                        targetHeight = maxHeight;
                        targetWidth = Math.round(maxHeight * (originWidth / originHeight));
                      }
                    }
                    
            
                    // canvas对图片进行缩放
                    canvas.width = targetWidth;
                    canvas.height = targetHeight;
                    // 清除画布
                    context.clearRect(0, 0, targetWidth, targetHeight);
                    // 图片压缩
                    context.drawImage(img, 0, 0, targetWidth, targetHeight);
            
            
                    //canvas直接转blob二进制文件,但是大部分浏览器不支持
                    // canvas.toBlob(function (blob) {
                    //   console.log(blob)
                    //   resolve(blob)
                    // }, "image/png");
            
                    let base64Data = canvas.toDataURL("image/png", 0.92);
                    
                    let blob = dataURItoBlob(base64Data);
                    //上传图片
                    
                    let params = new FormaData();
                    params.append("visit_file", blob, "cavas.png");
                    
                    uploadImage(params);
                    
            }
        }
        
          /**
           * base64 转二进制文件
           * @param {*} base64Data 
           */
          function dataURItoBlob(base64Data) {
            var bytes = window.atob(base64Data.split(",")[1]); //去掉url的头,并转换为byte
        
            //处理异常,将ascii码小于0的转换为大于0
            var ab = new ArrayBuffer(bytes.length);
            var ia = new Uint8Array(ab);
            for (var i = 0; i < bytes.length; i++) {
              ia[i] = bytes.charCodeAt(i);
            }
        
            return new Blob([ab], {
              type: "image/png"
            });
          }
        
    })()
这里通过FileReader获取本地base64文件,然后通过canvas对图片进行压缩,最终转为二进制的blob文件,传到服务器上。这里还可以做的更好,利用promise,对压缩功能进行返回,不必在压缩函内进行调用上传,降低耦合度。这里为了大家理解,我就没有分开。哈哈~~主要还是我嫌麻烦...

四、Vue+axios上传图片
这里的页面样式,图片压缩和预览都和上面一样,这里我主要配置一下axois的http,让post接口能够成功上传。
        /**
       * 
       * @param {路由} url 
       * @param {路由参数} params 
       * @param {文件数据} body 
       */
      upload(url, params = "", body = {}) {
        let path = config.host + url + params;
        // console.log(body);
        return axios({
          method: "POST",
          url: path,
          data: body,
          processData: false, //必不可少参数
          traditional: true, //比不可少参数
          contentType: false,//比不可少参数
          headers: {
            "token": localStorage["token"],
            "originno": config.originno,
            "Content-Type": false
          }
        }).then(
          res => res
        ).catch((error) => {
          console.log(error);
        })
      }

我也是最近在做项目时遇到需求,当时使用的就是vue+axois,怎么提交怎么报错,最后发现就是文件没有传过去,只要配置上那三个必不可少的参数,就可以上传成功。如果想让用户有更好的体验,可以对图片进行一下压缩和本地预览。还可以做的更好,如果大家有什么问题可以留言告诉我!也可以在我的基础上进行更多的扩展。
谢谢大家能够读完我的这篇文章!

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/52750.html

相关文章

  • 前端文件上传(js/vue.js/axios/canvas图片压缩)

    摘要:哈哈主要还是我嫌麻烦四上传图片这里的页面样式,图片压缩和预览都和上面一样,这里我主要配置一下的,让接口能够成功上传。如果想让用户有更好的体验,可以对图片进行一下压缩和本地预览。 一、通过Form表单提交上传 HTML enctype属性必不可少 上面一种方法通过表单自有属性进行提交,看似简单,但是也有其最大的缺点,那就是提交...

    Luosunce 评论0 收藏0
  • 结合Vue.js前端压缩图片方案

    摘要:图片文件大小减小后,上传速度自然会提升,在同样的并发下,后台处理的速度也会得到提升,用户体验得到提升。 这是一个很简单的方案。嗯,是真的。 为什么要这么做? 在移动Web蓬勃发展的今天,有太多太多的应用需要让用户在移动Web上传图片文件了,正因如此,我们有些困难必须去攻克: 低网速下上传进度缓慢,用户体验差 高并发下,后台处理较大的上传文件压力大 或许有更多... 在攻克上面的一些...

    sutaking 评论0 收藏0
  • input上传图片压缩vue前端js

    摘要:大家好,我是云皓,话不多说,直入正题,获取上传文件自行获取,也可通过的组件来获取,转化为文件,压缩,转换为文件,上传。 大家好,我是云皓,话不多说,直入正题 1,获取input上传file文件(自行获取,也可通过vant的upload组件来获取)2,转化为base64文件3,压缩4,转换为blob文件5,上传。下面直接上代码(本代码段是用用在vue&vantui 里面, 原理都在,可根...

    andot 评论0 收藏0
  • 说一说前端文件压缩

    摘要:因为异步压缩的时候我们上传的文件的数量不定,所以上面需要使用关键字来修饰压缩过程。而下面的压缩过程的实现,最终返回一个对象,当压缩过程已完成后,完整的生成的文件存于其中。 基于JSZip的前端文件压缩 1. 简介: 这段时间,项目需要做一个这样的功能:客户端在上传文件的时候(具体文件类型),需要对文件进行压缩再上传以节省带宽和服务器端资源,完成这个功能,我们选择了GitHub上的JSZ...

    Ververica 评论0 收藏0

发表评论

0条评论

Lsnsh

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<