Style block decorations
Co-Authored-By: Nathan Sobo <nathan@zed.dev>
This commit is contained in:
parent
65d4c33c0e
commit
7250974aa6
5 changed files with 117 additions and 66 deletions
|
@ -5,6 +5,7 @@ use collections::{BTreeSet, HashMap, HashSet};
|
||||||
use editor::{
|
use editor::{
|
||||||
diagnostic_block_renderer,
|
diagnostic_block_renderer,
|
||||||
display_map::{BlockDisposition, BlockId, BlockProperties, RenderBlock},
|
display_map::{BlockDisposition, BlockId, BlockProperties, RenderBlock},
|
||||||
|
highlight_diagnostic_message,
|
||||||
items::BufferItemHandle,
|
items::BufferItemHandle,
|
||||||
Autoscroll, BuildSettings, Editor, ExcerptId, ExcerptProperties, MultiBuffer, ToOffset,
|
Autoscroll, BuildSettings, Editor, ExcerptId, ExcerptProperties, MultiBuffer, ToOffset,
|
||||||
};
|
};
|
||||||
|
@ -703,10 +704,10 @@ fn diagnostic_header_renderer(
|
||||||
let style = &settings.style.diagnostic_header;
|
let style = &settings.style.diagnostic_header;
|
||||||
let icon = if diagnostic.severity == DiagnosticSeverity::ERROR {
|
let icon = if diagnostic.severity == DiagnosticSeverity::ERROR {
|
||||||
Svg::new("icons/diagnostic-error-10.svg")
|
Svg::new("icons/diagnostic-error-10.svg")
|
||||||
.with_color(settings.style.error_diagnostic.text)
|
.with_color(settings.style.error_diagnostic.message.text.color)
|
||||||
} else {
|
} else {
|
||||||
Svg::new("icons/diagnostic-warning-10.svg")
|
Svg::new("icons/diagnostic-warning-10.svg")
|
||||||
.with_color(settings.style.warning_diagnostic.text)
|
.with_color(settings.style.warning_diagnostic.message.text.color)
|
||||||
};
|
};
|
||||||
|
|
||||||
Flex::row()
|
Flex::row()
|
||||||
|
@ -741,28 +742,6 @@ fn diagnostic_header_renderer(
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn highlight_diagnostic_message(message: &str) -> (String, Vec<usize>) {
|
|
||||||
let mut message_without_backticks = String::new();
|
|
||||||
let mut prev_offset = 0;
|
|
||||||
let mut inside_block = false;
|
|
||||||
let mut highlights = Vec::new();
|
|
||||||
for (match_ix, (offset, _)) in message
|
|
||||||
.match_indices('`')
|
|
||||||
.chain([(message.len(), "")])
|
|
||||||
.enumerate()
|
|
||||||
{
|
|
||||||
message_without_backticks.push_str(&message[prev_offset..offset]);
|
|
||||||
if inside_block {
|
|
||||||
highlights.extend(prev_offset - match_ix..offset - match_ix);
|
|
||||||
}
|
|
||||||
|
|
||||||
inside_block = !inside_block;
|
|
||||||
prev_offset = offset + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
(message_without_backticks, highlights)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn context_header_renderer(build_settings: BuildSettings) -> RenderBlock {
|
fn context_header_renderer(build_settings: BuildSettings) -> RenderBlock {
|
||||||
Arc::new(move |cx| {
|
Arc::new(move |cx| {
|
||||||
let settings = build_settings(cx);
|
let settings = build_settings(cx);
|
||||||
|
|
|
@ -3825,6 +3825,10 @@ impl EditorSettings {
|
||||||
font_properties,
|
font_properties,
|
||||||
underline: None,
|
underline: None,
|
||||||
};
|
};
|
||||||
|
let default_diagnostic_style = DiagnosticStyle {
|
||||||
|
message: text.clone().into(),
|
||||||
|
header: Default::default(),
|
||||||
|
};
|
||||||
EditorStyle {
|
EditorStyle {
|
||||||
text: text.clone(),
|
text: text.clone(),
|
||||||
placeholder_text: None,
|
placeholder_text: None,
|
||||||
|
@ -3860,14 +3864,14 @@ impl EditorSettings {
|
||||||
},
|
},
|
||||||
icon: Default::default(),
|
icon: Default::default(),
|
||||||
},
|
},
|
||||||
error_diagnostic: Default::default(),
|
error_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_error_diagnostic: Default::default(),
|
invalid_error_diagnostic: default_diagnostic_style.clone(),
|
||||||
warning_diagnostic: Default::default(),
|
warning_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_warning_diagnostic: Default::default(),
|
invalid_warning_diagnostic: default_diagnostic_style.clone(),
|
||||||
information_diagnostic: Default::default(),
|
information_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_information_diagnostic: Default::default(),
|
invalid_information_diagnostic: default_diagnostic_style.clone(),
|
||||||
hint_diagnostic: Default::default(),
|
hint_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_hint_diagnostic: Default::default(),
|
invalid_hint_diagnostic: default_diagnostic_style.clone(),
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -4008,33 +4012,68 @@ pub fn diagnostic_block_renderer(
|
||||||
is_valid: bool,
|
is_valid: bool,
|
||||||
build_settings: BuildSettings,
|
build_settings: BuildSettings,
|
||||||
) -> RenderBlock {
|
) -> RenderBlock {
|
||||||
|
let mut highlighted_lines = Vec::new();
|
||||||
|
for line in diagnostic.message.lines() {
|
||||||
|
highlighted_lines.push(highlight_diagnostic_message(line));
|
||||||
|
}
|
||||||
|
|
||||||
Arc::new(move |cx: &BlockContext| {
|
Arc::new(move |cx: &BlockContext| {
|
||||||
let settings = build_settings(cx);
|
let settings = build_settings(cx);
|
||||||
let mut text_style = settings.style.text.clone();
|
let style = diagnostic_style(diagnostic.severity, is_valid, &settings.style).message;
|
||||||
text_style.color = diagnostic_style(diagnostic.severity, is_valid, &settings.style).text;
|
Flex::column()
|
||||||
Text::new(diagnostic.message.clone(), text_style)
|
.with_children(highlighted_lines.iter().map(|(line, highlights)| {
|
||||||
.with_soft_wrap(false)
|
Label::new(line.clone(), style.clone())
|
||||||
.contained()
|
.with_highlights(highlights.clone())
|
||||||
.with_margin_left(cx.anchor_x)
|
.contained()
|
||||||
|
.with_margin_left(cx.anchor_x)
|
||||||
|
.boxed()
|
||||||
|
}))
|
||||||
|
.aligned()
|
||||||
|
.left()
|
||||||
.boxed()
|
.boxed()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn highlight_diagnostic_message(message: &str) -> (String, Vec<usize>) {
|
||||||
|
let mut message_without_backticks = String::new();
|
||||||
|
let mut prev_offset = 0;
|
||||||
|
let mut inside_block = false;
|
||||||
|
let mut highlights = Vec::new();
|
||||||
|
for (match_ix, (offset, _)) in message
|
||||||
|
.match_indices('`')
|
||||||
|
.chain([(message.len(), "")])
|
||||||
|
.enumerate()
|
||||||
|
{
|
||||||
|
message_without_backticks.push_str(&message[prev_offset..offset]);
|
||||||
|
if inside_block {
|
||||||
|
highlights.extend(prev_offset - match_ix..offset - match_ix);
|
||||||
|
}
|
||||||
|
|
||||||
|
inside_block = !inside_block;
|
||||||
|
prev_offset = offset + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
(message_without_backticks, highlights)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn diagnostic_style(
|
pub fn diagnostic_style(
|
||||||
severity: DiagnosticSeverity,
|
severity: DiagnosticSeverity,
|
||||||
valid: bool,
|
valid: bool,
|
||||||
style: &EditorStyle,
|
style: &EditorStyle,
|
||||||
) -> DiagnosticStyle {
|
) -> DiagnosticStyle {
|
||||||
match (severity, valid) {
|
match (severity, valid) {
|
||||||
(DiagnosticSeverity::ERROR, true) => style.error_diagnostic,
|
(DiagnosticSeverity::ERROR, true) => style.error_diagnostic.clone(),
|
||||||
(DiagnosticSeverity::ERROR, false) => style.invalid_error_diagnostic,
|
(DiagnosticSeverity::ERROR, false) => style.invalid_error_diagnostic.clone(),
|
||||||
(DiagnosticSeverity::WARNING, true) => style.warning_diagnostic,
|
(DiagnosticSeverity::WARNING, true) => style.warning_diagnostic.clone(),
|
||||||
(DiagnosticSeverity::WARNING, false) => style.invalid_warning_diagnostic,
|
(DiagnosticSeverity::WARNING, false) => style.invalid_warning_diagnostic.clone(),
|
||||||
(DiagnosticSeverity::INFORMATION, true) => style.information_diagnostic,
|
(DiagnosticSeverity::INFORMATION, true) => style.information_diagnostic.clone(),
|
||||||
(DiagnosticSeverity::INFORMATION, false) => style.invalid_information_diagnostic,
|
(DiagnosticSeverity::INFORMATION, false) => style.invalid_information_diagnostic.clone(),
|
||||||
(DiagnosticSeverity::HINT, true) => style.hint_diagnostic,
|
(DiagnosticSeverity::HINT, true) => style.hint_diagnostic.clone(),
|
||||||
(DiagnosticSeverity::HINT, false) => style.invalid_hint_diagnostic,
|
(DiagnosticSeverity::HINT, false) => style.invalid_hint_diagnostic.clone(),
|
||||||
_ => Default::default(),
|
_ => DiagnosticStyle {
|
||||||
|
message: style.text.clone().into(),
|
||||||
|
header: Default::default(),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -549,7 +549,12 @@ impl EditorElement {
|
||||||
.chunks(rows.clone(), Some(&style.syntax))
|
.chunks(rows.clone(), Some(&style.syntax))
|
||||||
.map(|chunk| {
|
.map(|chunk| {
|
||||||
let highlight = if let Some(severity) = chunk.diagnostic {
|
let highlight = if let Some(severity) = chunk.diagnostic {
|
||||||
let underline = Some(super::diagnostic_style(severity, true, style).text);
|
let underline = Some(
|
||||||
|
super::diagnostic_style(severity, true, style)
|
||||||
|
.message
|
||||||
|
.text
|
||||||
|
.color,
|
||||||
|
);
|
||||||
if let Some(mut highlight) = chunk.highlight_style {
|
if let Some(mut highlight) = chunk.highlight_style {
|
||||||
highlight.underline = underline;
|
highlight.underline = underline;
|
||||||
Some(highlight)
|
Some(highlight)
|
||||||
|
|
|
@ -287,9 +287,9 @@ pub struct DiagnosticHeaderIcon {
|
||||||
pub width: f32,
|
pub width: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone, Deserialize, Default)]
|
#[derive(Clone, Deserialize, Default)]
|
||||||
pub struct DiagnosticStyle {
|
pub struct DiagnosticStyle {
|
||||||
pub text: Color,
|
pub message: LabelStyle,
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub header: ContainerStyle,
|
pub header: ContainerStyle,
|
||||||
}
|
}
|
||||||
|
@ -327,6 +327,10 @@ impl EditorStyle {
|
||||||
|
|
||||||
impl InputEditorStyle {
|
impl InputEditorStyle {
|
||||||
pub fn as_editor(&self) -> EditorStyle {
|
pub fn as_editor(&self) -> EditorStyle {
|
||||||
|
let default_diagnostic_style = DiagnosticStyle {
|
||||||
|
message: self.text.clone().into(),
|
||||||
|
header: Default::default(),
|
||||||
|
};
|
||||||
EditorStyle {
|
EditorStyle {
|
||||||
text: self.text.clone(),
|
text: self.text.clone(),
|
||||||
placeholder_text: self.placeholder_text.clone(),
|
placeholder_text: self.placeholder_text.clone(),
|
||||||
|
@ -365,14 +369,14 @@ impl InputEditorStyle {
|
||||||
},
|
},
|
||||||
icon: Default::default(),
|
icon: Default::default(),
|
||||||
},
|
},
|
||||||
error_diagnostic: Default::default(),
|
error_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_error_diagnostic: Default::default(),
|
invalid_error_diagnostic: default_diagnostic_style.clone(),
|
||||||
warning_diagnostic: Default::default(),
|
warning_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_warning_diagnostic: Default::default(),
|
invalid_warning_diagnostic: default_diagnostic_style.clone(),
|
||||||
information_diagnostic: Default::default(),
|
information_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_information_diagnostic: Default::default(),
|
invalid_information_diagnostic: default_diagnostic_style.clone(),
|
||||||
hint_diagnostic: Default::default(),
|
hint_diagnostic: default_diagnostic_style.clone(),
|
||||||
invalid_hint_diagnostic: Default::default(),
|
invalid_hint_diagnostic: default_diagnostic_style.clone(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -251,10 +251,6 @@ line_number_active = "$text.0.color"
|
||||||
selection = "$selection.host"
|
selection = "$selection.host"
|
||||||
guest_selections = "$selection.guests"
|
guest_selections = "$selection.guests"
|
||||||
error_color = "$status.bad"
|
error_color = "$status.bad"
|
||||||
invalid_error_diagnostic = { text = "$text.3.color" }
|
|
||||||
invalid_warning_diagnostic = { text = "$text.3.color" }
|
|
||||||
invalid_information_diagnostic = { text = "$text.3.color" }
|
|
||||||
invalid_hint_diagnostic = { text = "$text.3.color" }
|
|
||||||
|
|
||||||
[editor.diagnostic_path_header]
|
[editor.diagnostic_path_header]
|
||||||
filename = { extends = "$text.0", size = 14 }
|
filename = { extends = "$text.0", size = 14 }
|
||||||
|
@ -271,21 +267,49 @@ text = { extends = "$text.1", size = 14 }
|
||||||
highlight_text = { extends = "$text.0", size = 14, weight = "bold" }
|
highlight_text = { extends = "$text.0", size = 14, weight = "bold" }
|
||||||
|
|
||||||
[editor.error_diagnostic]
|
[editor.error_diagnostic]
|
||||||
text = "$status.bad"
|
|
||||||
header.border = { width = 1, top = true, color = "$border.0" }
|
header.border = { width = 1, top = true, color = "$border.0" }
|
||||||
|
|
||||||
|
[editor.error_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$status.bad" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$status.bad", weight = "bold" }
|
||||||
|
|
||||||
[editor.warning_diagnostic]
|
[editor.warning_diagnostic]
|
||||||
text = "$status.warn"
|
|
||||||
header.border = { width = 1, top = true, color = "$border.0" }
|
header.border = { width = 1, top = true, color = "$border.0" }
|
||||||
|
|
||||||
|
[editor.warning_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$status.warn" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$status.warn", weight = "bold" }
|
||||||
|
|
||||||
[editor.information_diagnostic]
|
[editor.information_diagnostic]
|
||||||
text = "$status.info"
|
|
||||||
border = { width = 1, top = true, color = "$border.0" }
|
border = { width = 1, top = true, color = "$border.0" }
|
||||||
|
|
||||||
|
[editor.information_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$status.info" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$status.info", weight = "bold" }
|
||||||
|
|
||||||
[editor.hint_diagnostic]
|
[editor.hint_diagnostic]
|
||||||
text = "$status.info"
|
|
||||||
border = { width = 1, top = true, color = "$border.0" }
|
border = { width = 1, top = true, color = "$border.0" }
|
||||||
|
|
||||||
|
[editor.hint_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$status.info" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$status.info", weight = "bold" }
|
||||||
|
|
||||||
|
[editor.invalid_error_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$text.3.color" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$text.3.color", weight = "bold" }
|
||||||
|
|
||||||
|
[editor.invalid_warning_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$text.3.color" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$text.3.color", weight = "bold" }
|
||||||
|
|
||||||
|
[editor.invalid_information_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$text.3.color" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$text.3.color", weight = "bold" }
|
||||||
|
|
||||||
|
[editor.invalid_hint_diagnostic.message]
|
||||||
|
text = { extends = "$editor.text", size = 14, color = "$text.3.color" }
|
||||||
|
highlight_text = { extends = "$editor.text", size = 14, color = "$text.3.color", weight = "bold" }
|
||||||
|
|
||||||
[project_diagnostics]
|
[project_diagnostics]
|
||||||
background = "$surface.1"
|
background = "$surface.1"
|
||||||
empty_message = "$text.0"
|
empty_message = "$text.0"
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue