├── .gitattributes ├── .gitignore ├── src ├── native-func.js ├── image-loaders.js ├── ImageBitmap-wrapper.js ├── constants.js ├── options-parser.js ├── check-usability.js ├── misc.js ├── monkeypatch.js ├── image-cropper.js └── test-support.js ├── LICENSE ├── README.md └── dist └── createImageBitmap.js /.gitattributes: -------------------------------------------------------------------------------- 1 | # Auto detect text files and perform LF normalization 2 | * text=auto 3 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules/* 2 | package-lock.json 3 | .DS_Store 4 | webpack.config.js 5 | package.json 6 | -------------------------------------------------------------------------------- /src/native-func.js: -------------------------------------------------------------------------------- 1 | const native_func = globalThis.createImageBitmap; 2 | const nativeCreateImageBitmap = native_func ? 3 | (...args) => native_func.call( globalThis, ...args ) : false; 4 | 5 | export { nativeCreateImageBitmap }; 6 | -------------------------------------------------------------------------------- /src/image-loaders.js: -------------------------------------------------------------------------------- 1 | import { XLINK_NS, error_messages } from "./constants.js"; 2 | import { checkUsability } from "./check-usability.js"; 3 | function loadImage( url ) { 4 | const img = new Image(); 5 | img.src = url; 6 | return new Promise( (resolve, reject) => { 7 | img.onload = (evt) => { 8 | // might be broken 9 | checkUsability( img ); 10 | resolve( img ); 11 | }; 12 | img.onerror = (evt) => { 13 | const error = new DOMException( 14 | error_messages.INVALID_STATE_IMAGE, 15 | "InvalidStateError" 16 | ); 17 | reject( error ); 18 | }; 19 | } ); 20 | } 21 | function loadSVGImage( elem ) { 22 | const url = elem.getAttribute( "href" ) || elem.getAttributeNS( XLINK_NS, "href" ); 23 | return loadImage( url ); 24 | } 25 | 26 | export { loadImage, loadSVGImage }; 27 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2021 Kaiido 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /src/ImageBitmap-wrapper.js: -------------------------------------------------------------------------------- 1 | import { cropImage } from "./image-cropper.js"; 2 | import { nativeCreateImageBitmap } from "./native-func.js"; 3 | 4 | const getCanvasProtoDesc = ( prop ) => 5 | Object.getOwnPropertyDescriptor( HTMLCanvasElement.prototype, prop ); 6 | 7 | // we can't use the 'class' syntax here 8 | // see https://esdiscuss.org/topic/classes-and-enumerability 9 | function ImageBitmapPolyfill() { 10 | throw new TypeError( "Illegal Constructor" ); 11 | } 12 | ImageBitmapPolyfill.prototype = Object.create( { 13 | close() { 14 | // there is no real way to "detach" a canvas's buffer 15 | // after a drawing context has been created. 16 | // the closest is to set its width and height to 0 17 | const width_setter = getCanvasProtoDesc( "width" ).set; 18 | width_setter.call( this, 0); 19 | const height_setter = getCanvasProtoDesc( "height" ).set; 20 | height_setter.call( this, 0 ); 21 | }, 22 | get width() { 23 | const canvas_getter = getCanvasProtoDesc( "width" ).get; 24 | return canvas_getter.call( this ); 25 | }, 26 | get height() { 27 | const canvas_getter = getCanvasProtoDesc( "height" ).get; 28 | return canvas_getter.call( this ); 29 | }, 30 | get [ Symbol.toStringTag ]() { 31 | return "ImageBitmap"; 32 | } 33 | } ); 34 | 35 | function customWrapper( source ) { 36 | if( !(source instanceof HTMLCanvasElement) ) { 37 | source = cropImage( source ); 38 | } 39 | Object.setPrototypeOf( source, ImageBitmapPolyfill.prototype ); 40 | return source; 41 | } 42 | 43 | function convertToImageBitmap( source ) { 44 | if( (source instanceof globalThis.ImageBitmap) ) { 45 | return source; 46 | } 47 | if( nativeCreateImageBitmap ) { 48 | return nativeCreateImageBitmap( source ); 49 | } 50 | return customWrapper( source ); 51 | } 52 | 53 | export { convertToImageBitmap, ImageBitmapPolyfill }; 54 | -------------------------------------------------------------------------------- /src/constants.js: -------------------------------------------------------------------------------- 1 | // the properties we look for 2 | // the ones that aren't real property names are prefixed with a _ 3 | // they are separated in two lists 4 | // so we can perform synchronous and async tests separately 5 | const synchronous_polyfills = [ 6 | "_imageBitmapOptions", 7 | "resizeWidth", 8 | "resizeHeight", 9 | "resizeQuality", 10 | "imageOrientation" 11 | ] 12 | const asynchronous_polyfills = [ 13 | "_Blob", 14 | "_ImageData", 15 | "_SVGBlob", 16 | "_SVGImageElement", 17 | ]; 18 | // imageBitmapOptions properties we know of, 19 | // we don't necessarily have a polyfill for all of these 20 | // we still test them anyway 21 | const known_properties = [ 22 | "resizeWidth", 23 | "resizeHeight", 24 | "resizeQuality", 25 | "imageOrientation", 26 | "premultiplyAlpha", 27 | "colorSpaceConversion" 28 | ]; 29 | // https://html.spec.whatwg.org/multipage/imagebitmap-and-animations.html#imagebitmap 30 | const enums = { 31 | ResizeQuality: [ "pixelated", "low", "medium", "high" ], 32 | ImageOrientation: [ "none", "flipY" ] 33 | }; 34 | 35 | const error_messages = { 36 | COMMON_HEADER: "Failed to execute 'createImageBitmap': ", 37 | INVALID_STATE_IMAGE: "Provided image was in an invalid state.", 38 | ARGUMENT_COUNT_1: "At least one argument is required.", 39 | ARGUMENT_COUNT_N: "%s is not a valid argument count for any overload", 40 | CROP_RECT_ZERO: "The crop rect width passed to createImageBitmap must be nonzero", 41 | ALLOCATION_FAILED: "The ImageBitmap could not be allocated.", 42 | INVALID_SOURCE: "Argument 1 could not be converted to any of: HTMLImageElement, " + 43 | "SVGImageElement, HTMLCanvasElement, HTMLVideoElement, ImageBitmap, Blob, " + 44 | "CanvasRenderingContext2D, ImageData.", 45 | ENUM: "'%v' is not a valid value for enumeration %e", 46 | ALLOCATION: "The ImageBitmap couldn't be allocated." 47 | } 48 | 49 | const canvas_sources = [ 50 | "HTMLImageElement", 51 | "SVGImageElement", 52 | "HTMLVideoElement", 53 | "HTMLCanvasElement", 54 | "OffscreenCanvas", 55 | "ImageBitmap" 56 | ]; 57 | 58 | const SVG_MIME = "image/svg+xml"; 59 | const SVG_NS = "http://www.w3.org/2000/svg"; 60 | const XLINK_NS = "http://www.w3.org/1999/xlink"; 61 | const GIF_URL = ""; 62 | 63 | export { 64 | synchronous_polyfills, 65 | asynchronous_polyfills, 66 | enums, 67 | known_properties, 68 | error_messages, 69 | canvas_sources, 70 | SVG_MIME, 71 | SVG_NS, 72 | XLINK_NS, 73 | GIF_URL 74 | }; 75 | -------------------------------------------------------------------------------- /src/options-parser.js: -------------------------------------------------------------------------------- 1 | import { convertToLong, enforceRangeLong } from "./misc.js"; 2 | import { enums, error_messages } from "./constants.js"; 3 | 4 | const common_error_header = error_messages.COMMON_HEADER; 5 | 6 | function testEnum( options, property, enum_name ) { 7 | const enum_list = enums[ enum_name ]; 8 | const value = options[ property ]; 9 | if( value !== undefined && !enum_list.includes( value ) ) { 10 | const error_msg = error_messages.ENUM 11 | .replace( "%v", value ) 12 | .replace( "%e", enum_name ); 13 | throw new TypeError( error_msg ); 14 | } 15 | } 16 | function parseOptions( source, ...args ) { 17 | 18 | let [ sx, sy, sw, sh ] = args.map( convertToLong ); 19 | let cropRect; // optional 20 | const options = (args.length === 1 ? args[ 0 ] : args[ 4 ]) || {}; 21 | 22 | const resizeWidth = ("resizeWidth" in options) && 23 | enforceRangeLong( options.resizeWidth ); 24 | const resizeHeight = ("resizeHeight" in options) && 25 | enforceRangeLong( options.resizeHeight ); 26 | 27 | testEnum( options, "resizeQuality", "ResizeQuality" ); 28 | testEnum( options, "imageOrientation", "ImageOrientation" ); 29 | 30 | const arguments_count = arguments.length; 31 | 32 | if( !arguments_count ) { 33 | throw new TypeError( common_error_header + error_messages.ARGUMENT_COUNT_1 ); 34 | } 35 | if( arguments_count > 2 && arguments_count < 5 ) { 36 | const error_count_msg = error_messages.ARGUMENT_COUNT_N.replace( "%s", arguments_count ); 37 | throw new TypeError( common_error_header + error_count_msg ); 38 | } 39 | if( arguments_count >= 5 ) { // 'sw' and 'sh' are defined 40 | if( !sw || !sh ) { // convertToLong converts invalid longs to NaN 41 | throw new RangeError( common_error_header + error_messages.CROP_RECT_ZERO ); 42 | } 43 | // invalid sx and sy are treated as 0 44 | sx = sx || 0; 45 | sy = sy || 0; 46 | cropRect = { sx, sy, sw, sh }; 47 | } 48 | if( resizeWidth === 0 || resizeHeight === 0 ) { 49 | const message = common_error_header + error_messages.ALLOCATION_FAILED; 50 | throw new DOMException( message, "InvalidStateError" ); 51 | } 52 | 53 | return { 54 | cropRect, 55 | ...options 56 | }; 57 | 58 | } 59 | function cleanArguments( missing_features, ...args ) { 60 | const number_of_arguments = args.length; 61 | if( 62 | missing_features.has( "_imageBitmapOptions" ) && 63 | [ 2, 6 ].includes( number_of_arguments ) 64 | ) { 65 | return args.slice( 0, number_of_arguments - 1 ); 66 | } 67 | return args; 68 | } 69 | 70 | export { parseOptions, cleanArguments }; 71 | -------------------------------------------------------------------------------- /src/check-usability.js: -------------------------------------------------------------------------------- 1 | import { error_messages } from "./constants.js"; 2 | import { get2dContext, buildSourceIs } from "./misc.js"; 3 | import { getSourceDimensions } from "./image-cropper.js"; 4 | 5 | function invalidStateImageError() { 6 | const message = error_messages.COMMON_HEADER + error_messages.INVALID_STATE_IMAGE; 7 | throw new DOMException( message, "InvalidStateError" ); 8 | } 9 | // 10 | // https://html.spec.whatwg.org/multipage/canvas.html#check-the-usability-of-the-image-argument 11 | // 12 | function checkUsability( source ) { 13 | 14 | const sourceIs = buildSourceIs( source ); 15 | 16 | // HTMLCanvasElement, and OffscreenCanvas are only invalid if their width or height is zero 17 | // ImageBitmap would be when detached, but detaching it will set its width and height to zero 18 | // so we can share their check 19 | if( 20 | sourceIs( "HTMLCanvasElement" ) || 21 | sourceIs( "OffscreenCanvas" ) || 22 | sourceIs( "ImageBitmap" ) 23 | ) { 24 | 25 | if( source.width === 0 || source.height === 0 ) { 26 | invalidStateImageError(); 27 | } 28 | 29 | } 30 | // there is no synchronous way to tell if an HTMLImageElement 31 | // is in a broken state, such an element could have a width and height 32 | // at the time we check it 33 | // HTMLImageElement::decode() could tell this asynchronously 34 | // but 1. we need a synchrounous method, 2. it's not available everywhere 35 | // and 3. it only concerns HTMLImageElement 36 | // SVGImageElements don't have any API letting us know (not even a width nor height) 37 | // HTMLVideoElements only have unreliable videoWidth and videoHeight 38 | // however, CanvasRenderingContext2d.createPattern is supposed to call this algorithm too 39 | // so we can (ab)use it, even though it works only in Firefox... 40 | // see https://wpt.fyi/results/html/canvas/element/fill-and-stroke-styles/2d.pattern.image.broken.html 41 | else if( 42 | sourceIs( "HTMLImageElement" ) || 43 | sourceIs( "HTMLVideoElement" ) || 44 | sourceIs( "SVGImageElement" ) 45 | ) { 46 | 47 | // we can still start by checking the size, if 0 no need to go farther 48 | const { width, height } = getSourceDimensions( source ); 49 | // look for strict equality, SVGImageElement returns NaN 50 | if( width === 0 || height === 0 ) { 51 | invalidStateImageError(); 52 | } 53 | 54 | let failed = false; 55 | try { 56 | const pat = get2dContext( 0, 0 ).createPattern( source, "no-repeat" ); 57 | failed = !pat; 58 | } 59 | catch( err ) { 60 | // Safari doesn't support drawing SVGImageElement on a canvas, 61 | // nor do they support creating a CanvasPattern from it 62 | // we thus return a "maybe" string to let the other side handle it as they wish 63 | return "maybe"; 64 | } 65 | if( failed ) { 66 | invalidStateImageError(); 67 | } 68 | 69 | } 70 | 71 | return true; 72 | 73 | } 74 | 75 | export { checkUsability, invalidStateImageError }; 76 | -------------------------------------------------------------------------------- /src/misc.js: -------------------------------------------------------------------------------- 1 | import { error_messages } from "./constants.js"; 2 | 3 | // we could be lazy an just check for length 0, 4 | // which would probably be enough in 99% of the case 5 | // but we won't call it many times anyway, 6 | // so better be precise 7 | function isDetachedBuffer( buffer ) { 8 | try { 9 | buffer.slice( 0,0 ); 10 | return false; 11 | } 12 | catch( err ) { return true; } 13 | } 14 | 15 | // https://heycam.github.io/webidl/#idl-long 16 | const long_tester = new Int32Array( 1 ); 17 | function convertToLong( value ) { 18 | if( !Number.isFinite( value ) ) { 19 | return NaN; 20 | } 21 | long_tester[ 0 ] = value; 22 | return long_tester[ 0 ]; 23 | } 24 | // https://heycam.github.io/webidl/#EnforceRange 25 | function enforceRangeLong( value ) { 26 | const as_long = convertToLong( value ); 27 | if( isNaN( as_long ) ) { 28 | throw new TypeError( "Invalid long value" ); 29 | } 30 | return as_long; 31 | } 32 | // we can be quite strict here 33 | // because to load it in a , 34 | // the type must be set to this value 35 | function blobCanBeSVG( blob ) { 36 | return blob && blob.type === "image/svg+xml"; 37 | } 38 | 39 | function get2dContext( width, height ) { 40 | const canvas = Object.assign( document.createElement( "canvas" ), { width, height } ); 41 | const ctx = canvas.getContext( "2d" ); 42 | // not yet in the specs 43 | // https://github.com/whatwg/html/issues/3323 44 | if( 45 | width && height && // 0 width and height should not throw here 46 | !contextIsCorrectlyAllocated( ctx ) 47 | ) { 48 | throw new DOMException( error_messages.ALLOCATION, "InvalidStateError" ); 49 | } 50 | return ctx; 51 | } 52 | 53 | function contextIsCorrectlyAllocated( ctx ) { 54 | // https://github.com/fserb/canvas2D/blob/master/spec/context-loss.md 55 | if( ctx.isContextLost ) { 56 | // weirdness in the API, 57 | // we need to interact with the context for the status to change 58 | ctx.translate( 0, 0 ); 59 | return !ctx.isContextLost(); 60 | } 61 | // Chrome has a isContextLost method, but it seems to only work async... 62 | let allocated = false; 63 | // Firefox does throw if the context is lost when we call getImageData 64 | try { 65 | ctx.fillRect( 0, 0, 1, 1 ); 66 | allocated = ctx.getImageData( 0, 0, 1, 1 ).data[ 3 ] !== 0; 67 | } 68 | finally { 69 | ctx.clearRect( 0, 0, 1, 1 ); 70 | return allocated; 71 | } 72 | } 73 | 74 | // ImageData constructor may not be available 75 | // fortunately we can fall back to ctx2D.createImageData 76 | function createImageData( width, height ) { 77 | try { 78 | // low finger-print if available 79 | return new ImageData( width, height ); 80 | } 81 | catch( err ) { 82 | const context = create2dContext( 0, 0 ); 83 | return context.createImageData( width, height ); 84 | } 85 | } 86 | 87 | // a helper to build a 'sourceIs' function 88 | function buildSourceIs( source ) { 89 | return ( type ) => { 90 | const constructor = globalThis[ type ]; 91 | return constructor && (source instanceof constructor); 92 | }; 93 | } 94 | 95 | export { 96 | isDetachedBuffer, 97 | convertToLong, 98 | enforceRangeLong, 99 | blobCanBeSVG, 100 | get2dContext, 101 | createImageData, 102 | buildSourceIs 103 | }; 104 | -------------------------------------------------------------------------------- /src/monkeypatch.js: -------------------------------------------------------------------------------- 1 | import { 2 | requiresPolyfill, 3 | requiresAsyncTests, 4 | tests_results, 5 | async_tests 6 | } from "./test-support.js"; 7 | import { checkUsability, invalidStateImageError } from "./check-usability.js"; 8 | import { parseOptions, cleanArguments } from "./options-parser.js"; 9 | import { nativeCreateImageBitmap } from "./native-func.js"; 10 | import { canvas_sources, error_messages } from "./constants.js"; 11 | import { loadImage, loadSVGImage } from "./image-loaders.js"; 12 | import { cropImage, cropBlobImage, cropImageData } from "./image-cropper.js"; 13 | import { convertToImageBitmap, ImageBitmapPolyfill } from "./ImageBitmap-wrapper.js"; 14 | import { isDetachedBuffer, buildSourceIs } from "./misc.js"; 15 | 16 | if( !nativeCreateImageBitmap ) { 17 | globalThis.ImageBitmap = ImageBitmapPolyfill; 18 | } 19 | 20 | // https://html.spec.whatwg.org/multipage/imagebitmap-and-animations.html#dom-createimagebitmap 21 | globalThis.createImageBitmap = async function( source, ...args ) { // length 1 22 | 23 | const options = parseOptions( ...arguments ); 24 | 25 | // with some sources (canvas & video) we need to stay synchronous until 26 | // the source's buffer gets painted 27 | // but some tests are async (e.g SVGImageElement) 28 | // so we only await for these tests when we are sure they are required 29 | const missing_features = requiresAsyncTests( source, options ) ? 30 | await async_tests : tests_results; 31 | 32 | // full compat 33 | if( missing_features.size === 0 ) { 34 | return nativeCreateImageBitmap( ...arguments ); 35 | } 36 | 37 | // polyfill is not required, but some features may not be supported 38 | if( 39 | nativeCreateImageBitmap && 40 | !requiresPolyfill( missing_features, options, source ) 41 | ) { 42 | return nativeCreateImageBitmap( ...cleanArguments( missing_features, ...arguments ) ); 43 | } 44 | 45 | const sourceIs = buildSourceIs( source ); 46 | 47 | let cropped_image; 48 | if( canvas_sources.some( sourceIs ) ) { 49 | 50 | const is_usable = checkUsability( source ); 51 | 52 | // checkUsability failed to determine if source is usable, 53 | // as of this writing, this only concerns SVGImageElement 54 | // and since we do convert SVGImageElements anyway right after 55 | // we don't have much to do now 56 | if( is_usable === "maybe" ) { 57 | // might want to do something someday. 58 | } 59 | 60 | // SVGImageElement have a very poor API to work with 61 | // (e.g no intrinsic dimension is exposed) 62 | // so we convert these to HTMLImageElement 63 | // must be called after 'checkUsability' for we correctly throw with unusable media 64 | // (that is, for UAs that do support SVGImageElement as CanvasSource) 65 | if( sourceIs( "SVGImageElement" ) ) { 66 | source = await loadSVGImage( source ); 67 | } 68 | 69 | if( sourceIs( "HTMLImageElement" ) ) { 70 | // Since we can't detect true "image has no dimensions" (see test-support.js) 71 | // we unfortunately can't fall here... 72 | // so while per specs this should work, it's just dead code... 73 | if( 74 | (!source.naturalWidth && !source.naturalHeight) && 75 | (!options.resizeWidth || !options.resizeHeight) 76 | ) { 77 | invalidStateImageError() 78 | } 79 | 80 | } 81 | cropped_image = cropImage( source, options ); 82 | 83 | } 84 | else if( sourceIs( "Blob" ) ) { 85 | cropped_image = await cropBlobImage( source, options ); 86 | } 87 | else if( sourceIs( "ImageData" ) ) { 88 | if( isDetachedBuffer( source.data.buffer ) ) { 89 | invalidStateImageError(); 90 | } 91 | cropped_image = await cropImageData( source, options ); 92 | } 93 | if( !cropped_image ) { 94 | throw new TypeError( error_messages.INVALID_SOURCE ); 95 | } 96 | 97 | return await convertToImageBitmap( cropped_image ); 98 | 99 | }; 100 | 101 | globalThis.createImageBitmap._original = nativeCreateImageBitmap; -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # createImageBitmap MonkeyPatch 2 | 3 | ### A monkey-patch to the HTML's [`createImageBitmap`][1] method. 4 | 5 | Only Chromium based browsers currently support the full API, Firefox supporting it only partially and Safari not at all. 6 | This means that for cross-browser projects, we can't use it, despite its great usefulness and ease of use. 7 | 8 | This monkey-patch aims at filling as much gaps as possible between all these browsers. 9 | 10 | However, truth be told, it can not polyfill everything in this API, here is a list of what it supports in every browsers and what it doesn't: 11 | 12 | Feature | ![Chromium based][2] | ![Firefox][3] | ![Safari][4] 13 | ----------------------------------------- | -------------------- | --------------------- | ---------------- 14 | createImageBitmap + cropping | native | native | ✔ (1) 15 | HTMLImageElement as source | native | native | ✔ 16 | HTMLCanvasElement as source | native | native | ✔ 17 | OffscreenCanvas as source | native | :x: | :x: 18 | SVGImageElement as source | native | native | ✔ 19 | ImageBitmap as source | native | native | ✔ 20 | ImageData as source | native | native | ✔ 21 | Raster image in Blob as source | native | native | ✔ 22 | SVG image in Blob as source (2)| ✔ | ✔ | ✔ 23 | ImageBitmapOptions | native | ✔ | ✔ 24 | resizeWidth & resizeHeight | native | ✔ | ✔ 25 | resizeQuality: pixelated | native | ✔ | ✔ 26 | resizeQuality: low | native | native | ✔ 27 | resizeQuality: medium | native | :x: | :x: 28 | resizeQuality: high | native | :x: | :x: 29 | imageOrientation | native | ✔ | ✔ 30 | colorSpaceConversion | native | :x: | :x: 31 | premultiplyAlpha | native | :x: | :x: 32 | Available in Workers | native | native (3) | :x: 33 | Transferrable | native | native | :x: 34 | 35 | (1) Returns a modified HTMLCanvasElement instance in place of an ImageBitmap object. 36 | (2) Only for SVG images with an intrinsic width and height, as per the specs recommendation. 37 | (3) Only the native implementation is available in Workers 38 | 39 | ### How to use 40 | 41 | You can include the bundled version available in `/src/` in a <script> tag before the first use of `createImageBitmap`, 42 | 43 | ```html 44 | 45 | ``` 46 | 47 | or you can use the ESM modules available in `/src/`. 48 | 49 | ```html 50 | 51 | ``` 52 | 53 | ### Why isn't the monkey-patch available in web Workers? 54 | 55 | The monkey-patch makes most of its work on an HTMLCanvasElement 2d context, and relies in various places in the ability to have DOM Elements (for instance HTMLImageElements are used to load images from Blobs). 56 | Since Web-Workers don't have access to the DOM nor any Elements, the scope of what this monkey-patch could do currently seems too limited to be worth the pain. 57 | 58 | ### Why isn't the monkey-patch available in node-js? 59 | 60 | People at [node-canvas][2] probably are the best to implement such a thing. 61 | It should take only a few tweaks from here to get a node-canvas + jsdom version working though, so feel free to fork this repo and propose them a PR if you wish. 62 | 63 | ### Why doesn't this monkey-patch add support for options like all resize qualities, *colorSpaceConversion* or *premultiplyAlpha**? 64 | 65 | Mainly lack of time. We accept PR though ¯\_(ツ)_/¯ 66 | 67 | ### Why no ImageBitmapRenderer at least? 68 | 69 | Still lack of time, and a Safari weirdness where they do expose this interface, even though they don't have the `ImageBitmap` that goes with it, and which won't accept our "fake" version. 70 | But hopefully, this will be one of the first next additions. 71 | 72 | ### What about my old Internet Explorer? Can I use it there? 73 | 74 | Not as-is. The scripts are written using ES2021 syntax and are targeting recent browsers. 75 | You should be able to transpile it to ES5 though, and it should work even in IE9, **but it hasn't been tested there**. 76 | 77 | ### Does this mean even my shiny Chromium browser will use the less than optimal 2D context instead of a true ImageBitmap? 78 | 79 | No, the monkey-patch only patches what it detects is not supported by the current browser. It leaves the native implementation do its job when it can do it. 80 | 81 | [1]: https://developer.mozilla.org/en-US/docs/Web/API/WindowOrWorkerGlobalScope/createImageBitmap 82 | [2]: https://raw.githubusercontent.com/alrra/browser-logos/main/src/chromium/chromium_48x48.png 83 | [3]: https://raw.githubusercontent.com/alrra/browser-logos/master/src/firefox/firefox_48x48.png 84 | [4]: https://raw.githubusercontent.com/alrra/browser-logos/master/src/safari/safari_48x48.png 85 | [5]: https://github.com/Automattic/node-canvas 86 | -------------------------------------------------------------------------------- /src/image-cropper.js: -------------------------------------------------------------------------------- 1 | import { nativeCreateImageBitmap } from "./native-func.js"; 2 | import { blobCanBeSVG, get2dContext, buildSourceIs } from "./misc.js"; 3 | import { loadImage } from "./image-loaders.js"; 4 | import { cleanArguments } from "./options-parser.js"; 5 | 6 | function getSourceDimensions( source ) { 7 | const sourceIs = buildSourceIs( source ); 8 | if( sourceIs( "HTMLImageElement" ) ) { 9 | return { width: source.naturalWidth, height: source.naturalHeight }; 10 | } 11 | else if( sourceIs( "HTMLVideoElement" ) ) { 12 | return { width: source.videoWidth, height: source.videoHeight }; 13 | } 14 | else if( sourceIs( "SVGImageElement" ) ) { 15 | return { width: NaN, height: NaN }; 16 | } 17 | else if( 18 | sourceIs( "HTMLCanvasElement" ) || 19 | sourceIs( "OffscreenCanvas" ) || 20 | sourceIs( "ImageBitmap" ) || 21 | sourceIs( "ImageData" ) 22 | ) { 23 | return source; 24 | } 25 | return { width: NaN, height: NaN }; 26 | } 27 | 28 | function cropImage( source, options ) { 29 | 30 | // merge passed cropRect with resize and orientation options 31 | const safe_rect = getFinalCropRect( source, options ); 32 | 33 | const { height } = getSourceDimensions( source ); 34 | const { sx, sy, sw, sh, dx, dy, dw, dh, flipY, resizeQuality } = safe_rect; 35 | const out_ctx = get2dContext( safe_rect.width, safe_rect.height ); 36 | const out_canvas = out_ctx.canvas; 37 | 38 | if( resizeQuality === "pixelated" ) { 39 | out_ctx.imageSmoothingEnabled = false; 40 | } 41 | else if( resizeQuality ) { 42 | // to do: better image quality resizers 43 | out_ctx.imageSmoothingQuality = resizeQuality; 44 | } 45 | out_ctx.drawImage( source, sx, sy, sw, sh, dx, dy, dw, dh ); 46 | 47 | if( flipY ) { 48 | out_ctx.globalCompositeOperation = "copy"; 49 | out_ctx.setTransform( 1, 0, 0, -1, 0, out_canvas.height ); 50 | out_ctx.drawImage( out_canvas, 0, 0 ); 51 | } 52 | 53 | return out_canvas; 54 | 55 | } 56 | 57 | 58 | async function cropBlobImage( blob, options ) { 59 | 60 | const url = globalThis.URL.createObjectURL( blob ); 61 | const img = await loadImage( url ); 62 | globalThis.URL.revokeObjectURL( blob ); 63 | 64 | return cropImage( img, options ); 65 | 66 | } 67 | async function cropImageData( img, options ) { 68 | 69 | const { resizeWidth, resizeHeight } = parseResizeOptions( img, options ); 70 | const { resizeQuality, imageOrientation } = options; 71 | const cropRect = options.cropRect || {}; 72 | // beware, sw and sh can be negative 73 | const sx = cropRect.sx || 0; 74 | const sy = cropRect.sy || 0; 75 | const sw = cropRect.sw || img.width; 76 | const sh = cropRect.sh || img.height; 77 | const dx = sw < 0 ? (sw * -1) - sx : -sx; 78 | const dy = sh < 0 ? (sh * -1) - sy : -sy; 79 | const dw = Math.abs( sw ); 80 | const dh = Math.abs( sh ); 81 | 82 | const a_ctx = get2dContext( dw, dh ) 83 | a_ctx.putImageData( img, dx, dy, sx, sy, sw, sh ); 84 | 85 | const flipY = imageOrientation === "flipY"; 86 | const should_redraw = resizeWidth || flipY; 87 | let out_ctx; 88 | 89 | if( resizeWidth ) { 90 | out_ctx = get2dContext( resizeWidth, resizeHeight ); 91 | if( resizeQuality === "pixelated" ) { 92 | out_ctx.imageSmoothingEnabled = false; 93 | } 94 | else { 95 | out_ctx.imageSmoothingQuality = resizeQuality; 96 | } 97 | } 98 | else { 99 | out_ctx = a_ctx; 100 | a_ctx.globalCompositeOperation = "copy"; 101 | } 102 | 103 | const a_canvas = a_ctx.canvas; 104 | const out_canvas = out_ctx.canvas; 105 | 106 | if( flipY ){ 107 | out_ctx.setTransform( 1, 0, 0, -1, 0, out_canvas.height ); 108 | } 109 | 110 | if( should_redraw ) { 111 | out_ctx.drawImage( a_canvas, 112 | 0, 0, a_canvas.width, a_canvas.height, 113 | 0, 0, out_canvas.width, out_canvas.height 114 | ); 115 | } 116 | 117 | return out_canvas; 118 | 119 | } 120 | 121 | function getFinalCropRect( source, options ) { 122 | 123 | const { width, height } = getSourceDimensions( source ); 124 | const { resizeWidth, resizeHeight } = parseResizeOptions( source, options ); 125 | 126 | const crop_rect = options.cropRect || 127 | { sx: 0, sy: 0, sw: width, sh: height }; 128 | const dest_rect = { 129 | dx: 0, 130 | dy: 0, 131 | dw: resizeWidth || Math.abs( crop_rect.sw ), // resizeXXX must be non-zero 132 | dh: resizeHeight || Math.abs( crop_rect.sh ) 133 | }; 134 | 135 | const safe_rect = getSafeRect( width, height, crop_rect, dest_rect ); 136 | safe_rect.resizeQuality = options.resizeQuality; 137 | safe_rect.flipY = options.imageOrientation === "flipY"; 138 | safe_rect.width = dest_rect.dw; 139 | safe_rect.height = dest_rect.dh; 140 | 141 | return safe_rect; 142 | 143 | } 144 | 145 | function getSafeRect( width, height, { sx, sy, sw, sh }, { dx, dy, dw, dh } ) { 146 | // Safari doesn't support cropping outside of source's boundaries through drawImage 147 | // so we need to provide a "safe" rect 148 | 149 | if( sw < 0 ) { 150 | sx += sw; 151 | sw = Math.abs( sw ); 152 | } 153 | if( sh < 0 ) { 154 | sy += sh; 155 | sh = Math.abs( sh ); 156 | } 157 | if( dw < 0 ) { 158 | dx += dw; 159 | dw = Math.abs( dw ); 160 | } 161 | if( dh < 0 ) { 162 | dy += dh; 163 | dh = Math.abs( dh ); 164 | } 165 | const x1 = Math.max( sx, 0 ); 166 | const x2 = Math.min( sx + sw, width ); 167 | const y1 = Math.max( sy, 0 ); 168 | const y2 = Math.min( sy + sh, height ); 169 | const w_ratio = dw / sw; 170 | const h_ratio = dh / sh; 171 | 172 | return { 173 | sx: x1, 174 | sy: y1, 175 | sw: x2 - x1, 176 | sh: y2 - y1, 177 | dx: sx < 0 ? dx - (sx * w_ratio) : dx, 178 | dy: sy < 0 ? dy - (sy * h_ratio) : dy, 179 | dw: (x2 - x1) * w_ratio, 180 | dh: (y2 - y1) * h_ratio 181 | }; 182 | 183 | } 184 | 185 | function parseResizeOptions( source, options ) { 186 | 187 | const { width, height } = getSourceDimensions( source ); 188 | let { resizeWidth, resizeHeight } = options; 189 | if( resizeWidth === undefined && resizeHeight !== undefined ) { 190 | const { width, height } = getSourceDimensions( source ); 191 | resizeWidth = Math.ceil( width * (resizeHeight / height) ); 192 | } 193 | if( resizeHeight === undefined && resizeWidth !== undefined ) { 194 | const { width, height } = getSourceDimensions( source ); 195 | resizeHeight = Math.ceil( height * (resizeWidth / width) ); 196 | } 197 | return { resizeWidth, resizeHeight }; 198 | 199 | } 200 | 201 | export { 202 | getSourceDimensions, 203 | cropImage, 204 | cropBlobImage, 205 | cropImageData 206 | }; 207 | -------------------------------------------------------------------------------- /src/test-support.js: -------------------------------------------------------------------------------- 1 | import { 2 | known_properties, 3 | synchronous_polyfills, 4 | asynchronous_polyfills, 5 | SVG_MIME, 6 | SVG_NS, 7 | GIF_URL 8 | } from "./constants.js"; 9 | import { checkUsability } from "./check-usability.js"; 10 | import { nativeCreateImageBitmap } from "./native-func.js"; 11 | import { createImageData, blobCanBeSVG, buildSourceIs } from "./misc.js"; 12 | 13 | // to avoid having the grabbing of the source's buffer delayed 14 | // we try to return the results of the tests synchronously when possible. 15 | // for this matter, we update a single list of results. 16 | let async_tests_are_done = false; 17 | 18 | // some tests can be performed synchronously 19 | // we initiate our Set with these directly 20 | const tests_results = performSynchronousTests(); 21 | const async_tests = performAsynchronousTests() 22 | 23 | function performSynchronousTests() { 24 | // nothing to test... 25 | if( !nativeCreateImageBitmap ) { 26 | const all_polyfills = synchronous_polyfills.concat( asynchronous_polyfills ); 27 | return new Set( all_polyfills ); 28 | } 29 | const synchronous_tests = [ testOptionsBag() ]; 30 | const supported_features = new Set( synchronous_tests.flat() ); 31 | const required_polyfills = synchronous_polyfills 32 | .filter( (type) => !supported_features.has( type ) ); 33 | 34 | return new Set( required_polyfills ); 35 | } 36 | async function performAsynchronousTests() { 37 | 38 | // nothing to test... 39 | if( !nativeCreateImageBitmap ) { 40 | async_tests_are_done = true; 41 | return tests_results; 42 | } 43 | 44 | const asynchronous_tests = [ 45 | testBitmapBlobSource(), 46 | testImageDataSource(), 47 | testSVGBlobSource(), 48 | testSVGImageElementSource() 49 | ]; 50 | 51 | const results = await Promise.all( asynchronous_tests ); 52 | const supported_features = new Set( results.flat() ); 53 | 54 | const required_polyfills = asynchronous_polyfills 55 | .filter( (type) => !supported_features.has( type ) ); 56 | 57 | required_polyfills.forEach( (prop) => tests_results.add( prop ) ); 58 | 59 | async_tests_are_done = true; 60 | 61 | return tests_results; 62 | 63 | } 64 | 65 | function testOptionsBag() { 66 | 67 | const out = []; 68 | const tester = {}; 69 | const addTestProperty = ( prop ) => { 70 | Object.defineProperty( tester, prop , { get(){ out.push( prop ) } } ); 71 | }; 72 | known_properties.forEach( addTestProperty ); 73 | 74 | nativeCreateImageBitmap( new ImageData(1,1), tester ) 75 | .then( (img) => img.close() ) 76 | .catch( () => {} ); 77 | 78 | // some implementations will throw in the Promise 79 | // if the option bag is passed 80 | // to avoid doing it ourselves when calling the native method 81 | // we need to mark its support here 82 | // however since we need this method to be synchronous 83 | // we can't wait for the Promise to reject 84 | // so we assume any implementation supporting the options bag 85 | // will support at least one of the currently known options 86 | if( out.length ) { 87 | out.push( "_imageBitmapOptions" ); 88 | } 89 | 90 | return out; 91 | 92 | } 93 | async function testBitmapBlobSource() { 94 | 95 | const out = []; 96 | const blob = await fetch( GIF_URL ) 97 | .then( resp => resp.ok && resp.blob() ); 98 | 99 | try { 100 | const img = await nativeCreateImageBitmap( blob ); 101 | if( img.width === 1 ) { 102 | out.push( "_Blob" ); 103 | } 104 | } 105 | catch( err ) { 106 | 107 | } 108 | 109 | return out; 110 | 111 | } 112 | async function testImageDataSource() { 113 | 114 | const out = []; 115 | const image = createImageData( 1, 1 ); 116 | 117 | try { 118 | const img = await nativeCreateImageBitmap( image ); 119 | if( img.width === 1 ) { 120 | out.push( "_ImageData" ); 121 | } 122 | } 123 | catch( err ) { 124 | 125 | } 126 | 127 | return out; 128 | 129 | } 130 | async function testSVGBlobSource() { 131 | 132 | const out = []; 133 | const svg_markup = ` 134 | 135 | `; 136 | const svg_blob = new Blob( [ svg_markup ], { type: SVG_MIME } ); 137 | 138 | try { 139 | const img = await nativeCreateImageBitmap( svg_blob ); 140 | if( img.width === 1 ) { 141 | out.push( '_SVGBlob' ); 142 | } 143 | } 144 | catch( err ) { 145 | 146 | } 147 | 148 | return out; 149 | 150 | } 151 | async function testSVGImageElementSource() { 152 | 153 | const out = []; 154 | if( !("SVGImageElement" in globalThis) ) { // Worker 155 | return out; 156 | } 157 | 158 | const wait = (time) => new Promise( res => setTimeout( res, time ) ); 159 | 160 | const img = document.createElementNS( SVG_NS, 'image' ); 161 | img.setAttributeNS( 'http://www.w3.org/1999/xlink', 'xlink:href', GIF_URL ); 162 | 163 | const imgIsNotAvailable = () => { 164 | try { 165 | checkUsability( img ); 166 | return false; 167 | } 168 | catch( err ) { 169 | return true; 170 | } 171 | }; 172 | 173 | let retries = 0; 174 | do { 175 | await wait( 10 ); 176 | } 177 | while( imgIsNotAvailable() && (++retries < 300) ); 178 | 179 | try { 180 | const bmp = await nativeCreateImageBitmap( img ); 181 | out.push( "_SVGImageElement" ); 182 | } 183 | catch( err ) { 184 | 185 | } 186 | 187 | return out; 188 | 189 | } 190 | 191 | function requiresPolyfill( missing_features, options, source ) { 192 | 193 | const sourceIs = buildSourceIs( source ); 194 | 195 | if( sourceIs( "Blob" ) ) { 196 | if( missing_features.has( "_Blob" ) ) { 197 | return true; 198 | } 199 | const is_svg_blob = blobCanBeSVG( source ); 200 | if( is_svg_blob && missing_features.has( "_SVGBlob" ) ) { 201 | return true; 202 | } 203 | } 204 | if( 205 | sourceIs( "ImageData" ) && 206 | missing_features.has( "_ImageData" ) 207 | ) { 208 | return true; 209 | } 210 | if( 211 | sourceIs( "SVGImageElement" ) && 212 | missing_features.has( "_SVGImageElement" ) 213 | ) { 214 | return true; 215 | } 216 | if( 217 | Object.keys( options ) 218 | .some( (key) => missing_features.has( key ) ) 219 | ) { 220 | return true; 221 | } 222 | 223 | return false; 224 | 225 | } 226 | 227 | function requiresAsyncTests( source, options ) { 228 | if( async_tests_are_done ) { 229 | return false; 230 | } 231 | // async tests are currently all for different types of source 232 | // so we can currently only check for these types 233 | const sourceIs = buildSourceIs( source ); 234 | const types_requiring_async_tests = [ 235 | "Blob", 236 | "ImageData", 237 | "SVGImageElement" 238 | ]; 239 | return types_requiring_async_tests.some( sourceIs ); 240 | } 241 | 242 | export { 243 | requiresPolyfill, 244 | requiresAsyncTests, 245 | tests_results, 246 | async_tests 247 | }; 248 | -------------------------------------------------------------------------------- /dist/createImageBitmap.js: -------------------------------------------------------------------------------- 1 | (()=>{"use strict";const t=["_imageBitmapOptions","resizeWidth","resizeHeight","resizeQuality","imageOrientation"],e=["_Blob","_ImageData","_SVGBlob","_SVGImageElement"],a=["resizeWidth","resizeHeight","resizeQuality","imageOrientation","premultiplyAlpha","colorSpaceConversion"],n={ResizeQuality:["pixelated","low","medium","high"],ImageOrientation:["none","flipY"]},i={COMMON_HEADER:"Failed to execute 'createImageBitmap': ",INVALID_STATE_IMAGE:"Provided image was in an invalid state.",ARGUMENT_COUNT_1:"At least one argument is required.",ARGUMENT_COUNT_N:"%s is not a valid argument count for any overload",CROP_RECT_ZERO:"The crop rect width passed to createImageBitmap must be nonzero",ALLOCATION_FAILED:"The ImageBitmap could not be allocated.",INVALID_SOURCE:"Argument 1 could not be converted to any of: HTMLImageElement, SVGImageElement, HTMLCanvasElement, HTMLVideoElement, ImageBitmap, Blob, CanvasRenderingContext2D, ImageData.",ENUM:"'%v' is not a valid value for enumeration %e",ALLOCATION:"The ImageBitmap couldn't be allocated."},r=["HTMLImageElement","SVGImageElement","HTMLVideoElement","HTMLCanvasElement","OffscreenCanvas","ImageBitmap"],o="http://www.w3.org/2000/svg",s="";function h(t){try{return t.slice(0,0),!1}catch(t){return!0}}const c=new Int32Array(1);function l(t){return Number.isFinite(t)?(c[0]=t,c[0]):NaN}function g(t){const e=l(t);if(isNaN(e))throw new TypeError("Invalid long value");return e}function m(t,e){const a=Object.assign(document.createElement("canvas"),{width:t,height:e}).getContext("2d");if(t&&e&&!function(t){if(t.isContextLost)return t.translate(0,0),!t.isContextLost();let e=!1;try{t.fillRect(0,0,1,1),e=0!==t.getImageData(0,0,1,1).data[3]}finally{return t.clearRect(0,0,1,1),e}}(a))throw new DOMException(i.ALLOCATION,"InvalidStateError");return a}function u(t){return e=>{const a=globalThis[e];return a&&t instanceof a}}const d=globalThis.createImageBitmap,w=!!d&&((...t)=>d.call(globalThis,...t));function f(t){const e=new Image;return e.src=t,new Promise(((t,a)=>{e.onload=a=>{N(e),t(e)},e.onerror=t=>{const e=new DOMException(i.INVALID_STATE_IMAGE,"InvalidStateError");a(e)}}))}function p(t){return f(t.getAttribute("href")||t.getAttributeNS("http://www.w3.org/1999/xlink","href"))}const I=i.COMMON_HEADER;function E(t,e,a){const r=n[a],o=t[e];if(void 0!==o&&!r.includes(o)){const t=i.ENUM.replace("%v",o).replace("%e",a);throw new TypeError(t)}}function A(t,...e){let a,[n,r,o,s]=e.map(l);const h=(1===e.length?e[0]:e[4])||{},c="resizeWidth"in h&&g(h.resizeWidth),m="resizeHeight"in h&&g(h.resizeHeight);E(h,"resizeQuality","ResizeQuality"),E(h,"imageOrientation","ImageOrientation");const u=arguments.length;if(!u)throw new TypeError(I+i.ARGUMENT_COUNT_1);if(u>2&&u<5){const t=i.ARGUMENT_COUNT_N.replace("%s",u);throw new TypeError(I+t)}if(u>=5){if(!o||!s)throw new RangeError(I+i.CROP_RECT_ZERO);n=n||0,r=r||0,a={sx:n,sy:r,sw:o,sh:s}}if(0===c||0===m){const t=I+i.ALLOCATION_FAILED;throw new DOMException(t,"InvalidStateError")}return{cropRect:a,...h}}function y(t,...e){const a=e.length;return t.has("_imageBitmapOptions")&&[2,6].includes(a)?e.slice(0,a-1):e}function T(t){const e=u(t);return e("HTMLImageElement")?{width:t.naturalWidth,height:t.naturalHeight}:e("HTMLVideoElement")?{width:t.videoWidth,height:t.videoHeight}:e("SVGImageElement")?{width:NaN,height:NaN}:e("HTMLCanvasElement")||e("OffscreenCanvas")||e("ImageBitmap")||e("ImageData")?t:{width:NaN,height:NaN}}function b(t,e){const a=function(t,e){const{width:a,height:n}=T(t),{resizeWidth:i,resizeHeight:r}=v(t,e),o=e.cropRect||{sx:0,sy:0,sw:a,sh:n},s={dx:0,dy:0,dw:i||Math.abs(o.sw),dh:r||Math.abs(o.sh)},h=function(t,e,{sx:a,sy:n,sw:i,sh:r},{dx:o,dy:s,dw:h,dh:c}){i<0&&(a+=i,i=Math.abs(i)),r<0&&(n+=r,r=Math.abs(r)),h<0&&(o+=h,h=Math.abs(h)),c<0&&(s+=c,c=Math.abs(c));const l=Math.max(a,0),g=Math.min(a+i,t),m=Math.max(n,0),u=Math.min(n+r,e),d=h/i,w=c/r;return{sx:l,sy:m,sw:g-l,sh:u-m,dx:a<0?o-a*d:o,dy:n<0?s-n*w:s,dw:(g-l)*d,dh:(u-m)*w}}(a,n,o,s);return h.resizeQuality=e.resizeQuality,h.flipY="flipY"===e.imageOrientation,h.width=s.dw,h.height=s.dh,h}(t,e),{height:n}=T(t),{sx:i,sy:r,sw:o,sh:s,dx:h,dy:c,dw:l,dh:g,flipY:u,resizeQuality:d}=a,w=m(a.width,a.height),f=w.canvas;return"pixelated"===d?w.imageSmoothingEnabled=!1:d&&(w.imageSmoothingQuality=d),w.drawImage(t,i,r,o,s,h,c,l,g),u&&(w.globalCompositeOperation="copy",w.setTransform(1,0,0,-1,0,f.height),w.drawImage(f,0,0)),f}async function O(t,e){const a=globalThis.URL.createObjectURL(t),n=await f(a);return globalThis.URL.revokeObjectURL(t),b(n,e)}async function M(t,e){const{resizeWidth:a,resizeHeight:n}=v(t,e),{resizeQuality:i,imageOrientation:r}=e,o=e.cropRect||{},s=o.sx||0,h=o.sy||0,c=o.sw||t.width,l=o.sh||t.height,g=c<0?-1*c-s:-s,u=l<0?-1*l-h:-h,d=m(Math.abs(c),Math.abs(l));d.putImageData(t,g,u,s,h,c,l);const w="flipY"===r,f=a||w;let p;a?(p=m(a,n),"pixelated"===i?p.imageSmoothingEnabled=!1:p.imageSmoothingQuality=i):(p=d,d.globalCompositeOperation="copy");const I=d.canvas,E=p.canvas;return w&&p.setTransform(1,0,0,-1,0,E.height),f&&p.drawImage(I,0,0,I.width,I.height,0,0,E.width,E.height),E}function v(t,e){const{width:a,height:n}=T(t);let{resizeWidth:i,resizeHeight:r}=e;if(void 0===i&&void 0!==r){const{width:e,height:a}=T(t);i=Math.ceil(e*(r/a))}if(void 0===r&&void 0!==i){const{width:e,height:a}=T(t);r=Math.ceil(a*(i/e))}return{resizeWidth:i,resizeHeight:r}}function _(){const t=i.COMMON_HEADER+i.INVALID_STATE_IMAGE;throw new DOMException(t,"InvalidStateError")}function N(t){const e=u(t);if(e("HTMLCanvasElement")||e("OffscreenCanvas")||e("ImageBitmap"))0!==t.width&&0!==t.height||_();else if(e("HTMLImageElement")||e("HTMLVideoElement")||e("SVGImageElement")){const{width:e,height:a}=T(t);0!==e&&0!==a||_();let n=!1;try{n=!m(0,0).createPattern(t,"no-repeat")}catch(t){return"maybe"}n&&_()}return!0}let L=!1;const S=function(){if(!w){const a=t.concat(e);return new Set(a)}const a=[B()],n=new Set(a.flat()),i=t.filter((t=>!n.has(t)));return new Set(i)}(),C=async function(){if(!w)return L=!0,S;const t=[x(),H(),D(),R()],a=await Promise.all(t),n=new Set(a.flat());return e.filter((t=>!n.has(t))).forEach((t=>S.add(t))),L=!0,S}();function B(){const t=[],e={};return a.forEach((a=>{Object.defineProperty(e,a,{get(){t.push(a)}})})),w(new ImageData(1,1),e).then((t=>t.close())).catch((()=>{})),t.length&&t.push("_imageBitmapOptions"),t}async function x(){const t=[],e=await fetch(s).then((t=>t.ok&&t.blob()));try{1===(await w(e)).width&&t.push("_Blob")}catch(t){}return t}async function H(){const t=[],e=function(t,e){try{return new ImageData(1,1)}catch(t){return create2dContext(0,0).createImageData(1,1)}}();try{1===(await w(e)).width&&t.push("_ImageData")}catch(t){}return t}async function D(){const t=[],e=new Blob([`\n \n `],{type:"image/svg+xml"});try{1===(await w(e)).width&&t.push("_SVGBlob")}catch(t){}return t}async function R(){const t=[];if(!("SVGImageElement"in globalThis))return t;const e=t=>new Promise((e=>setTimeout(e,t))),a=document.createElementNS(o,"image");a.setAttributeNS("http://www.w3.org/1999/xlink","xlink:href",s);const n=()=>{try{return N(a),!1}catch(t){return!0}};let i=0;do{await e(10)}while(n()&&++i<300);try{await w(a),t.push("_SVGImageElement")}catch(t){}return t}function z(t,e,a){const n=u(a);if(n("Blob")){if(t.has("_Blob"))return!0;if((i=a)&&"image/svg+xml"===i.type&&t.has("_SVGBlob"))return!0}var i;return!(!n("ImageData")||!t.has("_ImageData"))||!(!n("SVGImageElement")||!t.has("_SVGImageElement"))||!!Object.keys(e).some((e=>t.has(e)))}function V(t,e){if(L)return!1;const a=u(t);return["Blob","ImageData","SVGImageElement"].some(a)}const G=t=>Object.getOwnPropertyDescriptor(HTMLCanvasElement.prototype,t);function U(){throw new TypeError("Illegal Constructor")}function Q(t){return t instanceof globalThis.ImageBitmap?t:w?w(t):function(t){return t instanceof HTMLCanvasElement||(t=b(t)),Object.setPrototypeOf(t,U.prototype),t}(t)}U.prototype=Object.create({close(){G("width").set.call(this,0),G("height").set.call(this,0)},get width(){return G("width").get.call(this)},get height(){return G("height").get.call(this)},get[Symbol.toStringTag](){return"ImageBitmap"}}),w||(globalThis.ImageBitmap=U),globalThis.createImageBitmap=async function(t,...e){const a=A(...arguments),n=V(t)?await C:S;if(0===n.size)return w(...arguments);if(w&&!z(n,a,t))return w(...y(n,...arguments));const o=u(t);let s;if(r.some(o)?(N(t),o("SVGImageElement")&&(t=await p(t)),o("HTMLImageElement")&&(t.naturalWidth||t.naturalHeight||a.resizeWidth&&a.resizeHeight||_()),s=b(t,a)):o("Blob")?s=await O(t,a):o("ImageData")&&(h(t.data.buffer)&&_(),s=await M(t,a)),!s)throw new TypeError(i.INVALID_SOURCE);return await Q(s)},globalThis.createImageBitmap._original=w})(); --------------------------------------------------------------------------------