Associate each item in Items with a logical index

This commit is contained in:
+merlan #flirora 2025-02-15 01:03:38 -05:00
parent 662bccc42d
commit fa76ea6310
4 changed files with 30 additions and 17 deletions

View File

@ -26,6 +26,7 @@ const LINE_SEPARATOR: char = '\u{2028}'; // We use LS to distinguish justified b
/// first and last one since they may be broken apart by the start or end of the /// first and last one since they may be broken apart by the start or end of the
/// line, respectively. But even those can partially reuse previous results when /// line, respectively. But even those can partially reuse previous results when
/// the break index is safe-to-break per rustybuzz. /// the break index is safe-to-break per rustybuzz.
#[derive(Debug)]
pub struct Line<'a> { pub struct Line<'a> {
/// The items the line is made of. /// The items the line is made of.
pub items: Items<'a>, pub items: Items<'a>,
@ -219,7 +220,7 @@ fn collect_items<'a>(
// Add fallback text to expand the line height, if necessary. // Add fallback text to expand the line height, if necessary.
if !items.iter().any(|item| matches!(item, Item::Text(_))) { if !items.iter().any(|item| matches!(item, Item::Text(_))) {
if let Some(fallback) = fallback { if let Some(fallback) = fallback {
items.push(fallback); items.push(fallback, usize::MAX);
} }
} }
@ -273,7 +274,7 @@ fn collect_range<'a>(
for run in p.slice(range.clone()) { for run in p.slice(range.clone()) {
// All non-text items are just kept, they can't be split. // All non-text items are just kept, they can't be split.
let Item::Text(shaped) = &run.item else { let Item::Text(shaped) = &run.item else {
items.push(&run.item); items.push(&run.item, run.idx);
continue; continue;
}; };
let subrange = &run.range; let subrange = &run.range;
@ -294,10 +295,10 @@ fn collect_range<'a>(
} else if split { } else if split {
// When the item is split in half, reshape it. // When the item is split in half, reshape it.
let reshaped = shaped.reshape(engine, sliced); let reshaped = shaped.reshape(engine, sliced);
items.push(Item::Text(reshaped)); items.push(Item::Text(reshaped), run.idx);
} else { } else {
// When the item is fully contained, just keep it. // When the item is fully contained, just keep it.
items.push(&run.item); items.push(&run.item, run.idx);
} }
} }
} }
@ -628,7 +629,7 @@ fn overhang(c: char) -> f64 {
} }
/// A collection of owned or borrowed inline items. /// A collection of owned or borrowed inline items.
pub struct Items<'a>(Vec<ItemEntry<'a>>); pub struct Items<'a>(Vec<IndexedItemEntry<'a>>);
impl<'a> Items<'a> { impl<'a> Items<'a> {
/// Create empty items. /// Create empty items.
@ -637,33 +638,33 @@ impl<'a> Items<'a> {
} }
/// Push a new item. /// Push a new item.
pub fn push(&mut self, entry: impl Into<ItemEntry<'a>>) { pub fn push(&mut self, entry: impl Into<ItemEntry<'a>>, idx: usize) {
self.0.push(entry.into()); self.0.push(IndexedItemEntry { item: entry.into(), idx });
} }
/// Iterate over the items /// Iterate over the items
pub fn iter(&self) -> impl Iterator<Item = &Item<'a>> { pub fn iter(&self) -> impl Iterator<Item = &Item<'a>> {
self.0.iter().map(|item| &**item) self.0.iter().map(|item| &*item.item)
} }
/// Access the first item. /// Access the first item.
pub fn first(&self) -> Option<&Item<'a>> { pub fn first(&self) -> Option<&Item<'a>> {
self.0.first().map(|item| &**item) self.0.first().map(|item| &*item.item)
} }
/// Access the last item. /// Access the last item.
pub fn last(&self) -> Option<&Item<'a>> { pub fn last(&self) -> Option<&Item<'a>> {
self.0.last().map(|item| &**item) self.0.last().map(|item| &*item.item)
} }
/// Access the first item mutably, if it is text. /// Access the first item mutably, if it is text.
pub fn first_text_mut(&mut self) -> Option<&mut ShapedText<'a>> { pub fn first_text_mut(&mut self) -> Option<&mut ShapedText<'a>> {
self.0.first_mut()?.text_mut() self.0.first_mut()?.item.text_mut()
} }
/// Access the last item mutably, if it is text. /// Access the last item mutably, if it is text.
pub fn last_text_mut(&mut self) -> Option<&mut ShapedText<'a>> { pub fn last_text_mut(&mut self) -> Option<&mut ShapedText<'a>> {
self.0.last_mut()?.text_mut() self.0.last_mut()?.item.text_mut()
} }
/// Reorder the items starting at the given index to RTL. /// Reorder the items starting at the given index to RTL.
@ -674,12 +675,17 @@ impl<'a> Items<'a> {
impl<'a> FromIterator<ItemEntry<'a>> for Items<'a> { impl<'a> FromIterator<ItemEntry<'a>> for Items<'a> {
fn from_iter<I: IntoIterator<Item = ItemEntry<'a>>>(iter: I) -> Self { fn from_iter<I: IntoIterator<Item = ItemEntry<'a>>>(iter: I) -> Self {
Self(iter.into_iter().collect()) Self(
iter.into_iter()
.enumerate()
.map(|(idx, item)| IndexedItemEntry { item, idx })
.collect(),
)
} }
} }
impl<'a> Deref for Items<'a> { impl<'a> Deref for Items<'a> {
type Target = Vec<ItemEntry<'a>>; type Target = Vec<IndexedItemEntry<'a>>;
fn deref(&self) -> &Self::Target { fn deref(&self) -> &Self::Target {
&self.0 &self.0
@ -698,6 +704,13 @@ impl Debug for Items<'_> {
} }
} }
/// An item accompanied by its position within a line.
#[derive(Debug)]
pub struct IndexedItemEntry<'a> {
pub item: ItemEntry<'a>,
pub idx: usize,
}
/// A reference to or a boxed item. /// A reference to or a boxed item.
/// ///
/// This is conceptually similar to a [`Cow<'a, Item<'a>>`][std::borrow::Cow], /// This is conceptually similar to a [`Cow<'a, Item<'a>>`][std::borrow::Cow],

View File

@ -110,10 +110,10 @@ pub fn linebreak<'a>(
p: &'a Preparation<'a>, p: &'a Preparation<'a>,
width: Abs, width: Abs,
) -> Vec<Line<'a>> { ) -> Vec<Line<'a>> {
match p.config.linebreaks { dbg!(match p.config.linebreaks {
Linebreaks::Simple => linebreak_simple(engine, p, width), Linebreaks::Simple => linebreak_simple(engine, p, width),
Linebreaks::Optimized => linebreak_optimized(engine, p, width), Linebreaks::Optimized => linebreak_optimized(engine, p, width),
} })
} }
/// Performs line breaking in simple first-fit style. This means that we build /// Performs line breaking in simple first-fit style. This means that we build

View File

@ -118,7 +118,6 @@ pub fn prepare<'a>(
if config.cjk_latin_spacing { if config.cjk_latin_spacing {
add_cjk_latin_spacing(&mut items); add_cjk_latin_spacing(&mut items);
} }
dbg!(&items);
Ok(Preparation { Ok(Preparation {
config, config,

View File

@ -539,6 +539,7 @@ impl IntoValue for CslSource {
/// memoization) for the whole document. This setup is necessary because /// memoization) for the whole document. This setup is necessary because
/// citation formatting is inherently stateful and we need access to all /// citation formatting is inherently stateful and we need access to all
/// citations to do it. /// citations to do it.
#[derive(Debug)]
pub(super) struct Works { pub(super) struct Works {
/// Maps from the location of a citation group to its rendered content. /// Maps from the location of a citation group to its rendered content.
pub citations: HashMap<Location, SourceResult<Content>>, pub citations: HashMap<Location, SourceResult<Content>>,