Avoid composing edits together for now

This commit is contained in:
Antonio Scandurra 2021-11-12 08:55:17 +01:00
parent 068aa1adb3
commit 6e882bcd02

View file

@ -14,16 +14,13 @@ pub type Edit = buffer::Edit<u32>;
pub struct WrapMap { pub struct WrapMap {
snapshot: Snapshot, snapshot: Snapshot,
pending_edits: VecDeque<(TabSnapshot, Vec<TabEdit>)>, pending_edits: VecDeque<(TabSnapshot, Vec<TabEdit>)>,
interpolated_edits: Patch, interpolated_edits: Vec<Vec<Edit>>,
edits_since_sync: Patch, edits_since_sync: Vec<Vec<Edit>>,
wrap_width: Option<f32>, wrap_width: Option<f32>,
background_task: Option<Task<()>>, background_task: Option<Task<()>>,
font: (FontId, f32), font: (FontId, f32),
} }
#[derive(Default)]
struct Patch(Vec<Edit>);
impl Entity for WrapMap { impl Entity for WrapMap {
type Event = (); type Event = ();
} }
@ -106,13 +103,10 @@ impl WrapMap {
tab_snapshot: TabSnapshot, tab_snapshot: TabSnapshot,
edits: Vec<TabEdit>, edits: Vec<TabEdit>,
cx: &mut ModelContext<Self>, cx: &mut ModelContext<Self>,
) -> (Snapshot, Vec<Edit>) { ) -> (Snapshot, Vec<Vec<Edit>>) {
self.pending_edits.push_back((tab_snapshot, edits)); self.pending_edits.push_back((tab_snapshot, edits));
self.flush_edits(cx); self.flush_edits(cx);
( (self.snapshot.clone(), mem::take(&mut self.edits_since_sync))
self.snapshot.clone(),
mem::take(&mut self.edits_since_sync).0,
)
} }
pub fn set_font(&mut self, font_id: FontId, font_size: f32, cx: &mut ModelContext<Self>) { pub fn set_font(&mut self, font_id: FontId, font_size: f32, cx: &mut ModelContext<Self>) {
@ -143,7 +137,7 @@ impl WrapMap {
let mut line_wrapper = font_cache.line_wrapper(font_id, font_size); let mut line_wrapper = font_cache.line_wrapper(font_id, font_size);
let tab_snapshot = new_snapshot.tab_snapshot.clone(); let tab_snapshot = new_snapshot.tab_snapshot.clone();
let range = TabPoint::zero()..tab_snapshot.max_point(); let range = TabPoint::zero()..tab_snapshot.max_point();
new_snapshot let edits = new_snapshot
.update( .update(
tab_snapshot, tab_snapshot,
&[TabEdit { &[TabEdit {
@ -154,22 +148,28 @@ impl WrapMap {
&mut line_wrapper, &mut line_wrapper,
) )
.await; .await;
new_snapshot (new_snapshot, edits)
}); });
match cx match cx
.background() .background()
.block_with_timeout(Duration::from_millis(5), task) .block_with_timeout(Duration::from_millis(5), task)
{ {
Ok(snapshot) => { Ok((snapshot, edits)) => {
self.snapshot = snapshot; self.snapshot = snapshot;
self.edits_since_sync.push(edits);
cx.notify(); cx.notify();
} }
Err(wrap_task) => { Err(wrap_task) => {
self.background_task = Some(cx.spawn(|this, mut cx| async move { self.background_task = Some(cx.spawn(|this, mut cx| async move {
let snapshot = wrap_task.await; let (snapshot, edits) = wrap_task.await;
this.update(&mut cx, |this, cx| { this.update(&mut cx, |this, cx| {
this.snapshot = snapshot; this.snapshot = snapshot;
for mut edits in this.interpolated_edits.drain(..) {
invert(&mut edits);
this.edits_since_sync.push(edits);
}
this.edits_since_sync.push(edits);
this.background_task = None; this.background_task = None;
this.flush_edits(cx); this.flush_edits(cx);
cx.notify(); cx.notify();
@ -178,6 +178,7 @@ impl WrapMap {
} }
} }
} else { } else {
let old_rows = self.snapshot.transforms.summary().output.lines.row + 1;
self.snapshot.transforms = SumTree::new(); self.snapshot.transforms = SumTree::new();
let summary = self.snapshot.tab_snapshot.text_summary(); let summary = self.snapshot.tab_snapshot.text_summary();
if !summary.lines.is_zero() { if !summary.lines.is_zero() {
@ -185,6 +186,14 @@ impl WrapMap {
.transforms .transforms
.push(Transform::isomorphic(summary), &()); .push(Transform::isomorphic(summary), &());
} }
let new_rows = self.snapshot.transforms.summary().output.lines.row + 1;
self.snapshot.interpolated = false;
self.pending_edits.clear();
self.interpolated_edits.clear();
self.edits_since_sync.push(vec![Edit {
old: 0..old_rows,
new: 0..new_rows,
}]);
} }
} }
@ -214,12 +223,12 @@ impl WrapMap {
let update_task = cx.background().spawn(async move { let update_task = cx.background().spawn(async move {
let mut line_wrapper = font_cache.line_wrapper(font_id, font_size); let mut line_wrapper = font_cache.line_wrapper(font_id, font_size);
let mut output_edits = Patch::default(); let mut output_edits = Vec::new();
for (tab_snapshot, edits) in pending_edits { for (tab_snapshot, edits) in pending_edits {
let wrap_edits = snapshot let wrap_edits = snapshot
.update(tab_snapshot, &edits, wrap_width, &mut line_wrapper) .update(tab_snapshot, &edits, wrap_width, &mut line_wrapper)
.await; .await;
output_edits.compose(&wrap_edits); output_edits.push(wrap_edits);
} }
(snapshot, output_edits) (snapshot, output_edits)
}); });
@ -230,16 +239,18 @@ impl WrapMap {
{ {
Ok((snapshot, output_edits)) => { Ok((snapshot, output_edits)) => {
self.snapshot = snapshot; self.snapshot = snapshot;
self.edits_since_sync.compose(&output_edits); self.edits_since_sync.extend(output_edits);
} }
Err(update_task) => { Err(update_task) => {
self.background_task = Some(cx.spawn(|this, mut cx| async move { self.background_task = Some(cx.spawn(|this, mut cx| async move {
let (snapshot, output_edits) = update_task.await; let (snapshot, output_edits) = update_task.await;
this.update(&mut cx, |this, cx| { this.update(&mut cx, |this, cx| {
this.snapshot = snapshot; this.snapshot = snapshot;
this.edits_since_sync for mut edits in this.interpolated_edits.drain(..) {
.compose(mem::take(&mut this.interpolated_edits).invert()) invert(&mut edits);
.compose(&output_edits); this.edits_since_sync.push(edits);
}
this.edits_since_sync.extend(output_edits);
this.background_task = None; this.background_task = None;
this.flush_edits(cx); this.flush_edits(cx);
cx.notify(); cx.notify();
@ -257,8 +268,8 @@ impl WrapMap {
to_remove_len += 1; to_remove_len += 1;
} else { } else {
let interpolated_edits = self.snapshot.interpolate(tab_snapshot.clone(), &edits); let interpolated_edits = self.snapshot.interpolate(tab_snapshot.clone(), &edits);
self.edits_since_sync.compose(&interpolated_edits); self.edits_since_sync.push(interpolated_edits.clone());
self.interpolated_edits.compose(&interpolated_edits); self.interpolated_edits.push(interpolated_edits);
} }
} }
@ -268,156 +279,6 @@ impl WrapMap {
} }
} }
impl Patch {
fn compose(&mut self, new: &Self) -> &mut Self {
let mut new_edits = new.0.iter().peekable();
let mut old_delta = 0;
let mut new_delta = 0;
let mut ix = 0;
'outer: while ix < self.0.len() {
let old_edit = &mut self.0[ix];
old_edit.new.start = (old_edit.new.start as i32 + new_delta) as u32;
old_edit.new.end = (old_edit.new.end as i32 + new_delta) as u32;
while let Some(new_edit) = new_edits.peek() {
let new_edit_delta = new_edit.new.len() as i32 - new_edit.old.len() as i32;
if new_edit.old.end < self.0[ix].new.start {
let new_edit = new_edits.next().unwrap().clone();
self.0.insert(ix, new_edit);
ix += 1;
let old_edit = &mut self.0[ix];
old_edit.new.start = (old_edit.new.start as i32 + new_edit_delta) as u32;
old_edit.new.end = (old_edit.new.end as i32 + new_edit_delta) as u32;
new_delta += new_edit_delta;
} else if new_edit.old.start <= self.0[ix].new.end {
let new_edit = new_edits.next().unwrap().clone();
let old_edit = &mut self.0[ix];
if new_edit.old.start < old_edit.new.start {
old_edit.old.start -= old_edit.new.start - new_edit.old.start;
old_edit.new.start = new_edit.new.start;
}
if new_edit.old.end > old_edit.new.end {
old_edit.old.end += new_edit.old.end - old_edit.new.end;
old_edit.new.end = new_edit.old.end;
}
old_edit.new.end = (old_edit.new.end as i32 + new_edit_delta) as u32;
new_delta += new_edit_delta;
if old_edit.old.len() == 0 && old_edit.new.len() == 0 {
self.0.remove(ix);
continue 'outer;
}
} else {
break;
}
}
ix += 1;
}
self.0.extend(new_edits.cloned());
self
// let mut other_ranges = edit.ranges.iter().peekable();
// let mut new_ranges = Vec::new();
// let insertion_len = edit.new_text.as_ref().map_or(0, |t| t.len());
// let mut delta = 0;
// for mut self_range in self.ranges.iter().cloned() {
// self_range.start += delta;
// self_range.end += delta;
// while let Some(other_range) = other_ranges.peek() {
// let mut other_range = (*other_range).clone();
// other_range.start += delta;
// other_range.end += delta;
// if other_range.start <= self_range.end {
// other_ranges.next().unwrap();
// delta += insertion_len;
// if other_range.end < self_range.start {
// new_ranges.push(other_range.start..other_range.end + insertion_len);
// self_range.start += insertion_len;
// self_range.end += insertion_len;
// } else {
// self_range.start = cmp::min(self_range.start, other_range.start);
// self_range.end = cmp::max(self_range.end, other_range.end) + insertion_len;
// }
// } else {
// break;
// }
// }
// new_ranges.push(self_range);
// }
// for other_range in other_ranges {
// new_ranges.push(other_range.start + delta..other_range.end + delta + insertion_len);
// delta += insertion_len;
// }
// self.ranges = new_ranges;
}
fn compose2(&self, new: &Self) -> Patch {
let mut composed = Vec::new();
let mut new_edits = new.0.iter().cloned().peekable();
let mut old_delta = 0;
let mut new_delta = 0;
for mut old_edit in self.0.iter().cloned() {
let mut next_new_delta = new_delta;
while let Some(mut new_edit) = new_edits.peek().cloned() {
let new_edit_delta = new_edit.new.len() as i32 - new_edit.old.len() as i32;
if new_edit.old.end < old_edit.new.start {
new_edit.old.start = (new_edit.old.start as i32 - old_delta) as u32;
new_edit.old.end = (new_edit.old.end as i32 - old_delta) as u32;
new_edits.next();
new_delta += new_edit_delta;
next_new_delta += new_edit_delta;
composed.push(new_edit);
} else if new_edit.old.start <= old_edit.new.end {
if new_edit.old.start < old_edit.new.start {
old_edit.old.start -= old_edit.new.start - new_edit.old.start;
old_edit.new.start = new_edit.new.start;
}
if new_edit.old.end > old_edit.new.end {
old_edit.old.end += new_edit.old.end - old_edit.new.end;
old_edit.new.end = new_edit.old.end;
}
old_edit.new.end = (old_edit.new.end as i32 + new_edit_delta) as u32;
new_edits.next();
next_new_delta += new_edit_delta;
} else {
break;
}
}
old_edit.new.start = (old_edit.new.start as i32 + new_delta) as u32;
old_edit.new.end = (old_edit.new.end as i32 + new_delta) as u32;
old_delta += old_edit.new.len() as i32 - old_edit.old.len() as i32;
new_delta = next_new_delta;
composed.push(old_edit);
}
composed.extend(new_edits.map(|mut new_edit| {
new_edit.old.start = (new_edit.old.start as i32 - old_delta) as u32;
new_edit.old.end = (new_edit.old.end as i32 - old_delta) as u32;
new_edit
}));
Patch(composed)
}
fn invert(&mut self) -> &mut Self {
for edit in &mut self.0 {
mem::swap(&mut edit.old, &mut edit.new);
}
self
}
}
impl Snapshot { impl Snapshot {
fn new(tab_snapshot: TabSnapshot) -> Self { fn new(tab_snapshot: TabSnapshot) -> Self {
let mut transforms = SumTree::new(); let mut transforms = SumTree::new();
@ -432,7 +293,7 @@ impl Snapshot {
} }
} }
fn interpolate(&mut self, new_tab_snapshot: TabSnapshot, edits: &[TabEdit]) -> Patch { fn interpolate(&mut self, new_tab_snapshot: TabSnapshot, edits: &[TabEdit]) -> Vec<Edit> {
let mut new_transforms; let mut new_transforms;
if edits.is_empty() { if edits.is_empty() {
new_transforms = self.transforms.clone(); new_transforms = self.transforms.clone();
@ -497,7 +358,7 @@ impl Snapshot {
edits: &[TabEdit], edits: &[TabEdit],
wrap_width: f32, wrap_width: f32,
line_wrapper: &mut LineWrapper, line_wrapper: &mut LineWrapper,
) -> Patch { ) -> Vec<Edit> {
#[derive(Debug)] #[derive(Debug)]
struct RowEdit { struct RowEdit {
old_rows: Range<u32>, old_rows: Range<u32>,
@ -1066,7 +927,11 @@ impl<'a> sum_tree::Dimension<'a, TransformSummary> for WrapPoint {
} }
} }
fn compose(prev: &mut Vec<Edit>, next: &[Edit]) {} fn invert(edits: &mut Vec<Edit>) {
for edit in edits {
mem::swap(&mut edit.old, &mut edit.new);
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {