@workglow/util 0.2.14 → 0.2.16
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/browser.js +28 -18
- package/dist/browser.js.map +6 -5
- package/dist/bun.js +28 -18
- package/dist/bun.js.map +6 -5
- package/dist/logging/LoggerRegistry.d.ts.map +1 -1
- package/dist/media/Image.browser.d.ts +24 -0
- package/dist/media/Image.browser.d.ts.map +1 -0
- package/dist/media/Image.d.ts +100 -0
- package/dist/media/Image.d.ts.map +1 -0
- package/dist/media/MediaRawImage.d.ts +27 -0
- package/dist/media/MediaRawImage.d.ts.map +1 -0
- package/dist/media/color.d.ts +41 -0
- package/dist/media/color.d.ts.map +1 -0
- package/dist/media/imageRasterCodecRegistry.d.ts +13 -0
- package/dist/media/imageRasterCodecRegistry.d.ts.map +1 -0
- package/dist/media/{image.d.ts → imageTypes.d.ts} +5 -1
- package/dist/media/imageTypes.d.ts.map +1 -0
- package/dist/media-browser.d.ts +6 -2
- package/dist/media-browser.d.ts.map +1 -1
- package/dist/media-browser.js +521 -41
- package/dist/media-browser.js.map +9 -5
- package/dist/media-node.d.ts +5 -2
- package/dist/media-node.d.ts.map +1 -1
- package/dist/media-node.js +424 -23
- package/dist/media-node.js.map +8 -5
- package/dist/node.js +28 -18
- package/dist/node.js.map +6 -5
- package/dist/telemetry/TelemetryRegistry.d.ts.map +1 -1
- package/dist/utilities/runtimeEnv.d.ts +18 -0
- package/dist/utilities/runtimeEnv.d.ts.map +1 -0
- package/dist/worker-browser.js +26 -10
- package/dist/worker-browser.js.map +5 -4
- package/dist/worker-bun.js +26 -10
- package/dist/worker-bun.js.map +5 -4
- package/dist/worker-node.js +26 -10
- package/dist/worker-node.js.map +5 -4
- package/package.json +3 -3
- package/dist/media/image.browser.d.ts +0 -11
- package/dist/media/image.browser.d.ts.map +0 -1
- package/dist/media/image.d.ts.map +0 -1
- package/dist/media/image.node.d.ts +0 -11
- package/dist/media/image.node.d.ts.map +0 -1
|
@@ -1,11 +1,15 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
|
-
"sources": ["../src/media/
|
|
3
|
+
"sources": ["../src/media/imageTypes.ts", "../src/media/color.ts", "../src/media/imageRasterCodecRegistry.ts", "../src/media/MediaRawImage.ts", "../src/media/Image.ts", "../src/media/Image.browser.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"/**\n * @license\n * Copyright 2025 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n */\n\nexport type ImageChannels = 1 | 3 | 4; // grayscale, rgb, rgba\n\nexport type ImageDataSupport =\n | \"Blob\"\n | \"ImageBinary\"\n | \"ImageBitmap\"\n | \"OffscreenCanvas\"\n | \"VideoFrame\"\n | \"RawImage\"\n | \"DataUri\"\n | \"Sharp\";\n\nexport interface ImageBinary {\n data: Uint8ClampedArray;\n width: number;\n height: number;\n channels: ImageChannels;\n rawChannels?: number | undefined;\n}\n\nexport function parseDataUri(dataUri: string): {\n mimeType: string;\n base64: string;\n} {\n const match = dataUri.match(/^data:([^;]+);base64,(.+)$/);\n if (!match) {\n throw new Error(\"Invalid base64 data URI\");\n }\n\n return {\n mimeType: match[1], // e.g. \"image/png\"\n base64: match[2],\n };\n}\n",
|
|
6
|
-
"/**\n * @license\n * Copyright 2025 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n */\n\
|
|
5
|
+
"/**\n * @license\n * Copyright 2025 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n */\n\nexport type ImageChannels = 1 | 3 | 4; // grayscale, rgb, rgba\n\nexport type ImageDataSupport =\n | \"Blob\"\n | \"ImageBinary\"\n | \"ImageBitmap\"\n | \"OffscreenCanvas\"\n | \"VideoFrame\"\n | \"RawImage\"\n | \"DataUri\"\n | \"Sharp\";\n\nexport interface ImageBinary {\n data: Uint8ClampedArray;\n width: number;\n height: number;\n channels: ImageChannels;\n rawChannels?: number | undefined;\n}\n\n/** RGBA pixel buffer (`channels` is always 4). */\nexport type RgbaImageBinary = Readonly<\n Omit<ImageBinary, \"channels\" | \"rawChannels\"> & { readonly channels: 4 }\n>;\n\nexport function parseDataUri(dataUri: string): {\n mimeType: string;\n base64: string;\n} {\n const match = dataUri.match(/^data:([^;]+);base64,(.+)$/);\n if (!match) {\n throw new Error(\"Invalid base64 data URI\");\n }\n\n return {\n mimeType: match[1], // e.g. \"image/png\"\n base64: match[2],\n };\n}\n",
|
|
6
|
+
"/**\n * @license\n * Copyright 2025 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n */\n\nexport interface ColorObject {\n readonly r: number;\n readonly g: number;\n readonly b: number;\n readonly a: number;\n}\n\nconst HEX_PATTERN = /^#([0-9a-fA-F]{3,4}|[0-9a-fA-F]{6}|[0-9a-fA-F]{8})$/;\n\n/**\n * Parse a `#RGB` / `#RGBA` / `#RRGGBB` / `#RRGGBBAA` hex color into a {@link ColorObject}.\n * Case-insensitive on input. No whitespace tolerance. Shorthand nibbles are doubled.\n * Throws on any malformed input.\n */\nexport function parseHexColor(hex: string): ColorObject {\n if (typeof hex !== \"string\" || !HEX_PATTERN.test(hex)) {\n throw new Error(`Invalid hex color: ${String(hex)}`);\n }\n const body = hex.slice(1);\n const double = (nibble: string): number => parseInt(nibble + nibble, 16);\n if (body.length === 3) {\n return { r: double(body[0]!), g: double(body[1]!), b: double(body[2]!), a: 255 };\n }\n if (body.length === 4) {\n return {\n r: double(body[0]!),\n g: double(body[1]!),\n b: double(body[2]!),\n a: double(body[3]!),\n };\n }\n if (body.length === 6) {\n return {\n r: parseInt(body.slice(0, 2), 16),\n g: parseInt(body.slice(2, 4), 16),\n b: parseInt(body.slice(4, 6), 16),\n a: 255,\n };\n }\n return {\n r: parseInt(body.slice(0, 2), 16),\n g: parseInt(body.slice(2, 4), 16),\n b: parseInt(body.slice(4, 6), 16),\n a: parseInt(body.slice(6, 8), 16),\n };\n}\n\nconst CHANNEL_MIN = 0;\nconst CHANNEL_MAX = 255;\n\nfunction assertChannel(name: string, value: number): void {\n if (!Number.isInteger(value) || value < CHANNEL_MIN || value > CHANNEL_MAX) {\n throw new Error(`Color channel ${name} out of range (0-255 integer): ${value}`);\n }\n}\n\nfunction byteToHex(value: number): string {\n return value.toString(16).padStart(2, \"0\");\n}\n\n/**\n * Emit a {@link ColorObject} as `#RRGGBB` when `a === 255`, otherwise `#RRGGBBAA`.\n * Always lowercase, never shorthand. Throws on non-integer or out-of-range channels.\n */\nexport function toHexColor(c: ColorObject): string {\n assertChannel(\"r\", c.r);\n assertChannel(\"g\", c.g);\n assertChannel(\"b\", c.b);\n assertChannel(\"a\", c.a);\n const head = `#${byteToHex(c.r)}${byteToHex(c.g)}${byteToHex(c.b)}`;\n return c.a === 255 ? head : `${head}${byteToHex(c.a)}`;\n}\n\nfunction isInRangeByte(value: unknown): value is number {\n return typeof value === \"number\" && Number.isInteger(value) && value >= 0 && value <= 255;\n}\n\n/**\n * Type guard for a {@link ColorObject}-shaped value (alpha optional).\n * Does not reject extra properties — JSON Schema validation handles that separately.\n */\nexport function isColorObject(value: unknown): value is ColorObject {\n if (value === null || typeof value !== \"object\" || Array.isArray(value)) return false;\n const candidate = value as Record<string, unknown>;\n if (!isInRangeByte(candidate.r)) return false;\n if (!isInRangeByte(candidate.g)) return false;\n if (!isInRangeByte(candidate.b)) return false;\n if (candidate.a !== undefined && !isInRangeByte(candidate.a)) return false;\n return true;\n}\n\n/** Type guard for a hex color string (same regex as `parseHexColor`). */\nexport function isHexColor(value: unknown): value is string {\n return typeof value === \"string\" && HEX_PATTERN.test(value);\n}\n\n/**\n * Normalize either wire form to a full {@link ColorObject}. Object inputs default\n * `a` to 255. Throws on anything that's neither a valid hex string nor a valid\n * color object.\n */\nexport function resolveColor(\n value: string | { r: number; g: number; b: number; a?: number }\n): ColorObject {\n if (typeof value === \"string\") return parseHexColor(value);\n if (!isColorObject(value)) {\n throw new Error(`Invalid color value: ${JSON.stringify(value)}`);\n }\n return {\n r: value.r,\n g: value.g,\n b: value.b,\n a: value.a ?? 255,\n };\n}\n",
|
|
7
|
+
"/**\n * @license\n * Copyright 2026 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport type { ImageBinary } from \"./imageTypes\";\n\nexport interface ImageRasterCodec {\n decodeDataUri(dataUri: string): Promise<ImageBinary>;\n encodeDataUri(image: ImageBinary, mimeType: string): Promise<string>;\n}\n\nlet codec: ImageRasterCodec | null = null;\n\nexport function registerImageRasterCodec(next: ImageRasterCodec): void {\n codec = next;\n}\n\nexport function getImageRasterCodec(): ImageRasterCodec {\n if (!codec) {\n throw new Error(\n \"Image raster codec is not registered. Ensure you import @workglow/tasks from the browser or Node entry (dist/browser.js or dist/node.js), or call registerImageRasterCodec() during startup.\"\n );\n }\n return codec;\n}\n",
|
|
8
|
+
"/**\n * @license\n * Copyright 2026 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n */\n\n/**\n * Structural match for `RawImage` from `@huggingface/transformers`. The HF\n * class carries helpers (`save`, `toCanvas`, `rgb`, ...) that the provider\n * runtime does not invoke on inputs, so a minimal shim over the four data\n * fields is sufficient. If a consumer later needs HF-specific methods we can\n * widen this shim lazily — no need to pull the transformers dep into\n * `@workglow/util/media` today.\n */\nexport class MediaRawImage {\n readonly data: Uint8ClampedArray;\n readonly width: number;\n readonly height: number;\n readonly channels: number;\n\n constructor(data: Uint8ClampedArray, width: number, height: number, channels: number) {\n this.data = data;\n this.width = width;\n this.height = height;\n this.channels = channels;\n }\n}\n\nexport function isMediaRawImageShape(\n value: unknown\n): value is { data: Uint8ClampedArray; width: number; height: number; channels: number } {\n if (!value || typeof value !== \"object\") return false;\n const v = value as Record<string, unknown>;\n return (\n v.data instanceof Uint8ClampedArray &&\n typeof v.width === \"number\" &&\n typeof v.height === \"number\" &&\n typeof v.channels === \"number\"\n );\n}\n",
|
|
9
|
+
"/**\n * @license\n * Copyright 2026 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport type { ImageBinary, ImageChannels, ImageDataSupport } from \"./imageTypes\";\nimport { parseDataUri } from \"./imageTypes\";\nimport { getImageRasterCodec } from \"./imageRasterCodecRegistry\";\nimport { MediaRawImage, isMediaRawImageShape } from \"./MediaRawImage\";\n\n/**\n * Legacy wire format: either a raw `data:image/...;base64,...` URI or a\n * serialized {@link ImageBinary} with `data` as a plain number array\n * (`Array.from(Uint8ClampedArray)`) to avoid the indexed-object pitfall\n * when stringified through `JSON.stringify`.\n */\nexport type ImageJson =\n | string\n | {\n readonly data: readonly number[];\n readonly width: number;\n readonly height: number;\n readonly channels: ImageChannels;\n }\n | {\n // Sentinel for transient browser handles (bitmap / videoFrame /\n // offscreenCanvas) whose pixels haven't been materialized. Lets\n // JSON.stringify of a workflow holding such an image succeed instead\n // of throwing; callers must await getPixels() or getDataUri() before\n // serialization if they need a real round-trip.\n readonly unsynced: true;\n readonly kind: ImageSourceKind;\n };\n\nexport type ImageSourceKind =\n | \"dataUri\"\n | \"pixels\"\n | \"blob\"\n | \"bitmap\"\n | \"videoFrame\"\n | \"offscreenCanvas\";\n\ntype ImageSource =\n | { readonly kind: \"dataUri\"; readonly dataUri: string; readonly mimeType: string }\n | { readonly kind: \"pixels\"; readonly pixels: ImageBinary }\n | { readonly kind: \"blob\"; readonly blob: Blob }\n | { readonly kind: \"bitmap\"; readonly bitmap: ImageBitmap }\n | { readonly kind: \"videoFrame\"; readonly frame: VideoFrame }\n | { readonly kind: \"offscreenCanvas\"; readonly canvas: OffscreenCanvas };\n\nconst IMAGE_BRAND = Symbol.for(\"@workglow/util/media/Image\");\n\nfunction parseDataUriMimeType(dataUri: string): string {\n const match = dataUri.match(/^data:([^;,]+)/);\n const raw = match?.[1]?.trim();\n return raw ? raw.toLowerCase() : \"image/png\";\n}\n\nexport function dataUriToBlob(dataUri: string): Blob {\n const { mimeType, base64 } = parseDataUri(dataUri);\n const binary = atob(base64);\n const bytes = new Uint8Array(binary.length);\n for (let i = 0; i < binary.length; i++) {\n bytes[i] = binary.charCodeAt(i);\n }\n return new Blob([bytes], { type: mimeType });\n}\n\nfunction toImageBinary(value: {\n data: unknown;\n width: number;\n height: number;\n channels: number;\n rawChannels?: number | undefined;\n}): ImageBinary {\n const ch = value.channels;\n if (ch !== 1 && ch !== 3 && ch !== 4) {\n throw new Error(`Image: unsupported channel count ${ch}`);\n }\n const data = coerceToUint8ClampedArray(value.data);\n return {\n data,\n width: value.width,\n height: value.height,\n channels: ch,\n rawChannels: value.rawChannels,\n };\n}\n\nfunction coerceToUint8ClampedArray(data: unknown): Uint8ClampedArray {\n if (data instanceof Uint8ClampedArray) {\n return data;\n }\n if (ArrayBuffer.isView(data)) {\n return new Uint8ClampedArray(\n (data as ArrayBufferView).buffer,\n (data as ArrayBufferView).byteOffset,\n (data as ArrayBufferView).byteLength\n );\n }\n if (Array.isArray(data)) {\n return Uint8ClampedArray.from(data as ArrayLike<number>);\n }\n if (data && typeof data === \"object\") {\n // Indexed-object form produced by JSON.stringify(Uint8ClampedArray).\n const obj = data as Record<string, unknown>;\n const keys = Object.keys(obj);\n if (keys.length > 0 && keys.every((k) => /^\\d+$/.test(k))) {\n const arr = new Uint8ClampedArray(keys.length);\n for (let i = 0; i < keys.length; i++) {\n arr[i] = Number(obj[String(i)]);\n }\n return arr;\n }\n }\n throw new Error(\"Image: pixel data is not array-like\");\n}\n\n/**\n * Unified image representation: holds a single source (dataUri, pixels, blob,\n * bitmap, video frame, or offscreen canvas) and converts on demand.\n *\n * - `toJSON()` normalizes to `{ kind, ... }`; no `Uint8ClampedArray` indexed-\n * object serialization pitfall.\n * - `fromJSON()` accepts the new discriminated shape **and** legacy shapes\n * (raw data URI, raw `ImageBinary`, indexed-object `data`) for migration.\n * - `toFirstSupported(supports[])` replaces `convertImageDataToUseableForm`.\n */\nexport class Image {\n /** @internal brand so `Image.is` works across realms. */\n readonly [IMAGE_BRAND]: true = true;\n\n private source: ImageSource;\n private pixelsCache: ImageBinary | undefined;\n private dataUriCache: Map<string, string> = new Map();\n private blobCache: Map<string, Blob> = new Map();\n\n private constructor(source: ImageSource) {\n this.source = source;\n }\n\n static fromDataUri(dataUri: string): Image {\n if (!dataUri.startsWith(\"data:\")) {\n throw new Error(\"Image.fromDataUri: input must start with 'data:'\");\n }\n return new Image({ kind: \"dataUri\", dataUri, mimeType: parseDataUriMimeType(dataUri) });\n }\n\n static fromPixels(pixels: ImageBinary): Image {\n return new Image({ kind: \"pixels\", pixels });\n }\n\n static fromBlob(blob: Blob): Image {\n return new Image({ kind: \"blob\", blob });\n }\n\n /** Accepts anything `convertImageDataToUseableForm` accepted today, plus `Image`. */\n static from(value: unknown): Image {\n if (Image.is(value)) {\n return value;\n }\n if (typeof value === \"string\" && value.startsWith(\"data:\")) {\n return Image.fromDataUri(value);\n }\n if (typeof Blob !== \"undefined\" && value instanceof Blob) {\n return Image.fromBlob(value);\n }\n if (typeof ImageBitmap !== \"undefined\" && value instanceof ImageBitmap) {\n return new Image({ kind: \"bitmap\", bitmap: value });\n }\n if (typeof VideoFrame !== \"undefined\" && value instanceof VideoFrame) {\n return new Image({ kind: \"videoFrame\", frame: value });\n }\n if (typeof OffscreenCanvas !== \"undefined\" && value instanceof OffscreenCanvas) {\n return new Image({ kind: \"offscreenCanvas\", canvas: value });\n }\n if (\n value &&\n typeof value === \"object\" &&\n \"data\" in value &&\n \"width\" in value &&\n \"height\" in value &&\n \"channels\" in value\n ) {\n return Image.fromPixels(toImageBinary(value as never));\n }\n throw new Error(`Image.from: unsupported image value of type ${typeof value}`);\n }\n\n static is(value: unknown): value is Image {\n return (\n typeof value === \"object\" &&\n value !== null &&\n (value as Record<symbol, unknown>)[IMAGE_BRAND] === true\n );\n }\n\n /**\n * Reconstruct an {@link Image} from its serialized form. Accepts a raw\n * `data:` URI string, a pixel-backed object `{data,width,height,channels}`\n * (with `data` as typed array, plain array, or the indexed-object pitfall\n * produced by `JSON.stringify(Uint8ClampedArray)`), or an `Image` instance.\n */\n static fromJSON(value: unknown): Image {\n if (Image.is(value)) {\n return value;\n }\n if (typeof value === \"string\" && value.startsWith(\"data:\")) {\n return Image.fromDataUri(value);\n }\n if (\n value &&\n typeof value === \"object\" &&\n (value as { unsynced?: unknown }).unsynced === true\n ) {\n const kind = (value as { kind?: unknown }).kind;\n throw new Error(\n `Image.fromJSON: cannot reconstruct image from \"unsynced\" sentinel (kind=${String(kind)}); pixels were not materialized before serialization. Call await image.getPixels() before JSON.stringify.`\n );\n }\n if (\n value &&\n typeof value === \"object\" &&\n \"data\" in value &&\n typeof (value as { width?: unknown }).width === \"number\" &&\n typeof (value as { height?: unknown }).height === \"number\" &&\n typeof (value as { channels?: unknown }).channels === \"number\"\n ) {\n const v = value as {\n data: unknown;\n width: number;\n height: number;\n channels: number;\n };\n return Image.fromPixels(\n toImageBinary({\n data: v.data,\n width: v.width,\n height: v.height,\n channels: v.channels,\n })\n );\n }\n throw new Error(\"Image.fromJSON: value does not match any known Image shape\");\n }\n\n get kind(): ImageSourceKind {\n return this.source.kind;\n }\n\n get mimeType(): string | undefined {\n if (this.source.kind === \"dataUri\") return this.source.mimeType;\n if (this.source.kind === \"blob\") return this.source.blob.type || undefined;\n return undefined;\n }\n\n get width(): number | undefined {\n if (this.source.kind === \"pixels\") return this.source.pixels.width;\n if (this.source.kind === \"bitmap\") return this.source.bitmap.width;\n if (this.source.kind === \"offscreenCanvas\") return this.source.canvas.width;\n if (this.source.kind === \"videoFrame\") return this.source.frame.displayWidth;\n return this.pixelsCache?.width;\n }\n\n get height(): number | undefined {\n if (this.source.kind === \"pixels\") return this.source.pixels.height;\n if (this.source.kind === \"bitmap\") return this.source.bitmap.height;\n if (this.source.kind === \"offscreenCanvas\") return this.source.canvas.height;\n if (this.source.kind === \"videoFrame\") return this.source.frame.displayHeight;\n return this.pixelsCache?.height;\n }\n\n get channels(): ImageChannels | undefined {\n if (this.source.kind === \"pixels\") return this.source.pixels.channels;\n return this.pixelsCache?.channels;\n }\n\n async getPixels(): Promise<ImageBinary> {\n if (this.pixelsCache) return this.pixelsCache;\n if (this.source.kind === \"pixels\") {\n this.pixelsCache = this.source.pixels;\n return this.pixelsCache;\n }\n if (this.source.kind === \"dataUri\") {\n this.pixelsCache = await getImageRasterCodec().decodeDataUri(this.source.dataUri);\n return this.pixelsCache;\n }\n if (this.source.kind === \"blob\") {\n const dataUri = await blobToDataUri(this.source.blob);\n this.pixelsCache = await getImageRasterCodec().decodeDataUri(dataUri);\n return this.pixelsCache;\n }\n throw new Error(\n `Image.getPixels: browser-only source '${this.source.kind}' requires Image.browser augmentation`\n );\n }\n\n async getDataUri(mimeType: string = \"image/png\"): Promise<string> {\n if (this.source.kind === \"dataUri\") {\n if (mimeType === this.source.mimeType) {\n return this.source.dataUri;\n }\n }\n const cached = this.dataUriCache.get(mimeType);\n if (cached) return cached;\n const pixels = await this.getPixels();\n const dataUri = await getImageRasterCodec().encodeDataUri(pixels, mimeType);\n this.dataUriCache.set(mimeType, dataUri);\n return dataUri;\n }\n\n async getBlob(mimeType: string = \"image/png\"): Promise<Blob> {\n if (this.source.kind === \"blob\" && (!mimeType || this.source.blob.type === mimeType)) {\n return this.source.blob;\n }\n const cached = this.blobCache.get(mimeType);\n if (cached) return cached;\n if (this.source.kind === \"dataUri\" && this.source.mimeType === mimeType) {\n const blob = dataUriToBlob(this.source.dataUri);\n this.blobCache.set(mimeType, blob);\n return blob;\n }\n const dataUri = await this.getDataUri(mimeType);\n const blob = dataUriToBlob(dataUri);\n this.blobCache.set(mimeType, blob);\n return blob;\n }\n\n /**\n * Return the first representation in `supports` that can be produced on\n * this platform. Mirrors the list-order semantics of the former\n * `convertImageDataToUseableForm`: earlier entries win.\n */\n async toFirstSupported(supports: readonly ImageDataSupport[]): Promise<unknown> {\n const canonical = this.canonicalSupport();\n if (canonical && supports.includes(canonical)) {\n return this.currentSourceValue();\n }\n for (const want of supports) {\n switch (want) {\n case \"ImageBinary\":\n return this.getPixels();\n case \"Blob\":\n return this.getBlob();\n case \"DataUri\":\n return this.getDataUri();\n case \"RawImage\": {\n const p = await this.getPixels();\n return new MediaRawImage(p.data, p.width, p.height, p.channels);\n }\n case \"ImageBitmap\":\n case \"VideoFrame\":\n case \"OffscreenCanvas\": {\n const asBrowser = this as unknown as {\n toFirstSupportedBrowser?: (\n want: ImageDataSupport\n ) => Promise<unknown> | undefined;\n };\n if (asBrowser.toFirstSupportedBrowser) {\n const produced = await asBrowser.toFirstSupportedBrowser(want);\n if (produced !== undefined) return produced;\n }\n continue;\n }\n case \"Sharp\":\n continue;\n }\n }\n throw new Error(\n `Image.toFirstSupported: none of [${supports.join(\", \")}] can be produced on this platform`\n );\n }\n\n toJSON(): ImageJson {\n if (this.source.kind === \"dataUri\") {\n return this.source.dataUri;\n }\n const pixels =\n this.source.kind === \"pixels\" ? this.source.pixels : this.pixelsCache;\n if (pixels) {\n return {\n data: Array.from(pixels.data),\n width: pixels.width,\n height: pixels.height,\n channels: pixels.channels,\n };\n }\n return { unsynced: true, kind: this.source.kind };\n }\n\n private canonicalSupport(): ImageDataSupport | undefined {\n switch (this.source.kind) {\n case \"dataUri\":\n return \"DataUri\";\n case \"pixels\":\n return \"ImageBinary\";\n case \"blob\":\n return \"Blob\";\n case \"bitmap\":\n return \"ImageBitmap\";\n case \"videoFrame\":\n return \"VideoFrame\";\n case \"offscreenCanvas\":\n return \"OffscreenCanvas\";\n }\n }\n\n /** @internal */\n getSource(): ImageSource {\n return this.source;\n }\n\n /** @internal */\n setPixelsCache(pixels: ImageBinary): void {\n this.pixelsCache = pixels;\n }\n\n private currentSourceValue(): unknown {\n switch (this.source.kind) {\n case \"dataUri\":\n return this.source.dataUri;\n case \"pixels\":\n return this.source.pixels;\n case \"blob\":\n return this.source.blob;\n case \"bitmap\":\n return this.source.bitmap;\n case \"videoFrame\":\n return this.source.frame;\n case \"offscreenCanvas\":\n return this.source.canvas;\n }\n }\n}\n\nasync function blobToDataUri(blob: Blob): Promise<string> {\n const buffer = await blob.arrayBuffer();\n const bytes = new Uint8Array(buffer);\n let binary = \"\";\n const CHUNK = 8192;\n for (let i = 0; i < bytes.byteLength; i += CHUNK) {\n binary += String.fromCharCode(...bytes.subarray(i, i + CHUNK));\n }\n const mime = blob.type || \"image/png\";\n return `data:${mime};base64,${btoa(binary)}`;\n}\n\nexport { MediaRawImage, isMediaRawImageShape };\n",
|
|
10
|
+
"/**\n * @license\n * Copyright 2026 Steven Roussey <sroussey@gmail.com>\n * SPDX-License-Identifier: Apache-2.0\n *\n * Browser-only augmentation of {@link Image}. Adds `getImageData`,\n * `getImageBitmap`, `getVideoFrame`, `getOffscreenCanvas`, and a private\n * helper `toFirstSupportedBrowser` consulted by the base `toFirstSupported`.\n */\n\nimport type { ImageBinary, ImageDataSupport } from \"./imageTypes\";\nimport { Image, dataUriToBlob } from \"./Image\";\n\ndeclare module \"./Image\" {\n interface Image {\n getImageData(): Promise<ImageData>;\n getImageBitmap(): Promise<ImageBitmap>;\n getVideoFrame(): Promise<VideoFrame>;\n getOffscreenCanvas(): Promise<OffscreenCanvas>;\n }\n namespace Image {\n function fromBitmap(bitmap: ImageBitmap): Image;\n function fromVideoFrame(frame: VideoFrame): Image;\n function fromOffscreenCanvas(canvas: OffscreenCanvas): Image;\n }\n}\n\nImage.fromBitmap = function fromBitmap(bitmap: ImageBitmap): Image {\n return Image.from(bitmap);\n};\nImage.fromVideoFrame = function fromVideoFrame(frame: VideoFrame): Image {\n return Image.from(frame);\n};\nImage.fromOffscreenCanvas = function fromOffscreenCanvas(canvas: OffscreenCanvas): Image {\n return Image.from(canvas);\n};\n\nfunction rasterToImageData(image: ImageBinary): ImageData {\n const { width, height, channels, data } = image;\n const id = new ImageData(width, height);\n if (channels === 4) {\n id.data.set(data);\n return id;\n }\n if (channels === 3) {\n for (let i = 0; i < width * height; i++) {\n id.data[i * 4] = data[i * 3]!;\n id.data[i * 4 + 1] = data[i * 3 + 1]!;\n id.data[i * 4 + 2] = data[i * 3 + 2]!;\n id.data[i * 4 + 3] = 255;\n }\n return id;\n }\n if (channels === 1) {\n for (let i = 0; i < width * height; i++) {\n const v = data[i]!;\n id.data[i * 4] = v;\n id.data[i * 4 + 1] = v;\n id.data[i * 4 + 2] = v;\n id.data[i * 4 + 3] = 255;\n }\n return id;\n }\n throw new Error(`Image.getImageData: unsupported channel count ${channels}`);\n}\n\nasync function blobToOffscreenCanvas(blob: Blob): Promise<OffscreenCanvas> {\n const bitmap = await createImageBitmap(blob);\n try {\n const canvas = new OffscreenCanvas(bitmap.width, bitmap.height);\n const ctx = canvas.getContext(\"2d\");\n if (!ctx) throw new Error(\"Image: failed to get 2D context on OffscreenCanvas\");\n ctx.drawImage(bitmap, 0, 0);\n return canvas;\n } finally {\n bitmap.close();\n }\n}\n\nImage.prototype.getImageData = async function getImageData(this: Image): Promise<ImageData> {\n const pixels = await this.getPixels();\n return rasterToImageData(pixels);\n};\n\nImage.prototype.getImageBitmap = async function getImageBitmap(this: Image): Promise<ImageBitmap> {\n const source = this.getSource();\n if (source.kind === \"bitmap\") return source.bitmap;\n if (source.kind === \"blob\") return createImageBitmap(source.blob);\n if (source.kind === \"offscreenCanvas\") {\n return source.canvas.transferToImageBitmap();\n }\n if (source.kind === \"dataUri\") {\n return createImageBitmap(dataUriToBlob(source.dataUri));\n }\n const id = await this.getImageData();\n return createImageBitmap(id);\n};\n\nImage.prototype.getVideoFrame = async function getVideoFrame(this: Image): Promise<VideoFrame> {\n const source = this.getSource();\n if (source.kind === \"videoFrame\") return source.frame;\n const bitmap = await this.getImageBitmap();\n return new VideoFrame(bitmap, { timestamp: 0 });\n};\n\nImage.prototype.getOffscreenCanvas = async function getOffscreenCanvas(\n this: Image\n): Promise<OffscreenCanvas> {\n const source = this.getSource();\n if (source.kind === \"offscreenCanvas\") return source.canvas;\n if (source.kind === \"blob\") return blobToOffscreenCanvas(source.blob);\n if (source.kind === \"dataUri\") return blobToOffscreenCanvas(dataUriToBlob(source.dataUri));\n const id = await this.getImageData();\n const canvas = new OffscreenCanvas(id.width, id.height);\n const ctx = canvas.getContext(\"2d\");\n if (!ctx) throw new Error(\"Image.getOffscreenCanvas: failed to get 2D context\");\n ctx.putImageData(id, 0, 0);\n return canvas;\n};\n\n// Internal hook used by base `toFirstSupported` for browser-only tokens.\n(Image.prototype as unknown as {\n toFirstSupportedBrowser: (want: ImageDataSupport) => Promise<unknown>;\n}).toFirstSupportedBrowser = async function toFirstSupportedBrowser(\n this: Image,\n want: ImageDataSupport\n): Promise<unknown> {\n if (want === \"ImageBitmap\") return this.getImageBitmap();\n if (want === \"VideoFrame\") return this.getVideoFrame();\n if (want === \"OffscreenCanvas\") return this.getOffscreenCanvas();\n return undefined;\n};\n"
|
|
7
11
|
],
|
|
8
|
-
"mappings": ";AA0BO,SAAS,YAAY,CAAC,SAG3B;AAAA,EACA,MAAM,QAAQ,QAAQ,MAAM,4BAA4B;AAAA,EACxD,IAAI,CAAC,OAAO;AAAA,IACV,MAAM,IAAI,MAAM,yBAAyB;AAAA,EAC3C;AAAA,EAEA,OAAO;AAAA,IACL,UAAU,MAAM;AAAA,IAChB,QAAQ,MAAM;AAAA,EAChB;AAAA;;AC1BF,IAAM,+BAA+B,OAAO,SAAyC;AAAA,EACnF,MAAM,MAAM,MAAM,kBAAkB,IAAI;AAAA,EACxC,MAAM,MAAM,IAAI,gBAAgB,IAAI,OAAO,IAAI,MAAM,EAAE,WAAW,IAAI;AAAA,EACtE,IAAI,CAAC,KAAK;AAAA,IACR,MAAM,IAAI,MAAM,wBAAwB;AAAA,EAC1C;AAAA,EACA,IAAI,UAAU,KAAK,GAAG,CAAC;AAAA,EACvB,OAAO,IAAI;AAAA;AAGb,SAAS,aAAa,CAAC,SAAuB;AAAA,EAC5C,QAAQ,UAAU,WAAW,aAAa,OAAO;AAAA,EAEjD,MAAM,SAAS,KAAK,MAAM;AAAA,EAC1B,MAAM,QAAQ,WAAW,KAAK,QAAQ,CAAC,SAAS,KAAK,WAAW,CAAC,CAAC;AAAA,EAElE,MAAM,OAAO,IAAI,KAAK,CAAC,KAAK,GAAG,EAAE,MAAM,SAAS,CAAC;AAAA,EACjD,OAAO;AAAA;AAGT,eAAsB,6BAA6B,CACjD,WACA,UACkB;AAAA,EAClB,IAAI,cAAc,QAAQ,cAAc,WAAW;AAAA,IACjD,MAAM,IAAI,MAAM,iCAAiC;AAAA,EACnD;AAAA,EAGA,IAAI,SAAS,SAAS,aAAa,KAAK,qBAAqB,aAAa;AAAA,IACxE,OAAO;AAAA,EACT;AAAA,EACA,IAAI,SAAS,SAAS,YAAY,KAAK,qBAAqB,YAAY;AAAA,IACtE,OAAO;AAAA,EACT;AAAA,EACA,IAAI,SAAS,SAAS,MAAM,KAAK,qBAAqB,MAAM;AAAA,IAC1D,OAAO;AAAA,EACT;AAAA,EACA,IACE,SAAS,SAAS,aAAa,KAC/B,OAAO,cAAc,YACrB,UAAU,aACV,WAAW,aACX,YAAY,aACZ,cAAc,WACd;AAAA,IACA,OAAO;AAAA,EACT;AAAA,EAGA,IAAI,SAAS,SAAS,aAAa,KAAK,qBAAqB,MAAM;AAAA,IACjE,OAAO,kBAAkB,SAAS;AAAA,EACpC;AAAA,EACA,IAAI,SAAS,SAAS,iBAAiB,KAAK,qBAAqB,MAAM;AAAA,IACrE,OAAO,MAAM,6BAA6B,SAAS;AAAA,EACrD;AAAA,EACA,IAAI,SAAS,SAAS,aAAa,KAAK,qBAAqB,iBAAiB;AAAA,IAC5E,OAAO,UAAU,sBAAsB;AAAA,EACzC;AAAA,EAEA,IAAI,SAAS,SAAS,aAAa,KAAK,OAAO,cAAc,UAAU;AAAA,IACrE,OAAO,kBAAkB,cAAc,SAAS,CAAC;AAAA,EACnD;AAAA,EACA,IAAI,SAAS,SAAS,iBAAiB,KAAK,OAAO,cAAc,UAAU;AAAA,IACzE,OAAO,6BAA6B,cAAc,SAAS,CAAC;AAAA,EAC9D;AAAA,EACA,IAAI,SAAS,SAAS,MAAM,KAAK,OAAO,cAAc,UAAU;AAAA,IAC9D,OAAO,cAAc,SAAS;AAAA,EAChC;AAAA,EACA,IACE,SAAS,SAAS,SAAS,KAC3B,OAAO,cAAc,YACrB,UAAU,WAAW,OAAO,GAC5B;AAAA,IACA,OAAO;AAAA,EACT;AAAA,EACA,MAAM,IAAI,MAAM,gCAAgC,OAAO,YAAY;AAAA;",
|
|
9
|
-
"debugId": "
|
|
12
|
+
"mappings": "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",
|
|
13
|
+
"debugId": "9670F06EEC7694A064756E2164756E21",
|
|
10
14
|
"names": []
|
|
11
15
|
}
|
package/dist/media-node.d.ts
CHANGED
|
@@ -3,6 +3,9 @@
|
|
|
3
3
|
* Copyright 2025 Steven Roussey <sroussey@gmail.com>
|
|
4
4
|
* SPDX-License-Identifier: Apache-2.0
|
|
5
5
|
*/
|
|
6
|
-
export * from "./media/
|
|
7
|
-
export * from "./media/
|
|
6
|
+
export * from "./media/imageTypes";
|
|
7
|
+
export * from "./media/color";
|
|
8
|
+
export * from "./media/imageRasterCodecRegistry";
|
|
9
|
+
export * from "./media/MediaRawImage";
|
|
10
|
+
export * from "./media/Image";
|
|
8
11
|
//# sourceMappingURL=media-node.d.ts.map
|
package/dist/media-node.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"media-node.d.ts","sourceRoot":"","sources":["../src/media-node.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,cAAc,eAAe,CAAC;AAC9B,cAAc,
|
|
1
|
+
{"version":3,"file":"media-node.d.ts","sourceRoot":"","sources":["../src/media-node.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC;AAC9B,cAAc,kCAAkC,CAAC;AACjD,cAAc,uBAAuB,CAAC;AACtC,cAAc,eAAe,CAAC"}
|
package/dist/media-node.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
// src/media/
|
|
1
|
+
// src/media/imageTypes.ts
|
|
2
2
|
function parseDataUri(dataUri) {
|
|
3
3
|
const match = dataUri.match(/^data:([^;]+);base64,(.+)$/);
|
|
4
4
|
if (!match) {
|
|
@@ -9,9 +9,130 @@ function parseDataUri(dataUri) {
|
|
|
9
9
|
base64: match[2]
|
|
10
10
|
};
|
|
11
11
|
}
|
|
12
|
-
// src/media/
|
|
13
|
-
|
|
14
|
-
|
|
12
|
+
// src/media/color.ts
|
|
13
|
+
var HEX_PATTERN = /^#([0-9a-fA-F]{3,4}|[0-9a-fA-F]{6}|[0-9a-fA-F]{8})$/;
|
|
14
|
+
function parseHexColor(hex) {
|
|
15
|
+
if (typeof hex !== "string" || !HEX_PATTERN.test(hex)) {
|
|
16
|
+
throw new Error(`Invalid hex color: ${String(hex)}`);
|
|
17
|
+
}
|
|
18
|
+
const body = hex.slice(1);
|
|
19
|
+
const double = (nibble) => parseInt(nibble + nibble, 16);
|
|
20
|
+
if (body.length === 3) {
|
|
21
|
+
return { r: double(body[0]), g: double(body[1]), b: double(body[2]), a: 255 };
|
|
22
|
+
}
|
|
23
|
+
if (body.length === 4) {
|
|
24
|
+
return {
|
|
25
|
+
r: double(body[0]),
|
|
26
|
+
g: double(body[1]),
|
|
27
|
+
b: double(body[2]),
|
|
28
|
+
a: double(body[3])
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
if (body.length === 6) {
|
|
32
|
+
return {
|
|
33
|
+
r: parseInt(body.slice(0, 2), 16),
|
|
34
|
+
g: parseInt(body.slice(2, 4), 16),
|
|
35
|
+
b: parseInt(body.slice(4, 6), 16),
|
|
36
|
+
a: 255
|
|
37
|
+
};
|
|
38
|
+
}
|
|
39
|
+
return {
|
|
40
|
+
r: parseInt(body.slice(0, 2), 16),
|
|
41
|
+
g: parseInt(body.slice(2, 4), 16),
|
|
42
|
+
b: parseInt(body.slice(4, 6), 16),
|
|
43
|
+
a: parseInt(body.slice(6, 8), 16)
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
var CHANNEL_MIN = 0;
|
|
47
|
+
var CHANNEL_MAX = 255;
|
|
48
|
+
function assertChannel(name, value) {
|
|
49
|
+
if (!Number.isInteger(value) || value < CHANNEL_MIN || value > CHANNEL_MAX) {
|
|
50
|
+
throw new Error(`Color channel ${name} out of range (0-255 integer): ${value}`);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
function byteToHex(value) {
|
|
54
|
+
return value.toString(16).padStart(2, "0");
|
|
55
|
+
}
|
|
56
|
+
function toHexColor(c) {
|
|
57
|
+
assertChannel("r", c.r);
|
|
58
|
+
assertChannel("g", c.g);
|
|
59
|
+
assertChannel("b", c.b);
|
|
60
|
+
assertChannel("a", c.a);
|
|
61
|
+
const head = `#${byteToHex(c.r)}${byteToHex(c.g)}${byteToHex(c.b)}`;
|
|
62
|
+
return c.a === 255 ? head : `${head}${byteToHex(c.a)}`;
|
|
63
|
+
}
|
|
64
|
+
function isInRangeByte(value) {
|
|
65
|
+
return typeof value === "number" && Number.isInteger(value) && value >= 0 && value <= 255;
|
|
66
|
+
}
|
|
67
|
+
function isColorObject(value) {
|
|
68
|
+
if (value === null || typeof value !== "object" || Array.isArray(value))
|
|
69
|
+
return false;
|
|
70
|
+
const candidate = value;
|
|
71
|
+
if (!isInRangeByte(candidate.r))
|
|
72
|
+
return false;
|
|
73
|
+
if (!isInRangeByte(candidate.g))
|
|
74
|
+
return false;
|
|
75
|
+
if (!isInRangeByte(candidate.b))
|
|
76
|
+
return false;
|
|
77
|
+
if (candidate.a !== undefined && !isInRangeByte(candidate.a))
|
|
78
|
+
return false;
|
|
79
|
+
return true;
|
|
80
|
+
}
|
|
81
|
+
function isHexColor(value) {
|
|
82
|
+
return typeof value === "string" && HEX_PATTERN.test(value);
|
|
83
|
+
}
|
|
84
|
+
function resolveColor(value) {
|
|
85
|
+
if (typeof value === "string")
|
|
86
|
+
return parseHexColor(value);
|
|
87
|
+
if (!isColorObject(value)) {
|
|
88
|
+
throw new Error(`Invalid color value: ${JSON.stringify(value)}`);
|
|
89
|
+
}
|
|
90
|
+
return {
|
|
91
|
+
r: value.r,
|
|
92
|
+
g: value.g,
|
|
93
|
+
b: value.b,
|
|
94
|
+
a: value.a ?? 255
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
// src/media/imageRasterCodecRegistry.ts
|
|
98
|
+
var codec = null;
|
|
99
|
+
function registerImageRasterCodec(next) {
|
|
100
|
+
codec = next;
|
|
101
|
+
}
|
|
102
|
+
function getImageRasterCodec() {
|
|
103
|
+
if (!codec) {
|
|
104
|
+
throw new Error("Image raster codec is not registered. Ensure you import @workglow/tasks from the browser or Node entry (dist/browser.js or dist/node.js), or call registerImageRasterCodec() during startup.");
|
|
105
|
+
}
|
|
106
|
+
return codec;
|
|
107
|
+
}
|
|
108
|
+
// src/media/MediaRawImage.ts
|
|
109
|
+
class MediaRawImage {
|
|
110
|
+
data;
|
|
111
|
+
width;
|
|
112
|
+
height;
|
|
113
|
+
channels;
|
|
114
|
+
constructor(data, width, height, channels) {
|
|
115
|
+
this.data = data;
|
|
116
|
+
this.width = width;
|
|
117
|
+
this.height = height;
|
|
118
|
+
this.channels = channels;
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
function isMediaRawImageShape(value) {
|
|
122
|
+
if (!value || typeof value !== "object")
|
|
123
|
+
return false;
|
|
124
|
+
const v = value;
|
|
125
|
+
return v.data instanceof Uint8ClampedArray && typeof v.width === "number" && typeof v.height === "number" && typeof v.channels === "number";
|
|
126
|
+
}
|
|
127
|
+
// src/media/Image.ts
|
|
128
|
+
var IMAGE_BRAND = Symbol.for("@workglow/util/media/Image");
|
|
129
|
+
function parseDataUriMimeType(dataUri) {
|
|
130
|
+
const match = dataUri.match(/^data:([^;,]+)/);
|
|
131
|
+
const raw = match?.[1]?.trim();
|
|
132
|
+
return raw ? raw.toLowerCase() : "image/png";
|
|
133
|
+
}
|
|
134
|
+
function dataUriToBlob(dataUri) {
|
|
135
|
+
const { mimeType, base64 } = parseDataUri(dataUri);
|
|
15
136
|
const binary = atob(base64);
|
|
16
137
|
const bytes = new Uint8Array(binary.length);
|
|
17
138
|
for (let i = 0;i < binary.length; i++) {
|
|
@@ -19,32 +140,312 @@ async function dataUriToBlob(string) {
|
|
|
19
140
|
}
|
|
20
141
|
return new Blob([bytes], { type: mimeType });
|
|
21
142
|
}
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
143
|
+
function toImageBinary(value) {
|
|
144
|
+
const ch = value.channels;
|
|
145
|
+
if (ch !== 1 && ch !== 3 && ch !== 4) {
|
|
146
|
+
throw new Error(`Image: unsupported channel count ${ch}`);
|
|
25
147
|
}
|
|
26
|
-
|
|
27
|
-
|
|
148
|
+
const data = coerceToUint8ClampedArray(value.data);
|
|
149
|
+
return {
|
|
150
|
+
data,
|
|
151
|
+
width: value.width,
|
|
152
|
+
height: value.height,
|
|
153
|
+
channels: ch,
|
|
154
|
+
rawChannels: value.rawChannels
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
function coerceToUint8ClampedArray(data) {
|
|
158
|
+
if (data instanceof Uint8ClampedArray) {
|
|
159
|
+
return data;
|
|
28
160
|
}
|
|
29
|
-
if (
|
|
30
|
-
return
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
161
|
+
if (ArrayBuffer.isView(data)) {
|
|
162
|
+
return new Uint8ClampedArray(data.buffer, data.byteOffset, data.byteLength);
|
|
163
|
+
}
|
|
164
|
+
if (Array.isArray(data)) {
|
|
165
|
+
return Uint8ClampedArray.from(data);
|
|
166
|
+
}
|
|
167
|
+
if (data && typeof data === "object") {
|
|
168
|
+
const obj = data;
|
|
169
|
+
const keys = Object.keys(obj);
|
|
170
|
+
if (keys.length > 0 && keys.every((k) => /^\d+$/.test(k))) {
|
|
171
|
+
const arr = new Uint8ClampedArray(keys.length);
|
|
172
|
+
for (let i = 0;i < keys.length; i++) {
|
|
173
|
+
arr[i] = Number(obj[String(i)]);
|
|
174
|
+
}
|
|
175
|
+
return arr;
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
throw new Error("Image: pixel data is not array-like");
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
class Image {
|
|
182
|
+
[IMAGE_BRAND] = true;
|
|
183
|
+
source;
|
|
184
|
+
pixelsCache;
|
|
185
|
+
dataUriCache = new Map;
|
|
186
|
+
blobCache = new Map;
|
|
187
|
+
constructor(source) {
|
|
188
|
+
this.source = source;
|
|
189
|
+
}
|
|
190
|
+
static fromDataUri(dataUri) {
|
|
191
|
+
if (!dataUri.startsWith("data:")) {
|
|
192
|
+
throw new Error("Image.fromDataUri: input must start with 'data:'");
|
|
193
|
+
}
|
|
194
|
+
return new Image({ kind: "dataUri", dataUri, mimeType: parseDataUriMimeType(dataUri) });
|
|
195
|
+
}
|
|
196
|
+
static fromPixels(pixels) {
|
|
197
|
+
return new Image({ kind: "pixels", pixels });
|
|
198
|
+
}
|
|
199
|
+
static fromBlob(blob) {
|
|
200
|
+
return new Image({ kind: "blob", blob });
|
|
201
|
+
}
|
|
202
|
+
static from(value) {
|
|
203
|
+
if (Image.is(value)) {
|
|
204
|
+
return value;
|
|
205
|
+
}
|
|
206
|
+
if (typeof value === "string" && value.startsWith("data:")) {
|
|
207
|
+
return Image.fromDataUri(value);
|
|
208
|
+
}
|
|
209
|
+
if (typeof Blob !== "undefined" && value instanceof Blob) {
|
|
210
|
+
return Image.fromBlob(value);
|
|
211
|
+
}
|
|
212
|
+
if (typeof ImageBitmap !== "undefined" && value instanceof ImageBitmap) {
|
|
213
|
+
return new Image({ kind: "bitmap", bitmap: value });
|
|
214
|
+
}
|
|
215
|
+
if (typeof VideoFrame !== "undefined" && value instanceof VideoFrame) {
|
|
216
|
+
return new Image({ kind: "videoFrame", frame: value });
|
|
217
|
+
}
|
|
218
|
+
if (typeof OffscreenCanvas !== "undefined" && value instanceof OffscreenCanvas) {
|
|
219
|
+
return new Image({ kind: "offscreenCanvas", canvas: value });
|
|
220
|
+
}
|
|
221
|
+
if (value && typeof value === "object" && "data" in value && "width" in value && "height" in value && "channels" in value) {
|
|
222
|
+
return Image.fromPixels(toImageBinary(value));
|
|
223
|
+
}
|
|
224
|
+
throw new Error(`Image.from: unsupported image value of type ${typeof value}`);
|
|
225
|
+
}
|
|
226
|
+
static is(value) {
|
|
227
|
+
return typeof value === "object" && value !== null && value[IMAGE_BRAND] === true;
|
|
228
|
+
}
|
|
229
|
+
static fromJSON(value) {
|
|
230
|
+
if (Image.is(value)) {
|
|
231
|
+
return value;
|
|
232
|
+
}
|
|
233
|
+
if (typeof value === "string" && value.startsWith("data:")) {
|
|
234
|
+
return Image.fromDataUri(value);
|
|
235
|
+
}
|
|
236
|
+
if (value && typeof value === "object" && value.unsynced === true) {
|
|
237
|
+
const kind = value.kind;
|
|
238
|
+
throw new Error(`Image.fromJSON: cannot reconstruct image from "unsynced" sentinel (kind=${String(kind)}); pixels were not materialized before serialization. Call await image.getPixels() before JSON.stringify.`);
|
|
239
|
+
}
|
|
240
|
+
if (value && typeof value === "object" && "data" in value && typeof value.width === "number" && typeof value.height === "number" && typeof value.channels === "number") {
|
|
241
|
+
const v = value;
|
|
242
|
+
return Image.fromPixels(toImageBinary({
|
|
243
|
+
data: v.data,
|
|
244
|
+
width: v.width,
|
|
245
|
+
height: v.height,
|
|
246
|
+
channels: v.channels
|
|
247
|
+
}));
|
|
248
|
+
}
|
|
249
|
+
throw new Error("Image.fromJSON: value does not match any known Image shape");
|
|
36
250
|
}
|
|
37
|
-
|
|
38
|
-
return
|
|
251
|
+
get kind() {
|
|
252
|
+
return this.source.kind;
|
|
39
253
|
}
|
|
40
|
-
|
|
41
|
-
|
|
254
|
+
get mimeType() {
|
|
255
|
+
if (this.source.kind === "dataUri")
|
|
256
|
+
return this.source.mimeType;
|
|
257
|
+
if (this.source.kind === "blob")
|
|
258
|
+
return this.source.blob.type || undefined;
|
|
259
|
+
return;
|
|
260
|
+
}
|
|
261
|
+
get width() {
|
|
262
|
+
if (this.source.kind === "pixels")
|
|
263
|
+
return this.source.pixels.width;
|
|
264
|
+
if (this.source.kind === "bitmap")
|
|
265
|
+
return this.source.bitmap.width;
|
|
266
|
+
if (this.source.kind === "offscreenCanvas")
|
|
267
|
+
return this.source.canvas.width;
|
|
268
|
+
if (this.source.kind === "videoFrame")
|
|
269
|
+
return this.source.frame.displayWidth;
|
|
270
|
+
return this.pixelsCache?.width;
|
|
271
|
+
}
|
|
272
|
+
get height() {
|
|
273
|
+
if (this.source.kind === "pixels")
|
|
274
|
+
return this.source.pixels.height;
|
|
275
|
+
if (this.source.kind === "bitmap")
|
|
276
|
+
return this.source.bitmap.height;
|
|
277
|
+
if (this.source.kind === "offscreenCanvas")
|
|
278
|
+
return this.source.canvas.height;
|
|
279
|
+
if (this.source.kind === "videoFrame")
|
|
280
|
+
return this.source.frame.displayHeight;
|
|
281
|
+
return this.pixelsCache?.height;
|
|
282
|
+
}
|
|
283
|
+
get channels() {
|
|
284
|
+
if (this.source.kind === "pixels")
|
|
285
|
+
return this.source.pixels.channels;
|
|
286
|
+
return this.pixelsCache?.channels;
|
|
287
|
+
}
|
|
288
|
+
async getPixels() {
|
|
289
|
+
if (this.pixelsCache)
|
|
290
|
+
return this.pixelsCache;
|
|
291
|
+
if (this.source.kind === "pixels") {
|
|
292
|
+
this.pixelsCache = this.source.pixels;
|
|
293
|
+
return this.pixelsCache;
|
|
294
|
+
}
|
|
295
|
+
if (this.source.kind === "dataUri") {
|
|
296
|
+
this.pixelsCache = await getImageRasterCodec().decodeDataUri(this.source.dataUri);
|
|
297
|
+
return this.pixelsCache;
|
|
298
|
+
}
|
|
299
|
+
if (this.source.kind === "blob") {
|
|
300
|
+
const dataUri = await blobToDataUri(this.source.blob);
|
|
301
|
+
this.pixelsCache = await getImageRasterCodec().decodeDataUri(dataUri);
|
|
302
|
+
return this.pixelsCache;
|
|
303
|
+
}
|
|
304
|
+
throw new Error(`Image.getPixels: browser-only source '${this.source.kind}' requires Image.browser augmentation`);
|
|
305
|
+
}
|
|
306
|
+
async getDataUri(mimeType = "image/png") {
|
|
307
|
+
if (this.source.kind === "dataUri") {
|
|
308
|
+
if (mimeType === this.source.mimeType) {
|
|
309
|
+
return this.source.dataUri;
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
const cached = this.dataUriCache.get(mimeType);
|
|
313
|
+
if (cached)
|
|
314
|
+
return cached;
|
|
315
|
+
const pixels = await this.getPixels();
|
|
316
|
+
const dataUri = await getImageRasterCodec().encodeDataUri(pixels, mimeType);
|
|
317
|
+
this.dataUriCache.set(mimeType, dataUri);
|
|
318
|
+
return dataUri;
|
|
319
|
+
}
|
|
320
|
+
async getBlob(mimeType = "image/png") {
|
|
321
|
+
if (this.source.kind === "blob" && (!mimeType || this.source.blob.type === mimeType)) {
|
|
322
|
+
return this.source.blob;
|
|
323
|
+
}
|
|
324
|
+
const cached = this.blobCache.get(mimeType);
|
|
325
|
+
if (cached)
|
|
326
|
+
return cached;
|
|
327
|
+
if (this.source.kind === "dataUri" && this.source.mimeType === mimeType) {
|
|
328
|
+
const blob2 = dataUriToBlob(this.source.dataUri);
|
|
329
|
+
this.blobCache.set(mimeType, blob2);
|
|
330
|
+
return blob2;
|
|
331
|
+
}
|
|
332
|
+
const dataUri = await this.getDataUri(mimeType);
|
|
333
|
+
const blob = dataUriToBlob(dataUri);
|
|
334
|
+
this.blobCache.set(mimeType, blob);
|
|
335
|
+
return blob;
|
|
336
|
+
}
|
|
337
|
+
async toFirstSupported(supports) {
|
|
338
|
+
const canonical = this.canonicalSupport();
|
|
339
|
+
if (canonical && supports.includes(canonical)) {
|
|
340
|
+
return this.currentSourceValue();
|
|
341
|
+
}
|
|
342
|
+
for (const want of supports) {
|
|
343
|
+
switch (want) {
|
|
344
|
+
case "ImageBinary":
|
|
345
|
+
return this.getPixels();
|
|
346
|
+
case "Blob":
|
|
347
|
+
return this.getBlob();
|
|
348
|
+
case "DataUri":
|
|
349
|
+
return this.getDataUri();
|
|
350
|
+
case "RawImage": {
|
|
351
|
+
const p = await this.getPixels();
|
|
352
|
+
return new MediaRawImage(p.data, p.width, p.height, p.channels);
|
|
353
|
+
}
|
|
354
|
+
case "ImageBitmap":
|
|
355
|
+
case "VideoFrame":
|
|
356
|
+
case "OffscreenCanvas": {
|
|
357
|
+
const asBrowser = this;
|
|
358
|
+
if (asBrowser.toFirstSupportedBrowser) {
|
|
359
|
+
const produced = await asBrowser.toFirstSupportedBrowser(want);
|
|
360
|
+
if (produced !== undefined)
|
|
361
|
+
return produced;
|
|
362
|
+
}
|
|
363
|
+
continue;
|
|
364
|
+
}
|
|
365
|
+
case "Sharp":
|
|
366
|
+
continue;
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
throw new Error(`Image.toFirstSupported: none of [${supports.join(", ")}] can be produced on this platform`);
|
|
370
|
+
}
|
|
371
|
+
toJSON() {
|
|
372
|
+
if (this.source.kind === "dataUri") {
|
|
373
|
+
return this.source.dataUri;
|
|
374
|
+
}
|
|
375
|
+
const pixels = this.source.kind === "pixels" ? this.source.pixels : this.pixelsCache;
|
|
376
|
+
if (pixels) {
|
|
377
|
+
return {
|
|
378
|
+
data: Array.from(pixels.data),
|
|
379
|
+
width: pixels.width,
|
|
380
|
+
height: pixels.height,
|
|
381
|
+
channels: pixels.channels
|
|
382
|
+
};
|
|
383
|
+
}
|
|
384
|
+
return { unsynced: true, kind: this.source.kind };
|
|
385
|
+
}
|
|
386
|
+
canonicalSupport() {
|
|
387
|
+
switch (this.source.kind) {
|
|
388
|
+
case "dataUri":
|
|
389
|
+
return "DataUri";
|
|
390
|
+
case "pixels":
|
|
391
|
+
return "ImageBinary";
|
|
392
|
+
case "blob":
|
|
393
|
+
return "Blob";
|
|
394
|
+
case "bitmap":
|
|
395
|
+
return "ImageBitmap";
|
|
396
|
+
case "videoFrame":
|
|
397
|
+
return "VideoFrame";
|
|
398
|
+
case "offscreenCanvas":
|
|
399
|
+
return "OffscreenCanvas";
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
getSource() {
|
|
403
|
+
return this.source;
|
|
404
|
+
}
|
|
405
|
+
setPixelsCache(pixels) {
|
|
406
|
+
this.pixelsCache = pixels;
|
|
407
|
+
}
|
|
408
|
+
currentSourceValue() {
|
|
409
|
+
switch (this.source.kind) {
|
|
410
|
+
case "dataUri":
|
|
411
|
+
return this.source.dataUri;
|
|
412
|
+
case "pixels":
|
|
413
|
+
return this.source.pixels;
|
|
414
|
+
case "blob":
|
|
415
|
+
return this.source.blob;
|
|
416
|
+
case "bitmap":
|
|
417
|
+
return this.source.bitmap;
|
|
418
|
+
case "videoFrame":
|
|
419
|
+
return this.source.frame;
|
|
420
|
+
case "offscreenCanvas":
|
|
421
|
+
return this.source.canvas;
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
async function blobToDataUri(blob) {
|
|
426
|
+
const buffer = await blob.arrayBuffer();
|
|
427
|
+
const bytes = new Uint8Array(buffer);
|
|
428
|
+
let binary = "";
|
|
429
|
+
const CHUNK = 8192;
|
|
430
|
+
for (let i = 0;i < bytes.byteLength; i += CHUNK) {
|
|
431
|
+
binary += String.fromCharCode(...bytes.subarray(i, i + CHUNK));
|
|
42
432
|
}
|
|
43
|
-
|
|
433
|
+
const mime = blob.type || "image/png";
|
|
434
|
+
return `data:${mime};base64,${btoa(binary)}`;
|
|
44
435
|
}
|
|
45
436
|
export {
|
|
437
|
+
toHexColor,
|
|
438
|
+
resolveColor,
|
|
439
|
+
registerImageRasterCodec,
|
|
440
|
+
parseHexColor,
|
|
46
441
|
parseDataUri,
|
|
47
|
-
|
|
442
|
+
isMediaRawImageShape,
|
|
443
|
+
isHexColor,
|
|
444
|
+
isColorObject,
|
|
445
|
+
getImageRasterCodec,
|
|
446
|
+
dataUriToBlob,
|
|
447
|
+
MediaRawImage,
|
|
448
|
+
Image
|
|
48
449
|
};
|
|
49
450
|
|
|
50
|
-
//# debugId=
|
|
451
|
+
//# debugId=DC7CC1D941A4FA1F64756E2164756E21
|