Prepare to find diffs between inlay hint generations

This commit is contained in:
Kirill Bulatov 2023-06-07 18:59:39 +03:00
parent 78b3c9b88a
commit 9287634548
2 changed files with 85 additions and 12 deletions

View file

@ -288,13 +288,28 @@ impl DisplayMap {
.update(cx, |map, cx| map.set_wrap_width(width, cx)) .update(cx, |map, cx| map.set_wrap_width(width, cx))
} }
pub fn set_inlay_hints(&self, new_hints: &[project::InlayHint], cx: &mut ModelContext<Self>) { pub fn set_inlay_hints(
&mut self,
new_hints: &[project::InlayHint],
cx: &mut ModelContext<Self>,
) {
dbg!("---", new_hints.len());
let multi_buffer = self.buffer.read(cx); let multi_buffer = self.buffer.read(cx);
let zz = dbg!(multi_buffer
.all_buffers()
.into_iter()
.map(|buffer_handle| buffer_handle.id())
.collect::<HashSet<_>>());
self.editor_addition_map.set_inlay_hints( self.editor_addition_map.set_inlay_hints(
new_hints new_hints
.into_iter() .into_iter()
.filter_map(|hint| { .filter_map(|hint| {
let buffer = multi_buffer.buffer(hint.buffer_id)?.read(cx); // TODO kb this is all wrong, need to manage both(?) or at least the remote buffer id when needed.
// Here though, `.buffer(` requires remote buffer id, so use the workaround above.
dbg!(zz.contains(&(hint.buffer_id as usize)));
let buffer = dbg!(multi_buffer.buffer(dbg!(hint.buffer_id)))?.read(cx);
let snapshot = buffer.snapshot(); let snapshot = buffer.snapshot();
Some(InlayHintToRender { Some(InlayHintToRender {
position: editor_addition_map::EditorAdditionPoint( position: editor_addition_map::EditorAdditionPoint(

View file

@ -1,7 +1,10 @@
#![allow(unused)] #![allow(unused)]
// TODO kb // TODO kb
use std::ops::{Add, AddAssign, Range, Sub}; use std::{
ops::{Add, AddAssign, Range, Sub},
sync::atomic::{self, AtomicUsize},
};
use crate::MultiBufferSnapshot; use crate::MultiBufferSnapshot;
@ -12,21 +15,43 @@ use super::{
}, },
TextHighlights, TextHighlights,
}; };
use collections::HashMap;
use gpui::fonts::HighlightStyle; use gpui::fonts::HighlightStyle;
use language::{Chunk, Edit, Point, Rope, TextSummary}; use language::{Chunk, Edit, Point, Rope, TextSummary};
use parking_lot::Mutex; use parking_lot::Mutex;
use project::InlayHint; use project::InlayHint;
use rand::Rng; use rand::Rng;
use sum_tree::Bias; use sum_tree::{Bias, SumTree};
pub struct EditorAdditionMap(Mutex<EditorAdditionSnapshot>); #[derive(Clone, Copy, Debug, Default, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct InlayHintId(usize);
pub struct EditorAdditionMap {
snapshot: Mutex<EditorAdditionSnapshot>,
next_hint_id: AtomicUsize,
hints: HashMap<InlayHintId, InlayHintToRender>,
}
#[derive(Clone)] #[derive(Clone)]
pub struct EditorAdditionSnapshot { pub struct EditorAdditionSnapshot {
// TODO kb merge these two together // TODO kb merge these two together
pub suggestion_snapshot: SuggestionSnapshot, pub suggestion_snapshot: SuggestionSnapshot,
transforms: SumTree<Transform>,
pub version: usize, pub version: usize,
hints: Vec<InlayHintToRender>, }
#[derive(Clone)]
struct Transform {
input: TextSummary,
output: TextSummary,
}
impl sum_tree::Item for Transform {
type Summary = TextSummary;
fn summary(&self) -> Self::Summary {
self.output.clone()
}
} }
pub type EditorAdditionEdit = Edit<EditorAdditionOffset>; pub type EditorAdditionEdit = Edit<EditorAdditionOffset>;
@ -68,7 +93,7 @@ pub struct EditorAdditionChunks<'a> {
suggestion_chunks: SuggestionChunks<'a>, suggestion_chunks: SuggestionChunks<'a>,
} }
#[derive(Clone)] #[derive(Debug, Clone)]
pub struct InlayHintToRender { pub struct InlayHintToRender {
pub(super) position: EditorAdditionPoint, pub(super) position: EditorAdditionPoint,
pub(super) text: Rope, pub(super) text: Rope,
@ -109,9 +134,17 @@ impl EditorAdditionMap {
let snapshot = EditorAdditionSnapshot { let snapshot = EditorAdditionSnapshot {
suggestion_snapshot: suggestion_snapshot.clone(), suggestion_snapshot: suggestion_snapshot.clone(),
version: 0, version: 0,
hints: Vec::new(), transforms: SumTree::new(),
}; };
(Self(Mutex::new(snapshot.clone())), snapshot)
(
Self {
snapshot: Mutex::new(snapshot.clone()),
next_hint_id: AtomicUsize::new(0),
hints: HashMap::default(),
},
snapshot,
)
} }
pub fn sync( pub fn sync(
@ -119,13 +152,15 @@ impl EditorAdditionMap {
suggestion_snapshot: SuggestionSnapshot, suggestion_snapshot: SuggestionSnapshot,
suggestion_edits: Vec<SuggestionEdit>, suggestion_edits: Vec<SuggestionEdit>,
) -> (EditorAdditionSnapshot, Vec<EditorAdditionEdit>) { ) -> (EditorAdditionSnapshot, Vec<EditorAdditionEdit>) {
let mut snapshot = self.0.lock(); let mut snapshot = self.snapshot.lock();
if snapshot.suggestion_snapshot.version != suggestion_snapshot.version { if snapshot.suggestion_snapshot.version != suggestion_snapshot.version {
snapshot.version += 1; snapshot.version += 1;
} }
let mut editor_addition_edits = Vec::new(); let mut editor_addition_edits = Vec::new();
dbg!(&suggestion_edits);
for suggestion_edit in suggestion_edits { for suggestion_edit in suggestion_edits {
let old = suggestion_edit.old; let old = suggestion_edit.old;
let new = suggestion_edit.new; let new = suggestion_edit.new;
@ -141,8 +176,31 @@ impl EditorAdditionMap {
(snapshot.clone(), editor_addition_edits) (snapshot.clone(), editor_addition_edits)
} }
pub fn set_inlay_hints(&self, new_hints: Vec<InlayHintToRender>) { // TODO kb replace set_inlay_hints with this
self.0.lock().hints = new_hints; pub fn splice(
&mut self,
to_remove: Vec<InlayHintId>,
to_insert: Vec<InlayHintToRender>,
) -> Vec<InlayHintId> {
// Order removals and insertions by position.
// let anchors;
// Remove and insert inlays in a single traversal across the tree.
todo!("TODO kb")
}
pub fn set_inlay_hints(&mut self, new_hints: Vec<InlayHintToRender>) {
dbg!(new_hints.len());
// TODO kb reuse ids for hints that did not change and similar things
self.hints = new_hints
.into_iter()
.map(|hint| {
(
InlayHintId(self.next_hint_id.fetch_add(1, atomic::Ordering::SeqCst)),
hint,
)
})
.collect();
} }
} }