在iOS Safari中使用XHR下载文件

时间:2019-02-22 10:31:04

标签: javascript ios safari data-uri

我正在尝试添加功能来下载服务器上托管的文件。要访问文件,我必须发送Authorization头,因此我必须发送XHR请求以从服务器获取文件。由于文件内容位于变量中,因此我必须创建一个数据url以使其可用作定位标记的href属性,然后以编程方式单击它以下载文件。 在几乎所有浏览器(IE11除外,我为它编写了单独的代码)中,它都运行良好,但是在iOS Safari(在某些版本的iOS中)中,它给出了错误。这是我正在使用的代码-

var isBrowserIE = window.navigator && window.navigator.msSaveOrOpenBlob;
var dataHref = 'https://example.com/doc.pdf';
var xhr = new XMLHttpRequest();
xhr.open('GET', dataHref, true);
xhr.setRequestHeader('Content-Type', 'application/pdf');
xhr.setRequestHeader('Authorization', 'Bearer ' + token);
xhr.responseType = isBrowserIE ? 'blob' : 'arraybuffer';
xhr.onload = function (e) {
    if (this.status == 200) {
        //For IE11
        if (isBrowserIE) {
            // Create a new Blob object using the response data of the onload object
            var blob = new Blob([this.response], { type: 'application/pdf' });

            var bool = window.navigator.msSaveOrOpenBlob(blob, docName);
            if (!bool) {
                alert("Download failed, Please try again later");
            }
        } else {
            var uInt8Array = new Uint8Array(this.response);
            var i = uInt8Array.length;
            var binaryString = new Array(i);
            while (i--) {
                binaryString[i] = String.fromCharCode(uInt8Array[i]);
            }
            var data = binaryString.join('');

            var base64 = window.btoa(data);

            var dataUrl = 'data:application/octet-stream;charset=utf-16le;base64,' + base64;
            var element = document.createElement('a');
            element.setAttribute('href', dataUrl);
            element.setAttribute('download', 'doc.pdf');
            element.style.display = 'none';
            document.body.appendChild(element);
            element.click();
            document.body.removeChild(element);
        }
    } else {
        alert("Download failed, Please try again later");
        closeWindow();
    }
};

xhr.send();

这是与我有关的可能错误-

Safari cannot open the page.<br><br>The error was: “Data URL decoding&nbsp;failed”.

我错过了什么,这是此错误的原因吗?该错误仅在iPad 4和iPad 5中发生,但在iPad mini和iPhone XR中起作用。不知道为什么在某些版本的iOS设备上不能正常工作。

1 个答案:

答案 0 :(得分:2)

所以,我终于弄清楚了。这是我的最终代码,并带有注释说明(对不起,ES5代码,我需要支持IE11,并且当前项目尚未使用babel)-

/* exported DownloadHandler */
/* global Uint8Array*/
var DownloadHandler = (function() {
  function isMobileDevice() {
    return navigator.userAgent.match(/Android|iPhone|iPad|iPod|BlackBerry|Opera Mini|IEMobile/i);
  }
  function isChromeBrowser() {
    return navigator.userAgent.match(/Crios|Chrome/i);
  }
  function isIEBrowser() {
    return window.navigator && window.navigator.msSaveOrOpenBlob;
  }
  function isSafariBrowser() {
    return navigator.userAgent.match(/Safari/i);
  }
  function getResponseType() {
    // Both Desktop Chrome and IE supports blob properly
    // Chrome also supports Data URI way, but it fails miserably when the file size is more than 2 MB (Not sure about the exact limit though).
    if (isIEBrowser() || isChromeBrowser()) {
      return 'blob';
    } else if (isMobileDevice()) {
      return 'arraybuffer';
    }
    return 'blob';
  }
  function getBlobUriFromResponse(response) {
    var blob = new Blob([response], { type: 'application/pdf' });
    var downloadUrl = URL.createObjectURL(blob);
    return downloadUrl;
  }

  function getDataUriFromResponse(response) {
    var uInt8Array = new Uint8Array(response);
    var i = uInt8Array.length;
    var binaryString = new Array(i);
    while (i--) {
      binaryString[i] = String.fromCharCode(uInt8Array[i]);
    }
    var data = binaryString.join('');

    var base64 = window.btoa(data);

    var dataUrl = 'data:application/octet-stream;charset=utf-16le;base64,' + base64;
    return dataUrl;
  }
  function downloadFileUsingXHR(fileName, fileUrl, fileMimeType, requestType, headersList) {
    var xhr = new XMLHttpRequest();
    xhr.open(requestType, fileUrl, true);
    xhr.setRequestHeader('Content-Type', fileMimeType);
    for (var i = 0; i < headersList.length; i++) {
      var header = headersList[i];
      xhr.setRequestHeader(header.key, header.value);
    }
    xhr.responseType = getResponseType();
    xhr.onload = function() {
      if (this.status == 200) {
        //For IE11
        //IE uses blob with vendor specific code
        if (isIEBrowser()) {
          // Create a new Blob object using the response data of the onload object
          var blob = new Blob([this.response], { type: fileMimeType });

          var bool = window.navigator.msSaveOrOpenBlob(blob, fileName);
          if (!bool) {
            alert('Download failed, Please try again later');
          }
        } else {
          var dataUrl;
          if (this.responseType === 'blob') {
            dataUrl = getBlobUriFromResponse(this.response);
          } else {
            dataUrl = getDataUriFromResponse(this.response);
          }
          var element = document.createElement('a');
          // Safari doesn't work well with blank targets
          if (!isSafariBrowser()) {
            element.setAttribute('target', '_blank');
          }
          element.setAttribute('href', dataUrl);
          element.setAttribute('download', fileName);
          element.style.display = 'none';
          document.body.appendChild(element);
          element.click();
          document.body.removeChild(element);
        }
      } else {
        alert('Download failed, Please try again later');
      }
    };

    xhr.send();
  }
  return {
    downloadFileUsingXHR: downloadFileUsingXHR
  };
})();

以下是使用上述代码的方法:

DownloadHandler.downloadFileUsingXHR('example.pdf', 'https://example.com/doc.pdf', 'application/pdf','GET',[{key:'Authorization',value:'Bearer ' + token}]);

稍后我可能会将其转换为库,并在此处发布链接。我也将有机会完善代码