This commit is contained in:
Nathan Sobo 2021-07-24 08:40:48 -06:00
parent 3bf47be51e
commit 9edc8b9a06
4 changed files with 80 additions and 98 deletions

View File

@ -632,8 +632,8 @@ impl Editor {
scroll_position: Vector2F,
cx: &mut ViewContext<Self>,
) {
let buffer = self.buffer.read(cx);
let display_map = self.display_map.snapshot(cx);
let buffer = self.buffer.read(cx);
let cursor = display_map.anchor_before(position, Bias::Left);
if let Some(selection) = self.pending_selection.as_mut() {
selection.set_head(buffer, cursor);
@ -832,13 +832,13 @@ impl Editor {
pub fn delete_line(&mut self, _: &(), cx: &mut ViewContext<Self>) {
self.start_transaction(cx);
let display_map = self.display_map.snapshot(cx);
let app = cx.as_ref();
let buffer = self.buffer.read(app);
let mut new_cursors = Vec::new();
let mut edit_ranges = Vec::new();
let display_map = self.display_map.snapshot(cx);
let mut selections = self.selections(app).iter().peekable();
while let Some(selection) = selections.next() {
let (mut rows, _) = selection.buffer_rows_for_display_rows(false, &display_map);
@ -918,8 +918,8 @@ impl Editor {
}
self.update_selections(selections.clone(), false, cx);
let buffer = self.buffer.read(cx);
let display_map = self.display_map.snapshot(cx);
let buffer = self.buffer.read(cx);
let mut edits = Vec::new();
let mut selections_iter = selections.iter_mut().peekable();
@ -967,9 +967,9 @@ impl Editor {
pub fn move_line_up(&mut self, _: &(), cx: &mut ViewContext<Self>) {
self.start_transaction(cx);
let display_map = self.display_map.snapshot(cx);
let app = cx.as_ref();
let buffer = self.buffer.read(cx);
let display_map = self.display_map.snapshot(cx);
let mut edits = Vec::new();
let mut new_selection_ranges = Vec::new();
@ -1052,9 +1052,9 @@ impl Editor {
pub fn move_line_down(&mut self, _: &(), cx: &mut ViewContext<Self>) {
self.start_transaction(cx);
let display_map = self.display_map.snapshot(cx);
let app = cx.as_ref();
let buffer = self.buffer.read(cx);
let display_map = self.display_map.snapshot(cx);
let mut edits = Vec::new();
let mut new_selection_ranges = Vec::new();
@ -2258,6 +2258,10 @@ impl Snapshot {
self.display_snapshot.max_point()
}
pub fn longest_row(&self) -> u32 {
self.display_snapshot.longest_row()
}
pub fn font_ascent(&self, font_cache: &FontCache) -> f32 {
let font_id = font_cache.default_font(self.font_family);
let ascent = font_cache.metric(font_id, |m| m.ascent);
@ -2285,7 +2289,6 @@ impl Snapshot {
&self,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
cx: &AppContext,
) -> Result<f32> {
let font_size = self.font_size;
let font_id = font_cache.select_font(self.font_family, &FontProperties::new())?;
@ -2308,7 +2311,6 @@ impl Snapshot {
viewport_height: f32,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
cx: &mut MutableAppContext,
) -> Result<Vec<text_layout::Line>> {
let font_id = font_cache.select_font(self.font_family, &FontProperties::new())?;
@ -2339,11 +2341,10 @@ impl Snapshot {
}
pub fn layout_lines(
&self,
&mut self,
mut rows: Range<u32>,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
cx: &mut MutableAppContext,
) -> Result<Vec<text_layout::Line>> {
rows.end = cmp::min(rows.end, self.display_snapshot.max_point().row() + 1);
if rows.start >= rows.end {
@ -2399,7 +2400,6 @@ impl Snapshot {
row: u32,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
cx: &mut MutableAppContext,
) -> Result<text_layout::Line> {
let font_id = font_cache.select_font(self.font_family, &FontProperties::new())?;
@ -2713,7 +2713,7 @@ mod tests {
let layouts = view
.read(cx)
.snapshot(cx)
.layout_line_numbers(1000.0, &font_cache, &layout_cache, cx)
.layout_line_numbers(1000.0, &font_cache, &layout_cache)
.unwrap();
assert_eq!(layouts.len(), 6);
}

View File

@ -584,7 +584,7 @@ impl Snapshot {
}
}
pub fn highlighted_chunks(&mut self, range: Range<OutputOffset>) -> HighlightedChunks {
pub fn highlighted_chunks(&self, range: Range<OutputOffset>) -> HighlightedChunks {
let mut transform_cursor = self.transforms.cursor::<OutputOffset, usize>();
transform_cursor.seek(&range.end, Bias::Right, &());

View File

@ -144,7 +144,7 @@ impl Snapshot {
}
}
pub fn highlighted_chunks(&mut self, range: Range<OutputPoint>) -> HighlightedChunks {
pub fn highlighted_chunks(&self, range: Range<OutputPoint>) -> HighlightedChunks {
let (input_start, expanded_char_column, to_next_stop) =
self.to_input_point(range.start, Bias::Left);
let input_start = self.input.to_output_offset(input_start);

View File

@ -1,6 +1,6 @@
use crate::time::ReplicaId;
use super::{DisplayPoint, Editor, SelectAction};
use super::{DisplayPoint, Editor, SelectAction, Snapshot};
use gpui::{
color::ColorU,
geometry::{
@ -83,7 +83,7 @@ impl EditorElement {
let rect = paint.text_bounds;
let mut scroll_delta = Vector2F::zero();
let vertical_margin = view.line_height(cx.font_cache).min(rect.height() / 3.0);
let vertical_margin = layout.line_height.min(rect.height() / 3.0);
let top = rect.origin_y() + vertical_margin;
let bottom = rect.lower_left().y() - vertical_margin;
if position.y() < top {
@ -93,7 +93,7 @@ impl EditorElement {
scroll_delta.set_y(scale_vertical_mouse_autoscroll_delta(position.y() - bottom))
}
let horizontal_margin = view.line_height(cx.font_cache).min(rect.width() / 3.0);
let horizontal_margin = layout.line_height.min(rect.width() / 3.0);
let left = rect.origin_x() + horizontal_margin;
let right = rect.upper_right().x() - horizontal_margin;
if position.x() < left {
@ -113,7 +113,7 @@ impl EditorElement {
position: paint.point_for_position(view, layout, position, &font_cache, cx),
scroll_position: (view.scroll_position() + scroll_delta).clamp(
Vector2F::zero(),
layout.scroll_max(view, &font_cache, &text_layout_cache, cx),
layout.scroll_max(&font_cache, &text_layout_cache),
),
});
@ -159,19 +159,16 @@ impl EditorElement {
let view = self.view(cx.app);
let font_cache = &cx.font_cache;
let layout_cache = &cx.text_layout_cache;
let max_glyph_width = view.em_width(font_cache);
let line_height = view.line_height(font_cache);
let max_glyph_width = layout.em_width;
if !precise {
delta *= vec2f(max_glyph_width, line_height);
delta *= vec2f(max_glyph_width, layout.line_height);
}
let x = (view.scroll_position().x() * max_glyph_width - delta.x()) / max_glyph_width;
let y = (view.scroll_position().y() * line_height - delta.y()) / line_height;
let y = (view.scroll_position().y() * layout.line_height - delta.y()) / layout.line_height;
let scroll_position = vec2f(x, y).clamp(
Vector2F::zero(),
self.update_view(cx.app, |view, cx| {
layout.scroll_max(view, font_cache, layout_cache, cx)
}),
layout.scroll_max(font_cache, layout_cache),
);
cx.dispatch_action("buffer:scroll", scroll_position);
@ -180,9 +177,7 @@ impl EditorElement {
}
fn paint_gutter(&mut self, rect: RectF, layout: &LayoutState, cx: &mut PaintContext) {
let view = self.view(cx.app);
let line_height = view.line_height(cx.font_cache);
let scroll_top = view.scroll_position().y() * line_height;
let scroll_top = layout.snapshot.scroll_position.y() * layout.line_height;
cx.scene.push_layer(Some(rect));
cx.scene.push_quad(Quad {
@ -196,11 +191,11 @@ impl EditorElement {
let line_origin = rect.origin()
+ vec2f(
rect.width() - line.width() - layout.gutter_padding,
ix as f32 * line_height - (scroll_top % line_height),
ix as f32 * layout.line_height - (scroll_top % layout.line_height),
);
line.paint(
line_origin,
RectF::new(vec2f(0., 0.), vec2f(line.width(), line_height)),
RectF::new(vec2f(0., 0.), vec2f(line.width(), layout.line_height)),
cx,
);
}
@ -210,12 +205,12 @@ impl EditorElement {
fn paint_text(&mut self, bounds: RectF, layout: &LayoutState, cx: &mut PaintContext) {
let view = self.view(cx.app);
let line_height = view.line_height(cx.font_cache);
let start_row = view.scroll_position().y() as u32;
let scroll_top = view.scroll_position().y() * line_height;
let end_row = ((scroll_top + bounds.height()) / line_height).ceil() as u32 + 1; // Add 1 to ensure selections bleed off screen
let max_glyph_width = view.em_width(cx.font_cache);
let scroll_left = view.scroll_position().x() * max_glyph_width;
let scroll_position = layout.snapshot.scroll_position;
let start_row = scroll_position.y() as u32;
let scroll_top = scroll_position.y() * layout.line_height;
let end_row = ((scroll_top + bounds.height()) / layout.line_height).ceil() as u32 + 1; // Add 1 to ensure selections bleed off screen
let max_glyph_width = layout.em_width;
let scroll_left = scroll_position.x() * max_glyph_width;
cx.scene.push_layer(Some(bounds));
cx.scene.push_quad(Quad {
@ -254,8 +249,8 @@ impl EditorElement {
let selection = Selection {
color: selection_color,
line_height,
start_y: content_origin.y() + row_range.start as f32 * line_height
line_height: layout.line_height,
start_y: content_origin.y() + row_range.start as f32 * layout.line_height
- scroll_top,
lines: row_range
.into_iter()
@ -294,11 +289,11 @@ impl EditorElement {
&layout.line_layouts[(selection.end.row() - start_row) as usize];
let x = cursor_row_layout.x_for_index(selection.end.column() as usize)
- scroll_left;
let y = selection.end.row() as f32 * line_height - scroll_top;
let y = selection.end.row() as f32 * layout.line_height - scroll_top;
cursors.push(Cursor {
color: cursor_color,
origin: content_origin + vec2f(x, y),
line_height,
line_height: layout.line_height,
});
}
}
@ -309,8 +304,11 @@ impl EditorElement {
for (ix, line) in layout.line_layouts.iter().enumerate() {
let row = start_row + ix as u32;
line.paint(
content_origin + vec2f(-scroll_left, row as f32 * line_height - scroll_top),
RectF::new(vec2f(scroll_left, 0.), vec2f(bounds.width(), line_height)),
content_origin + vec2f(-scroll_left, row as f32 * layout.line_height - scroll_top),
RectF::new(
vec2f(scroll_left, 0.),
vec2f(bounds.width(), layout.line_height),
),
cx,
);
}
@ -348,7 +346,7 @@ impl Element for EditorElement {
let gutter_width;
if snapshot.gutter_visible {
gutter_padding = snapshot.em_width(cx.font_cache);
match snapshot.max_line_number_width(cx.font_cache, cx.text_layout_cache, cx.app) {
match snapshot.max_line_number_width(cx.font_cache, cx.text_layout_cache) {
Err(error) => {
log::error!("error computing max line number width: {}", error);
return (size, None);
@ -385,12 +383,7 @@ impl Element for EditorElement {
});
let line_number_layouts = if snapshot.gutter_visible {
match snapshot.layout_line_numbers(
size.y(),
cx.font_cache,
cx.text_layout_cache,
cx.app,
) {
match snapshot.layout_line_numbers(size.y(), cx.font_cache, cx.text_layout_cache) {
Err(error) => {
log::error!("error laying out line numbers: {}", error);
return (size, None);
@ -406,22 +399,22 @@ impl Element for EditorElement {
let end_row = ((scroll_top + size.y()) / line_height).ceil() as u32 + 1; // Add 1 to ensure selections bleed off screen
let mut max_visible_line_width = 0.0;
let line_layouts =
match snapshot.layout_lines(start_row..end_row, font_cache, layout_cache, cx.app) {
Err(error) => {
log::error!("error laying out lines: {}", error);
return (size, None);
}
Ok(layouts) => {
for line in &layouts {
if line.width() > max_visible_line_width {
max_visible_line_width = line.width();
}
let line_layouts = match snapshot.layout_lines(start_row..end_row, font_cache, layout_cache)
{
Err(error) => {
log::error!("error laying out lines: {}", error);
return (size, None);
}
Ok(layouts) => {
for line in &layouts {
if line.width() > max_visible_line_width {
max_visible_line_width = line.width();
}
layouts
}
};
layouts
}
};
let view = self.view(cx.app);
let mut selections = HashMap::new();
@ -444,24 +437,23 @@ impl Element for EditorElement {
text_size,
overscroll,
text_offset,
snapshot,
line_layouts,
line_number_layouts,
line_height,
em_width,
selections,
max_visible_line_width,
};
// TODO: If any of the below changes the editor's scroll state, update the layout's snapshot to reflect it.
let view = self.view(cx.app);
view.clamp_scroll_left(
layout
.scroll_max(view, cx.font_cache, cx.text_layout_cache, cx.app)
.x(),
);
view.clamp_scroll_left(layout.scroll_max(cx.font_cache, cx.text_layout_cache).x());
if autoscroll_horizontally {
view.autoscroll_horizontally(
view.scroll_position().y() as u32,
layout.text_size.x(),
layout.scroll_width(view, cx.font_cache, cx.text_layout_cache, cx.app),
layout.scroll_width(cx.font_cache, cx.text_layout_cache),
snapshot.em_width(cx.font_cache),
&layout.line_layouts,
cx.app,
@ -471,12 +463,7 @@ impl Element for EditorElement {
(size, Some(layout))
}
fn after_layout(
&mut self,
size: Vector2F,
layout: &mut Self::LayoutState,
cx: &mut AfterLayoutContext,
) {
fn after_layout(&mut self, _: Vector2F, _: &mut Self::LayoutState, _: &mut AfterLayoutContext) {
}
fn paint(
@ -555,8 +542,11 @@ pub struct LayoutState {
gutter_size: Vector2F,
gutter_padding: f32,
text_size: Vector2F,
snapshot: Snapshot,
line_layouts: Vec<text_layout::Line>,
line_number_layouts: Vec<text_layout::Line>,
line_height: f32,
em_width: f32,
selections: HashMap<ReplicaId, Vec<Range<DisplayPoint>>>,
overscroll: Vector2F,
text_offset: Vector2F,
@ -564,33 +554,25 @@ pub struct LayoutState {
}
impl LayoutState {
fn scroll_width(
&self,
view: &Editor,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
cx: &mut MutableAppContext,
) -> f32 {
let row = view.longest_row(cx);
let longest_line_width = view
.layout_line(row, font_cache, layout_cache, cx)
fn scroll_width(&self, font_cache: &FontCache, layout_cache: &TextLayoutCache) -> f32 {
let row = self.snapshot.longest_row();
let longest_line_width = self
.snapshot
.layout_line(row, font_cache, layout_cache)
.unwrap()
.width();
longest_line_width.max(self.max_visible_line_width) + self.overscroll.x()
}
fn scroll_max(
&self,
view: &Editor,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
cx: &mut MutableAppContext,
) -> Vector2F {
fn scroll_max(&self, font_cache: &FontCache, layout_cache: &TextLayoutCache) -> Vector2F {
let text_width = self.text_size.x();
let scroll_width = self.scroll_width(font_cache, layout_cache);
let em_width = self.snapshot.em_width(font_cache);
let max_row = self.snapshot.max_point().row();
vec2f(
((self.scroll_width(view, font_cache, layout_cache, cx) - self.text_size.x())
/ view.em_width(font_cache))
.max(0.0),
view.max_point(cx).row().saturating_sub(1) as f32,
((scroll_width - text_width) / em_width).max(0.0),
max_row.saturating_sub(1) as f32,
)
}
}
@ -612,10 +594,10 @@ impl PaintState {
let scroll_position = view.scroll_position();
let position = position - self.text_bounds.origin();
let y = position.y().max(0.0).min(layout.size.y());
let row = ((y / line_h) + scroll_position.y()) as u32;
let row = ((y / layout.line_height) + scroll_position.y()) as u32;
let row = cmp::min(row, view.max_point(cx).row());
let line = &layout.line_layouts[(row - scroll_position.y() as u32) as usize];
let x = position.x() + (scroll_position.x() * view.em_width(font_cache));
let x = position.x() + (scroll_position.x() * layout.em_width);
let column = if x >= 0.0 {
line.index_for_x(x)