@prose-reader/shared 1.303.0 → 1.304.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.
package/dist/index.js CHANGED
@@ -1,228 +1,134 @@
1
- const y = (e, t) => e === t ? !0 : e.length !== t.length ? !1 : e.every((n, o) => n === t[o]), f = (e) => e.split(/[#?]/)[0]?.split(".").pop()?.trim() || "";
2
- function d(e) {
3
- const t = [];
4
- if (e.length === 0)
5
- return "";
6
- if (typeof e[0] != "string")
7
- throw new TypeError(`Url must be a string. Received ${e[0]}`);
8
- e[0].match(/^[^/:]+:\/*$/) && e.length > 1 && (e[0] = e.shift() + e[0]), e[0].match(/^file:\/\/\//) ? e[0] = e[0].replace(/^([^/:]+):\/*/, "$1:///") : e[0] = e[0].replace(/^([^/:]+):\/*/, "$1://");
9
- for (let r = 0; r < e.length; r++) {
10
- let l = e[r];
11
- if (typeof l != "string")
12
- throw new TypeError(`Url must be a string. Received ${l}`);
13
- l !== "" && (r > 0 && (l = l.replace(/^[/]+/, "")), r < e.length - 1 ? l = l.replace(/[/]+$/, "") : l = l.replace(/[/]+$/, "/"), t.push(l));
14
- }
15
- let n = t.join("/");
16
- n = n.replace(/\/(\?|&|#[^!])/g, "$1");
17
- const o = n.split("?");
18
- return n = o.shift() + (o.length > 0 ? "?" : "") + o.join("&"), n;
1
+ //#region src/array.ts
2
+ var e = (e, t) => e === t ? !0 : e.length === t.length ? e.every((e, n) => e === t[n]) : !1, t = (e) => e.split(/[#?]/)[0]?.split(".").pop()?.trim() || "";
3
+ function n(e) {
4
+ let t = [];
5
+ if (e.length === 0) return "";
6
+ if (typeof e[0] != "string") throw TypeError(`Url must be a string. Received ${e[0]}`);
7
+ e[0].match(/^[^/:]+:\/*$/) && e.length > 1 && (e[0] = e.shift() + e[0]), e[0].match(/^file:\/\/\//) ? e[0] = e[0].replace(/^([^/:]+):\/*/, "$1:///") : e[0] = e[0].replace(/^([^/:]+):\/*/, "$1://");
8
+ for (let n = 0; n < e.length; n++) {
9
+ let r = e[n];
10
+ if (typeof r != "string") throw TypeError(`Url must be a string. Received ${r}`);
11
+ r !== "" && (n > 0 && (r = r.replace(/^[/]+/, "")), r = n < e.length - 1 ? r.replace(/[/]+$/, "") : r.replace(/[/]+$/, "/"), t.push(r));
12
+ }
13
+ let n = t.join("/");
14
+ n = n.replace(/\/(\?|&|#[^!])/g, "$1");
15
+ let r = n.split("?");
16
+ return n = r.shift() + (r.length > 0 ? "?" : "") + r.join("&"), n;
19
17
  }
20
- function R(...e) {
21
- const t = Array.from(Array.isArray(e[0]) ? e[0] : e);
22
- return d(t);
18
+ function r(...e) {
19
+ return n(Array.from(Array.isArray(e[0]) ? e[0] : e));
23
20
  }
24
- function $(e) {
25
- const t = e.split("/");
26
- return t.pop(), t.join("/");
21
+ function i(e) {
22
+ let t = e.split("/");
23
+ return t.pop(), t.join("/");
27
24
  }
28
- const h = (e) => {
29
- switch (f(e)) {
30
- case "png":
31
- return "image/png";
32
- case "jpg":
33
- return "image/jpg";
34
- case "jpeg":
35
- return "image/jpeg";
36
- case "txt":
37
- return "text/plain";
38
- case "webp":
39
- return "image/webp";
40
- case "xhtml":
41
- return "application/xhtml+xml";
42
- case "mp3":
43
- return "audio/mpeg";
44
- case "m4a":
45
- case "m4b":
46
- return "audio/mp4";
47
- case "aac":
48
- return "audio/aac";
49
- case "ogg":
50
- case "oga":
51
- return "audio/ogg";
52
- case "wav":
53
- return "audio/wav";
54
- case "flac":
55
- return "audio/flac";
56
- case "opus":
57
- return "audio/opus";
58
- }
59
- }, x = ({
60
- mimeType: e,
61
- uri: t
62
- }) => (e ?? h(t ?? ""))?.startsWith("application/xhtml+xml"), _ = (e) => e.startsWith("image/") || e.startsWith("audio/") || e.startsWith("video/"), C = (e) => {
63
- if (!e.length) return;
64
- const t = e.indexOf(";");
65
- return t >= 0 ? e.substring(0, t) : e;
66
- }, v = (e) => {
67
- const t = e.pageSpreadLeft === !0, n = e.pageSpreadRight === !0;
68
- if (t !== n)
69
- return t ? "left" : "right";
70
- }, F = (e, t, n) => {
71
- if (Object.is(e, t))
72
- return !0;
73
- if (typeof e != "object" || e === null || typeof t != "object" || t === null)
74
- return !1;
75
- const o = Object.keys(e), r = Object.keys(t);
76
- if (o.length !== r.length)
77
- return !1;
78
- const l = n?.customEqual ?? Object.is, c = e, s = t;
79
- for (const a of o)
80
- if (!Object.hasOwn(s, a) || !l(c[a], s[a]))
81
- return !1;
82
- return !0;
83
- }, T = (e, t) => e.reduce(
84
- (n, o) => {
85
- const r = t(o);
86
- return n[r] || (n[r] = []), n[r].push(o), n;
87
- },
88
- {}
89
- );
90
- function M(e, t) {
91
- const n = { ...e };
92
- for (const o in t)
93
- if (Object.hasOwn(t, o)) {
94
- const r = t[o];
95
- (r !== void 0 || !(o in e)) && (n[o] = r);
96
- }
97
- return n;
25
+ //#endregion
26
+ //#region src/contentType.ts
27
+ var a = (e) => {
28
+ switch (t(e)) {
29
+ case "png": return "image/png";
30
+ case "jpg": return "image/jpg";
31
+ case "jpeg": return "image/jpeg";
32
+ case "txt": return "text/plain";
33
+ case "webp": return "image/webp";
34
+ case "xhtml": return "application/xhtml+xml";
35
+ case "mp3": return "audio/mpeg";
36
+ case "m4a":
37
+ case "m4b": return "audio/mp4";
38
+ case "aac": return "audio/aac";
39
+ case "ogg":
40
+ case "oga": return "audio/ogg";
41
+ case "wav": return "audio/wav";
42
+ case "flac": return "audio/flac";
43
+ case "opus": return "audio/opus";
44
+ }
45
+ }, o = ({ mimeType: e, uri: t }) => (e ?? a(t ?? ""))?.startsWith("application/xhtml+xml"), s = (e) => e.startsWith("image/") || e.startsWith("audio/") || e.startsWith("video/"), c = (e) => {
46
+ if (!e.length) return;
47
+ let t = e.indexOf(";");
48
+ return t >= 0 ? e.substring(0, t) : e;
49
+ }, l = (e) => {
50
+ let t = e.pageSpreadLeft === !0;
51
+ if (t !== (e.pageSpreadRight === !0)) return t ? "left" : "right";
52
+ }, u = (e, t, n) => {
53
+ if (Object.is(e, t)) return !0;
54
+ if (typeof e != "object" || !e || typeof t != "object" || !t) return !1;
55
+ let r = Object.keys(e), i = Object.keys(t);
56
+ if (r.length !== i.length) return !1;
57
+ let a = n?.customEqual ?? Object.is, o = e, s = t;
58
+ for (let e of r) if (!Object.hasOwn(s, e) || !a(o[e], s[e])) return !1;
59
+ return !0;
60
+ }, d = (e, t) => e.reduce((e, n) => {
61
+ let r = t(n);
62
+ return e[r] || (e[r] = []), e[r].push(n), e;
63
+ }, {});
64
+ function f(e, t) {
65
+ let n = { ...e };
66
+ for (let r in t) if (Object.hasOwn(t, r)) {
67
+ let i = t[r];
68
+ (i !== void 0 || !(r in e)) && (n[r] = i);
69
+ }
70
+ return n;
98
71
  }
99
- const m = () => {
100
- if (!(typeof window > "u"))
101
- return window;
72
+ //#endregion
73
+ //#region src/report.ts
74
+ var p = () => {
75
+ if (!(typeof window > "u")) return window;
102
76
  };
103
- function b() {
104
- const e = m()?.__PROSE_READER_DEBUG;
105
- return e === !0 || e === "true";
77
+ function m() {
78
+ let e = p()?.__PROSE_READER_DEBUG;
79
+ return e === !0 || e === "true";
106
80
  }
107
- const E = (e, t) => typeof e == "boolean" ? {
108
- enabled: e,
109
- options: t
81
+ var h = (e, t) => typeof e == "boolean" ? {
82
+ enabled: e,
83
+ options: t
110
84
  } : {
111
- enabled: e?.enabled,
112
- options: e ?? t
113
- }, w = (e) => e.at(-1)?.color, u = (e) => {
114
- if (!e.length)
115
- return;
116
- if (!e.some((o) => o.color))
117
- return [e.map((o) => o.label).join(" ")];
118
- let t = "";
119
- const n = [];
120
- for (const o of e)
121
- t += `%c${t ? ` ${o.label}` : o.label}`, n.push(
122
- o.color ? `color: ${o.color}` : ""
123
- );
124
- return [t, ...n];
125
- }, p = (e = [], t = b()) => {
126
- let n = t;
127
- const o = {
128
- enable: (c) => {
129
- l(c);
130
- },
131
- namespace: (c, s, a) => {
132
- const i = E(
133
- s,
134
- a
135
- ), g = i.options?.color ?? w(e);
136
- return p(
137
- [
138
- ...e,
139
- {
140
- label: `[${c}]`,
141
- color: g
142
- }
143
- ],
144
- i.enabled ?? n
145
- );
146
- },
147
- isEnabled: () => n,
148
- debug: () => {
149
- },
150
- info: () => {
151
- },
152
- log: () => {
153
- },
154
- groupCollapsed: () => {
155
- },
156
- groupEnd: () => {
157
- },
158
- getGroupArgs: (c) => {
159
- const s = u(e);
160
- if (!s)
161
- return [c];
162
- const [a, ...i] = s;
163
- return [`${a} ${c}`, ...i];
164
- },
165
- warn: () => {
166
- },
167
- error: () => {
168
- }
169
- }, r = (c) => {
170
- if (!c) {
171
- o.debug = () => {
172
- }, o.info = () => {
173
- }, o.log = () => {
174
- }, o.groupCollapsed = () => {
175
- }, o.groupEnd = () => {
176
- }, o.warn = () => {
177
- }, o.error = () => {
178
- };
179
- return;
180
- }
181
- const s = u(e) ?? [];
182
- o.debug = Function.prototype.bind.call(
183
- console.debug,
184
- console,
185
- ...s
186
- ), o.info = Function.prototype.bind.call(
187
- console.info,
188
- console,
189
- ...s
190
- ), o.log = Function.prototype.bind.call(
191
- console.log,
192
- console,
193
- ...s
194
- ), o.groupCollapsed = Function.prototype.bind.call(
195
- console.groupCollapsed,
196
- console
197
- ), o.groupEnd = Function.prototype.bind.call(console.groupEnd, console), o.warn = Function.prototype.bind.call(
198
- console.warn,
199
- console,
200
- ...s
201
- ), o.error = Function.prototype.bind.call(
202
- console.error,
203
- console,
204
- ...s
205
- );
206
- }, l = (c) => {
207
- n !== c && (n = c, r(n));
208
- };
209
- return r(n), o;
210
- }, O = p(), P = "prose-reader-resource-error", S = (e) => e.replaceAll("&", "&amp;").replaceAll('"', "&quot;").replaceAll("'", "&apos;").replaceAll("<", "&lt;").replaceAll(">", "&gt;");
211
- export {
212
- P as PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME,
213
- O as Report,
214
- y as arrayEqual,
215
- h as detectMimeTypeFromName,
216
- S as escapeXmlAttributeValue,
217
- v as getItemSpreadPosition,
218
- $ as getParentPath,
219
- f as getUrlExtension,
220
- T as groupBy,
221
- _ as isMediaContentMimeType,
222
- F as isShallowEqual,
223
- x as isXmlBasedMimeType,
224
- C as parseContentType,
225
- M as shallowMergeIfDefined,
226
- R as urlJoin
227
- };
228
- //# sourceMappingURL=index.js.map
85
+ enabled: e?.enabled,
86
+ options: e ?? t
87
+ }, g = (e) => e.at(-1)?.color, _ = (e) => {
88
+ if (!e.length) return;
89
+ if (!e.some((e) => e.color)) return [e.map((e) => e.label).join(" ")];
90
+ let t = "", n = [];
91
+ for (let r of e) t += `%c${t ? ` ${r.label}` : r.label}`, n.push(r.color ? `color: ${r.color}` : "");
92
+ return [t, ...n];
93
+ }, v = (e = [], t = m()) => {
94
+ let n = t, r = {
95
+ enable: (e) => {
96
+ a(e);
97
+ },
98
+ namespace: (t, r, i) => {
99
+ let a = h(r, i), o = a.options?.color ?? g(e);
100
+ return v([...e, {
101
+ label: `[${t}]`,
102
+ color: o
103
+ }], a.enabled ?? n);
104
+ },
105
+ isEnabled: () => n,
106
+ debug: () => {},
107
+ info: () => {},
108
+ log: () => {},
109
+ groupCollapsed: () => {},
110
+ groupEnd: () => {},
111
+ getGroupArgs: (t) => {
112
+ let n = _(e);
113
+ if (!n) return [t];
114
+ let [r, ...i] = n;
115
+ return [`${r} ${t}`, ...i];
116
+ },
117
+ warn: () => {},
118
+ error: () => {}
119
+ }, i = (t) => {
120
+ if (!t) {
121
+ r.debug = () => {}, r.info = () => {}, r.log = () => {}, r.groupCollapsed = () => {}, r.groupEnd = () => {}, r.warn = () => {}, r.error = () => {};
122
+ return;
123
+ }
124
+ let n = _(e) ?? [];
125
+ r.debug = Function.prototype.bind.call(console.debug, console, ...n), r.info = Function.prototype.bind.call(console.info, console, ...n), r.log = Function.prototype.bind.call(console.log, console, ...n), r.groupCollapsed = Function.prototype.bind.call(console.groupCollapsed, console), r.groupEnd = Function.prototype.bind.call(console.groupEnd, console), r.warn = Function.prototype.bind.call(console.warn, console, ...n), r.error = Function.prototype.bind.call(console.error, console, ...n);
126
+ }, a = (e) => {
127
+ n !== e && (n = e, i(n));
128
+ };
129
+ return i(n), r;
130
+ }, y = v(), b = "prose-reader-resource-error", x = (e) => e.replaceAll("&", "&amp;").replaceAll("\"", "&quot;").replaceAll("'", "&apos;").replaceAll("<", "&lt;").replaceAll(">", "&gt;");
131
+ //#endregion
132
+ export { b as PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME, y as Report, e as arrayEqual, a as detectMimeTypeFromName, x as escapeXmlAttributeValue, l as getItemSpreadPosition, i as getParentPath, t as getUrlExtension, d as groupBy, s as isMediaContentMimeType, u as isShallowEqual, o as isXmlBasedMimeType, c as parseContentType, f as shallowMergeIfDefined, r as urlJoin };
133
+
134
+ //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/array.ts","../src/url.ts","../src/contentType.ts","../src/manifest/utils.ts","../src/objects.ts","../src/report.ts","../src/resources.ts","../src/xml.ts"],"sourcesContent":["// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const arrayEqual = <A extends any[], B extends any[]>(a: A, b: B) => {\n // @ts-expect-error\n if (a === b) return true\n if (a.length !== b.length) return false\n\n return a.every((v, i) => v === b[i])\n}\n","export const getUrlExtension = (url: string) => {\n return url.split(/[#?]/)[0]?.split(`.`).pop()?.trim() || ``\n}\n\nfunction normalize(strArray: string[]) {\n const resultArray = []\n if (strArray.length === 0) {\n return \"\"\n }\n\n if (typeof strArray[0] !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${strArray[0]}`)\n }\n\n // If the first part is a plain protocol, we combine it with the next part.\n if (strArray[0].match(/^[^/:]+:\\/*$/) && strArray.length > 1) {\n strArray[0] = strArray.shift() + strArray[0]\n }\n\n // There must be two or three slashes in the file protocol, two slashes in anything else.\n if (strArray[0].match(/^file:\\/\\/\\//)) {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1:///\")\n } else {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1://\")\n }\n\n for (let i = 0; i < strArray.length; i++) {\n let component = strArray[i]\n\n if (typeof component !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${component}`)\n }\n\n if (component === \"\") {\n continue\n }\n\n if (i > 0) {\n // Removing the starting slashes for each component but the first.\n component = component.replace(/^[/]+/, \"\")\n }\n if (i < strArray.length - 1) {\n // Removing the ending slashes for each component but the last.\n component = component.replace(/[/]+$/, \"\")\n } else {\n // For the last component we will combine multiple slashes to a single one.\n component = component.replace(/[/]+$/, \"/\")\n }\n\n resultArray.push(component)\n }\n\n let str = resultArray.join(\"/\")\n // Each input component is now separated by a single slash except the possible first plain protocol part.\n\n // remove trailing slash before parameters or hash\n str = str.replace(/\\/(\\?|&|#[^!])/g, \"$1\")\n\n // replace ? in parameters with &\n const parts = str.split(\"?\")\n str = parts.shift() + (parts.length > 0 ? \"?\" : \"\") + parts.join(\"&\")\n\n return str\n}\n\nexport function urlJoin(...args: (string | string[])[]) {\n const parts = Array.from(Array.isArray(args[0]) ? args[0] : args) as string[]\n\n return normalize(parts)\n}\n\nexport function getParentPath(uri: string): string {\n const parts = uri.split(\"/\")\n\n parts.pop() // Remove the last segment\n\n return parts.join(\"/\")\n}\n","import { getUrlExtension } from \"./url\"\n\nexport const detectMimeTypeFromName = (name: string) => {\n const extension = getUrlExtension(name)\n\n switch (extension) {\n case `png`:\n return `image/png`\n case `jpg`:\n return `image/jpg`\n case `jpeg`:\n return `image/jpeg`\n case `txt`:\n return `text/plain`\n case `webp`:\n return `image/webp`\n case `xhtml`:\n return `application/xhtml+xml`\n case `mp3`:\n return `audio/mpeg`\n case `m4a`:\n case `m4b`:\n return `audio/mp4`\n case `aac`:\n return `audio/aac`\n case `ogg`:\n case `oga`:\n return `audio/ogg`\n case `wav`:\n return `audio/wav`\n case `flac`:\n return `audio/flac`\n case `opus`:\n return `audio/opus`\n }\n\n return undefined\n}\n\nexport const isXmlBasedMimeType = ({\n mimeType,\n uri,\n}: {\n uri?: string\n mimeType?: string\n}) => {\n const _mimeType = mimeType ?? detectMimeTypeFromName(uri ?? \"\")\n\n return _mimeType?.startsWith(`application/xhtml+xml`)\n}\n\n/**\n * Discrete media MIME types represent a single self-contained unit\n * (an image, an audio track, a video clip) as opposed to document\n * content that may need reflowable layout.\n */\nexport const isMediaContentMimeType = (mimeType: string) =>\n mimeType.startsWith(\"image/\") ||\n mimeType.startsWith(\"audio/\") ||\n mimeType.startsWith(\"video/\")\n\nexport const parseContentType = (str: string) => {\n if (!str.length) return undefined\n\n const cut = str.indexOf(`;`)\n\n return cut >= 0 ? str.substring(0, cut) : str\n}\n","import type { Manifest } from \"./Manifest\"\n\nexport type SpreadPosition = \"left\" | \"right\" | \"none\"\n\nexport const getItemSpreadPosition = (\n item: Manifest[`spineItems`][number],\n): Exclude<SpreadPosition, \"none\"> | undefined => {\n const hasPageSpreadLeft = item.pageSpreadLeft === true\n const hasPageSpreadRight = item.pageSpreadRight === true\n\n if (hasPageSpreadLeft === hasPageSpreadRight) return undefined\n\n return hasPageSpreadLeft ? `left` : `right`\n}\n","/**\n * Return true if `objectA` is shallow equal to `objectB`.\n *\n * - Primitives are compared with `Object.is` (so `NaN === NaN`, `+0 ≠ -0`).\n * - Objects and arrays are compared by their own enumerable string keys,\n * each value pair compared with `Object.is` (or `customEqual`).\n * - Two objects with the same keys but different prototypes are considered\n * equal — class identity is not checked.\n *\n * Pass `customEqual` to override the per-value comparator (e.g. to recurse\n * one level deeper, or to compare specific value types by content).\n */\nexport const isShallowEqual = <A = unknown, B = unknown>(\n objectA: A,\n objectB: B,\n options?: {\n customEqual?: (a: unknown, b: unknown) => boolean\n },\n): boolean => {\n if (Object.is(objectA, objectB)) {\n return true\n }\n if (\n typeof objectA !== \"object\" ||\n objectA === null ||\n typeof objectB !== \"object\" ||\n objectB === null\n ) {\n return false\n }\n\n const keysA = Object.keys(objectA)\n const keysB = Object.keys(objectB)\n\n if (keysA.length !== keysB.length) {\n return false\n }\n\n const isEqual = options?.customEqual ?? Object.is\n\n // Indexable cast after narrowing to non-null `object`: TypeScript doesn't\n // widen `object` itself, but every iterated key is verified as own via\n // `Object.keys` / `Object.hasOwn`.\n const a = objectA as Record<string, unknown>\n const b = objectB as Record<string, unknown>\n\n for (const key of keysA) {\n if (!Object.hasOwn(b, key) || !isEqual(a[key], b[key])) {\n return false\n }\n }\n\n return true\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const groupBy = <T, K extends keyof any>(\n list: T[],\n getKey: (item: T) => K,\n) =>\n list.reduce(\n (previous, currentItem) => {\n const group = getKey(currentItem)\n if (!previous[group]) previous[group] = []\n previous[group].push(currentItem)\n return previous\n },\n {} as Record<K, T[]>,\n )\n\nexport function shallowMergeIfDefined<T extends object, U extends object>(\n obj1: T,\n obj2: U,\n): T & U {\n const result = { ...obj1 } as T & U\n\n for (const key in obj2) {\n if (Object.hasOwn(obj2, key)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n const value = (obj2 as any)[key]\n if (value !== undefined || !(key in obj1)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n ;(result as any)[key] = value\n }\n }\n }\n\n return result\n}\n","/**\n * This report class needs to be fast because it is used in hot path. The disabled state\n * should not trigger any i/o and be no op operations.\n * The logging should keep the original stack trace.\n */\n\ndeclare global {\n interface Window {\n __PROSE_READER_DEBUG?: boolean | string\n }\n}\n\nconst getWindow = () => {\n if (typeof window === \"undefined\") {\n return undefined\n }\n\n return window\n}\n\nfunction isGlobalDebugEnabled() {\n const debug = getWindow()?.__PROSE_READER_DEBUG\n\n return debug === true || debug === \"true\"\n}\n\nexport type ReportOptions = {\n color?: string\n}\n\nexport type ReportNamespaceOptions = ReportOptions & {\n enabled?: boolean\n}\n\ntype ReportNamespace = {\n (namespace: string, enabled?: boolean, options?: ReportOptions): Report\n (namespace: string, options?: ReportNamespaceOptions): Report\n}\n\ntype ReportGroupArgs = [label: string, ...styles: string[]]\n\ntype NamespaceSegment = {\n label: string\n color?: string\n}\n\nexport type Report = {\n enable: (enabled: boolean) => void\n isEnabled: () => boolean\n namespace: ReportNamespace\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n debug: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n info: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n log: (...args: any[]) => void\n /** Use with getGroupArgs for namespace + color. Preserves call site. */\n groupCollapsed: (...args: unknown[]) => void\n groupEnd: () => void\n /** Returns [label, ...styles] to pass to groupCollapsed for namespace + color. */\n getGroupArgs: (title: string) => ReportGroupArgs\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n warn: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n error: (...args: any[]) => void\n}\n\nconst getNamespaceArgs = (\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n options?: ReportOptions,\n) => {\n if (typeof enabledOrOptions === \"boolean\") {\n return {\n enabled: enabledOrOptions,\n options,\n }\n }\n\n return {\n enabled: enabledOrOptions?.enabled,\n options: enabledOrOptions ?? options,\n }\n}\n\nconst getCurrentColor = (namespaceSegments: NamespaceSegment[]) =>\n namespaceSegments.at(-1)?.color\n\nconst getStyledNamespaceArgs = (\n namespaceSegments: NamespaceSegment[],\n): ReportGroupArgs | undefined => {\n if (!namespaceSegments.length) {\n return undefined\n }\n\n if (!namespaceSegments.some((segment) => segment.color)) {\n return [namespaceSegments.map((segment) => segment.label).join(\" \")]\n }\n\n let label = \"\"\n const styles: string[] = []\n\n for (const namespaceSegment of namespaceSegments) {\n label += `%c${label ? ` ${namespaceSegment.label}` : namespaceSegment.label}`\n styles.push(\n namespaceSegment.color ? `color: ${namespaceSegment.color}` : \"\",\n )\n }\n\n return [label, ...styles]\n}\n\nconst createReport = (\n namespaceSegments: NamespaceSegment[] = [],\n enabled = isGlobalDebugEnabled(),\n): Report => {\n let reportEnabled = enabled\n\n const report: Report = {\n enable: (enabled: boolean) => {\n setEnabled(enabled)\n },\n namespace: (\n _namespace: string,\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n namespaceOptions?: ReportOptions,\n ) => {\n const childNamespaceArgs = getNamespaceArgs(\n enabledOrOptions,\n namespaceOptions,\n )\n\n const childColor =\n childNamespaceArgs.options?.color ?? getCurrentColor(namespaceSegments)\n\n return createReport(\n [\n ...namespaceSegments,\n {\n label: `[${_namespace}]`,\n color: childColor,\n },\n ],\n childNamespaceArgs.enabled ?? reportEnabled,\n )\n },\n isEnabled: () => reportEnabled,\n debug: () => {},\n info: () => {},\n log: () => {},\n groupCollapsed: () => {},\n groupEnd: () => {},\n getGroupArgs: (title: string) => {\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments)\n\n if (!namespaceArgs) {\n return [title]\n }\n\n const [namespaceLabel, ...namespaceStyles] = namespaceArgs\n\n return [`${namespaceLabel} ${title}`, ...namespaceStyles]\n },\n warn: () => {},\n error: () => {},\n }\n\n // Keep direct prototype-bound console functions for enabled mode.\n // Wrapping console calls in closures changes call-site/stack behavior in devtools,\n // which makes runtime debugging harder to interpret.\n const applyEnabledState = (enabled: boolean) => {\n if (!enabled) {\n report.debug = () => {}\n report.info = () => {}\n report.log = () => {}\n report.groupCollapsed = () => {}\n report.groupEnd = () => {}\n report.warn = () => {}\n report.error = () => {}\n\n return\n }\n\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments) ?? []\n\n report.debug = Function.prototype.bind.call(\n console.debug,\n console,\n ...namespaceArgs,\n )\n report.info = Function.prototype.bind.call(\n console.info,\n console,\n ...namespaceArgs,\n )\n report.log = Function.prototype.bind.call(\n console.log,\n console,\n ...namespaceArgs,\n )\n report.groupCollapsed = Function.prototype.bind.call(\n console.groupCollapsed,\n console,\n )\n report.groupEnd = Function.prototype.bind.call(console.groupEnd, console)\n report.warn = Function.prototype.bind.call(\n console.warn,\n console,\n ...namespaceArgs,\n )\n report.error = Function.prototype.bind.call(\n console.error,\n console,\n ...namespaceArgs,\n )\n }\n\n const setEnabled = (enabled: boolean) => {\n if (reportEnabled === enabled) {\n return\n }\n\n reportEnabled = enabled\n applyEnabledState(reportEnabled)\n }\n\n applyEnabledState(reportEnabled)\n\n return report\n}\n\nexport const Report = createReport()\n","export const PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME =\n \"prose-reader-resource-error\"\n","export const escapeXmlAttributeValue = (value: string) =>\n value\n .replaceAll(`&`, `&amp;`)\n .replaceAll(`\"`, `&quot;`)\n .replaceAll(`'`, `&apos;`)\n .replaceAll(`<`, `&lt;`)\n .replaceAll(`>`, `&gt;`)\n"],"names":["arrayEqual","a","b","v","i","getUrlExtension","url","normalize","strArray","resultArray","component","str","parts","urlJoin","args","getParentPath","uri","detectMimeTypeFromName","name","isXmlBasedMimeType","mimeType","isMediaContentMimeType","parseContentType","cut","getItemSpreadPosition","item","hasPageSpreadLeft","hasPageSpreadRight","isShallowEqual","objectA","objectB","options","keysA","keysB","isEqual","key","groupBy","list","getKey","previous","currentItem","group","shallowMergeIfDefined","obj1","obj2","result","value","getWindow","isGlobalDebugEnabled","debug","getNamespaceArgs","enabledOrOptions","getCurrentColor","namespaceSegments","getStyledNamespaceArgs","segment","label","styles","namespaceSegment","createReport","enabled","reportEnabled","report","setEnabled","_namespace","namespaceOptions","childNamespaceArgs","childColor","title","namespaceArgs","namespaceLabel","namespaceStyles","applyEnabledState","Report","PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME","escapeXmlAttributeValue"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":[],"sources":["../src/array.ts","../src/url.ts","../src/contentType.ts","../src/manifest/utils.ts","../src/objects.ts","../src/report.ts","../src/resources.ts","../src/xml.ts"],"sourcesContent":["// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const arrayEqual = <A extends any[], B extends any[]>(a: A, b: B) => {\n // @ts-expect-error\n if (a === b) return true\n if (a.length !== b.length) return false\n\n return a.every((v, i) => v === b[i])\n}\n","export const getUrlExtension = (url: string) => {\n return url.split(/[#?]/)[0]?.split(`.`).pop()?.trim() || ``\n}\n\nfunction normalize(strArray: string[]) {\n const resultArray = []\n if (strArray.length === 0) {\n return \"\"\n }\n\n if (typeof strArray[0] !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${strArray[0]}`)\n }\n\n // If the first part is a plain protocol, we combine it with the next part.\n if (strArray[0].match(/^[^/:]+:\\/*$/) && strArray.length > 1) {\n strArray[0] = strArray.shift() + strArray[0]\n }\n\n // There must be two or three slashes in the file protocol, two slashes in anything else.\n if (strArray[0].match(/^file:\\/\\/\\//)) {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1:///\")\n } else {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1://\")\n }\n\n for (let i = 0; i < strArray.length; i++) {\n let component = strArray[i]\n\n if (typeof component !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${component}`)\n }\n\n if (component === \"\") {\n continue\n }\n\n if (i > 0) {\n // Removing the starting slashes for each component but the first.\n component = component.replace(/^[/]+/, \"\")\n }\n if (i < strArray.length - 1) {\n // Removing the ending slashes for each component but the last.\n component = component.replace(/[/]+$/, \"\")\n } else {\n // For the last component we will combine multiple slashes to a single one.\n component = component.replace(/[/]+$/, \"/\")\n }\n\n resultArray.push(component)\n }\n\n let str = resultArray.join(\"/\")\n // Each input component is now separated by a single slash except the possible first plain protocol part.\n\n // remove trailing slash before parameters or hash\n str = str.replace(/\\/(\\?|&|#[^!])/g, \"$1\")\n\n // replace ? in parameters with &\n const parts = str.split(\"?\")\n str = parts.shift() + (parts.length > 0 ? \"?\" : \"\") + parts.join(\"&\")\n\n return str\n}\n\nexport function urlJoin(...args: (string | string[])[]) {\n const parts = Array.from(Array.isArray(args[0]) ? args[0] : args) as string[]\n\n return normalize(parts)\n}\n\nexport function getParentPath(uri: string): string {\n const parts = uri.split(\"/\")\n\n parts.pop() // Remove the last segment\n\n return parts.join(\"/\")\n}\n","import { getUrlExtension } from \"./url\"\n\nexport const detectMimeTypeFromName = (name: string) => {\n const extension = getUrlExtension(name)\n\n switch (extension) {\n case `png`:\n return `image/png`\n case `jpg`:\n return `image/jpg`\n case `jpeg`:\n return `image/jpeg`\n case `txt`:\n return `text/plain`\n case `webp`:\n return `image/webp`\n case `xhtml`:\n return `application/xhtml+xml`\n case `mp3`:\n return `audio/mpeg`\n case `m4a`:\n case `m4b`:\n return `audio/mp4`\n case `aac`:\n return `audio/aac`\n case `ogg`:\n case `oga`:\n return `audio/ogg`\n case `wav`:\n return `audio/wav`\n case `flac`:\n return `audio/flac`\n case `opus`:\n return `audio/opus`\n }\n\n return undefined\n}\n\nexport const isXmlBasedMimeType = ({\n mimeType,\n uri,\n}: {\n uri?: string\n mimeType?: string\n}) => {\n const _mimeType = mimeType ?? detectMimeTypeFromName(uri ?? \"\")\n\n return _mimeType?.startsWith(`application/xhtml+xml`)\n}\n\n/**\n * Discrete media MIME types represent a single self-contained unit\n * (an image, an audio track, a video clip) as opposed to document\n * content that may need reflowable layout.\n */\nexport const isMediaContentMimeType = (mimeType: string) =>\n mimeType.startsWith(\"image/\") ||\n mimeType.startsWith(\"audio/\") ||\n mimeType.startsWith(\"video/\")\n\nexport const parseContentType = (str: string) => {\n if (!str.length) return undefined\n\n const cut = str.indexOf(`;`)\n\n return cut >= 0 ? str.substring(0, cut) : str\n}\n","import type { Manifest } from \"./Manifest\"\n\nexport type SpreadPosition = \"left\" | \"right\" | \"none\"\n\nexport const getItemSpreadPosition = (\n item: Manifest[`spineItems`][number],\n): Exclude<SpreadPosition, \"none\"> | undefined => {\n const hasPageSpreadLeft = item.pageSpreadLeft === true\n const hasPageSpreadRight = item.pageSpreadRight === true\n\n if (hasPageSpreadLeft === hasPageSpreadRight) return undefined\n\n return hasPageSpreadLeft ? `left` : `right`\n}\n","/**\n * Return true if `objectA` is shallow equal to `objectB`.\n *\n * - Primitives are compared with `Object.is` (so `NaN === NaN`, `+0 ≠ -0`).\n * - Objects and arrays are compared by their own enumerable string keys,\n * each value pair compared with `Object.is` (or `customEqual`).\n * - Two objects with the same keys but different prototypes are considered\n * equal — class identity is not checked.\n *\n * Pass `customEqual` to override the per-value comparator (e.g. to recurse\n * one level deeper, or to compare specific value types by content).\n */\nexport const isShallowEqual = <A = unknown, B = unknown>(\n objectA: A,\n objectB: B,\n options?: {\n customEqual?: (a: unknown, b: unknown) => boolean\n },\n): boolean => {\n if (Object.is(objectA, objectB)) {\n return true\n }\n if (\n typeof objectA !== \"object\" ||\n objectA === null ||\n typeof objectB !== \"object\" ||\n objectB === null\n ) {\n return false\n }\n\n const keysA = Object.keys(objectA)\n const keysB = Object.keys(objectB)\n\n if (keysA.length !== keysB.length) {\n return false\n }\n\n const isEqual = options?.customEqual ?? Object.is\n\n // Indexable cast after narrowing to non-null `object`: TypeScript doesn't\n // widen `object` itself, but every iterated key is verified as own via\n // `Object.keys` / `Object.hasOwn`.\n const a = objectA as Record<string, unknown>\n const b = objectB as Record<string, unknown>\n\n for (const key of keysA) {\n if (!Object.hasOwn(b, key) || !isEqual(a[key], b[key])) {\n return false\n }\n }\n\n return true\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const groupBy = <T, K extends keyof any>(\n list: T[],\n getKey: (item: T) => K,\n) =>\n list.reduce(\n (previous, currentItem) => {\n const group = getKey(currentItem)\n if (!previous[group]) previous[group] = []\n previous[group].push(currentItem)\n return previous\n },\n {} as Record<K, T[]>,\n )\n\nexport function shallowMergeIfDefined<T extends object, U extends object>(\n obj1: T,\n obj2: U,\n): T & U {\n const result = { ...obj1 } as T & U\n\n for (const key in obj2) {\n if (Object.hasOwn(obj2, key)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n const value = (obj2 as any)[key]\n if (value !== undefined || !(key in obj1)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n ;(result as any)[key] = value\n }\n }\n }\n\n return result\n}\n","/**\n * This report class needs to be fast because it is used in hot path. The disabled state\n * should not trigger any i/o and be no op operations.\n * The logging should keep the original stack trace.\n */\n\ndeclare global {\n interface Window {\n __PROSE_READER_DEBUG?: boolean | string\n }\n}\n\nconst getWindow = () => {\n if (typeof window === \"undefined\") {\n return undefined\n }\n\n return window\n}\n\nfunction isGlobalDebugEnabled() {\n const debug = getWindow()?.__PROSE_READER_DEBUG\n\n return debug === true || debug === \"true\"\n}\n\nexport type ReportOptions = {\n color?: string\n}\n\nexport type ReportNamespaceOptions = ReportOptions & {\n enabled?: boolean\n}\n\ntype ReportNamespace = {\n (namespace: string, enabled?: boolean, options?: ReportOptions): Report\n (namespace: string, options?: ReportNamespaceOptions): Report\n}\n\ntype ReportGroupArgs = [label: string, ...styles: string[]]\n\ntype NamespaceSegment = {\n label: string\n color?: string\n}\n\nexport type Report = {\n enable: (enabled: boolean) => void\n isEnabled: () => boolean\n namespace: ReportNamespace\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n debug: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n info: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n log: (...args: any[]) => void\n /** Use with getGroupArgs for namespace + color. Preserves call site. */\n groupCollapsed: (...args: unknown[]) => void\n groupEnd: () => void\n /** Returns [label, ...styles] to pass to groupCollapsed for namespace + color. */\n getGroupArgs: (title: string) => ReportGroupArgs\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n warn: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n error: (...args: any[]) => void\n}\n\nconst getNamespaceArgs = (\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n options?: ReportOptions,\n) => {\n if (typeof enabledOrOptions === \"boolean\") {\n return {\n enabled: enabledOrOptions,\n options,\n }\n }\n\n return {\n enabled: enabledOrOptions?.enabled,\n options: enabledOrOptions ?? options,\n }\n}\n\nconst getCurrentColor = (namespaceSegments: NamespaceSegment[]) =>\n namespaceSegments.at(-1)?.color\n\nconst getStyledNamespaceArgs = (\n namespaceSegments: NamespaceSegment[],\n): ReportGroupArgs | undefined => {\n if (!namespaceSegments.length) {\n return undefined\n }\n\n if (!namespaceSegments.some((segment) => segment.color)) {\n return [namespaceSegments.map((segment) => segment.label).join(\" \")]\n }\n\n let label = \"\"\n const styles: string[] = []\n\n for (const namespaceSegment of namespaceSegments) {\n label += `%c${label ? ` ${namespaceSegment.label}` : namespaceSegment.label}`\n styles.push(\n namespaceSegment.color ? `color: ${namespaceSegment.color}` : \"\",\n )\n }\n\n return [label, ...styles]\n}\n\nconst createReport = (\n namespaceSegments: NamespaceSegment[] = [],\n enabled = isGlobalDebugEnabled(),\n): Report => {\n let reportEnabled = enabled\n\n const report: Report = {\n enable: (enabled: boolean) => {\n setEnabled(enabled)\n },\n namespace: (\n _namespace: string,\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n namespaceOptions?: ReportOptions,\n ) => {\n const childNamespaceArgs = getNamespaceArgs(\n enabledOrOptions,\n namespaceOptions,\n )\n\n const childColor =\n childNamespaceArgs.options?.color ?? getCurrentColor(namespaceSegments)\n\n return createReport(\n [\n ...namespaceSegments,\n {\n label: `[${_namespace}]`,\n color: childColor,\n },\n ],\n childNamespaceArgs.enabled ?? reportEnabled,\n )\n },\n isEnabled: () => reportEnabled,\n debug: () => {},\n info: () => {},\n log: () => {},\n groupCollapsed: () => {},\n groupEnd: () => {},\n getGroupArgs: (title: string) => {\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments)\n\n if (!namespaceArgs) {\n return [title]\n }\n\n const [namespaceLabel, ...namespaceStyles] = namespaceArgs\n\n return [`${namespaceLabel} ${title}`, ...namespaceStyles]\n },\n warn: () => {},\n error: () => {},\n }\n\n // Keep direct prototype-bound console functions for enabled mode.\n // Wrapping console calls in closures changes call-site/stack behavior in devtools,\n // which makes runtime debugging harder to interpret.\n const applyEnabledState = (enabled: boolean) => {\n if (!enabled) {\n report.debug = () => {}\n report.info = () => {}\n report.log = () => {}\n report.groupCollapsed = () => {}\n report.groupEnd = () => {}\n report.warn = () => {}\n report.error = () => {}\n\n return\n }\n\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments) ?? []\n\n report.debug = Function.prototype.bind.call(\n console.debug,\n console,\n ...namespaceArgs,\n )\n report.info = Function.prototype.bind.call(\n console.info,\n console,\n ...namespaceArgs,\n )\n report.log = Function.prototype.bind.call(\n console.log,\n console,\n ...namespaceArgs,\n )\n report.groupCollapsed = Function.prototype.bind.call(\n console.groupCollapsed,\n console,\n )\n report.groupEnd = Function.prototype.bind.call(console.groupEnd, console)\n report.warn = Function.prototype.bind.call(\n console.warn,\n console,\n ...namespaceArgs,\n )\n report.error = Function.prototype.bind.call(\n console.error,\n console,\n ...namespaceArgs,\n )\n }\n\n const setEnabled = (enabled: boolean) => {\n if (reportEnabled === enabled) {\n return\n }\n\n reportEnabled = enabled\n applyEnabledState(reportEnabled)\n }\n\n applyEnabledState(reportEnabled)\n\n return report\n}\n\nexport const Report = createReport()\n","export const PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME =\n \"prose-reader-resource-error\"\n","export const escapeXmlAttributeValue = (value: string) =>\n value\n .replaceAll(`&`, `&amp;`)\n .replaceAll(`\"`, `&quot;`)\n .replaceAll(`'`, `&apos;`)\n .replaceAll(`<`, `&lt;`)\n .replaceAll(`>`, `&gt;`)\n"],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- (function(r,u){typeof exports=="object"&&typeof module<"u"?u(exports):typeof define=="function"&&define.amd?define(["exports"],u):(r=typeof globalThis<"u"?globalThis:r||self,u(r["prose-shared"]={}))})(this,(function(r){"use strict";const u=(e,n)=>e===n?!0:e.length!==n.length?!1:e.every((o,t)=>o===n[t]),d=e=>e.split(/[#?]/)[0]?.split(".").pop()?.trim()||"";function E(e){const n=[];if(e.length===0)return"";if(typeof e[0]!="string")throw new TypeError(`Url must be a string. Received ${e[0]}`);e[0].match(/^[^/:]+:\/*$/)&&e.length>1&&(e[0]=e.shift()+e[0]),e[0].match(/^file:\/\/\//)?e[0]=e[0].replace(/^([^/:]+):\/*/,"$1:///"):e[0]=e[0].replace(/^([^/:]+):\/*/,"$1://");for(let l=0;l<e.length;l++){let i=e[l];if(typeof i!="string")throw new TypeError(`Url must be a string. Received ${i}`);i!==""&&(l>0&&(i=i.replace(/^[/]+/,"")),l<e.length-1?i=i.replace(/[/]+$/,""):i=i.replace(/[/]+$/,"/"),n.push(i))}let o=n.join("/");o=o.replace(/\/(\?|&|#[^!])/g,"$1");const t=o.split("?");return o=t.shift()+(t.length>0?"?":"")+t.join("&"),o}function m(...e){const n=Array.from(Array.isArray(e[0])?e[0]:e);return E(n)}function b(e){const n=e.split("/");return n.pop(),n.join("/")}const f=e=>{switch(d(e)){case"png":return"image/png";case"jpg":return"image/jpg";case"jpeg":return"image/jpeg";case"txt":return"text/plain";case"webp":return"image/webp";case"xhtml":return"application/xhtml+xml";case"mp3":return"audio/mpeg";case"m4a":case"m4b":return"audio/mp4";case"aac":return"audio/aac";case"ogg":case"oga":return"audio/ogg";case"wav":return"audio/wav";case"flac":return"audio/flac";case"opus":return"audio/opus"}},y=({mimeType:e,uri:n})=>(e??f(n??""))?.startsWith("application/xhtml+xml"),R=e=>e.startsWith("image/")||e.startsWith("audio/")||e.startsWith("video/"),w=e=>{if(!e.length)return;const n=e.indexOf(";");return n>=0?e.substring(0,n):e},_=e=>{const n=e.pageSpreadLeft===!0,o=e.pageSpreadRight===!0;if(n!==o)return n?"left":"right"},T=(e,n,o)=>{if(Object.is(e,n))return!0;if(typeof e!="object"||e===null||typeof n!="object"||n===null)return!1;const t=Object.keys(e),l=Object.keys(n);if(t.length!==l.length)return!1;const i=o?.customEqual??Object.is,a=e,c=n;for(const s of t)if(!Object.hasOwn(c,s)||!i(a[s],c[s]))return!1;return!0},C=(e,n)=>e.reduce((o,t)=>{const l=n(t);return o[l]||(o[l]=[]),o[l].push(t),o},{});function M(e,n){const o={...e};for(const t in n)if(Object.hasOwn(n,t)){const l=n[t];(l!==void 0||!(t in e))&&(o[t]=l)}return o}const $=()=>{if(!(typeof window>"u"))return window};function S(){const e=$()?.__PROSE_READER_DEBUG;return e===!0||e==="true"}const P=(e,n)=>typeof e=="boolean"?{enabled:e,options:n}:{enabled:e?.enabled,options:e??n},v=e=>e.at(-1)?.color,g=e=>{if(!e.length)return;if(!e.some(t=>t.color))return[e.map(t=>t.label).join(" ")];let n="";const o=[];for(const t of e)n+=`%c${n?` ${t.label}`:t.label}`,o.push(t.color?`color: ${t.color}`:"");return[n,...o]},h=(e=[],n=S())=>{let o=n;const t={enable:a=>{i(a)},namespace:(a,c,s)=>{const p=P(c,s),N=p.options?.color??v(e);return h([...e,{label:`[${a}]`,color:N}],p.enabled??o)},isEnabled:()=>o,debug:()=>{},info:()=>{},log:()=>{},groupCollapsed:()=>{},groupEnd:()=>{},getGroupArgs:a=>{const c=g(e);if(!c)return[a];const[s,...p]=c;return[`${s} ${a}`,...p]},warn:()=>{},error:()=>{}},l=a=>{if(!a){t.debug=()=>{},t.info=()=>{},t.log=()=>{},t.groupCollapsed=()=>{},t.groupEnd=()=>{},t.warn=()=>{},t.error=()=>{};return}const c=g(e)??[];t.debug=Function.prototype.bind.call(console.debug,console,...c),t.info=Function.prototype.bind.call(console.info,console,...c),t.log=Function.prototype.bind.call(console.log,console,...c),t.groupCollapsed=Function.prototype.bind.call(console.groupCollapsed,console),t.groupEnd=Function.prototype.bind.call(console.groupEnd,console),t.warn=Function.prototype.bind.call(console.warn,console,...c),t.error=Function.prototype.bind.call(console.error,console,...c)},i=a=>{o!==a&&(o=a,l(o))};return l(o),t},O=h(),D="prose-reader-resource-error",F=e=>e.replaceAll("&","&amp;").replaceAll('"',"&quot;").replaceAll("'","&apos;").replaceAll("<","&lt;").replaceAll(">","&gt;");r.PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME=D,r.Report=O,r.arrayEqual=u,r.detectMimeTypeFromName=f,r.escapeXmlAttributeValue=F,r.getItemSpreadPosition=_,r.getParentPath=b,r.getUrlExtension=d,r.groupBy=C,r.isMediaContentMimeType=R,r.isShallowEqual=T,r.isXmlBasedMimeType=y,r.parseContentType=w,r.shallowMergeIfDefined=M,r.urlJoin=m,Object.defineProperty(r,Symbol.toStringTag,{value:"Module"})}));
2
- //# sourceMappingURL=index.umd.cjs.map
1
+ (function(e,t){typeof exports==`object`&&typeof module<`u`?t(exports):typeof define==`function`&&define.amd?define([`exports`],t):(e=typeof globalThis<`u`?globalThis:e||self,t(e[`prose-shared`]={}))})(this,function(e){Object.defineProperty(e,Symbol.toStringTag,{value:`Module`});var t=(e,t)=>e===t?!0:e.length===t.length?e.every((e,n)=>e===t[n]):!1,n=e=>e.split(/[#?]/)[0]?.split(`.`).pop()?.trim()||``;function r(e){let t=[];if(e.length===0)return``;if(typeof e[0]!=`string`)throw TypeError(`Url must be a string. Received ${e[0]}`);e[0].match(/^[^/:]+:\/*$/)&&e.length>1&&(e[0]=e.shift()+e[0]),e[0].match(/^file:\/\/\//)?e[0]=e[0].replace(/^([^/:]+):\/*/,`$1:///`):e[0]=e[0].replace(/^([^/:]+):\/*/,`$1://`);for(let n=0;n<e.length;n++){let r=e[n];if(typeof r!=`string`)throw TypeError(`Url must be a string. Received ${r}`);r!==``&&(n>0&&(r=r.replace(/^[/]+/,``)),r=n<e.length-1?r.replace(/[/]+$/,``):r.replace(/[/]+$/,`/`),t.push(r))}let n=t.join(`/`);n=n.replace(/\/(\?|&|#[^!])/g,`$1`);let r=n.split(`?`);return n=r.shift()+(r.length>0?`?`:``)+r.join(`&`),n}function i(...e){return r(Array.from(Array.isArray(e[0])?e[0]:e))}function a(e){let t=e.split(`/`);return t.pop(),t.join(`/`)}var o=e=>{switch(n(e)){case`png`:return`image/png`;case`jpg`:return`image/jpg`;case`jpeg`:return`image/jpeg`;case`txt`:return`text/plain`;case`webp`:return`image/webp`;case`xhtml`:return`application/xhtml+xml`;case`mp3`:return`audio/mpeg`;case`m4a`:case`m4b`:return`audio/mp4`;case`aac`:return`audio/aac`;case`ogg`:case`oga`:return`audio/ogg`;case`wav`:return`audio/wav`;case`flac`:return`audio/flac`;case`opus`:return`audio/opus`}},s=({mimeType:e,uri:t})=>(e??o(t??``))?.startsWith(`application/xhtml+xml`),c=e=>e.startsWith(`image/`)||e.startsWith(`audio/`)||e.startsWith(`video/`),l=e=>{if(!e.length)return;let t=e.indexOf(`;`);return t>=0?e.substring(0,t):e},u=e=>{let t=e.pageSpreadLeft===!0;if(t!==(e.pageSpreadRight===!0))return t?`left`:`right`},d=(e,t,n)=>{if(Object.is(e,t))return!0;if(typeof e!=`object`||!e||typeof t!=`object`||!t)return!1;let r=Object.keys(e),i=Object.keys(t);if(r.length!==i.length)return!1;let a=n?.customEqual??Object.is,o=e,s=t;for(let e of r)if(!Object.hasOwn(s,e)||!a(o[e],s[e]))return!1;return!0},f=(e,t)=>e.reduce((e,n)=>{let r=t(n);return e[r]||(e[r]=[]),e[r].push(n),e},{});function p(e,t){let n={...e};for(let r in t)if(Object.hasOwn(t,r)){let i=t[r];(i!==void 0||!(r in e))&&(n[r]=i)}return n}var m=()=>{if(!(typeof window>`u`))return window};function h(){let e=m()?.__PROSE_READER_DEBUG;return e===!0||e===`true`}var g=(e,t)=>typeof e==`boolean`?{enabled:e,options:t}:{enabled:e?.enabled,options:e??t},_=e=>e.at(-1)?.color,v=e=>{if(!e.length)return;if(!e.some(e=>e.color))return[e.map(e=>e.label).join(` `)];let t=``,n=[];for(let r of e)t+=`%c${t?` ${r.label}`:r.label}`,n.push(r.color?`color: ${r.color}`:``);return[t,...n]},y=(e=[],t=h())=>{let n=t,r={enable:e=>{a(e)},namespace:(t,r,i)=>{let a=g(r,i),o=a.options?.color??_(e);return y([...e,{label:`[${t}]`,color:o}],a.enabled??n)},isEnabled:()=>n,debug:()=>{},info:()=>{},log:()=>{},groupCollapsed:()=>{},groupEnd:()=>{},getGroupArgs:t=>{let n=v(e);if(!n)return[t];let[r,...i]=n;return[`${r} ${t}`,...i]},warn:()=>{},error:()=>{}},i=t=>{if(!t){r.debug=()=>{},r.info=()=>{},r.log=()=>{},r.groupCollapsed=()=>{},r.groupEnd=()=>{},r.warn=()=>{},r.error=()=>{};return}let n=v(e)??[];r.debug=Function.prototype.bind.call(console.debug,console,...n),r.info=Function.prototype.bind.call(console.info,console,...n),r.log=Function.prototype.bind.call(console.log,console,...n),r.groupCollapsed=Function.prototype.bind.call(console.groupCollapsed,console),r.groupEnd=Function.prototype.bind.call(console.groupEnd,console),r.warn=Function.prototype.bind.call(console.warn,console,...n),r.error=Function.prototype.bind.call(console.error,console,...n)},a=e=>{n!==e&&(n=e,i(n))};return i(n),r},b=y();e.PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME=`prose-reader-resource-error`,e.Report=b,e.arrayEqual=t,e.detectMimeTypeFromName=o,e.escapeXmlAttributeValue=e=>e.replaceAll(`&`,`&amp;`).replaceAll(`"`,`&quot;`).replaceAll(`'`,`&apos;`).replaceAll(`<`,`&lt;`).replaceAll(`>`,`&gt;`),e.getItemSpreadPosition=u,e.getParentPath=a,e.getUrlExtension=n,e.groupBy=f,e.isMediaContentMimeType=c,e.isShallowEqual=d,e.isXmlBasedMimeType=s,e.parseContentType=l,e.shallowMergeIfDefined=p,e.urlJoin=i});
2
+ //# sourceMappingURL=index.umd.cjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.umd.cjs","sources":["../src/array.ts","../src/url.ts","../src/contentType.ts","../src/manifest/utils.ts","../src/objects.ts","../src/report.ts","../src/resources.ts","../src/xml.ts"],"sourcesContent":["// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const arrayEqual = <A extends any[], B extends any[]>(a: A, b: B) => {\n // @ts-expect-error\n if (a === b) return true\n if (a.length !== b.length) return false\n\n return a.every((v, i) => v === b[i])\n}\n","export const getUrlExtension = (url: string) => {\n return url.split(/[#?]/)[0]?.split(`.`).pop()?.trim() || ``\n}\n\nfunction normalize(strArray: string[]) {\n const resultArray = []\n if (strArray.length === 0) {\n return \"\"\n }\n\n if (typeof strArray[0] !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${strArray[0]}`)\n }\n\n // If the first part is a plain protocol, we combine it with the next part.\n if (strArray[0].match(/^[^/:]+:\\/*$/) && strArray.length > 1) {\n strArray[0] = strArray.shift() + strArray[0]\n }\n\n // There must be two or three slashes in the file protocol, two slashes in anything else.\n if (strArray[0].match(/^file:\\/\\/\\//)) {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1:///\")\n } else {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1://\")\n }\n\n for (let i = 0; i < strArray.length; i++) {\n let component = strArray[i]\n\n if (typeof component !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${component}`)\n }\n\n if (component === \"\") {\n continue\n }\n\n if (i > 0) {\n // Removing the starting slashes for each component but the first.\n component = component.replace(/^[/]+/, \"\")\n }\n if (i < strArray.length - 1) {\n // Removing the ending slashes for each component but the last.\n component = component.replace(/[/]+$/, \"\")\n } else {\n // For the last component we will combine multiple slashes to a single one.\n component = component.replace(/[/]+$/, \"/\")\n }\n\n resultArray.push(component)\n }\n\n let str = resultArray.join(\"/\")\n // Each input component is now separated by a single slash except the possible first plain protocol part.\n\n // remove trailing slash before parameters or hash\n str = str.replace(/\\/(\\?|&|#[^!])/g, \"$1\")\n\n // replace ? in parameters with &\n const parts = str.split(\"?\")\n str = parts.shift() + (parts.length > 0 ? \"?\" : \"\") + parts.join(\"&\")\n\n return str\n}\n\nexport function urlJoin(...args: (string | string[])[]) {\n const parts = Array.from(Array.isArray(args[0]) ? args[0] : args) as string[]\n\n return normalize(parts)\n}\n\nexport function getParentPath(uri: string): string {\n const parts = uri.split(\"/\")\n\n parts.pop() // Remove the last segment\n\n return parts.join(\"/\")\n}\n","import { getUrlExtension } from \"./url\"\n\nexport const detectMimeTypeFromName = (name: string) => {\n const extension = getUrlExtension(name)\n\n switch (extension) {\n case `png`:\n return `image/png`\n case `jpg`:\n return `image/jpg`\n case `jpeg`:\n return `image/jpeg`\n case `txt`:\n return `text/plain`\n case `webp`:\n return `image/webp`\n case `xhtml`:\n return `application/xhtml+xml`\n case `mp3`:\n return `audio/mpeg`\n case `m4a`:\n case `m4b`:\n return `audio/mp4`\n case `aac`:\n return `audio/aac`\n case `ogg`:\n case `oga`:\n return `audio/ogg`\n case `wav`:\n return `audio/wav`\n case `flac`:\n return `audio/flac`\n case `opus`:\n return `audio/opus`\n }\n\n return undefined\n}\n\nexport const isXmlBasedMimeType = ({\n mimeType,\n uri,\n}: {\n uri?: string\n mimeType?: string\n}) => {\n const _mimeType = mimeType ?? detectMimeTypeFromName(uri ?? \"\")\n\n return _mimeType?.startsWith(`application/xhtml+xml`)\n}\n\n/**\n * Discrete media MIME types represent a single self-contained unit\n * (an image, an audio track, a video clip) as opposed to document\n * content that may need reflowable layout.\n */\nexport const isMediaContentMimeType = (mimeType: string) =>\n mimeType.startsWith(\"image/\") ||\n mimeType.startsWith(\"audio/\") ||\n mimeType.startsWith(\"video/\")\n\nexport const parseContentType = (str: string) => {\n if (!str.length) return undefined\n\n const cut = str.indexOf(`;`)\n\n return cut >= 0 ? str.substring(0, cut) : str\n}\n","import type { Manifest } from \"./Manifest\"\n\nexport type SpreadPosition = \"left\" | \"right\" | \"none\"\n\nexport const getItemSpreadPosition = (\n item: Manifest[`spineItems`][number],\n): Exclude<SpreadPosition, \"none\"> | undefined => {\n const hasPageSpreadLeft = item.pageSpreadLeft === true\n const hasPageSpreadRight = item.pageSpreadRight === true\n\n if (hasPageSpreadLeft === hasPageSpreadRight) return undefined\n\n return hasPageSpreadLeft ? `left` : `right`\n}\n","/**\n * Return true if `objectA` is shallow equal to `objectB`.\n *\n * - Primitives are compared with `Object.is` (so `NaN === NaN`, `+0 ≠ -0`).\n * - Objects and arrays are compared by their own enumerable string keys,\n * each value pair compared with `Object.is` (or `customEqual`).\n * - Two objects with the same keys but different prototypes are considered\n * equal — class identity is not checked.\n *\n * Pass `customEqual` to override the per-value comparator (e.g. to recurse\n * one level deeper, or to compare specific value types by content).\n */\nexport const isShallowEqual = <A = unknown, B = unknown>(\n objectA: A,\n objectB: B,\n options?: {\n customEqual?: (a: unknown, b: unknown) => boolean\n },\n): boolean => {\n if (Object.is(objectA, objectB)) {\n return true\n }\n if (\n typeof objectA !== \"object\" ||\n objectA === null ||\n typeof objectB !== \"object\" ||\n objectB === null\n ) {\n return false\n }\n\n const keysA = Object.keys(objectA)\n const keysB = Object.keys(objectB)\n\n if (keysA.length !== keysB.length) {\n return false\n }\n\n const isEqual = options?.customEqual ?? Object.is\n\n // Indexable cast after narrowing to non-null `object`: TypeScript doesn't\n // widen `object` itself, but every iterated key is verified as own via\n // `Object.keys` / `Object.hasOwn`.\n const a = objectA as Record<string, unknown>\n const b = objectB as Record<string, unknown>\n\n for (const key of keysA) {\n if (!Object.hasOwn(b, key) || !isEqual(a[key], b[key])) {\n return false\n }\n }\n\n return true\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const groupBy = <T, K extends keyof any>(\n list: T[],\n getKey: (item: T) => K,\n) =>\n list.reduce(\n (previous, currentItem) => {\n const group = getKey(currentItem)\n if (!previous[group]) previous[group] = []\n previous[group].push(currentItem)\n return previous\n },\n {} as Record<K, T[]>,\n )\n\nexport function shallowMergeIfDefined<T extends object, U extends object>(\n obj1: T,\n obj2: U,\n): T & U {\n const result = { ...obj1 } as T & U\n\n for (const key in obj2) {\n if (Object.hasOwn(obj2, key)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n const value = (obj2 as any)[key]\n if (value !== undefined || !(key in obj1)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n ;(result as any)[key] = value\n }\n }\n }\n\n return result\n}\n","/**\n * This report class needs to be fast because it is used in hot path. The disabled state\n * should not trigger any i/o and be no op operations.\n * The logging should keep the original stack trace.\n */\n\ndeclare global {\n interface Window {\n __PROSE_READER_DEBUG?: boolean | string\n }\n}\n\nconst getWindow = () => {\n if (typeof window === \"undefined\") {\n return undefined\n }\n\n return window\n}\n\nfunction isGlobalDebugEnabled() {\n const debug = getWindow()?.__PROSE_READER_DEBUG\n\n return debug === true || debug === \"true\"\n}\n\nexport type ReportOptions = {\n color?: string\n}\n\nexport type ReportNamespaceOptions = ReportOptions & {\n enabled?: boolean\n}\n\ntype ReportNamespace = {\n (namespace: string, enabled?: boolean, options?: ReportOptions): Report\n (namespace: string, options?: ReportNamespaceOptions): Report\n}\n\ntype ReportGroupArgs = [label: string, ...styles: string[]]\n\ntype NamespaceSegment = {\n label: string\n color?: string\n}\n\nexport type Report = {\n enable: (enabled: boolean) => void\n isEnabled: () => boolean\n namespace: ReportNamespace\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n debug: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n info: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n log: (...args: any[]) => void\n /** Use with getGroupArgs for namespace + color. Preserves call site. */\n groupCollapsed: (...args: unknown[]) => void\n groupEnd: () => void\n /** Returns [label, ...styles] to pass to groupCollapsed for namespace + color. */\n getGroupArgs: (title: string) => ReportGroupArgs\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n warn: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n error: (...args: any[]) => void\n}\n\nconst getNamespaceArgs = (\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n options?: ReportOptions,\n) => {\n if (typeof enabledOrOptions === \"boolean\") {\n return {\n enabled: enabledOrOptions,\n options,\n }\n }\n\n return {\n enabled: enabledOrOptions?.enabled,\n options: enabledOrOptions ?? options,\n }\n}\n\nconst getCurrentColor = (namespaceSegments: NamespaceSegment[]) =>\n namespaceSegments.at(-1)?.color\n\nconst getStyledNamespaceArgs = (\n namespaceSegments: NamespaceSegment[],\n): ReportGroupArgs | undefined => {\n if (!namespaceSegments.length) {\n return undefined\n }\n\n if (!namespaceSegments.some((segment) => segment.color)) {\n return [namespaceSegments.map((segment) => segment.label).join(\" \")]\n }\n\n let label = \"\"\n const styles: string[] = []\n\n for (const namespaceSegment of namespaceSegments) {\n label += `%c${label ? ` ${namespaceSegment.label}` : namespaceSegment.label}`\n styles.push(\n namespaceSegment.color ? `color: ${namespaceSegment.color}` : \"\",\n )\n }\n\n return [label, ...styles]\n}\n\nconst createReport = (\n namespaceSegments: NamespaceSegment[] = [],\n enabled = isGlobalDebugEnabled(),\n): Report => {\n let reportEnabled = enabled\n\n const report: Report = {\n enable: (enabled: boolean) => {\n setEnabled(enabled)\n },\n namespace: (\n _namespace: string,\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n namespaceOptions?: ReportOptions,\n ) => {\n const childNamespaceArgs = getNamespaceArgs(\n enabledOrOptions,\n namespaceOptions,\n )\n\n const childColor =\n childNamespaceArgs.options?.color ?? getCurrentColor(namespaceSegments)\n\n return createReport(\n [\n ...namespaceSegments,\n {\n label: `[${_namespace}]`,\n color: childColor,\n },\n ],\n childNamespaceArgs.enabled ?? reportEnabled,\n )\n },\n isEnabled: () => reportEnabled,\n debug: () => {},\n info: () => {},\n log: () => {},\n groupCollapsed: () => {},\n groupEnd: () => {},\n getGroupArgs: (title: string) => {\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments)\n\n if (!namespaceArgs) {\n return [title]\n }\n\n const [namespaceLabel, ...namespaceStyles] = namespaceArgs\n\n return [`${namespaceLabel} ${title}`, ...namespaceStyles]\n },\n warn: () => {},\n error: () => {},\n }\n\n // Keep direct prototype-bound console functions for enabled mode.\n // Wrapping console calls in closures changes call-site/stack behavior in devtools,\n // which makes runtime debugging harder to interpret.\n const applyEnabledState = (enabled: boolean) => {\n if (!enabled) {\n report.debug = () => {}\n report.info = () => {}\n report.log = () => {}\n report.groupCollapsed = () => {}\n report.groupEnd = () => {}\n report.warn = () => {}\n report.error = () => {}\n\n return\n }\n\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments) ?? []\n\n report.debug = Function.prototype.bind.call(\n console.debug,\n console,\n ...namespaceArgs,\n )\n report.info = Function.prototype.bind.call(\n console.info,\n console,\n ...namespaceArgs,\n )\n report.log = Function.prototype.bind.call(\n console.log,\n console,\n ...namespaceArgs,\n )\n report.groupCollapsed = Function.prototype.bind.call(\n console.groupCollapsed,\n console,\n )\n report.groupEnd = Function.prototype.bind.call(console.groupEnd, console)\n report.warn = Function.prototype.bind.call(\n console.warn,\n console,\n ...namespaceArgs,\n )\n report.error = Function.prototype.bind.call(\n console.error,\n console,\n ...namespaceArgs,\n )\n }\n\n const setEnabled = (enabled: boolean) => {\n if (reportEnabled === enabled) {\n return\n }\n\n reportEnabled = enabled\n applyEnabledState(reportEnabled)\n }\n\n applyEnabledState(reportEnabled)\n\n return report\n}\n\nexport const Report = createReport()\n","export const PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME =\n \"prose-reader-resource-error\"\n","export const escapeXmlAttributeValue = (value: string) =>\n value\n .replaceAll(`&`, `&amp;`)\n .replaceAll(`\"`, `&quot;`)\n .replaceAll(`'`, `&apos;`)\n .replaceAll(`<`, `&lt;`)\n .replaceAll(`>`, `&gt;`)\n"],"names":["arrayEqual","a","b","v","i","getUrlExtension","url","normalize","strArray","resultArray","component","str","parts","urlJoin","args","getParentPath","uri","detectMimeTypeFromName","name","isXmlBasedMimeType","mimeType","isMediaContentMimeType","parseContentType","cut","getItemSpreadPosition","item","hasPageSpreadLeft","hasPageSpreadRight","isShallowEqual","objectA","objectB","options","keysA","keysB","isEqual","key","groupBy","list","getKey","previous","currentItem","group","shallowMergeIfDefined","obj1","obj2","result","value","getWindow","isGlobalDebugEnabled","debug","getNamespaceArgs","enabledOrOptions","getCurrentColor","namespaceSegments","getStyledNamespaceArgs","segment","label","styles","namespaceSegment","createReport","enabled","reportEnabled","report","setEnabled","_namespace","namespaceOptions","childNamespaceArgs","childColor","title","namespaceArgs","namespaceLabel","namespaceStyles","applyEnabledState","Report","PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME","escapeXmlAttributeValue"],"mappings":"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"}
1
+ {"version":3,"file":"index.umd.cjs","names":[],"sources":["../src/array.ts","../src/url.ts","../src/contentType.ts","../src/manifest/utils.ts","../src/objects.ts","../src/report.ts","../src/resources.ts","../src/xml.ts"],"sourcesContent":["// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const arrayEqual = <A extends any[], B extends any[]>(a: A, b: B) => {\n // @ts-expect-error\n if (a === b) return true\n if (a.length !== b.length) return false\n\n return a.every((v, i) => v === b[i])\n}\n","export const getUrlExtension = (url: string) => {\n return url.split(/[#?]/)[0]?.split(`.`).pop()?.trim() || ``\n}\n\nfunction normalize(strArray: string[]) {\n const resultArray = []\n if (strArray.length === 0) {\n return \"\"\n }\n\n if (typeof strArray[0] !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${strArray[0]}`)\n }\n\n // If the first part is a plain protocol, we combine it with the next part.\n if (strArray[0].match(/^[^/:]+:\\/*$/) && strArray.length > 1) {\n strArray[0] = strArray.shift() + strArray[0]\n }\n\n // There must be two or three slashes in the file protocol, two slashes in anything else.\n if (strArray[0].match(/^file:\\/\\/\\//)) {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1:///\")\n } else {\n strArray[0] = strArray[0].replace(/^([^/:]+):\\/*/, \"$1://\")\n }\n\n for (let i = 0; i < strArray.length; i++) {\n let component = strArray[i]\n\n if (typeof component !== \"string\") {\n throw new TypeError(`Url must be a string. Received ${component}`)\n }\n\n if (component === \"\") {\n continue\n }\n\n if (i > 0) {\n // Removing the starting slashes for each component but the first.\n component = component.replace(/^[/]+/, \"\")\n }\n if (i < strArray.length - 1) {\n // Removing the ending slashes for each component but the last.\n component = component.replace(/[/]+$/, \"\")\n } else {\n // For the last component we will combine multiple slashes to a single one.\n component = component.replace(/[/]+$/, \"/\")\n }\n\n resultArray.push(component)\n }\n\n let str = resultArray.join(\"/\")\n // Each input component is now separated by a single slash except the possible first plain protocol part.\n\n // remove trailing slash before parameters or hash\n str = str.replace(/\\/(\\?|&|#[^!])/g, \"$1\")\n\n // replace ? in parameters with &\n const parts = str.split(\"?\")\n str = parts.shift() + (parts.length > 0 ? \"?\" : \"\") + parts.join(\"&\")\n\n return str\n}\n\nexport function urlJoin(...args: (string | string[])[]) {\n const parts = Array.from(Array.isArray(args[0]) ? args[0] : args) as string[]\n\n return normalize(parts)\n}\n\nexport function getParentPath(uri: string): string {\n const parts = uri.split(\"/\")\n\n parts.pop() // Remove the last segment\n\n return parts.join(\"/\")\n}\n","import { getUrlExtension } from \"./url\"\n\nexport const detectMimeTypeFromName = (name: string) => {\n const extension = getUrlExtension(name)\n\n switch (extension) {\n case `png`:\n return `image/png`\n case `jpg`:\n return `image/jpg`\n case `jpeg`:\n return `image/jpeg`\n case `txt`:\n return `text/plain`\n case `webp`:\n return `image/webp`\n case `xhtml`:\n return `application/xhtml+xml`\n case `mp3`:\n return `audio/mpeg`\n case `m4a`:\n case `m4b`:\n return `audio/mp4`\n case `aac`:\n return `audio/aac`\n case `ogg`:\n case `oga`:\n return `audio/ogg`\n case `wav`:\n return `audio/wav`\n case `flac`:\n return `audio/flac`\n case `opus`:\n return `audio/opus`\n }\n\n return undefined\n}\n\nexport const isXmlBasedMimeType = ({\n mimeType,\n uri,\n}: {\n uri?: string\n mimeType?: string\n}) => {\n const _mimeType = mimeType ?? detectMimeTypeFromName(uri ?? \"\")\n\n return _mimeType?.startsWith(`application/xhtml+xml`)\n}\n\n/**\n * Discrete media MIME types represent a single self-contained unit\n * (an image, an audio track, a video clip) as opposed to document\n * content that may need reflowable layout.\n */\nexport const isMediaContentMimeType = (mimeType: string) =>\n mimeType.startsWith(\"image/\") ||\n mimeType.startsWith(\"audio/\") ||\n mimeType.startsWith(\"video/\")\n\nexport const parseContentType = (str: string) => {\n if (!str.length) return undefined\n\n const cut = str.indexOf(`;`)\n\n return cut >= 0 ? str.substring(0, cut) : str\n}\n","import type { Manifest } from \"./Manifest\"\n\nexport type SpreadPosition = \"left\" | \"right\" | \"none\"\n\nexport const getItemSpreadPosition = (\n item: Manifest[`spineItems`][number],\n): Exclude<SpreadPosition, \"none\"> | undefined => {\n const hasPageSpreadLeft = item.pageSpreadLeft === true\n const hasPageSpreadRight = item.pageSpreadRight === true\n\n if (hasPageSpreadLeft === hasPageSpreadRight) return undefined\n\n return hasPageSpreadLeft ? `left` : `right`\n}\n","/**\n * Return true if `objectA` is shallow equal to `objectB`.\n *\n * - Primitives are compared with `Object.is` (so `NaN === NaN`, `+0 ≠ -0`).\n * - Objects and arrays are compared by their own enumerable string keys,\n * each value pair compared with `Object.is` (or `customEqual`).\n * - Two objects with the same keys but different prototypes are considered\n * equal — class identity is not checked.\n *\n * Pass `customEqual` to override the per-value comparator (e.g. to recurse\n * one level deeper, or to compare specific value types by content).\n */\nexport const isShallowEqual = <A = unknown, B = unknown>(\n objectA: A,\n objectB: B,\n options?: {\n customEqual?: (a: unknown, b: unknown) => boolean\n },\n): boolean => {\n if (Object.is(objectA, objectB)) {\n return true\n }\n if (\n typeof objectA !== \"object\" ||\n objectA === null ||\n typeof objectB !== \"object\" ||\n objectB === null\n ) {\n return false\n }\n\n const keysA = Object.keys(objectA)\n const keysB = Object.keys(objectB)\n\n if (keysA.length !== keysB.length) {\n return false\n }\n\n const isEqual = options?.customEqual ?? Object.is\n\n // Indexable cast after narrowing to non-null `object`: TypeScript doesn't\n // widen `object` itself, but every iterated key is verified as own via\n // `Object.keys` / `Object.hasOwn`.\n const a = objectA as Record<string, unknown>\n const b = objectB as Record<string, unknown>\n\n for (const key of keysA) {\n if (!Object.hasOwn(b, key) || !isEqual(a[key], b[key])) {\n return false\n }\n }\n\n return true\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: TODO\nexport const groupBy = <T, K extends keyof any>(\n list: T[],\n getKey: (item: T) => K,\n) =>\n list.reduce(\n (previous, currentItem) => {\n const group = getKey(currentItem)\n if (!previous[group]) previous[group] = []\n previous[group].push(currentItem)\n return previous\n },\n {} as Record<K, T[]>,\n )\n\nexport function shallowMergeIfDefined<T extends object, U extends object>(\n obj1: T,\n obj2: U,\n): T & U {\n const result = { ...obj1 } as T & U\n\n for (const key in obj2) {\n if (Object.hasOwn(obj2, key)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n const value = (obj2 as any)[key]\n if (value !== undefined || !(key in obj1)) {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n ;(result as any)[key] = value\n }\n }\n }\n\n return result\n}\n","/**\n * This report class needs to be fast because it is used in hot path. The disabled state\n * should not trigger any i/o and be no op operations.\n * The logging should keep the original stack trace.\n */\n\ndeclare global {\n interface Window {\n __PROSE_READER_DEBUG?: boolean | string\n }\n}\n\nconst getWindow = () => {\n if (typeof window === \"undefined\") {\n return undefined\n }\n\n return window\n}\n\nfunction isGlobalDebugEnabled() {\n const debug = getWindow()?.__PROSE_READER_DEBUG\n\n return debug === true || debug === \"true\"\n}\n\nexport type ReportOptions = {\n color?: string\n}\n\nexport type ReportNamespaceOptions = ReportOptions & {\n enabled?: boolean\n}\n\ntype ReportNamespace = {\n (namespace: string, enabled?: boolean, options?: ReportOptions): Report\n (namespace: string, options?: ReportNamespaceOptions): Report\n}\n\ntype ReportGroupArgs = [label: string, ...styles: string[]]\n\ntype NamespaceSegment = {\n label: string\n color?: string\n}\n\nexport type Report = {\n enable: (enabled: boolean) => void\n isEnabled: () => boolean\n namespace: ReportNamespace\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n debug: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n info: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n log: (...args: any[]) => void\n /** Use with getGroupArgs for namespace + color. Preserves call site. */\n groupCollapsed: (...args: unknown[]) => void\n groupEnd: () => void\n /** Returns [label, ...styles] to pass to groupCollapsed for namespace + color. */\n getGroupArgs: (title: string) => ReportGroupArgs\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n warn: (...args: any[]) => void\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n error: (...args: any[]) => void\n}\n\nconst getNamespaceArgs = (\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n options?: ReportOptions,\n) => {\n if (typeof enabledOrOptions === \"boolean\") {\n return {\n enabled: enabledOrOptions,\n options,\n }\n }\n\n return {\n enabled: enabledOrOptions?.enabled,\n options: enabledOrOptions ?? options,\n }\n}\n\nconst getCurrentColor = (namespaceSegments: NamespaceSegment[]) =>\n namespaceSegments.at(-1)?.color\n\nconst getStyledNamespaceArgs = (\n namespaceSegments: NamespaceSegment[],\n): ReportGroupArgs | undefined => {\n if (!namespaceSegments.length) {\n return undefined\n }\n\n if (!namespaceSegments.some((segment) => segment.color)) {\n return [namespaceSegments.map((segment) => segment.label).join(\" \")]\n }\n\n let label = \"\"\n const styles: string[] = []\n\n for (const namespaceSegment of namespaceSegments) {\n label += `%c${label ? ` ${namespaceSegment.label}` : namespaceSegment.label}`\n styles.push(\n namespaceSegment.color ? `color: ${namespaceSegment.color}` : \"\",\n )\n }\n\n return [label, ...styles]\n}\n\nconst createReport = (\n namespaceSegments: NamespaceSegment[] = [],\n enabled = isGlobalDebugEnabled(),\n): Report => {\n let reportEnabled = enabled\n\n const report: Report = {\n enable: (enabled: boolean) => {\n setEnabled(enabled)\n },\n namespace: (\n _namespace: string,\n enabledOrOptions?: boolean | ReportNamespaceOptions,\n namespaceOptions?: ReportOptions,\n ) => {\n const childNamespaceArgs = getNamespaceArgs(\n enabledOrOptions,\n namespaceOptions,\n )\n\n const childColor =\n childNamespaceArgs.options?.color ?? getCurrentColor(namespaceSegments)\n\n return createReport(\n [\n ...namespaceSegments,\n {\n label: `[${_namespace}]`,\n color: childColor,\n },\n ],\n childNamespaceArgs.enabled ?? reportEnabled,\n )\n },\n isEnabled: () => reportEnabled,\n debug: () => {},\n info: () => {},\n log: () => {},\n groupCollapsed: () => {},\n groupEnd: () => {},\n getGroupArgs: (title: string) => {\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments)\n\n if (!namespaceArgs) {\n return [title]\n }\n\n const [namespaceLabel, ...namespaceStyles] = namespaceArgs\n\n return [`${namespaceLabel} ${title}`, ...namespaceStyles]\n },\n warn: () => {},\n error: () => {},\n }\n\n // Keep direct prototype-bound console functions for enabled mode.\n // Wrapping console calls in closures changes call-site/stack behavior in devtools,\n // which makes runtime debugging harder to interpret.\n const applyEnabledState = (enabled: boolean) => {\n if (!enabled) {\n report.debug = () => {}\n report.info = () => {}\n report.log = () => {}\n report.groupCollapsed = () => {}\n report.groupEnd = () => {}\n report.warn = () => {}\n report.error = () => {}\n\n return\n }\n\n const namespaceArgs = getStyledNamespaceArgs(namespaceSegments) ?? []\n\n report.debug = Function.prototype.bind.call(\n console.debug,\n console,\n ...namespaceArgs,\n )\n report.info = Function.prototype.bind.call(\n console.info,\n console,\n ...namespaceArgs,\n )\n report.log = Function.prototype.bind.call(\n console.log,\n console,\n ...namespaceArgs,\n )\n report.groupCollapsed = Function.prototype.bind.call(\n console.groupCollapsed,\n console,\n )\n report.groupEnd = Function.prototype.bind.call(console.groupEnd, console)\n report.warn = Function.prototype.bind.call(\n console.warn,\n console,\n ...namespaceArgs,\n )\n report.error = Function.prototype.bind.call(\n console.error,\n console,\n ...namespaceArgs,\n )\n }\n\n const setEnabled = (enabled: boolean) => {\n if (reportEnabled === enabled) {\n return\n }\n\n reportEnabled = enabled\n applyEnabledState(reportEnabled)\n }\n\n applyEnabledState(reportEnabled)\n\n return report\n}\n\nexport const Report = createReport()\n","export const PROSE_READER_RESOURCE_ERROR_INJECTED_META_NAME =\n \"prose-reader-resource-error\"\n","export const escapeXmlAttributeValue = (value: string) =>\n value\n .replaceAll(`&`, `&amp;`)\n .replaceAll(`\"`, `&quot;`)\n .replaceAll(`'`, `&apos;`)\n .replaceAll(`<`, `&lt;`)\n .replaceAll(`>`, `&gt;`)\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@prose-reader/shared",
3
- "version": "1.303.0",
3
+ "version": "1.304.0",
4
4
  "type": "module",
5
5
  "main": "./dist/index.umd.cjs",
6
6
  "module": "./dist/index.js",
@@ -20,5 +20,5 @@
20
20
  "build": "tsc && vite build",
21
21
  "test": "vitest run --coverage"
22
22
  },
23
- "gitHead": "dddc06869ba852fa280e050c1c1929433bb91c5e"
23
+ "gitHead": "1f4e4822d4d74fc2f6025409133db0f4f7f152af"
24
24
  }