use crate::library::layout::BlockSpacing; use crate::library::prelude::*; use crate::library::text::{FontFamily, TextNode, TextSize}; /// A section heading. #[derive(Debug, Hash)] pub struct HeadingNode { /// The logical nesting depth of the section, starting from one. In the /// default style, this controls the text size of the heading. pub level: NonZeroUsize, /// The heading's contents. pub body: Content, } #[node(showable)] impl HeadingNode { /// The heading's font family. Just the normal text family if `auto`. #[property(referenced)] pub const FAMILY: Leveled> = Leveled::Value(Smart::Auto); /// The color of text in the heading. Just the normal text color if `auto`. #[property(referenced)] pub const FILL: Leveled> = Leveled::Value(Smart::Auto); /// The size of text in the heading. #[property(referenced)] pub const SIZE: Leveled = Leveled::Mapping(|level| { let size = match level.get() { 1 => 1.4, 2 => 1.2, _ => 1.0, }; TextSize(Em::new(size).into()) }); /// Whether text in the heading is strengthend. #[property(referenced)] pub const STRONG: Leveled = Leveled::Value(true); /// Whether text in the heading is emphasized. #[property(referenced)] pub const EMPH: Leveled = Leveled::Value(false); /// Whether the heading is underlined. #[property(referenced)] pub const UNDERLINE: Leveled = Leveled::Value(false); /// The spacing above the heading. #[property(referenced, shorthand(around))] pub const ABOVE: Leveled> = Leveled::Mapping(|level| { let ratio = match level.get() { 1 => 1.5, _ => 1.2, }; Some(Ratio::new(ratio).into()) }); /// The spacing below the heading. #[property(referenced, shorthand(around))] pub const BELOW: Leveled> = Leveled::Value(Some(Ratio::new(0.55).into())); fn construct(_: &mut Machine, args: &mut Args) -> TypResult { Ok(Content::show(Self { body: args.expect("body")?, level: args.named("level")?.unwrap_or(NonZeroUsize::new(1).unwrap()), })) } } impl Show for HeadingNode { fn unguard(&self, sel: Selector) -> ShowNode { let body = self.body.unguard(sel).role(Role::Heading(self.level.get())); Self { body, ..*self }.pack() } fn encode(&self, _: StyleChain) -> Dict { dict! { "level" => Value::Int(self.level.get() as i64), "body" => Value::Content(self.body.clone()), } } fn realize(&self, _: &mut Context, _: StyleChain) -> TypResult { Ok(Content::block(self.body.clone())) } fn finalize( &self, ctx: &mut Context, styles: StyleChain, mut realized: Content, ) -> TypResult { macro_rules! resolve { ($key:expr) => { styles.get($key).resolve(ctx, self.level)? }; } let mut map = StyleMap::with_role(Role::Heading(self.level.get())); map.set(TextNode::SIZE, resolve!(Self::SIZE)); if let Smart::Custom(family) = resolve!(Self::FAMILY) { map.set_family(family, styles); } if let Smart::Custom(fill) = resolve!(Self::FILL) { map.set(TextNode::FILL, fill); } if resolve!(Self::STRONG) { realized = realized.strong(); } if resolve!(Self::EMPH) { realized = realized.emph(); } if resolve!(Self::UNDERLINE) { realized = realized.underlined(); } realized = realized.styled_with_map(map); realized = realized.spaced( resolve!(Self::ABOVE).resolve(styles), resolve!(Self::BELOW).resolve(styles), ); Ok(realized) } } /// Either the value or a closure mapping to the value. #[derive(Debug, Clone, PartialEq, Hash)] pub enum Leveled { /// A bare value. Value(T), /// A simple mapping from a heading level to a value. Mapping(fn(NonZeroUsize) -> T), /// A closure mapping from a heading level to a value. Func(Func, Span), } impl Leveled { /// Resolve the value based on the level. pub fn resolve(&self, ctx: &mut Context, level: NonZeroUsize) -> TypResult { Ok(match self { Self::Value(value) => value.clone(), Self::Mapping(mapping) => mapping(level), Self::Func(func, span) => { let args = Args::new(*span, [Value::Int(level.get() as i64)]); func.call_detached(ctx, args)?.cast().at(*span)? } }) } } impl Cast> for Leveled { fn is(value: &Spanned) -> bool { matches!(&value.v, Value::Func(_)) || T::is(&value.v) } fn cast(value: Spanned) -> StrResult { match value.v { Value::Func(v) => Ok(Self::Func(v, value.span)), v => T::cast(v) .map(Self::Value) .map_err(|msg| with_alternative(msg, "function")), } } }