WIP
This commit is contained in:
parent
118f137f18
commit
7dcf30c954
1 changed files with 136 additions and 30 deletions
|
@ -25,7 +25,7 @@ pub struct InjectionMap {
|
||||||
next_injection_id: usize,
|
next_injection_id: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct InjectionSnapshot {
|
pub struct Snapshot {
|
||||||
transforms: SumTree<Transform>,
|
transforms: SumTree<Transform>,
|
||||||
injections: SumTree<Injection>,
|
injections: SumTree<Injection>,
|
||||||
buffer_snapshot: language::Snapshot,
|
buffer_snapshot: language::Snapshot,
|
||||||
|
@ -103,7 +103,7 @@ struct TransformSummary {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, Default)]
|
#[derive(Copy, Clone, Debug, Default)]
|
||||||
struct InjectionOffset(usize);
|
pub struct InjectionOffset(usize);
|
||||||
|
|
||||||
impl sum_tree::Summary for InjectionId {
|
impl sum_tree::Summary for InjectionId {
|
||||||
type Context = ();
|
type Context = ();
|
||||||
|
@ -114,10 +114,32 @@ impl sum_tree::Summary for InjectionId {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl InjectionMap {
|
impl InjectionMap {
|
||||||
pub fn read(&self, cx: &AppContext) -> (InjectionSnapshot, Vec<Edit<InjectionOffset>>) {
|
pub fn new(buffer_handle: ModelHandle<Buffer>, cx: &AppContext) -> (Self, Snapshot) {
|
||||||
|
let buffer = buffer_handle.read(cx);
|
||||||
|
let this = Self {
|
||||||
|
buffer: buffer_handle,
|
||||||
|
injections: Default::default(),
|
||||||
|
injection_sites: Default::default(),
|
||||||
|
transforms: Mutex::new(SumTree::from_item(
|
||||||
|
Transform::isomorphic(buffer.text_summary()),
|
||||||
|
&(),
|
||||||
|
)),
|
||||||
|
last_sync: Mutex::new(SyncState {
|
||||||
|
version: buffer.version(),
|
||||||
|
parse_count: buffer.parse_count(),
|
||||||
|
diagnostics_update_count: buffer.diagnostics_update_count(),
|
||||||
|
}),
|
||||||
|
version: AtomicUsize::new(0),
|
||||||
|
next_injection_id: 0,
|
||||||
|
};
|
||||||
|
let (snapshot, _) = this.read(cx);
|
||||||
|
(this, snapshot)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn read(&self, cx: &AppContext) -> (Snapshot, Vec<Edit<InjectionOffset>>) {
|
||||||
let edits = self.sync(cx);
|
let edits = self.sync(cx);
|
||||||
// self.check_invariants(cx);
|
// self.check_invariants(cx);
|
||||||
let snapshot = InjectionSnapshot {
|
let snapshot = Snapshot {
|
||||||
transforms: self.transforms.lock().clone(),
|
transforms: self.transforms.lock().clone(),
|
||||||
injections: self.injections.clone(),
|
injections: self.injections.clone(),
|
||||||
buffer_snapshot: self.buffer.read(cx).snapshot(),
|
buffer_snapshot: self.buffer.read(cx).snapshot(),
|
||||||
|
@ -129,11 +151,7 @@ impl InjectionMap {
|
||||||
pub fn write(
|
pub fn write(
|
||||||
&mut self,
|
&mut self,
|
||||||
cx: &AppContext,
|
cx: &AppContext,
|
||||||
) -> (
|
) -> (InjectionMapWriter, Snapshot, Vec<Edit<InjectionOffset>>) {
|
||||||
InjectionMapWriter,
|
|
||||||
InjectionSnapshot,
|
|
||||||
Vec<Edit<InjectionOffset>>,
|
|
||||||
) {
|
|
||||||
let (snapshot, edits) = self.read(cx);
|
let (snapshot, edits) = self.read(cx);
|
||||||
(InjectionMapWriter(self), snapshot, edits)
|
(InjectionMapWriter(self), snapshot, edits)
|
||||||
}
|
}
|
||||||
|
@ -174,31 +192,47 @@ impl InjectionMap {
|
||||||
|
|
||||||
let mut new_transforms = SumTree::<Transform>::new();
|
let mut new_transforms = SumTree::<Transform>::new();
|
||||||
let mut transforms = self.transforms.lock();
|
let mut transforms = self.transforms.lock();
|
||||||
|
let old_max_point = transforms.summary().input.lines;
|
||||||
|
let new_max_point = buffer.max_point();
|
||||||
let mut cursor = transforms.cursor::<Point>();
|
let mut cursor = transforms.cursor::<Point>();
|
||||||
let mut injection_sites = self.injection_sites.cursor::<InjectionSitePosition>();
|
let mut injection_sites = self.injection_sites.cursor::<InjectionSitePosition>();
|
||||||
let mut pending_after_injections: Vec<InjectionId> = Vec::new();
|
let mut pending_after_injections: Vec<InjectionId> = Vec::new();
|
||||||
|
|
||||||
while let Some(mut edit) = buffer_edits_iter.next() {
|
while let Some(mut edit) = buffer_edits_iter.next() {
|
||||||
// Expand this edit to line boundaries
|
dbg!(&edit);
|
||||||
|
// Expand this edit to line boundaries.
|
||||||
edit.old.start.column = 0;
|
edit.old.start.column = 0;
|
||||||
edit.old.end += Point::new(1, 0);
|
edit.old.end += Point::new(1, 0);
|
||||||
edit.new.start.column = 0;
|
edit.new.start.column = 0;
|
||||||
edit.new.end += Point::new(1, 0);
|
edit.new.end += Point::new(1, 0);
|
||||||
|
|
||||||
// Merge with subsequent edits that intersect the same lines
|
// Push any transforms preceding the edit.
|
||||||
while let Some(next_edit) = buffer_edits_iter.peek() {
|
new_transforms.push_tree(cursor.slice(&edit.old.start, Bias::Left, &()), &());
|
||||||
if next_edit.old.start.row > edit.old.end.row {
|
|
||||||
break;
|
// Snap edits to row boundaries of intersecting transforms.
|
||||||
|
loop {
|
||||||
|
if cmp::min(edit.old.end, old_max_point) <= cursor.end(&()) {
|
||||||
|
cursor.seek(&edit.old.end, Bias::Left, &());
|
||||||
|
cursor.next(&());
|
||||||
|
let new_old_end = *cursor.start() + Point::new(1, 0);
|
||||||
|
edit.new.end += new_old_end - edit.old.end;
|
||||||
|
edit.old.end = new_old_end;
|
||||||
}
|
}
|
||||||
|
|
||||||
let next_edit = buffer_edits_iter.next().unwrap();
|
if buffer_edits_iter.peek().map_or(false, |next_edit| {
|
||||||
edit.old.end.row = next_edit.old.end.row + 1;
|
edit.old.end.row >= next_edit.old.start.row
|
||||||
let row_delta = next_edit.new.end.row as i32 - next_edit.old.end.row as i32;
|
}) {
|
||||||
edit.new.end.row = (edit.new.end.row as i32 + row_delta) as u32;
|
let next_edit = buffer_edits_iter.next().unwrap();
|
||||||
|
edit.old.end = cmp::max(edit.old.end, next_edit.old.end + Point::new(1, 0));
|
||||||
|
let row_delta = (next_edit.new.end.row as i32 - next_edit.new.start.row as i32)
|
||||||
|
- (next_edit.old.end.row as i32 - next_edit.old.start.row as i32);
|
||||||
|
edit.new.end.row = (edit.new.end.row as i32 + row_delta) as u32;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Push any transforms preceding the edit
|
dbg!(&edit);
|
||||||
new_transforms.push_tree(cursor.slice(&edit.old.start, Bias::Left, &()), &());
|
|
||||||
|
|
||||||
// Find and insert all injections on the lines spanned by the edit, interleaved with isomorphic regions
|
// Find and insert all injections on the lines spanned by the edit, interleaved with isomorphic regions
|
||||||
injection_sites.seek(
|
injection_sites.seek(
|
||||||
|
@ -264,13 +298,38 @@ impl InjectionMap {
|
||||||
last_injection_row = Some(injection_row);
|
last_injection_row = Some(injection_row);
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(last_injection_row) = injection_row {}
|
if let Some(last_injection_row) = last_injection_row {
|
||||||
|
let injection_point = Point::new(last_injection_row + 1, 0);
|
||||||
|
if injection_point > new_transforms.summary().input.lines {
|
||||||
|
let injection_offset = injection_point.to_offset(buffer);
|
||||||
|
new_transforms.push(
|
||||||
|
Transform::isomorphic(buffer.text_summary_for_range(
|
||||||
|
new_transforms.summary().input.bytes..injection_offset,
|
||||||
|
)),
|
||||||
|
&(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
for injection_id in pending_after_injections.drain(..) {
|
||||||
|
new_transforms.push(
|
||||||
|
Transform::for_injection(self.injections.get(&injection_id, &()).unwrap()),
|
||||||
|
&(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let sum = new_transforms.summary();
|
||||||
|
let new_end = cmp::min(edit.new.end, new_max_point);
|
||||||
|
if sum.input.lines < new_end {
|
||||||
|
let text_summary =
|
||||||
|
buffer.text_summary_for_range(sum.input.bytes..new_end.to_offset(buffer));
|
||||||
|
new_transforms.push(Transform::isomorphic(text_summary), &());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
new_transforms.push_tree(cursor.suffix(&()), &());
|
new_transforms.push_tree(cursor.suffix(&()), &());
|
||||||
drop(cursor);
|
drop(cursor);
|
||||||
|
|
||||||
*transforms = new_transforms;
|
*transforms = new_transforms;
|
||||||
todo!()
|
Vec::new()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -279,11 +338,7 @@ impl<'a> InjectionMapWriter<'a> {
|
||||||
&mut self,
|
&mut self,
|
||||||
injections: T,
|
injections: T,
|
||||||
cx: &AppContext,
|
cx: &AppContext,
|
||||||
) -> (
|
) -> (Vec<InjectionId>, Snapshot, Vec<Edit<InjectionOffset>>)
|
||||||
Vec<InjectionId>,
|
|
||||||
InjectionSnapshot,
|
|
||||||
Vec<Edit<InjectionOffset>>,
|
|
||||||
)
|
|
||||||
where
|
where
|
||||||
T: IntoIterator<Item = (Anchor, InjectionProps)>,
|
T: IntoIterator<Item = (Anchor, InjectionProps)>,
|
||||||
{
|
{
|
||||||
|
@ -314,6 +369,7 @@ impl<'a> InjectionMapWriter<'a> {
|
||||||
InjectionSite {
|
InjectionSite {
|
||||||
injection_id: id,
|
injection_id: id,
|
||||||
position,
|
position,
|
||||||
|
disposition: props.disposition,
|
||||||
},
|
},
|
||||||
buffer,
|
buffer,
|
||||||
);
|
);
|
||||||
|
@ -332,7 +388,7 @@ impl<'a> InjectionMapWriter<'a> {
|
||||||
self.0.injection_sites = new_sites;
|
self.0.injection_sites = new_sites;
|
||||||
|
|
||||||
let edits = self.0.apply_edits(edits, cx);
|
let edits = self.0.apply_edits(edits, cx);
|
||||||
let snapshot = InjectionSnapshot {
|
let snapshot = Snapshot {
|
||||||
transforms: self.0.transforms.lock().clone(),
|
transforms: self.0.transforms.lock().clone(),
|
||||||
injections: self.0.injections.clone(),
|
injections: self.0.injections.clone(),
|
||||||
buffer_snapshot: buffer.snapshot(),
|
buffer_snapshot: buffer.snapshot(),
|
||||||
|
@ -384,13 +440,13 @@ impl sum_tree::Summary for InjectionSiteSummary {
|
||||||
fn add_summary(&mut self, summary: &Self, _: &Self::Context) {
|
fn add_summary(&mut self, summary: &Self, _: &Self::Context) {
|
||||||
self.min_injection_id = cmp::min(self.min_injection_id, summary.min_injection_id);
|
self.min_injection_id = cmp::min(self.min_injection_id, summary.min_injection_id);
|
||||||
self.max_injection_id = cmp::max(self.max_injection_id, summary.max_injection_id);
|
self.max_injection_id = cmp::max(self.max_injection_id, summary.max_injection_id);
|
||||||
self.max_position = summary.max_position;
|
self.max_position = summary.max_position.clone();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> sum_tree::Dimension<'a, InjectionSiteSummary> for InjectionSitePosition {
|
impl<'a> sum_tree::Dimension<'a, InjectionSiteSummary> for InjectionSitePosition {
|
||||||
fn add_summary(&mut self, summary: &'a InjectionSiteSummary, _: &buffer::Buffer) {
|
fn add_summary(&mut self, summary: &'a InjectionSiteSummary, _: &buffer::Buffer) {
|
||||||
self.0 = summary.max_position;
|
self.0 = summary.max_position.clone();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -478,3 +534,53 @@ impl<'a> sum_tree::Dimension<'a, TransformSummary> for InjectionOffset {
|
||||||
self.0 += summary.output.bytes
|
self.0 += summary.output.bytes
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use std::env;
|
||||||
|
|
||||||
|
use super::*;
|
||||||
|
use buffer::RandomCharIter;
|
||||||
|
use rand::prelude::*;
|
||||||
|
|
||||||
|
#[gpui::test(iterations = 1000)]
|
||||||
|
fn test_random(cx: &mut gpui::MutableAppContext, mut rng: StdRng) {
|
||||||
|
let operations = env::var("OPERATIONS")
|
||||||
|
.map(|i| i.parse().expect("invalid `OPERATIONS` variable"))
|
||||||
|
.unwrap_or(1);
|
||||||
|
|
||||||
|
let buffer = cx.add_model(|cx| {
|
||||||
|
let len = rng.gen_range(0..10);
|
||||||
|
let text = RandomCharIter::new(&mut rng).take(len).collect::<String>();
|
||||||
|
Buffer::new(0, text, cx)
|
||||||
|
});
|
||||||
|
let (map, initial_snapshot) = InjectionMap::new(buffer.clone(), cx.as_ref());
|
||||||
|
assert_eq!(
|
||||||
|
initial_snapshot.transforms.summary().input,
|
||||||
|
buffer.read(cx).text_summary()
|
||||||
|
);
|
||||||
|
|
||||||
|
for _ in 0..operations {
|
||||||
|
log::info!("text: {:?}", buffer.read(cx).text());
|
||||||
|
match rng.gen_range(0..=100) {
|
||||||
|
_ => {
|
||||||
|
let edits = buffer.update(cx, |buffer, _| {
|
||||||
|
let start_version = buffer.version.clone();
|
||||||
|
let edit_count = rng.gen_range(1..=5);
|
||||||
|
buffer.randomly_edit(&mut rng, edit_count);
|
||||||
|
buffer
|
||||||
|
.edits_since::<Point>(&start_version)
|
||||||
|
.collect::<Vec<_>>()
|
||||||
|
});
|
||||||
|
log::info!("editing {:?}", edits);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let (snapshot, edits) = map.read(cx.as_ref());
|
||||||
|
assert_eq!(
|
||||||
|
snapshot.transforms.summary().input,
|
||||||
|
buffer.read(cx).text_summary()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue