mirror of
https://github.com/typst/typst
synced 2025-05-14 17:15:28 +08:00
Indent configuration for multiline headings (#3459)
This commit is contained in:
parent
102e6717e8
commit
6d0c159e97
@ -3,12 +3,14 @@ use std::num::NonZeroUsize;
|
|||||||
use crate::diag::SourceResult;
|
use crate::diag::SourceResult;
|
||||||
use crate::engine::Engine;
|
use crate::engine::Engine;
|
||||||
use crate::foundations::{
|
use crate::foundations::{
|
||||||
elem, Content, NativeElement, Packed, Show, ShowSet, Smart, StyleChain, Styles,
|
elem, Content, NativeElement, Packed, Resolve, Show, ShowSet, Smart, StyleChain,
|
||||||
Synthesize,
|
Styles, Synthesize,
|
||||||
};
|
};
|
||||||
use crate::introspection::{Count, Counter, CounterUpdate, Locatable};
|
use crate::introspection::{Count, Counter, CounterUpdate, Locatable};
|
||||||
use crate::layout::{BlockElem, Em, HElem, VElem};
|
use crate::layout::{
|
||||||
use crate::model::{Numbering, Outlinable, Refable, Supplement};
|
Abs, Axes, BlockElem, Em, HElem, LayoutMultiple, Length, Regions, VElem,
|
||||||
|
};
|
||||||
|
use crate::model::{Numbering, Outlinable, ParElem, Refable, Supplement};
|
||||||
use crate::text::{FontWeight, LocalName, SpaceElem, TextElem, TextSize};
|
use crate::text::{FontWeight, LocalName, SpaceElem, TextElem, TextSize};
|
||||||
use crate::util::NonZeroExt;
|
use crate::util::NonZeroExt;
|
||||||
|
|
||||||
@ -163,6 +165,18 @@ pub struct HeadingElem {
|
|||||||
#[default(Smart::Auto)]
|
#[default(Smart::Auto)]
|
||||||
pub bookmarked: Smart<bool>,
|
pub bookmarked: Smart<bool>,
|
||||||
|
|
||||||
|
/// The indent all but the first line of a heading should have.
|
||||||
|
///
|
||||||
|
/// The default value of `{auto}` indicates that the subsequent heading
|
||||||
|
/// lines will be indented based on the width of the numbering.
|
||||||
|
///
|
||||||
|
/// ```example
|
||||||
|
/// #set heading(numbering: "1.")
|
||||||
|
/// #heading[A very, very, very, very, very, very long heading]
|
||||||
|
/// ```
|
||||||
|
#[default(Smart::Auto)]
|
||||||
|
pub hanging_indent: Smart<Length>,
|
||||||
|
|
||||||
/// The heading's title.
|
/// The heading's title.
|
||||||
#[required]
|
#[required]
|
||||||
pub body: Content,
|
pub body: Content,
|
||||||
@ -201,15 +215,39 @@ impl Synthesize for Packed<HeadingElem> {
|
|||||||
impl Show for Packed<HeadingElem> {
|
impl Show for Packed<HeadingElem> {
|
||||||
#[typst_macros::time(name = "heading", span = self.span())]
|
#[typst_macros::time(name = "heading", span = self.span())]
|
||||||
fn show(&self, engine: &mut Engine, styles: StyleChain) -> SourceResult<Content> {
|
fn show(&self, engine: &mut Engine, styles: StyleChain) -> SourceResult<Content> {
|
||||||
|
const SPACING_TO_NUMBERING: Em = Em::new(0.3);
|
||||||
|
|
||||||
let span = self.span();
|
let span = self.span();
|
||||||
let mut realized = self.body().clone();
|
let mut realized = self.body().clone();
|
||||||
|
|
||||||
|
let hanging_indent = self.hanging_indent(styles);
|
||||||
|
|
||||||
|
let mut indent = match hanging_indent {
|
||||||
|
Smart::Custom(length) => length.resolve(styles),
|
||||||
|
Smart::Auto => Abs::zero(),
|
||||||
|
};
|
||||||
|
|
||||||
if let Some(numbering) = (**self).numbering(styles).as_ref() {
|
if let Some(numbering) = (**self).numbering(styles).as_ref() {
|
||||||
realized = Counter::of(HeadingElem::elem())
|
let numbering = Counter::of(HeadingElem::elem())
|
||||||
.display_at_loc(engine, self.location().unwrap(), styles, numbering)?
|
.display_at_loc(engine, self.location().unwrap(), styles, numbering)?
|
||||||
.spanned(span)
|
.spanned(span);
|
||||||
+ HElem::new(Em::new(0.3).into()).with_weak(true).pack()
|
|
||||||
|
if hanging_indent.is_auto() {
|
||||||
|
let pod = Regions::one(Axes::splat(Abs::inf()), Axes::splat(false));
|
||||||
|
let size = numbering.measure(engine, styles, pod)?.into_frame().size();
|
||||||
|
|
||||||
|
indent = size.x + SPACING_TO_NUMBERING.resolve(styles);
|
||||||
|
}
|
||||||
|
|
||||||
|
realized = numbering
|
||||||
|
+ HElem::new(SPACING_TO_NUMBERING.into()).with_weak(true).pack()
|
||||||
+ realized;
|
+ realized;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if indent != Abs::zero() {
|
||||||
|
realized = realized.styled(ParElem::set_hanging_indent(indent.into()));
|
||||||
|
}
|
||||||
|
|
||||||
Ok(BlockElem::new().with_body(Some(realized)).pack().spanned(span))
|
Ok(BlockElem::new().with_body(Some(realized)).pack().spanned(span))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 11 KiB |
Loading…
x
Reference in New Issue
Block a user