mulmocast 2.2.0 → 2.2.2

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 (49) hide show
  1. package/assets/schemas/html_prompt.json +60 -0
  2. package/assets/schemas/mulmo_script.json +283 -0
  3. package/assets/slide_themes/corporate.json +18 -0
  4. package/assets/slide_themes/creative.json +18 -0
  5. package/assets/slide_themes/dark.json +18 -0
  6. package/assets/slide_themes/minimal.json +18 -0
  7. package/assets/slide_themes/pop.json +18 -0
  8. package/assets/slide_themes/warm.json +18 -0
  9. package/assets/styles/akira_comic.json +22 -0
  10. package/assets/styles/ani.json +43 -0
  11. package/assets/styles/children_book.json +13 -0
  12. package/assets/styles/comic_strips.json +13 -0
  13. package/assets/styles/drslump_comic.json +22 -0
  14. package/assets/styles/ghibli_comic.json +22 -0
  15. package/assets/styles/ghibli_shorts.json +28 -0
  16. package/assets/styles/ghost_comic.json +29 -0
  17. package/assets/styles/leda.json +29 -0
  18. package/assets/styles/onepiece_comic.json +22 -0
  19. package/assets/styles/slide_corporate.json +30 -0
  20. package/assets/styles/slide_creative.json +30 -0
  21. package/assets/styles/slide_dark.json +30 -0
  22. package/assets/styles/slide_minimal.json +30 -0
  23. package/assets/styles/slide_pop.json +30 -0
  24. package/assets/styles/slide_warm.json +30 -0
  25. package/lib/data/index.d.ts +2 -0
  26. package/lib/data/index.js +2 -0
  27. package/lib/data/slideStyles.d.ts +206 -0
  28. package/lib/data/slideStyles.js +206 -0
  29. package/lib/data/slideThemes.d.ts +134 -0
  30. package/lib/data/slideThemes.js +134 -0
  31. package/lib/slide/layouts/columns.js +3 -4
  32. package/lib/slide/layouts/comparison.js +3 -5
  33. package/lib/slide/layouts/grid.js +1 -1
  34. package/lib/slide/layouts/stats.js +20 -7
  35. package/lib/slide/layouts/timeline.js +17 -4
  36. package/lib/slide/render.d.ts +1 -1
  37. package/lib/slide/render.js +5 -1
  38. package/lib/slide/schema.d.ts +1 -0
  39. package/lib/slide/schema.js +1 -0
  40. package/lib/types/schema.d.ts +4 -0
  41. package/lib/types/slide.d.ts +1 -0
  42. package/lib/types/slide.js +1 -0
  43. package/lib/utils/context.d.ts +2 -0
  44. package/lib/utils/image_plugins/slide.js +4 -2
  45. package/package.json +14 -4
  46. package/scripts/test/test_slide_image_ref_en.json +2 -1
  47. package/scripts/test/test_slide_image_ref_gemini_en.json +289 -0
  48. package/scripts/test/test_slide_image_ref_gemini_en.json~ +289 -0
  49. package/scripts/test/test_tts_speed.json +252 -0
@@ -0,0 +1,134 @@
1
+ export declare const slideThemes: {
2
+ corporate: {
3
+ colors: {
4
+ accent: string;
5
+ bg: string;
6
+ bgCard: string;
7
+ bgCardAlt: string;
8
+ danger: string;
9
+ highlight: string;
10
+ info: string;
11
+ primary: string;
12
+ success: string;
13
+ text: string;
14
+ textDim: string;
15
+ textMuted: string;
16
+ warning: string;
17
+ };
18
+ fonts: {
19
+ body: string;
20
+ mono: string;
21
+ title: string;
22
+ };
23
+ };
24
+ creative: {
25
+ colors: {
26
+ accent: string;
27
+ bg: string;
28
+ bgCard: string;
29
+ bgCardAlt: string;
30
+ danger: string;
31
+ highlight: string;
32
+ info: string;
33
+ primary: string;
34
+ success: string;
35
+ text: string;
36
+ textDim: string;
37
+ textMuted: string;
38
+ warning: string;
39
+ };
40
+ fonts: {
41
+ body: string;
42
+ mono: string;
43
+ title: string;
44
+ };
45
+ };
46
+ dark: {
47
+ colors: {
48
+ accent: string;
49
+ bg: string;
50
+ bgCard: string;
51
+ bgCardAlt: string;
52
+ danger: string;
53
+ highlight: string;
54
+ info: string;
55
+ primary: string;
56
+ success: string;
57
+ text: string;
58
+ textDim: string;
59
+ textMuted: string;
60
+ warning: string;
61
+ };
62
+ fonts: {
63
+ body: string;
64
+ mono: string;
65
+ title: string;
66
+ };
67
+ };
68
+ minimal: {
69
+ colors: {
70
+ accent: string;
71
+ bg: string;
72
+ bgCard: string;
73
+ bgCardAlt: string;
74
+ danger: string;
75
+ highlight: string;
76
+ info: string;
77
+ primary: string;
78
+ success: string;
79
+ text: string;
80
+ textDim: string;
81
+ textMuted: string;
82
+ warning: string;
83
+ };
84
+ fonts: {
85
+ body: string;
86
+ mono: string;
87
+ title: string;
88
+ };
89
+ };
90
+ pop: {
91
+ colors: {
92
+ accent: string;
93
+ bg: string;
94
+ bgCard: string;
95
+ bgCardAlt: string;
96
+ danger: string;
97
+ highlight: string;
98
+ info: string;
99
+ primary: string;
100
+ success: string;
101
+ text: string;
102
+ textDim: string;
103
+ textMuted: string;
104
+ warning: string;
105
+ };
106
+ fonts: {
107
+ body: string;
108
+ mono: string;
109
+ title: string;
110
+ };
111
+ };
112
+ warm: {
113
+ colors: {
114
+ accent: string;
115
+ bg: string;
116
+ bgCard: string;
117
+ bgCardAlt: string;
118
+ danger: string;
119
+ highlight: string;
120
+ info: string;
121
+ primary: string;
122
+ success: string;
123
+ text: string;
124
+ textDim: string;
125
+ textMuted: string;
126
+ warning: string;
127
+ };
128
+ fonts: {
129
+ body: string;
130
+ mono: string;
131
+ title: string;
132
+ };
133
+ };
134
+ };
@@ -0,0 +1,134 @@
1
+ export const slideThemes = {
2
+ corporate: {
3
+ colors: {
4
+ accent: "7C3AED",
5
+ bg: "F8FAFC",
6
+ bgCard: "FFFFFF",
7
+ bgCardAlt: "F1F5F9",
8
+ danger: "DC2626",
9
+ highlight: "DB2777",
10
+ info: "0891B2",
11
+ primary: "2563EB",
12
+ success: "16A34A",
13
+ text: "0F172A",
14
+ textDim: "94A3B8",
15
+ textMuted: "475569",
16
+ warning: "EA580C",
17
+ },
18
+ fonts: {
19
+ body: "Helvetica",
20
+ mono: "Menlo",
21
+ title: "Georgia",
22
+ },
23
+ },
24
+ creative: {
25
+ colors: {
26
+ accent: "FB923C",
27
+ bg: "1C1917",
28
+ bgCard: "292524",
29
+ bgCardAlt: "3D3733",
30
+ danger: "EF4444",
31
+ highlight: "E879F9",
32
+ info: "38BDF8",
33
+ primary: "F43F5E",
34
+ success: "4ADE80",
35
+ text: "FAFAF9",
36
+ textDim: "A8A29E",
37
+ textMuted: "D6D3D1",
38
+ warning: "FBBF24",
39
+ },
40
+ fonts: {
41
+ body: "Helvetica",
42
+ mono: "Consolas",
43
+ title: "Georgia",
44
+ },
45
+ },
46
+ dark: {
47
+ colors: {
48
+ accent: "8B5CF6",
49
+ bg: "0F172A",
50
+ bgCard: "1E293B",
51
+ bgCardAlt: "334155",
52
+ danger: "EF4444",
53
+ highlight: "EC4899",
54
+ info: "14B8A6",
55
+ primary: "3B82F6",
56
+ success: "22C55E",
57
+ text: "F8FAFC",
58
+ textDim: "64748B",
59
+ textMuted: "CBD5E1",
60
+ warning: "F59E0B",
61
+ },
62
+ fonts: {
63
+ body: "Calibri",
64
+ mono: "Consolas",
65
+ title: "Georgia",
66
+ },
67
+ },
68
+ minimal: {
69
+ colors: {
70
+ accent: "7C3AED",
71
+ bg: "F8FAFC",
72
+ bgCard: "FFFFFF",
73
+ bgCardAlt: "F1F5F9",
74
+ danger: "DC2626",
75
+ highlight: "DB2777",
76
+ info: "0891B2",
77
+ primary: "2563EB",
78
+ success: "059669",
79
+ text: "0F172A",
80
+ textDim: "94A3B8",
81
+ textMuted: "475569",
82
+ warning: "D97706",
83
+ },
84
+ fonts: {
85
+ body: "Calibri",
86
+ mono: "Consolas",
87
+ title: "Trebuchet MS",
88
+ },
89
+ },
90
+ pop: {
91
+ colors: {
92
+ accent: "7C3AED",
93
+ bg: "FFF0F5",
94
+ bgCard: "FFFFFF",
95
+ bgCardAlt: "FDE8EF",
96
+ danger: "EF4444",
97
+ highlight: "F97316",
98
+ info: "06B6D4",
99
+ primary: "E91E63",
100
+ success: "10B981",
101
+ text: "1A1A2E",
102
+ textDim: "9090B0",
103
+ textMuted: "4A4A6A",
104
+ warning: "F59E0B",
105
+ },
106
+ fonts: {
107
+ body: "Calibri",
108
+ mono: "Consolas",
109
+ title: "Georgia",
110
+ },
111
+ },
112
+ warm: {
113
+ colors: {
114
+ accent: "D946EF",
115
+ bg: "FFFBEB",
116
+ bgCard: "FFFFFF",
117
+ bgCardAlt: "FEF3C7",
118
+ danger: "DC2626",
119
+ highlight: "E11D48",
120
+ info: "0284C7",
121
+ primary: "EA580C",
122
+ success: "16A34A",
123
+ text: "1C1917",
124
+ textDim: "A8A29E",
125
+ textMuted: "57534E",
126
+ warning: "CA8A04",
127
+ },
128
+ fonts: {
129
+ body: "Calibri",
130
+ mono: "Consolas",
131
+ title: "Georgia",
132
+ },
133
+ },
134
+ };
@@ -23,13 +23,12 @@ const buildColumnCard = (col) => {
23
23
  }
24
24
  if (col.content) {
25
25
  const centerCls = col.icon ? "text-center" : "";
26
- inner.push(`<div class="mt-3 space-y-3 flex-1 min-h-0 overflow-hidden flex flex-col ${centerCls}">`);
26
+ inner.push(`<div class="mt-4 space-y-4 flex-1 min-h-0 overflow-auto flex flex-col ${centerCls}">`);
27
27
  inner.push(renderCardContentBlocks(col.content));
28
28
  inner.push(`</div>`);
29
29
  }
30
30
  if (col.footer) {
31
- inner.push(`<div class="flex-1"></div>`);
32
- inner.push(`<p class="text-sm text-d-dim font-body mt-3">${escapeHtml(col.footer)}</p>`);
31
+ inner.push(`<p class="text-sm text-d-dim font-body mt-auto pt-3">${escapeHtml(col.footer)}</p>`);
33
32
  }
34
33
  return cardWrap(accent, inner.join("\n"), "flex-1");
35
34
  };
@@ -43,7 +42,7 @@ export const layoutColumns = (data) => {
43
42
  colElements.push(`<div class="flex items-center shrink-0"><span class="text-2xl text-d-dim">\u25B6</span></div>`);
44
43
  }
45
44
  });
46
- parts.push(`<div class="flex gap-4 px-12 mt-5 flex-1 min-h-0 items-stretch">`);
45
+ parts.push(`<div class="flex gap-4 px-12 mt-5 flex-1 min-h-0 items-start">`);
47
46
  parts.push(colElements.join("\n"));
48
47
  parts.push(`</div>`);
49
48
  if (data.callout) {
@@ -5,20 +5,18 @@ const buildPanel = (panel) => {
5
5
  const inner = [];
6
6
  inner.push(`<h3 class="text-xl font-bold text-${c(accent)} font-body">${escapeHtml(panel.title)}</h3>`);
7
7
  if (panel.content) {
8
- inner.push(`<div class="mt-4 space-y-3 flex-1 min-h-0 overflow-hidden flex flex-col">`);
8
+ inner.push(`<div class="mt-5 space-y-4 flex-1 min-h-0 overflow-auto flex flex-col">`);
9
9
  inner.push(renderContentBlocks(panel.content));
10
10
  inner.push(`</div>`);
11
11
  }
12
12
  if (panel.footer) {
13
- if (!panel.content)
14
- inner.push(`<div class="flex-1"></div>`);
15
- inner.push(`<p class="text-sm text-d-dim font-body mt-3">${escapeHtml(panel.footer)}</p>`);
13
+ inner.push(`<p class="text-sm text-d-dim font-body mt-auto pt-3">${escapeHtml(panel.footer)}</p>`);
16
14
  }
17
15
  return cardWrap(accent, inner.join("\n"), "flex-1");
18
16
  };
19
17
  export const layoutComparison = (data) => {
20
18
  const parts = [slideHeader(data)];
21
- parts.push(`<div class="flex gap-5 px-12 mt-5 flex-1 min-h-0 items-stretch">`);
19
+ parts.push(`<div class="flex gap-5 px-12 mt-5 flex-1 min-h-0 items-start">`);
22
20
  parts.push(buildPanel(data.left));
23
21
  parts.push(buildPanel(data.right));
24
22
  parts.push(`</div>`);
@@ -8,7 +8,7 @@ export const layoutGrid = (data) => {
8
8
  parts.push(`<div class="px-12 pt-5 shrink-0">`);
9
9
  parts.push(` <h2 class="text-[42px] leading-tight font-title font-bold text-d-text">${nl2br(data.title)}</h2>`);
10
10
  parts.push(`</div>`);
11
- parts.push(`<div class="grid grid-cols-${nCols} gap-4 px-12 mt-5 flex-1 min-h-0 overflow-hidden content-start">`);
11
+ parts.push(`<div class="grid grid-cols-${nCols} gap-4 px-12 mt-5 flex-1 min-h-0 overflow-hidden content-center">`);
12
12
  (data.items || []).forEach((item) => {
13
13
  const itemAccent = item.accentColor || "primary";
14
14
  const inner = [];
@@ -1,21 +1,34 @@
1
- import { escapeHtml, c, slideHeader, renderCalloutBar } from "../utils.js";
1
+ import { escapeHtml, nl2br, c, renderCalloutBar } from "../utils.js";
2
2
  export const layoutStats = (data) => {
3
- const parts = [slideHeader(data)];
3
+ const accent = data.accentColor || "primary";
4
4
  const stats = data.stats || [];
5
- parts.push(`<div class="flex gap-6 px-12 mt-8 flex-1 items-start">`);
5
+ const parts = [];
6
+ parts.push(`<div class="h-[3px] bg-${c(accent)} shrink-0"></div>`);
7
+ parts.push(`<div class="flex-1 flex flex-col justify-center px-12 min-h-0">`);
8
+ // Header inside centering wrapper
9
+ if (data.stepLabel) {
10
+ parts.push(`<p class="text-sm font-bold text-${c(accent)} font-body">${escapeHtml(data.stepLabel)}</p>`);
11
+ }
12
+ parts.push(`<h2 class="text-[42px] leading-tight font-title font-bold text-d-text">${nl2br(data.title)}</h2>`);
13
+ if (data.subtitle) {
14
+ parts.push(`<p class="text-[15px] text-d-dim mt-2 font-body">${nl2br(data.subtitle)}</p>`);
15
+ }
16
+ // Stats cards
17
+ parts.push(`<div class="flex gap-6 mt-10">`);
6
18
  stats.forEach((stat) => {
7
19
  const color = stat.color || data.accentColor || "primary";
8
- parts.push(`<div class="flex-1 bg-d-card rounded-lg shadow-lg p-8 text-center">`);
20
+ parts.push(`<div class="flex-1 bg-d-card rounded-lg shadow-lg p-10 text-center">`);
9
21
  parts.push(` <div class="h-[3px] bg-${c(color)} rounded-full w-12 mx-auto mb-6"></div>`);
10
- parts.push(` <p class="text-[48px] font-bold text-${c(color)} font-body leading-none">${escapeHtml(stat.value)}</p>`);
11
- parts.push(` <p class="text-base text-d-muted font-body mt-3">${escapeHtml(stat.label)}</p>`);
22
+ parts.push(` <p class="text-[52px] font-bold text-${c(color)} font-body leading-none">${escapeHtml(stat.value)}</p>`);
23
+ parts.push(` <p class="text-lg text-d-muted font-body mt-4">${escapeHtml(stat.label)}</p>`);
12
24
  if (stat.change) {
13
25
  const changeColor = stat.change.startsWith("+") ? "success" : "danger";
14
- parts.push(` <p class="text-sm font-bold text-${c(changeColor)} font-body mt-2">${escapeHtml(stat.change)}</p>`);
26
+ parts.push(` <p class="text-base font-bold text-${c(changeColor)} font-body mt-3">${escapeHtml(stat.change)}</p>`);
15
27
  }
16
28
  parts.push(`</div>`);
17
29
  });
18
30
  parts.push(`</div>`);
31
+ parts.push(`</div>`);
19
32
  if (data.callout) {
20
33
  parts.push(`<div class="mt-auto pb-4">${renderCalloutBar(data.callout)}</div>`);
21
34
  }
@@ -1,9 +1,21 @@
1
- import { escapeHtml, nl2br, c, slideHeader } from "../utils.js";
1
+ import { escapeHtml, nl2br, c } from "../utils.js";
2
2
  export const layoutTimeline = (data) => {
3
- const parts = [slideHeader(data)];
3
+ const accent = data.accentColor || "primary";
4
+ const parts = [];
4
5
  const items = data.items || [];
5
- parts.push(`<div class="flex items-start px-12 mt-8 flex-1 relative">`);
6
- parts.push(`<div class="absolute left-16 right-16 top-[52px] h-[2px] bg-d-alt"></div>`);
6
+ parts.push(`<div class="h-[3px] bg-${c(accent)} shrink-0"></div>`);
7
+ parts.push(`<div class="flex-1 flex flex-col justify-center px-12 min-h-0">`);
8
+ // Header inside centering wrapper
9
+ if (data.stepLabel) {
10
+ parts.push(`<p class="text-sm font-bold text-${c(accent)} font-body">${escapeHtml(data.stepLabel)}</p>`);
11
+ }
12
+ parts.push(`<h2 class="text-[42px] leading-tight font-title font-bold text-d-text">${nl2br(data.title)}</h2>`);
13
+ if (data.subtitle) {
14
+ parts.push(`<p class="text-[15px] text-d-dim mt-2 font-body">${nl2br(data.subtitle)}</p>`);
15
+ }
16
+ // Timeline items
17
+ parts.push(`<div class="flex items-start mt-10 relative">`);
18
+ parts.push(`<div class="absolute left-4 right-4 top-[52px] h-[2px] bg-d-alt"></div>`);
7
19
  items.forEach((item) => {
8
20
  const color = item.color || data.accentColor || "primary";
9
21
  const dotBorder = item.done ? `bg-${c(color)}` : `bg-d-alt`;
@@ -20,5 +32,6 @@ export const layoutTimeline = (data) => {
20
32
  parts.push(`</div>`);
21
33
  });
22
34
  parts.push(`</div>`);
35
+ parts.push(`</div>`);
23
36
  return parts.join("\n");
24
37
  };
@@ -1,3 +1,3 @@
1
1
  import type { SlideTheme, SlideLayout } from "./schema.js";
2
2
  /** Generate a complete HTML document for a single slide */
3
- export declare const generateSlideHTML: (theme: SlideTheme, slide: SlideLayout) => string;
3
+ export declare const generateSlideHTML: (theme: SlideTheme, slide: SlideLayout, reference?: string) => string;
@@ -22,7 +22,7 @@ const buildCdnScripts = (theme, slide) => {
22
22
  return scripts.join("\n");
23
23
  };
24
24
  /** Generate a complete HTML document for a single slide */
25
- export const generateSlideHTML = (theme, slide) => {
25
+ export const generateSlideHTML = (theme, slide, reference) => {
26
26
  const content = renderSlideContent(slide);
27
27
  const twConfig = buildTailwindConfig(theme);
28
28
  const cdnScripts = buildCdnScripts(theme, slide);
@@ -30,6 +30,9 @@ export const generateSlideHTML = (theme, slide) => {
30
30
  const bgCls = slideStyle?.bgColor ? "" : "bg-d-bg";
31
31
  const inlineStyle = slideStyle?.bgColor ? ` style="background-color:#${sanitizeHex(slideStyle.bgColor)}"` : "";
32
32
  const footer = slideStyle?.footer ? `<p class="absolute bottom-2 right-4 text-xs text-d-dim font-body">${escapeHtml(slideStyle.footer)}</p>` : "";
33
+ const referenceHtml = reference
34
+ ? `<div class="mt-auto px-4 pb-2"><p class="text-sm text-d-muted font-body opacity-80">${escapeHtml(reference)}</p></div>`
35
+ : "";
33
36
  return `<!DOCTYPE html>
34
37
  <html lang="en" class="h-full">
35
38
  <head>
@@ -45,6 +48,7 @@ ${cdnScripts}
45
48
  <body class="h-full">
46
49
  <div class="relative overflow-hidden ${bgCls} w-full h-full flex flex-col"${inlineStyle}>
47
50
  ${content}
51
+ ${referenceHtml}
48
52
  ${footer}
49
53
  </div>
50
54
  </body>
@@ -4421,6 +4421,7 @@ export declare const mulmoSlideMediaSchema: z.ZodObject<{
4421
4421
  }, z.core.$strip>>;
4422
4422
  layout: z.ZodLiteral<"funnel">;
4423
4423
  }, z.core.$strip>], "layout">;
4424
+ reference: z.ZodOptional<z.ZodString>;
4424
4425
  }, z.core.$strict>;
4425
4426
  export type AccentColorKey = z.infer<typeof accentColorKeySchema>;
4426
4427
  export type SlideThemeColors = z.infer<typeof slideThemeColorsSchema>;
@@ -345,5 +345,6 @@ export const mulmoSlideMediaSchema = z
345
345
  type: z.literal("slide"),
346
346
  theme: slideThemeSchema.optional(),
347
347
  slide: slideLayoutSchema,
348
+ reference: z.string().optional(),
348
349
  })
349
350
  .strict();
@@ -1740,6 +1740,7 @@ export declare const mulmoImageAssetSchema: z.ZodUnion<readonly [z.ZodObject<{
1740
1740
  }, z.core.$strip>>;
1741
1741
  layout: z.ZodLiteral<"funnel">;
1742
1742
  }, z.core.$strip>], "layout">;
1743
+ reference: z.ZodOptional<z.ZodString>;
1743
1744
  }, z.core.$strict>]>;
1744
1745
  export declare const mulmoAudioAssetSchema: z.ZodUnion<readonly [z.ZodObject<{
1745
1746
  type: z.ZodLiteral<"audio">;
@@ -3566,6 +3567,7 @@ export declare const mulmoBeatSchema: z.ZodObject<{
3566
3567
  }, z.core.$strip>>;
3567
3568
  layout: z.ZodLiteral<"funnel">;
3568
3569
  }, z.core.$strip>], "layout">;
3570
+ reference: z.ZodOptional<z.ZodString>;
3569
3571
  }, z.core.$strict>]>>;
3570
3572
  audio: z.ZodOptional<z.ZodUnion<readonly [z.ZodObject<{
3571
3573
  type: z.ZodLiteral<"audio">;
@@ -6093,6 +6095,7 @@ export declare const mulmoScriptSchema: z.ZodObject<{
6093
6095
  }, z.core.$strip>>;
6094
6096
  layout: z.ZodLiteral<"funnel">;
6095
6097
  }, z.core.$strip>], "layout">;
6098
+ reference: z.ZodOptional<z.ZodString>;
6096
6099
  }, z.core.$strict>]>>;
6097
6100
  audio: z.ZodOptional<z.ZodUnion<readonly [z.ZodObject<{
6098
6101
  type: z.ZodLiteral<"audio">;
@@ -8281,6 +8284,7 @@ export declare const mulmoStudioSchema: z.ZodObject<{
8281
8284
  }, z.core.$strip>>;
8282
8285
  layout: z.ZodLiteral<"funnel">;
8283
8286
  }, z.core.$strip>], "layout">;
8287
+ reference: z.ZodOptional<z.ZodString>;
8284
8288
  }, z.core.$strict>]>>;
8285
8289
  audio: z.ZodOptional<z.ZodUnion<readonly [z.ZodObject<{
8286
8290
  type: z.ZodLiteral<"audio">;
@@ -4421,6 +4421,7 @@ export declare const mulmoSlideMediaSchema: z.ZodObject<{
4421
4421
  }, z.core.$strip>>;
4422
4422
  layout: z.ZodLiteral<"funnel">;
4423
4423
  }, z.core.$strip>], "layout">;
4424
+ reference: z.ZodOptional<z.ZodString>;
4424
4425
  }, z.core.$strict>;
4425
4426
  export type AccentColorKey = z.infer<typeof accentColorKeySchema>;
4426
4427
  export type SlideThemeColors = z.infer<typeof slideThemeColorsSchema>;
@@ -345,5 +345,6 @@ export const mulmoSlideMediaSchema = z
345
345
  type: z.literal("slide"),
346
346
  theme: slideThemeSchema.optional(),
347
347
  slide: slideLayoutSchema,
348
+ reference: z.string().optional(),
348
349
  })
349
350
  .strict();
@@ -909,6 +909,7 @@ export declare const createStudioData: (_mulmoScript: MulmoScript, fileName: str
909
909
  mono: string;
910
910
  };
911
911
  } | undefined;
912
+ reference?: string | undefined;
912
913
  } | {
913
914
  type: "image";
914
915
  source: {
@@ -2292,6 +2293,7 @@ export declare const initializeContextFromFiles: (files: FileObject, raiseError:
2292
2293
  mono: string;
2293
2294
  };
2294
2295
  } | undefined;
2296
+ reference?: string | undefined;
2295
2297
  } | {
2296
2298
  type: "image";
2297
2299
  source: {
@@ -117,7 +117,8 @@ const processSlide = async (params) => {
117
117
  return;
118
118
  const theme = resolveTheme(params);
119
119
  const slide = resolveSlide(params, toFileUrl);
120
- const html = generateSlideHTML(theme, slide);
120
+ const reference = beat.image.reference;
121
+ const html = generateSlideHTML(theme, slide, reference);
121
122
  await renderHTMLToImage(html, imagePath, canvasSize.width, canvasSize.height);
122
123
  return imagePath;
123
124
  };
@@ -127,7 +128,8 @@ const dumpHtml = async (params) => {
127
128
  return;
128
129
  const theme = resolveTheme(params);
129
130
  const slide = resolveSlide(params);
130
- return generateSlideHTML(theme, slide);
131
+ const reference = beat.image.reference;
132
+ return generateSlideHTML(theme, slide, reference);
131
133
  };
132
134
  export const process = processSlide;
133
135
  export const path = parrotingImagePath;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "mulmocast",
3
- "version": "2.2.0",
3
+ "version": "2.2.2",
4
4
  "description": "",
5
5
  "type": "module",
6
6
  "main": "lib/index.node.js",
@@ -17,7 +17,14 @@
17
17
  "default": "./lib/data/index.js"
18
18
  },
19
19
  "./assets/*": "./assets/*",
20
- "./scripts/*": "./scripts/*"
20
+ "./scripts/*": "./scripts/*",
21
+ "./tools/complete_script": {
22
+ "types": "./lib/tools/complete_script.d.ts",
23
+ "default": "./lib/tools/complete_script.js"
24
+ }
25
+ },
26
+ "resolutions": {
27
+ "minimatch": "^10.2.2"
21
28
  },
22
29
  "bin": {
23
30
  "mulmo": "lib/cli/bin.js",
@@ -30,6 +37,9 @@
30
37
  "./assets/audio/silent60sec.mp3",
31
38
  "./assets/html/",
32
39
  "./assets/images/",
40
+ "./assets/schemas/",
41
+ "./assets/slide_themes/",
42
+ "./assets/styles/",
33
43
  "./assets/templates/"
34
44
  ],
35
45
  "directories": {
@@ -100,7 +110,7 @@
100
110
  "marked": "^17.0.3",
101
111
  "mulmocast-vision": "^1.0.8",
102
112
  "ora": "^9.3.0",
103
- "puppeteer": "^24.37.4",
113
+ "puppeteer": "^24.37.5",
104
114
  "replicate": "^1.4.0",
105
115
  "yaml": "^2.8.2",
106
116
  "yargs": "^18.0.0",
@@ -114,7 +124,7 @@
114
124
  "@types/jsdom": "^27.0.0",
115
125
  "@types/yargs": "^17.0.35",
116
126
  "cross-env": "^10.1.0",
117
- "eslint": "^10.0.0",
127
+ "eslint": "^10.0.1",
118
128
  "eslint-config-prettier": "^10.1.8",
119
129
  "eslint-plugin-prettier": "^5.5.5",
120
130
  "eslint-plugin-sonarjs": "^4.0.0",
@@ -280,7 +280,8 @@
280
280
  "text": "5σ means the probability of a fluke is 1 in 3.5 million — the threshold for 'discovery' in particle physics.",
281
281
  "color": "warning"
282
282
  }
283
- }
283
+ },
284
+ "reference": "Source: CERN Annual Report 2024 — https://home.cern/resources/report"
284
285
  }
285
286
  }
286
287
  ]