Get Editor compiling with MultiBuffer as its buffer

There's a bunch of unimplemented methods in MultiBuffer, but everything compiles.

Co-Authored-By: Max Brunsfeld <maxbrunsfeld@gmail.com>
This commit is contained in:
Nathan Sobo 2021-12-08 19:23:04 -07:00
parent daedf179b2
commit 87d16c271e
18 changed files with 931 additions and 450 deletions

View file

@ -6,7 +6,10 @@ mod wrap_map;
use block_map::{BlockMap, BlockPoint}; use block_map::{BlockMap, BlockPoint};
use fold_map::{FoldMap, ToFoldPoint as _}; use fold_map::{FoldMap, ToFoldPoint as _};
use gpui::{fonts::FontId, ElementBox, Entity, ModelContext, ModelHandle}; use gpui::{fonts::FontId, ElementBox, Entity, ModelContext, ModelHandle};
use language::{Anchor, Buffer, Point, Subscription as BufferSubscription, ToOffset, ToPoint}; use language::{
multi_buffer::{Anchor, MultiBuffer, MultiBufferSnapshot, ToOffset, ToPoint},
Point, Subscription as BufferSubscription,
};
use std::{ use std::{
collections::{HashMap, HashSet}, collections::{HashMap, HashSet},
ops::Range, ops::Range,
@ -26,7 +29,7 @@ pub trait ToDisplayPoint {
} }
pub struct DisplayMap { pub struct DisplayMap {
buffer: ModelHandle<Buffer>, buffer: ModelHandle<MultiBuffer>,
buffer_subscription: BufferSubscription, buffer_subscription: BufferSubscription,
fold_map: FoldMap, fold_map: FoldMap,
tab_map: TabMap, tab_map: TabMap,
@ -40,7 +43,7 @@ impl Entity for DisplayMap {
impl DisplayMap { impl DisplayMap {
pub fn new( pub fn new(
buffer: ModelHandle<Buffer>, buffer: ModelHandle<MultiBuffer>,
tab_size: usize, tab_size: usize,
font_id: FontId, font_id: FontId,
font_size: f32, font_size: f32,
@ -48,7 +51,7 @@ impl DisplayMap {
cx: &mut ModelContext<Self>, cx: &mut ModelContext<Self>,
) -> Self { ) -> Self {
let buffer_subscription = buffer.update(cx, |buffer, _| buffer.subscribe()); let buffer_subscription = buffer.update(cx, |buffer, _| buffer.subscribe());
let (fold_map, snapshot) = FoldMap::new(buffer.read(cx).snapshot()); let (fold_map, snapshot) = FoldMap::new(buffer.read(cx).snapshot(cx));
let (tab_map, snapshot) = TabMap::new(snapshot, tab_size); let (tab_map, snapshot) = TabMap::new(snapshot, tab_size);
let (wrap_map, snapshot) = WrapMap::new(snapshot, font_id, font_size, wrap_width, cx); let (wrap_map, snapshot) = WrapMap::new(snapshot, font_id, font_size, wrap_width, cx);
let block_map = BlockMap::new(buffer.clone(), snapshot); let block_map = BlockMap::new(buffer.clone(), snapshot);
@ -64,7 +67,7 @@ impl DisplayMap {
} }
pub fn snapshot(&self, cx: &mut ModelContext<Self>) -> DisplaySnapshot { pub fn snapshot(&self, cx: &mut ModelContext<Self>) -> DisplaySnapshot {
let buffer_snapshot = self.buffer.read(cx).snapshot(); let buffer_snapshot = self.buffer.read(cx).snapshot(cx);
let edits = self.buffer_subscription.consume().into_inner(); let edits = self.buffer_subscription.consume().into_inner();
let (folds_snapshot, edits) = self.fold_map.read(buffer_snapshot, edits); let (folds_snapshot, edits) = self.fold_map.read(buffer_snapshot, edits);
let (tabs_snapshot, edits) = self.tab_map.sync(folds_snapshot.clone(), edits); let (tabs_snapshot, edits) = self.tab_map.sync(folds_snapshot.clone(), edits);
@ -74,7 +77,7 @@ impl DisplayMap {
let blocks_snapshot = self.block_map.read(wraps_snapshot.clone(), edits, cx); let blocks_snapshot = self.block_map.read(wraps_snapshot.clone(), edits, cx);
DisplaySnapshot { DisplaySnapshot {
buffer_snapshot: self.buffer.read(cx).snapshot(), buffer_snapshot: self.buffer.read(cx).snapshot(cx),
folds_snapshot, folds_snapshot,
tabs_snapshot, tabs_snapshot,
wraps_snapshot, wraps_snapshot,
@ -87,7 +90,7 @@ impl DisplayMap {
ranges: impl IntoIterator<Item = Range<T>>, ranges: impl IntoIterator<Item = Range<T>>,
cx: &mut ModelContext<Self>, cx: &mut ModelContext<Self>,
) { ) {
let snapshot = self.buffer.read(cx).snapshot(); let snapshot = self.buffer.read(cx).snapshot(cx);
let edits = self.buffer_subscription.consume().into_inner(); let edits = self.buffer_subscription.consume().into_inner();
let (mut fold_map, snapshot, edits) = self.fold_map.write(snapshot, edits); let (mut fold_map, snapshot, edits) = self.fold_map.write(snapshot, edits);
let (snapshot, edits) = self.tab_map.sync(snapshot, edits); let (snapshot, edits) = self.tab_map.sync(snapshot, edits);
@ -108,7 +111,7 @@ impl DisplayMap {
ranges: impl IntoIterator<Item = Range<T>>, ranges: impl IntoIterator<Item = Range<T>>,
cx: &mut ModelContext<Self>, cx: &mut ModelContext<Self>,
) { ) {
let snapshot = self.buffer.read(cx).snapshot(); let snapshot = self.buffer.read(cx).snapshot(cx);
let edits = self.buffer_subscription.consume().into_inner(); let edits = self.buffer_subscription.consume().into_inner();
let (mut fold_map, snapshot, edits) = self.fold_map.write(snapshot, edits); let (mut fold_map, snapshot, edits) = self.fold_map.write(snapshot, edits);
let (snapshot, edits) = self.tab_map.sync(snapshot, edits); let (snapshot, edits) = self.tab_map.sync(snapshot, edits);
@ -132,7 +135,7 @@ impl DisplayMap {
where where
P: ToOffset + Clone, P: ToOffset + Clone,
{ {
let snapshot = self.buffer.read(cx).snapshot(); let snapshot = self.buffer.read(cx).snapshot(cx);
let edits = self.buffer_subscription.consume().into_inner(); let edits = self.buffer_subscription.consume().into_inner();
let (snapshot, edits) = self.fold_map.read(snapshot, edits); let (snapshot, edits) = self.fold_map.read(snapshot, edits);
let (snapshot, edits) = self.tab_map.sync(snapshot, edits); let (snapshot, edits) = self.tab_map.sync(snapshot, edits);
@ -151,7 +154,7 @@ impl DisplayMap {
} }
pub fn remove_blocks(&mut self, ids: HashSet<BlockId>, cx: &mut ModelContext<Self>) { pub fn remove_blocks(&mut self, ids: HashSet<BlockId>, cx: &mut ModelContext<Self>) {
let snapshot = self.buffer.read(cx).snapshot(); let snapshot = self.buffer.read(cx).snapshot(cx);
let edits = self.buffer_subscription.consume().into_inner(); let edits = self.buffer_subscription.consume().into_inner();
let (snapshot, edits) = self.fold_map.read(snapshot, edits); let (snapshot, edits) = self.fold_map.read(snapshot, edits);
let (snapshot, edits) = self.tab_map.sync(snapshot, edits); let (snapshot, edits) = self.tab_map.sync(snapshot, edits);
@ -179,7 +182,7 @@ impl DisplayMap {
} }
pub struct DisplaySnapshot { pub struct DisplaySnapshot {
pub buffer_snapshot: language::BufferSnapshot, pub buffer_snapshot: MultiBufferSnapshot,
folds_snapshot: fold_map::FoldSnapshot, folds_snapshot: fold_map::FoldSnapshot,
tabs_snapshot: tab_map::TabSnapshot, tabs_snapshot: tab_map::TabSnapshot,
wraps_snapshot: wrap_map::WrapSnapshot, wraps_snapshot: wrap_map::WrapSnapshot,
@ -457,7 +460,7 @@ mod tests {
use super::*; use super::*;
use crate::{movement, test::*}; use crate::{movement, test::*};
use gpui::{color::Color, MutableAppContext}; use gpui::{color::Color, MutableAppContext};
use language::{Language, LanguageConfig, RandomCharIter, SelectionGoal}; use language::{Buffer, Language, LanguageConfig, RandomCharIter, SelectionGoal};
use rand::{prelude::StdRng, Rng}; use rand::{prelude::StdRng, Rng};
use std::{env, sync::Arc}; use std::{env, sync::Arc};
use theme::SyntaxTheme; use theme::SyntaxTheme;
@ -489,10 +492,10 @@ mod tests {
log::info!("tab size: {}", tab_size); log::info!("tab size: {}", tab_size);
log::info!("wrap width: {:?}", wrap_width); log::info!("wrap width: {:?}", wrap_width);
let buffer = cx.add_model(|cx| { let buffer = cx.update(|cx| {
let len = rng.gen_range(0..10); let len = rng.gen_range(0..10);
let text = RandomCharIter::new(&mut rng).take(len).collect::<String>(); let text = RandomCharIter::new(&mut rng).take(len).collect::<String>();
Buffer::new(0, text, cx) MultiBuffer::build_simple(&text, cx)
}); });
let map = cx.add_model(|cx| { let map = cx.add_model(|cx| {
@ -563,8 +566,10 @@ mod tests {
assert_eq!(prev_display_bound.column(), 0); assert_eq!(prev_display_bound.column(), 0);
if next_display_bound < snapshot.max_point() { if next_display_bound < snapshot.max_point() {
assert_eq!( assert_eq!(
buffer buffer.read_with(&cx, |buffer, _| buffer
.read_with(&cx, |buffer, _| buffer.chars_at(next_buffer_bound).next()), .as_snapshot()
.chars_at(next_buffer_bound)
.next()),
Some('\n') Some('\n')
) )
} }
@ -651,7 +656,7 @@ mod tests {
let wrap_width = Some(64.); let wrap_width = Some(64.);
let text = "one two three four five\nsix seven eight"; let text = "one two three four five\nsix seven eight";
let buffer = cx.add_model(|cx| Buffer::new(0, text.to_string(), cx)); let buffer = MultiBuffer::build_simple(text, cx);
let map = cx.add_model(|cx| { let map = cx.add_model(|cx| {
DisplayMap::new(buffer.clone(), tab_size, font_id, font_size, wrap_width, cx) DisplayMap::new(buffer.clone(), tab_size, font_id, font_size, wrap_width, cx)
}); });
@ -724,7 +729,7 @@ mod tests {
#[gpui::test] #[gpui::test]
fn test_text_chunks(cx: &mut gpui::MutableAppContext) { fn test_text_chunks(cx: &mut gpui::MutableAppContext) {
let text = sample_text(6, 6, 'a'); let text = sample_text(6, 6, 'a');
let buffer = cx.add_model(|cx| Buffer::new(0, text, cx)); let buffer = MultiBuffer::build_simple(&text, cx);
let tab_size = 4; let tab_size = 4;
let family_id = cx.font_cache().load_family(&["Helvetica"]).unwrap(); let family_id = cx.font_cache().load_family(&["Helvetica"]).unwrap();
let font_id = cx let font_id = cx
@ -803,6 +808,7 @@ mod tests {
let buffer = let buffer =
cx.add_model(|cx| Buffer::new(0, text, cx).with_language(Some(lang), None, cx)); cx.add_model(|cx| Buffer::new(0, text, cx).with_language(Some(lang), None, cx));
buffer.condition(&cx, |buf, _| !buf.is_parsing()).await; buffer.condition(&cx, |buf, _| !buf.is_parsing()).await;
let buffer = cx.add_model(|cx| MultiBuffer::singleton(buffer, cx));
let tab_size = 2; let tab_size = 2;
let font_cache = cx.font_cache(); let font_cache = cx.font_cache();
@ -890,6 +896,7 @@ mod tests {
let buffer = let buffer =
cx.add_model(|cx| Buffer::new(0, text, cx).with_language(Some(lang), None, cx)); cx.add_model(|cx| Buffer::new(0, text, cx).with_language(Some(lang), None, cx));
buffer.condition(&cx, |buf, _| !buf.is_parsing()).await; buffer.condition(&cx, |buf, _| !buf.is_parsing()).await;
let buffer = cx.add_model(|cx| MultiBuffer::singleton(buffer, cx));
let font_cache = cx.font_cache(); let font_cache = cx.font_cache();
@ -935,7 +942,7 @@ mod tests {
let text = "\n'a', 'α',\t'✋',\t'❎', '🍐'\n"; let text = "\n'a', 'α',\t'✋',\t'❎', '🍐'\n";
let display_text = "\n'a', 'α', '✋', '❎', '🍐'\n"; let display_text = "\n'a', 'α', '✋', '❎', '🍐'\n";
let buffer = cx.add_model(|cx| Buffer::new(0, text, cx)); let buffer = MultiBuffer::build_simple(text, cx);
let tab_size = 4; let tab_size = 4;
let font_cache = cx.font_cache(); let font_cache = cx.font_cache();
@ -979,7 +986,7 @@ mod tests {
#[gpui::test] #[gpui::test]
fn test_tabs_with_multibyte_chars(cx: &mut gpui::MutableAppContext) { fn test_tabs_with_multibyte_chars(cx: &mut gpui::MutableAppContext) {
let text = "\t\tα\nβ\t\n🏀β\t\tγ"; let text = "\t\tα\nβ\t\n🏀β\t\tγ";
let buffer = cx.add_model(|cx| Buffer::new(0, text, cx)); let buffer = MultiBuffer::build_simple(text, cx);
let tab_size = 4; let tab_size = 4;
let font_cache = cx.font_cache(); let font_cache = cx.font_cache();
let family_id = font_cache.load_family(&["Helvetica"]).unwrap(); let family_id = font_cache.load_family(&["Helvetica"]).unwrap();
@ -1038,7 +1045,7 @@ mod tests {
#[gpui::test] #[gpui::test]
fn test_max_point(cx: &mut gpui::MutableAppContext) { fn test_max_point(cx: &mut gpui::MutableAppContext) {
let buffer = cx.add_model(|cx| Buffer::new(0, "aaa\n\t\tbbb", cx)); let buffer = MultiBuffer::build_simple("aaa\n\t\tbbb", cx);
let tab_size = 4; let tab_size = 4;
let font_cache = cx.font_cache(); let font_cache = cx.font_cache();
let family_id = font_cache.load_family(&["Helvetica"]).unwrap(); let family_id = font_cache.load_family(&["Helvetica"]).unwrap();

View file

@ -1,6 +1,9 @@
use super::wrap_map::{self, WrapEdit, WrapPoint, WrapSnapshot}; use super::wrap_map::{self, WrapEdit, WrapPoint, WrapSnapshot};
use gpui::{AppContext, ElementBox, ModelHandle}; use gpui::{AppContext, ElementBox, ModelHandle};
use language::{Buffer, Chunk}; use language::{
multi_buffer::{Anchor, MultiBuffer, ToOffset, ToPoint as _},
Chunk,
};
use parking_lot::Mutex; use parking_lot::Mutex;
use std::{ use std::{
cmp::{self, Ordering}, cmp::{self, Ordering},
@ -12,14 +15,14 @@ use std::{
Arc, Arc,
}, },
}; };
use sum_tree::SumTree; use sum_tree::{Bias, SumTree};
use text::{Anchor, Bias, Edit, Point, ToOffset, ToPoint as _}; use text::{Edit, Point};
use theme::SyntaxTheme; use theme::SyntaxTheme;
const NEWLINES: &'static [u8] = &[b'\n'; u8::MAX as usize]; const NEWLINES: &'static [u8] = &[b'\n'; u8::MAX as usize];
pub struct BlockMap { pub struct BlockMap {
buffer: ModelHandle<Buffer>, buffer: ModelHandle<MultiBuffer>,
next_block_id: AtomicUsize, next_block_id: AtomicUsize,
wrap_snapshot: Mutex<WrapSnapshot>, wrap_snapshot: Mutex<WrapSnapshot>,
blocks: Vec<Arc<Block>>, blocks: Vec<Arc<Block>>,
@ -109,7 +112,7 @@ pub struct BlockBufferRows<'a> {
} }
impl BlockMap { impl BlockMap {
pub fn new(buffer: ModelHandle<Buffer>, wrap_snapshot: WrapSnapshot) -> Self { pub fn new(buffer: ModelHandle<MultiBuffer>, wrap_snapshot: WrapSnapshot) -> Self {
Self { Self {
buffer, buffer,
next_block_id: AtomicUsize::new(0), next_block_id: AtomicUsize::new(0),
@ -153,6 +156,7 @@ impl BlockMap {
} }
let buffer = self.buffer.read(cx); let buffer = self.buffer.read(cx);
let buffer = buffer.as_snapshot();
let mut transforms = self.transforms.lock(); let mut transforms = self.transforms.lock();
let mut new_transforms = SumTree::new(); let mut new_transforms = SumTree::new();
let old_row_count = transforms.summary().input_rows; let old_row_count = transforms.summary().input_rows;
@ -241,7 +245,7 @@ impl BlockMap {
let start_block_ix = match self.blocks[last_block_ix..].binary_search_by(|probe| { let start_block_ix = match self.blocks[last_block_ix..].binary_search_by(|probe| {
probe probe
.position .position
.cmp(&start_anchor, buffer) .cmp(&start_anchor, &buffer)
.unwrap() .unwrap()
.then(Ordering::Greater) .then(Ordering::Greater)
}) { }) {
@ -255,7 +259,7 @@ impl BlockMap {
match self.blocks[start_block_ix..].binary_search_by(|probe| { match self.blocks[start_block_ix..].binary_search_by(|probe| {
probe probe
.position .position
.cmp(&end_anchor, buffer) .cmp(&end_anchor, &buffer)
.unwrap() .unwrap()
.then(Ordering::Greater) .then(Ordering::Greater)
}) { }) {
@ -268,7 +272,7 @@ impl BlockMap {
self.blocks[start_block_ix..end_block_ix] self.blocks[start_block_ix..end_block_ix]
.iter() .iter()
.map(|block| { .map(|block| {
let mut position = block.position.to_point(buffer); let mut position = block.position.to_point(&buffer);
let column = wrap_snapshot.from_point(position, Bias::Left).column(); let column = wrap_snapshot.from_point(position, Bias::Left).column();
match block.disposition { match block.disposition {
BlockDisposition::Above => position.column = 0, BlockDisposition::Above => position.column = 0,
@ -380,6 +384,7 @@ impl<'a> BlockMapWriter<'a> {
P: ToOffset + Clone, P: ToOffset + Clone,
{ {
let buffer = self.0.buffer.read(cx); let buffer = self.0.buffer.read(cx);
let buffer = buffer.as_snapshot();
let mut ids = Vec::new(); let mut ids = Vec::new();
let mut edits = Vec::<Edit<u32>>::new(); let mut edits = Vec::<Edit<u32>>::new();
let wrap_snapshot = &*self.0.wrap_snapshot.lock(); let wrap_snapshot = &*self.0.wrap_snapshot.lock();
@ -389,7 +394,7 @@ impl<'a> BlockMapWriter<'a> {
ids.push(id); ids.push(id);
let position = buffer.anchor_after(block.position); let position = buffer.anchor_after(block.position);
let point = position.to_point(buffer); let point = position.to_point(&buffer);
let start_row = wrap_snapshot let start_row = wrap_snapshot
.from_point(Point::new(point.row, 0), Bias::Left) .from_point(Point::new(point.row, 0), Bias::Left)
.row(); .row();
@ -404,7 +409,7 @@ impl<'a> BlockMapWriter<'a> {
let block_ix = match self let block_ix = match self
.0 .0
.blocks .blocks
.binary_search_by(|probe| probe.position.cmp(&position, buffer).unwrap()) .binary_search_by(|probe| probe.position.cmp(&position, &buffer).unwrap())
{ {
Ok(ix) | Err(ix) => ix, Ok(ix) | Err(ix) => ix,
}; };
@ -436,12 +441,13 @@ impl<'a> BlockMapWriter<'a> {
pub fn remove(&mut self, block_ids: HashSet<BlockId>, cx: &AppContext) { pub fn remove(&mut self, block_ids: HashSet<BlockId>, cx: &AppContext) {
let buffer = self.0.buffer.read(cx); let buffer = self.0.buffer.read(cx);
let buffer = buffer.as_snapshot();
let wrap_snapshot = &*self.0.wrap_snapshot.lock(); let wrap_snapshot = &*self.0.wrap_snapshot.lock();
let mut edits = Vec::new(); let mut edits = Vec::new();
let mut last_block_buffer_row = None; let mut last_block_buffer_row = None;
self.0.blocks.retain(|block| { self.0.blocks.retain(|block| {
if block_ids.contains(&block.id) { if block_ids.contains(&block.id) {
let buffer_row = block.position.to_point(buffer).row; let buffer_row = block.position.to_point(&buffer).row;
if last_block_buffer_row != Some(buffer_row) { if last_block_buffer_row != Some(buffer_row) {
last_block_buffer_row = Some(buffer_row); last_block_buffer_row = Some(buffer_row);
let start_row = wrap_snapshot let start_row = wrap_snapshot
@ -877,7 +883,6 @@ mod tests {
use super::*; use super::*;
use crate::display_map::{fold_map::FoldMap, tab_map::TabMap, wrap_map::WrapMap}; use crate::display_map::{fold_map::FoldMap, tab_map::TabMap, wrap_map::WrapMap};
use gpui::{elements::Empty, Element}; use gpui::{elements::Empty, Element};
use language::Buffer;
use rand::prelude::*; use rand::prelude::*;
use std::env; use std::env;
use text::RandomCharIter; use text::RandomCharIter;
@ -906,8 +911,9 @@ mod tests {
let text = "aaa\nbbb\nccc\nddd"; let text = "aaa\nbbb\nccc\nddd";
let buffer = cx.add_model(|cx| Buffer::new(0, text, cx)); let buffer = MultiBuffer::build_simple(text, cx);
let (fold_map, folds_snapshot) = FoldMap::new(buffer.read(cx).snapshot()); let subscription = buffer.update(cx, |buffer, _| buffer.subscribe());
let (fold_map, folds_snapshot) = FoldMap::new(buffer.read(cx).snapshot(cx));
let (tab_map, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), 1); let (tab_map, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), 1);
let (wrap_map, wraps_snapshot) = WrapMap::new(tabs_snapshot, font_id, 14.0, None, cx); let (wrap_map, wraps_snapshot) = WrapMap::new(tabs_snapshot, font_id, 14.0, None, cx);
let mut block_map = BlockMap::new(buffer.clone(), wraps_snapshot.clone()); let mut block_map = BlockMap::new(buffer.clone(), wraps_snapshot.clone());
@ -1050,15 +1056,14 @@ mod tests {
] ]
); );
// Insert a line break, separating two block decorations into separate // Insert a line break, separating two block decorations into separate lines.
// lines. let buffer_snapshot = buffer.update(cx, |buffer, cx| {
let (buffer_snapshot, buffer_edits) = buffer.update(cx, |buffer, cx| {
let v0 = buffer.version();
buffer.edit([Point::new(1, 1)..Point::new(1, 1)], "!!!\n", cx); buffer.edit([Point::new(1, 1)..Point::new(1, 1)], "!!!\n", cx);
(buffer.snapshot(), buffer.edits_since(&v0).collect()) buffer.snapshot(cx)
}); });
let (folds_snapshot, fold_edits) = fold_map.read(buffer_snapshot, buffer_edits); let (folds_snapshot, fold_edits) =
fold_map.read(buffer_snapshot, subscription.consume().into_inner());
let (tabs_snapshot, tab_edits) = tab_map.sync(folds_snapshot, fold_edits); let (tabs_snapshot, tab_edits) = tab_map.sync(folds_snapshot, fold_edits);
let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| { let (wraps_snapshot, wrap_edits) = wrap_map.update(cx, |wrap_map, cx| {
wrap_map.sync(tabs_snapshot, tab_edits, cx) wrap_map.sync(tabs_snapshot, tab_edits, cx)
@ -1077,8 +1082,8 @@ mod tests {
let text = "one two three\nfour five six\nseven eight"; let text = "one two three\nfour five six\nseven eight";
let buffer = cx.add_model(|cx| Buffer::new(0, text, cx)); let buffer = MultiBuffer::build_simple(text, cx);
let (_, folds_snapshot) = FoldMap::new(buffer.read(cx).snapshot()); let (_, folds_snapshot) = FoldMap::new(buffer.read(cx).snapshot(cx));
let (_, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), 1); let (_, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), 1);
let (_, wraps_snapshot) = WrapMap::new(tabs_snapshot, font_id, 14.0, Some(60.), cx); let (_, wraps_snapshot) = WrapMap::new(tabs_snapshot, font_id, 14.0, Some(60.), cx);
let mut block_map = BlockMap::new(buffer.clone(), wraps_snapshot.clone()); let mut block_map = BlockMap::new(buffer.clone(), wraps_snapshot.clone());
@ -1132,13 +1137,12 @@ mod tests {
log::info!("Wrap width: {:?}", wrap_width); log::info!("Wrap width: {:?}", wrap_width);
let buffer = cx.add_model(|cx| { let len = rng.gen_range(0..10);
let len = rng.gen_range(0..10); let text = RandomCharIter::new(&mut rng).take(len).collect::<String>();
let text = RandomCharIter::new(&mut rng).take(len).collect::<String>(); log::info!("initial buffer text: {:?}", text);
log::info!("initial buffer text: {:?}", text); let buffer = MultiBuffer::build_simple(&text, cx);
Buffer::new(0, text, cx)
}); let mut buffer_snapshot = buffer.read(cx).snapshot(cx);
let mut buffer_snapshot = buffer.read(cx).snapshot();
let (fold_map, folds_snapshot) = FoldMap::new(buffer_snapshot.clone()); let (fold_map, folds_snapshot) = FoldMap::new(buffer_snapshot.clone());
let (tab_map, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), tab_size); let (tab_map, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), tab_size);
let (wrap_map, wraps_snapshot) = let (wrap_map, wraps_snapshot) =
@ -1176,7 +1180,7 @@ mod tests {
log::info!( log::info!(
"inserting block {:?} {:?} with height {}", "inserting block {:?} {:?} with height {}",
disposition, disposition,
position.to_point(buffer), position.to_point(&buffer.as_snapshot()),
height height
); );
BlockProperties { BlockProperties {
@ -1221,12 +1225,12 @@ mod tests {
} }
_ => { _ => {
buffer.update(cx, |buffer, cx| { buffer.update(cx, |buffer, cx| {
let v0 = buffer.version();
let edit_count = rng.gen_range(1..=5); let edit_count = rng.gen_range(1..=5);
let subscription = buffer.subscribe();
buffer.randomly_edit(&mut rng, edit_count, cx); buffer.randomly_edit(&mut rng, edit_count, cx);
log::info!("buffer text: {:?}", buffer.text()); log::info!("buffer text: {:?}", buffer.text());
buffer_edits.extend(buffer.edits_since(&v0)); buffer_edits.extend(subscription.consume());
buffer_snapshot = buffer.snapshot(); buffer_snapshot = buffer.snapshot(cx);
}); });
} }
} }
@ -1248,7 +1252,7 @@ mod tests {
.iter() .iter()
.cloned() .cloned()
.map(|(id, block)| { .map(|(id, block)| {
let mut position = block.position.to_point(buffer); let mut position = block.position.to_point(&buffer.as_snapshot());
let column = wraps_snapshot.from_point(position, Bias::Left).column(); let column = wraps_snapshot.from_point(position, Bias::Left).column();
match block.disposition { match block.disposition {
BlockDisposition::Above => { BlockDisposition::Above => {

View file

@ -1,5 +1,6 @@
use language::{ use language::{
Anchor, AnchorRangeExt, BufferSnapshot, Chunk, Edit, Point, PointUtf16, TextSummary, ToOffset, multi_buffer::{Anchor, AnchorRangeExt, MultiBufferChunks, MultiBufferSnapshot, ToOffset},
Chunk, Edit, Point, PointUtf16, TextSummary,
}; };
use parking_lot::Mutex; use parking_lot::Mutex;
use std::{ use std::{
@ -189,14 +190,14 @@ impl<'a> FoldMapWriter<'a> {
} }
pub struct FoldMap { pub struct FoldMap {
buffer: Mutex<BufferSnapshot>, buffer: Mutex<MultiBufferSnapshot>,
transforms: Mutex<SumTree<Transform>>, transforms: Mutex<SumTree<Transform>>,
folds: SumTree<Fold>, folds: SumTree<Fold>,
version: AtomicUsize, version: AtomicUsize,
} }
impl FoldMap { impl FoldMap {
pub fn new(buffer: BufferSnapshot) -> (Self, FoldSnapshot) { pub fn new(buffer: MultiBufferSnapshot) -> (Self, FoldSnapshot) {
let this = Self { let this = Self {
buffer: Mutex::new(buffer.clone()), buffer: Mutex::new(buffer.clone()),
folds: Default::default(), folds: Default::default(),
@ -224,7 +225,7 @@ impl FoldMap {
pub fn read( pub fn read(
&self, &self,
buffer: BufferSnapshot, buffer: MultiBufferSnapshot,
edits: Vec<Edit<usize>>, edits: Vec<Edit<usize>>,
) -> (FoldSnapshot, Vec<FoldEdit>) { ) -> (FoldSnapshot, Vec<FoldEdit>) {
let edits = self.sync(buffer, edits); let edits = self.sync(buffer, edits);
@ -240,7 +241,7 @@ impl FoldMap {
pub fn write( pub fn write(
&mut self, &mut self,
buffer: BufferSnapshot, buffer: MultiBufferSnapshot,
edits: Vec<Edit<usize>>, edits: Vec<Edit<usize>>,
) -> (FoldMapWriter, FoldSnapshot, Vec<FoldEdit>) { ) -> (FoldMapWriter, FoldSnapshot, Vec<FoldEdit>) {
let (snapshot, edits) = self.read(buffer, edits); let (snapshot, edits) = self.read(buffer, edits);
@ -259,7 +260,7 @@ impl FoldMap {
fn sync( fn sync(
&self, &self,
new_buffer: BufferSnapshot, new_buffer: MultiBufferSnapshot,
buffer_edits: Vec<text::Edit<usize>>, buffer_edits: Vec<text::Edit<usize>>,
) -> Vec<FoldEdit> { ) -> Vec<FoldEdit> {
if buffer_edits.is_empty() { if buffer_edits.is_empty() {
@ -476,7 +477,7 @@ impl FoldMap {
pub struct FoldSnapshot { pub struct FoldSnapshot {
transforms: SumTree<Transform>, transforms: SumTree<Transform>,
folds: SumTree<Fold>, folds: SumTree<Fold>,
buffer_snapshot: language::BufferSnapshot, buffer_snapshot: MultiBufferSnapshot,
pub version: usize, pub version: usize,
} }
@ -699,7 +700,7 @@ impl FoldSnapshot {
} }
fn intersecting_folds<'a, T>( fn intersecting_folds<'a, T>(
buffer: &'a text::BufferSnapshot, buffer: &'a MultiBufferSnapshot,
folds: &'a SumTree<Fold>, folds: &'a SumTree<Fold>,
range: Range<T>, range: Range<T>,
inclusive: bool, inclusive: bool,
@ -850,9 +851,9 @@ impl Default for FoldSummary {
} }
impl sum_tree::Summary for FoldSummary { impl sum_tree::Summary for FoldSummary {
type Context = text::BufferSnapshot; type Context = MultiBufferSnapshot;
fn add_summary(&mut self, other: &Self, buffer: &text::BufferSnapshot) { fn add_summary(&mut self, other: &Self, buffer: &MultiBufferSnapshot) {
if other.min_start.cmp(&self.min_start, buffer).unwrap() == Ordering::Less { if other.min_start.cmp(&self.min_start, buffer).unwrap() == Ordering::Less {
self.min_start = other.min_start.clone(); self.min_start = other.min_start.clone();
} }
@ -876,20 +877,20 @@ impl sum_tree::Summary for FoldSummary {
} }
impl<'a> sum_tree::Dimension<'a, FoldSummary> for Fold { impl<'a> sum_tree::Dimension<'a, FoldSummary> for Fold {
fn add_summary(&mut self, summary: &'a FoldSummary, _: &text::BufferSnapshot) { fn add_summary(&mut self, summary: &'a FoldSummary, _: &MultiBufferSnapshot) {
self.0.start = summary.start.clone(); self.0.start = summary.start.clone();
self.0.end = summary.end.clone(); self.0.end = summary.end.clone();
} }
} }
impl<'a> sum_tree::SeekTarget<'a, FoldSummary, Fold> for Fold { impl<'a> sum_tree::SeekTarget<'a, FoldSummary, Fold> for Fold {
fn cmp(&self, other: &Self, buffer: &text::BufferSnapshot) -> Ordering { fn cmp(&self, other: &Self, buffer: &MultiBufferSnapshot) -> Ordering {
self.0.cmp(&other.0, buffer).unwrap() self.0.cmp(&other.0, buffer).unwrap()
} }
} }
impl<'a> sum_tree::Dimension<'a, FoldSummary> for usize { impl<'a> sum_tree::Dimension<'a, FoldSummary> for usize {
fn add_summary(&mut self, summary: &'a FoldSummary, _: &text::BufferSnapshot) { fn add_summary(&mut self, summary: &'a FoldSummary, _: &MultiBufferSnapshot) {
*self += summary.count; *self += summary.count;
} }
} }
@ -924,7 +925,7 @@ impl<'a> Iterator for FoldBufferRows<'a> {
pub struct FoldChunks<'a> { pub struct FoldChunks<'a> {
transform_cursor: Cursor<'a, Transform, (FoldOffset, usize)>, transform_cursor: Cursor<'a, Transform, (FoldOffset, usize)>,
buffer_chunks: language::BufferChunks<'a>, buffer_chunks: MultiBufferChunks<'a>,
buffer_chunk: Option<(usize, Chunk<'a>)>, buffer_chunk: Option<(usize, Chunk<'a>)>,
buffer_offset: usize, buffer_offset: usize,
output_offset: usize, output_offset: usize,
@ -1053,7 +1054,7 @@ pub type FoldEdit = Edit<FoldOffset>;
mod tests { mod tests {
use super::*; use super::*;
use crate::ToPoint; use crate::ToPoint;
use language::Buffer; use language::multi_buffer::MultiBuffer;
use rand::prelude::*; use rand::prelude::*;
use std::{env, mem}; use std::{env, mem};
use text::RandomCharIter; use text::RandomCharIter;
@ -1062,8 +1063,9 @@ mod tests {
#[gpui::test] #[gpui::test]
fn test_basic_folds(cx: &mut gpui::MutableAppContext) { fn test_basic_folds(cx: &mut gpui::MutableAppContext) {
let buffer = cx.add_model(|cx| Buffer::new(0, sample_text(5, 6, 'a'), cx)); let buffer = MultiBuffer::build_simple(&sample_text(5, 6, 'a'), cx);
let buffer_snapshot = buffer.read(cx).snapshot(); let subscription = buffer.update(cx, |buffer, _| buffer.subscribe());
let buffer_snapshot = buffer.read(cx).snapshot(cx);
let mut map = FoldMap::new(buffer_snapshot.clone()).0; let mut map = FoldMap::new(buffer_snapshot.clone()).0;
let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]); let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]);
@ -1086,8 +1088,7 @@ mod tests {
] ]
); );
let (buffer_snapshot, edits) = buffer.update(cx, |buffer, cx| { let buffer_snapshot = buffer.update(cx, |buffer, cx| {
let v0 = buffer.version();
buffer.edit( buffer.edit(
vec![ vec![
Point::new(0, 0)..Point::new(0, 1), Point::new(0, 0)..Point::new(0, 1),
@ -1096,9 +1097,10 @@ mod tests {
"123", "123",
cx, cx,
); );
(buffer.snapshot(), buffer.edits_since(&v0).collect()) buffer.snapshot(cx)
}); });
let (snapshot3, edits) = map.read(buffer_snapshot.clone(), edits); let (snapshot3, edits) =
map.read(buffer_snapshot.clone(), subscription.consume().into_inner());
assert_eq!(snapshot3.text(), "123a…c123c…eeeee"); assert_eq!(snapshot3.text(), "123a…c123c…eeeee");
assert_eq!( assert_eq!(
edits, edits,
@ -1114,12 +1116,11 @@ mod tests {
] ]
); );
let (buffer_snapshot, edits) = buffer.update(cx, |buffer, cx| { let buffer_snapshot = buffer.update(cx, |buffer, cx| {
let v0 = buffer.version();
buffer.edit(vec![Point::new(2, 6)..Point::new(4, 3)], "456", cx); buffer.edit(vec![Point::new(2, 6)..Point::new(4, 3)], "456", cx);
(buffer.snapshot(), buffer.edits_since(&v0).collect()) buffer.snapshot(cx)
}); });
let (snapshot4, _) = map.read(buffer_snapshot.clone(), edits); let (snapshot4, _) = map.read(buffer_snapshot.clone(), subscription.consume().into_inner());
assert_eq!(snapshot4.text(), "123a…c123456eee"); assert_eq!(snapshot4.text(), "123a…c123456eee");
let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]); let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]);
@ -1130,8 +1131,9 @@ mod tests {
#[gpui::test] #[gpui::test]
fn test_adjacent_folds(cx: &mut gpui::MutableAppContext) { fn test_adjacent_folds(cx: &mut gpui::MutableAppContext) {
let buffer = cx.add_model(|cx| Buffer::new(0, "abcdefghijkl", cx)); let buffer = MultiBuffer::build_simple("abcdefghijkl", cx);
let buffer_snapshot = buffer.read(cx).snapshot(); let subscription = buffer.update(cx, |buffer, _| buffer.subscribe());
let buffer_snapshot = buffer.read(cx).snapshot(cx);
{ {
let mut map = FoldMap::new(buffer_snapshot.clone()).0; let mut map = FoldMap::new(buffer_snapshot.clone()).0;
@ -1164,20 +1166,20 @@ mod tests {
assert_eq!(snapshot.text(), "…fghijkl"); assert_eq!(snapshot.text(), "…fghijkl");
// Edit within one of the folds. // Edit within one of the folds.
let (buffer_snapshot, edits) = buffer.update(cx, |buffer, cx| { let buffer_snapshot = buffer.update(cx, |buffer, cx| {
let v0 = buffer.version();
buffer.edit(vec![0..1], "12345", cx); buffer.edit(vec![0..1], "12345", cx);
(buffer.snapshot(), buffer.edits_since(&v0).collect()) buffer.snapshot(cx)
}); });
let (snapshot, _) = map.read(buffer_snapshot.clone(), edits); let (snapshot, _) =
map.read(buffer_snapshot.clone(), subscription.consume().into_inner());
assert_eq!(snapshot.text(), "12345…fghijkl"); assert_eq!(snapshot.text(), "12345…fghijkl");
} }
} }
#[gpui::test] #[gpui::test]
fn test_overlapping_folds(cx: &mut gpui::MutableAppContext) { fn test_overlapping_folds(cx: &mut gpui::MutableAppContext) {
let buffer = cx.add_model(|cx| Buffer::new(0, sample_text(5, 6, 'a'), cx)); let buffer = MultiBuffer::build_simple(&sample_text(5, 6, 'a'), cx);
let buffer_snapshot = buffer.read(cx).snapshot(); let buffer_snapshot = buffer.read(cx).snapshot(cx);
let mut map = FoldMap::new(buffer_snapshot.clone()).0; let mut map = FoldMap::new(buffer_snapshot.clone()).0;
let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]); let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]);
writer.fold(vec![ writer.fold(vec![
@ -1192,8 +1194,9 @@ mod tests {
#[gpui::test] #[gpui::test]
fn test_merging_folds_via_edit(cx: &mut gpui::MutableAppContext) { fn test_merging_folds_via_edit(cx: &mut gpui::MutableAppContext) {
let buffer = cx.add_model(|cx| Buffer::new(0, sample_text(5, 6, 'a'), cx)); let buffer = MultiBuffer::build_simple(&sample_text(5, 6, 'a'), cx);
let buffer_snapshot = buffer.read(cx).snapshot(); let subscription = buffer.update(cx, |buffer, _| buffer.subscribe());
let buffer_snapshot = buffer.read(cx).snapshot(cx);
let mut map = FoldMap::new(buffer_snapshot.clone()).0; let mut map = FoldMap::new(buffer_snapshot.clone()).0;
let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]); let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]);
@ -1204,19 +1207,18 @@ mod tests {
let (snapshot, _) = map.read(buffer_snapshot.clone(), vec![]); let (snapshot, _) = map.read(buffer_snapshot.clone(), vec![]);
assert_eq!(snapshot.text(), "aa…cccc\nd…eeeee"); assert_eq!(snapshot.text(), "aa…cccc\nd…eeeee");
let (buffer_snapshot, edits) = buffer.update(cx, |buffer, cx| { let buffer_snapshot = buffer.update(cx, |buffer, cx| {
let v0 = buffer.version();
buffer.edit(Some(Point::new(2, 2)..Point::new(3, 1)), "", cx); buffer.edit(Some(Point::new(2, 2)..Point::new(3, 1)), "", cx);
(buffer.snapshot(), buffer.edits_since(&v0).collect()) buffer.snapshot(cx)
}); });
let (snapshot, _) = map.read(buffer_snapshot.clone(), edits); let (snapshot, _) = map.read(buffer_snapshot.clone(), subscription.consume().into_inner());
assert_eq!(snapshot.text(), "aa…eeeee"); assert_eq!(snapshot.text(), "aa…eeeee");
} }
#[gpui::test] #[gpui::test]
fn test_folds_in_range(cx: &mut gpui::MutableAppContext) { fn test_folds_in_range(cx: &mut gpui::MutableAppContext) {
let buffer = cx.add_model(|cx| Buffer::new(0, sample_text(5, 6, 'a'), cx)); let buffer = MultiBuffer::build_simple(&sample_text(5, 6, 'a'), cx);
let buffer_snapshot = buffer.read(cx).snapshot(); let buffer_snapshot = buffer.read(cx).snapshot(cx);
let mut map = FoldMap::new(buffer_snapshot.clone()).0; let mut map = FoldMap::new(buffer_snapshot.clone()).0;
let buffer = buffer.read(cx); let buffer = buffer.read(cx);
@ -1230,7 +1232,9 @@ mod tests {
let (snapshot, _) = map.read(buffer_snapshot.clone(), vec![]); let (snapshot, _) = map.read(buffer_snapshot.clone(), vec![]);
let fold_ranges = snapshot let fold_ranges = snapshot
.folds_in_range(Point::new(1, 0)..Point::new(1, 3)) .folds_in_range(Point::new(1, 0)..Point::new(1, 3))
.map(|fold| fold.start.to_point(buffer)..fold.end.to_point(buffer)) .map(|fold| {
fold.start.to_point(&buffer.as_snapshot())..fold.end.to_point(&buffer.as_snapshot())
})
.collect::<Vec<_>>(); .collect::<Vec<_>>();
assert_eq!( assert_eq!(
fold_ranges, fold_ranges,
@ -1247,12 +1251,10 @@ mod tests {
.map(|i| i.parse().expect("invalid `OPERATIONS` variable")) .map(|i| i.parse().expect("invalid `OPERATIONS` variable"))
.unwrap_or(10); .unwrap_or(10);
let buffer = cx.add_model(|cx| { let len = rng.gen_range(0..10);
let len = rng.gen_range(0..10); let text = RandomCharIter::new(&mut rng).take(len).collect::<String>();
let text = RandomCharIter::new(&mut rng).take(len).collect::<String>(); let buffer = MultiBuffer::build_simple(&text, cx);
Buffer::new(0, text, cx) let buffer_snapshot = buffer.read(cx).snapshot(cx);
});
let buffer_snapshot = buffer.read(cx).snapshot();
let mut map = FoldMap::new(buffer_snapshot.clone()).0; let mut map = FoldMap::new(buffer_snapshot.clone()).0;
let (mut initial_snapshot, _) = map.read(buffer_snapshot.clone(), vec![]); let (mut initial_snapshot, _) = map.read(buffer_snapshot.clone(), vec![]);
@ -1260,23 +1262,21 @@ mod tests {
for _ in 0..operations { for _ in 0..operations {
log::info!("text: {:?}", buffer.read(cx).text()); log::info!("text: {:?}", buffer.read(cx).text());
let buffer_edits = match rng.gen_range(0..=100) { let mut buffer_edits = Vec::new();
match rng.gen_range(0..=100) {
0..=59 => { 0..=59 => {
snapshot_edits.extend(map.randomly_mutate(&mut rng)); snapshot_edits.extend(map.randomly_mutate(&mut rng));
vec![]
} }
_ => buffer.update(cx, |buffer, cx| { _ => buffer.update(cx, |buffer, cx| {
let start_version = buffer.version.clone(); let subscription = buffer.subscribe();
let edit_count = rng.gen_range(1..=5); let edit_count = rng.gen_range(1..=5);
buffer.randomly_edit(&mut rng, edit_count, cx); buffer.randomly_edit(&mut rng, edit_count, cx);
let edits = buffer let edits = subscription.consume().into_inner();
.edits_since::<Point>(&start_version)
.collect::<Vec<_>>();
log::info!("editing {:?}", edits); log::info!("editing {:?}", edits);
buffer.edits_since::<usize>(&start_version).collect() buffer_edits.extend(edits);
}), }),
}; };
let buffer_snapshot = buffer.read(cx).snapshot(); let buffer_snapshot = buffer.read(cx).snapshot(cx);
let (snapshot, edits) = map.read(buffer_snapshot.clone(), buffer_edits); let (snapshot, edits) = map.read(buffer_snapshot.clone(), buffer_edits);
snapshot_edits.push((snapshot.clone(), edits)); snapshot_edits.push((snapshot.clone(), edits));
@ -1285,8 +1285,8 @@ mod tests {
let mut expected_buffer_rows = Vec::new(); let mut expected_buffer_rows = Vec::new();
let mut next_row = buffer_snapshot.max_point().row; let mut next_row = buffer_snapshot.max_point().row;
for fold_range in map.merged_fold_ranges().into_iter().rev() { for fold_range in map.merged_fold_ranges().into_iter().rev() {
let fold_start = buffer_snapshot.point_for_offset(fold_range.start).unwrap(); let fold_start = buffer_snapshot.offset_to_point(fold_range.start);
let fold_end = buffer_snapshot.point_for_offset(fold_range.end).unwrap(); let fold_end = buffer_snapshot.offset_to_point(fold_range.end);
expected_buffer_rows.extend((fold_end.row + 1..=next_row).rev()); expected_buffer_rows.extend((fold_end.row + 1..=next_row).rev());
next_row = fold_start.row; next_row = fold_start.row;
@ -1458,9 +1458,9 @@ mod tests {
#[gpui::test] #[gpui::test]
fn test_buffer_rows(cx: &mut gpui::MutableAppContext) { fn test_buffer_rows(cx: &mut gpui::MutableAppContext) {
let text = sample_text(6, 6, 'a') + "\n"; let text = sample_text(6, 6, 'a') + "\n";
let buffer = cx.add_model(|cx| Buffer::new(0, text, cx)); let buffer = MultiBuffer::build_simple(&text, cx);
let buffer_snapshot = buffer.read(cx).snapshot(); let buffer_snapshot = buffer.read(cx).snapshot(cx);
let mut map = FoldMap::new(buffer_snapshot.clone()).0; let mut map = FoldMap::new(buffer_snapshot.clone()).0;
let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]); let (mut writer, _, _) = map.write(buffer_snapshot.clone(), vec![]);

View file

@ -435,7 +435,7 @@ impl<'a> Iterator for TabChunks<'a> {
mod tests { mod tests {
use super::*; use super::*;
use crate::display_map::fold_map::FoldMap; use crate::display_map::fold_map::FoldMap;
use language::Buffer; use language::multi_buffer::MultiBuffer;
use rand::{prelude::StdRng, Rng}; use rand::{prelude::StdRng, Rng};
use text::{RandomCharIter, Rope}; use text::{RandomCharIter, Rope};
@ -449,12 +449,10 @@ mod tests {
#[gpui::test(iterations = 100)] #[gpui::test(iterations = 100)]
fn test_random(cx: &mut gpui::MutableAppContext, mut rng: StdRng) { fn test_random(cx: &mut gpui::MutableAppContext, mut rng: StdRng) {
let tab_size = rng.gen_range(1..=4); let tab_size = rng.gen_range(1..=4);
let buffer = cx.add_model(|cx| { let len = rng.gen_range(0..30);
let len = rng.gen_range(0..30); let text = RandomCharIter::new(&mut rng).take(len).collect::<String>();
let text = RandomCharIter::new(&mut rng).take(len).collect::<String>(); let buffer = MultiBuffer::build_simple(&text, cx);
Buffer::new(0, text, cx) let buffer_snapshot = buffer.read(cx).snapshot(cx);
});
let buffer_snapshot = buffer.read(cx).snapshot();
log::info!("Buffer text: {:?}", buffer.read(cx).text()); log::info!("Buffer text: {:?}", buffer.read(cx).text());
let (mut fold_map, _) = FoldMap::new(buffer_snapshot.clone()); let (mut fold_map, _) = FoldMap::new(buffer_snapshot.clone());

View file

@ -974,7 +974,7 @@ mod tests {
display_map::{fold_map::FoldMap, tab_map::TabMap}, display_map::{fold_map::FoldMap, tab_map::TabMap},
test::Observer, test::Observer,
}; };
use language::{Buffer, RandomCharIter}; use language::{multi_buffer::MultiBuffer, RandomCharIter};
use rand::prelude::*; use rand::prelude::*;
use std::{cmp, env}; use std::{cmp, env};
use text::Rope; use text::Rope;
@ -1004,12 +1004,12 @@ mod tests {
log::info!("Tab size: {}", tab_size); log::info!("Tab size: {}", tab_size);
log::info!("Wrap width: {:?}", wrap_width); log::info!("Wrap width: {:?}", wrap_width);
let buffer = cx.add_model(|cx| { let buffer = cx.update(|cx| {
let len = rng.gen_range(0..10); let len = rng.gen_range(0..10);
let text = RandomCharIter::new(&mut rng).take(len).collect::<String>(); let text = RandomCharIter::new(&mut rng).take(len).collect::<String>();
Buffer::new(0, text, cx) MultiBuffer::build_simple(&text, cx)
}); });
let buffer_snapshot = buffer.read_with(&cx, |buffer, _| buffer.snapshot()); let buffer_snapshot = buffer.read_with(&cx, |buffer, cx| buffer.snapshot(cx));
let (mut fold_map, folds_snapshot) = FoldMap::new(buffer_snapshot.clone()); let (mut fold_map, folds_snapshot) = FoldMap::new(buffer_snapshot.clone());
let (tab_map, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), tab_size); let (tab_map, tabs_snapshot) = TabMap::new(folds_snapshot.clone(), tab_size);
log::info!( log::info!(
@ -1074,15 +1074,15 @@ mod tests {
} }
_ => { _ => {
buffer.update(&mut cx, |buffer, cx| { buffer.update(&mut cx, |buffer, cx| {
let v0 = buffer.version(); let subscription = buffer.subscribe();
let edit_count = rng.gen_range(1..=5); let edit_count = rng.gen_range(1..=5);
buffer.randomly_edit(&mut rng, edit_count, cx); buffer.randomly_edit(&mut rng, edit_count, cx);
buffer_edits.extend(buffer.edits_since(&v0)); buffer_edits.extend(subscription.consume());
}); });
} }
} }
let buffer_snapshot = buffer.read_with(&cx, |buffer, _| buffer.snapshot()); let buffer_snapshot = buffer.read_with(&cx, |buffer, cx| buffer.snapshot(cx));
log::info!("Unwrapped text (no folds): {:?}", buffer_snapshot.text()); log::info!("Unwrapped text (no folds): {:?}", buffer_snapshot.text());
let (folds_snapshot, fold_edits) = fold_map.read(buffer_snapshot, buffer_edits); let (folds_snapshot, fold_edits) = fold_map.read(buffer_snapshot, buffer_edits);
log::info!( log::info!(

File diff suppressed because it is too large Load diff

View file

@ -19,7 +19,7 @@ use gpui::{
MutableAppContext, PaintContext, Quad, Scene, SizeConstraint, ViewContext, WeakViewHandle, MutableAppContext, PaintContext, Quad, Scene, SizeConstraint, ViewContext, WeakViewHandle,
}; };
use json::json; use json::json;
use language::{Chunk, ToPoint}; use language::{multi_buffer::ToPoint, Chunk};
use smallvec::SmallVec; use smallvec::SmallVec;
use std::{ use std::{
cmp::{self, Ordering}, cmp::{self, Ordering},
@ -738,13 +738,11 @@ impl Element for EditorElement {
self.update_view(cx.app, |view, cx| { self.update_view(cx.app, |view, cx| {
highlighted_row = view.highlighted_row(); highlighted_row = view.highlighted_row();
for selection_set_id in view.active_selection_sets(cx).collect::<Vec<_>>() { for selection_set_id in view.active_selection_sets(cx).collect::<Vec<_>>() {
let replica_selections = view let replica_selections = view.intersecting_selections(
.intersecting_selections( selection_set_id,
selection_set_id, DisplayPoint::new(start_row, 0)..DisplayPoint::new(end_row, 0),
DisplayPoint::new(start_row, 0)..DisplayPoint::new(end_row, 0), cx,
cx, );
)
.collect::<Vec<_>>();
for selection in &replica_selections { for selection in &replica_selections {
if selection_set_id == view.selection_set_id { if selection_set_id == view.selection_set_id {
let is_empty = selection.start == selection.end; let is_empty = selection.start == selection.end;
@ -1165,14 +1163,13 @@ fn scale_horizontal_mouse_autoscroll_delta(delta: f32) -> f32 {
mod tests { mod tests {
use super::*; use super::*;
use crate::{Editor, EditorSettings}; use crate::{Editor, EditorSettings};
use language::Buffer; use language::{MultiBuffer};
use util::test::sample_text; use util::test::sample_text;
#[gpui::test] #[gpui::test]
fn test_layout_line_numbers(cx: &mut gpui::MutableAppContext) { fn test_layout_line_numbers(cx: &mut gpui::MutableAppContext) {
let settings = EditorSettings::test(cx); let settings = EditorSettings::test(cx);
let buffer = MultiBuffer::build_simple(&sample_text(6, 6, 'a'), cx);
let buffer = cx.add_model(|cx| Buffer::new(0, sample_text(6, 6, 'a'), cx));
let (window_id, editor) = cx.add_window(Default::default(), |cx| { let (window_id, editor) = cx.add_window(Default::default(), |cx| {
Editor::for_buffer( Editor::for_buffer(
buffer, buffer,

View file

@ -5,12 +5,15 @@ use gpui::{
MutableAppContext, RenderContext, Subscription, Task, View, ViewContext, ViewHandle, MutableAppContext, RenderContext, Subscription, Task, View, ViewContext, ViewHandle,
WeakModelHandle, WeakModelHandle,
}; };
use language::{Buffer, Diagnostic, File as _}; use language::{
multi_buffer::{MultiBuffer, ToPoint as _},
Diagnostic, File as _,
};
use postage::watch; use postage::watch;
use project::{ProjectPath, Worktree}; use project::{ProjectPath, Worktree};
use std::fmt::Write; use std::fmt::Write;
use std::path::Path; use std::path::Path;
use text::{Point, Selection, ToPoint}; use text::{Point, Selection};
use workspace::{ use workspace::{
settings, EntryOpener, ItemHandle, ItemView, ItemViewHandle, Settings, StatusItemView, settings, EntryOpener, ItemHandle, ItemView, ItemViewHandle, Settings, StatusItemView,
WeakItemHandle, WeakItemHandle,
@ -19,10 +22,10 @@ use workspace::{
pub struct BufferOpener; pub struct BufferOpener;
#[derive(Clone)] #[derive(Clone)]
pub struct BufferItemHandle(pub ModelHandle<Buffer>); pub struct BufferItemHandle(pub ModelHandle<MultiBuffer>);
#[derive(Clone)] #[derive(Clone)]
struct WeakBufferItemHandle(WeakModelHandle<Buffer>); struct WeakBufferItemHandle(WeakModelHandle<MultiBuffer>);
impl EntryOpener for BufferOpener { impl EntryOpener for BufferOpener {
fn open( fn open(
@ -32,10 +35,10 @@ impl EntryOpener for BufferOpener {
cx: &mut ModelContext<Worktree>, cx: &mut ModelContext<Worktree>,
) -> Option<Task<Result<Box<dyn ItemHandle>>>> { ) -> Option<Task<Result<Box<dyn ItemHandle>>>> {
let buffer = worktree.open_buffer(project_path.path, cx); let buffer = worktree.open_buffer(project_path.path, cx);
let task = cx.spawn(|_, _| async move { let task = cx.spawn(|_, mut cx| async move {
buffer let buffer = buffer.await?;
.await let buffer = cx.add_model(|cx| MultiBuffer::singleton(buffer, cx));
.map(|buffer| Box::new(BufferItemHandle(buffer)) as Box<dyn ItemHandle>) Ok(Box::new(BufferItemHandle(buffer)) as Box<dyn ItemHandle>)
}); });
Some(task) Some(task)
} }
@ -102,7 +105,7 @@ impl ItemHandle for BufferItemHandle {
} }
fn project_path(&self, cx: &AppContext) -> Option<ProjectPath> { fn project_path(&self, cx: &AppContext) -> Option<ProjectPath> {
self.0.read(cx).file().map(|f| ProjectPath { self.0.read(cx).file(cx).map(|f| ProjectPath {
worktree_id: f.worktree_id(), worktree_id: f.worktree_id(),
path: f.path().clone(), path: f.path().clone(),
}) })
@ -137,7 +140,7 @@ impl ItemView for Editor {
let filename = self let filename = self
.buffer() .buffer()
.read(cx) .read(cx)
.file() .file(cx)
.and_then(|file| file.file_name()); .and_then(|file| file.file_name());
if let Some(name) = filename { if let Some(name) = filename {
name.to_string_lossy().into() name.to_string_lossy().into()
@ -147,7 +150,7 @@ impl ItemView for Editor {
} }
fn project_path(&self, cx: &AppContext) -> Option<ProjectPath> { fn project_path(&self, cx: &AppContext) -> Option<ProjectPath> {
self.buffer().read(cx).file().map(|file| ProjectPath { self.buffer().read(cx).file(cx).map(|file| ProjectPath {
worktree_id: file.worktree_id(), worktree_id: file.worktree_id(),
path: file.path().clone(), path: file.path().clone(),
}) })
@ -174,7 +177,14 @@ impl ItemView for Editor {
path: &Path, path: &Path,
cx: &mut ViewContext<Self>, cx: &mut ViewContext<Self>,
) -> Task<Result<()>> { ) -> Task<Result<()>> {
self.buffer().update(cx, |buffer, cx| { let buffer = self
.buffer()
.read(cx)
.as_singleton()
.expect("cannot call save_as on an excerpt list")
.clone();
buffer.update(cx, |buffer, cx| {
let handle = cx.handle(); let handle = cx.handle();
let text = buffer.as_rope().clone(); let text = buffer.as_rope().clone();
let version = buffer.version(); let version = buffer.version();
@ -237,7 +247,7 @@ impl CursorPosition {
fn update_position(&mut self, editor: ViewHandle<Editor>, cx: &mut ViewContext<Self>) { fn update_position(&mut self, editor: ViewHandle<Editor>, cx: &mut ViewContext<Self>) {
let editor = editor.read(cx); let editor = editor.read(cx);
let buffer = editor.buffer().read(cx); let buffer = editor.buffer().read(cx).snapshot(cx);
self.selected_count = 0; self.selected_count = 0;
let mut last_selection: Option<Selection<usize>> = None; let mut last_selection: Option<Selection<usize>> = None;
@ -250,7 +260,7 @@ impl CursorPosition {
last_selection = Some(selection); last_selection = Some(selection);
} }
} }
self.position = last_selection.map(|s| s.head().to_point(buffer)); self.position = last_selection.map(|s| s.head().to_point(&buffer));
cx.notify(); cx.notify();
} }

View file

@ -1,7 +1,7 @@
use super::{Bias, DisplayPoint, DisplaySnapshot, SelectionGoal, ToDisplayPoint}; use super::{Bias, DisplayPoint, DisplaySnapshot, SelectionGoal, ToDisplayPoint};
use anyhow::Result; use anyhow::Result;
use language::multi_buffer::ToPoint;
use std::{cmp, ops::Range}; use std::{cmp, ops::Range};
use text::ToPoint;
pub fn left(map: &DisplaySnapshot, mut point: DisplayPoint) -> Result<DisplayPoint> { pub fn left(map: &DisplaySnapshot, mut point: DisplayPoint) -> Result<DisplayPoint> {
if point.column() > 0 { if point.column() > 0 {
@ -244,7 +244,8 @@ fn char_kind(c: char) -> CharKind {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
use crate::{display_map::DisplayMap, Buffer}; use crate::display_map::DisplayMap;
use language::MultiBuffer;
#[gpui::test] #[gpui::test]
fn test_prev_next_word_boundary_multibyte(cx: &mut gpui::MutableAppContext) { fn test_prev_next_word_boundary_multibyte(cx: &mut gpui::MutableAppContext) {
@ -256,7 +257,7 @@ mod tests {
.unwrap(); .unwrap();
let font_size = 14.0; let font_size = 14.0;
let buffer = cx.add_model(|cx| Buffer::new(0, "a bcΔ defγ hi—jk", cx)); let buffer = MultiBuffer::build_simple("a bcΔ defγ hi—jk", cx);
let display_map = let display_map =
cx.add_model(|cx| DisplayMap::new(buffer, tab_size, font_id, font_size, None, cx)); cx.add_model(|cx| DisplayMap::new(buffer, tab_size, font_id, font_size, None, cx));
let snapshot = display_map.update(cx, |map, cx| map.snapshot(cx)); let snapshot = display_map.update(cx, |map, cx| map.snapshot(cx));
@ -312,7 +313,7 @@ mod tests {
.select_font(family_id, &Default::default()) .select_font(family_id, &Default::default())
.unwrap(); .unwrap();
let font_size = 14.0; let font_size = 14.0;
let buffer = cx.add_model(|cx| Buffer::new(0, "lorem ipsum dolor\n sit", cx)); let buffer = MultiBuffer::build_simple("lorem ipsum dolor\n sit", cx);
let display_map = let display_map =
cx.add_model(|cx| DisplayMap::new(buffer, tab_size, font_id, font_size, None, cx)); cx.add_model(|cx| DisplayMap::new(buffer, tab_size, font_id, font_size, None, cx));
let snapshot = display_map.update(cx, |map, cx| map.snapshot(cx)); let snapshot = display_map.update(cx, |map, cx| map.snapshot(cx));

View file

@ -67,7 +67,7 @@ impl GoToLine {
let (restore_state, cursor_point, max_point) = active_editor.update(cx, |editor, cx| { let (restore_state, cursor_point, max_point) = active_editor.update(cx, |editor, cx| {
let restore_state = Some(RestoreState { let restore_state = Some(RestoreState {
scroll_position: editor.scroll_position(cx), scroll_position: editor.scroll_position(cx),
selections: editor.selections::<usize>(cx).collect(), selections: editor.selections::<usize>(cx),
}); });
( (

View file

@ -1,6 +1,6 @@
mod buffer; mod buffer;
mod highlight_map; mod highlight_map;
mod multi_buffer; pub mod multi_buffer;
pub mod proto; pub mod proto;
#[cfg(test)] #[cfg(test)]
mod tests; mod tests;
@ -12,6 +12,7 @@ use gpui::{executor::Background, AppContext};
use highlight_map::HighlightMap; use highlight_map::HighlightMap;
use lazy_static::lazy_static; use lazy_static::lazy_static;
use lsp::LanguageServer; use lsp::LanguageServer;
pub use multi_buffer::MultiBuffer;
use parking_lot::Mutex; use parking_lot::Mutex;
use serde::Deserialize; use serde::Deserialize;
use std::{collections::HashSet, path::Path, str, sync::Arc}; use std::{collections::HashSet, path::Path, str, sync::Arc};

View file

@ -5,20 +5,25 @@ mod selection;
use self::location::*; use self::location::*;
use crate::{ use crate::{
buffer::{self, Buffer, Chunk, ToOffset as _, ToPoint as _}, buffer::{self, Buffer, Chunk, ToOffset as _, ToPoint as _},
BufferSnapshot, BufferSnapshot, Diagnostic, File, Language,
}; };
use anyhow::Result;
use clock::ReplicaId;
use collections::HashMap; use collections::HashMap;
use gpui::{AppContext, Entity, ModelContext, ModelHandle}; use gpui::{AppContext, Entity, ModelContext, ModelHandle, MutableAppContext, Task};
use parking_lot::Mutex; use parking_lot::{Mutex, MutexGuard};
use std::{cmp, ops::Range}; use std::{cmp, io, ops::Range, sync::Arc, time::SystemTime};
use sum_tree::{Bias, Cursor, SumTree}; use sum_tree::{Bias, Cursor, SumTree};
use text::{ use text::{
rope::TextDimension, rope::TextDimension,
subscription::{Subscription, Topic}, subscription::{Subscription, Topic},
AnchorRangeExt, Edit, Point, PointUtf16, TextSummary, AnchorRangeExt as _, Edit, Point, PointUtf16, Selection, SelectionSetId, TextSummary,
}; };
use theme::SyntaxTheme; use theme::SyntaxTheme;
pub use anchor::{Anchor, AnchorRangeExt, AnchorRangeMap, AnchorRangeSet};
pub use selection::SelectionSet;
const NEWLINES: &'static [u8] = &[b'\n'; u8::MAX as usize]; const NEWLINES: &'static [u8] = &[b'\n'; u8::MAX as usize];
#[derive(Default)] #[derive(Default)]
@ -28,11 +33,11 @@ pub struct MultiBuffer {
subscriptions: Topic, subscriptions: Topic,
} }
pub trait ToOffset { pub trait ToOffset: 'static {
fn to_offset<'a>(&self, snapshot: &MultiBufferSnapshot) -> usize; fn to_offset<'a>(&self, snapshot: &MultiBufferSnapshot) -> usize;
} }
pub trait ToPoint { pub trait ToPoint: 'static {
fn to_point<'a>(&self, snapshot: &MultiBufferSnapshot) -> Point; fn to_point<'a>(&self, snapshot: &MultiBufferSnapshot) -> Point;
} }
@ -69,7 +74,7 @@ struct ExcerptSummary {
text: TextSummary, text: TextSummary,
} }
pub struct Chunks<'a> { pub struct MultiBufferChunks<'a> {
range: Range<usize>, range: Range<usize>,
cursor: Cursor<'a, Excerpt, usize>, cursor: Cursor<'a, Excerpt, usize>,
header_height: u8, header_height: u8,
@ -77,20 +82,155 @@ pub struct Chunks<'a> {
theme: Option<&'a SyntaxTheme>, theme: Option<&'a SyntaxTheme>,
} }
pub struct MultiBufferBytes<'a> {
chunks: MultiBufferChunks<'a>,
}
impl MultiBuffer { impl MultiBuffer {
pub fn new() -> Self { pub fn new() -> Self {
Self::default() Self::default()
} }
pub fn singleton(buffer: ModelHandle<Buffer>, cx: &mut ModelContext<Self>) -> Self {
let mut this = Self::new();
this.push(
ExcerptProperties {
buffer: &buffer,
range: text::Anchor::min()..text::Anchor::max(),
header_height: 0,
},
cx,
);
this
}
pub fn build_simple(text: &str, cx: &mut MutableAppContext) -> ModelHandle<Self> {
let buffer = cx.add_model(|cx| Buffer::new(0, text, cx));
cx.add_model(|cx| Self::singleton(buffer, cx))
}
pub fn snapshot(&self, cx: &AppContext) -> MultiBufferSnapshot { pub fn snapshot(&self, cx: &AppContext) -> MultiBufferSnapshot {
self.sync(cx); self.sync(cx);
self.snapshot.lock().clone() self.snapshot.lock().clone()
} }
pub fn as_snapshot(&self) -> MutexGuard<MultiBufferSnapshot> {
self.snapshot.lock()
}
pub fn as_singleton(&self) -> Option<&ModelHandle<Buffer>> {
if self.buffers.len() == 1 {
return Some(&self.buffers.values().next().unwrap().buffer);
} else {
None
}
}
pub fn subscribe(&mut self) -> Subscription { pub fn subscribe(&mut self) -> Subscription {
self.subscriptions.subscribe() self.subscriptions.subscribe()
} }
pub fn edit<I, S, T>(&mut self, ranges_iter: I, new_text: T, cx: &mut ModelContext<Self>)
where
I: IntoIterator<Item = Range<S>>,
S: ToOffset,
T: Into<String>,
{
self.edit_internal(ranges_iter, new_text, false, cx)
}
pub fn edit_with_autoindent<I, S, T>(
&mut self,
ranges_iter: I,
new_text: T,
cx: &mut ModelContext<Self>,
) where
I: IntoIterator<Item = Range<S>>,
S: ToOffset,
T: Into<String>,
{
self.edit_internal(ranges_iter, new_text, true, cx)
}
pub fn edit_internal<I, S, T>(
&mut self,
ranges_iter: I,
new_text: T,
autoindent: bool,
cx: &mut ModelContext<Self>,
) where
I: IntoIterator<Item = Range<S>>,
S: ToOffset,
T: Into<String>,
{
todo!()
}
pub fn start_transaction(
&mut self,
selection_set_ids: impl IntoIterator<Item = SelectionSetId>,
) -> Result<()> {
todo!()
}
pub fn end_transaction(
&mut self,
selection_set_ids: impl IntoIterator<Item = SelectionSetId>,
cx: &mut ModelContext<Self>,
) -> Result<()> {
todo!()
}
pub fn undo(&mut self, cx: &mut ModelContext<Self>) {
todo!()
}
pub fn redo(&mut self, cx: &mut ModelContext<Self>) {
todo!()
}
pub fn selection_set(&self, set_id: SelectionSetId) -> Result<&SelectionSet> {
todo!()
}
pub fn add_selection_set<T: ToOffset>(
&mut self,
selections: &[Selection<T>],
cx: &mut ModelContext<Self>,
) -> SelectionSetId {
todo!()
}
pub fn remove_selection_set(
&mut self,
set_id: SelectionSetId,
cx: &mut ModelContext<Self>,
) -> Result<()> {
todo!()
}
pub fn update_selection_set<T: ToOffset>(
&mut self,
set_id: SelectionSetId,
selections: &[Selection<T>],
cx: &mut ModelContext<Self>,
) -> Result<()> {
todo!()
}
pub fn set_active_selection_set(
&mut self,
set_id: Option<SelectionSetId>,
cx: &mut ModelContext<Self>,
) -> Result<()> {
todo!()
}
pub fn selection_sets(&self) -> impl Iterator<Item = (&SelectionSetId, &SelectionSet)> {
todo!();
None.into_iter()
}
pub fn push<O>(&mut self, props: ExcerptProperties<O>, cx: &mut ModelContext<Self>) -> ExcerptId pub fn push<O>(&mut self, props: ExcerptProperties<O>, cx: &mut ModelContext<Self>) -> ExcerptId
where where
O: text::ToOffset, O: text::ToOffset,
@ -125,6 +265,30 @@ impl MultiBuffer {
id id
} }
pub fn save(
&mut self,
cx: &mut ModelContext<Self>,
) -> Result<Task<Result<(clock::Global, SystemTime)>>> {
todo!()
}
pub fn file<'a>(&self, cx: &'a AppContext) -> Option<&'a dyn File> {
self.as_singleton()
.and_then(|buffer| buffer.read(cx).file())
}
pub fn is_dirty(&self) -> bool {
todo!()
}
pub fn has_conflict(&self) -> bool {
todo!()
}
pub fn is_parsing(&self, _: &AppContext) -> bool {
todo!()
}
fn sync(&self, cx: &AppContext) { fn sync(&self, cx: &AppContext) {
let mut snapshot = self.snapshot.lock(); let mut snapshot = self.snapshot.lock();
let mut excerpts_to_edit = Vec::new(); let mut excerpts_to_edit = Vec::new();
@ -194,17 +358,141 @@ impl MultiBuffer {
} }
} }
// Methods delegating to the snapshot
impl MultiBuffer {
pub fn replica_id(&self) -> ReplicaId {
self.snapshot.lock().replica_id()
}
pub fn text(&self) -> String {
self.snapshot.lock().text()
}
pub fn text_for_range<'a, T: ToOffset>(
&'a self,
range: Range<T>,
) -> impl Iterator<Item = &'a str> {
todo!();
[].into_iter()
}
pub fn max_point(&self) -> Point {
self.snapshot.lock().max_point()
}
pub fn len(&self) -> usize {
self.snapshot.lock().len()
}
pub fn line_len(&self, row: u32) -> u32 {
self.snapshot.lock().line_len(row)
}
pub fn is_line_blank(&self, row: u32) -> bool {
self.snapshot.lock().is_line_blank(row)
}
pub fn indent_column_for_line(&self, row: u32) -> u32 {
self.snapshot.lock().indent_column_for_line(row)
}
pub fn anchor_before<T: ToOffset>(&self, position: T) -> Anchor {
self.snapshot.lock().anchor_before(position)
}
pub fn anchor_after<T: ToOffset>(&self, position: T) -> Anchor {
self.snapshot.lock().anchor_after(position)
}
pub fn anchor_at<T: ToOffset>(&self, position: T, bias: Bias) -> Anchor {
self.snapshot.lock().anchor_at(position, bias)
}
pub fn anchor_range_set<E>(
&self,
start_bias: Bias,
end_bias: Bias,
entries: E,
) -> AnchorRangeSet
where
E: IntoIterator<Item = Range<usize>>,
{
todo!()
}
pub fn clip_offset(&self, offset: usize, bias: Bias) -> usize {
self.snapshot.lock().clip_offset(offset, bias)
}
pub fn clip_point(&self, point: Point, bias: Bias) -> Point {
self.snapshot.lock().clip_point(point, bias)
}
pub fn language<'a>(&self) -> Option<&'a Arc<Language>> {
todo!()
}
pub fn parse_count(&self) -> usize {
self.snapshot.lock().parse_count()
}
pub fn diagnostics_update_count(&self) -> usize {
self.snapshot.lock().diagnostics_update_count()
}
pub fn diagnostics_in_range<'a, T, O>(
&'a self,
search_range: Range<T>,
) -> impl Iterator<Item = (Range<O>, &Diagnostic)> + 'a
where
T: 'a + ToOffset,
O: 'a,
{
todo!();
None.into_iter()
}
}
#[cfg(any(test, feature = "test-support"))]
impl MultiBuffer {
pub fn randomly_edit<R: rand::Rng>(&mut self, _: &mut R, _: usize, _: &mut ModelContext<Self>) {
todo!()
}
pub fn randomly_mutate<R: rand::Rng>(&mut self, rng: &mut R, cx: &mut ModelContext<Self>) {
todo!()
}
}
impl Entity for MultiBuffer { impl Entity for MultiBuffer {
type Event = (); type Event = super::Event;
} }
impl MultiBufferSnapshot { impl MultiBufferSnapshot {
pub fn replica_id(&self) -> ReplicaId {
todo!()
}
pub fn text(&self) -> String { pub fn text(&self) -> String {
self.chunks(0..self.len(), None) self.chunks(0..self.len(), None)
.map(|chunk| chunk.text) .map(|chunk| chunk.text)
.collect() .collect()
} }
pub fn reversed_chars_at<'a, T: ToOffset>(
&'a self,
position: T,
) -> impl Iterator<Item = char> + 'a {
todo!();
None.into_iter()
}
pub fn chars_at<'a, T: ToOffset>(&'a self, position: T) -> impl Iterator<Item = char> + 'a {
let offset = position.to_offset(self);
self.text_for_range(offset..self.len())
.flat_map(|chunk| chunk.chars())
}
pub fn text_for_range<'a, T: ToOffset>( pub fn text_for_range<'a, T: ToOffset>(
&'a self, &'a self,
range: Range<T>, range: Range<T>,
@ -212,6 +500,18 @@ impl MultiBufferSnapshot {
self.chunks(range, None).map(|chunk| chunk.text) self.chunks(range, None).map(|chunk| chunk.text)
} }
pub fn is_line_blank(&self, row: u32) -> bool {
self.text_for_range(Point::new(row, 0)..Point::new(row, self.line_len(row)))
.all(|chunk| chunk.matches(|c: char| !c.is_whitespace()).next().is_none())
}
pub fn contains_str_at<T>(&self, _: T, _: &str) -> bool
where
T: ToOffset,
{
todo!()
}
pub fn len(&self) -> usize { pub fn len(&self) -> usize {
self.excerpts.summary().text.bytes self.excerpts.summary().text.bytes
} }
@ -291,11 +591,15 @@ impl MultiBufferSnapshot {
} }
} }
pub fn bytes_in_range<'a, T: ToOffset>(&'a self, range: Range<T>) -> MultiBufferBytes<'a> {
todo!()
}
pub fn chunks<'a, T: ToOffset>( pub fn chunks<'a, T: ToOffset>(
&'a self, &'a self,
range: Range<T>, range: Range<T>,
theme: Option<&'a SyntaxTheme>, theme: Option<&'a SyntaxTheme>,
) -> Chunks<'a> { ) -> MultiBufferChunks<'a> {
let range = range.start.to_offset(self)..range.end.to_offset(self); let range = range.start.to_offset(self)..range.end.to_offset(self);
let mut cursor = self.excerpts.cursor::<usize>(); let mut cursor = self.excerpts.cursor::<usize>();
cursor.seek(&range.start, Bias::Right, &()); cursor.seek(&range.start, Bias::Right, &());
@ -331,7 +635,7 @@ impl MultiBufferSnapshot {
excerpt.buffer.chunks(buffer_start..buffer_end, theme) excerpt.buffer.chunks(buffer_start..buffer_end, theme)
}); });
Chunks { MultiBufferChunks {
range, range,
cursor, cursor,
header_height, header_height,
@ -413,6 +717,10 @@ impl MultiBufferSnapshot {
} }
} }
pub fn indent_column_for_line(&self, row: u32) -> u32 {
todo!()
}
pub fn line_len(&self, row: u32) -> u32 { pub fn line_len(&self, row: u32) -> u32 {
let mut cursor = self.excerpts.cursor::<Point>(); let mut cursor = self.excerpts.cursor::<Point>();
cursor.seek(&Point::new(row, 0), Bias::Right, &()); cursor.seek(&Point::new(row, 0), Bias::Right, &());
@ -534,18 +842,62 @@ impl MultiBufferSnapshot {
summary summary
} }
fn resolve_excerpt<'a, D: TextDimension>( pub fn anchor_before<T: ToOffset>(&self, position: T) -> Anchor {
self.anchor_at(position, Bias::Left)
}
pub fn anchor_after<T: ToOffset>(&self, position: T) -> Anchor {
self.anchor_at(position, Bias::Right)
}
pub fn anchor_at<T: ToOffset>(&self, position: T, bias: Bias) -> Anchor {
todo!()
}
pub fn parse_count(&self) -> usize {
todo!()
}
pub fn enclosing_bracket_ranges<T: ToOffset>(
&self,
range: Range<T>,
) -> Option<(Range<usize>, Range<usize>)> {
todo!()
}
pub fn diagnostics_update_count(&self) -> usize {
todo!()
}
pub fn language<'a>(&self) -> Option<&'a Arc<Language>> {
todo!()
}
pub fn diagnostic_group<'a, O>(
&'a self, &'a self,
excerpt_id: &ExcerptId, group_id: usize,
) -> Option<(D, &'a BufferSnapshot)> { ) -> impl Iterator<Item = (Range<O>, &Diagnostic)> + 'a
let mut cursor = self.excerpts.cursor::<(ExcerptId, TextSummary)>(); where
cursor.seek(excerpt_id, Bias::Left, &()); O: 'a,
if let Some(excerpt) = cursor.item() { {
if cursor.start().0 == *excerpt_id { todo!();
return Some((D::from_text_summary(&cursor.start().1), &excerpt.buffer)); None.into_iter()
} }
}
None pub fn diagnostics_in_range<'a, T, O>(
&'a self,
search_range: Range<T>,
) -> impl Iterator<Item = (Range<O>, &Diagnostic)> + 'a
where
T: 'a + ToOffset,
O: 'a,
{
todo!();
None.into_iter()
}
pub fn range_for_syntax_ancestor<T: ToOffset>(&self, range: Range<T>) -> Option<Range<usize>> {
todo!()
} }
fn buffer_snapshot_for_excerpt<'a>( fn buffer_snapshot_for_excerpt<'a>(
@ -672,7 +1024,17 @@ impl<'a> sum_tree::Dimension<'a, ExcerptSummary> for Location {
} }
} }
impl<'a> Iterator for Chunks<'a> { impl<'a> MultiBufferChunks<'a> {
pub fn offset(&self) -> usize {
todo!()
}
pub fn seek(&mut self, offset: usize) {
todo!()
}
}
impl<'a> Iterator for MultiBufferChunks<'a> {
type Item = Chunk<'a>; type Item = Chunk<'a>;
fn next(&mut self) -> Option<Self::Item> { fn next(&mut self) -> Option<Self::Item> {
@ -726,6 +1088,20 @@ impl<'a> Iterator for Chunks<'a> {
} }
} }
impl<'a> Iterator for MultiBufferBytes<'a> {
type Item = &'a [u8];
fn next(&mut self) -> Option<Self::Item> {
todo!()
}
}
impl<'a> io::Read for MultiBufferBytes<'a> {
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
todo!()
}
}
impl ToOffset for Point { impl ToOffset for Point {
fn to_offset<'a>(&self, snapshot: &MultiBufferSnapshot) -> usize { fn to_offset<'a>(&self, snapshot: &MultiBufferSnapshot) -> usize {
snapshot.point_to_offset(*self) snapshot.point_to_offset(*self)

View file

@ -1,9 +1,12 @@
use super::{location::*, ExcerptSummary, MultiBufferSnapshot, ToOffset}; use super::{location::*, ExcerptSummary, MultiBufferSnapshot, ToOffset, ToPoint};
use anyhow::{anyhow, Result}; use anyhow::{anyhow, Result};
use smallvec::SmallVec; use smallvec::SmallVec;
use std::{cmp::Ordering, ops::Range}; use std::{
cmp::Ordering,
ops::{Range, Sub},
};
use sum_tree::Bias; use sum_tree::Bias;
use text::{rope::TextDimension, AnchorRangeExt, ToOffset as _}; use text::{rope::TextDimension, AnchorRangeExt as _, Point};
#[derive(Clone, Eq, PartialEq, Debug, Hash)] #[derive(Clone, Eq, PartialEq, Debug, Hash)]
pub struct Anchor { pub struct Anchor {
@ -16,6 +19,9 @@ pub struct AnchorRangeMap<T> {
entries: SmallVec<[(ExcerptId, text::AnchorRangeMap<T>); 1]>, entries: SmallVec<[(ExcerptId, text::AnchorRangeMap<T>); 1]>,
} }
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct AnchorRangeSet(AnchorRangeMap<()>);
impl Anchor { impl Anchor {
pub fn min() -> Self { pub fn min() -> Self {
Self { Self {
@ -68,6 +74,27 @@ impl Anchor {
} }
self.clone() self.clone()
} }
pub fn summary<'a, D>(&self, snapshot: &'a MultiBufferSnapshot) -> D
where
D: TextDimension + Ord + Sub<D, Output = D>,
{
let mut cursor = snapshot.excerpts.cursor::<ExcerptSummary>();
cursor.seek(&self.excerpt_id, Bias::Left, &());
if let Some(excerpt) = cursor.item() {
if excerpt.id == self.excerpt_id {
let mut excerpt_start = D::from_text_summary(&cursor.start().text);
excerpt_start.add_summary(&excerpt.header_summary(), &());
let excerpt_buffer_start = excerpt.range.start.summary::<D>(&excerpt.buffer);
let buffer_point = self.text_anchor.summary::<D>(&excerpt.buffer);
if buffer_point > excerpt_buffer_start {
excerpt_start.add_assign(&(buffer_point - excerpt_buffer_start));
}
return excerpt_start;
}
}
D::from_text_summary(&cursor.start().text)
}
} }
impl<T> AnchorRangeMap<T> { impl<T> AnchorRangeMap<T> {
@ -263,18 +290,48 @@ impl<T> AnchorRangeMap<T> {
} }
} }
impl ToOffset for Anchor { impl AnchorRangeSet {
fn to_offset<'a>(&self, snapshot: &MultiBufferSnapshot) -> usize { pub fn len(&self) -> usize {
let mut cursor = snapshot.excerpts.cursor::<ExcerptSummary>(); self.0.len()
cursor.seek(&self.excerpt_id, Bias::Left, &()); }
if let Some(excerpt) = cursor.item() {
if excerpt.id == self.excerpt_id { pub fn ranges<'a, D>(
let buffer_offset = self.text_anchor.to_offset(&excerpt.buffer); &'a self,
return cursor.start().text.bytes content: &'a MultiBufferSnapshot,
+ excerpt.header_height as usize ) -> impl 'a + Iterator<Item = Range<Point>>
+ buffer_offset.saturating_sub(excerpt.range.start.to_offset(&excerpt.buffer)); where
} D: TextDimension,
} {
cursor.start().text.bytes self.0.ranges(content).map(|(range, _)| range)
}
}
impl ToOffset for Anchor {
fn to_offset<'a>(&self, snapshot: &MultiBufferSnapshot) -> usize {
self.summary(snapshot)
}
}
impl ToPoint for Anchor {
fn to_point<'a>(&self, snapshot: &MultiBufferSnapshot) -> Point {
self.summary(snapshot)
}
}
pub trait AnchorRangeExt {
fn cmp(&self, b: &Range<Anchor>, buffer: &MultiBufferSnapshot) -> Result<Ordering>;
fn to_offset(&self, content: &MultiBufferSnapshot) -> Range<usize>;
}
impl AnchorRangeExt for Range<Anchor> {
fn cmp(&self, other: &Range<Anchor>, buffer: &MultiBufferSnapshot) -> Result<Ordering> {
Ok(match self.start.cmp(&other.start, buffer)? {
Ordering::Equal => other.end.cmp(&self.end, buffer)?,
ord @ _ => ord,
})
}
fn to_offset(&self, content: &MultiBufferSnapshot) -> Range<usize> {
self.start.to_offset(&content)..self.end.to_offset(&content)
} }
} }

View file

@ -376,7 +376,7 @@ fn test_autoindent_moves_selections(cx: &mut MutableAppContext) {
.selection_set(selection_set_id) .selection_set(selection_set_id)
.unwrap() .unwrap()
.selections::<Point>(&buffer) .selections::<Point>(&buffer)
.map(|selection| selection.point_range(&buffer)) .map(|selection| selection.start.to_point(&buffer)..selection.end.to_point(&buffer))
.collect::<Vec<_>>(); .collect::<Vec<_>>();
assert_eq!(selection_ranges[0], empty(Point::new(1, 4))); assert_eq!(selection_ranges[0], empty(Point::new(1, 4)));

View file

@ -948,7 +948,7 @@ mod tests {
fs::{FakeFs, Fs as _}, fs::{FakeFs, Fs as _},
language::{ language::{
tree_sitter_rust, Diagnostic, Language, LanguageConfig, LanguageRegistry, tree_sitter_rust, Diagnostic, Language, LanguageConfig, LanguageRegistry,
LanguageServerConfig, Point, LanguageServerConfig, MultiBuffer, Point,
}, },
lsp, lsp,
project::{ProjectPath, Worktree}, project::{ProjectPath, Worktree},
@ -1035,6 +1035,7 @@ mod tests {
.update(&mut cx_b, |worktree, cx| worktree.open_buffer("b.txt", cx)) .update(&mut cx_b, |worktree, cx| worktree.open_buffer("b.txt", cx))
.await .await
.unwrap(); .unwrap();
let buffer_b = cx_b.add_model(|cx| MultiBuffer::singleton(buffer_b, cx));
buffer_b.read_with(&cx_b, |buf, _| assert_eq!(buf.text(), "b-contents")); buffer_b.read_with(&cx_b, |buf, _| assert_eq!(buf.text(), "b-contents"));
worktree_a.read_with(&cx_a, |tree, cx| assert!(tree.has_open_buffer("b.txt", cx))); worktree_a.read_with(&cx_a, |tree, cx| assert!(tree.has_open_buffer("b.txt", cx)));
let buffer_a = worktree_a let buffer_a = worktree_a

View file

@ -201,6 +201,15 @@ where
} }
} }
impl<T: Clone> IntoIterator for Patch<T> {
type Item = Edit<T>;
type IntoIter = std::vec::IntoIter<Edit<T>>;
fn into_iter(self) -> Self::IntoIter {
self.0.into_iter()
}
}
impl<'a, T: Clone> IntoIterator for &'a Patch<T> { impl<'a, T: Clone> IntoIterator for &'a Patch<T> {
type Item = Edit<T>; type Item = Edit<T>;
type IntoIter = std::iter::Cloned<std::slice::Iter<'a, Edit<T>>>; type IntoIter = std::iter::Cloned<std::slice::Iter<'a, Edit<T>>>;

View file

@ -1,6 +1,4 @@
use crate::{ use crate::{rope::TextDimension, AnchorRangeMap, BufferSnapshot, ToOffset, ToPoint};
rope::TextDimension, AnchorRangeMap, Buffer, BufferSnapshot, Point, ToOffset, ToPoint,
};
use std::{cmp::Ordering, ops::Range, sync::Arc}; use std::{cmp::Ordering, ops::Range, sync::Arc};
use sum_tree::Bias; use sum_tree::Bias;
@ -75,26 +73,6 @@ impl<T: ToOffset + ToPoint + Copy + Ord> Selection<T> {
self.end = head; self.end = head;
} }
} }
pub fn point_range(&self, buffer: &Buffer) -> Range<Point> {
let start = self.start.to_point(buffer);
let end = self.end.to_point(buffer);
if self.reversed {
end..start
} else {
start..end
}
}
pub fn offset_range(&self, buffer: &Buffer) -> Range<usize> {
let start = self.start.to_offset(buffer);
let end = self.end.to_offset(buffer);
if self.reversed {
end..start
} else {
start..end
}
}
} }
impl SelectionSet { impl SelectionSet {

View file

@ -1850,13 +1850,13 @@ impl BufferSnapshot {
self.visible_text.clip_point_utf16(point, bias) self.visible_text.clip_point_utf16(point, bias)
} }
pub fn point_for_offset(&self, offset: usize) -> Result<Point> { // pub fn point_for_offset(&self, offset: usize) -> Result<Point> {
if offset <= self.len() { // if offset <= self.len() {
Ok(self.text_summary_for_range(0..offset)) // Ok(self.text_summary_for_range(0..offset))
} else { // } else {
Err(anyhow!("offset out of bounds")) // Err(anyhow!("offset out of bounds"))
} // }
} // }
pub fn edits_since<'a, D>( pub fn edits_since<'a, D>(
&'a self, &'a self,