/* 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 } })()