diff options
| author | 2024-07-17 13:00:00 +0200 | |
|---|---|---|
| committer | 2024-07-17 13:00:00 +0200 | |
| commit | 616689ca54942a13aac3615e571ae995ad4571b6 (patch) | |
| tree | 763d8926cfa2be97dffa858a04036e207fd06ec8 /graphics | |
| parent | b518e30610fa53691c727852f70b497dd19cfc7a (diff) | |
| download | iced-616689ca54942a13aac3615e571ae995ad4571b6.tar.gz iced-616689ca54942a13aac3615e571ae995ad4571b6.tar.bz2 iced-616689ca54942a13aac3615e571ae995ad4571b6.zip | |
Update `cosmic-text` and `resvg` (#2416)
* Update `cosmic-text`, `glyphon`, and `resvg`
* Fix slow font fallback with `Shaping::Basic` in `cosmic-text`
* Update `cosmic-text` and `resvg`
* Update `cosmic-text`
* Fix `SelectAll` action in `editor`
* Fix some panics in `graphics::text::editor`
* Remove empty `if` statement in `tiny_skia::vector`
* Update `cosmic-text`, `glyphon`, and `rustc-hash`
Diffstat (limited to '')
| -rw-r--r-- | graphics/src/geometry/text.rs | 5 | ||||
| -rw-r--r-- | graphics/src/text/cache.rs | 4 | ||||
| -rw-r--r-- | graphics/src/text/editor.rs | 289 | ||||
| -rw-r--r-- | graphics/src/text/paragraph.rs | 8 | 
4 files changed, 136 insertions, 170 deletions
| diff --git a/graphics/src/geometry/text.rs b/graphics/src/geometry/text.rs index d314e85e..90147f87 100644 --- a/graphics/src/geometry/text.rs +++ b/graphics/src/geometry/text.rs @@ -43,6 +43,7 @@ impl Text {          let mut buffer = cosmic_text::BufferLine::new(              &self.content, +            cosmic_text::LineEnding::default(),              cosmic_text::AttrsList::new(text::to_attributes(self.font)),              text::to_shaping(self.shaping),          ); @@ -50,8 +51,10 @@ impl Text {          let layout = buffer.layout(              font_system.raw(),              self.size.0, -            f32::MAX, +            None,              cosmic_text::Wrap::None, +            None, +            4,          );          let translation_x = match self.horizontal_alignment { diff --git a/graphics/src/text/cache.rs b/graphics/src/text/cache.rs index 822b61c4..e64d93f1 100644 --- a/graphics/src/text/cache.rs +++ b/graphics/src/text/cache.rs @@ -48,8 +48,8 @@ impl Cache {              buffer.set_size(                  font_system, -                key.bounds.width, -                key.bounds.height.max(key.line_height), +                Some(key.bounds.width), +                Some(key.bounds.height.max(key.line_height)),              );              buffer.set_text(                  font_system, diff --git a/graphics/src/text/editor.rs b/graphics/src/text/editor.rs index 36b4ca6e..3e6ef70c 100644 --- a/graphics/src/text/editor.rs +++ b/graphics/src/text/editor.rs @@ -17,7 +17,7 @@ use std::sync::{self, Arc};  pub struct Editor(Option<Arc<Internal>>);  struct Internal { -    editor: cosmic_text::Editor, +    editor: cosmic_text::Editor<'static>,      font: Font,      bounds: Size,      topmost_line_changed: Option<usize>, @@ -32,7 +32,7 @@ impl Editor {      /// Returns the buffer of the [`Editor`].      pub fn buffer(&self) -> &cosmic_text::Buffer { -        self.internal().editor.buffer() +        buffer_from_editor(&self.internal().editor)      }      /// Creates a [`Weak`] reference to the [`Editor`]. @@ -101,16 +101,10 @@ impl editor::Editor for Editor {          let internal = self.internal();          let cursor = internal.editor.cursor(); -        let buffer = internal.editor.buffer(); - -        match internal.editor.select_opt() { -            Some(selection) => { -                let (start, end) = if cursor < selection { -                    (cursor, selection) -                } else { -                    (selection, cursor) -                }; +        let buffer = buffer_from_editor(&internal.editor); +        match internal.editor.selection_bounds() { +            Some((start, end)) => {                  let line_height = buffer.metrics().line_height;                  let selected_lines = end.line - start.line + 1; @@ -142,7 +136,8 @@ impl editor::Editor for Editor {                                  width,                                  y: (visual_line as i32 + visual_lines_offset)                                      as f32 -                                    * line_height, +                                    * line_height +                                    - buffer.scroll().vertical,                                  height: line_height,                              })                          } else { @@ -224,7 +219,8 @@ impl editor::Editor for Editor {                  Cursor::Caret(Point::new(                      offset,                      (visual_lines_offset + visual_line as i32) as f32 -                        * line_height, +                        * line_height +                        - buffer.scroll().vertical,                  ))              }          } @@ -252,16 +248,8 @@ impl editor::Editor for Editor {          match action {              // Motion events              Action::Move(motion) => { -                if let Some(selection) = editor.select_opt() { -                    let cursor = editor.cursor(); - -                    let (left, right) = if cursor < selection { -                        (cursor, selection) -                    } else { -                        (selection, cursor) -                    }; - -                    editor.set_select_opt(None); +                if let Some((start, end)) = editor.selection_bounds() { +                    editor.set_selection(cosmic_text::Selection::None);                      match motion {                          // These motions are performed as-is even when a selection @@ -272,17 +260,20 @@ impl editor::Editor for Editor {                          | Motion::DocumentEnd => {                              editor.action(                                  font_system.raw(), -                                motion_to_action(motion), +                                cosmic_text::Action::Motion(to_motion(motion)),                              );                          }                          // Other motions simply move the cursor to one end of the selection                          _ => editor.set_cursor(match motion.direction() { -                            Direction::Left => left, -                            Direction::Right => right, +                            Direction::Left => start, +                            Direction::Right => end,                          }),                      }                  } else { -                    editor.action(font_system.raw(), motion_to_action(motion)); +                    editor.action( +                        font_system.raw(), +                        cosmic_text::Action::Motion(to_motion(motion)), +                    );                  }              } @@ -290,103 +281,36 @@ impl editor::Editor for Editor {              Action::Select(motion) => {                  let cursor = editor.cursor(); -                if editor.select_opt().is_none() { -                    editor.set_select_opt(Some(cursor)); +                if editor.selection_bounds().is_none() { +                    editor +                        .set_selection(cosmic_text::Selection::Normal(cursor));                  } -                editor.action(font_system.raw(), motion_to_action(motion)); +                editor.action( +                    font_system.raw(), +                    cosmic_text::Action::Motion(to_motion(motion)), +                );                  // Deselect if selection matches cursor position -                if let Some(selection) = editor.select_opt() { -                    let cursor = editor.cursor(); - -                    if cursor.line == selection.line -                        && cursor.index == selection.index -                    { -                        editor.set_select_opt(None); +                if let Some((start, end)) = editor.selection_bounds() { +                    if start.line == end.line && start.index == end.index { +                        editor.set_selection(cosmic_text::Selection::None);                      }                  }              }              Action::SelectWord => { -                use unicode_segmentation::UnicodeSegmentation; -                  let cursor = editor.cursor(); -                if let Some(line) = editor.buffer().lines.get(cursor.line) { -                    let (start, end) = -                        UnicodeSegmentation::unicode_word_indices(line.text()) -                            // Split words with dots -                            .flat_map(|(i, word)| { -                                word.split('.').scan(i, |current, word| { -                                    let start = *current; -                                    *current += word.len() + 1; - -                                    Some((start, word)) -                                }) -                            }) -                            // Turn words into ranges -                            .map(|(i, word)| (i, i + word.len())) -                            // Find the word at cursor -                            .find(|&(start, end)| { -                                start <= cursor.index && cursor.index < end -                            }) -                            // Cursor is not in a word. Let's select its punctuation cluster. -                            .unwrap_or_else(|| { -                                let start = line.text()[..cursor.index] -                                    .char_indices() -                                    .rev() -                                    .take_while(|(_, c)| { -                                        c.is_ascii_punctuation() -                                    }) -                                    .map(|(i, _)| i) -                                    .last() -                                    .unwrap_or(cursor.index); - -                                let end = line.text()[cursor.index..] -                                    .char_indices() -                                    .skip_while(|(_, c)| { -                                        c.is_ascii_punctuation() -                                    }) -                                    .map(|(i, _)| i + cursor.index) -                                    .next() -                                    .unwrap_or(cursor.index); - -                                (start, end) -                            }); - -                    if start != end { -                        editor.set_cursor(cosmic_text::Cursor { -                            index: start, -                            ..cursor -                        }); - -                        editor.set_select_opt(Some(cosmic_text::Cursor { -                            index: end, -                            ..cursor -                        })); -                    } -                } +                editor.set_selection(cosmic_text::Selection::Word(cursor));              }              Action::SelectLine => {                  let cursor = editor.cursor(); -                if let Some(line_length) = editor -                    .buffer() -                    .lines -                    .get(cursor.line) -                    .map(|line| line.text().len()) -                { -                    editor -                        .set_cursor(cosmic_text::Cursor { index: 0, ..cursor }); - -                    editor.set_select_opt(Some(cosmic_text::Cursor { -                        index: line_length, -                        ..cursor -                    })); -                } +                editor.set_selection(cosmic_text::Selection::Line(cursor));              }              Action::SelectAll => { -                let buffer = editor.buffer(); +                let buffer = buffer_from_editor(editor); +                  if buffer.lines.len() > 1                      || buffer                          .lines @@ -394,15 +318,20 @@ impl editor::Editor for Editor {                          .is_some_and(|line| !line.text().is_empty())                  {                      let cursor = editor.cursor(); -                    editor.set_select_opt(Some(cosmic_text::Cursor { -                        line: 0, -                        index: 0, -                        ..cursor -                    })); + +                    editor.set_selection(cosmic_text::Selection::Normal( +                        cosmic_text::Cursor { +                            line: 0, +                            index: 0, +                            ..cursor +                        }, +                    ));                      editor.action(                          font_system.raw(), -                        motion_to_action(Motion::DocumentEnd), +                        cosmic_text::Action::Motion( +                            cosmic_text::Motion::BufferEnd, +                        ),                      );                  }              } @@ -440,10 +369,12 @@ impl editor::Editor for Editor {                  }                  let cursor = editor.cursor(); -                let selection = editor.select_opt().unwrap_or(cursor); +                let selection_start = editor +                    .selection_bounds() +                    .map(|(start, _)| start) +                    .unwrap_or(cursor); -                internal.topmost_line_changed = -                    Some(cursor.min(selection).line); +                internal.topmost_line_changed = Some(selection_start.line);              }              // Mouse events @@ -466,13 +397,9 @@ impl editor::Editor for Editor {                  );                  // Deselect if selection matches cursor position -                if let Some(selection) = editor.select_opt() { -                    let cursor = editor.cursor(); - -                    if cursor.line == selection.line -                        && cursor.index == selection.index -                    { -                        editor.set_select_opt(None); +                if let Some((start, end)) = editor.selection_bounds() { +                    if start.line == end.line && start.index == end.index { +                        editor.set_selection(cosmic_text::Selection::None);                      }                  }              } @@ -494,7 +421,7 @@ impl editor::Editor for Editor {      fn min_bounds(&self) -> Size {          let internal = self.internal(); -        text::measure(internal.editor.buffer()) +        text::measure(buffer_from_editor(&internal.editor))      }      fn update( @@ -517,7 +444,10 @@ impl editor::Editor for Editor {          if font_system.version() != internal.version {              log::trace!("Updating `FontSystem` of `Editor`..."); -            for line in internal.editor.buffer_mut().lines.iter_mut() { +            for line in buffer_mut_from_editor(&mut internal.editor) +                .lines +                .iter_mut() +            {                  line.reset();              } @@ -528,7 +458,10 @@ impl editor::Editor for Editor {          if new_font != internal.font {              log::trace!("Updating font of `Editor`..."); -            for line in internal.editor.buffer_mut().lines.iter_mut() { +            for line in buffer_mut_from_editor(&mut internal.editor) +                .lines +                .iter_mut() +            {                  let _ = line.set_attrs_list(cosmic_text::AttrsList::new(                      text::to_attributes(new_font),                  )); @@ -538,7 +471,7 @@ impl editor::Editor for Editor {              internal.topmost_line_changed = Some(0);          } -        let metrics = internal.editor.buffer().metrics(); +        let metrics = buffer_from_editor(&internal.editor).metrics();          let new_line_height = new_line_height.to_absolute(new_size);          if new_size.0 != metrics.font_size @@ -546,7 +479,7 @@ impl editor::Editor for Editor {          {              log::trace!("Updating `Metrics` of `Editor`..."); -            internal.editor.buffer_mut().set_metrics( +            buffer_mut_from_editor(&mut internal.editor).set_metrics(                  font_system.raw(),                  cosmic_text::Metrics::new(new_size.0, new_line_height.0),              ); @@ -555,10 +488,10 @@ impl editor::Editor for Editor {          if new_bounds != internal.bounds {              log::trace!("Updating size of `Editor`..."); -            internal.editor.buffer_mut().set_size( +            buffer_mut_from_editor(&mut internal.editor).set_size(                  font_system.raw(), -                new_bounds.width, -                new_bounds.height, +                Some(new_bounds.width), +                Some(new_bounds.height),              );              internal.bounds = new_bounds; @@ -573,7 +506,7 @@ impl editor::Editor for Editor {              new_highlighter.change_line(topmost_line_changed);          } -        internal.editor.shape_as_needed(font_system.raw()); +        internal.editor.shape_as_needed(font_system.raw(), false);          self.0 = Some(Arc::new(internal));      } @@ -585,12 +518,13 @@ impl editor::Editor for Editor {          format_highlight: impl Fn(&H::Highlight) -> highlighter::Format<Self::Font>,      ) {          let internal = self.internal(); -        let buffer = internal.editor.buffer(); +        let buffer = buffer_from_editor(&internal.editor); -        let mut window = buffer.scroll() + buffer.visible_lines(); +        let scroll = buffer.scroll(); +        let mut window = (internal.bounds.height / buffer.metrics().line_height) +            .ceil() as i32; -        let last_visible_line = buffer -            .lines +        let last_visible_line = buffer.lines[scroll.line..]              .iter()              .enumerate()              .find_map(|(i, line)| { @@ -604,7 +538,7 @@ impl editor::Editor for Editor {                      window -= visible_lines;                      None                  } else { -                    Some(i) +                    Some(scroll.line + i)                  }              })              .unwrap_or(buffer.lines.len().saturating_sub(1)); @@ -626,7 +560,7 @@ impl editor::Editor for Editor {          let attributes = text::to_attributes(font); -        for line in &mut internal.editor.buffer_mut().lines +        for line in &mut buffer_mut_from_editor(&mut internal.editor).lines              [current_line..=last_visible_line]          {              let mut list = cosmic_text::AttrsList::new(attributes); @@ -652,7 +586,7 @@ impl editor::Editor for Editor {              let _ = line.set_attrs_list(list);          } -        internal.editor.shape_as_needed(font_system.raw()); +        internal.editor.shape_as_needed(font_system.raw(), false);          self.0 = Some(Arc::new(internal));      } @@ -668,7 +602,8 @@ impl PartialEq for Internal {      fn eq(&self, other: &Self) -> bool {          self.font == other.font              && self.bounds == other.bounds -            && self.editor.buffer().metrics() == other.editor.buffer().metrics() +            && buffer_from_editor(&self.editor).metrics() +                == buffer_from_editor(&other.editor).metrics()      }  } @@ -730,7 +665,8 @@ fn highlight_line(      let layout = line          .layout_opt()          .as_ref() -        .expect("Line layout should be cached"); +        .map(Vec::as_slice) +        .unwrap_or_default();      layout.iter().map(move |visual_line| {          let start = visual_line @@ -773,34 +709,61 @@ fn highlight_line(  }  fn visual_lines_offset(line: usize, buffer: &cosmic_text::Buffer) -> i32 { -    let visual_lines_before_start: usize = buffer -        .lines +    let scroll = buffer.scroll(); + +    let start = scroll.line.min(line); +    let end = scroll.line.max(line); + +    let visual_lines_offset: usize = buffer.lines[start..]          .iter() -        .take(line) +        .take(end - start)          .map(|line| { -            line.layout_opt() -                .as_ref() -                .expect("Line layout should be cached") -                .len() +            line.layout_opt().as_ref().map(Vec::len).unwrap_or_default()          })          .sum(); -    visual_lines_before_start as i32 - buffer.scroll() +    visual_lines_offset as i32 * if scroll.line < line { 1 } else { -1 }  } -fn motion_to_action(motion: Motion) -> cosmic_text::Action { +fn to_motion(motion: Motion) -> cosmic_text::Motion {      match motion { -        Motion::Left => cosmic_text::Action::Left, -        Motion::Right => cosmic_text::Action::Right, -        Motion::Up => cosmic_text::Action::Up, -        Motion::Down => cosmic_text::Action::Down, -        Motion::WordLeft => cosmic_text::Action::LeftWord, -        Motion::WordRight => cosmic_text::Action::RightWord, -        Motion::Home => cosmic_text::Action::Home, -        Motion::End => cosmic_text::Action::End, -        Motion::PageUp => cosmic_text::Action::PageUp, -        Motion::PageDown => cosmic_text::Action::PageDown, -        Motion::DocumentStart => cosmic_text::Action::BufferStart, -        Motion::DocumentEnd => cosmic_text::Action::BufferEnd, +        Motion::Left => cosmic_text::Motion::Left, +        Motion::Right => cosmic_text::Motion::Right, +        Motion::Up => cosmic_text::Motion::Up, +        Motion::Down => cosmic_text::Motion::Down, +        Motion::WordLeft => cosmic_text::Motion::LeftWord, +        Motion::WordRight => cosmic_text::Motion::RightWord, +        Motion::Home => cosmic_text::Motion::Home, +        Motion::End => cosmic_text::Motion::End, +        Motion::PageUp => cosmic_text::Motion::PageUp, +        Motion::PageDown => cosmic_text::Motion::PageDown, +        Motion::DocumentStart => cosmic_text::Motion::BufferStart, +        Motion::DocumentEnd => cosmic_text::Motion::BufferEnd, +    } +} + +fn buffer_from_editor<'a, 'b>( +    editor: &'a impl cosmic_text::Edit<'b>, +) -> &'a cosmic_text::Buffer +where +    'b: 'a, +{ +    match editor.buffer_ref() { +        cosmic_text::BufferRef::Owned(buffer) => buffer, +        cosmic_text::BufferRef::Borrowed(buffer) => buffer, +        cosmic_text::BufferRef::Arc(buffer) => buffer, +    } +} + +fn buffer_mut_from_editor<'a, 'b>( +    editor: &'a mut impl cosmic_text::Edit<'b>, +) -> &'a mut cosmic_text::Buffer +where +    'b: 'a, +{ +    match editor.buffer_ref_mut() { +        cosmic_text::BufferRef::Owned(buffer) => buffer, +        cosmic_text::BufferRef::Borrowed(buffer) => buffer, +        cosmic_text::BufferRef::Arc(_buffer) => unreachable!(),      }  } diff --git a/graphics/src/text/paragraph.rs b/graphics/src/text/paragraph.rs index 31a323ac..a5fefe8f 100644 --- a/graphics/src/text/paragraph.rs +++ b/graphics/src/text/paragraph.rs @@ -77,8 +77,8 @@ impl core::text::Paragraph for Paragraph {          buffer.set_size(              font_system.raw(), -            text.bounds.width, -            text.bounds.height, +            Some(text.bounds.width), +            Some(text.bounds.height),          );          buffer.set_text( @@ -116,8 +116,8 @@ impl core::text::Paragraph for Paragraph {                  internal.buffer.set_size(                      font_system.raw(), -                    new_bounds.width, -                    new_bounds.height, +                    Some(new_bounds.width), +                    Some(new_bounds.height),                  );                  internal.bounds = new_bounds; | 
