From a890787923636194eff0dc4d6a9d6fdd84db66f8 Mon Sep 17 00:00:00 2001 From: Max Brunsfeld Date: Tue, 25 Jan 2022 17:19:38 -0800 Subject: [PATCH] Render header blocks as horizontall fixed (like the gutter) Tweak editor layout to perform horizontal autoscroll before laying out the blocks, so that they can access the scroll position. --- crates/diagnostics/src/diagnostics.rs | 7 +- crates/editor/src/display_map/block_map.rs | 2 + crates/editor/src/element.rs | 122 ++++++++++----------- 3 files changed, 62 insertions(+), 69 deletions(-) diff --git a/crates/diagnostics/src/diagnostics.rs b/crates/diagnostics/src/diagnostics.rs index ea5441b01e..299eed77c1 100644 --- a/crates/diagnostics/src/diagnostics.rs +++ b/crates/diagnostics/src/diagnostics.rs @@ -692,7 +692,7 @@ fn path_header_renderer(buffer: ModelHandle, build_settings: BuildSettin .left() .contained() .with_style(style.container) - .with_padding_left(cx.gutter_padding) + .with_padding_left(cx.gutter_padding + cx.scroll_x * cx.em_width) .expanded() .named("path header block") }) @@ -745,7 +745,7 @@ fn diagnostic_header_renderer( })) .contained() .with_style(style.container) - .with_padding_left(cx.gutter_padding) + .with_padding_left(cx.gutter_padding + cx.scroll_x * cx.em_width) .expanded() .named("diagnostic header") }) @@ -757,7 +757,7 @@ fn context_header_renderer(build_settings: BuildSettings) -> RenderBlock { let text_style = settings.style.text.clone(); Label::new("…".to_string(), text_style) .contained() - .with_padding_left(cx.gutter_padding) + .with_padding_left(cx.gutter_padding + cx.scroll_x * cx.em_width) .named("collapsed context") }) } @@ -1252,6 +1252,7 @@ mod tests { .render(&BlockContext { cx, anchor_x: 0., + scroll_x: 0., gutter_padding: 0., gutter_width: 0., line_height: 0., diff --git a/crates/editor/src/display_map/block_map.rs b/crates/editor/src/display_map/block_map.rs index 5f3c8dc83e..f00dd89651 100644 --- a/crates/editor/src/display_map/block_map.rs +++ b/crates/editor/src/display_map/block_map.rs @@ -69,6 +69,7 @@ where pub struct BlockContext<'a> { pub cx: &'a AppContext, pub anchor_x: f32, + pub scroll_x: f32, pub gutter_width: f32, pub gutter_padding: f32, pub em_width: f32, @@ -951,6 +952,7 @@ mod tests { cx, anchor_x: 0., gutter_padding: 0., + scroll_x: 0., gutter_width: 0., line_height: 0., em_width: 0., diff --git a/crates/editor/src/element.rs b/crates/editor/src/element.rs index fbe7183e4c..ffbefbf035 100644 --- a/crates/editor/src/element.rs +++ b/crates/editor/src/element.rs @@ -17,7 +17,7 @@ use gpui::{ json::{self, ToJson}, keymap::Keystroke, text_layout::{self, RunStyle, TextLayoutCache}, - AppContext, Axis, Border, Element, ElementBox, Event, EventContext, FontCache, LayoutContext, + AppContext, Axis, Border, Element, ElementBox, Event, EventContext, LayoutContext, MutableAppContext, PaintContext, Quad, Scene, SizeConstraint, ViewContext, WeakViewHandle, }; use json::json; @@ -140,7 +140,6 @@ impl EditorElement { )) } - let font_cache = cx.font_cache.clone(); let snapshot = self.snapshot(cx.app); let (position, overshoot) = paint.point_for_position(&snapshot, layout, position); @@ -148,7 +147,7 @@ impl EditorElement { position, overshoot, scroll_position: (snapshot.scroll_position() + scroll_delta) - .clamp(Vector2F::zero(), layout.scroll_max(&font_cache)), + .clamp(Vector2F::zero(), layout.scroll_max), })); true } else { @@ -189,7 +188,6 @@ impl EditorElement { } let snapshot = self.snapshot(cx.app); - let font_cache = &cx.font_cache; let max_glyph_width = layout.em_width; if !precise { delta *= vec2f(max_glyph_width, layout.line_height); @@ -198,7 +196,7 @@ impl EditorElement { let scroll_position = snapshot.scroll_position(); let x = (scroll_position.x() * max_glyph_width - delta.x()) / max_glyph_width; let y = (scroll_position.y() * layout.line_height - delta.y()) / layout.line_height; - let scroll_position = vec2f(x, y).clamp(Vector2F::zero(), layout.scroll_max(font_cache)); + let scroll_position = vec2f(x, y).clamp(Vector2F::zero(), layout.scroll_max); cx.dispatch_action(Scroll(scroll_position)); @@ -452,7 +450,7 @@ impl EditorElement { .width() } - fn layout_rows( + fn layout_line_numbers( &self, rows: Range, active_rows: &BTreeMap, @@ -611,6 +609,7 @@ impl EditorElement { anchor_x, gutter_padding, line_height, + scroll_x: snapshot.scroll_position.x(), gutter_width, em_width, }); @@ -764,7 +763,8 @@ impl Element for EditorElement { } }); - let line_number_layouts = self.layout_rows(start_row..end_row, &active_rows, &snapshot, cx); + let line_number_layouts = + self.layout_line_numbers(start_row..end_row, &active_rows, &snapshot, cx); let mut max_visible_line_width = 0.0; let line_layouts = self.layout_lines(start_row..end_row, &mut snapshot, cx); @@ -783,7 +783,33 @@ impl Element for EditorElement { ) .width(); let scroll_width = longest_line_width.max(max_visible_line_width) + overscroll.x(); - let max_glyph_width = style.text.em_width(&cx.font_cache); + let em_width = style.text.em_width(cx.font_cache); + let max_row = snapshot.max_point().row(); + let scroll_max = vec2f( + ((scroll_width - text_size.x()) / em_width).max(0.0), + max_row.saturating_sub(1) as f32, + ); + + self.update_view(cx.app, |view, cx| { + let clamped = view.clamp_scroll_left(scroll_max.x()); + let autoscrolled; + if autoscroll_horizontally { + autoscrolled = view.autoscroll_horizontally( + start_row, + text_size.x(), + scroll_width, + em_width, + &line_layouts, + cx, + ); + } else { + autoscrolled = false; + } + + if clamped || autoscrolled { + snapshot = view.snapshot(cx); + } + }); let blocks = self.layout_blocks( start_row..end_row, @@ -799,49 +825,27 @@ impl Element for EditorElement { cx, ); - let mut layout = LayoutState { + ( size, - scroll_width, - gutter_size, - gutter_padding, - text_size, - text_offset, - snapshot, - style, - active_rows, - highlighted_rows, - line_layouts, - line_number_layouts, - blocks, - line_height, - em_width, - em_advance, - selections, - }; - - let scroll_max = layout.scroll_max(cx.font_cache).x(); - self.update_view(cx.app, |view, cx| { - let clamped = view.clamp_scroll_left(scroll_max); - let autoscrolled; - if autoscroll_horizontally { - autoscrolled = view.autoscroll_horizontally( - start_row, - layout.text_size.x(), - scroll_width, - max_glyph_width, - &layout.line_layouts, - cx, - ); - } else { - autoscrolled = false; - } - - if clamped || autoscrolled { - layout.snapshot = view.snapshot(cx); - } - }); - - (size, Some(layout)) + Some(LayoutState { + size, + scroll_max, + gutter_size, + gutter_padding, + text_size, + text_offset, + snapshot, + active_rows, + highlighted_rows, + line_layouts, + line_number_layouts, + blocks, + line_height, + em_width, + em_advance, + selections, + }), + ) } fn paint( @@ -931,11 +935,10 @@ impl Element for EditorElement { pub struct LayoutState { size: Vector2F, - scroll_width: f32, + scroll_max: Vector2F, gutter_size: Vector2F, gutter_padding: f32, text_size: Vector2F, - style: EditorStyle, snapshot: EditorSnapshot, active_rows: BTreeMap, highlighted_rows: Option>, @@ -949,19 +952,6 @@ pub struct LayoutState { text_offset: Vector2F, } -impl LayoutState { - fn scroll_max(&self, font_cache: &FontCache) -> Vector2F { - let text_width = self.text_size.x(); - let em_width = self.style.text.em_width(font_cache); - let max_row = self.snapshot.max_point().row(); - - vec2f( - ((self.scroll_width - text_width) / em_width).max(0.0), - max_row.saturating_sub(1) as f32, - ) - } -} - fn layout_line( row: u32, snapshot: &EditorSnapshot, @@ -1191,7 +1181,7 @@ mod tests { let snapshot = editor.snapshot(cx); let mut presenter = cx.build_presenter(window_id, 30.); let mut layout_cx = presenter.build_layout_context(false, cx); - element.layout_rows(0..6, &Default::default(), &snapshot, &mut layout_cx) + element.layout_line_numbers(0..6, &Default::default(), &snapshot, &mut layout_cx) }); assert_eq!(layouts.len(), 6); }