如何在浏览器中通过Javascript压缩图像?

TL;DR;

有没有办法在上传图像之前直接在浏览器端压缩图像(主要是jpeg,png和gif)?我很确定JavaScript可以做到这一点,但我找不到实现它的方法。


以下是我想实现的完整方案:

  • 用户转到我的网站,并通过元素选择图像,input type="file"
  • 此图像通过JavaScript检索,我们进行一些验证,例如正确的文件格式,最大文件大小等,
  • 如果一切都正常,则页面上显示图像的预览,
  • 用户可以执行一些基本操作,例如将图像旋转90°/-90°,按照预定义的比例裁剪图像等,或者用户可以上传另一个图像并返回步骤1,
  • 当用户满意时,编辑后的图像被压缩并“保存”到本地(不是保存到文件中,而是在浏览器内存/页面中),-
  • 用户用姓名,年龄等数据填写表格,
  • 用户单击“完成”按钮,然后将包含数据+压缩图像的表单发送到服务器(不含AJAX),

直到最后一步的整个过程应该在客户端完成,并且应该在最新的Chrome和Firefox,Safari 5 +和IE 8 +上兼容。如果可能的话,只应该使用JavaScript(但我非常确定这是不可能的)。

我现在还没有编写任何代码,但我已经考虑过了。可以通过File API在本地读取文件,可以使用Canvas元素完成图像预览和编辑,但我找不到一种方法来执行图像压缩部分

根据 html5please.comcaniuse.com,支持这些浏览器非常困难(多亏了IE),但可以使用FlashCanvasFileReader等polyfill来完成。

实际上,目标是减小文件大小,因此我认为图像压缩是一种解决方案。但是,我知道上传的图像将在我的网站上显示,每次都在同一个地方,并且我知道这个显示区域的尺寸(例如200x400)。因此,我可以调整图像大小以适应这些尺寸,从而减小文件大小。我不知道这种技术的压缩比是多少。

你觉得怎么样?你有什么建议要告诉我吗?你知道在JavaScript中压缩图像浏览器端的方法吗?感谢您的回复。


答案 1

总之:

  • 使用 HTML5 FileReader API 和 .readAsArrayBuffer 读取文件
  • 使用文件数据创建 Blob,并通过窗口获取其 url。URL.createObjectURL(blob)
  • 创建新的图像元素并将其 src 设置为文件 blob url
  • 将图像发送到画布。画布大小设置为所需的输出大小
  • 通过 canvas.toDataURL(“image/jpeg”,0.7)从 canvas 获取缩小的数据(设置自己的输出格式和质量)
  • 将新的隐藏输入附加到原始表单,并基本上将dataURI图像作为普通文本传输
  • 在后端,读取 dataURI,从 Base64 解码并保存

来源:代码


答案 2

我看到其他答案中缺少两件事:

  • canvas.toBlob(如果可用)比 更高性能,并且还具有异步性。canvas.toDataURL
  • 文件 -> 图像 -> 画布 -> 文件转换丢失 EXIF 数据;特别是,有关图像旋转的数据通常由现代手机/平板电脑设置。

以下脚本处理这两点:

// From https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob, needed for Safari:
if (!HTMLCanvasElement.prototype.toBlob) {
    Object.defineProperty(HTMLCanvasElement.prototype, 'toBlob', {
        value: function(callback, type, quality) {

            var binStr = atob(this.toDataURL(type, quality).split(',')[1]),
                len = binStr.length,
                arr = new Uint8Array(len);

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

            callback(new Blob([arr], {type: type || 'image/png'}));
        }
    });
}

window.URL = window.URL || window.webkitURL;

// Modified from https://stackoverflow.com/a/32490603, cc by-sa 3.0
// -2 = not jpeg, -1 = no data, 1..8 = orientations
function getExifOrientation(file, callback) {
    // Suggestion from http://code.flickr.net/2012/06/01/parsing-exif-client-side-using-javascript-2/:
    if (file.slice) {
        file = file.slice(0, 131072);
    } else if (file.webkitSlice) {
        file = file.webkitSlice(0, 131072);
    }

    var reader = new FileReader();
    reader.onload = function(e) {
        var view = new DataView(e.target.result);
        if (view.getUint16(0, false) != 0xFFD8) {
            callback(-2);
            return;
        }
        var length = view.byteLength, offset = 2;
        while (offset < length) {
            var marker = view.getUint16(offset, false);
            offset += 2;
            if (marker == 0xFFE1) {
                if (view.getUint32(offset += 2, false) != 0x45786966) {
                    callback(-1);
                    return;
                }
                var little = view.getUint16(offset += 6, false) == 0x4949;
                offset += view.getUint32(offset + 4, little);
                var tags = view.getUint16(offset, little);
                offset += 2;
                for (var i = 0; i < tags; i++)
                    if (view.getUint16(offset + (i * 12), little) == 0x0112) {
                        callback(view.getUint16(offset + (i * 12) + 8, little));
                        return;
                    }
            }
            else if ((marker & 0xFF00) != 0xFF00) break;
            else offset += view.getUint16(offset, false);
        }
        callback(-1);
    };
    reader.readAsArrayBuffer(file);
}

// Derived from https://stackoverflow.com/a/40867559, cc by-sa
function imgToCanvasWithOrientation(img, rawWidth, rawHeight, orientation) {
    var canvas = document.createElement('canvas');
    if (orientation > 4) {
        canvas.width = rawHeight;
        canvas.height = rawWidth;
    } else {
        canvas.width = rawWidth;
        canvas.height = rawHeight;
    }

    if (orientation > 1) {
        console.log("EXIF orientation = " + orientation + ", rotating picture");
    }

    var ctx = canvas.getContext('2d');
    switch (orientation) {
        case 2: ctx.transform(-1, 0, 0, 1, rawWidth, 0); break;
        case 3: ctx.transform(-1, 0, 0, -1, rawWidth, rawHeight); break;
        case 4: ctx.transform(1, 0, 0, -1, 0, rawHeight); break;
        case 5: ctx.transform(0, 1, 1, 0, 0, 0); break;
        case 6: ctx.transform(0, 1, -1, 0, rawHeight, 0); break;
        case 7: ctx.transform(0, -1, -1, 0, rawHeight, rawWidth); break;
        case 8: ctx.transform(0, -1, 1, 0, 0, rawWidth); break;
    }
    ctx.drawImage(img, 0, 0, rawWidth, rawHeight);
    return canvas;
}

function reduceFileSize(file, acceptFileSize, maxWidth, maxHeight, quality, callback) {
    if (file.size <= acceptFileSize) {
        callback(file);
        return;
    }
    var img = new Image();
    img.onerror = function() {
        URL.revokeObjectURL(this.src);
        callback(file);
    };
    img.onload = function() {
        URL.revokeObjectURL(this.src);
        getExifOrientation(file, function(orientation) {
            var w = img.width, h = img.height;
            var scale = (orientation > 4 ?
                Math.min(maxHeight / w, maxWidth / h, 1) :
                Math.min(maxWidth / w, maxHeight / h, 1));
            h = Math.round(h * scale);
            w = Math.round(w * scale);

            var canvas = imgToCanvasWithOrientation(img, w, h, orientation);
            canvas.toBlob(function(blob) {
                console.log("Resized image to " + w + "x" + h + ", " + (blob.size >> 10) + "kB");
                callback(blob);
            }, 'image/jpeg', quality);
        });
    };
    img.src = URL.createObjectURL(file);
}

用法示例:

inputfile.onchange = function() {
    // If file size > 500kB, resize such that width <= 1000, quality = 0.9
    reduceFileSize(this.files[0], 500*1024, 1000, Infinity, 0.9, blob => {
        let body = new FormData();
        body.set('file', blob, blob.name || "file.jpg");
        fetch('/upload-image', {method: 'POST', body}).then(...);
    });
};