diff --git a/config-file-schema.json b/config-file-schema.json index b78ccc4f..a11fe342 100644 --- a/config-file-schema.json +++ b/config-file-schema.json @@ -14,16 +14,14 @@ } ] }, + "mermaid": { + "$ref": "#/definitions/MermaidConfig" + }, "options": { "$ref": "#/definitions/OptionsConfig" }, "typst": { - "allOf": [ - { - "$ref": "#/definitions/TypstConfig" - } - ], - "minimum": 1.0 + "$ref": "#/definitions/TypstConfig" } }, "additionalProperties": false, @@ -210,6 +208,19 @@ }, "additionalProperties": false }, + "MermaidConfig": { + "type": "object", + "properties": { + "scale": { + "description": "The scaling parameter to be used in the mermaid CLI.", + "default": 2, + "type": "integer", + "format": "uint32", + "minimum": 0.0 + } + }, + "additionalProperties": false + }, "OptionsConfig": { "type": "object", "properties": { diff --git a/src/custom.rs b/src/custom.rs index a5779a8c..f44697c2 100644 --- a/src/custom.rs +++ b/src/custom.rs @@ -16,9 +16,11 @@ pub struct Config { pub defaults: DefaultsConfig, #[serde(default)] - #[validate(range(min = 1))] pub typst: TypstConfig, + #[serde(default)] + pub mermaid: MermaidConfig, + #[serde(default)] pub options: OptionsConfig, @@ -126,10 +128,28 @@ impl Default for TypstConfig { } } -fn default_typst_ppi() -> u32 { +pub(crate) fn default_typst_ppi() -> u32 { 300 } +#[derive(Clone, Debug, Deserialize, JsonSchema)] +#[serde(deny_unknown_fields)] +pub struct MermaidConfig { + /// The scaling parameter to be used in the mermaid CLI. + #[serde(default = "default_mermaid_scale")] + pub scale: u32, +} + +impl Default for MermaidConfig { + fn default() -> Self { + Self { scale: default_mermaid_scale() } + } +} + +pub(crate) fn default_mermaid_scale() -> u32 { + 2 +} + #[derive(Clone, Debug, Default, Deserialize, ValueEnum, JsonSchema)] #[serde(rename_all = "kebab-case")] pub enum ImageProtocol { diff --git a/src/demo.rs b/src/demo.rs index 6ee342ca..1b811aaf 100644 --- a/src/demo.rs +++ b/src/demo.rs @@ -8,7 +8,7 @@ use crate::{ presentation::Presentation, processing::builder::{BuildError, PresentationBuilder}, render::{draw::TerminalDrawer, terminal::TerminalWrite}, - ImageRegistry, MarkdownParser, PresentationBuilderOptions, PresentationTheme, Resources, Themes, TypstRender, + ImageRegistry, MarkdownParser, PresentationBuilderOptions, PresentationTheme, Resources, Themes, ThirdPartyRender, }; use std::{io, rc::Rc}; @@ -100,14 +100,14 @@ impl ThemesDemo { ) -> Result { let image_registry = ImageRegistry::default(); let mut resources = Resources::new("non_existent", image_registry.clone()); - let mut typst = TypstRender::default(); + let mut third_party = ThirdPartyRender::default(); let options = PresentationBuilderOptions::default(); let executer = Rc::new(CodeExecutor::default()); let bindings_config = Default::default(); let builder = PresentationBuilder::new( theme, &mut resources, - &mut typst, + &mut third_party, executer, &self.themes, image_registry, diff --git a/src/export.rs b/src/export.rs index 31d2d722..5cc4788f 100644 --- a/src/export.rs +++ b/src/export.rs @@ -8,8 +8,8 @@ use crate::{ }, presentation::{Presentation, RenderOperation}, processing::builder::{BuildError, PresentationBuilder, PresentationBuilderOptions, Themes}, + third_party::ThirdPartyRender, tools::{ExecutionError, ThirdPartyTools}, - typst::TypstRender, MarkdownParser, PresentationTheme, Resources, }; use base64::{engine::general_purpose::STANDARD, Engine}; @@ -29,7 +29,7 @@ pub struct Exporter<'a> { parser: MarkdownParser<'a>, default_theme: &'a PresentationTheme, resources: Resources, - typst: TypstRender, + third_party: ThirdPartyRender, code_executor: Rc, themes: Themes, options: PresentationBuilderOptions, @@ -41,12 +41,12 @@ impl<'a> Exporter<'a> { parser: MarkdownParser<'a>, default_theme: &'a PresentationTheme, resources: Resources, - typst: TypstRender, + third_party: ThirdPartyRender, code_executor: Rc, themes: Themes, options: PresentationBuilderOptions, ) -> Self { - Self { parser, default_theme, resources, typst, code_executor, themes, options } + Self { parser, default_theme, resources, third_party, code_executor, themes, options } } /// Export the given presentation into PDF. @@ -85,7 +85,7 @@ impl<'a> Exporter<'a> { let mut presentation = PresentationBuilder::new( self.default_theme, &mut self.resources, - &mut self.typst, + &mut self.third_party, self.code_executor.clone(), &self.themes, Default::default(), @@ -302,11 +302,11 @@ mod test { let parser = MarkdownParser::new(&arena); let theme = PresentationThemeSet::default().load_by_name("dark").unwrap(); let resources = Resources::new("examples", Default::default()); - let typst = TypstRender::default(); + let third_party = ThirdPartyRender::default(); let code_executor = Default::default(); let themes = Themes::default(); let options = PresentationBuilderOptions { allow_mutations: false, ..Default::default() }; - let mut exporter = Exporter::new(parser, &theme, resources, typst, code_executor, themes, options); + let mut exporter = Exporter::new(parser, &theme, resources, third_party, code_executor, themes, options); exporter.extract_metadata(content, Path::new(path)).expect("metadata extraction failed") } diff --git a/src/lib.rs b/src/lib.rs index a42fb608..d16d5e56 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -17,8 +17,8 @@ pub(crate) mod render; pub(crate) mod resource; pub(crate) mod style; pub(crate) mod theme; +pub(crate) mod third_party; pub(crate) mod tools; -pub(crate) mod typst; pub use crate::{ custom::{Config, ImageProtocol, ValidateOverflows}, @@ -33,5 +33,5 @@ pub use crate::{ render::highlighting::{CodeHighlighter, HighlightThemeSet}, resource::Resources, theme::{LoadThemeError, PresentationTheme, PresentationThemeSet}, - typst::TypstRender, + third_party::{ThirdPartyConfigs, ThirdPartyRender}, }; diff --git a/src/main.rs b/src/main.rs index 451ad63e..13906df6 100644 --- a/src/main.rs +++ b/src/main.rs @@ -4,7 +4,7 @@ use directories::ProjectDirs; use presenterm::{ CodeExecutor, CommandSource, Config, Exporter, GraphicsMode, HighlightThemeSet, ImagePrinter, ImageProtocol, ImageRegistry, MarkdownParser, PresentMode, PresentationBuilderOptions, PresentationTheme, PresentationThemeSet, - Presenter, PresenterOptions, Resources, Themes, ThemesDemo, TypstRender, ValidateOverflows, + Presenter, PresenterOptions, Resources, Themes, ThemesDemo, ThirdPartyConfigs, ThirdPartyRender, ValidateOverflows, }; use std::{ env, io, @@ -211,10 +211,13 @@ fn run(mut cli: Cli) -> Result<(), Box> { let printer = Rc::new(ImagePrinter::new(graphics_mode.clone())?); let registry = ImageRegistry(printer.clone()); let resources = Resources::new(resources_path, registry.clone()); - let typst = TypstRender::new(config.typst.ppi, registry, resources_path); + let third_party_config = + ThirdPartyConfigs { typst_ppi: config.typst.ppi.to_string(), mermaid_scale: config.mermaid.scale.to_string() }; + let third_party = ThirdPartyRender::new(third_party_config, registry, resources_path); let code_executor = Rc::new(code_executor); if cli.export_pdf || cli.generate_pdf_metadata { - let mut exporter = Exporter::new(parser, &default_theme, resources, typst, code_executor, themes, options); + let mut exporter = + Exporter::new(parser, &default_theme, resources, third_party, code_executor, themes, options); let mut args = Vec::new(); if let Some(theme) = cli.theme.as_ref() { args.extend(["--theme", theme]); @@ -239,8 +242,17 @@ fn run(mut cli: Cli) -> Result<(), Box> { bindings: config.bindings, validate_overflows, }; - let presenter = - Presenter::new(&default_theme, commands, parser, resources, typst, code_executor, themes, printer, options); + let presenter = Presenter::new( + &default_theme, + commands, + parser, + resources, + third_party, + code_executor, + themes, + printer, + options, + ); presenter.present(&path)?; } Ok(()) diff --git a/src/markdown/elements.rs b/src/markdown/elements.rs index 2f3ec3f0..216d4b1f 100644 --- a/src/markdown/elements.rs +++ b/src/markdown/elements.rs @@ -217,6 +217,7 @@ pub enum CodeLanguage { Latex, Lua, Makefile, + Mermaid, Markdown, Nix, OCaml, @@ -245,7 +246,7 @@ pub enum CodeLanguage { impl CodeLanguage { pub(crate) fn supports_auto_render(&self) -> bool { - matches!(self, Self::Latex | Self::Typst) + matches!(self, Self::Latex | Self::Typst | Self::Mermaid) } } @@ -283,6 +284,7 @@ impl FromStr for CodeLanguage { "lua" => Lua, "make" => Makefile, "markdown" => Markdown, + "mermaid" => Mermaid, "nix" => Nix, "ocaml" => OCaml, "perl" => Perl, diff --git a/src/presenter.rs b/src/presenter.rs index 8eb858ee..572f8186 100644 --- a/src/presenter.rs +++ b/src/presenter.rs @@ -15,7 +15,7 @@ use crate::{ }, resource::Resources, theme::PresentationTheme, - typst::TypstRender, + third_party::ThirdPartyRender, }; use std::{ collections::HashSet, @@ -43,7 +43,7 @@ pub struct Presenter<'a> { commands: CommandSource, parser: MarkdownParser<'a>, resources: Resources, - typst: TypstRender, + third_party: ThirdPartyRender, code_executor: Rc, state: PresenterState, slides_with_pending_widgets: HashSet, @@ -60,7 +60,7 @@ impl<'a> Presenter<'a> { commands: CommandSource, parser: MarkdownParser<'a>, resources: Resources, - typst: TypstRender, + third_party: ThirdPartyRender, code_executor: Rc, themes: Themes, image_printer: Rc, @@ -71,7 +71,7 @@ impl<'a> Presenter<'a> { commands, parser, resources, - typst, + third_party, code_executor, state: PresenterState::Empty, slides_with_pending_widgets: HashSet::new(), @@ -260,7 +260,7 @@ impl<'a> Presenter<'a> { let mut presentation = PresentationBuilder::new( self.default_theme, &mut self.resources, - &mut self.typst, + &mut self.third_party, self.code_executor.clone(), &self.themes, ImageRegistry(self.image_printer.clone()), diff --git a/src/processing/builder.rs b/src/processing/builder.rs index 7d44a161..94ee4607 100644 --- a/src/processing/builder.rs +++ b/src/processing/builder.rs @@ -26,7 +26,7 @@ use crate::{ theme::{ Alignment, AuthorPositioning, ElementType, LoadThemeError, Margin, PresentationTheme, PresentationThemeSet, }, - typst::{TypstRender, TypstRenderError}, + third_party::{ThirdPartyRender, TypstRenderError}, }; use image::DynamicImage; use serde::Deserialize; @@ -98,7 +98,7 @@ pub(crate) struct PresentationBuilder<'a> { code_executor: Rc, theme: Cow<'a, PresentationTheme>, resources: &'a mut Resources, - typst: &'a mut TypstRender, + third_party: &'a mut ThirdPartyRender, slide_state: SlideState, footer_context: Rc>, themes: &'a Themes, @@ -114,7 +114,7 @@ impl<'a> PresentationBuilder<'a> { pub(crate) fn new( default_theme: &'a PresentationTheme, resources: &'a mut Resources, - typst: &'a mut TypstRender, + third_party: &'a mut ThirdPartyRender, code_executor: Rc, themes: &'a Themes, image_registry: ImageRegistry, @@ -130,7 +130,7 @@ impl<'a> PresentationBuilder<'a> { code_executor, theme: Cow::Borrowed(default_theme), resources, - typst, + third_party, slide_state: Default::default(), footer_context: Default::default(), themes, @@ -694,8 +694,9 @@ impl<'a> PresentationBuilder<'a> { fn push_rendered_code(&mut self, code: Code) -> Result<(), BuildError> { let image = match code.language { - CodeLanguage::Typst => self.typst.render_typst(&code.contents, &self.theme.typst)?, - CodeLanguage::Latex => self.typst.render_latex(&code.contents, &self.theme.typst)?, + CodeLanguage::Typst => self.third_party.render_typst(&code.contents, &self.theme.typst)?, + CodeLanguage::Latex => self.third_party.render_latex(&code.contents, &self.theme.typst)?, + CodeLanguage::Mermaid => self.third_party.render_mermaid(&code.contents, &self.theme.mermaid)?, _ => panic!("language {:?} should not be renderable", code.language), }; self.push_image(image); @@ -1065,14 +1066,14 @@ mod test { ) -> Result { let theme = PresentationTheme::default(); let mut resources = Resources::new("/tmp", Default::default()); - let mut typst = TypstRender::default(); + let mut third_party = ThirdPartyRender::default(); let code_executor = Rc::new(CodeExecutor::default()); let themes = Themes::default(); let bindings = KeyBindingsConfig::default(); let builder = PresentationBuilder::new( &theme, &mut resources, - &mut typst, + &mut third_party, code_executor, &themes, Default::default(), diff --git a/src/render/highlighting.rs b/src/render/highlighting.rs index d20caee5..9b4f92d8 100644 --- a/src/render/highlighting.rs +++ b/src/render/highlighting.rs @@ -137,6 +137,7 @@ impl CodeHighlighter { Lua => "lua", Makefile => "make", Markdown => "md", + Mermaid => "txt", Nix => "nix", OCaml => "ml", Perl => "pl", diff --git a/src/theme.rs b/src/theme.rs index b5629ab9..11af0f62 100644 --- a/src/theme.rs +++ b/src/theme.rs @@ -170,7 +170,11 @@ pub struct PresentationTheme { #[serde(default)] pub(crate) typst: TypstStyle, - /// The style for typst auto-rendered code blocks. + /// The style for mermaid auto-rendered code blocks. + #[serde(default)] + pub(crate) mermaid: MermaidStyle, + + /// The style for modals. #[serde(default)] pub(crate) modals: ModalStyle, } @@ -607,6 +611,16 @@ pub(crate) struct TypstStyle { pub(crate) colors: Colors, } +/// Where to position the author's name in the intro slide. +#[derive(Clone, Debug, Default, Deserialize, Serialize)] +pub(crate) struct MermaidStyle { + /// The mermaidjs theme to use. + pub(crate) theme: Option, + + /// The background color to use. + pub(crate) background: Option, +} + /// Modals style. #[derive(Clone, Debug, Default, Deserialize, Serialize)] pub(crate) struct ModalStyle { diff --git a/src/typst.rs b/src/third_party.rs similarity index 68% rename from src/typst.rs rename to src/third_party.rs index b10005e5..f88936a7 100644 --- a/src/typst.rs +++ b/src/third_party.rs @@ -1,31 +1,36 @@ use crate::{ + custom::{default_mermaid_scale, default_typst_ppi}, media::{image::Image, printer::RegisterImageError}, style::Color, - theme::TypstStyle, + theme::{MermaidStyle, TypstStyle}, tools::{ExecutionError, ThirdPartyTools}, ImageRegistry, }; use std::{fs, io, path::Path}; use tempfile::tempdir_in; -const DEFAULT_PPI: u32 = 300; const DEFAULT_HORIZONTAL_MARGIN: u16 = 5; const DEFAULT_VERTICAL_MARGIN: u16 = 7; -pub struct TypstRender { - ppi: String, +pub struct ThirdPartyConfigs { + pub typst_ppi: String, + pub mermaid_scale: String, +} + +pub struct ThirdPartyRender { + config: ThirdPartyConfigs, image_registry: ImageRegistry, root_dir: String, } -impl TypstRender { - pub fn new(ppi: u32, image_registry: ImageRegistry, root_dir: &Path) -> Self { +impl ThirdPartyRender { + pub fn new(config: ThirdPartyConfigs, image_registry: ImageRegistry, root_dir: &Path) -> Self { // typst complains about empty paths so we give it a "." if we don't have one. let root_dir = match root_dir.to_string_lossy().to_string() { path if path.is_empty() => ".".into(), path => path, }; - Self { ppi: ppi.to_string(), image_registry, root_dir } + Self { config, image_registry, root_dir } } pub(crate) fn render_typst(&self, input: &str, style: &TypstStyle) -> Result { @@ -47,6 +52,32 @@ impl TypstRender { self.render_typst(&input, style) } + pub(crate) fn render_mermaid(&self, input: &str, style: &MermaidStyle) -> Result { + let workdir = tempdir_in(&self.root_dir)?; + let output_path = workdir.path().join("output.png"); + let input_path = workdir.path().join("input.mmd"); + fs::write(&input_path, input)?; + + ThirdPartyTools::mermaid(&[ + "-i", + &input_path.to_string_lossy(), + "-o", + &output_path.to_string_lossy(), + "-s", + &self.config.mermaid_scale, + "-t", + style.theme.as_deref().unwrap_or("default"), + "-b", + style.background.as_deref().unwrap_or("white"), + ]) + .run()?; + + let png_contents = fs::read(&output_path)?; + let image = image::load_from_memory(&png_contents)?; + let image = self.image_registry.register_image(image)?; + Ok(image) + } + fn render_to_image(&self, base_path: &Path, path: &Path) -> Result { let output_path = base_path.join("output.png"); ThirdPartyTools::typst(&[ @@ -56,7 +87,7 @@ impl TypstRender { "--root", &self.root_dir, "--ppi", - &self.ppi, + &self.config.typst_ppi, &path.to_string_lossy(), &output_path.to_string_lossy(), ]) @@ -91,9 +122,13 @@ impl TypstRender { } } -impl Default for TypstRender { +impl Default for ThirdPartyRender { fn default() -> Self { - Self::new(DEFAULT_PPI, Default::default(), Path::new(".")) + let config = ThirdPartyConfigs { + typst_ppi: default_typst_ppi().to_string(), + mermaid_scale: default_mermaid_scale().to_string(), + }; + Self::new(config, Default::default(), Path::new(".")) } } diff --git a/src/tools.rs b/src/tools.rs index e0df0230..f00b783e 100644 --- a/src/tools.rs +++ b/src/tools.rs @@ -17,6 +17,10 @@ impl ThirdPartyTools { Tool::new("typst", args) } + pub(crate) fn mermaid(args: &[&str]) -> Tool { + Tool::new("mmdc", args) + } + pub(crate) fn presenterm_export(args: &[&str]) -> Tool { Tool::new("presenterm-export", args).inherit_stdout() }