yummies 7.11.0 → 7.13.0

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.
Files changed (159) hide show
  1. package/README.md +5 -87
  2. package/async.cjs +179 -48
  3. package/async.cjs.map +1 -1
  4. package/async.d.ts +125 -7
  5. package/async.js +180 -54
  6. package/async.js.map +1 -1
  7. package/chunk-CVq3Gv4J.cjs +50 -0
  8. package/chunk-YKewjYmz.js +37 -0
  9. package/common.cjs +48 -8
  10. package/common.cjs.map +1 -1
  11. package/common.d.ts +53 -2
  12. package/common.js +49 -11
  13. package/common.js.map +1 -1
  14. package/complex.cjs +275 -128
  15. package/complex.cjs.map +1 -1
  16. package/complex.d.ts +66 -0
  17. package/complex.js +275 -133
  18. package/complex.js.map +1 -1
  19. package/cookie.cjs +17 -7
  20. package/cookie.cjs.map +1 -1
  21. package/cookie.d.ts +26 -0
  22. package/cookie.js +18 -9
  23. package/cookie.js.map +1 -1
  24. package/css.cjs +163 -39
  25. package/css.cjs.map +1 -1
  26. package/css.d.ts +115 -6
  27. package/css.js +159 -41
  28. package/css.js.map +1 -1
  29. package/data.cjs +90 -55
  30. package/data.cjs.map +1 -1
  31. package/data.d.ts +50 -0
  32. package/data.js +91 -61
  33. package/data.js.map +1 -1
  34. package/date-time.cjs +594 -412
  35. package/date-time.cjs.map +1 -1
  36. package/date-time.d.ts +105 -0
  37. package/date-time.js +591 -421
  38. package/date-time.js.map +1 -1
  39. package/device.cjs +65 -23
  40. package/device.cjs.map +1 -1
  41. package/device.d.ts +49 -0
  42. package/device.js +66 -31
  43. package/device.js.map +1 -1
  44. package/encodings.cjs +275 -266
  45. package/encodings.cjs.map +1 -1
  46. package/encodings.d.ts +25 -0
  47. package/encodings.js +276 -268
  48. package/encodings.js.map +1 -1
  49. package/errors.cjs +36 -18
  50. package/errors.cjs.map +1 -1
  51. package/errors.d.ts +17 -0
  52. package/errors.js +35 -19
  53. package/errors.js.map +1 -1
  54. package/file.cjs +58 -24
  55. package/file.cjs.map +1 -1
  56. package/file.d.ts +32 -0
  57. package/file.js +59 -27
  58. package/file.js.map +1 -1
  59. package/format.cjs +125 -83
  60. package/format.cjs.map +1 -1
  61. package/format.d.ts +18 -0
  62. package/format.js +118 -82
  63. package/format.js.map +1 -1
  64. package/html.cjs +242 -137
  65. package/html.cjs.map +1 -1
  66. package/html.d.ts +81 -0
  67. package/html.js +239 -150
  68. package/html.js.map +1 -1
  69. package/id.cjs +90 -17
  70. package/id.cjs.map +1 -1
  71. package/id.d.ts +16 -0
  72. package/id.js +89 -24
  73. package/id.js.map +1 -1
  74. package/imports.cjs +57 -29
  75. package/imports.cjs.map +1 -1
  76. package/imports.d.ts +24 -0
  77. package/imports.js +56 -31
  78. package/imports.js.map +1 -1
  79. package/math.cjs +32 -6
  80. package/math.cjs.map +1 -1
  81. package/math.d.ts +33 -0
  82. package/math.js +33 -10
  83. package/math.js.map +1 -1
  84. package/media.cjs +291 -84
  85. package/media.cjs.map +1 -1
  86. package/media.d.ts +204 -2
  87. package/media.js +290 -93
  88. package/media.js.map +1 -1
  89. package/mobx.cjs +449 -193
  90. package/mobx.cjs.map +1 -1
  91. package/mobx.d.ts +108 -0
  92. package/mobx.js +447 -200
  93. package/mobx.js.map +1 -1
  94. package/ms.cjs +37 -10
  95. package/ms.cjs.map +1 -1
  96. package/ms.d.ts +16 -0
  97. package/ms.js +38 -13
  98. package/ms.js.map +1 -1
  99. package/number.cjs +29 -7
  100. package/number.cjs.map +1 -1
  101. package/number.d.ts +16 -0
  102. package/number.js +30 -9
  103. package/number.js.map +1 -1
  104. package/package.json +11 -3
  105. package/parser.cjs +117 -64
  106. package/parser.cjs.map +1 -1
  107. package/parser.d.ts +17 -0
  108. package/parser.js +111 -64
  109. package/parser.js.map +1 -1
  110. package/price.cjs +24 -18
  111. package/price.cjs.map +1 -1
  112. package/price.d.ts +24 -0
  113. package/price.js +25 -20
  114. package/price.js.map +1 -1
  115. package/random.cjs +95 -13
  116. package/random.cjs.map +1 -1
  117. package/random.d.ts +80 -0
  118. package/random.js +96 -22
  119. package/random.js.map +1 -1
  120. package/react.cjs +673 -214
  121. package/react.cjs.map +1 -1
  122. package/react.d.ts +21 -0
  123. package/react.js +674 -239
  124. package/react.js.map +1 -1
  125. package/sound.cjs +30 -9
  126. package/sound.cjs.map +1 -1
  127. package/sound.d.ts +16 -0
  128. package/sound.js +31 -11
  129. package/sound.js.map +1 -1
  130. package/storage.cjs +49 -50
  131. package/storage.cjs.map +1 -1
  132. package/storage.d.ts +24 -0
  133. package/storage.js +50 -53
  134. package/storage.js.map +1 -1
  135. package/text.cjs +67 -34
  136. package/text.cjs.map +1 -1
  137. package/text.d.ts +16 -0
  138. package/text.js +68 -37
  139. package/text.js.map +1 -1
  140. package/type-guard.cjs +292 -72
  141. package/type-guard.cjs.map +1 -1
  142. package/type-guard.d.ts +18 -0
  143. package/type-guard.js +288 -73
  144. package/type-guard.js.map +1 -1
  145. package/types.cjs +0 -2
  146. package/types.d.ts +41 -0
  147. package/types.global.cjs +0 -2
  148. package/types.global.d.ts +41 -0
  149. package/types.global.js +0 -2
  150. package/types.js +0 -2
  151. package/vibrate.cjs +47 -6
  152. package/vibrate.cjs.map +1 -1
  153. package/vibrate.d.ts +39 -1
  154. package/vibrate.js +48 -8
  155. package/vibrate.js.map +1 -1
  156. package/types.cjs.map +0 -1
  157. package/types.global.cjs.map +0 -1
  158. package/types.global.js.map +0 -1
  159. package/types.js.map +0 -1
package/media.js CHANGED
@@ -1,106 +1,303 @@
1
1
  import { degToRad } from "yummies/math";
2
+ //#region src/media.ts
3
+ /**
4
+ * ---header-docs-section---
5
+ * # yummies/media
6
+ *
7
+ * ## Description
8
+ *
9
+ * Binary and media helpers: Base64 data URLs, object URLs, image orientation fixes, and related
10
+ * browser APIs wrapped in promises. Centralizes `FileReader`/`URL.createObjectURL` usage so image
11
+ * pipelines and uploads stay consistent and easier to test or mock.
12
+ *
13
+ * ## Usage
14
+ *
15
+ * ```ts
16
+ * import { blobToBase64 } from "yummies/media";
17
+ * ```
18
+ */
19
+ /**
20
+ * Reads a {@link Blob} as a **data URL** string (`data:<mime>;base64,...`) using {@link FileReader#readAsDataURL}.
21
+ *
22
+ * Useful for previewing uploads, embedding small assets inline, or serializing binary for APIs that
23
+ * expect Base64-in-JSON. The result includes the MIME prefix, not raw Base64 alone — use
24
+ * {@link decodeDataUrl} if you need the payload and type separately.
25
+ *
26
+ * @param blob - Any `Blob` or `File` (files are blobs).
27
+ * @returns Resolves to the data URL string; rejects if reading fails.
28
+ *
29
+ * @example
30
+ * ```ts
31
+ * const dataUrl = await blobToBase64(file);
32
+ * previewImg.src = dataUrl;
33
+ * ```
34
+ *
35
+ * @example
36
+ * ```ts
37
+ * const fromFetch = await fetch('/api/export').then((r) => r.blob());
38
+ * const dataUrl = await blobToBase64(fromFetch);
39
+ * ```
40
+ */
2
41
  function blobToBase64(blob) {
3
- return new Promise((resolve, reject) => {
4
- const reader = new FileReader();
5
- reader.onloadend = () => resolve(reader.result);
6
- reader.onerror = reject;
7
- reader.readAsDataURL(blob);
8
- });
42
+ return new Promise((resolve, reject) => {
43
+ const reader = new FileReader();
44
+ reader.onloadend = () => resolve(reader.result);
45
+ reader.onerror = reject;
46
+ reader.readAsDataURL(blob);
47
+ });
9
48
  }
10
- const blobToUrl = (urlOrBlob) => urlOrBlob instanceof Blob ? URL.createObjectURL(urlOrBlob) : urlOrBlob;
11
- const fileToBlob = (file) => {
12
- return new Blob([file], { type: file.type });
49
+ /**
50
+ * If `urlOrBlob` is already a string, returns it unchanged. If it is a {@link Blob}, returns
51
+ * `URL.createObjectURL(blob)` a short-lived `blob:` URL valid in this document until
52
+ * {@link URL.revokeObjectURL} is called.
53
+ *
54
+ * Pair with {@link renderImage} or `<img src>` without re-fetching binary data. **Remember to
55
+ * `revokeObjectURL`** when the URL is no longer needed to avoid retaining blob memory.
56
+ *
57
+ * @param urlOrBlob - Remote/http(s) URL string, data URL, or a `Blob` / `File`.
58
+ * @returns A string suitable for `HTMLImageElement.src` or similar.
59
+ *
60
+ * @example
61
+ * ```ts
62
+ * const src = blobToUrl(uploadedFile);
63
+ * img.src = src;
64
+ * // when done:
65
+ * URL.revokeObjectURL(src);
66
+ * ```
67
+ *
68
+ * @example
69
+ * ```ts
70
+ * blobToUrl('https://cdn.example.com/logo.png'); // passed through as-is
71
+ * ```
72
+ */
73
+ var blobToUrl = (urlOrBlob) => urlOrBlob instanceof Blob ? URL.createObjectURL(urlOrBlob) : urlOrBlob;
74
+ /**
75
+ * Creates a new {@link Blob} from a {@link File}, copying the bytes and keeping `file.type` as
76
+ * the blob’s MIME type. Handy when an API accepts `Blob` but you only have `File`, or you want a
77
+ * plain blob without the `File` name/lastModified metadata.
78
+ *
79
+ * @param file - Source file from an `<input type="file">` or drag-and-drop.
80
+ * @returns A `Blob` with the same content and `type` as the file.
81
+ *
82
+ * @example
83
+ * ```ts
84
+ * const blob = fileToBlob(fileFromInput);
85
+ * await uploadEndpoint(blob);
86
+ * ```
87
+ *
88
+ * @example
89
+ * ```ts
90
+ * const blob = fileToBlob(imageFile);
91
+ * const url = URL.createObjectURL(blob);
92
+ * ```
93
+ */
94
+ var fileToBlob = (file) => {
95
+ return new Blob([file], { type: file.type });
13
96
  };
14
- const imageToBlob = (imageElement, mimeType = "image/png") => {
15
- const canvas = document.createElement("canvas");
16
- canvas.width = imageElement.naturalWidth || 300;
17
- canvas.height = imageElement.naturalHeight || 300;
18
- canvas.getContext("2d").drawImage(imageElement, 0, 0);
19
- const dataUri = canvas.toDataURL(mimeType, 1);
20
- const base64data = dataUri.split(",")[1];
21
- const base64MimeType = dataUri.split(";")[0].slice(5);
22
- const bytes = globalThis.atob(base64data);
23
- const buf = new ArrayBuffer(bytes.length);
24
- const array = new Uint8Array(buf);
25
- for (let index = 0; index < bytes.length; index++) {
26
- array[index] = bytes.charCodeAt(index);
27
- }
28
- const blob = new Blob([array], { type: base64MimeType });
29
- return blob;
97
+ /**
98
+ * Draws an {@link HTMLImageElement} onto an offscreen canvas, then builds a {@link Blob} from the
99
+ * raster (via `canvas.toDataURL` + binary decode). Dimensions use `naturalWidth` / `naturalHeight`,
100
+ * falling back to `300×300` if those are zero (e.g. not yet decoded).
101
+ *
102
+ * **CORS:** if the image is cross-origin without proper CORS headers, the canvas may be
103
+ * *tainted* and `toDataURL` can throw — same browser rules as any canvas export.
104
+ *
105
+ * @param imageElement - Loaded image (`complete` / decoded recommended).
106
+ * @param mimeType - Output MIME type, e.g. `'image/png'` (default) or `'image/jpeg'`. Encoder support is browser-dependent.
107
+ * @returns Encoded image as a `Blob` with a matching `type` when possible.
108
+ *
109
+ * @example
110
+ * ```ts
111
+ * const img = await renderImage('/photo.jpg');
112
+ * const jpegBlob = imageToBlob(img, 'image/jpeg');
113
+ * ```
114
+ *
115
+ * @example
116
+ * ```ts
117
+ * const pngBlob = imageToBlob(cachedImg); // default image/png
118
+ * ```
119
+ */
120
+ var imageToBlob = (imageElement, mimeType = "image/png") => {
121
+ const canvas = document.createElement("canvas");
122
+ canvas.width = imageElement.naturalWidth || 300;
123
+ canvas.height = imageElement.naturalHeight || 300;
124
+ canvas.getContext("2d").drawImage(imageElement, 0, 0);
125
+ const dataUri = canvas.toDataURL(mimeType, 1);
126
+ const base64data = dataUri.split(",")[1];
127
+ const base64MimeType = dataUri.split(";")[0].slice(5);
128
+ const bytes = globalThis.atob(base64data);
129
+ const buf = new ArrayBuffer(bytes.length);
130
+ const array = new Uint8Array(buf);
131
+ for (let index = 0; index < bytes.length; index++) array[index] = bytes.charCodeAt(index);
132
+ return new Blob([array], { type: base64MimeType });
30
133
  };
31
- const renderImage = (urlOrBlob) => new Promise((resolve, reject) => {
32
- const image = new Image();
33
- image.src = blobToUrl(urlOrBlob);
34
- image.onload = () => resolve(image);
35
- image.onerror = () => reject();
134
+ /**
135
+ * Loads a resource into a new `HTMLImageElement`: `src` is set via {@link blobToUrl} (so blobs get
136
+ * object URLs, strings are used directly). Resolves on `load` with the same element; rejects on
137
+ * `error` (e.g. bad URL, network failure, corrupt image) **with no rejection value**.
138
+ *
139
+ * Does not add the node to the DOM — use the returned element for canvas, measuring, or
140
+ * {@link imageToBlob}.
141
+ *
142
+ * @param urlOrBlob - Remote URL, data URL, or `Blob` / `File`.
143
+ * @returns Promise that fulfills with the loaded `HTMLImageElement`.
144
+ *
145
+ * @example
146
+ * ```ts
147
+ * const img = await renderImage('https://example.com/pic.png');
148
+ * document.body.appendChild(img);
149
+ * ```
150
+ *
151
+ * @example
152
+ * ```ts
153
+ * const img = await renderImage(pickedFile);
154
+ * const blob = imageToBlob(img, 'image/webp');
155
+ * ```
156
+ */
157
+ var renderImage = (urlOrBlob) => new Promise((resolve, reject) => {
158
+ const image = new Image();
159
+ image.src = blobToUrl(urlOrBlob);
160
+ image.onload = () => resolve(image);
161
+ image.onerror = () => reject();
36
162
  });
37
163
  function cropImageFromCanvas(context) {
38
- const canvas = context.canvas;
39
- let w = canvas.width;
40
- let h = canvas.height;
41
- const pix = { x: [], y: [] };
42
- const imageData = context.getImageData(0, 0, canvas.width, canvas.height);
43
- let x;
44
- let y;
45
- let index;
46
- for (y = 0; y < h; y++) {
47
- for (x = 0; x < w; x++) {
48
- index = (y * w + x) * 4;
49
- if (imageData.data[index + 3] > 0) {
50
- pix.x.push(x);
51
- pix.y.push(y);
52
- }
53
- }
54
- }
55
- pix.x.sort((a, b) => a - b);
56
- pix.y.sort((a, b) => a - b);
57
- const n = pix.x.length - 1;
58
- w = 1 + pix.x[n] - pix.x[0];
59
- h = 1 + pix.y[n] - pix.y[0];
60
- const cut = context.getImageData(pix.x[0], pix.y[0], w, h);
61
- canvas.width = w;
62
- canvas.height = h;
63
- context.putImageData(cut, 0, 0);
64
- return canvas;
164
+ const canvas = context.canvas;
165
+ let w = canvas.width;
166
+ let h = canvas.height;
167
+ const pix = {
168
+ x: [],
169
+ y: []
170
+ };
171
+ const imageData = context.getImageData(0, 0, canvas.width, canvas.height);
172
+ let x;
173
+ let y;
174
+ let index;
175
+ for (y = 0; y < h; y++) for (x = 0; x < w; x++) {
176
+ index = (y * w + x) * 4;
177
+ if (imageData.data[index + 3] > 0) {
178
+ pix.x.push(x);
179
+ pix.y.push(y);
180
+ }
181
+ }
182
+ pix.x.sort((a, b) => a - b);
183
+ pix.y.sort((a, b) => a - b);
184
+ const n = pix.x.length - 1;
185
+ w = 1 + pix.x[n] - pix.x[0];
186
+ h = 1 + pix.y[n] - pix.y[0];
187
+ const cut = context.getImageData(pix.x[0], pix.y[0], w, h);
188
+ canvas.width = w;
189
+ canvas.height = h;
190
+ context.putImageData(cut, 0, 0);
191
+ return canvas;
65
192
  }
66
- const rotateImage = (image, angle) => {
67
- const maxSize = Math.max(image.width, image.height);
68
- const canvas = document.createElement("canvas");
69
- canvas.width = maxSize;
70
- canvas.height = maxSize;
71
- const context = canvas.getContext("2d");
72
- context.save();
73
- context.translate(canvas.width / 2, canvas.height / 2);
74
- context.rotate(degToRad(angle));
75
- context.drawImage(image, -image.width / 2, -image.height / 2);
76
- context.restore();
77
- cropImageFromCanvas(context);
78
- return renderImage(canvas.toDataURL("image/png"));
193
+ /**
194
+ * Rotates `image` around its center on a square canvas (side = max(width, height)), then **crops**
195
+ * transparent margins by trimming to the bounding box of pixels with non-zero alpha.
196
+ * Returns a **new** loaded `HTMLImageElement` (PNG data URL under the hood via {@link renderImage}).
197
+ *
198
+ * `angle` is in **degrees**; converted with {@link degToRad} from `yummies/math`.
199
+ *
200
+ * Very large sources can stress memory on some mobile browsers (known iPhone issues see TODO in source).
201
+ *
202
+ * @param image - Source image (should be decoded; uses `width` / `height`).
203
+ * @param angle - Rotation in degrees (e.g. `90` for quarter turn).
204
+ * @returns Promise of a new `HTMLImageElement` showing the rotated, cropped result.
205
+ *
206
+ * @example
207
+ * ```ts
208
+ * const upright = await rotateImage(landscapeImg, 90);
209
+ * ```
210
+ *
211
+ * @example
212
+ * ```ts
213
+ * const fixed = await rotateImage(await renderImage(file), -15);
214
+ * ```
215
+ */
216
+ var rotateImage = (image, angle) => {
217
+ const maxSize = Math.max(image.width, image.height);
218
+ const canvas = document.createElement("canvas");
219
+ canvas.width = maxSize;
220
+ canvas.height = maxSize;
221
+ const context = canvas.getContext("2d");
222
+ context.save();
223
+ context.translate(canvas.width / 2, canvas.height / 2);
224
+ context.rotate(degToRad(angle));
225
+ context.drawImage(image, -image.width / 2, -image.height / 2);
226
+ context.restore();
227
+ cropImageFromCanvas(context);
228
+ return renderImage(canvas.toDataURL("image/png"));
79
229
  };
230
+ /**
231
+ * Parses a `data:` URL of the form `data:<mime>;base64,<payload>` into its MIME type and raw
232
+ * Base64 body (without the `data:...;base64,` prefix). Non-matching strings yield `undefined`
233
+ * fields in the result object.
234
+ *
235
+ * @param url - Full data URL string.
236
+ * @returns `{ mimeType, data }` — both optional when the regex does not match.
237
+ *
238
+ * @example
239
+ * ```ts
240
+ * const { mimeType, data } = decodeDataUrl('data:image/png;base64,iVBORw0KGgo=');
241
+ * // mimeType === 'image/png', data === 'iVBORw0KGgo='
242
+ * ```
243
+ *
244
+ * @example
245
+ * ```ts
246
+ * decodeDataUrl('not-a-data-url'); // { mimeType: undefined, data: undefined }
247
+ * ```
248
+ */
80
249
  function decodeDataUrl(url) {
81
- const regex = /^data:(.*);base64,\s?(.*)$/;
82
- const matches = new RegExp(regex).exec(url);
83
- return {
84
- mimeType: matches?.[1],
85
- data: matches?.[2]
86
- };
250
+ const matches = (/* @__PURE__ */ new RegExp(/^data:(.*);base64,\s?(.*)$/)).exec(url);
251
+ return {
252
+ mimeType: matches?.[1],
253
+ data: matches?.[2]
254
+ };
87
255
  }
88
- const isHttpUrl = (url) => {
89
- return url.startsWith("https://") || url.startsWith("http://");
256
+ /**
257
+ * Returns `true` if `url` starts with `https://` or `http://` (case-sensitive, as in
258
+ * `String#startsWith`). Does not validate that the rest is a well-formed URL — only the scheme
259
+ * prefix. `blob:`, `data:`, and relative paths return `false`.
260
+ *
261
+ * @param url - String to test (often a user-provided or configured href).
262
+ *
263
+ * @example
264
+ * ```ts
265
+ * isHttpUrl('https://example.com/path'); // true
266
+ * isHttpUrl('http://localhost:3000'); // true
267
+ * ```
268
+ *
269
+ * @example
270
+ * ```ts
271
+ * isHttpUrl('//cdn.example.com'); // false — no explicit http(s) prefix
272
+ * isHttpUrl('/assets/logo.png'); // false — relative path
273
+ * ```
274
+ */
275
+ var isHttpUrl = (url) => {
276
+ return url.startsWith("https://") || url.startsWith("http://");
90
277
  };
91
- const isBase64Image = (str) => {
92
- const { mimeType, data } = decodeDataUrl(str);
93
- return !!(data && mimeType?.startsWith("image/"));
278
+ /**
279
+ * Returns `true` when `str` is a data URL that {@link decodeDataUrl} can parse **and** the MIME
280
+ * type starts with `image/` (e.g. `image/png`, `image/jpeg`). Requires both a non-empty Base64
281
+ * payload and an image MIME — arbitrary `data:text/plain;base64,...` is `false`.
282
+ *
283
+ * @param str - Candidate string (often `img.src` or API payload).
284
+ *
285
+ * @example
286
+ * ```ts
287
+ * isBase64Image('data:image/png;base64,iVBORw0KGgo='); // true
288
+ * ```
289
+ *
290
+ * @example
291
+ * ```ts
292
+ * isBase64Image('https://example.com/x.png'); // false
293
+ * isBase64Image('data:text/plain;base64,SGk='); // false
294
+ * ```
295
+ */
296
+ var isBase64Image = (str) => {
297
+ const { mimeType, data } = decodeDataUrl(str);
298
+ return !!(data && mimeType?.startsWith("image/"));
94
299
  };
95
- export {
96
- blobToBase64,
97
- blobToUrl,
98
- decodeDataUrl,
99
- fileToBlob,
100
- imageToBlob,
101
- isBase64Image,
102
- isHttpUrl,
103
- renderImage,
104
- rotateImage
105
- };
106
- //# sourceMappingURL=media.js.map
300
+ //#endregion
301
+ export { blobToBase64, blobToUrl, decodeDataUrl, fileToBlob, imageToBlob, isBase64Image, isHttpUrl, renderImage, rotateImage };
302
+
303
+ //# sourceMappingURL=media.js.map
package/media.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"media.js","sources":["../src/media.ts"],"sourcesContent":["import { degToRad } from 'yummies/math';\n\nexport function blobToBase64(blob: Blob): Promise<string> {\n return new Promise((resolve, reject) => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result as string);\n reader.onerror = reject;\n reader.readAsDataURL(blob);\n });\n}\n\nexport const blobToUrl = (urlOrBlob: string | Blob) =>\n urlOrBlob instanceof Blob ? URL.createObjectURL(urlOrBlob) : urlOrBlob;\n\nexport const fileToBlob = (file: File) => {\n return new Blob([file], { type: file.type });\n};\n\nexport const imageToBlob = (\n imageElement: HTMLImageElement,\n mimeType: string = 'image/png',\n) => {\n const canvas = document.createElement('canvas');\n\n canvas.width = imageElement.naturalWidth || 300;\n canvas.height = imageElement.naturalHeight || 300;\n\n canvas.getContext('2d')!.drawImage(imageElement, 0, 0);\n\n const dataUri = canvas.toDataURL(mimeType, 1);\n const base64data = dataUri.split(',')[1];\n const base64MimeType = dataUri.split(';')[0].slice(5);\n\n const bytes = globalThis.atob(base64data);\n const buf = new ArrayBuffer(bytes.length);\n const array = new Uint8Array(buf);\n\n for (let index = 0; index < bytes.length; index++) {\n array[index] = bytes.charCodeAt(index);\n }\n\n const blob = new Blob([array], { type: base64MimeType });\n\n return blob;\n};\n\n/**\n * Loads and renders an image using `Image`.\n *\n * @returns {Promise<HTMLImageElement>}\n */\nexport const renderImage = (urlOrBlob: Blob | string) =>\n new Promise<HTMLImageElement>((resolve, reject) => {\n const image = new Image();\n image.src = blobToUrl(urlOrBlob);\n image.onload = () => resolve(image);\n image.onerror = () => reject();\n });\n\nfunction cropImageFromCanvas(context: CanvasRenderingContext2D) {\n const canvas = context.canvas;\n let w = canvas.width;\n let h = canvas.height;\n const pix: { x: number[]; y: number[] } = { x: [], y: [] };\n const imageData = context.getImageData(0, 0, canvas.width, canvas.height);\n let x: number;\n let y: number;\n let index: number;\n\n for (y = 0; y < h; y++) {\n for (x = 0; x < w; x++) {\n index = (y * w + x) * 4;\n if (imageData.data[index + 3] > 0) {\n pix.x.push(x);\n pix.y.push(y);\n }\n }\n }\n pix.x.sort((a, b) => a - b);\n pix.y.sort((a, b) => a - b);\n const n = pix.x.length - 1;\n\n w = 1 + pix.x[n] - pix.x[0];\n h = 1 + pix.y[n] - pix.y[0];\n const cut = context.getImageData(pix.x[0], pix.y[0], w, h);\n\n canvas.width = w;\n canvas.height = h;\n context.putImageData(cut, 0, 0);\n return canvas;\n}\n\n// TODO: ломает iphone с огромными изображениями\nexport const rotateImage = (image: HTMLImageElement, angle: number) => {\n const maxSize = Math.max(image.width, image.height);\n const canvas = document.createElement('canvas');\n canvas.width = maxSize;\n canvas.height = maxSize;\n const context = canvas.getContext('2d')!;\n context.save();\n context.translate(canvas.width / 2, canvas.height / 2);\n context.rotate(degToRad(angle));\n context.drawImage(image, -image.width / 2, -image.height / 2);\n context.restore();\n cropImageFromCanvas(context);\n return renderImage(canvas.toDataURL('image/png'));\n};\n\ninterface DecodedDataUrl {\n mimeType?: string;\n data?: string;\n}\n\n/*\n * Returning object which contains base64 data and mime type of passed data url string.\n * */\nexport function decodeDataUrl(url: string): DecodedDataUrl {\n const regex = /^data:(.*);base64,\\s?(.*)$/;\n const matches = new RegExp(regex).exec(url);\n\n return {\n mimeType: matches?.[1],\n data: matches?.[2],\n };\n}\n\nexport const isHttpUrl = (url: string): boolean => {\n return url.startsWith('https://') || url.startsWith('http://');\n};\n\nexport const isBase64Image = (str: string): boolean => {\n const { mimeType, data } = decodeDataUrl(str);\n return !!(data && mimeType?.startsWith('image/'));\n};\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"media.js","names":[],"sources":["../src/media.ts"],"sourcesContent":["/**\n * ---header-docs-section---\n * # yummies/media\n *\n * ## Description\n *\n * Binary and media helpers: Base64 data URLs, object URLs, image orientation fixes, and related\n * browser APIs wrapped in promises. Centralizes `FileReader`/`URL.createObjectURL` usage so image\n * pipelines and uploads stay consistent and easier to test or mock.\n *\n * ## Usage\n *\n * ```ts\n * import { blobToBase64 } from \"yummies/media\";\n * ```\n */\n\nimport { degToRad } from 'yummies/math';\n\n/**\n * Reads a {@link Blob} as a **data URL** string (`data:<mime>;base64,...`) using {@link FileReader#readAsDataURL}.\n *\n * Useful for previewing uploads, embedding small assets inline, or serializing binary for APIs that\n * expect Base64-in-JSON. The result includes the MIME prefix, not raw Base64 alone — use\n * {@link decodeDataUrl} if you need the payload and type separately.\n *\n * @param blob - Any `Blob` or `File` (files are blobs).\n * @returns Resolves to the data URL string; rejects if reading fails.\n *\n * @example\n * ```ts\n * const dataUrl = await blobToBase64(file);\n * previewImg.src = dataUrl;\n * ```\n *\n * @example\n * ```ts\n * const fromFetch = await fetch('/api/export').then((r) => r.blob());\n * const dataUrl = await blobToBase64(fromFetch);\n * ```\n */\nexport function blobToBase64(blob: Blob): Promise<string> {\n return new Promise((resolve, reject) => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result as string);\n reader.onerror = reject;\n reader.readAsDataURL(blob);\n });\n}\n\n/**\n * If `urlOrBlob` is already a string, returns it unchanged. If it is a {@link Blob}, returns\n * `URL.createObjectURL(blob)` — a short-lived `blob:` URL valid in this document until\n * {@link URL.revokeObjectURL} is called.\n *\n * Pair with {@link renderImage} or `<img src>` without re-fetching binary data. **Remember to\n * `revokeObjectURL`** when the URL is no longer needed to avoid retaining blob memory.\n *\n * @param urlOrBlob - Remote/http(s) URL string, data URL, or a `Blob` / `File`.\n * @returns A string suitable for `HTMLImageElement.src` or similar.\n *\n * @example\n * ```ts\n * const src = blobToUrl(uploadedFile);\n * img.src = src;\n * // when done:\n * URL.revokeObjectURL(src);\n * ```\n *\n * @example\n * ```ts\n * blobToUrl('https://cdn.example.com/logo.png'); // passed through as-is\n * ```\n */\nexport const blobToUrl = (urlOrBlob: string | Blob) =>\n urlOrBlob instanceof Blob ? URL.createObjectURL(urlOrBlob) : urlOrBlob;\n\n/**\n * Creates a new {@link Blob} from a {@link File}, copying the bytes and keeping `file.type` as\n * the blob’s MIME type. Handy when an API accepts `Blob` but you only have `File`, or you want a\n * plain blob without the `File` name/lastModified metadata.\n *\n * @param file - Source file from an `<input type=\"file\">` or drag-and-drop.\n * @returns A `Blob` with the same content and `type` as the file.\n *\n * @example\n * ```ts\n * const blob = fileToBlob(fileFromInput);\n * await uploadEndpoint(blob);\n * ```\n *\n * @example\n * ```ts\n * const blob = fileToBlob(imageFile);\n * const url = URL.createObjectURL(blob);\n * ```\n */\nexport const fileToBlob = (file: File) => {\n return new Blob([file], { type: file.type });\n};\n\n/**\n * Draws an {@link HTMLImageElement} onto an offscreen canvas, then builds a {@link Blob} from the\n * raster (via `canvas.toDataURL` + binary decode). Dimensions use `naturalWidth` / `naturalHeight`,\n * falling back to `300×300` if those are zero (e.g. not yet decoded).\n *\n * **CORS:** if the image is cross-origin without proper CORS headers, the canvas may be\n * *tainted* and `toDataURL` can throw — same browser rules as any canvas export.\n *\n * @param imageElement - Loaded image (`complete` / decoded recommended).\n * @param mimeType - Output MIME type, e.g. `'image/png'` (default) or `'image/jpeg'`. Encoder support is browser-dependent.\n * @returns Encoded image as a `Blob` with a matching `type` when possible.\n *\n * @example\n * ```ts\n * const img = await renderImage('/photo.jpg');\n * const jpegBlob = imageToBlob(img, 'image/jpeg');\n * ```\n *\n * @example\n * ```ts\n * const pngBlob = imageToBlob(cachedImg); // default image/png\n * ```\n */\nexport const imageToBlob = (\n imageElement: HTMLImageElement,\n mimeType: string = 'image/png',\n) => {\n const canvas = document.createElement('canvas');\n\n canvas.width = imageElement.naturalWidth || 300;\n canvas.height = imageElement.naturalHeight || 300;\n\n canvas.getContext('2d')!.drawImage(imageElement, 0, 0);\n\n const dataUri = canvas.toDataURL(mimeType, 1);\n const base64data = dataUri.split(',')[1];\n const base64MimeType = dataUri.split(';')[0].slice(5);\n\n const bytes = globalThis.atob(base64data);\n const buf = new ArrayBuffer(bytes.length);\n const array = new Uint8Array(buf);\n\n for (let index = 0; index < bytes.length; index++) {\n array[index] = bytes.charCodeAt(index);\n }\n\n const blob = new Blob([array], { type: base64MimeType });\n\n return blob;\n};\n\n/**\n * Loads a resource into a new `HTMLImageElement`: `src` is set via {@link blobToUrl} (so blobs get\n * object URLs, strings are used directly). Resolves on `load` with the same element; rejects on\n * `error` (e.g. bad URL, network failure, corrupt image) **with no rejection value**.\n *\n * Does not add the node to the DOM — use the returned element for canvas, measuring, or\n * {@link imageToBlob}.\n *\n * @param urlOrBlob - Remote URL, data URL, or `Blob` / `File`.\n * @returns Promise that fulfills with the loaded `HTMLImageElement`.\n *\n * @example\n * ```ts\n * const img = await renderImage('https://example.com/pic.png');\n * document.body.appendChild(img);\n * ```\n *\n * @example\n * ```ts\n * const img = await renderImage(pickedFile);\n * const blob = imageToBlob(img, 'image/webp');\n * ```\n */\nexport const renderImage = (urlOrBlob: Blob | string) =>\n new Promise<HTMLImageElement>((resolve, reject) => {\n const image = new Image();\n image.src = blobToUrl(urlOrBlob);\n image.onload = () => resolve(image);\n image.onerror = () => reject();\n });\n\nfunction cropImageFromCanvas(context: CanvasRenderingContext2D) {\n const canvas = context.canvas;\n let w = canvas.width;\n let h = canvas.height;\n const pix: { x: number[]; y: number[] } = { x: [], y: [] };\n const imageData = context.getImageData(0, 0, canvas.width, canvas.height);\n let x: number;\n let y: number;\n let index: number;\n\n for (y = 0; y < h; y++) {\n for (x = 0; x < w; x++) {\n index = (y * w + x) * 4;\n if (imageData.data[index + 3] > 0) {\n pix.x.push(x);\n pix.y.push(y);\n }\n }\n }\n pix.x.sort((a, b) => a - b);\n pix.y.sort((a, b) => a - b);\n const n = pix.x.length - 1;\n\n w = 1 + pix.x[n] - pix.x[0];\n h = 1 + pix.y[n] - pix.y[0];\n const cut = context.getImageData(pix.x[0], pix.y[0], w, h);\n\n canvas.width = w;\n canvas.height = h;\n context.putImageData(cut, 0, 0);\n return canvas;\n}\n\n// TODO: ломает iphone с огромными изображениями\n/**\n * Rotates `image` around its center on a square canvas (side = max(width, height)), then **crops**\n * transparent margins by trimming to the bounding box of pixels with non-zero alpha.\n * Returns a **new** loaded `HTMLImageElement` (PNG data URL under the hood via {@link renderImage}).\n *\n * `angle` is in **degrees**; converted with {@link degToRad} from `yummies/math`.\n *\n * Very large sources can stress memory on some mobile browsers (known iPhone issues — see TODO in source).\n *\n * @param image - Source image (should be decoded; uses `width` / `height`).\n * @param angle - Rotation in degrees (e.g. `90` for quarter turn).\n * @returns Promise of a new `HTMLImageElement` showing the rotated, cropped result.\n *\n * @example\n * ```ts\n * const upright = await rotateImage(landscapeImg, 90);\n * ```\n *\n * @example\n * ```ts\n * const fixed = await rotateImage(await renderImage(file), -15);\n * ```\n */\nexport const rotateImage = (image: HTMLImageElement, angle: number) => {\n const maxSize = Math.max(image.width, image.height);\n const canvas = document.createElement('canvas');\n canvas.width = maxSize;\n canvas.height = maxSize;\n const context = canvas.getContext('2d')!;\n context.save();\n context.translate(canvas.width / 2, canvas.height / 2);\n context.rotate(degToRad(angle));\n context.drawImage(image, -image.width / 2, -image.height / 2);\n context.restore();\n cropImageFromCanvas(context);\n return renderImage(canvas.toDataURL('image/png'));\n};\n\ninterface DecodedDataUrl {\n mimeType?: string;\n data?: string;\n}\n\n/**\n * Parses a `data:` URL of the form `data:<mime>;base64,<payload>` into its MIME type and raw\n * Base64 body (without the `data:...;base64,` prefix). Non-matching strings yield `undefined`\n * fields in the result object.\n *\n * @param url - Full data URL string.\n * @returns `{ mimeType, data }` — both optional when the regex does not match.\n *\n * @example\n * ```ts\n * const { mimeType, data } = decodeDataUrl('data:image/png;base64,iVBORw0KGgo=');\n * // mimeType === 'image/png', data === 'iVBORw0KGgo='\n * ```\n *\n * @example\n * ```ts\n * decodeDataUrl('not-a-data-url'); // { mimeType: undefined, data: undefined }\n * ```\n */\nexport function decodeDataUrl(url: string): DecodedDataUrl {\n const regex = /^data:(.*);base64,\\s?(.*)$/;\n const matches = new RegExp(regex).exec(url);\n\n return {\n mimeType: matches?.[1],\n data: matches?.[2],\n };\n}\n\n/**\n * Returns `true` if `url` starts with `https://` or `http://` (case-sensitive, as in\n * `String#startsWith`). Does not validate that the rest is a well-formed URL — only the scheme\n * prefix. `blob:`, `data:`, and relative paths return `false`.\n *\n * @param url - String to test (often a user-provided or configured href).\n *\n * @example\n * ```ts\n * isHttpUrl('https://example.com/path'); // true\n * isHttpUrl('http://localhost:3000'); // true\n * ```\n *\n * @example\n * ```ts\n * isHttpUrl('//cdn.example.com'); // false — no explicit http(s) prefix\n * isHttpUrl('/assets/logo.png'); // false — relative path\n * ```\n */\nexport const isHttpUrl = (url: string): boolean => {\n return url.startsWith('https://') || url.startsWith('http://');\n};\n\n/**\n * Returns `true` when `str` is a data URL that {@link decodeDataUrl} can parse **and** the MIME\n * type starts with `image/` (e.g. `image/png`, `image/jpeg`). Requires both a non-empty Base64\n * payload and an image MIME — arbitrary `data:text/plain;base64,...` is `false`.\n *\n * @param str - Candidate string (often `img.src` or API payload).\n *\n * @example\n * ```ts\n * isBase64Image('data:image/png;base64,iVBORw0KGgo='); // true\n * ```\n *\n * @example\n * ```ts\n * isBase64Image('https://example.com/x.png'); // false\n * isBase64Image('data:text/plain;base64,SGk='); // false\n * ```\n */\nexport const isBase64Image = (str: string): boolean => {\n const { mimeType, data } = decodeDataUrl(str);\n return !!(data && mimeType?.startsWith('image/'));\n};\n"],"mappings":"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"}