解决IOS系统手机拍照图片Canvas压缩上传后图片旋转的bug

有时候我们要手机上传图片到网上提交数据,但是可能图片的文件比较大,像拍照拍出来的就有可能有2m。这个时候就需要把图片压缩一下,如果在微信中的话可以调用微信的接口上传文件,但是有个更加通用的办法,就是用canvas来压缩上传。

用canvas压缩图片的原理就是读取图片的文件,然后把图片画在画布上,再用canvas自带的一个接口:

canvas.toDataURL(type,encoderOptions);

这里的第一个参数type就是图片的格式,默认会读取image/png。

第二个参数就比较重要了,这个参数就决定了图片压缩的程度,可以在0-1中选择一个值,当然推荐不要低于0.5,因为这样保存后的图片质量不太好。

这个方法执行完后会返回一个值,就是图片的base64格式的字符串,这个字符串就可以上传到服务端。

前段时间我在用这个方法上传图片的时候发现在ios下的图片上传会有一个bug,就是canvas转换后的图片会逆时针旋转90°,这样的话就不是我想要的效果了。所以在网上搜集了很多解决的方案,发现可以用一个插件EXIF.js来解决这个功能。

先来讲讲这个插件是用来做什么的,这个插件会读取图片的信息,然后返回一个值,这个值就是图片的旋转位置,6代表图片需要顺时针(向左)90度旋转才是正常的,8代表需要逆时针(向右)90度旋转,3表示图片是上下颠倒的。

下面是具体的代码:

function compress(img, fileType) {
            var canvas = document.createElement("canvas");
            var rotateshow;
            EXIF.getData(img, function(){
                EXIF.getAllTags(img);
                Orientation = EXIF.getTag(img,'Orientation');
                switch (Orientation){
                    case 6:
                        rotateshow = rotateImg(img,'left',canvas,fileType);
                        break;
                    case 8:
                        rotateshow = rotateImg(img,'right',canvas,fileType);
                        break;
                    case 3:
                        rotateImg(img,'right',canvas,fileType);
                        rotateshow = rotateImg(img,'right',canvas,fileType);
                        break;
                    default:
                        rotateshow = img.src;
                }
                toPreviewer(rotateshow);
            });
        }

不过之前我在的EXIF.js里发现一个问题就是getData获取不到img的属性,然后在我在这个插件里找到了相关的代码发现有一段代码判断图片的时候判断不对,导致代码不能执行下去:

EXIF.getData = function(img, callback) {
        /*if ((self.Image && img instanceof self.Image)
            || (self.HTMLImageElement && img instanceof self.HTMLImageElement)
            && !img.complete)
            return false;*/
        if (!imageHasData(img)) {
            getImageData(img, callback);
        } else {
            if (callback) {
                callback.call(img);
            }
        }
        return true;
    };

所以就把这里注释掉了,在判断完图片的朝向后就要对图片进行处理,图片的旋转可以把图片画在canvas里然后用canvas的rotate方法进行旋转然后调用toDataURL来进行压缩,具体的代码如下:

            var min_step = 0;
            var max_step = 3;
            //var img = document.getElementById(pid);
            if (img == null)return;
            //img的高度和宽度不能在img元素隐藏后获取,否则会出错
            var height = img.height;
            var width = img.width;
            //var step = img.getAttribute('step');
            var step = 2;
            if (step == null) {
                step = min_step;
            }
            if (direction == 'right') {
                step++;
                //旋转到原位置,即超过最大值
                step > max_step && (step = min_step);
            } else {
                step--;
                step < min_step && (step = max_step);
            }
            var degree = step * 90 * Math.PI / 180;
            var ctx = canvas.getContext('2d');
            switch (step) {
                case 0:
                    canvas.width = width;
                    canvas.height = height;
                    ctx.drawImage(img, 0, 0);
                    break;
                case 1:
                    canvas.width = height;
                    canvas.height = width;
                    ctx.rotate(degree);
                    ctx.drawImage(img, 0, -height);
                    break;
                case 2:
                    canvas.width = width;
                    canvas.height = height;
                    ctx.rotate(degree);
                    ctx.drawImage(img, -width, -height);
                    break;
                case 3:
                    canvas.width = height;
                    canvas.height = width;
                    ctx.rotate(degree);
                    ctx.drawImage(img, -width, 0);
                    break;
            }
            return canvas.toDataURL(fileType,0.75);
        }

至此图片上传bug就解决了,可以到我的github上查看我的源码:

https://github.com/Tinsson/Blog_demo/blob/master/picUpload/sign.html
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,875评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,569评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,475评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,459评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,537评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,563评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,580评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,326评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,773评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,086评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,252评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,921评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,566评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,190评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,435评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,129评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,125评论 2 352

推荐阅读更多精彩内容