mirror of
https://github.com/typst/typst
synced 2025-05-13 12:36:23 +08:00
301 lines
9.3 KiB
Rust
301 lines
9.3 KiB
Rust
use once_cell::sync::Lazy;
|
|
use syntect::highlighting as synt;
|
|
use typst::syntax::{self, LinkedNode};
|
|
|
|
use super::{
|
|
FontFamily, FontList, Hyphenate, LinebreakNode, SmartQuoteNode, TextNode, TextSize,
|
|
};
|
|
use crate::layout::BlockNode;
|
|
use crate::prelude::*;
|
|
|
|
/// Raw text with optional syntax highlighting.
|
|
///
|
|
/// Displays the text verbatim and in a monospace font. This is typically used
|
|
/// to embed computer code into your document.
|
|
///
|
|
/// ## Syntax
|
|
/// This function also has dedicated syntax. You can enclose text in 1 or 3+
|
|
/// backticks (`` ` ``) to make it raw. Two backticks produce empty raw text.
|
|
/// When you use three or more backticks, you can additionally specify a
|
|
/// language tag for syntax highlighting directly after the opening backticks.
|
|
/// Within raw blocks, everything is rendered as is, in particular, there are no
|
|
/// escape sequences.
|
|
///
|
|
/// ## Example
|
|
/// ````example
|
|
/// Adding `rbx` to `rcx` gives
|
|
/// the desired result.
|
|
///
|
|
/// ```rust
|
|
/// fn main() {
|
|
/// println!("Hello World!");
|
|
/// }
|
|
/// ```
|
|
/// ````
|
|
///
|
|
/// Display: Raw Text / Code
|
|
/// Category: text
|
|
#[node(Prepare, Show, Finalize)]
|
|
pub struct RawNode {
|
|
/// The raw text.
|
|
///
|
|
/// You can also use raw blocks creatively to create custom syntaxes for
|
|
/// your automations.
|
|
///
|
|
/// ````example
|
|
/// // Parse numbers in raw blocks with the
|
|
/// // `mydsl` tag and sum them up.
|
|
/// #show raw.where(lang: "mydsl"): it => {
|
|
/// let sum = 0
|
|
/// for part in it.text.split("+") {
|
|
/// sum += int(part.trim())
|
|
/// }
|
|
/// sum
|
|
/// }
|
|
///
|
|
/// ```mydsl
|
|
/// 1 + 2 + 3 + 4 + 5
|
|
/// ```
|
|
/// ````
|
|
#[required]
|
|
pub text: EcoString,
|
|
|
|
/// Whether the raw text is displayed as a separate block.
|
|
///
|
|
/// ````example
|
|
/// // Display inline code in a small box
|
|
/// // that retains the correct baseline.
|
|
/// #show raw.where(block: false): box.with(
|
|
/// fill: luma(240),
|
|
/// inset: (x: 3pt, y: 0pt),
|
|
/// outset: (y: 3pt),
|
|
/// radius: 2pt,
|
|
/// )
|
|
///
|
|
/// // Display block code in a larger block
|
|
/// // with more padding.
|
|
/// #show raw.where(block: true): block.with(
|
|
/// fill: luma(240),
|
|
/// inset: 10pt,
|
|
/// radius: 4pt,
|
|
/// )
|
|
///
|
|
/// With `rg`, you can search through your files quickly.
|
|
///
|
|
/// ```bash
|
|
/// rg "Hello World"
|
|
/// ```
|
|
/// ````
|
|
#[default(false)]
|
|
pub block: bool,
|
|
|
|
/// The language to syntax-highlight in.
|
|
///
|
|
/// Apart from typical language tags known from Markdown, this supports the
|
|
/// `{"typ"}` and `{"typc"}` tags for Typst markup and Typst code,
|
|
/// respectively.
|
|
///
|
|
/// ````example
|
|
/// ```typ
|
|
/// This is *Typst!*
|
|
/// ```
|
|
/// ````
|
|
pub lang: Option<EcoString>,
|
|
}
|
|
|
|
impl Prepare for RawNode {
|
|
fn prepare(
|
|
&self,
|
|
_: &mut Vt,
|
|
mut this: Content,
|
|
styles: StyleChain,
|
|
) -> SourceResult<Content> {
|
|
this.push_field("lang", self.lang(styles).clone());
|
|
Ok(this)
|
|
}
|
|
}
|
|
|
|
impl Show for RawNode {
|
|
fn show(&self, _: &mut Vt, _: &Content, styles: StyleChain) -> SourceResult<Content> {
|
|
let text = self.text();
|
|
let lang = self.lang(styles).as_ref().map(|s| s.to_lowercase());
|
|
let foreground = THEME
|
|
.settings
|
|
.foreground
|
|
.map(to_typst)
|
|
.map_or(Color::BLACK, Color::from)
|
|
.into();
|
|
|
|
let mut realized = if matches!(lang.as_deref(), Some("typ" | "typst" | "typc")) {
|
|
let root = match lang.as_deref() {
|
|
Some("typc") => syntax::parse_code(&text),
|
|
_ => syntax::parse(&text),
|
|
};
|
|
|
|
let mut seq = vec![];
|
|
let highlighter = synt::Highlighter::new(&THEME);
|
|
highlight_themed(
|
|
&LinkedNode::new(&root),
|
|
vec![],
|
|
&highlighter,
|
|
&mut |node, style| {
|
|
seq.push(styled(&text[node.range()], foreground, style));
|
|
},
|
|
);
|
|
|
|
Content::sequence(seq)
|
|
} else if let Some(syntax) =
|
|
lang.and_then(|token| SYNTAXES.find_syntax_by_token(&token))
|
|
{
|
|
let mut seq = vec![];
|
|
let mut highlighter = syntect::easy::HighlightLines::new(syntax, &THEME);
|
|
for (i, line) in text.lines().enumerate() {
|
|
if i != 0 {
|
|
seq.push(LinebreakNode::new().pack());
|
|
}
|
|
|
|
for (style, piece) in
|
|
highlighter.highlight_line(line, &SYNTAXES).into_iter().flatten()
|
|
{
|
|
seq.push(styled(piece, foreground, style));
|
|
}
|
|
}
|
|
|
|
Content::sequence(seq)
|
|
} else {
|
|
TextNode::packed(text)
|
|
};
|
|
|
|
if self.block(styles) {
|
|
realized = BlockNode::new().with_body(Some(realized)).pack();
|
|
}
|
|
|
|
Ok(realized)
|
|
}
|
|
}
|
|
|
|
impl Finalize for RawNode {
|
|
fn finalize(&self, realized: Content, _: StyleChain) -> Content {
|
|
let mut map = StyleMap::new();
|
|
map.set(TextNode::set_overhang(false));
|
|
map.set(TextNode::set_hyphenate(Hyphenate(Smart::Custom(false))));
|
|
map.set(TextNode::set_size(TextSize(Em::new(0.8).into())));
|
|
map.set(TextNode::set_font(FontList(vec![FontFamily::new("DejaVu Sans Mono")])));
|
|
map.set(SmartQuoteNode::set_enabled(false));
|
|
realized.styled_with_map(map)
|
|
}
|
|
}
|
|
|
|
/// Highlight a syntax node in a theme by calling `f` with ranges and their
|
|
/// styles.
|
|
fn highlight_themed<F>(
|
|
node: &LinkedNode,
|
|
scopes: Vec<syntect::parsing::Scope>,
|
|
highlighter: &synt::Highlighter,
|
|
f: &mut F,
|
|
) where
|
|
F: FnMut(&LinkedNode, synt::Style),
|
|
{
|
|
if node.children().len() == 0 {
|
|
let style = highlighter.style_for_stack(&scopes);
|
|
f(node, style);
|
|
return;
|
|
}
|
|
|
|
for child in node.children() {
|
|
let mut scopes = scopes.clone();
|
|
if let Some(tag) = typst::ide::highlight(&child) {
|
|
scopes.push(syntect::parsing::Scope::new(tag.tm_scope()).unwrap())
|
|
}
|
|
highlight_themed(&child, scopes, highlighter, f);
|
|
}
|
|
}
|
|
|
|
/// Style a piece of text with a syntect style.
|
|
fn styled(piece: &str, foreground: Paint, style: synt::Style) -> Content {
|
|
let mut body = TextNode::packed(piece);
|
|
|
|
let paint = to_typst(style.foreground).into();
|
|
if paint != foreground {
|
|
body = body.styled(TextNode::set_fill(paint));
|
|
}
|
|
|
|
if style.font_style.contains(synt::FontStyle::BOLD) {
|
|
body = body.strong();
|
|
}
|
|
|
|
if style.font_style.contains(synt::FontStyle::ITALIC) {
|
|
body = body.emph();
|
|
}
|
|
|
|
if style.font_style.contains(synt::FontStyle::UNDERLINE) {
|
|
body = body.underlined();
|
|
}
|
|
|
|
body
|
|
}
|
|
|
|
fn to_typst(synt::Color { r, g, b, a }: synt::Color) -> RgbaColor {
|
|
RgbaColor { r, g, b, a }
|
|
}
|
|
|
|
fn to_syn(RgbaColor { r, g, b, a }: RgbaColor) -> synt::Color {
|
|
synt::Color { r, g, b, a }
|
|
}
|
|
|
|
/// The syntect syntax definitions.
|
|
static SYNTAXES: Lazy<syntect::parsing::SyntaxSet> =
|
|
Lazy::new(|| syntect::parsing::SyntaxSet::load_defaults_nonewlines());
|
|
|
|
/// The default theme used for syntax highlighting.
|
|
pub static THEME: Lazy<synt::Theme> = Lazy::new(|| synt::Theme {
|
|
name: Some("Typst Light".into()),
|
|
author: Some("The Typst Project Developers".into()),
|
|
settings: synt::ThemeSettings::default(),
|
|
scopes: vec![
|
|
item("comment", Some("#8a8a8a"), None),
|
|
item("constant.character.escape", Some("#1d6c76"), None),
|
|
item("markup.bold", None, Some(synt::FontStyle::BOLD)),
|
|
item("markup.italic", None, Some(synt::FontStyle::ITALIC)),
|
|
item("markup.underline", None, Some(synt::FontStyle::UNDERLINE)),
|
|
item("markup.raw", Some("#818181"), None),
|
|
item("string.other.math.typst", None, None),
|
|
item("punctuation.definition.math", Some("#298e0d"), None),
|
|
item("keyword.operator.math", Some("#1d6c76"), None),
|
|
item("markup.heading, entity.name.section", None, Some(synt::FontStyle::BOLD)),
|
|
item(
|
|
"markup.heading.typst",
|
|
None,
|
|
Some(synt::FontStyle::BOLD | synt::FontStyle::UNDERLINE),
|
|
),
|
|
item("punctuation.definition.list", Some("#8b41b1"), None),
|
|
item("markup.list.term", None, Some(synt::FontStyle::BOLD)),
|
|
item("entity.name.label, markup.other.reference", Some("#1d6c76"), None),
|
|
item("keyword, constant.language, variable.language", Some("#d73a49"), None),
|
|
item("storage.type, storage.modifier", Some("#d73a49"), None),
|
|
item("constant", Some("#b60157"), None),
|
|
item("string", Some("#298e0d"), None),
|
|
item("entity.name, variable.function, support", Some("#4b69c6"), None),
|
|
item("support.macro", Some("#16718d"), None),
|
|
item("meta.annotation", Some("#301414"), None),
|
|
item("entity.other, meta.interpolation", Some("#8b41b1"), None),
|
|
item("invalid", Some("#ff0000"), None),
|
|
],
|
|
});
|
|
|
|
/// Create a syntect theme item.
|
|
fn item(
|
|
scope: &str,
|
|
color: Option<&str>,
|
|
font_style: Option<synt::FontStyle>,
|
|
) -> synt::ThemeItem {
|
|
synt::ThemeItem {
|
|
scope: scope.parse().unwrap(),
|
|
style: synt::StyleModifier {
|
|
foreground: color.map(|s| to_syn(s.parse::<RgbaColor>().unwrap())),
|
|
background: None,
|
|
font_style,
|
|
},
|
|
}
|
|
}
|