231 lines
7.5 KiB
Rust
231 lines
7.5 KiB
Rust
use std::ops::Range;
|
|
|
|
use gpui::{FontWeight, HighlightStyle, StyledText};
|
|
|
|
use crate::{LabelCommon, LabelLike, LabelSize, LineHeightStyle, prelude::*};
|
|
|
|
#[derive(IntoElement, RegisterComponent)]
|
|
pub struct HighlightedLabel {
|
|
base: LabelLike,
|
|
label: SharedString,
|
|
highlight_indices: Vec<usize>,
|
|
}
|
|
|
|
impl HighlightedLabel {
|
|
/// Constructs a label with the given characters highlighted.
|
|
/// Characters are identified by UTF-8 byte position.
|
|
pub fn new(label: impl Into<SharedString>, highlight_indices: Vec<usize>) -> Self {
|
|
Self {
|
|
base: LabelLike::new(),
|
|
label: label.into(),
|
|
highlight_indices,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl LabelCommon for HighlightedLabel {
|
|
fn size(mut self, size: LabelSize) -> Self {
|
|
self.base = self.base.size(size);
|
|
self
|
|
}
|
|
|
|
fn weight(mut self, weight: FontWeight) -> Self {
|
|
self.base = self.base.weight(weight);
|
|
self
|
|
}
|
|
|
|
fn line_height_style(mut self, line_height_style: LineHeightStyle) -> Self {
|
|
self.base = self.base.line_height_style(line_height_style);
|
|
self
|
|
}
|
|
|
|
fn color(mut self, color: Color) -> Self {
|
|
self.base = self.base.color(color);
|
|
self
|
|
}
|
|
|
|
fn strikethrough(mut self) -> Self {
|
|
self.base = self.base.strikethrough();
|
|
self
|
|
}
|
|
|
|
fn italic(mut self) -> Self {
|
|
self.base = self.base.italic();
|
|
self
|
|
}
|
|
|
|
fn alpha(mut self, alpha: f32) -> Self {
|
|
self.base = self.base.alpha(alpha);
|
|
self
|
|
}
|
|
|
|
fn underline(mut self) -> Self {
|
|
self.base = self.base.underline();
|
|
self
|
|
}
|
|
|
|
fn truncate(mut self) -> Self {
|
|
self.base = self.base.truncate();
|
|
self
|
|
}
|
|
|
|
fn single_line(mut self) -> Self {
|
|
self.base = self.base.single_line();
|
|
self
|
|
}
|
|
|
|
fn buffer_font(mut self, cx: &App) -> Self {
|
|
self.base = self.base.buffer_font(cx);
|
|
self
|
|
}
|
|
|
|
fn inline_code(mut self, cx: &App) -> Self {
|
|
self.base = self.base.inline_code(cx);
|
|
self
|
|
}
|
|
}
|
|
|
|
pub fn highlight_ranges(
|
|
text: &str,
|
|
indices: &[usize],
|
|
style: HighlightStyle,
|
|
) -> Vec<(Range<usize>, HighlightStyle)> {
|
|
let mut highlight_indices = indices.iter().copied().peekable();
|
|
let mut highlights: Vec<(Range<usize>, HighlightStyle)> = Vec::new();
|
|
|
|
while let Some(start_ix) = highlight_indices.next() {
|
|
let mut end_ix = start_ix;
|
|
|
|
loop {
|
|
end_ix = end_ix + text[end_ix..].chars().next().unwrap().len_utf8();
|
|
if let Some(&next_ix) = highlight_indices.peek()
|
|
&& next_ix == end_ix {
|
|
end_ix = next_ix;
|
|
highlight_indices.next();
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
|
|
highlights.push((start_ix..end_ix, style));
|
|
}
|
|
|
|
highlights
|
|
}
|
|
|
|
impl RenderOnce for HighlightedLabel {
|
|
fn render(self, window: &mut Window, cx: &mut App) -> impl IntoElement {
|
|
let highlight_color = cx.theme().colors().text_accent;
|
|
|
|
let highlights = highlight_ranges(
|
|
&self.label,
|
|
&self.highlight_indices,
|
|
HighlightStyle {
|
|
color: Some(highlight_color),
|
|
..Default::default()
|
|
},
|
|
);
|
|
|
|
let mut text_style = window.text_style();
|
|
text_style.color = self.base.color.color(cx);
|
|
|
|
self.base
|
|
.child(StyledText::new(self.label).with_default_highlights(&text_style, highlights))
|
|
}
|
|
}
|
|
|
|
impl Component for HighlightedLabel {
|
|
fn scope() -> ComponentScope {
|
|
ComponentScope::Typography
|
|
}
|
|
|
|
fn name() -> &'static str {
|
|
"HighlightedLabel"
|
|
}
|
|
|
|
fn description() -> Option<&'static str> {
|
|
Some("A label with highlighted characters based on specified indices.")
|
|
}
|
|
|
|
fn preview(_window: &mut Window, _cx: &mut App) -> Option<AnyElement> {
|
|
Some(
|
|
v_flex()
|
|
.gap_6()
|
|
.children(vec![
|
|
example_group_with_title(
|
|
"Basic Usage",
|
|
vec![
|
|
single_example(
|
|
"Default",
|
|
HighlightedLabel::new("Highlighted Text", vec![0, 1, 2, 3]).into_any_element(),
|
|
),
|
|
single_example(
|
|
"Custom Color",
|
|
HighlightedLabel::new("Colored Highlight", vec![0, 1, 7, 8, 9])
|
|
.color(Color::Accent)
|
|
.into_any_element(),
|
|
),
|
|
],
|
|
),
|
|
example_group_with_title(
|
|
"Styles",
|
|
vec![
|
|
single_example(
|
|
"Bold",
|
|
HighlightedLabel::new("Bold Highlight", vec![0, 1, 2, 3])
|
|
.weight(FontWeight::BOLD)
|
|
.into_any_element(),
|
|
),
|
|
single_example(
|
|
"Italic",
|
|
HighlightedLabel::new("Italic Highlight", vec![0, 1, 6, 7, 8])
|
|
.italic()
|
|
.into_any_element(),
|
|
),
|
|
single_example(
|
|
"Underline",
|
|
HighlightedLabel::new("Underlined Highlight", vec![0, 1, 10, 11, 12])
|
|
.underline()
|
|
.into_any_element(),
|
|
),
|
|
],
|
|
),
|
|
example_group_with_title(
|
|
"Sizes",
|
|
vec![
|
|
single_example(
|
|
"Small",
|
|
HighlightedLabel::new("Small Highlight", vec![0, 1, 5, 6, 7])
|
|
.size(LabelSize::Small)
|
|
.into_any_element(),
|
|
),
|
|
single_example(
|
|
"Large",
|
|
HighlightedLabel::new("Large Highlight", vec![0, 1, 5, 6, 7])
|
|
.size(LabelSize::Large)
|
|
.into_any_element(),
|
|
),
|
|
],
|
|
),
|
|
example_group_with_title(
|
|
"Special Cases",
|
|
vec![
|
|
single_example(
|
|
"Single Line",
|
|
HighlightedLabel::new("Single Line Highlight\nWith Newline", vec![0, 1, 7, 8, 9])
|
|
.single_line()
|
|
.into_any_element(),
|
|
),
|
|
single_example(
|
|
"Truncate",
|
|
HighlightedLabel::new("This is a very long text that should be truncated with highlights", vec![0, 1, 2, 3, 4, 5])
|
|
.truncate()
|
|
.into_any_element(),
|
|
),
|
|
],
|
|
),
|
|
])
|
|
.into_any_element()
|
|
)
|
|
}
|
|
}
|