mirror of
https://github.com/helix-editor/helix.git
synced 2024-11-22 09:26:19 +04:00
Simplify selection rendering by injecting highlight scopes
This commit is contained in:
parent
44566ea812
commit
057bd630d8
@ -9,7 +9,7 @@
|
|||||||
use helix_core::{
|
use helix_core::{
|
||||||
coords_at_pos,
|
coords_at_pos,
|
||||||
graphemes::ensure_grapheme_boundary,
|
graphemes::ensure_grapheme_boundary,
|
||||||
syntax::{self, HighlightEvent},
|
syntax::{self, Highlight, HighlightEvent},
|
||||||
LineEnding, Position, Range,
|
LineEnding, Position, Range,
|
||||||
};
|
};
|
||||||
use helix_lsp::LspProgressMap;
|
use helix_lsp::LspProgressMap;
|
||||||
@ -44,6 +44,124 @@ fn default() -> Self {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct Merge<I> {
|
||||||
|
iter: I,
|
||||||
|
spans: Box<dyn Iterator<Item = (usize, std::ops::Range<usize>)>>,
|
||||||
|
|
||||||
|
next_event: Option<HighlightEvent>,
|
||||||
|
next_span: Option<(usize, std::ops::Range<usize>)>,
|
||||||
|
|
||||||
|
queue: Vec<HighlightEvent>,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn merge<I: Iterator<Item = HighlightEvent>>(
|
||||||
|
iter: I,
|
||||||
|
spans: Vec<(usize, std::ops::Range<usize>)>,
|
||||||
|
) -> impl Iterator<Item = HighlightEvent> {
|
||||||
|
let spans = Box::new(spans.into_iter());
|
||||||
|
let mut merge = Merge {
|
||||||
|
iter,
|
||||||
|
spans,
|
||||||
|
next_event: None,
|
||||||
|
next_span: None,
|
||||||
|
queue: Vec::new(),
|
||||||
|
};
|
||||||
|
merge.next_event = merge.iter.next();
|
||||||
|
merge.next_span = merge.spans.next();
|
||||||
|
merge
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<I: Iterator<Item = HighlightEvent>> Iterator for Merge<I> {
|
||||||
|
type Item = HighlightEvent;
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
use HighlightEvent::*;
|
||||||
|
if let Some(event) = self.queue.pop() {
|
||||||
|
return Some(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
loop {
|
||||||
|
match (self.next_event, &self.next_span) {
|
||||||
|
// this happens when range is partially or fully offscreen
|
||||||
|
(Some(Source { start, end }), Some((span, range))) if start > range.start => {
|
||||||
|
if start > range.end {
|
||||||
|
self.next_span = self.spans.next();
|
||||||
|
} else {
|
||||||
|
self.next_span = Some((*span, start..range.end));
|
||||||
|
};
|
||||||
|
}
|
||||||
|
_ => break,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
match (self.next_event, &self.next_span) {
|
||||||
|
(Some(HighlightStart(i)), _) => {
|
||||||
|
self.next_event = self.iter.next();
|
||||||
|
return Some(HighlightStart(i));
|
||||||
|
}
|
||||||
|
(Some(HighlightEnd), _) => {
|
||||||
|
self.next_event = self.iter.next();
|
||||||
|
return Some(HighlightEnd);
|
||||||
|
}
|
||||||
|
(Some(Source { start, end }), Some((span, range))) if start < range.start => {
|
||||||
|
let intersect = range.start.min(end);
|
||||||
|
let event = Source {
|
||||||
|
start,
|
||||||
|
end: intersect,
|
||||||
|
};
|
||||||
|
|
||||||
|
if end == intersect {
|
||||||
|
// the event is complete
|
||||||
|
self.next_event = self.iter.next();
|
||||||
|
} else {
|
||||||
|
// subslice the event
|
||||||
|
self.next_event = Some(Source {
|
||||||
|
start: intersect,
|
||||||
|
end,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
Some(event)
|
||||||
|
}
|
||||||
|
(Some(Source { start, end }), Some((span, range))) if start == range.start => {
|
||||||
|
let intersect = range.end.min(end);
|
||||||
|
let event = HighlightStart(Highlight(*span));
|
||||||
|
|
||||||
|
// enqueue in reverse order
|
||||||
|
self.queue.push(HighlightEnd);
|
||||||
|
self.queue.push(Source {
|
||||||
|
start,
|
||||||
|
end: intersect,
|
||||||
|
});
|
||||||
|
|
||||||
|
if end == intersect {
|
||||||
|
// the event is complete
|
||||||
|
self.next_event = self.iter.next();
|
||||||
|
} else {
|
||||||
|
// subslice the event
|
||||||
|
self.next_event = Some(Source {
|
||||||
|
start: intersect,
|
||||||
|
end,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
if intersect == range.end {
|
||||||
|
self.next_span = self.spans.next();
|
||||||
|
} else {
|
||||||
|
self.next_span = Some((*span, intersect..range.end));
|
||||||
|
}
|
||||||
|
|
||||||
|
Some(event)
|
||||||
|
}
|
||||||
|
(Some(event), None) => {
|
||||||
|
self.next_event = self.iter.next();
|
||||||
|
return Some(event);
|
||||||
|
}
|
||||||
|
(None, None) => return None,
|
||||||
|
e => unreachable!("{:?}", e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl EditorView {
|
impl EditorView {
|
||||||
pub fn new(keymaps: Keymaps) -> Self {
|
pub fn new(keymaps: Keymaps) -> Self {
|
||||||
Self {
|
Self {
|
||||||
@ -140,9 +258,91 @@ pub fn render_buffer(
|
|||||||
let mut line = 0u16;
|
let mut line = 0u16;
|
||||||
let tab_width = doc.tab_width();
|
let tab_width = doc.tab_width();
|
||||||
|
|
||||||
|
let highlights = highlights.into_iter().map(|event| match event.unwrap() {
|
||||||
|
// convert byte offsets to char offset
|
||||||
|
HighlightEvent::Source { start, end } => {
|
||||||
|
let start = ensure_grapheme_boundary(text, text.byte_to_char(start));
|
||||||
|
let end = ensure_grapheme_boundary(text, text.byte_to_char(end));
|
||||||
|
HighlightEvent::Source { start, end }
|
||||||
|
}
|
||||||
|
event => event,
|
||||||
|
});
|
||||||
|
|
||||||
|
let selections = doc.selection(view.id);
|
||||||
|
let primary_idx = selections.primary_index();
|
||||||
|
|
||||||
|
let selection_scope = theme
|
||||||
|
.find_scope_index("ui.selection")
|
||||||
|
.expect("no selection scope found!");
|
||||||
|
|
||||||
|
let base_cursor_scope = theme
|
||||||
|
.find_scope_index("ui.cursor")
|
||||||
|
.unwrap_or(selection_scope);
|
||||||
|
|
||||||
|
let cursor_scope = match doc.mode() {
|
||||||
|
Mode::Insert => theme.find_scope_index("ui.cursor.insert"),
|
||||||
|
Mode::Select => theme.find_scope_index("ui.cursor.select"),
|
||||||
|
Mode::Normal => Some(base_cursor_scope),
|
||||||
|
}
|
||||||
|
.unwrap_or(base_cursor_scope);
|
||||||
|
|
||||||
|
let primary_selection_scope = theme
|
||||||
|
.find_scope_index("ui.selection.primary")
|
||||||
|
.unwrap_or(selection_scope);
|
||||||
|
|
||||||
|
// TODO: primary + insert mode patching
|
||||||
|
// let primary_cursor_style = theme
|
||||||
|
// .try_get("ui.cursor.primary")
|
||||||
|
// .map(|style| {
|
||||||
|
// if mode != Mode::Normal {
|
||||||
|
// // we want to make sure that the insert and select highlights
|
||||||
|
// // also affect the primary cursor if set
|
||||||
|
// style.patch(cursor_style)
|
||||||
|
// } else {
|
||||||
|
// style
|
||||||
|
// }
|
||||||
|
// })
|
||||||
|
// .unwrap_or(cursor_style);
|
||||||
|
|
||||||
|
let primary_cursor_scope = theme
|
||||||
|
.find_scope_index("ui.cursor.primary")
|
||||||
|
.unwrap_or(cursor_scope);
|
||||||
|
|
||||||
|
let highlights: Box<dyn Iterator<Item = HighlightEvent>> = if is_focused {
|
||||||
|
// inject selections as highlight scopes
|
||||||
|
let mut spans_: Vec<(usize, std::ops::Range<usize>)> = Vec::new();
|
||||||
|
|
||||||
|
for (i, range) in selections.iter().enumerate() {
|
||||||
|
let (cursor_scope, selection_scope) = if i == primary_idx {
|
||||||
|
(primary_cursor_scope, primary_selection_scope)
|
||||||
|
} else {
|
||||||
|
(cursor_scope, selection_scope)
|
||||||
|
};
|
||||||
|
|
||||||
|
if range.head == range.anchor {
|
||||||
|
spans_.push((cursor_scope, range.head..range.head + 1));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let reverse = range.head < range.anchor;
|
||||||
|
|
||||||
|
if reverse {
|
||||||
|
spans_.push((cursor_scope, range.head..range.head + 1));
|
||||||
|
spans_.push((selection_scope, range.head + 1..range.anchor + 1));
|
||||||
|
} else {
|
||||||
|
spans_.push((selection_scope, range.anchor..range.head));
|
||||||
|
spans_.push((cursor_scope, range.head..range.head + 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Box::new(merge(highlights, spans_))
|
||||||
|
} else {
|
||||||
|
Box::new(highlights)
|
||||||
|
};
|
||||||
|
|
||||||
'outer: for event in highlights {
|
'outer: for event in highlights {
|
||||||
match event.unwrap() {
|
match event {
|
||||||
HighlightEvent::HighlightStart(mut span) => {
|
HighlightEvent::HighlightStart(span) => {
|
||||||
spans.push(span);
|
spans.push(span);
|
||||||
}
|
}
|
||||||
HighlightEvent::HighlightEnd => {
|
HighlightEvent::HighlightEnd => {
|
||||||
@ -150,29 +350,14 @@ pub fn render_buffer(
|
|||||||
}
|
}
|
||||||
HighlightEvent::Source { start, end } => {
|
HighlightEvent::Source { start, end } => {
|
||||||
// TODO: filter out spans out of viewport for now..
|
// TODO: filter out spans out of viewport for now..
|
||||||
|
|
||||||
// TODO: do these before iterating
|
|
||||||
let start = ensure_grapheme_boundary(text, text.byte_to_char(start));
|
|
||||||
let end = ensure_grapheme_boundary(text, text.byte_to_char(end));
|
|
||||||
|
|
||||||
let text = text.slice(start..end);
|
let text = text.slice(start..end);
|
||||||
|
|
||||||
use helix_core::graphemes::{grapheme_width, RopeGraphemes};
|
use helix_core::graphemes::{grapheme_width, RopeGraphemes};
|
||||||
|
|
||||||
// TODO: scope matching: biggest union match? [string] & [html, string], [string, html] & [ string, html]
|
let style = spans.iter().fold(theme.get("ui.text"), |acc, span| {
|
||||||
// can do this by sorting our theme matches based on array len (longest first) then stopping at the
|
let style = theme.get(theme.scopes()[span.0].as_str());
|
||||||
// first rule that matches (rule.all(|scope| scopes.contains(scope)))
|
acc.patch(style)
|
||||||
// log::info!(
|
});
|
||||||
// "scopes: {:?}",
|
|
||||||
// spans
|
|
||||||
// .iter()
|
|
||||||
// .map(|span| theme.scopes()[span.0].as_str())
|
|
||||||
// .collect::<Vec<_>>()
|
|
||||||
// );
|
|
||||||
let style = match spans.first() {
|
|
||||||
Some(span) => theme.get(theme.scopes()[span.0].as_str()),
|
|
||||||
None => theme.get("ui.text"),
|
|
||||||
};
|
|
||||||
|
|
||||||
// TODO: we could render the text to a surface, then cache that, that
|
// TODO: we could render the text to a surface, then cache that, that
|
||||||
// way if only the selection/cursor changes we can copy from cache
|
// way if only the selection/cursor changes we can copy from cache
|
||||||
@ -183,7 +368,19 @@ pub fn render_buffer(
|
|||||||
|
|
||||||
// iterate over range char by char
|
// iterate over range char by char
|
||||||
for grapheme in RopeGraphemes::new(text) {
|
for grapheme in RopeGraphemes::new(text) {
|
||||||
|
let out_of_bounds = visual_x < view.first_col as u16
|
||||||
|
|| visual_x >= viewport.width + view.first_col as u16;
|
||||||
if LineEnding::from_rope_slice(&grapheme).is_some() {
|
if LineEnding::from_rope_slice(&grapheme).is_some() {
|
||||||
|
if !out_of_bounds {
|
||||||
|
// we still want to render an empty cell with the style
|
||||||
|
surface.set_string(
|
||||||
|
viewport.x + visual_x - view.first_col as u16,
|
||||||
|
viewport.y + line,
|
||||||
|
" ",
|
||||||
|
style,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
visual_x = 0;
|
visual_x = 0;
|
||||||
line += 1;
|
line += 1;
|
||||||
|
|
||||||
@ -192,11 +389,18 @@ pub fn render_buffer(
|
|||||||
break 'outer;
|
break 'outer;
|
||||||
}
|
}
|
||||||
} else if grapheme == "\t" {
|
} else if grapheme == "\t" {
|
||||||
|
if !out_of_bounds {
|
||||||
|
// we still want to render an empty cell with the style
|
||||||
|
surface.set_string(
|
||||||
|
viewport.x + visual_x - view.first_col as u16,
|
||||||
|
viewport.y + line,
|
||||||
|
" ".repeat(tab_width),
|
||||||
|
style,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
visual_x = visual_x.saturating_add(tab_width as u16);
|
visual_x = visual_x.saturating_add(tab_width as u16);
|
||||||
} else {
|
} else {
|
||||||
let out_of_bounds = visual_x < view.first_col as u16
|
|
||||||
|| visual_x >= viewport.width + view.first_col as u16;
|
|
||||||
|
|
||||||
// Cow will prevent allocations if span contained in a single slice
|
// Cow will prevent allocations if span contained in a single slice
|
||||||
// which should really be the majority case
|
// which should really be the majority case
|
||||||
let grapheme = Cow::from(grapheme);
|
let grapheme = Cow::from(grapheme);
|
||||||
@ -283,126 +487,11 @@ pub fn render_buffer(
|
|||||||
Range::new(start, end)
|
Range::new(start, end)
|
||||||
};
|
};
|
||||||
|
|
||||||
let mode = doc.mode();
|
|
||||||
let base_cursor_style = theme
|
|
||||||
.try_get("ui.cursor")
|
|
||||||
.unwrap_or_else(|| Style::default().add_modifier(Modifier::REVERSED));
|
|
||||||
let cursor_style = match mode {
|
|
||||||
Mode::Insert => theme.try_get("ui.cursor.insert"),
|
|
||||||
Mode::Select => theme.try_get("ui.cursor.select"),
|
|
||||||
Mode::Normal => Some(base_cursor_style),
|
|
||||||
}
|
|
||||||
.unwrap_or(base_cursor_style);
|
|
||||||
let primary_cursor_style = theme
|
|
||||||
.try_get("ui.cursor.primary")
|
|
||||||
.map(|style| {
|
|
||||||
if mode != Mode::Normal {
|
|
||||||
// we want to make sure that the insert and select highlights
|
|
||||||
// also affect the primary cursor if set
|
|
||||||
style.patch(cursor_style)
|
|
||||||
} else {
|
|
||||||
style
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.unwrap_or(cursor_style);
|
|
||||||
|
|
||||||
let selection_style = theme.get("ui.selection");
|
|
||||||
let primary_selection_style = theme
|
|
||||||
.try_get("ui.selection.primary")
|
|
||||||
.unwrap_or(selection_style);
|
|
||||||
|
|
||||||
let selection = doc.selection(view.id);
|
let selection = doc.selection(view.id);
|
||||||
let primary_idx = selection.primary_index();
|
|
||||||
|
|
||||||
for (i, selection) in selection
|
for selection in selection.iter().filter(|range| range.overlaps(&screen)) {
|
||||||
.iter()
|
let head = view.screen_coords_at_pos(doc, text, selection.head);
|
||||||
.enumerate()
|
|
||||||
.filter(|(_, range)| range.overlaps(&screen))
|
|
||||||
{
|
|
||||||
// TODO: render also if only one of the ranges is in viewport
|
|
||||||
let mut start = view.screen_coords_at_pos(doc, text, selection.anchor);
|
|
||||||
let mut end = view.screen_coords_at_pos(doc, text, selection.head);
|
|
||||||
|
|
||||||
let (cursor_style, selection_style) = if i == primary_idx {
|
|
||||||
(primary_cursor_style, primary_selection_style)
|
|
||||||
} else {
|
|
||||||
(cursor_style, selection_style)
|
|
||||||
};
|
|
||||||
|
|
||||||
let head = end;
|
|
||||||
|
|
||||||
if selection.head < selection.anchor {
|
|
||||||
std::mem::swap(&mut start, &mut end);
|
|
||||||
}
|
|
||||||
let start = start.unwrap_or_else(|| Position::new(0, 0));
|
|
||||||
let end = end.unwrap_or_else(|| {
|
|
||||||
Position::new(viewport.height as usize, viewport.width as usize)
|
|
||||||
});
|
|
||||||
|
|
||||||
if start.row == end.row {
|
|
||||||
surface.set_style(
|
|
||||||
Rect::new(
|
|
||||||
viewport.x + start.col as u16,
|
|
||||||
viewport.y + start.row as u16,
|
|
||||||
// .min is important, because set_style does a
|
|
||||||
// for i in area.left()..area.right() and
|
|
||||||
// area.right = x + width !!! which shouldn't be > then surface.area.right()
|
|
||||||
// This is checked by a debug_assert! in Buffer::index_of
|
|
||||||
((end.col - start.col) as u16 + 1).min(
|
|
||||||
surface
|
|
||||||
.area
|
|
||||||
.width
|
|
||||||
.saturating_sub(viewport.x + start.col as u16),
|
|
||||||
),
|
|
||||||
1,
|
|
||||||
),
|
|
||||||
selection_style,
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
surface.set_style(
|
|
||||||
Rect::new(
|
|
||||||
viewport.x + start.col as u16,
|
|
||||||
viewport.y + start.row as u16,
|
|
||||||
// text.line(view.first_line).len_chars() as u16 - start.col as u16,
|
|
||||||
viewport.width.saturating_sub(start.col as u16),
|
|
||||||
1,
|
|
||||||
),
|
|
||||||
selection_style,
|
|
||||||
);
|
|
||||||
for i in start.row + 1..end.row {
|
|
||||||
surface.set_style(
|
|
||||||
Rect::new(
|
|
||||||
viewport.x,
|
|
||||||
viewport.y + i as u16,
|
|
||||||
// text.line(view.first_line + i).len_chars() as u16,
|
|
||||||
viewport.width,
|
|
||||||
1,
|
|
||||||
),
|
|
||||||
selection_style,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
surface.set_style(
|
|
||||||
Rect::new(
|
|
||||||
viewport.x,
|
|
||||||
viewport.y + end.row as u16,
|
|
||||||
(end.col as u16).min(viewport.width),
|
|
||||||
1,
|
|
||||||
),
|
|
||||||
selection_style,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// cursor
|
|
||||||
if let Some(head) = head {
|
if let Some(head) = head {
|
||||||
surface.set_style(
|
|
||||||
Rect::new(
|
|
||||||
viewport.x + head.col as u16,
|
|
||||||
viewport.y + head.row as u16,
|
|
||||||
1,
|
|
||||||
1,
|
|
||||||
),
|
|
||||||
cursor_style,
|
|
||||||
);
|
|
||||||
surface.set_stringn(
|
surface.set_stringn(
|
||||||
viewport.x + 1 - OFFSET,
|
viewport.x + 1 - OFFSET,
|
||||||
viewport.y + head.row as u16,
|
viewport.y + head.row as u16,
|
||||||
@ -410,6 +499,7 @@ pub fn render_buffer(
|
|||||||
5,
|
5,
|
||||||
linenr_select,
|
linenr_select,
|
||||||
);
|
);
|
||||||
|
|
||||||
// TODO: set cursor position for IME
|
// TODO: set cursor position for IME
|
||||||
if let Some(syntax) = doc.syntax() {
|
if let Some(syntax) = doc.syntax() {
|
||||||
use helix_core::match_brackets;
|
use helix_core::match_brackets;
|
||||||
|
@ -215,6 +215,10 @@ pub fn try_get(&self, scope: &str) -> Option<Style> {
|
|||||||
pub fn scopes(&self) -> &[String] {
|
pub fn scopes(&self) -> &[String] {
|
||||||
&self.scopes
|
&self.scopes
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn find_scope_index(&self, scope: &str) -> Option<usize> {
|
||||||
|
self.scopes().iter().position(|s| s == scope)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -52,9 +52,12 @@
|
|||||||
|
|
||||||
"ui.selection" = { bg = "#540099" }
|
"ui.selection" = { bg = "#540099" }
|
||||||
"ui.selection.primary" = { bg = "#540099" }
|
"ui.selection.primary" = { bg = "#540099" }
|
||||||
|
# TODO: namespace ui.cursor as ui.selection.cursor?
|
||||||
"ui.cursor.select" = { bg = "#6F44F0" }
|
"ui.cursor.select" = { bg = "#6F44F0" }
|
||||||
"ui.cursor.insert" = { bg = "#802F00" }
|
"ui.cursor.insert" = { bg = "#ffffff" }
|
||||||
"ui.cursor.match" = { fg = "#212121", bg = "#6C6999" }
|
"ui.cursor.match" = { fg = "#212121", bg = "#6C6999" }
|
||||||
|
"ui.cursor" = { modifiers = ["reversed"] }
|
||||||
|
|
||||||
"ui.menu.selected" = { fg = "#281733", bg = "#ffffff" } # revolver
|
"ui.menu.selected" = { fg = "#281733", bg = "#ffffff" } # revolver
|
||||||
|
|
||||||
"warning" = "#ffcd1c"
|
"warning" = "#ffcd1c"
|
||||||
|
Loading…
Reference in New Issue
Block a user