Detailed changes
@@ -679,9 +679,8 @@ impl Buffer {
if let Some(ctx) = ctx {
ctx.notify();
- let changes = self.edits_since(since).collect::<Vec<_>>();
- if !changes.is_empty() {
- self.did_edit(changes, was_dirty, ctx);
+ if self.edits_since(since).next().is_some() {
+ self.did_edit(was_dirty, ctx);
}
}
}
@@ -742,8 +741,8 @@ impl Buffer {
Ok(ops)
}
- fn did_edit(&self, changes: Vec<Edit>, was_dirty: bool, ctx: &mut ModelContext<Self>) {
- ctx.emit(Event::Edited(changes));
+ fn did_edit(&self, was_dirty: bool, ctx: &mut ModelContext<Self>) {
+ ctx.emit(Event::Edited);
if !was_dirty {
ctx.emit(Event::Dirtied);
}
@@ -892,9 +891,8 @@ impl Buffer {
if let Some(ctx) = ctx {
ctx.notify();
- let changes = self.edits_since(old_version).collect::<Vec<_>>();
- if !changes.is_empty() {
- self.did_edit(changes, was_dirty, ctx);
+ if self.edits_since(old_version).next().is_some() {
+ self.did_edit(was_dirty, ctx);
}
}
@@ -1086,9 +1084,8 @@ impl Buffer {
if let Some(ctx) = ctx {
ctx.notify();
- let changes = self.edits_since(old_version).collect::<Vec<_>>();
- if !changes.is_empty() {
- self.did_edit(changes, was_dirty, ctx);
+ if self.edits_since(old_version).next().is_some() {
+ self.did_edit(was_dirty, ctx);
}
}
@@ -1113,9 +1110,8 @@ impl Buffer {
if let Some(ctx) = ctx {
ctx.notify();
- let changes = self.edits_since(old_version).collect::<Vec<_>>();
- if !changes.is_empty() {
- self.did_edit(changes, was_dirty, ctx);
+ if self.edits_since(old_version).next().is_some() {
+ self.did_edit(was_dirty, ctx);
}
}
@@ -1794,7 +1790,7 @@ impl Snapshot {
#[derive(Clone, Debug, Eq, PartialEq)]
pub enum Event {
- Edited(Vec<Edit>),
+ Edited,
Dirtied,
Saved,
FileHandleChanged,
@@ -2403,34 +2399,11 @@ mod tests {
let buffer_1_events = buffer_1_events.borrow();
assert_eq!(
*buffer_1_events,
- vec![
- Event::Edited(vec![Edit {
- old_range: 2..4,
- new_range: 2..5
- }]),
- Event::Dirtied,
- Event::Edited(vec![Edit {
- old_range: 5..5,
- new_range: 5..7
- }]),
- Event::Edited(vec![Edit {
- old_range: 5..7,
- new_range: 5..5
- }]),
- ]
+ vec![Event::Edited, Event::Dirtied, Event::Edited, Event::Edited]
);
let buffer_2_events = buffer_2_events.borrow();
- assert_eq!(
- *buffer_2_events,
- vec![
- Event::Edited(vec![Edit {
- old_range: 2..4,
- new_range: 2..5
- },]),
- Event::Dirtied
- ]
- );
+ assert_eq!(*buffer_2_events, vec![Event::Edited, Event::Dirtied]);
});
}
@@ -2970,16 +2943,7 @@ mod tests {
model.update(app, |buffer, ctx| {
assert!(buffer.text() == "ac");
assert!(buffer.is_dirty());
- assert_eq!(
- *events.borrow(),
- &[
- Event::Edited(vec![Edit {
- old_range: 1..2,
- new_range: 1..1
- }]),
- Event::Dirtied
- ]
- );
+ assert_eq!(*events.borrow(), &[Event::Edited, Event::Dirtied]);
events.borrow_mut().clear();
buffer.did_save(buffer.version(), ctx);
@@ -3001,17 +2965,7 @@ mod tests {
assert!(buffer.is_dirty());
assert_eq!(
*events.borrow(),
- &[
- Event::Edited(vec![Edit {
- old_range: 1..1,
- new_range: 1..2
- }]),
- Event::Dirtied,
- Event::Edited(vec![Edit {
- old_range: 2..2,
- new_range: 2..3
- }]),
- ],
+ &[Event::Edited, Event::Dirtied, Event::Edited],
);
events.borrow_mut().clear();
@@ -3023,13 +2977,7 @@ mod tests {
});
model.update(app, |_, _| {
- assert_eq!(
- *events.borrow(),
- &[Event::Edited(vec![Edit {
- old_range: 1..3,
- new_range: 1..1
- },])]
- );
+ assert_eq!(*events.borrow(), &[Event::Edited]);
});
});
}
@@ -254,7 +254,7 @@ pub enum SelectAction {
pub struct BufferView {
handle: WeakViewHandle<Self>,
buffer: ModelHandle<Buffer>,
- display_map: ModelHandle<DisplayMap>,
+ display_map: DisplayMap,
selection_set_id: SelectionSetId,
pending_selection: Option<Selection>,
scroll_position: Mutex<Vector2F>,
@@ -290,14 +290,11 @@ impl BufferView {
ctx.observe(&buffer, Self::on_buffer_changed);
ctx.subscribe_to_model(&buffer, Self::on_buffer_event);
- let display_map = ctx.add_model(|ctx| {
- DisplayMap::new(
- buffer.clone(),
- smol::block_on(settings.read()).tab_size,
- ctx,
- )
- });
- ctx.observe(&display_map, Self::on_display_map_changed);
+ let display_map = DisplayMap::new(
+ buffer.clone(),
+ smol::block_on(settings.read()).tab_size,
+ ctx.as_ref(),
+ );
let (selection_set_id, _) = buffer.update(ctx, |buffer, ctx| {
buffer.add_selection_set(
@@ -367,14 +364,13 @@ impl BufferView {
return false;
}
- let map = self.display_map.read(app);
let visible_lines = viewport_height / line_height;
let first_cursor_top = self
.selections(app)
.first()
.unwrap()
.head()
- .to_display_point(map, app)
+ .to_display_point(&self.display_map, app)
.unwrap()
.row() as f32;
let last_cursor_bottom = self
@@ -382,7 +378,7 @@ impl BufferView {
.last()
.unwrap()
.head()
- .to_display_point(map, app)
+ .to_display_point(&self.display_map, app)
.unwrap()
.row() as f32
+ 1.0;
@@ -415,16 +411,20 @@ impl BufferView {
scroll_width: f32,
max_glyph_width: f32,
layouts: &[Arc<text_layout::Line>],
- app: &AppContext,
+ ctx: &AppContext,
) {
- let map = self.display_map.read(app);
-
let mut target_left = std::f32::INFINITY;
let mut target_right = 0.0_f32;
- for selection in self.selections(app) {
- let head = selection.head().to_display_point(map, app).unwrap();
+ for selection in self.selections(ctx) {
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, ctx)
+ .unwrap();
let start_column = head.column().saturating_sub(3);
- let end_column = cmp::min(map.line_len(head.row(), app).unwrap(), head.column() + 3);
+ let end_column = cmp::min(
+ self.display_map.line_len(head.row(), ctx).unwrap(),
+ head.column() + 3,
+ );
target_left = target_left
.min(layouts[(head.row() - start_row) as usize].x_for_index(start_column as usize));
target_right = target_right.max(
@@ -466,8 +466,8 @@ impl BufferView {
ctx.emit(Event::Activate);
}
- let display_map = self.display_map.read(ctx);
- let cursor = display_map
+ let cursor = self
+ .display_map
.anchor_before(position, Bias::Left, ctx.as_ref())
.unwrap();
let selection = Selection {
@@ -492,8 +492,8 @@ impl BufferView {
ctx: &mut ViewContext<Self>,
) {
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(ctx);
- let cursor = map
+ let cursor = self
+ .display_map
.anchor_before(position, Bias::Left, ctx.as_ref())
.unwrap();
if let Some(selection) = self.pending_selection.as_mut() {
@@ -554,7 +554,6 @@ impl BufferView {
where
T: IntoIterator<Item = &'a Range<DisplayPoint>>,
{
- let map = self.display_map.read(ctx);
let mut selections = Vec::new();
for range in ranges {
let mut start = range.start;
@@ -567,8 +566,12 @@ impl BufferView {
};
selections.push(Selection {
- start: map.anchor_before(start, Bias::Left, ctx.as_ref())?,
- end: map.anchor_before(end, Bias::Left, ctx.as_ref())?,
+ start: self
+ .display_map
+ .anchor_before(start, Bias::Left, ctx.as_ref())?,
+ end: self
+ .display_map
+ .anchor_before(end, Bias::Left, ctx.as_ref())?,
reversed,
goal_column: None,
});
@@ -634,17 +637,17 @@ impl BufferView {
let mut selections = self.selections(ctx.as_ref()).to_vec();
{
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(ctx);
for selection in &mut selections {
let range = selection.range(buffer);
if range.start == range.end {
let head = selection
.head()
- .to_display_point(map, ctx.as_ref())
+ .to_display_point(&self.display_map, ctx.as_ref())
.unwrap();
- let cursor = map
+ let cursor = self
+ .display_map
.anchor_before(
- movement::left(map, head, ctx.as_ref()).unwrap(),
+ movement::left(&self.display_map, head, ctx.as_ref()).unwrap(),
Bias::Left,
ctx.as_ref(),
)
@@ -665,17 +668,17 @@ impl BufferView {
let mut selections = self.selections(ctx.as_ref()).to_vec();
{
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(ctx);
for selection in &mut selections {
let range = selection.range(buffer);
if range.start == range.end {
let head = selection
.head()
- .to_display_point(map, ctx.as_ref())
+ .to_display_point(&self.display_map, ctx.as_ref())
.unwrap();
- let cursor = map
+ let cursor = self
+ .display_map
.anchor_before(
- movement::right(map, head, ctx.as_ref()).unwrap(),
+ movement::right(&self.display_map, head, ctx.as_ref()).unwrap(),
Bias::Right,
ctx.as_ref(),
)
@@ -695,7 +698,6 @@ impl BufferView {
self.start_transaction(ctx);
let app = ctx.as_ref();
- let map = self.display_map.read(app);
let buffer = self.buffer.read(app);
let mut new_cursors = Vec::new();
@@ -703,16 +705,17 @@ impl BufferView {
let mut selections = self.selections(app).iter().peekable();
while let Some(selection) = selections.next() {
- let (mut rows, _) = selection.buffer_rows_for_display_rows(map, app);
+ let (mut rows, _) = selection.buffer_rows_for_display_rows(&self.display_map, app);
let goal_display_column = selection
.head()
- .to_display_point(map, app)
+ .to_display_point(&self.display_map, app)
.unwrap()
.column();
// Accumulate contiguous regions of rows that we want to delete.
while let Some(next_selection) = selections.peek() {
- let (next_rows, _) = next_selection.buffer_rows_for_display_rows(map, app);
+ let (next_rows, _) =
+ next_selection.buffer_rows_for_display_rows(&self.display_map, app);
if next_rows.start <= rows.end {
rows.end = next_rows.end;
selections.next().unwrap();
@@ -738,14 +741,18 @@ impl BufferView {
}
let mut cursor = Point::new(cursor_buffer_row, 0)
- .to_display_point(map, app)
+ .to_display_point(&self.display_map, app)
.unwrap();
*cursor.column_mut() = cmp::min(
goal_display_column,
- map.line_len(cursor.row(), app).unwrap(),
+ self.display_map.line_len(cursor.row(), app).unwrap(),
);
- new_cursors.push(cursor.to_buffer_point(map, Bias::Left, app).unwrap());
+ new_cursors.push(
+ cursor
+ .to_buffer_point(&self.display_map, Bias::Left, app)
+ .unwrap(),
+ );
edit_ranges.push(edit_start..edit_end);
}
@@ -784,15 +791,15 @@ impl BufferView {
let app = ctx.as_ref();
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(ctx);
let mut edits = Vec::new();
let mut selections_iter = selections.iter_mut().peekable();
while let Some(selection) = selections_iter.next() {
// Avoid duplicating the same lines twice.
- let (mut rows, _) = selection.buffer_rows_for_display_rows(map, app);
+ let (mut rows, _) = selection.buffer_rows_for_display_rows(&self.display_map, app);
while let Some(next_selection) = selections_iter.peek() {
- let (next_rows, _) = next_selection.buffer_rows_for_display_rows(map, app);
+ let (next_rows, _) =
+ next_selection.buffer_rows_for_display_rows(&self.display_map, app);
if next_rows.start <= rows.end - 1 {
rows.end = next_rows.end;
selections_iter.next().unwrap();
@@ -834,7 +841,6 @@ impl BufferView {
let app = ctx.as_ref();
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(ctx);
let mut edits = Vec::new();
let mut new_selection_ranges = Vec::new();
@@ -847,10 +853,10 @@ impl BufferView {
// Accumulate contiguous regions of rows that we want to move.
contiguous_selections.push(selection.range(buffer));
let (mut buffer_rows, mut display_rows) =
- selection.buffer_rows_for_display_rows(map, app);
+ selection.buffer_rows_for_display_rows(&self.display_map, app);
while let Some(next_selection) = selections.peek() {
let (next_buffer_rows, next_display_rows) =
- next_selection.buffer_rows_for_display_rows(map, app);
+ next_selection.buffer_rows_for_display_rows(&self.display_map, app);
if next_buffer_rows.start <= buffer_rows.end {
buffer_rows.end = next_buffer_rows.end;
display_rows.end = next_display_rows.end;
@@ -873,7 +879,7 @@ impl BufferView {
let prev_row_display_start = DisplayPoint::new(display_rows.start - 1, 0);
let prev_row_start = prev_row_display_start
- .to_buffer_offset(map, Bias::Left, app)
+ .to_buffer_offset(&self.display_map, Bias::Left, app)
.unwrap();
let mut text = String::new();
@@ -884,7 +890,7 @@ impl BufferView {
let row_delta = buffer_rows.start
- prev_row_display_start
- .to_buffer_point(map, Bias::Left, app)
+ .to_buffer_point(&self.display_map, Bias::Left, app)
.unwrap()
.row;
@@ -896,7 +902,7 @@ impl BufferView {
// Move folds up.
old_folds.push(start..end);
- for fold in map.folds_in_range(start..end, app).unwrap() {
+ for fold in self.display_map.folds_in_range(start..end, app).unwrap() {
let mut start = fold.start.to_point(buffer).unwrap();
let mut end = fold.end.to_point(buffer).unwrap();
start.row -= row_delta;
@@ -925,7 +931,6 @@ impl BufferView {
let app = ctx.as_ref();
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(ctx);
let mut edits = Vec::new();
let mut new_selection_ranges = Vec::new();
@@ -938,10 +943,10 @@ impl BufferView {
// Accumulate contiguous regions of rows that we want to move.
contiguous_selections.push(selection.range(buffer));
let (mut buffer_rows, mut display_rows) =
- selection.buffer_rows_for_display_rows(map, app);
+ selection.buffer_rows_for_display_rows(&self.display_map, app);
while let Some(next_selection) = selections.peek() {
let (next_buffer_rows, next_display_rows) =
- next_selection.buffer_rows_for_display_rows(map, app);
+ next_selection.buffer_rows_for_display_rows(&self.display_map, app);
if next_buffer_rows.start <= buffer_rows.end {
buffer_rows.end = next_buffer_rows.end;
display_rows.end = next_display_rows.end;
@@ -953,7 +958,7 @@ impl BufferView {
}
// Cut the text from the selected rows and paste it at the end of the next line.
- if display_rows.end <= map.max_point(app).row() {
+ if display_rows.end <= self.display_map.max_point(app).row() {
let start = Point::new(buffer_rows.start, 0).to_offset(buffer).unwrap();
let end = Point::new(
buffer_rows.end - 1,
@@ -964,10 +969,10 @@ impl BufferView {
let next_row_display_end = DisplayPoint::new(
display_rows.end,
- map.line_len(display_rows.end, app).unwrap(),
+ self.display_map.line_len(display_rows.end, app).unwrap(),
);
let next_row_end = next_row_display_end
- .to_buffer_offset(map, Bias::Right, app)
+ .to_buffer_offset(&self.display_map, Bias::Right, app)
.unwrap();
let mut text = String::new();
@@ -977,7 +982,7 @@ impl BufferView {
edits.push((next_row_end..next_row_end, text));
let row_delta = next_row_display_end
- .to_buffer_point(map, Bias::Right, app)
+ .to_buffer_point(&self.display_map, Bias::Right, app)
.unwrap()
.row
- buffer_rows.end
@@ -991,7 +996,7 @@ impl BufferView {
// Move folds down.
old_folds.push(start..end);
- for fold in map.folds_in_range(start..end, app).unwrap() {
+ for fold in self.display_map.folds_in_range(start..end, app).unwrap() {
let mut start = fold.start.to_point(buffer).unwrap();
let mut end = fold.end.to_point(buffer).unwrap();
start.row += row_delta;
@@ -1155,16 +1160,26 @@ impl BufferView {
let app = ctx.as_ref();
let mut selections = self.selections(app).to_vec();
{
- let map = self.display_map.read(app);
for selection in &mut selections {
- let start = selection.start.to_display_point(map, app).unwrap();
- let end = selection.end.to_display_point(map, app).unwrap();
+ let start = selection
+ .start
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let end = selection
+ .end
+ .to_display_point(&self.display_map, app)
+ .unwrap();
if start != end {
selection.end = selection.start.clone();
} else {
- let cursor = map
- .anchor_before(movement::left(map, start, app).unwrap(), Bias::Left, app)
+ let cursor = self
+ .display_map
+ .anchor_before(
+ movement::left(&self.display_map, start, app).unwrap(),
+ Bias::Left,
+ app,
+ )
.unwrap();
selection.start = cursor.clone();
selection.end = cursor;
@@ -1180,15 +1195,15 @@ impl BufferView {
let mut selections = self.selections(ctx.as_ref()).to_vec();
{
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(ctx);
for selection in &mut selections {
let head = selection
.head()
- .to_display_point(map, ctx.as_ref())
+ .to_display_point(&self.display_map, ctx.as_ref())
.unwrap();
- let cursor = map
+ let cursor = self
+ .display_map
.anchor_before(
- movement::left(map, head, ctx.as_ref()).unwrap(),
+ movement::left(&self.display_map, head, ctx.as_ref()).unwrap(),
Bias::Left,
ctx.as_ref(),
)
@@ -1204,16 +1219,26 @@ impl BufferView {
let mut selections = self.selections(ctx.as_ref()).to_vec();
{
let app = ctx.as_ref();
- let map = self.display_map.read(app);
for selection in &mut selections {
- let start = selection.start.to_display_point(map, app).unwrap();
- let end = selection.end.to_display_point(map, app).unwrap();
+ let start = selection
+ .start
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let end = selection
+ .end
+ .to_display_point(&self.display_map, app)
+ .unwrap();
if start != end {
selection.start = selection.end.clone();
} else {
- let cursor = map
- .anchor_before(movement::right(map, end, app).unwrap(), Bias::Right, app)
+ let cursor = self
+ .display_map
+ .anchor_before(
+ movement::right(&self.display_map, end, app).unwrap(),
+ Bias::Right,
+ app,
+ )
.unwrap();
selection.start = cursor.clone();
selection.end = cursor;
@@ -1230,14 +1255,18 @@ impl BufferView {
{
let app = ctx.as_ref();
let buffer = self.buffer.read(app);
- let map = self.display_map.read(app);
for selection in &mut selections {
let head = selection
.head()
- .to_display_point(map, ctx.as_ref())
+ .to_display_point(&self.display_map, ctx.as_ref())
.unwrap();
- let cursor = map
- .anchor_before(movement::right(map, head, app).unwrap(), Bias::Right, app)
+ let cursor = self
+ .display_map
+ .anchor_before(
+ movement::right(&self.display_map, head, app).unwrap(),
+ Bias::Right,
+ app,
+ )
.unwrap();
selection.set_head(&buffer, cursor);
selection.goal_column = None;
@@ -1253,17 +1282,25 @@ impl BufferView {
let mut selections = self.selections(ctx.as_ref()).to_vec();
{
let app = ctx.as_ref();
- let map = self.display_map.read(app);
for selection in &mut selections {
- let start = selection.start.to_display_point(map, app).unwrap();
- let end = selection.end.to_display_point(map, app).unwrap();
+ let start = selection
+ .start
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let end = selection
+ .end
+ .to_display_point(&self.display_map, app)
+ .unwrap();
if start != end {
selection.goal_column = None;
}
let (start, goal_column) =
- movement::up(map, start, selection.goal_column, app).unwrap();
- let cursor = map.anchor_before(start, Bias::Left, app).unwrap();
+ movement::up(&self.display_map, start, selection.goal_column, app).unwrap();
+ let cursor = self
+ .display_map
+ .anchor_before(start, Bias::Left, app)
+ .unwrap();
selection.start = cursor.clone();
selection.end = cursor;
selection.goal_column = goal_column;
@@ -1279,12 +1316,19 @@ impl BufferView {
{
let app = ctx.as_ref();
let buffer = self.buffer.read(app);
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
let (head, goal_column) =
- movement::up(map, head, selection.goal_column, app).unwrap();
- selection.set_head(&buffer, map.anchor_before(head, Bias::Left, app).unwrap());
+ movement::up(&self.display_map, head, selection.goal_column, app).unwrap();
+ selection.set_head(
+ &buffer,
+ self.display_map
+ .anchor_before(head, Bias::Left, app)
+ .unwrap(),
+ );
selection.goal_column = goal_column;
}
}
@@ -1298,17 +1342,25 @@ impl BufferView {
let mut selections = self.selections(ctx.as_ref()).to_vec();
{
let app = ctx.as_ref();
- let map = self.display_map.read(app);
for selection in &mut selections {
- let start = selection.start.to_display_point(map, app).unwrap();
- let end = selection.end.to_display_point(map, app).unwrap();
+ let start = selection
+ .start
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let end = selection
+ .end
+ .to_display_point(&self.display_map, app)
+ .unwrap();
if start != end {
selection.goal_column = None;
}
let (start, goal_column) =
- movement::down(map, end, selection.goal_column, app).unwrap();
- let cursor = map.anchor_before(start, Bias::Right, app).unwrap();
+ movement::down(&self.display_map, end, selection.goal_column, app).unwrap();
+ let cursor = self
+ .display_map
+ .anchor_before(start, Bias::Right, app)
+ .unwrap();
selection.start = cursor.clone();
selection.end = cursor;
selection.goal_column = goal_column;
@@ -1324,12 +1376,19 @@ impl BufferView {
{
let app = ctx.as_ref();
let buffer = self.buffer.read(app);
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
let (head, goal_column) =
- movement::down(map, head, selection.goal_column, app).unwrap();
- selection.set_head(&buffer, map.anchor_before(head, Bias::Right, app).unwrap());
+ movement::down(&self.display_map, head, selection.goal_column, app).unwrap();
+ selection.set_head(
+ &buffer,
+ self.display_map
+ .anchor_before(head, Bias::Right, app)
+ .unwrap(),
+ );
selection.goal_column = goal_column;
}
}
@@ -1340,11 +1399,16 @@ impl BufferView {
let app = ctx.as_ref();
let mut selections = self.selections(app).to_vec();
{
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::prev_word_boundary(map, head, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head = movement::prev_word_boundary(&self.display_map, head, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.start = anchor.clone();
selection.end = anchor;
selection.reversed = false;
@@ -1359,11 +1423,16 @@ impl BufferView {
let mut selections = self.selections(app).to_vec();
{
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::prev_word_boundary(map, head, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head = movement::prev_word_boundary(&self.display_map, head, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.set_head(buffer, anchor);
selection.goal_column = None;
}
@@ -1382,11 +1451,16 @@ impl BufferView {
let app = ctx.as_ref();
let mut selections = self.selections(app).to_vec();
{
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::next_word_boundary(map, head, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head = movement::next_word_boundary(&self.display_map, head, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.start = anchor.clone();
selection.end = anchor;
selection.reversed = false;
@@ -1401,11 +1475,16 @@ impl BufferView {
let mut selections = self.selections(app).to_vec();
{
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::next_word_boundary(map, head, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head = movement::next_word_boundary(&self.display_map, head, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.set_head(buffer, anchor);
selection.goal_column = None;
}
@@ -1424,11 +1503,17 @@ impl BufferView {
let app = ctx.as_ref();
let mut selections = self.selections(app).to_vec();
{
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::line_beginning(map, head, true, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head =
+ movement::line_beginning(&self.display_map, head, true, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.start = anchor.clone();
selection.end = anchor;
selection.reversed = false;
@@ -1447,11 +1532,17 @@ impl BufferView {
let mut selections = self.selections(app).to_vec();
{
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::line_beginning(map, head, *toggle_indent, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head =
+ movement::line_beginning(&self.display_map, head, *toggle_indent, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.set_head(buffer, anchor);
selection.goal_column = None;
}
@@ -1470,11 +1561,16 @@ impl BufferView {
let app = ctx.as_ref();
let mut selections = self.selections(app).to_vec();
{
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::line_end(map, head, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head = movement::line_end(&self.display_map, head, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.start = anchor.clone();
selection.end = anchor;
selection.reversed = false;
@@ -1489,11 +1585,16 @@ impl BufferView {
let mut selections = self.selections(app).to_vec();
{
let buffer = self.buffer.read(ctx);
- let map = self.display_map.read(app);
for selection in &mut selections {
- let head = selection.head().to_display_point(map, app).unwrap();
- let new_head = movement::line_end(map, head, app).unwrap();
- let anchor = map.anchor_before(new_head, Bias::Left, app).unwrap();
+ let head = selection
+ .head()
+ .to_display_point(&self.display_map, app)
+ .unwrap();
+ let new_head = movement::line_end(&self.display_map, head, app).unwrap();
+ let anchor = self
+ .display_map
+ .anchor_before(new_head, Bias::Left, app)
+ .unwrap();
selection.set_head(buffer, anchor);
selection.goal_column = None;
}
@@ -1559,12 +1660,13 @@ impl BufferView {
range: Range<DisplayPoint>,
app: &'a AppContext,
) -> impl 'a + Iterator<Item = Range<DisplayPoint>> {
- let map = self.display_map.read(app);
-
- let start = map.anchor_before(range.start, Bias::Left, app).unwrap();
+ let start = self
+ .display_map
+ .anchor_before(range.start, Bias::Left, app)
+ .unwrap();
let start_index = self.selection_insertion_index(&start, app);
let pending_selection = self.pending_selection.as_ref().and_then(|s| {
- let selection_range = s.display_range(map, app);
+ let selection_range = s.display_range(&self.display_map, app);
if selection_range.start <= range.end || selection_range.end <= range.end {
Some(selection_range)
} else {
@@ -1573,7 +1675,7 @@ impl BufferView {
});
self.selections(app)[start_index..]
.iter()
- .map(move |s| s.display_range(map, app))
+ .map(move |s| s.display_range(&self.display_map, app))
.take_while(move |r| r.start <= range.end || r.end <= range.end)
.chain(pending_selection)
}
@@ -1673,17 +1775,18 @@ impl BufferView {
let mut fold_ranges = Vec::new();
let app = ctx.as_ref();
- let map = self.display_map.read(app);
for selection in self.selections(app) {
- let range = selection.display_range(map, app).sorted();
+ let range = selection.display_range(&self.display_map, app).sorted();
let buffer_start_row = range
.start
- .to_buffer_point(map, Bias::Left, app)
+ .to_buffer_point(&self.display_map, Bias::Left, app)
.unwrap()
.row;
for row in (0..=range.end.row()).rev() {
- if self.is_line_foldable(row, app) && !map.is_line_folded(row) {
+ if self.is_line_foldable(row, app)
+ && !self.display_map.is_line_folded(row, ctx.as_ref())
+ {
let fold_range = self.foldable_range_for_line(row, app).unwrap();
if fold_range.end.row >= buffer_start_row {
fold_ranges.push(fold_range);
@@ -1702,15 +1805,20 @@ impl BufferView {
use super::RangeExt;
let app = ctx.as_ref();
- let map = self.display_map.read(app);
let buffer = self.buffer.read(app);
let ranges = self
.selections(app)
.iter()
.map(|s| {
- let range = s.display_range(map, app).sorted();
- let mut start = range.start.to_buffer_point(map, Bias::Left, app).unwrap();
- let mut end = range.end.to_buffer_point(map, Bias::Left, app).unwrap();
+ let range = s.display_range(&self.display_map, app).sorted();
+ let mut start = range
+ .start
+ .to_buffer_point(&self.display_map, Bias::Left, app)
+ .unwrap();
+ let mut end = range
+ .end
+ .to_buffer_point(&self.display_map, Bias::Left, app)
+ .unwrap();
start.column = 0;
end.column = buffer.line_len(end.row).unwrap();
start..end
@@ -1720,17 +1828,17 @@ impl BufferView {
}
fn is_line_foldable(&self, display_row: u32, app: &AppContext) -> bool {
- let map = self.display_map.read(app);
let max_point = self.max_point(app);
if display_row >= max_point.row() {
false
} else {
- let (start_indent, is_blank) = map.line_indent(display_row, app).unwrap();
+ let (start_indent, is_blank) = self.display_map.line_indent(display_row, app).unwrap();
if is_blank {
false
} else {
for display_row in display_row + 1..=max_point.row() {
- let (indent, is_blank) = map.line_indent(display_row, app).unwrap();
+ let (indent, is_blank) =
+ self.display_map.line_indent(display_row, app).unwrap();
if !is_blank {
return indent > start_indent;
}
@@ -1741,14 +1849,13 @@ impl BufferView {
}
fn foldable_range_for_line(&self, start_row: u32, app: &AppContext) -> Result<Range<Point>> {
- let map = self.display_map.read(app);
let max_point = self.max_point(app);
- let (start_indent, _) = map.line_indent(start_row, app)?;
+ let (start_indent, _) = self.display_map.line_indent(start_row, app)?;
let start = DisplayPoint::new(start_row, self.line_len(start_row, app)?);
let mut end = None;
for row in start_row + 1..=max_point.row() {
- let (indent, is_blank) = map.line_indent(row, app)?;
+ let (indent, is_blank) = self.display_map.line_indent(row, app)?;
if !is_blank && indent <= start_indent {
end = Some(DisplayPoint::new(row - 1, self.line_len(row - 1, app)?));
break;
@@ -1756,60 +1863,57 @@ impl BufferView {
}
let end = end.unwrap_or(max_point);
- return Ok(start.to_buffer_point(map, Bias::Left, app)?
- ..end.to_buffer_point(map, Bias::Left, app)?);
+ return Ok(start.to_buffer_point(&self.display_map, Bias::Left, app)?
+ ..end.to_buffer_point(&self.display_map, Bias::Left, app)?);
}
pub fn fold_selected_ranges(&mut self, _: &(), ctx: &mut ViewContext<Self>) {
use super::RangeExt;
- self.display_map.update(ctx, |map, ctx| {
- let buffer = self.buffer.read(ctx);
- let ranges = self
- .selections(ctx.as_ref())
- .iter()
- .map(|s| s.range(buffer).sorted())
- .collect::<Vec<_>>();
- map.fold(ranges, ctx).unwrap();
- });
+ let buffer = self.buffer.read(ctx);
+ let ranges = self
+ .selections(ctx.as_ref())
+ .iter()
+ .map(|s| s.range(buffer).sorted())
+ .collect::<Vec<_>>();
+ self.display_map.fold(ranges, ctx.as_ref()).unwrap();
+ ctx.notify();
}
fn fold_ranges<T: ToOffset>(&mut self, ranges: Vec<Range<T>>, ctx: &mut ViewContext<Self>) {
if !ranges.is_empty() {
- self.display_map.update(ctx, |map, ctx| {
- map.fold(ranges, ctx).unwrap();
- });
+ self.display_map.fold(ranges, ctx.as_ref()).unwrap();
*self.autoscroll_requested.lock() = true;
+ ctx.notify();
}
}
fn unfold_ranges<T: ToOffset>(&mut self, ranges: Vec<Range<T>>, ctx: &mut ViewContext<Self>) {
if !ranges.is_empty() {
- self.display_map.update(ctx, |map, ctx| {
- map.unfold(ranges, ctx).unwrap();
- });
+ self.display_map.unfold(ranges, ctx.as_ref()).unwrap();
*self.autoscroll_requested.lock() = true;
+ ctx.notify();
}
}
- pub fn line(&self, display_row: u32, app: &AppContext) -> Result<String> {
- self.display_map.read(app).line(display_row, app)
+ pub fn line(&self, display_row: u32, ctx: &AppContext) -> Result<String> {
+ self.display_map.line(display_row, ctx)
}
- pub fn line_len(&self, display_row: u32, app: &AppContext) -> Result<u32> {
- self.display_map.read(app).line_len(display_row, app)
+ pub fn line_len(&self, display_row: u32, ctx: &AppContext) -> Result<u32> {
+ self.display_map.line_len(display_row, ctx)
}
- pub fn rightmost_point(&self, app: &AppContext) -> DisplayPoint {
- self.display_map.read(app).rightmost_point()
+ pub fn rightmost_point(&self, ctx: &AppContext) -> DisplayPoint {
+ self.display_map.rightmost_point(ctx)
}
- pub fn max_point(&self, app: &AppContext) -> DisplayPoint {
- self.display_map.read(app).max_point(app)
+ pub fn max_point(&self, ctx: &AppContext) -> DisplayPoint {
+ self.display_map.max_point(ctx)
}
- pub fn text(&self, app: &AppContext) -> String {
- self.display_map.read(app).text(app)
+ pub fn text(&self, ctx: &AppContext) -> String {
+ self.display_map.text(ctx)
}
pub fn font_size(&self) -> f32 {
@@ -1872,10 +1976,8 @@ impl BufferView {
viewport_height: f32,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
- app: &AppContext,
+ ctx: &AppContext,
) -> Result<Vec<Arc<text_layout::Line>>> {
- let display_map = self.display_map.read(app);
-
let settings = smol::block_on(self.settings.read());
let font_size = settings.buffer_font_size;
let font_id =
@@ -1883,14 +1985,19 @@ impl BufferView {
let start_row = self.scroll_position().y() as usize;
let end_row = cmp::min(
- self.max_point(app).row() as usize,
+ self.max_point(ctx).row() as usize,
start_row + (viewport_height / self.line_height(font_cache)).ceil() as usize,
);
let line_count = end_row - start_row + 1;
let mut layouts = Vec::with_capacity(line_count);
let mut line_number = String::new();
- for buffer_row in display_map.buffer_rows(start_row as u32)?.take(line_count) {
+ for buffer_row in self
+ .display_map
+ .snapshot(ctx)
+ .buffer_rows(start_row as u32)?
+ .take(line_count)
+ {
line_number.clear();
write!(&mut line_number, "{}", buffer_row + 1).unwrap();
layouts.push(layout_cache.layout_str(
@@ -1908,11 +2015,9 @@ impl BufferView {
mut rows: Range<u32>,
font_cache: &FontCache,
layout_cache: &TextLayoutCache,
- app: &AppContext,
+ ctx: &AppContext,
) -> Result<Vec<Arc<text_layout::Line>>> {
- let display_map = self.display_map.read(app);
-
- rows.end = cmp::min(rows.end, display_map.max_point(app).row() + 1);
+ rows.end = cmp::min(rows.end, self.display_map.max_point(ctx).row() + 1);
if rows.start >= rows.end {
return Ok(Vec::new());
}
@@ -1926,8 +2031,9 @@ impl BufferView {
let mut line = String::new();
let mut line_len = 0;
let mut row = rows.start;
- let chars = display_map
- .chars_at(DisplayPoint::new(rows.start, 0), app)
+ let snapshot = self.display_map.snapshot(ctx);
+ let chars = snapshot
+ .chars_at(DisplayPoint::new(rows.start, 0), ctx)
.unwrap();
for char in chars.chain(Some('\n')) {
if char == '\n' {
@@ -2019,10 +2125,6 @@ impl BufferView {
ctx.notify();
}
- fn on_display_map_changed(&mut self, _: ModelHandle<DisplayMap>, ctx: &mut ViewContext<Self>) {
- ctx.notify();
- }
-
fn on_buffer_event(
&mut self,
_: ModelHandle<Buffer>,
@@ -3,10 +3,12 @@ use super::{
};
use crate::{
sum_tree::{self, Cursor, SumTree},
+ time,
util::find_insertion_index,
};
use anyhow::{anyhow, Result};
use gpui::{AppContext, ModelHandle};
+use parking_lot::{Mutex, MutexGuard};
use std::{
cmp::{self, Ordering},
iter::Take,
@@ -16,49 +18,44 @@ use sum_tree::{Dimension, SeekBias};
pub struct FoldMap {
buffer: ModelHandle<Buffer>,
- transforms: SumTree<Transform>,
+ transforms: Mutex<SumTree<Transform>>,
folds: Vec<Range<Anchor>>,
+ last_sync: Mutex<time::Global>,
}
impl FoldMap {
- pub fn new(buffer: ModelHandle<Buffer>, app: &AppContext) -> Self {
- let text_summary = buffer.read(app).text_summary();
+ pub fn new(buffer_handle: ModelHandle<Buffer>, ctx: &AppContext) -> Self {
+ let buffer = buffer_handle.read(ctx);
+ let text_summary = buffer.text_summary();
Self {
- buffer,
+ buffer: buffer_handle,
folds: Vec::new(),
- transforms: SumTree::from_item(Transform {
+ transforms: Mutex::new(SumTree::from_item(Transform {
summary: TransformSummary {
buffer: text_summary.clone(),
display: text_summary,
},
display_text: None,
- }),
+ })),
+ last_sync: Mutex::new(buffer.version()),
}
}
- pub fn buffer_rows(&self, start_row: u32) -> Result<BufferRows> {
- if start_row > self.transforms.summary().display.lines.row {
- return Err(anyhow!("invalid display row {}", start_row));
+ pub fn snapshot(&self, ctx: &AppContext) -> FoldMapSnapshot {
+ FoldMapSnapshot {
+ transforms: self.sync(ctx).clone(),
+ buffer: self.buffer.clone(),
}
-
- let display_point = Point::new(start_row, 0);
- let mut cursor = self.transforms.cursor();
- cursor.seek(&DisplayPoint(display_point), SeekBias::Left);
-
- Ok(BufferRows {
- display_point,
- cursor,
- })
}
- pub fn len(&self) -> usize {
- self.transforms.summary().display.chars
+ pub fn len(&self, ctx: &AppContext) -> usize {
+ self.sync(ctx).summary().display.chars
}
pub fn line_len(&self, row: u32, ctx: &AppContext) -> Result<u32> {
let line_start = self.to_display_offset(DisplayPoint::new(row, 0), ctx)?.0;
- let line_end = if row >= self.max_point().row() {
- self.len()
+ let line_end = if row >= self.max_point(ctx).row() {
+ self.len(ctx)
} else {
self.to_display_offset(DisplayPoint::new(row + 1, 0), ctx)?
.0
@@ -68,25 +65,12 @@ impl FoldMap {
Ok((line_end - line_start) as u32)
}
- pub fn chars_at<'a>(&'a self, point: DisplayPoint, app: &'a AppContext) -> Result<Chars<'a>> {
- let offset = self.to_display_offset(point, app)?;
- let mut cursor = self.transforms.cursor();
- cursor.seek(&offset, SeekBias::Right);
- let buffer = self.buffer.read(app);
- Ok(Chars {
- cursor,
- offset: offset.0,
- buffer,
- buffer_chars: None,
- })
+ pub fn max_point(&self, ctx: &AppContext) -> DisplayPoint {
+ DisplayPoint(self.sync(ctx).summary().display.lines)
}
- pub fn max_point(&self) -> DisplayPoint {
- DisplayPoint(self.transforms.summary().display.lines)
- }
-
- pub fn rightmost_point(&self) -> DisplayPoint {
- DisplayPoint(self.transforms.summary().display.rightmost_point)
+ pub fn rightmost_point(&self, ctx: &AppContext) -> DisplayPoint {
+ DisplayPoint(self.sync(ctx).summary().display.rightmost_point)
}
pub fn folds_in_range<'a, T>(
@@ -108,10 +92,12 @@ impl FoldMap {
pub fn fold<T: ToOffset>(
&mut self,
ranges: impl IntoIterator<Item = Range<T>>,
- app: &AppContext,
+ ctx: &AppContext,
) -> Result<()> {
+ let _ = self.sync(ctx);
+
let mut edits = Vec::new();
- let buffer = self.buffer.read(app);
+ let buffer = self.buffer.read(ctx);
for range in ranges.into_iter() {
let start = range.start.to_offset(buffer)?;
let end = range.end.to_offset(buffer)?;
@@ -131,16 +117,18 @@ impl FoldMap {
.then_with(|| b.old_range.end.cmp(&a.old_range.end))
});
- self.apply_edits(&edits, app)?;
+ self.apply_edits(edits, ctx);
Ok(())
}
pub fn unfold<T: ToOffset>(
&mut self,
ranges: impl IntoIterator<Item = Range<T>>,
- app: &AppContext,
+ ctx: &AppContext,
) -> Result<()> {
- let buffer = self.buffer.read(app);
+ let _ = self.sync(ctx);
+
+ let buffer = self.buffer.read(ctx);
let mut edits = Vec::new();
for range in ranges.into_iter() {
@@ -165,12 +153,13 @@ impl FoldMap {
});
}
- self.apply_edits(&edits, app)?;
+ self.apply_edits(edits, ctx);
Ok(())
}
- pub fn is_line_folded(&self, display_row: u32) -> bool {
- let mut cursor = self.transforms.cursor::<DisplayPoint, DisplayPoint>();
+ pub fn is_line_folded(&self, display_row: u32, ctx: &AppContext) -> bool {
+ let transforms = self.sync(ctx);
+ let mut cursor = transforms.cursor::<DisplayPoint, DisplayPoint>();
cursor.seek(&DisplayPoint::new(display_row, 0), SeekBias::Right);
while let Some(transform) = cursor.item() {
if transform.display_text.is_some() {
@@ -185,43 +174,33 @@ impl FoldMap {
false
}
- pub fn to_buffer_offset(&self, point: DisplayPoint, app: &AppContext) -> Result<usize> {
- let mut cursor = self.transforms.cursor::<DisplayPoint, TransformSummary>();
+ pub fn to_buffer_offset(&self, point: DisplayPoint, ctx: &AppContext) -> Result<usize> {
+ let transforms = self.sync(ctx);
+ let mut cursor = transforms.cursor::<DisplayPoint, TransformSummary>();
cursor.seek(&point, SeekBias::Right);
let overshoot = point.0 - cursor.start().display.lines;
- (cursor.start().buffer.lines + overshoot).to_offset(self.buffer.read(app))
+ (cursor.start().buffer.lines + overshoot).to_offset(self.buffer.read(ctx))
}
pub fn to_display_offset(
&self,
point: DisplayPoint,
- app: &AppContext,
+ ctx: &AppContext,
) -> Result<DisplayOffset> {
- let mut cursor = self.transforms.cursor::<DisplayPoint, TransformSummary>();
- cursor.seek(&point, SeekBias::Right);
- let overshoot = point.0 - cursor.start().display.lines;
- let mut offset = cursor.start().display.chars;
- if !overshoot.is_zero() {
- let transform = cursor
- .item()
- .ok_or_else(|| anyhow!("display point {:?} is out of range", point))?;
- assert!(transform.display_text.is_none());
- let end_buffer_offset =
- (cursor.start().buffer.lines + overshoot).to_offset(self.buffer.read(app))?;
- offset += end_buffer_offset - cursor.start().buffer.chars;
- }
- Ok(DisplayOffset(offset))
+ self.snapshot(ctx).to_display_offset(point, ctx)
}
- pub fn to_buffer_point(&self, display_point: DisplayPoint) -> Point {
- let mut cursor = self.transforms.cursor::<DisplayPoint, TransformSummary>();
+ pub fn to_buffer_point(&self, display_point: DisplayPoint, ctx: &AppContext) -> Point {
+ let transforms = self.sync(ctx);
+ let mut cursor = transforms.cursor::<DisplayPoint, TransformSummary>();
cursor.seek(&display_point, SeekBias::Right);
let overshoot = display_point.0 - cursor.start().display.lines;
cursor.start().buffer.lines + overshoot
}
- pub fn to_display_point(&self, point: Point) -> DisplayPoint {
- let mut cursor = self.transforms.cursor::<Point, TransformSummary>();
+ pub fn to_display_point(&self, point: Point, ctx: &AppContext) -> DisplayPoint {
+ let transforms = self.sync(ctx);
+ let mut cursor = transforms.cursor::<Point, TransformSummary>();
cursor.seek(&point, SeekBias::Right);
let overshoot = point - cursor.start().buffer.lines;
DisplayPoint(cmp::min(
@@ -230,12 +209,23 @@ impl FoldMap {
))
}
- pub fn apply_edits(&mut self, edits: &[Edit], app: &AppContext) -> Result<()> {
- let buffer = self.buffer.read(app);
- let mut edits = edits.iter().cloned().peekable();
+ fn sync(&self, ctx: &AppContext) -> MutexGuard<SumTree<Transform>> {
+ let buffer = self.buffer.read(ctx);
+ let mut edits = buffer.edits_since(self.last_sync.lock().clone()).peekable();
+ if edits.peek().is_some() {
+ self.apply_edits(edits, ctx);
+ }
+ *self.last_sync.lock() = buffer.version();
+ self.transforms.lock()
+ }
+
+ fn apply_edits(&self, edits: impl IntoIterator<Item = Edit>, ctx: &AppContext) {
+ let buffer = self.buffer.read(ctx);
+ let mut edits = edits.into_iter().peekable();
let mut new_transforms = SumTree::new();
- let mut cursor = self.transforms.cursor::<usize, usize>();
+ let mut transforms = self.transforms.lock();
+ let mut cursor = transforms.cursor::<usize, usize>();
cursor.seek(&0, SeekBias::Right);
while let Some(mut edit) = edits.next() {
@@ -271,9 +261,10 @@ impl FoldMap {
edit.new_range.end =
((edit.new_range.start + edit.old_extent()) as isize + delta) as usize;
- let anchor = buffer.anchor_before(edit.new_range.start)?;
+ let anchor = buffer.anchor_before(edit.new_range.start).unwrap();
let folds_start =
- find_insertion_index(&self.folds, |probe| probe.start.cmp(&anchor, buffer))?;
+ find_insertion_index(&self.folds, |probe| probe.start.cmp(&anchor, buffer))
+ .unwrap();
let mut folds = self.folds[folds_start..]
.iter()
.map(|fold| {
@@ -355,9 +346,58 @@ impl FoldMap {
}
drop(cursor);
- self.transforms = new_transforms;
+ *transforms = new_transforms;
+ }
+}
- Ok(())
+pub struct FoldMapSnapshot {
+ transforms: SumTree<Transform>,
+ buffer: ModelHandle<Buffer>,
+}
+
+impl FoldMapSnapshot {
+ pub fn buffer_rows(&self, start_row: u32) -> Result<BufferRows> {
+ if start_row > self.transforms.summary().display.lines.row {
+ return Err(anyhow!("invalid display row {}", start_row));
+ }
+
+ let display_point = Point::new(start_row, 0);
+ let mut cursor = self.transforms.cursor();
+ cursor.seek(&DisplayPoint(display_point), SeekBias::Left);
+
+ Ok(BufferRows {
+ display_point,
+ cursor,
+ })
+ }
+
+ pub fn chars_at<'a>(&'a self, point: DisplayPoint, ctx: &'a AppContext) -> Result<Chars<'a>> {
+ let offset = self.to_display_offset(point, ctx)?;
+ let mut cursor = self.transforms.cursor();
+ cursor.seek(&offset, SeekBias::Right);
+ Ok(Chars {
+ cursor,
+ offset: offset.0,
+ buffer: self.buffer.read(ctx),
+ buffer_chars: None,
+ })
+ }
+
+ fn to_display_offset(&self, point: DisplayPoint, ctx: &AppContext) -> Result<DisplayOffset> {
+ let mut cursor = self.transforms.cursor::<DisplayPoint, TransformSummary>();
+ cursor.seek(&point, SeekBias::Right);
+ let overshoot = point.0 - cursor.start().display.lines;
+ let mut offset = cursor.start().display.chars;
+ if !overshoot.is_zero() {
+ let transform = cursor
+ .item()
+ .ok_or_else(|| anyhow!("display point {:?} is out of range", point))?;
+ assert!(transform.display_text.is_none());
+ let end_buffer_offset =
+ (cursor.start().buffer.lines + overshoot).to_offset(self.buffer.read(ctx))?;
+ offset += end_buffer_offset - cursor.start().buffer.chars;
+ }
+ Ok(DisplayOffset(offset))
}
}
@@ -508,8 +548,7 @@ mod tests {
.unwrap();
assert_eq!(map.text(app.as_ref()), "aa…cc…eeeee");
- let edits = buffer.update(app, |buffer, ctx| {
- let start_version = buffer.version.clone();
+ buffer.update(app, |buffer, ctx| {
buffer
.edit(
vec![
@@ -520,21 +559,16 @@ mod tests {
Some(ctx),
)
.unwrap();
- buffer.edits_since(start_version).collect::<Vec<_>>()
});
-
- map.apply_edits(&edits, app.as_ref()).unwrap();
assert_eq!(map.text(app.as_ref()), "123a…c123c…eeeee");
- let edits = buffer.update(app, |buffer, ctx| {
+ buffer.update(app, |buffer, ctx| {
let start_version = buffer.version.clone();
buffer
.edit(Some(Point::new(2, 6)..Point::new(4, 3)), "456", Some(ctx))
.unwrap();
buffer.edits_since(start_version).collect::<Vec<_>>()
});
-
- map.apply_edits(&edits, app.as_ref()).unwrap();
assert_eq!(map.text(app.as_ref()), "123a…c123456eee");
map.unfold(Some(Point::new(0, 4)..Point::new(0, 4)), app.as_ref())
@@ -575,12 +609,11 @@ mod tests {
assert_eq!(map.text(app.as_ref()), "…fghijkl");
// Edit within one of the folds.
- let edits = buffer.update(app, |buffer, ctx| {
+ buffer.update(app, |buffer, ctx| {
let version = buffer.version();
buffer.edit(vec![0..1], "12345", Some(ctx)).unwrap();
buffer.edits_since(version).collect::<Vec<_>>()
});
- map.apply_edits(edits.as_slice(), app.as_ref()).unwrap();
map.check_invariants(app.as_ref());
assert_eq!(map.text(app.as_ref()), "12345…fghijkl");
}
@@ -622,15 +655,11 @@ mod tests {
.unwrap();
assert_eq!(map.text(app.as_ref()), "aa…cccc\nd…eeeee");
- let edits = buffer.update(app, |buffer, ctx| {
- let start_version = buffer.version.clone();
+ buffer.update(app, |buffer, ctx| {
buffer
.edit(Some(Point::new(2, 2)..Point::new(3, 1)), "", Some(ctx))
.unwrap();
- buffer.edits_since(start_version).collect::<Vec<_>>()
});
-
- map.apply_edits(&edits, app.as_ref()).unwrap();
assert_eq!(map.text(app.as_ref()), "aa…eeeee");
});
}
@@ -723,7 +752,6 @@ mod tests {
buffer.edits_since(start_version).collect::<Vec<_>>()
});
log::info!("editing {:?}", edits);
- map.apply_edits(&edits, app.as_ref()).unwrap();
}
map.check_invariants(app.as_ref());
@@ -752,9 +780,12 @@ mod tests {
let mut display_point = DisplayPoint::new(0, 0);
let mut display_offset = DisplayOffset(0);
for c in expected_text.chars() {
- let buffer_point = map.to_buffer_point(display_point);
+ let buffer_point = map.to_buffer_point(display_point, app.as_ref());
let buffer_offset = buffer_point.to_offset(buffer).unwrap();
- assert_eq!(map.to_display_point(buffer_point), display_point);
+ assert_eq!(
+ map.to_display_point(buffer_point, app.as_ref()),
+ display_point
+ );
assert_eq!(
map.to_buffer_offset(display_point, app.as_ref()).unwrap(),
buffer_offset
@@ -774,13 +805,14 @@ mod tests {
}
for _ in 0..5 {
- let row = rng.gen_range(0..=map.max_point().row());
+ let row = rng.gen_range(0..=map.max_point(app.as_ref()).row());
let column = rng.gen_range(0..=map.line_len(row, app.as_ref()).unwrap());
let point = DisplayPoint::new(row, column);
let offset = map.to_display_offset(point, app.as_ref()).unwrap().0;
- let len = rng.gen_range(0..=map.len() - offset);
+ let len = rng.gen_range(0..=map.len(app.as_ref()) - offset);
assert_eq!(
- map.chars_at(point, app.as_ref())
+ map.snapshot(app.as_ref())
+ .chars_at(point, app.as_ref())
.unwrap()
.take(len)
.collect::<String>(),
@@ -793,17 +825,24 @@ mod tests {
}
for (idx, buffer_row) in expected_buffer_rows.iter().enumerate() {
- let display_row = map.to_display_point(Point::new(*buffer_row, 0)).row();
+ let display_row = map
+ .to_display_point(Point::new(*buffer_row, 0), app.as_ref())
+ .row();
assert_eq!(
- map.buffer_rows(display_row).unwrap().collect::<Vec<_>>(),
+ map.snapshot(app.as_ref())
+ .buffer_rows(display_row)
+ .unwrap()
+ .collect::<Vec<_>>(),
expected_buffer_rows[idx..],
);
}
for fold_range in map.merged_fold_ranges(app.as_ref()) {
- let display_point =
- map.to_display_point(fold_range.start.to_point(buffer).unwrap());
- assert!(map.is_line_folded(display_point.row()));
+ let display_point = map.to_display_point(
+ fold_range.start.to_point(buffer).unwrap(),
+ app.as_ref(),
+ );
+ assert!(map.is_line_folded(display_point.row(), app.as_ref()));
}
}
});
@@ -829,16 +868,26 @@ mod tests {
assert_eq!(map.text(app.as_ref()), "aa…cccc\nd…eeeee\nffffff\n");
assert_eq!(
- map.buffer_rows(0).unwrap().collect::<Vec<_>>(),
+ map.snapshot(app.as_ref())
+ .buffer_rows(0)
+ .unwrap()
+ .collect::<Vec<_>>(),
vec![0, 3, 5, 6]
);
- assert_eq!(map.buffer_rows(3).unwrap().collect::<Vec<_>>(), vec![6]);
+ assert_eq!(
+ map.snapshot(app.as_ref())
+ .buffer_rows(3)
+ .unwrap()
+ .collect::<Vec<_>>(),
+ vec![6]
+ );
});
}
impl FoldMap {
fn text(&self, app: &AppContext) -> String {
- self.chars_at(DisplayPoint(Point::zero()), app)
+ self.snapshot(app)
+ .chars_at(DisplayPoint(Point::zero()), app)
.unwrap()
.collect()
}
@@ -874,10 +923,11 @@ mod tests {
merged_ranges
}
- fn check_invariants(&self, app: &AppContext) {
- let buffer = self.buffer.read(app);
+ fn check_invariants(&self, ctx: &AppContext) {
+ let transforms = self.sync(ctx);
+ let buffer = self.buffer.read(ctx);
assert_eq!(
- self.transforms.summary().buffer.chars,
+ transforms.summary().buffer.chars,
buffer.len(),
"transform tree does not match buffer's length"
);
@@ -3,8 +3,8 @@ mod fold_map;
use super::{buffer, Anchor, AnchorRangeExt, Buffer, Edit, Point, TextSummary, ToOffset, ToPoint};
use anyhow::Result;
pub use fold_map::BufferRows;
-use fold_map::FoldMap;
-use gpui::{AppContext, Entity, ModelContext, ModelHandle};
+use fold_map::{FoldMap, FoldMapSnapshot};
+use gpui::{AppContext, ModelHandle};
use std::ops::Range;
#[derive(Copy, Clone)]
@@ -19,21 +19,22 @@ pub struct DisplayMap {
tab_size: usize,
}
-impl Entity for DisplayMap {
- type Event = ();
-}
-
impl DisplayMap {
- pub fn new(buffer: ModelHandle<Buffer>, tab_size: usize, ctx: &mut ModelContext<Self>) -> Self {
- ctx.subscribe(&buffer, Self::handle_buffer_event);
-
+ pub fn new(buffer: ModelHandle<Buffer>, tab_size: usize, ctx: &AppContext) -> Self {
DisplayMap {
buffer: buffer.clone(),
- fold_map: FoldMap::new(buffer, ctx.as_ref()),
+ fold_map: FoldMap::new(buffer, ctx),
tab_size,
}
}
+ pub fn snapshot(&self, ctx: &AppContext) -> DisplayMapSnapshot {
+ DisplayMapSnapshot {
+ folds_snapshot: self.fold_map.snapshot(ctx),
+ tab_size: self.tab_size,
+ }
+ }
+
pub fn folds_in_range<'a, T>(
&'a self,
range: Range<T>,
@@ -48,40 +49,45 @@ impl DisplayMap {
pub fn fold<T: ToOffset>(
&mut self,
ranges: impl IntoIterator<Item = Range<T>>,
- ctx: &mut ModelContext<Self>,
+ ctx: &AppContext,
) -> Result<()> {
- self.fold_map.fold(ranges, ctx.as_ref())?;
- ctx.notify();
- Ok(())
+ self.fold_map.fold(ranges, ctx)
}
pub fn unfold<T: ToOffset>(
&mut self,
ranges: impl IntoIterator<Item = Range<T>>,
- ctx: &mut ModelContext<Self>,
+ ctx: &AppContext,
) -> Result<()> {
- self.fold_map.unfold(ranges, ctx.as_ref())?;
- ctx.notify();
- Ok(())
+ self.fold_map.unfold(ranges, ctx)
}
- pub fn is_line_folded(&self, display_row: u32) -> bool {
- self.fold_map.is_line_folded(display_row)
+ pub fn is_line_folded(&self, display_row: u32, ctx: &AppContext) -> bool {
+ self.fold_map.is_line_folded(display_row, ctx)
}
- pub fn text(&self, app: &AppContext) -> String {
- self.chars_at(DisplayPoint::zero(), app).unwrap().collect()
+ pub fn text(&self, ctx: &AppContext) -> String {
+ self.snapshot(ctx)
+ .chars_at(DisplayPoint::zero(), ctx)
+ .unwrap()
+ .collect()
}
- pub fn line(&self, display_row: u32, app: &AppContext) -> Result<String> {
- let chars = self.chars_at(DisplayPoint::new(display_row, 0), app)?;
- Ok(chars.take_while(|c| *c != '\n').collect())
+ pub fn line(&self, display_row: u32, ctx: &AppContext) -> Result<String> {
+ Ok(self
+ .snapshot(ctx)
+ .chars_at(DisplayPoint::new(display_row, 0), ctx)?
+ .take_while(|c| *c != '\n')
+ .collect())
}
- pub fn line_indent(&self, display_row: u32, app: &AppContext) -> Result<(u32, bool)> {
+ pub fn line_indent(&self, display_row: u32, ctx: &AppContext) -> Result<(u32, bool)> {
let mut indent = 0;
let mut is_blank = true;
- for c in self.chars_at(DisplayPoint::new(display_row, 0), app)? {
+ for c in self
+ .snapshot(ctx)
+ .chars_at(DisplayPoint::new(display_row, 0), ctx)?
+ {
if c == ' ' {
indent += 1;
} else {
@@ -92,38 +98,18 @@ impl DisplayMap {
Ok((indent, is_blank))
}
- pub fn chars_at<'a>(&'a self, point: DisplayPoint, app: &'a AppContext) -> Result<Chars<'a>> {
- let column = point.column() as usize;
- let (point, to_next_stop) = point.collapse_tabs(self, Bias::Left, app)?;
- let mut fold_chars = self.fold_map.chars_at(point, app)?;
- if to_next_stop > 0 {
- fold_chars.next();
- }
-
- Ok(Chars {
- fold_chars,
- column,
- to_next_stop,
- tab_size: self.tab_size,
- })
- }
-
- pub fn buffer_rows(&self, start_row: u32) -> Result<BufferRows> {
- self.fold_map.buffer_rows(start_row)
- }
-
pub fn line_len(&self, row: u32, ctx: &AppContext) -> Result<u32> {
DisplayPoint::new(row, self.fold_map.line_len(row, ctx)?)
.expand_tabs(self, ctx)
.map(|point| point.column())
}
- pub fn max_point(&self, app: &AppContext) -> DisplayPoint {
- self.fold_map.max_point().expand_tabs(self, app).unwrap()
+ pub fn max_point(&self, ctx: &AppContext) -> DisplayPoint {
+ self.fold_map.max_point(ctx).expand_tabs(self, ctx).unwrap()
}
- pub fn rightmost_point(&self) -> DisplayPoint {
- self.fold_map.rightmost_point()
+ pub fn rightmost_point(&self, ctx: &AppContext) -> DisplayPoint {
+ self.fold_map.rightmost_point(ctx)
}
pub fn anchor_before(
@@ -147,12 +133,57 @@ impl DisplayMap {
.read(app)
.anchor_after(point.to_buffer_point(self, bias, app)?)
}
+}
- fn handle_buffer_event(&mut self, event: &buffer::Event, ctx: &mut ModelContext<Self>) {
- match event {
- buffer::Event::Edited(edits) => self.fold_map.apply_edits(edits, ctx.as_ref()).unwrap(),
- _ => {}
+pub struct DisplayMapSnapshot {
+ folds_snapshot: FoldMapSnapshot,
+ tab_size: usize,
+}
+
+impl DisplayMapSnapshot {
+ pub fn buffer_rows(&self, start_row: u32) -> Result<BufferRows> {
+ self.folds_snapshot.buffer_rows(start_row)
+ }
+
+ pub fn chars_at<'a>(&'a self, point: DisplayPoint, app: &'a AppContext) -> Result<Chars<'a>> {
+ let column = point.column() as usize;
+ let (point, to_next_stop) = self.collapse_tabs(point, Bias::Left, app)?;
+ let mut fold_chars = self.folds_snapshot.chars_at(point, app)?;
+ if to_next_stop > 0 {
+ fold_chars.next();
}
+
+ Ok(Chars {
+ fold_chars,
+ column,
+ to_next_stop,
+ tab_size: self.tab_size,
+ })
+ }
+
+ fn expand_tabs(&self, mut point: DisplayPoint, ctx: &AppContext) -> Result<DisplayPoint> {
+ let chars = self
+ .folds_snapshot
+ .chars_at(DisplayPoint(Point::new(point.row(), 0)), ctx)?;
+ let expanded = expand_tabs(chars, point.column() as usize, self.tab_size);
+ *point.column_mut() = expanded as u32;
+ Ok(point)
+ }
+
+ fn collapse_tabs(
+ &self,
+ mut point: DisplayPoint,
+ bias: Bias,
+ ctx: &AppContext,
+ ) -> Result<(DisplayPoint, usize)> {
+ let chars = self
+ .folds_snapshot
+ .chars_at(DisplayPoint(Point::new(point.row(), 0)), ctx)?;
+ let expanded = point.column() as usize;
+ let (collapsed, to_next_stop) = collapse_tabs(chars, expanded, bias, self.tab_size);
+ *point.column_mut() = collapsed as u32;
+
+ Ok((point, to_next_stop))
}
}
@@ -184,50 +215,36 @@ impl DisplayPoint {
&mut self.0.column
}
- pub fn to_buffer_point(self, map: &DisplayMap, bias: Bias, app: &AppContext) -> Result<Point> {
+ pub fn to_buffer_point(self, map: &DisplayMap, bias: Bias, ctx: &AppContext) -> Result<Point> {
Ok(map
.fold_map
- .to_buffer_point(self.collapse_tabs(map, bias, app)?.0))
+ .to_buffer_point(self.collapse_tabs(map, bias, ctx)?.0, ctx))
}
- pub fn to_buffer_offset(self, map: &DisplayMap, bias: Bias, app: &AppContext) -> Result<usize> {
+ pub fn to_buffer_offset(self, map: &DisplayMap, bias: Bias, ctx: &AppContext) -> Result<usize> {
map.fold_map
- .to_buffer_offset(self.collapse_tabs(map, bias, app)?.0, app)
+ .to_buffer_offset(self.collapse_tabs(&map, bias, ctx)?.0, ctx)
}
- fn expand_tabs(mut self, map: &DisplayMap, app: &AppContext) -> Result<Self> {
- let chars = map
- .fold_map
- .chars_at(DisplayPoint(Point::new(self.row(), 0)), app)?;
- let expanded = expand_tabs(chars, self.column() as usize, map.tab_size);
- *self.column_mut() = expanded as u32;
-
- Ok(self)
+ fn expand_tabs(self, map: &DisplayMap, ctx: &AppContext) -> Result<Self> {
+ map.snapshot(ctx).expand_tabs(self, ctx)
}
fn collapse_tabs(
- mut self,
+ self,
map: &DisplayMap,
bias: Bias,
- app: &AppContext,
+ ctx: &AppContext,
) -> Result<(Self, usize)> {
- let chars = map
- .fold_map
- .chars_at(DisplayPoint(Point::new(self.0.row, 0)), app)?;
- let expanded = self.column() as usize;
- let (collapsed, to_next_stop) = collapse_tabs(chars, expanded, bias, map.tab_size);
- *self.column_mut() = collapsed as u32;
-
- Ok((self, to_next_stop))
+ map.snapshot(ctx).collapse_tabs(self, bias, ctx)
}
}
impl Point {
- pub fn to_display_point(self, map: &DisplayMap, app: &AppContext) -> Result<DisplayPoint> {
- let mut display_point = map.fold_map.to_display_point(self);
- let chars = map
- .fold_map
- .chars_at(DisplayPoint::new(display_point.row(), 0), app)?;
+ pub fn to_display_point(self, map: &DisplayMap, ctx: &AppContext) -> Result<DisplayPoint> {
+ let mut display_point = map.fold_map.to_display_point(self, ctx);
+ let snapshot = map.fold_map.snapshot(ctx);
+ let chars = snapshot.chars_at(DisplayPoint::new(display_point.row(), 0), ctx)?;
*display_point.column_mut() =
expand_tabs(chars, display_point.column() as usize, map.tab_size) as u32;
Ok(display_point)
@@ -329,7 +346,7 @@ mod tests {
App::test((), |app| {
let text = sample_text(6, 6);
let buffer = app.add_model(|ctx| Buffer::new(0, text, ctx));
- let map = app.add_model(|ctx| DisplayMap::new(buffer.clone(), 4, ctx));
+ let map = DisplayMap::new(buffer.clone(), 4, app.as_ref());
buffer
.update(app, |buffer, ctx| {
buffer.edit(
@@ -344,23 +361,25 @@ mod tests {
})
.unwrap();
- let map = map.read(app);
assert_eq!(
- map.chars_at(DisplayPoint::new(1, 0), app.as_ref())
+ map.snapshot(app.as_ref())
+ .chars_at(DisplayPoint::new(1, 0), app.as_ref())
.unwrap()
.take(10)
.collect::<String>(),
" b bb"
);
assert_eq!(
- map.chars_at(DisplayPoint::new(1, 2), app.as_ref())
+ map.snapshot(app.as_ref())
+ .chars_at(DisplayPoint::new(1, 2), app.as_ref())
.unwrap()
.take(10)
.collect::<String>(),
" b bbbb"
);
assert_eq!(
- map.chars_at(DisplayPoint::new(1, 6), app.as_ref())
+ map.snapshot(app.as_ref())
+ .chars_at(DisplayPoint::new(1, 6), app.as_ref())
.unwrap()
.take(13)
.collect::<String>(),
@@ -396,11 +415,8 @@ mod tests {
fn test_max_point() {
App::test((), |app| {
let buffer = app.add_model(|ctx| Buffer::new(0, "aaa\n\t\tbbb", ctx));
- let map = app.add_model(|ctx| DisplayMap::new(buffer.clone(), 4, ctx));
- assert_eq!(
- map.read(app).max_point(app.as_ref()),
- DisplayPoint::new(1, 11)
- )
+ let map = DisplayMap::new(buffer.clone(), 4, app.as_ref());
+ assert_eq!(map.max_point(app.as_ref()), DisplayPoint::new(1, 11))
});
}
}
@@ -96,7 +96,7 @@ pub fn prev_word_boundary(
let mut boundary = DisplayPoint::new(point.row(), 0);
let mut column = 0;
let mut prev_c = None;
- for c in map.chars_at(boundary, app)? {
+ for c in map.snapshot(app).chars_at(boundary, app)? {
if column >= point.column() {
break;
}
@@ -118,7 +118,7 @@ pub fn next_word_boundary(
app: &AppContext,
) -> Result<DisplayPoint> {
let mut prev_c = None;
- for c in map.chars_at(point, app)? {
+ for c in map.snapshot(app).chars_at(point, app)? {
if prev_c.is_some() && (c == '\n' || char_kind(prev_c.unwrap()) != char_kind(c)) {
break;
}