菜鸟教程小白 发表于 2022-12-13 05:23:08

javascript - 在 iOS Safari 中使用 XHR 下载文件


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

<pre><code>var isBrowserIE = window.navigator &amp;&amp; window.navigator.msSaveOrOpenBlob;
var dataHref = &#39;https://example.com/doc.pdf&#39;;
var xhr = new XMLHttpRequest();
xhr.open(&#39;GET&#39;, dataHref, true);
xhr.setRequestHeader(&#39;Content-Type&#39;, &#39;application/pdf&#39;);
xhr.setRequestHeader(&#39;Authorization&#39;, &#39;Bearer &#39; + token);
xhr.responseType = isBrowserIE ? &#39;blob&#39; : &#39;arraybuffer&#39;;
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(, { type: &#39;application/pdf&#39; });

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

            var base64 = window.btoa(data);

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

xhr.send();
</code></pre>

<p>这是我可能遇到的相关错误 - </p>

<p> <code>Safari cannot open the page.&lt;br&gt;&lt;br&gt;The error was: “Data URL decoding&amp;nbsp;failed”.</code> </p>

<p>我是否遗漏了什么导致此错误的原因?该错误仅发生在 iPad 4 和 iPad 5 中,但适用于 iPad mini 和 iPhone XR。不确定为什么它可以在某些版本的 iOS 设备上运行,而在其他版本上不能运行。</p></p>
                                    <br><hr><h1><strong>Best Answer-推荐答案</ strong></h1><br>
                                            <p><p>所以,我终于想通了。这是我的最终代码,在注释中有解释(对不起,ES5 代码,我需要支持 IE11,而当前项目还没有使用 babel)- </p>

<pre><code>/* 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 &amp;&amp; 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 &#39;blob&#39;;
    } else if (isMobileDevice()) {
      return &#39;arraybuffer&#39;;
    }
    return &#39;blob&#39;;
}
function getBlobUriFromResponse(response) {
    var blob = new Blob(, { type: &#39;application/pdf&#39; });
    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 = String.fromCharCode(uInt8Array);
    }
    var data = binaryString.join(&#39;&#39;);

    var base64 = window.btoa(data);

    var dataUrl = &#39;data:application/octet-stream;charset=utf-16le;base64,&#39; + base64;
    return dataUrl;
}
function downloadFileUsingXHR(fileName, fileUrl, fileMimeType, requestType, headersList) {
    var xhr = new XMLHttpRequest();
    xhr.open(requestType, fileUrl, true);
    xhr.setRequestHeader(&#39;Content-Type&#39;, fileMimeType);
    for (var i = 0; i &lt; headersList.length; i++) {
      var header = headersList;
      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(, { type: fileMimeType });

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

    xhr.send();
}
return {
    downloadFileUsingXHR: downloadFileUsingXHR
};
})();
</code></pre>

<p>下面是如何使用上面的代码:</p>

<pre><code>DownloadHandler.downloadFileUsingXHR(&#39;example.pdf&#39;, &#39;https://example.com/doc.pdf&#39;, &#39;application/pdf&#39;,&#39;GET&#39;,[{key:&#39;Authorization&#39;,value:&#39;Bearer &#39; + token}]);
</code></pre>

<p>稍后我可能会将其转换为一个库并在此处发布一个链接。我也会有机会完善代码</p></p>
                                   
                                                <p style="font-size: 20px;">关于javascript - 在 iOS Safari 中使用 XHR 下载文件,我们在Stack Overflow上找到一个类似的问题:
                                                        <a href="https://stackoverflow.com/questions/54825077/" rel="noreferrer noopener nofollow" style="color: red;">
                                                                https://stackoverflow.com/questions/54825077/
                                                        </a>
                                                </p>
                                       
页: [1]
查看完整版本: javascript - 在 iOS Safari 中使用 XHR 下载文件