@analogjs/content 3.0.0-alpha.2 → 3.0.0-alpha.21

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 (93) hide show
  1. package/LICENSE +21 -0
  2. package/fesm2022/analogjs-content-md4x.mjs +290 -0
  3. package/fesm2022/analogjs-content-md4x.mjs.map +1 -0
  4. package/fesm2022/analogjs-content-mdc.mjs +170 -0
  5. package/fesm2022/analogjs-content-mdc.mjs.map +1 -0
  6. package/fesm2022/analogjs-content-og.mjs +34 -46
  7. package/fesm2022/analogjs-content-og.mjs.map +1 -1
  8. package/fesm2022/analogjs-content-prism-highlighter.mjs +81 -68
  9. package/fesm2022/analogjs-content-prism-highlighter.mjs.map +1 -1
  10. package/fesm2022/analogjs-content-resources.mjs +113 -124
  11. package/fesm2022/analogjs-content-resources.mjs.map +1 -1
  12. package/fesm2022/analogjs-content-shiki-highlighter.mjs +9 -14
  13. package/fesm2022/analogjs-content-shiki-highlighter.mjs.map +1 -1
  14. package/fesm2022/analogjs-content.mjs +658 -636
  15. package/fesm2022/analogjs-content.mjs.map +1 -1
  16. package/fesm2022/content-list-loader.mjs +284 -0
  17. package/fesm2022/content-list-loader.mjs.map +1 -0
  18. package/md4x/package.json +4 -0
  19. package/mdc/package.json +4 -0
  20. package/og/package.json +2 -2
  21. package/package.json +67 -38
  22. package/plugin/package.json +2 -22
  23. package/plugin/src/index.d.ts +4 -2
  24. package/plugin/src/index.js +6 -3
  25. package/plugin/src/index.js.map +1 -1
  26. package/plugin/src/migrations/update-markdown-version/compat.d.ts +6 -2
  27. package/plugin/src/migrations/update-markdown-version/compat.js +8 -5
  28. package/plugin/src/migrations/update-markdown-version/compat.js.map +1 -1
  29. package/plugin/src/migrations/update-markdown-version/update-markdown-version.d.ts +6 -2
  30. package/plugin/src/migrations/update-markdown-version/update-markdown-version.js +18 -20
  31. package/plugin/src/migrations/update-markdown-version/update-markdown-version.js.map +1 -1
  32. package/prism-highlighter/package.json +2 -2
  33. package/resources/package.json +2 -2
  34. package/shiki-highlighter/package.json +2 -2
  35. package/src/lib/devtools/content-devtools-client.ts +215 -0
  36. package/src/lib/devtools/content-devtools.styles.css +194 -0
  37. package/types/md4x/src/index.d.ts +5 -0
  38. package/types/md4x/src/lib/md4x-content-renderer.service.d.ts +33 -0
  39. package/types/md4x/src/lib/md4x-wasm-content-renderer.service.d.ts +16 -0
  40. package/types/md4x/src/lib/provide-md4x.d.ts +26 -0
  41. package/types/md4x/src/lib/streaming-markdown-renderer.d.ts +21 -0
  42. package/types/mdc/src/index.d.ts +2 -0
  43. package/types/mdc/src/lib/mdc-component-registry.d.ts +25 -0
  44. package/types/mdc/src/lib/mdc-renderer.directive.d.ts +33 -0
  45. package/types/og/src/index.d.ts +2 -0
  46. package/types/og/src/lib/og.d.ts +5 -0
  47. package/types/og/src/lib/options.d.ts +11 -0
  48. package/types/prism-highlighter/src/index.d.ts +8 -0
  49. package/types/prism-highlighter/src/lib/prism/angular.d.ts +1 -0
  50. package/types/prism-highlighter/src/lib/prism-highlighter.d.ts +8 -0
  51. package/types/resources/src/content-file-resource.d.ts +39 -0
  52. package/types/resources/src/content-files-resource.d.ts +3 -0
  53. package/types/resources/src/index.d.ts +2 -0
  54. package/types/shiki-highlighter/src/index.d.ts +7 -0
  55. package/types/src/index.d.ts +18 -0
  56. package/types/src/lib/anchor-navigation.directive.d.ts +9 -0
  57. package/types/src/lib/content-file-loader.d.ts +6 -0
  58. package/types/src/lib/content-file.d.ts +8 -0
  59. package/types/src/lib/content-files-list-token.d.ts +3 -0
  60. package/types/src/lib/content-files-token.d.ts +3 -0
  61. package/types/src/lib/content-list-loader.d.ts +7 -0
  62. package/types/src/lib/content-renderer.d.ts +33 -0
  63. package/types/src/lib/content.d.ts +14 -0
  64. package/types/src/lib/devtools/content-devtools-plugin.d.ts +23 -0
  65. package/types/src/lib/devtools/content-devtools-renderer.d.ts +23 -0
  66. package/types/src/lib/devtools/index.d.ts +23 -0
  67. package/types/src/lib/get-content-files.d.ts +14 -0
  68. package/types/src/lib/inject-content-files.d.ts +4 -0
  69. package/types/src/lib/markdown-content-renderer.service.d.ts +10 -0
  70. package/types/src/lib/markdown-route.component.d.ts +15 -0
  71. package/types/src/lib/markdown.component.d.ts +26 -0
  72. package/types/src/lib/marked-content-highlighter.d.ts +17 -0
  73. package/types/src/lib/marked-setup.service.d.ts +10 -0
  74. package/types/src/lib/parse-raw-content-file.d.ts +18 -0
  75. package/types/src/lib/provide-content.d.ts +7 -0
  76. package/types/src/lib/render-task.service.d.ts +8 -0
  77. package/types/src/lib/utils/zone-wait-for.d.ts +2 -0
  78. package/og/README.md +0 -3
  79. package/plugin/README.md +0 -11
  80. package/plugin/src/migrations/update-markdown-renderer-feature/compat.d.ts +0 -2
  81. package/plugin/src/migrations/update-markdown-renderer-feature/compat.js +0 -6
  82. package/plugin/src/migrations/update-markdown-renderer-feature/compat.js.map +0 -1
  83. package/plugin/src/migrations/update-markdown-renderer-feature/update-markdown-renderer-feature.d.ts +0 -2
  84. package/plugin/src/migrations/update-markdown-renderer-feature/update-markdown-renderer-feature.js +0 -48
  85. package/plugin/src/migrations/update-markdown-renderer-feature/update-markdown-renderer-feature.js.map +0 -1
  86. package/prism-highlighter/README.md +0 -3
  87. package/resources/README.md +0 -3
  88. package/shiki-highlighter/README.md +0 -3
  89. package/types/analogjs-content-og.d.ts +0 -19
  90. package/types/analogjs-content-prism-highlighter.d.ts +0 -15
  91. package/types/analogjs-content-resources.d.ts +0 -20
  92. package/types/analogjs-content-shiki-highlighter.d.ts +0 -11
  93. package/types/analogjs-content.d.ts +0 -168
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2022 Brandon Roberts
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
@@ -0,0 +1,290 @@
1
+ import { d as ContentRenderer, o as withContentFileLoader, r as withContentListLoader } from "./content-list-loader.mjs";
2
+ import * as i0 from "@angular/core";
3
+ import { Injectable, InjectionToken, inject } from "@angular/core";
4
+ //#region packages/content/md4x/src/lib/md4x-content-renderer.service.ts
5
+ var MD4X_RENDERER_OPTIONS = new InjectionToken("md4x_renderer_options");
6
+ function makeSlug$1(text, slugCounts) {
7
+ const baseSlug = text.toLowerCase().replace(/[^\w\s-]/g, "").trim().replace(/\s+/g, "-");
8
+ const count = slugCounts.get(baseSlug) ?? 0;
9
+ slugCounts.set(baseSlug, count + 1);
10
+ return count === 0 ? baseSlug : `${baseSlug}-${count}`;
11
+ }
12
+ function headingsToToc$1(headings) {
13
+ const slugCounts = /* @__PURE__ */ new Map();
14
+ return headings.map((h) => ({
15
+ id: makeSlug$1(h.text, slugCounts),
16
+ level: h.level,
17
+ text: h.text
18
+ }));
19
+ }
20
+ function injectHeadingIds$1(html, toc) {
21
+ let tocIndex = 0;
22
+ return html.replace(/<h([1-6])>/g, (match, level) => {
23
+ const item = toc[tocIndex++];
24
+ return item ? `<h${level} id="${item.id}">` : match;
25
+ });
26
+ }
27
+ /**
28
+ * Content renderer backed by md4x (C-based CommonMark parser compiled with Zig).
29
+ * 50-70x faster than marked for complex documents.
30
+ *
31
+ * @experimental md4x integration is experimental and may change in future releases.
32
+ */
33
+ var Md4xContentRendererService = class Md4xContentRendererService extends ContentRenderer {
34
+ constructor() {
35
+ super(...arguments);
36
+ this.options = inject(MD4X_RENDERER_OPTIONS, { optional: true });
37
+ }
38
+ async render(content) {
39
+ const { renderToHtml, parseMeta } = await import("md4x/napi");
40
+ const html = renderToHtml(content, {
41
+ heal: this.options?.heal,
42
+ highlighter: this.options?.highlighter
43
+ });
44
+ const toc = headingsToToc$1(parseMeta(content).headings);
45
+ return {
46
+ content: injectHeadingIds$1(html, toc),
47
+ toc
48
+ };
49
+ }
50
+ getContentHeadings(content) {
51
+ const lines = content.split("\n");
52
+ const toc = [];
53
+ const slugCounts = /* @__PURE__ */ new Map();
54
+ for (const line of lines) {
55
+ const match = /^(#{1,6})\s+(.+?)\s*$/.exec(line);
56
+ if (!match) continue;
57
+ const level = match[1].length;
58
+ const text = match[2].trim();
59
+ if (!text) continue;
60
+ toc.push({
61
+ id: makeSlug$1(text, slugCounts),
62
+ level,
63
+ text
64
+ });
65
+ }
66
+ return toc;
67
+ }
68
+ static {
69
+ this.ɵfac = i0.ɵɵngDeclareFactory({
70
+ minVersion: "12.0.0",
71
+ version: "21.2.6",
72
+ ngImport: i0,
73
+ type: Md4xContentRendererService,
74
+ deps: null,
75
+ target: i0.ɵɵFactoryTarget.Injectable
76
+ });
77
+ }
78
+ static {
79
+ this.ɵprov = i0.ɵɵngDeclareInjectable({
80
+ minVersion: "12.0.0",
81
+ version: "21.2.6",
82
+ ngImport: i0,
83
+ type: Md4xContentRendererService
84
+ });
85
+ }
86
+ };
87
+ i0.ɵɵngDeclareClassMetadata({
88
+ minVersion: "12.0.0",
89
+ version: "21.2.6",
90
+ ngImport: i0,
91
+ type: Md4xContentRendererService,
92
+ decorators: [{ type: Injectable }]
93
+ });
94
+ //#endregion
95
+ //#region packages/content/md4x/src/lib/md4x-wasm-content-renderer.service.ts
96
+ function makeSlug(text, slugCounts) {
97
+ const baseSlug = text.toLowerCase().replace(/[^\w\s-]/g, "").trim().replace(/\s+/g, "-");
98
+ const count = slugCounts.get(baseSlug) ?? 0;
99
+ slugCounts.set(baseSlug, count + 1);
100
+ return count === 0 ? baseSlug : `${baseSlug}-${count}`;
101
+ }
102
+ function headingsToToc(headings) {
103
+ const slugCounts = /* @__PURE__ */ new Map();
104
+ return headings.map((h) => ({
105
+ id: makeSlug(h.text, slugCounts),
106
+ level: h.level,
107
+ text: h.text
108
+ }));
109
+ }
110
+ function injectHeadingIds(html, toc) {
111
+ let tocIndex = 0;
112
+ return html.replace(/<h([1-6])>/g, (match, level) => {
113
+ const item = toc[tocIndex++];
114
+ return item ? `<h${level} id="${item.id}">` : match;
115
+ });
116
+ }
117
+ /**
118
+ * Content renderer backed by md4x/wasm for client-side (browser) rendering.
119
+ * ~100KB gzip, 3-6x faster than marked in the browser.
120
+ *
121
+ * @experimental md4x integration is experimental and may change in future releases.
122
+ */
123
+ var Md4xWasmContentRendererService = class Md4xWasmContentRendererService extends ContentRenderer {
124
+ constructor() {
125
+ super(...arguments);
126
+ this.options = inject(MD4X_RENDERER_OPTIONS, { optional: true });
127
+ this.initPromise = null;
128
+ }
129
+ async render(content) {
130
+ const wasm = await import("md4x/wasm");
131
+ if (!this.initPromise) this.initPromise = wasm.init();
132
+ await this.initPromise;
133
+ const html = wasm.renderToHtml(content, {
134
+ heal: this.options?.heal,
135
+ highlighter: this.options?.highlighter
136
+ });
137
+ const toc = headingsToToc(wasm.parseMeta(content).headings);
138
+ return {
139
+ content: injectHeadingIds(html, toc),
140
+ toc
141
+ };
142
+ }
143
+ getContentHeadings(content) {
144
+ const lines = content.split("\n");
145
+ const toc = [];
146
+ const slugCounts = /* @__PURE__ */ new Map();
147
+ for (const line of lines) {
148
+ const match = /^(#{1,6})\s+(.+?)\s*$/.exec(line);
149
+ if (!match) continue;
150
+ const level = match[1].length;
151
+ const text = match[2].trim();
152
+ if (!text) continue;
153
+ toc.push({
154
+ id: makeSlug(text, slugCounts),
155
+ level,
156
+ text
157
+ });
158
+ }
159
+ return toc;
160
+ }
161
+ static {
162
+ this.ɵfac = i0.ɵɵngDeclareFactory({
163
+ minVersion: "12.0.0",
164
+ version: "21.2.6",
165
+ ngImport: i0,
166
+ type: Md4xWasmContentRendererService,
167
+ deps: null,
168
+ target: i0.ɵɵFactoryTarget.Injectable
169
+ });
170
+ }
171
+ static {
172
+ this.ɵprov = i0.ɵɵngDeclareInjectable({
173
+ minVersion: "12.0.0",
174
+ version: "21.2.6",
175
+ ngImport: i0,
176
+ type: Md4xWasmContentRendererService
177
+ });
178
+ }
179
+ };
180
+ i0.ɵɵngDeclareClassMetadata({
181
+ minVersion: "12.0.0",
182
+ version: "21.2.6",
183
+ ngImport: i0,
184
+ type: Md4xWasmContentRendererService,
185
+ decorators: [{ type: Injectable }]
186
+ });
187
+ //#endregion
188
+ //#region packages/content/md4x/src/lib/provide-md4x.ts
189
+ /**
190
+ * Provides the experimental md4x-based content renderer (NAPI, server/build-time).
191
+ *
192
+ * @experimental md4x integration is experimental and may change in future releases.
193
+ *
194
+ * @example
195
+ * ```typescript
196
+ * provideContent(withMd4xRenderer());
197
+ * provideContent(withMd4xRenderer({ heal: true }));
198
+ * ```
199
+ */
200
+ function withMd4xRenderer(options) {
201
+ return [
202
+ {
203
+ provide: ContentRenderer,
204
+ useClass: Md4xContentRendererService
205
+ },
206
+ options ? {
207
+ provide: MD4X_RENDERER_OPTIONS,
208
+ useValue: options
209
+ } : [],
210
+ withContentFileLoader(),
211
+ withContentListLoader()
212
+ ];
213
+ }
214
+ /**
215
+ * Provides the experimental md4x WASM content renderer (browser/CSR).
216
+ * ~100KB gzip, 3-6x faster than marked in the browser.
217
+ *
218
+ * @experimental md4x integration is experimental and may change in future releases.
219
+ *
220
+ * @example
221
+ * ```typescript
222
+ * provideContent(withMd4xWasmRenderer());
223
+ * ```
224
+ */
225
+ function withMd4xWasmRenderer(options) {
226
+ return [
227
+ {
228
+ provide: ContentRenderer,
229
+ useClass: Md4xWasmContentRendererService
230
+ },
231
+ options ? {
232
+ provide: MD4X_RENDERER_OPTIONS,
233
+ useValue: options
234
+ } : [],
235
+ withContentFileLoader(),
236
+ withContentListLoader()
237
+ ];
238
+ }
239
+ //#endregion
240
+ //#region packages/content/md4x/src/lib/streaming-markdown-renderer.ts
241
+ /**
242
+ * Transforms a stream of markdown chunks into a stream of rendered HTML.
243
+ * Uses md4x's `heal()` to fix incomplete markdown from streaming sources
244
+ * (LLMs, collaborative editing) so each emitted HTML chunk is valid.
245
+ *
246
+ * @experimental Streaming markdown support is experimental and may change in future releases.
247
+ *
248
+ * @example
249
+ * ```typescript
250
+ * // In a Nitro API route
251
+ * import { streamMarkdown } from '@analogjs/content';
252
+ *
253
+ * export default defineEventHandler(async (event) => {
254
+ * const llmStream = getAIStream(prompt);
255
+ * return streamMarkdown(llmStream, { heal: true });
256
+ * });
257
+ * ```
258
+ */
259
+ async function streamMarkdown(input, options) {
260
+ const { renderToHtml, heal } = await import("md4x/napi");
261
+ let buffer = "";
262
+ const reader = input.getReader();
263
+ return new ReadableStream({
264
+ async pull(controller) {
265
+ try {
266
+ const { done, value } = await reader.read();
267
+ if (done) {
268
+ if (buffer) {
269
+ const source = options?.heal ? heal(buffer) : buffer;
270
+ controller.enqueue(renderToHtml(source));
271
+ }
272
+ controller.close();
273
+ return;
274
+ }
275
+ buffer += value;
276
+ const source = options?.heal ? heal(buffer) : buffer;
277
+ controller.enqueue(renderToHtml(source));
278
+ } catch (error) {
279
+ controller.error(error);
280
+ }
281
+ },
282
+ cancel() {
283
+ reader.cancel();
284
+ }
285
+ });
286
+ }
287
+ //#endregion
288
+ export { MD4X_RENDERER_OPTIONS, Md4xContentRendererService, Md4xWasmContentRendererService, streamMarkdown, withMd4xRenderer, withMd4xWasmRenderer };
289
+
290
+ //# sourceMappingURL=analogjs-content-md4x.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"analogjs-content-md4x.mjs","names":[],"sources":["../../md4x/src/lib/md4x-content-renderer.service.ts","../../md4x/src/lib/md4x-wasm-content-renderer.service.ts","../../md4x/src/lib/provide-md4x.ts","../../md4x/src/lib/streaming-markdown-renderer.ts"],"sourcesContent":["import { Injectable, inject, InjectionToken } from '@angular/core';\n\nimport {\n ContentRenderer,\n RenderedContent,\n TableOfContentItem,\n} from '../../../src/lib/content-renderer';\n\n/**\n * Options for the experimental md4x-based content renderer.\n *\n * @experimental md4x integration is experimental and may change in future releases.\n */\nexport interface Md4xRendererOptions {\n /** Heal incomplete markdown (useful for streaming/LLM content). */\n heal?: boolean;\n /** Custom code block highlighter. Receives raw code and block metadata,\n * returns highlighted HTML or undefined to keep default rendering. */\n highlighter?: (\n code: string,\n block: { lang: string; filename?: string; highlights?: number[] },\n ) => string | undefined;\n}\n\nexport const MD4X_RENDERER_OPTIONS: InjectionToken<Md4xRendererOptions> =\n new InjectionToken<Md4xRendererOptions>('md4x_renderer_options');\n\nfunction makeSlug(text: string, slugCounts: Map<string, number>): string {\n const baseSlug = text\n .toLowerCase()\n .replace(/[^\\w\\s-]/g, '')\n .trim()\n .replace(/\\s+/g, '-');\n const count = slugCounts.get(baseSlug) ?? 0;\n slugCounts.set(baseSlug, count + 1);\n return count === 0 ? baseSlug : `${baseSlug}-${count}`;\n}\n\nfunction headingsToToc(\n headings: Array<{ level: number; text: string }>,\n): TableOfContentItem[] {\n const slugCounts = new Map<string, number>();\n return headings.map((h) => ({\n id: makeSlug(h.text, slugCounts),\n level: h.level,\n text: h.text,\n }));\n}\n\nfunction injectHeadingIds(html: string, toc: TableOfContentItem[]): string {\n let tocIndex = 0;\n return html.replace(/<h([1-6])>/g, (match, level) => {\n const item = toc[tocIndex++];\n return item ? `<h${level} id=\"${item.id}\">` : match;\n });\n}\n\n/**\n * Content renderer backed by md4x (C-based CommonMark parser compiled with Zig).\n * 50-70x faster than marked for complex documents.\n *\n * @experimental md4x integration is experimental and may change in future releases.\n */\n@Injectable()\nexport class Md4xContentRendererService extends ContentRenderer {\n private options = inject(MD4X_RENDERER_OPTIONS, { optional: true });\n\n override async render(content: string): Promise<RenderedContent> {\n const { renderToHtml, parseMeta } = await import('md4x/napi');\n const html = renderToHtml(content, {\n heal: this.options?.heal,\n highlighter: this.options?.highlighter,\n });\n const meta = parseMeta(content);\n const toc = headingsToToc(meta.headings);\n return {\n content: injectHeadingIds(html, toc),\n toc,\n };\n }\n\n override getContentHeadings(content: string): TableOfContentItem[] {\n // Synchronous fallback — md4x is async-imported so we use regex extraction\n // matching NoopContentRenderer's algorithm for consistency.\n const lines = content.split('\\n');\n const toc: TableOfContentItem[] = [];\n const slugCounts = new Map<string, number>();\n\n for (const line of lines) {\n const match = /^(#{1,6})\\s+(.+?)\\s*$/.exec(line);\n if (!match) continue;\n const level = match[1].length;\n const text = match[2].trim();\n if (!text) continue;\n toc.push({ id: makeSlug(text, slugCounts), level, text });\n }\n\n return toc;\n }\n}\n","import { Injectable, inject } from '@angular/core';\n\nimport {\n ContentRenderer,\n RenderedContent,\n TableOfContentItem,\n} from '../../../src/lib/content-renderer';\nimport {\n MD4X_RENDERER_OPTIONS,\n Md4xRendererOptions,\n} from './md4x-content-renderer.service';\n\nfunction makeSlug(text: string, slugCounts: Map<string, number>): string {\n const baseSlug = text\n .toLowerCase()\n .replace(/[^\\w\\s-]/g, '')\n .trim()\n .replace(/\\s+/g, '-');\n const count = slugCounts.get(baseSlug) ?? 0;\n slugCounts.set(baseSlug, count + 1);\n return count === 0 ? baseSlug : `${baseSlug}-${count}`;\n}\n\nfunction headingsToToc(\n headings: Array<{ level: number; text: string }>,\n): TableOfContentItem[] {\n const slugCounts = new Map<string, number>();\n return headings.map((h) => ({\n id: makeSlug(h.text, slugCounts),\n level: h.level,\n text: h.text,\n }));\n}\n\nfunction injectHeadingIds(html: string, toc: TableOfContentItem[]): string {\n let tocIndex = 0;\n return html.replace(/<h([1-6])>/g, (match, level) => {\n const item = toc[tocIndex++];\n return item ? `<h${level} id=\"${item.id}\">` : match;\n });\n}\n\n/**\n * Content renderer backed by md4x/wasm for client-side (browser) rendering.\n * ~100KB gzip, 3-6x faster than marked in the browser.\n *\n * @experimental md4x integration is experimental and may change in future releases.\n */\n@Injectable()\nexport class Md4xWasmContentRendererService extends ContentRenderer {\n private options = inject(MD4X_RENDERER_OPTIONS, { optional: true });\n private initPromise: Promise<void> | null = null;\n\n override async render(content: string): Promise<RenderedContent> {\n const wasm = await import('md4x/wasm');\n if (!this.initPromise) {\n this.initPromise = wasm.init();\n }\n await this.initPromise;\n\n const html = wasm.renderToHtml(content, {\n heal: this.options?.heal,\n highlighter: this.options?.highlighter,\n });\n const meta = wasm.parseMeta(content);\n const toc = headingsToToc(meta.headings);\n return {\n content: injectHeadingIds(html, toc),\n toc,\n };\n }\n\n override getContentHeadings(content: string): TableOfContentItem[] {\n const lines = content.split('\\n');\n const toc: TableOfContentItem[] = [];\n const slugCounts = new Map<string, number>();\n\n for (const line of lines) {\n const match = /^(#{1,6})\\s+(.+?)\\s*$/.exec(line);\n if (!match) continue;\n const level = match[1].length;\n const text = match[2].trim();\n if (!text) continue;\n toc.push({ id: makeSlug(text, slugCounts), level, text });\n }\n\n return toc;\n }\n}\n","import { Provider } from '@angular/core';\n\nimport { ContentRenderer } from '../../../src/lib/content-renderer';\nimport { withContentFileLoader } from '../../../src/lib/content-file-loader';\nimport { withContentListLoader } from '../../../src/lib/content-list-loader';\nimport {\n Md4xContentRendererService,\n MD4X_RENDERER_OPTIONS,\n} from './md4x-content-renderer.service';\nimport { Md4xWasmContentRendererService } from './md4x-wasm-content-renderer.service';\nimport type { Md4xRendererOptions } from './md4x-content-renderer.service';\n\n/**\n * Provides the experimental md4x-based content renderer (NAPI, server/build-time).\n *\n * @experimental md4x integration is experimental and may change in future releases.\n *\n * @example\n * ```typescript\n * provideContent(withMd4xRenderer());\n * provideContent(withMd4xRenderer({ heal: true }));\n * ```\n */\nexport function withMd4xRenderer(options?: Md4xRendererOptions): Provider {\n return [\n { provide: ContentRenderer, useClass: Md4xContentRendererService },\n options ? { provide: MD4X_RENDERER_OPTIONS, useValue: options } : [],\n withContentFileLoader(),\n withContentListLoader(),\n ];\n}\n\n/**\n * Provides the experimental md4x WASM content renderer (browser/CSR).\n * ~100KB gzip, 3-6x faster than marked in the browser.\n *\n * @experimental md4x integration is experimental and may change in future releases.\n *\n * @example\n * ```typescript\n * provideContent(withMd4xWasmRenderer());\n * ```\n */\nexport function withMd4xWasmRenderer(options?: Md4xRendererOptions): Provider {\n return [\n { provide: ContentRenderer, useClass: Md4xWasmContentRendererService },\n options ? { provide: MD4X_RENDERER_OPTIONS, useValue: options } : [],\n withContentFileLoader(),\n withContentListLoader(),\n ];\n}\n","/**\n * Transforms a stream of markdown chunks into a stream of rendered HTML.\n * Uses md4x's `heal()` to fix incomplete markdown from streaming sources\n * (LLMs, collaborative editing) so each emitted HTML chunk is valid.\n *\n * @experimental Streaming markdown support is experimental and may change in future releases.\n *\n * @example\n * ```typescript\n * // In a Nitro API route\n * import { streamMarkdown } from '@analogjs/content';\n *\n * export default defineEventHandler(async (event) => {\n * const llmStream = getAIStream(prompt);\n * return streamMarkdown(llmStream, { heal: true });\n * });\n * ```\n */\nexport async function streamMarkdown(\n input: ReadableStream<string>,\n options?: { heal?: boolean },\n): Promise<ReadableStream<string>> {\n const { renderToHtml, heal } = await import('md4x/napi');\n\n let buffer = '';\n const reader = input.getReader();\n\n return new ReadableStream<string>({\n async pull(controller) {\n try {\n const { done, value } = await reader.read();\n\n if (done) {\n if (buffer) {\n const source = options?.heal ? heal(buffer) : buffer;\n controller.enqueue(renderToHtml(source));\n }\n controller.close();\n return;\n }\n\n buffer += value;\n const source = options?.heal ? heal(buffer) : buffer;\n controller.enqueue(renderToHtml(source));\n } catch (error) {\n controller.error(error);\n }\n },\n cancel() {\n reader.cancel();\n },\n });\n}\n"],"mappings":"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"}
@@ -0,0 +1,170 @@
1
+ import * as i0 from "@angular/core";
2
+ import { Directive, ElementRef, InjectionToken, Renderer2, ViewContainerRef, effect, inject, input } from "@angular/core";
3
+ //#region packages/content/mdc/src/lib/mdc-component-registry.ts
4
+ /**
5
+ * Registry mapping MDC component names to lazy-loaded Angular components.
6
+ *
7
+ * @experimental MDC component support is experimental and may change in future releases.
8
+ */
9
+ var MDC_COMPONENTS = new InjectionToken("mdc_components");
10
+ /**
11
+ * Provides a registry of Angular components that can be used in MDC
12
+ * (Markdown Components) syntax within markdown content.
13
+ *
14
+ * @experimental MDC component support is experimental and may change in future releases.
15
+ *
16
+ * @example
17
+ * ```typescript
18
+ * provideContent(
19
+ * withMd4xRenderer(),
20
+ * withMdcComponents({
21
+ * alert: () => import('./components/alert.component').then(m => m.AlertComponent),
22
+ * card: () => import('./components/card.component').then(m => m.CardComponent),
23
+ * }),
24
+ * );
25
+ * ```
26
+ */
27
+ function withMdcComponents(components) {
28
+ return {
29
+ provide: MDC_COMPONENTS,
30
+ useValue: new Map(Object.entries(components))
31
+ };
32
+ }
33
+ //#endregion
34
+ //#region packages/content/mdc/src/lib/mdc-renderer.directive.ts
35
+ /**
36
+ * Directive that renders MDC (Markdown Components) AST nodes as Angular components.
37
+ *
38
+ * Walks the ComarkTree AST from md4x's `parseAST()`, matches component nodes
39
+ * to the registered MDC_COMPONENTS map, and instantiates them via
40
+ * `ViewContainerRef.createComponent()` with MDC attributes bound as inputs.
41
+ *
42
+ * @experimental MDC component support is experimental and may change in future releases.
43
+ *
44
+ * @example
45
+ * ```html
46
+ * <div [mdcAst]="parsedAst"></div>
47
+ * ```
48
+ */
49
+ var MdcRendererDirective = class MdcRendererDirective {
50
+ constructor() {
51
+ this.ast = input(null, { alias: "mdcAst" });
52
+ this.viewContainer = inject(ViewContainerRef);
53
+ this.renderer = inject(Renderer2);
54
+ this.el = inject(ElementRef);
55
+ this.components = inject(MDC_COMPONENTS, { optional: true });
56
+ this.renderId = 0;
57
+ effect(() => {
58
+ const ast = this.ast();
59
+ const currentRenderId = ++this.renderId;
60
+ this.viewContainer.clear();
61
+ const host = this.el.nativeElement;
62
+ host.innerHTML = "";
63
+ if (!ast?.nodes) return;
64
+ this.renderNodes(ast.nodes, host, currentRenderId);
65
+ });
66
+ }
67
+ async renderNodes(nodes, parent, renderId) {
68
+ for (const node of nodes) {
69
+ if (this.renderId !== renderId) return;
70
+ await this.renderNode(node, parent, renderId);
71
+ }
72
+ }
73
+ async renderNode(node, parent, renderId) {
74
+ if (this.renderId !== renderId) return;
75
+ if (typeof node === "string") {
76
+ const text = this.renderer.createText(node);
77
+ this.renderer.appendChild(parent, text);
78
+ return;
79
+ }
80
+ const [tag, props, ...children] = node;
81
+ if (!tag) {
82
+ for (const child of children) {
83
+ if (this.renderId !== renderId) return;
84
+ await this.renderNode(child, parent, renderId);
85
+ }
86
+ return;
87
+ }
88
+ const componentLoader = this.components?.get(tag);
89
+ if (componentLoader) {
90
+ try {
91
+ const componentType = await componentLoader();
92
+ if (this.renderId !== renderId) return;
93
+ const tempContainer = this.renderer.createElement("div");
94
+ for (const child of children) {
95
+ if (this.renderId !== renderId) return;
96
+ await this.renderNode(child, tempContainer, renderId);
97
+ }
98
+ if (this.renderId !== renderId) return;
99
+ const projectableNodes = [Array.from(tempContainer.childNodes)];
100
+ const componentRef = this.viewContainer.createComponent(componentType, { projectableNodes });
101
+ for (const [key, value] of Object.entries(props)) componentRef.setInput(key, value);
102
+ const componentEl = componentRef.location.nativeElement;
103
+ this.renderer.appendChild(parent, componentEl);
104
+ } catch (e) {
105
+ console.error(`[MdcRenderer] Failed to load component "${tag}":`, e);
106
+ }
107
+ return;
108
+ }
109
+ const el = this.renderer.createElement(tag);
110
+ for (const [key, value] of Object.entries(props)) this.renderer.setAttribute(el, key, String(value));
111
+ for (const child of children) {
112
+ if (this.renderId !== renderId) return;
113
+ await this.renderNode(child, el, renderId);
114
+ }
115
+ this.renderer.appendChild(parent, el);
116
+ }
117
+ static {
118
+ this.ɵfac = i0.ɵɵngDeclareFactory({
119
+ minVersion: "12.0.0",
120
+ version: "21.2.6",
121
+ ngImport: i0,
122
+ type: MdcRendererDirective,
123
+ deps: [],
124
+ target: i0.ɵɵFactoryTarget.Directive
125
+ });
126
+ }
127
+ static {
128
+ this.ɵdir = i0.ɵɵngDeclareDirective({
129
+ minVersion: "17.1.0",
130
+ version: "21.2.6",
131
+ type: MdcRendererDirective,
132
+ isStandalone: true,
133
+ selector: "[mdcAst]",
134
+ inputs: { ast: {
135
+ classPropertyName: "ast",
136
+ publicName: "mdcAst",
137
+ isSignal: true,
138
+ isRequired: false,
139
+ transformFunction: null
140
+ } },
141
+ ngImport: i0
142
+ });
143
+ }
144
+ };
145
+ i0.ɵɵngDeclareClassMetadata({
146
+ minVersion: "12.0.0",
147
+ version: "21.2.6",
148
+ ngImport: i0,
149
+ type: MdcRendererDirective,
150
+ decorators: [{
151
+ type: Directive,
152
+ args: [{
153
+ selector: "[mdcAst]",
154
+ standalone: true
155
+ }]
156
+ }],
157
+ ctorParameters: () => [],
158
+ propDecorators: { ast: [{
159
+ type: i0.Input,
160
+ args: [{
161
+ isSignal: true,
162
+ alias: "mdcAst",
163
+ required: false
164
+ }]
165
+ }] }
166
+ });
167
+ //#endregion
168
+ export { MDC_COMPONENTS, MdcRendererDirective, withMdcComponents };
169
+
170
+ //# sourceMappingURL=analogjs-content-mdc.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"analogjs-content-mdc.mjs","names":[],"sources":["../../mdc/src/lib/mdc-component-registry.ts","../../mdc/src/lib/mdc-renderer.directive.ts"],"sourcesContent":["import { InjectionToken, Type, Provider } from '@angular/core';\n\n/**\n * Registry mapping MDC component names to lazy-loaded Angular components.\n *\n * @experimental MDC component support is experimental and may change in future releases.\n */\nexport const MDC_COMPONENTS: InjectionToken<\n Map<string, () => Promise<Type<unknown>>>\n> = new InjectionToken('mdc_components');\n\n/**\n * Provides a registry of Angular components that can be used in MDC\n * (Markdown Components) syntax within markdown content.\n *\n * @experimental MDC component support is experimental and may change in future releases.\n *\n * @example\n * ```typescript\n * provideContent(\n * withMd4xRenderer(),\n * withMdcComponents({\n * alert: () => import('./components/alert.component').then(m => m.AlertComponent),\n * card: () => import('./components/card.component').then(m => m.CardComponent),\n * }),\n * );\n * ```\n */\nexport function withMdcComponents(\n components: Record<string, () => Promise<Type<unknown>>>,\n): Provider {\n return {\n provide: MDC_COMPONENTS,\n useValue: new Map(Object.entries(components)),\n };\n}\n","import {\n Directive,\n ElementRef,\n effect,\n inject,\n input,\n InputSignal,\n Renderer2,\n ViewContainerRef,\n} from '@angular/core';\n\nimport { MDC_COMPONENTS } from './mdc-component-registry';\n\ntype ComarkNode =\n | string\n | [string | null, Record<string, unknown>, ...ComarkNode[]];\n\n/**\n * Directive that renders MDC (Markdown Components) AST nodes as Angular components.\n *\n * Walks the ComarkTree AST from md4x's `parseAST()`, matches component nodes\n * to the registered MDC_COMPONENTS map, and instantiates them via\n * `ViewContainerRef.createComponent()` with MDC attributes bound as inputs.\n *\n * @experimental MDC component support is experimental and may change in future releases.\n *\n * @example\n * ```html\n * <div [mdcAst]=\"parsedAst\"></div>\n * ```\n */\n@Directive({\n // eslint-disable-next-line @angular-eslint/directive-selector\n selector: '[mdcAst]',\n standalone: true,\n})\nexport class MdcRendererDirective {\n readonly ast: InputSignal<{ nodes: ComarkNode[] } | null> = input<{\n nodes: ComarkNode[];\n } | null>(null, {\n alias: 'mdcAst',\n });\n\n private readonly viewContainer = inject(ViewContainerRef);\n private readonly renderer = inject(Renderer2);\n private readonly el = inject(ElementRef);\n private readonly components = inject(MDC_COMPONENTS, { optional: true });\n private renderId = 0;\n\n constructor() {\n effect(() => {\n const ast = this.ast();\n const currentRenderId = ++this.renderId;\n\n this.viewContainer.clear();\n const host = this.el.nativeElement as HTMLElement;\n host.innerHTML = '';\n\n if (!ast?.nodes) return;\n\n // Fire-and-forget: Angular doesn't await effects, so we schedule\n // the async rendering and let it complete in the background.\n void this.renderNodes(ast.nodes, host, currentRenderId);\n });\n }\n\n private async renderNodes(\n nodes: ComarkNode[],\n parent: HTMLElement,\n renderId: number,\n ): Promise<void> {\n for (const node of nodes) {\n if (this.renderId !== renderId) return;\n await this.renderNode(node, parent, renderId);\n }\n }\n\n private async renderNode(\n node: ComarkNode,\n parent: HTMLElement,\n renderId: number,\n ): Promise<void> {\n if (this.renderId !== renderId) return;\n\n if (typeof node === 'string') {\n const text = this.renderer.createText(node);\n this.renderer.appendChild(parent, text);\n return;\n }\n\n const [tag, props, ...children] = node;\n\n if (!tag) {\n for (const child of children) {\n if (this.renderId !== renderId) return;\n await this.renderNode(child, parent, renderId);\n }\n return;\n }\n\n // Check if this tag is a registered MDC component\n const componentLoader = this.components?.get(tag);\n if (componentLoader) {\n try {\n const componentType = await componentLoader();\n if (this.renderId !== renderId) return;\n\n const tempContainer = this.renderer.createElement('div');\n for (const child of children) {\n if (this.renderId !== renderId) return;\n await this.renderNode(child, tempContainer, renderId);\n }\n if (this.renderId !== renderId) return;\n\n const projectableNodes = [\n Array.from((tempContainer as HTMLElement).childNodes),\n ];\n const componentRef = this.viewContainer.createComponent(componentType, {\n projectableNodes,\n });\n\n // Bind MDC attributes as component inputs\n for (const [key, value] of Object.entries(props)) {\n componentRef.setInput(key, value);\n }\n\n const componentEl = componentRef.location.nativeElement as HTMLElement;\n this.renderer.appendChild(parent, componentEl);\n } catch (e) {\n console.error(`[MdcRenderer] Failed to load component \"${tag}\":`, e);\n }\n return;\n }\n\n // Fall back to rendering as a standard HTML element\n const el = this.renderer.createElement(tag);\n for (const [key, value] of Object.entries(props)) {\n this.renderer.setAttribute(el, key, String(value));\n }\n for (const child of children) {\n if (this.renderId !== renderId) return;\n await this.renderNode(child, el, renderId);\n }\n this.renderer.appendChild(parent, el);\n }\n}\n"],"mappings":";;;;;;;;AAOA,IAAa,iBAET,IAAI,eAAe,iBAAiB;;;;;;;;;;;;;;;;;;AAmBxC,SAAgB,kBACd,YACU;AACV,QAAO;EACL,SAAS;EACT,UAAc,IAAI,IAAA,OAAO,QAAQ,WAAW,CAAA;EAC7C;;;;;;;;;;;;;;;;;;ACsBG,IAAK,uBAAL,MAAK,qBAAY;CAEjB,cAAU;AAIL,OAAK,MAAA,MAAY,MAAI,EAA6B,OAAA,UAAA,CAAA;AACvD,OAAA,gBAAA,OAAA,iBAAA;;AAGU,OAAA,KAAA,OACZ,WAEA;AAEK,OAAM,aAAQ,OAAO,gBAAA,EAAA,UAAA,MAAA,CAAA;AACpB,OAAK,WAAA;AACT,eAAW;;;AAID,QAAA,cAEZ,OACA;GAES,MAAA,OAAa,KAAA,GAAU;AAE5B,QAAO,YAAS;AACZ,OAAO,CAAA,KAAK,MACb;AAIK,QAAO,YAAG,IAAY,OAAA,MAAA,gBAAA;IAE7B;;CAED,MAAI,YAAK,OAAa,QAAU,UAAA;AAChC,OAAM,MAAK,QAAW,OAAO;kCAE/B;;;;CAMA,MAAI,WAAA,MAAA,QAAA,UAAA;AACF,MAAM,KAAA,aAAgB,SAClB;AAEJ,MAAM,OAAA,SAAgB,UAAK;GACtB,MAAM,OAAS,KAAA,SAAU,WAAA,KAAA;AACxB,QAAK,SAAA,YAAuB,QAAA,KAAA;AAC1B;;EAEJ,MAAK,CAAA,KAAA,OAAa,GAAA,YAAU;AAEhC,MAAM,CAAA,KAAA;AAGA,QAAA,MAAA,SAAoB,UAAA;AAKd,QAAA,KAAK,aAAiB,SACnB;;;AAIV;;;AAIP,MAAA,iBAAA;;IAIS,MAAK,gBAAS,MAAc,iBAAI;AAC/B,QAAK,KAAA,aAAiB,SAClB;;AAEL,SAAA,MAAS,SAAU,UAAA;AACnB,SAAA,KAAa,aAAU,SACrB;;;;6BA9GhB,CAEW,MAAA,KAAA,cAAA,WAAA,CACE;IACZ,MAAA,eAAA,KAAA,cAAA,gBAAA,eAAA,EAAA,kBAAA,CAAA"}
@@ -1,48 +1,36 @@
1
- import satori from 'satori';
2
- import { html } from 'satori-html';
3
- import sharp from 'sharp';
4
-
5
- // Credit for modified source: https://github.com/etherCorps/sveltekit-og/blob/main/src/lib/api.ts
6
- const generateImage = async (element, options) => {
7
- const elementHtml = html(element);
8
- const svg = await satori(elementHtml, {
9
- width: options.width || 1200,
10
- height: options.height || 630,
11
- fonts: options.fonts?.length ? options.fonts : [],
12
- tailwindConfig: options.tailwindConfig,
13
- });
14
- const svgBuffer = Buffer.from(svg);
15
- const png = sharp(svgBuffer).png().toBuffer();
16
- const pngBuffer = await png;
17
- return pngBuffer;
1
+ import satori from "satori";
2
+ import { html } from "satori-html";
3
+ import sharp from "sharp";
4
+ //#region packages/content/og/src/lib/og.ts
5
+ var generateImage = async (element, options) => {
6
+ const svg = await satori(html(element), {
7
+ width: options.width || 1200,
8
+ height: options.height || 630,
9
+ fonts: options.fonts?.length ? options.fonts : [],
10
+ tailwindConfig: options.tailwindConfig
11
+ });
12
+ return await sharp(Buffer.from(svg)).png().toBuffer();
18
13
  };
19
- class ImageResponse extends Response {
20
- constructor(element, options = {}) {
21
- super();
22
- const body = new ReadableStream({
23
- async start(controller) {
24
- const buffer = await generateImage(element, options);
25
- controller.enqueue(buffer);
26
- controller.close();
27
- },
28
- });
29
- return new Response(body, {
30
- headers: {
31
- 'Content-Type': 'image/png',
32
- 'Cache-Control': options.debug
33
- ? 'no-cache, no-store'
34
- : 'public, immutable, no-transform, max-age=31536000',
35
- ...options.headers,
36
- },
37
- status: options.status || 200,
38
- statusText: options.statusText,
39
- });
40
- }
41
- }
42
-
43
- /**
44
- * Generated bundle index. Do not edit.
45
- */
46
-
14
+ var ImageResponse = class extends Response {
15
+ constructor(element, options = {}) {
16
+ super();
17
+ const body = new ReadableStream({ async start(controller) {
18
+ const buffer = await generateImage(element, options);
19
+ controller.enqueue(buffer);
20
+ controller.close();
21
+ } });
22
+ return new Response(body, {
23
+ headers: {
24
+ "Content-Type": "image/png",
25
+ "Cache-Control": options.debug ? "no-cache, no-store" : "public, immutable, no-transform, max-age=31536000",
26
+ ...options.headers
27
+ },
28
+ status: options.status || 200,
29
+ statusText: options.statusText
30
+ });
31
+ }
32
+ };
33
+ //#endregion
47
34
  export { ImageResponse };
48
- //# sourceMappingURL=analogjs-content-og.mjs.map
35
+
36
+ //# sourceMappingURL=analogjs-content-og.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"analogjs-content-og.mjs","sources":["../../../../packages/content/og/src/lib/og.ts","../../../../packages/content/og/src/analogjs-content-og.ts"],"sourcesContent":["// Credit for modified source: https://github.com/etherCorps/sveltekit-og/blob/main/src/lib/api.ts\n\nimport satori from 'satori';\nimport { html as toReactElement } from 'satori-html';\nimport sharp from 'sharp';\n\nimport { ImageResponseOptions } from './options';\n\nexport const generateImage = async (\n element: string,\n options: ImageResponseOptions,\n) => {\n const elementHtml = toReactElement(element);\n const svg = await satori(elementHtml as any, {\n width: options.width || 1200,\n height: options.height || 630,\n fonts: options.fonts?.length ? options.fonts : [],\n tailwindConfig: options.tailwindConfig,\n });\n const svgBuffer = Buffer.from(svg);\n const png = sharp(svgBuffer).png().toBuffer();\n\n const pngBuffer = await png;\n\n return pngBuffer;\n};\n\nexport class ImageResponse extends Response {\n constructor(element: string, options: ImageResponseOptions = {}) {\n super();\n\n const body = new ReadableStream({\n async start(controller) {\n const buffer = await generateImage(element, options);\n controller.enqueue(buffer);\n controller.close();\n },\n });\n\n return new Response(body, {\n headers: {\n 'Content-Type': 'image/png',\n 'Cache-Control': options.debug\n ? 'no-cache, no-store'\n : 'public, immutable, no-transform, max-age=31536000',\n ...options.headers,\n },\n status: options.status || 200,\n statusText: options.statusText,\n });\n }\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":["toReactElement"],"mappings":";;;;AAAA;AAQO,MAAM,aAAa,GAAG,OAC3B,OAAe,EACf,OAA6B,KAC3B;AACF,IAAA,MAAM,WAAW,GAAGA,IAAc,CAAC,OAAO,CAAC;AAC3C,IAAA,MAAM,GAAG,GAAG,MAAM,MAAM,CAAC,WAAkB,EAAE;AAC3C,QAAA,KAAK,EAAE,OAAO,CAAC,KAAK,IAAI,IAAI;AAC5B,QAAA,MAAM,EAAE,OAAO,CAAC,MAAM,IAAI,GAAG;AAC7B,QAAA,KAAK,EAAE,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,KAAK,GAAG,EAAE;QACjD,cAAc,EAAE,OAAO,CAAC,cAAc;AACvC,KAAA,CAAC;IACF,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC;AAClC,IAAA,MAAM,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE;AAE7C,IAAA,MAAM,SAAS,GAAG,MAAM,GAAG;AAE3B,IAAA,OAAO,SAAS;AAClB,CAAC;AAEK,MAAO,aAAc,SAAQ,QAAQ,CAAA;IACzC,WAAA,CAAY,OAAe,EAAE,OAAA,GAAgC,EAAE,EAAA;AAC7D,QAAA,KAAK,EAAE;AAEP,QAAA,MAAM,IAAI,GAAG,IAAI,cAAc,CAAC;YAC9B,MAAM,KAAK,CAAC,UAAU,EAAA;gBACpB,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,OAAO,EAAE,OAAO,CAAC;AACpD,gBAAA,UAAU,CAAC,OAAO,CAAC,MAAM,CAAC;gBAC1B,UAAU,CAAC,KAAK,EAAE;YACpB,CAAC;AACF,SAAA,CAAC;AAEF,QAAA,OAAO,IAAI,QAAQ,CAAC,IAAI,EAAE;AACxB,YAAA,OAAO,EAAE;AACP,gBAAA,cAAc,EAAE,WAAW;gBAC3B,eAAe,EAAE,OAAO,CAAC;AACvB,sBAAE;AACF,sBAAE,mDAAmD;gBACvD,GAAG,OAAO,CAAC,OAAO;AACnB,aAAA;AACD,YAAA,MAAM,EAAE,OAAO,CAAC,MAAM,IAAI,GAAG;YAC7B,UAAU,EAAE,OAAO,CAAC,UAAU;AAC/B,SAAA,CAAC;IACJ;AACD;;ACnDD;;AAEG;;;;"}
1
+ {"version":3,"file":"analogjs-content-og.mjs","names":[],"sources":["../../og/src/lib/og.ts"],"sourcesContent":["// Credit for modified source: https://github.com/etherCorps/sveltekit-og/blob/main/src/lib/api.ts\n\nimport satori from 'satori';\nimport { html as toReactElement } from 'satori-html';\nimport sharp from 'sharp';\n\nimport { ImageResponseOptions } from './options';\n\nexport const generateImage = async (\n element: string,\n options: ImageResponseOptions,\n): Promise<Buffer> => {\n const elementHtml = toReactElement(element);\n const svg = await satori(elementHtml as any, {\n width: options.width || 1200,\n height: options.height || 630,\n fonts: options.fonts?.length ? options.fonts : [],\n tailwindConfig: options.tailwindConfig,\n });\n const svgBuffer = Buffer.from(svg);\n const png = sharp(svgBuffer).png().toBuffer();\n\n const pngBuffer = await png;\n\n return pngBuffer;\n};\n\nexport class ImageResponse extends Response {\n constructor(element: string, options: ImageResponseOptions = {}) {\n super();\n\n const body = new ReadableStream({\n async start(controller) {\n const buffer = await generateImage(element, options);\n controller.enqueue(buffer);\n controller.close();\n },\n });\n\n return new Response(body, {\n headers: {\n 'Content-Type': 'image/png',\n 'Cache-Control': options.debug\n ? 'no-cache, no-store'\n : 'public, immutable, no-transform, max-age=31536000',\n ...options.headers,\n },\n status: options.status || 200,\n statusText: options.statusText,\n });\n }\n}\n"],"mappings":";;;;IAYQ,gBAAc,OAAe,SAAQ,YAAA;CAEzC,MAAO,MAAA,MAAQ,OADC,KAA2B,QAAA,EACnB;EACxB,OAAQ,QAAQ,SAAU;EAC1B,QAAO,QAAe,UAAS;EAC/B,OAAA,QAAgB,OAAQ,SAAA,QAAA,QAAA,EAAA;EACxB,gBAAA,QAAA;EACF,CAAA;QAKO,MAFD,MAFY,OAAU,KAAC,IAAM,CAEX,CAAA,KAAA,CAAA,UAAA;;IAMZ,gBAAA,cAAmD,SAAE;CAC/D,YAAO,SAAA,UAAA,EAAA,EAAA;AAED,SAAO;EAET,MAAM,OAAS,IAAM,eAAc,EACnC,MAAW,MAAQ,YAAO;GACf,MAAO,SAAA,MAAA,cAAA,SAAA,QAAA;AAEpB,cAAA,QAAA,OAAA;AAES,cAAS,OAAM;KAEtB,CAAA;AACA,SAAA,IAAA,SAAiB,MAAQ;GAGtB,SAAQ;IACZ,gBAAA;IACO,iBAAkB,QAAA,QACd,uBACZ"}