highlighted_label.rs

  1use std::ops::Range;
  2
  3use gpui::{HighlightStyle, StyledText};
  4
  5use crate::{prelude::*, LabelCommon, LabelLike, LabelSize, LineHeightStyle};
  6
  7#[derive(IntoElement)]
  8pub struct HighlightedLabel {
  9    base: LabelLike,
 10    label: SharedString,
 11    highlight_indices: Vec<usize>,
 12}
 13
 14impl HighlightedLabel {
 15    /// Constructs a label with the given characters highlighted.
 16    /// Characters are identified by UTF-8 byte position.
 17    pub fn new(label: impl Into<SharedString>, highlight_indices: Vec<usize>) -> Self {
 18        Self {
 19            base: LabelLike::new(),
 20            label: label.into(),
 21            highlight_indices,
 22        }
 23    }
 24}
 25
 26impl LabelCommon for HighlightedLabel {
 27    fn size(mut self, size: LabelSize) -> Self {
 28        self.base = self.base.size(size);
 29        self
 30    }
 31
 32    fn line_height_style(mut self, line_height_style: LineHeightStyle) -> Self {
 33        self.base = self.base.line_height_style(line_height_style);
 34        self
 35    }
 36
 37    fn color(mut self, color: Color) -> Self {
 38        self.base = self.base.color(color);
 39        self
 40    }
 41
 42    fn strikethrough(mut self, strikethrough: bool) -> Self {
 43        self.base = self.base.strikethrough(strikethrough);
 44        self
 45    }
 46
 47    fn italic(mut self, italic: bool) -> Self {
 48        self.base = self.base.italic(italic);
 49        self
 50    }
 51}
 52
 53pub fn highlight_ranges(
 54    text: &str,
 55    indices: &Vec<usize>,
 56    style: HighlightStyle,
 57) -> Vec<(Range<usize>, HighlightStyle)> {
 58    let mut highlight_indices = indices.iter().copied().peekable();
 59    let mut highlights: Vec<(Range<usize>, HighlightStyle)> = Vec::new();
 60
 61    while let Some(start_ix) = highlight_indices.next() {
 62        let mut end_ix = start_ix;
 63
 64        loop {
 65            end_ix = end_ix + text[end_ix..].chars().next().unwrap().len_utf8();
 66            if let Some(&next_ix) = highlight_indices.peek() {
 67                if next_ix == end_ix {
 68                    end_ix = next_ix;
 69                    highlight_indices.next();
 70                    continue;
 71                }
 72            }
 73            break;
 74        }
 75
 76        highlights.push((start_ix..end_ix, style));
 77    }
 78
 79    highlights
 80}
 81
 82impl RenderOnce for HighlightedLabel {
 83    fn render(self, cx: &mut WindowContext) -> impl IntoElement {
 84        let highlight_color = cx.theme().colors().text_accent;
 85
 86        let highlights = highlight_ranges(
 87            &self.label,
 88            &self.highlight_indices,
 89            HighlightStyle {
 90                color: Some(highlight_color),
 91                ..Default::default()
 92            },
 93        );
 94
 95        let mut text_style = cx.text_style();
 96        text_style.color = self.base.color.color(cx);
 97
 98        self.base
 99            .child(StyledText::new(self.label).with_highlights(&text_style, highlights))
100    }
101}