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.
- package/assets/schemas/html_prompt.json +60 -0
- package/assets/schemas/mulmo_script.json +283 -0
- package/assets/slide_themes/corporate.json +18 -0
- package/assets/slide_themes/creative.json +18 -0
- package/assets/slide_themes/dark.json +18 -0
- package/assets/slide_themes/minimal.json +18 -0
- package/assets/slide_themes/pop.json +18 -0
- package/assets/slide_themes/warm.json +18 -0
- package/assets/styles/akira_comic.json +22 -0
- package/assets/styles/ani.json +43 -0
- package/assets/styles/children_book.json +13 -0
- package/assets/styles/comic_strips.json +13 -0
- package/assets/styles/drslump_comic.json +22 -0
- package/assets/styles/ghibli_comic.json +22 -0
- package/assets/styles/ghibli_shorts.json +28 -0
- package/assets/styles/ghost_comic.json +29 -0
- package/assets/styles/leda.json +29 -0
- package/assets/styles/onepiece_comic.json +22 -0
- package/assets/styles/slide_corporate.json +30 -0
- package/assets/styles/slide_creative.json +30 -0
- package/assets/styles/slide_dark.json +30 -0
- package/assets/styles/slide_minimal.json +30 -0
- package/assets/styles/slide_pop.json +30 -0
- package/assets/styles/slide_warm.json +30 -0
- package/lib/data/index.d.ts +2 -0
- package/lib/data/index.js +2 -0
- package/lib/data/slideStyles.d.ts +206 -0
- package/lib/data/slideStyles.js +206 -0
- package/lib/data/slideThemes.d.ts +134 -0
- package/lib/data/slideThemes.js +134 -0
- package/lib/slide/layouts/columns.js +3 -4
- package/lib/slide/layouts/comparison.js +3 -5
- package/lib/slide/layouts/grid.js +1 -1
- package/lib/slide/layouts/stats.js +20 -7
- package/lib/slide/layouts/timeline.js +17 -4
- package/lib/slide/render.d.ts +1 -1
- package/lib/slide/render.js +5 -1
- package/lib/slide/schema.d.ts +1 -0
- package/lib/slide/schema.js +1 -0
- package/lib/types/schema.d.ts +4 -0
- package/lib/types/slide.d.ts +1 -0
- package/lib/types/slide.js +1 -0
- package/lib/utils/context.d.ts +2 -0
- package/lib/utils/image_plugins/slide.js +4 -2
- package/package.json +14 -4
- package/scripts/test/test_slide_image_ref_en.json +2 -1
- package/scripts/test/test_slide_image_ref_gemini_en.json +289 -0
- package/scripts/test/test_slide_image_ref_gemini_en.json~ +289 -0
- 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-
|
|
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(`<
|
|
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-
|
|
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-
|
|
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
|
-
|
|
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-
|
|
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-
|
|
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,
|
|
1
|
+
import { escapeHtml, nl2br, c, renderCalloutBar } from "../utils.js";
|
|
2
2
|
export const layoutStats = (data) => {
|
|
3
|
-
const
|
|
3
|
+
const accent = data.accentColor || "primary";
|
|
4
4
|
const stats = data.stats || [];
|
|
5
|
-
parts
|
|
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-
|
|
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-[
|
|
11
|
-
parts.push(` <p class="text-
|
|
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-
|
|
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
|
|
1
|
+
import { escapeHtml, nl2br, c } from "../utils.js";
|
|
2
2
|
export const layoutTimeline = (data) => {
|
|
3
|
-
const
|
|
3
|
+
const accent = data.accentColor || "primary";
|
|
4
|
+
const parts = [];
|
|
4
5
|
const items = data.items || [];
|
|
5
|
-
parts.push(`<div class="
|
|
6
|
-
parts.push(`<div class="
|
|
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
|
};
|
package/lib/slide/render.d.ts
CHANGED
|
@@ -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;
|
package/lib/slide/render.js
CHANGED
|
@@ -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>
|
package/lib/slide/schema.d.ts
CHANGED
|
@@ -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>;
|
package/lib/slide/schema.js
CHANGED
package/lib/types/schema.d.ts
CHANGED
|
@@ -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">;
|
package/lib/types/slide.d.ts
CHANGED
|
@@ -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>;
|
package/lib/types/slide.js
CHANGED
package/lib/utils/context.d.ts
CHANGED
|
@@ -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
|
|
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
|
-
|
|
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.
|
|
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.
|
|
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.
|
|
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
|
]
|