| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728 | /* Blob.js * A Blob, File, FileReader & URL implementation. * 2019-04-19 * * By Eli Grey, http://eligrey.com * By Jimmy Wärting, https://github.com/jimmywarting * License: MIT *   See https://github.com/eligrey/Blob.js/blob/master/LICENSE.md */;(function () {  var global =    typeof window === "object" ? window : typeof self === "object" ? self : this  var BlobBuilder =    global.BlobBuilder ||    global.WebKitBlobBuilder ||    global.MSBlobBuilder ||    global.MozBlobBuilder  global.URL =    global.URL ||    global.webkitURL ||    function (href, a) {      a = document.createElement("a")      a.href = href      return a    }  var origBlob = global.Blob  var createObjectURL = URL.createObjectURL  var revokeObjectURL = URL.revokeObjectURL  var strTag = global.Symbol && global.Symbol.toStringTag  var blobSupported = false  var blobSupportsArrayBufferView = false  var arrayBufferSupported = !!global.ArrayBuffer  var blobBuilderSupported =    BlobBuilder && BlobBuilder.prototype.append && BlobBuilder.prototype.getBlob  try {    // Check if Blob constructor is supported    blobSupported = new Blob(["ä"]).size === 2    // Check if Blob constructor supports ArrayBufferViews    // Fails in Safari 6, so we need to map to ArrayBuffers there.    blobSupportsArrayBufferView = new Blob([new Uint8Array([1, 2])]).size === 2  } catch (e) {}  /**   * Helper function that maps ArrayBufferViews to ArrayBuffers   * Used by BlobBuilder constructor and old browsers that didn't   * support it in the Blob constructor.   */  function mapArrayBufferViews (ary) {    return ary.map(function (chunk) {      if (chunk.buffer instanceof ArrayBuffer) {        var buf = chunk.buffer        // if this is a subarray, make a copy so we only        // include the subarray region from the underlying buffer        if (chunk.byteLength !== buf.byteLength) {          var copy = new Uint8Array(chunk.byteLength)          copy.set(new Uint8Array(buf, chunk.byteOffset, chunk.byteLength))          buf = copy.buffer        }        return buf      }      return chunk    })  }  function BlobBuilderConstructor (ary, options) {    options = options || {}    var bb = new BlobBuilder()    mapArrayBufferViews(ary).forEach(function (part) {      bb.append(part)    })    return options.type ? bb.getBlob(options.type) : bb.getBlob()  }  function BlobConstructor (ary, options) {    return new origBlob(mapArrayBufferViews(ary), options || {})  }  if (global.Blob) {    BlobBuilderConstructor.prototype = Blob.prototype    BlobConstructor.prototype = Blob.prototype  }  /********************************************************/  /*               String Encoder fallback                */  /********************************************************/  function stringEncode (string) {    var pos = 0    var len = string.length    var out = []    var Arr = global.Uint8Array || Array // Use byte array when possible    var at = 0 // output position    var tlen = Math.max(32, len + (len >> 1) + 7) // 1.5x size    var target = new Arr((tlen >> 3) << 3) // ... but at 8 byte offset    while (pos < len) {      var value = string.charCodeAt(pos++)      if (value >= 0xd800 && value <= 0xdbff) {        // high surrogate        if (pos < len) {          var extra = string.charCodeAt(pos)          if ((extra & 0xfc00) === 0xdc00) {            ++pos            value = ((value & 0x3ff) << 10) + (extra & 0x3ff) + 0x10000          }        }        if (value >= 0xd800 && value <= 0xdbff) {          continue // drop lone surrogate        }      }      // expand the buffer if we couldn't write 4 bytes      if (at + 4 > target.length) {        tlen += 8 // minimum extra        tlen *= 1.0 + (pos / string.length) * 2 // take 2x the remaining        tlen = (tlen >> 3) << 3 // 8 byte offset        const update = new Uint8Array(tlen)        update.set(target)        target = update      }      if ((value & 0xffffff80) === 0) {        // 1-byte        target[at++] = value // ASCII        continue      } else if ((value & 0xfffff800) === 0) {        // 2-byte        target[at++] = ((value >> 6) & 0x1f) | 0xc0      } else if ((value & 0xffff0000) === 0) {        // 3-byte        target[at++] = ((value >> 12) & 0x0f) | 0xe0        target[at++] = ((value >> 6) & 0x3f) | 0x80      } else if ((value & 0xffe00000) === 0) {        // 4-byte        target[at++] = ((value >> 18) & 0x07) | 0xf0        target[at++] = ((value >> 12) & 0x3f) | 0x80        target[at++] = ((value >> 6) & 0x3f) | 0x80      } else {        // FIXME: do we care        continue      }      target[at++] = (value & 0x3f) | 0x80    }    return target.slice(0, at)  }  /********************************************************/  /*               String Decoder fallback                */  /********************************************************/  function stringDecode (buf) {    var end = buf.length    var res = []    var i = 0    while (i < end) {      var firstByte = buf[i]      var codePoint = null      var bytesPerSequence =        firstByte > 0xef ? 4 : firstByte > 0xdf ? 3 : firstByte > 0xbf ? 2 : 1      if (i + bytesPerSequence <= end) {        var secondByte, thirdByte, fourthByte, tempCodePoint        switch (bytesPerSequence) {          case 1:            if (firstByte < 0x80) {              codePoint = firstByte            }            break          case 2:            secondByte = buf[i + 1]            if ((secondByte & 0xc0) === 0x80) {              tempCodePoint = ((firstByte & 0x1f) << 0x6) | (secondByte & 0x3f)              if (tempCodePoint > 0x7f) {                codePoint = tempCodePoint              }            }            break          case 3:            secondByte = buf[i + 1]            thirdByte = buf[i + 2]            if ((secondByte & 0xc0) === 0x80 && (thirdByte & 0xc0) === 0x80) {              tempCodePoint =                ((firstByte & 0xf) << 0xc) |                ((secondByte & 0x3f) << 0x6) |                (thirdByte & 0x3f)              if (                tempCodePoint > 0x7ff &&                (tempCodePoint < 0xd800 || tempCodePoint > 0xdfff)              ) {                codePoint = tempCodePoint              }            }            break          case 4:            secondByte = buf[i + 1]            thirdByte = buf[i + 2]            fourthByte = buf[i + 3]            if (              (secondByte & 0xc0) === 0x80 &&              (thirdByte & 0xc0) === 0x80 &&              (fourthByte & 0xc0) === 0x80            ) {              tempCodePoint =                ((firstByte & 0xf) << 0x12) |                ((secondByte & 0x3f) << 0xc) |                ((thirdByte & 0x3f) << 0x6) |                (fourthByte & 0x3f)              if (tempCodePoint > 0xffff && tempCodePoint < 0x110000) {                codePoint = tempCodePoint              }            }        }      }      if (codePoint === null) {        // we did not generate a valid codePoint so insert a        // replacement char (U+FFFD) and advance only 1 byte        codePoint = 0xfffd        bytesPerSequence = 1      } else if (codePoint > 0xffff) {        // encode to utf16 (surrogate pair dance)        codePoint -= 0x10000        res.push(((codePoint >>> 10) & 0x3ff) | 0xd800)        codePoint = 0xdc00 | (codePoint & 0x3ff)      }      res.push(codePoint)      i += bytesPerSequence    }    var len = res.length    var str = ""    var i = 0    while (i < len) {      str += String.fromCharCode.apply(String, res.slice(i, (i += 0x1000)))    }    return str  }  // string -> buffer  var textEncode =    typeof TextEncoder === "function"      ? TextEncoder.prototype.encode.bind(new TextEncoder())      : stringEncode  // buffer -> string  var textDecode =    typeof TextDecoder === "function"      ? TextDecoder.prototype.decode.bind(new TextDecoder())      : stringDecode  function FakeBlobBuilder () {    function isDataView (obj) {      return obj && DataView.prototype.isPrototypeOf(obj)    }    function bufferClone (buf) {      var view = new Array(buf.byteLength)      var array = new Uint8Array(buf)      var i = view.length      while (i--) {        view[i] = array[i]      }      return view    }    function array2base64 (input) {      var byteToCharMap =        "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="      var output = []      for (var i = 0; i < input.length; i += 3) {        var byte1 = input[i]        var haveByte2 = i + 1 < input.length        var byte2 = haveByte2 ? input[i + 1] : 0        var haveByte3 = i + 2 < input.length        var byte3 = haveByte3 ? input[i + 2] : 0        var outByte1 = byte1 >> 2        var outByte2 = ((byte1 & 0x03) << 4) | (byte2 >> 4)        var outByte3 = ((byte2 & 0x0f) << 2) | (byte3 >> 6)        var outByte4 = byte3 & 0x3f        if (!haveByte3) {          outByte4 = 64          if (!haveByte2) {            outByte3 = 64          }        }        output.push(          byteToCharMap[outByte1],          byteToCharMap[outByte2],          byteToCharMap[outByte3],          byteToCharMap[outByte4]        )      }      return output.join("")    }    var create =      Object.create ||      function (a) {        function c () {}        c.prototype = a        return new c()      }    if (arrayBufferSupported) {      var viewClasses = [        "[object Int8Array]",        "[object Uint8Array]",        "[object Uint8ClampedArray]",        "[object Int16Array]",        "[object Uint16Array]",        "[object Int32Array]",        "[object Uint32Array]",        "[object Float32Array]",        "[object Float64Array]"      ]      var isArrayBufferView =        ArrayBuffer.isView ||        function (obj) {          return (            obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1          )        }    }    function concatTypedarrays (chunks) {      var size = 0      var i = chunks.length      while (i--) {        size += chunks[i].length      }      var b = new Uint8Array(size)      var offset = 0      for (i = 0, l = chunks.length; i < l; i++) {        var chunk = chunks[i]        b.set(chunk, offset)        offset += chunk.byteLength || chunk.length      }      return b    }    /********************************************************/    /*                   Blob constructor                   */    /********************************************************/    function Blob (chunks, opts) {      chunks = chunks || []      opts = opts == null ? {} : opts      for (var i = 0, len = chunks.length; i < len; i++) {        var chunk = chunks[i]        if (chunk instanceof Blob) {          chunks[i] = chunk._buffer        } else if (typeof chunk === "string") {          chunks[i] = textEncode(chunk)        } else if (          arrayBufferSupported &&          (ArrayBuffer.prototype.isPrototypeOf(chunk) ||            isArrayBufferView(chunk))        ) {          chunks[i] = bufferClone(chunk)        } else if (arrayBufferSupported && isDataView(chunk)) {          chunks[i] = bufferClone(chunk.buffer)        } else {          chunks[i] = textEncode(String(chunk))        }      }      this._buffer = global.Uint8Array        ? concatTypedarrays(chunks)        : [].concat.apply([], chunks)      this.size = this._buffer.length      this.type = opts.type || ""      if (/[^\u0020-\u007E]/.test(this.type)) {        this.type = ""      } else {        this.type = this.type.toLowerCase()      }    }    Blob.prototype.arrayBuffer = function () {      return Promise.resolve(this._buffer)    }    Blob.prototype.text = function () {      return Promise.resolve(textDecode(this._buffer))    }    Blob.prototype.slice = function (start, end, type) {      var slice = this._buffer.slice(start || 0, end || this._buffer.length)      return new Blob([slice], { type: type })    }    Blob.prototype.toString = function () {      return "[object Blob]"    }    /********************************************************/    /*                   File constructor                   */    /********************************************************/    function File (chunks, name, opts) {      opts = opts || {}      var a = Blob.call(this, chunks, opts) || this      a.name = name.replace(/\//g, ":")      a.lastModifiedDate = opts.lastModified        ? new Date(opts.lastModified)        : new Date()      a.lastModified = +a.lastModifiedDate      return a    }    File.prototype = create(Blob.prototype)    File.prototype.constructor = File    if (Object.setPrototypeOf) {      Object.setPrototypeOf(File, Blob)    } else {      try {        File.__proto__ = Blob      } catch (e) {}    }    File.prototype.toString = function () {      return "[object File]"    }    /********************************************************/    /*                FileReader constructor                */    /********************************************************/    function FileReader () {      if (!(this instanceof FileReader)) {        throw new TypeError(          "Failed to construct 'FileReader': Please use the 'new' operator, this DOM object constructor cannot be called as a function."        )      }      var delegate = document.createDocumentFragment()      this.addEventListener = delegate.addEventListener      this.dispatchEvent = function (evt) {        var local = this["on" + evt.type]        if (typeof local === "function") local(evt)        delegate.dispatchEvent(evt)      }      this.removeEventListener = delegate.removeEventListener    }    function _read (fr, blob, kind) {      if (!(blob instanceof Blob)) {        throw new TypeError(          "Failed to execute '" +            kind +            "' on 'FileReader': parameter 1 is not of type 'Blob'."        )      }      fr.result = ""      setTimeout(function () {        this.readyState = FileReader.LOADING        fr.dispatchEvent(new Event("load"))        fr.dispatchEvent(new Event("loadend"))      })    }    FileReader.EMPTY = 0    FileReader.LOADING = 1    FileReader.DONE = 2    FileReader.prototype.error = null    FileReader.prototype.onabort = null    FileReader.prototype.onerror = null    FileReader.prototype.onload = null    FileReader.prototype.onloadend = null    FileReader.prototype.onloadstart = null    FileReader.prototype.onprogress = null    FileReader.prototype.readAsDataURL = function (blob) {      _read(this, blob, "readAsDataURL")      this.result =        "data:" + blob.type + ";base64," + array2base64(blob._buffer)    }    FileReader.prototype.readAsText = function (blob) {      _read(this, blob, "readAsText")      this.result = textDecode(blob._buffer)    }    FileReader.prototype.readAsArrayBuffer = function (blob) {      _read(this, blob, "readAsText")      // return ArrayBuffer when possible      this.result = (blob._buffer.buffer || blob._buffer).slice()    }    FileReader.prototype.abort = function () {}    /********************************************************/    /*                         URL                          */    /********************************************************/    URL.createObjectURL = function (blob) {      return blob instanceof Blob        ? "data:" + blob.type + ";base64," + array2base64(blob._buffer)        : createObjectURL.call(URL, blob)    }    URL.revokeObjectURL = function (url) {      revokeObjectURL && revokeObjectURL.call(URL, url)    }    /********************************************************/    /*                         XHR                          */    /********************************************************/    var _send = global.XMLHttpRequest && global.XMLHttpRequest.prototype.send    if (_send) {      XMLHttpRequest.prototype.send = function (data) {        if (data instanceof Blob) {          this.setRequestHeader("Content-Type", data.type)          _send.call(this, textDecode(data._buffer))        } else {          _send.call(this, data)        }      }    }    global.FileReader = FileReader    global.File = File    global.Blob = Blob  }  function fixFileAndXHR () {    var isIE =      !!global.ActiveXObject ||      ("-ms-scroll-limit" in document.documentElement.style &&        "-ms-ime-align" in document.documentElement.style)    // Monkey patched    // IE don't set Content-Type header on XHR whose body is a typed Blob    // https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/6047383    var _send = global.XMLHttpRequest && global.XMLHttpRequest.prototype.send    if (isIE && _send) {      XMLHttpRequest.prototype.send = function (data) {        if (data instanceof Blob) {          this.setRequestHeader("Content-Type", data.type)          _send.call(this, data)        } else {          _send.call(this, data)        }      }    }    try {      new File([], "")    } catch (e) {      try {        var klass = new Function(          "class File extends Blob {" +            "constructor(chunks, name, opts) {" +            "opts = opts || {};" +            "super(chunks, opts || {});" +            "this.name = name.replace(///g, \":\");" +            "this.lastModifiedDate = opts.lastModified ? new Date(opts.lastModified) : new Date();" +            "this.lastModified = +this.lastModifiedDate;" +            "}};" +            "return new File([], \"\"), File"        )()        global.File = klass      } catch (e) {        var klass = function (b, d, c) {          var blob = new Blob(b, c)          var t =            c && void 0 !== c.lastModified              ? new Date(c.lastModified)              : new Date()          blob.name = d.replace(/\//g, ":")          blob.lastModifiedDate = t          blob.lastModified = +t          blob.toString = function () {            return "[object File]"          }          if (strTag) {            blob[strTag] = "File"          }          return blob        }        global.File = klass      }    }  }  if (blobSupported) {    fixFileAndXHR()    global.Blob = blobSupportsArrayBufferView ? global.Blob : BlobConstructor  } else if (blobBuilderSupported) {    fixFileAndXHR()    global.Blob = BlobBuilderConstructor  } else {    FakeBlobBuilder()  }  if (strTag) {    File.prototype[strTag] = "File"    Blob.prototype[strTag] = "Blob"    FileReader.prototype[strTag] = "FileReader"  }  var blob = global.Blob.prototype  var stream  function promisify (obj) {    return new Promise(function (resolve, reject) {      obj.onload = obj.onerror = function (evt) {        obj.onload = obj.onerror = null        evt.type === "load"          ? resolve(obj.result || obj)          : reject(new Error("Failed to read the blob/file"))      }    })  }  try {    new ReadableStream({ type: "bytes" })    stream = function stream () {      var position = 0      var blob = this      return new ReadableStream({        type: "bytes",        autoAllocateChunkSize: 524288,        pull: function (controller) {          var v = controller.byobRequest.view          var chunk = blob.slice(position, position + v.byteLength)          return chunk.arrayBuffer().then(function (buffer) {            var uint8array = new Uint8Array(buffer)            var bytesRead = uint8array.byteLength            position += bytesRead            v.set(uint8array)            controller.byobRequest.respond(bytesRead)            if (position >= blob.size) controller.close()          })        }      })    }  } catch (e) {    try {      new ReadableStream({})      stream = function stream (blob) {        var position = 0        var blob = this        return new ReadableStream({          pull: function (controller) {            var chunk = blob.slice(position, position + 524288)            return chunk.arrayBuffer().then(function (buffer) {              position += buffer.byteLength              var uint8array = new Uint8Array(buffer)              controller.enqueue(uint8array)              if (position == blob.size) controller.close()            })          }        })      }    } catch (e) {      try {        new Response("").body.getReader().read()        stream = function stream () {          return new Response(this).body        }      } catch (e) {        stream = function stream () {          throw new Error(            "Include https://github.com/MattiasBuelens/web-streams-polyfill"          )        }      }    }  }  if (!blob.arrayBuffer) {    blob.arrayBuffer = function arrayBuffer () {      var fr = new FileReader()      fr.readAsArrayBuffer(this)      return promisify(fr)    }  }  if (!blob.text) {    blob.text = function text () {      var fr = new FileReader()      fr.readAsText(this)      return promisify(fr)    }  }  if (!blob.stream) {    blob.stream = stream  }})()
 |