我一直在尝试重新实现一个HTML5图像上传器,就像Mozilla Hacks网站上的那个,但它适用于WebKit浏览器。该任务的一部分是从canvas对象中提取图像文件,并将其附加到FormData对象中以便上传。

问题是,虽然canvas有toDataURL函数来返回图像文件的表示,但FormData对象只接受file API中的file或Blob对象。

Mozilla解决方案在画布上使用了以下仅限firefox的功能:

var file = canvas.mozGetAsFile("foo.png");

...这在WebKit浏览器上是不可用的。我能想到的最佳解决方案是找到某种方法将Data URI转换为File对象,我认为这可能是File API的一部分,但我无法找到这样做的方法。

这可能吗?如果没有,有什么替代方案吗?


当前回答

不断发展的标准看起来是canvas.toBlob(),而不是Mozilla冒险猜测的canvas.getAsFile()。

我还没有看到任何浏览器支持它:(

谢谢这个很棒的帖子!

此外,任何人尝试接受的答案应该小心BlobBuilder,因为我发现支持是有限的(和命名空间):

    var bb;
    try {
        bb = new BlobBuilder();
    } catch(e) {
        try {
            bb = new WebKitBlobBuilder();
        } catch(e) {
            bb = new MozBlobBuilder();
        }
    }

你在BlobBuilder中使用了其他库的polyfill吗?

其他回答

我和拉文德·帕亚尔遇到了同样的问题,而我找到了答案。试试这个:

var dataURL = canvas.toDataURL("image/jpeg");

var name = "image.jpg";
var parseFile = new Parse.File(name, {base64: dataURL.substring(23)});

这个可以在iOS和Safari中使用。

你需要使用Stoive的ArrayBuffer解决方案,但你不能使用BlobBuilder,正如vava720所指出的,所以这里是两者的混搭。

function dataURItoBlob(dataURI) {
    var byteString = atob(dataURI.split(',')[1]);
    var ab = new ArrayBuffer(byteString.length);
    var ia = new Uint8Array(ab);
    for (var i = 0; i < byteString.length; i++) {
        ia[i] = byteString.charCodeAt(i);
    }
    return new Blob([ab], { type: 'image/jpeg' });
}

不断发展的标准看起来是canvas.toBlob(),而不是Mozilla冒险猜测的canvas.getAsFile()。

我还没有看到任何浏览器支持它:(

谢谢这个很棒的帖子!

此外,任何人尝试接受的答案应该小心BlobBuilder,因为我发现支持是有限的(和命名空间):

    var bb;
    try {
        bb = new BlobBuilder();
    } catch(e) {
        try {
            bb = new WebKitBlobBuilder();
        } catch(e) {
            bb = new MozBlobBuilder();
        }
    }

你在BlobBuilder中使用了其他库的polyfill吗?

在摆弄了一些东西之后,我自己设法弄明白了这一点。

首先,这将把一个dataURI转换为一个Blob:

function dataURItoBlob(dataURI) {
    // convert base64/URLEncoded data component to raw binary data held in a string
    var byteString;
    if (dataURI.split(',')[0].indexOf('base64') >= 0)
        byteString = atob(dataURI.split(',')[1]);
    else
        byteString = unescape(dataURI.split(',')[1]);

    // separate out the mime component
    var mimeString = dataURI.split(',')[0].split(':')[1].split(';')[0];

    // write the bytes of the string to a typed array
    var ia = new Uint8Array(byteString.length);
    for (var i = 0; i < byteString.length; i++) {
        ia[i] = byteString.charCodeAt(i);
    }

    return new Blob([ia], {type:mimeString});
}

从那里,将数据追加到表单,以便将其作为文件上传是很容易的:

var dataURL = canvas.toDataURL('image/jpeg', 0.5);
var blob = dataURItoBlob(dataURL);
var fd = new FormData(document.forms[0]);
fd.append("canvasImage", blob);

以下是Stoive回答的ES6版本:

export class ImageDataConverter {
  constructor(dataURI) {
    this.dataURI = dataURI;
  }

  getByteString() {
    let byteString;
    if (this.dataURI.split(',')[0].indexOf('base64') >= 0) {
      byteString = atob(this.dataURI.split(',')[1]);
    } else {
      byteString = decodeURI(this.dataURI.split(',')[1]);
    }
    return byteString;
  }

  getMimeString() {
    return this.dataURI.split(',')[0].split(':')[1].split(';')[0];
  }

  convertToTypedArray() {
    let byteString = this.getByteString();
    let ia = new Uint8Array(byteString.length);
    for (let i = 0; i < byteString.length; i++) {
      ia[i] = byteString.charCodeAt(i);
    }
    return ia;
  }

  dataURItoBlob() {
    let mimeString = this.getMimeString();
    let intArray = this.convertToTypedArray();
    return new Blob([intArray], {type: mimeString});
  }
}

用法:

const dataURL = canvas.toDataURL('image/jpeg', 0.5);
const blob = new ImageDataConverter(dataURL).dataURItoBlob();
let fd = new FormData(document.forms[0]);
fd.append("canvasImage", blob);