mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-28 04:45:05 +00:00
support replacing root node
This commit is contained in:
parent
b565d8db74
commit
41dbaa415a
3 changed files with 200 additions and 14 deletions
|
@ -51,18 +51,28 @@ impl SyntaxEditor {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn insert(&mut self, position: Position, element: impl Element) {
|
pub fn insert(&mut self, position: Position, element: impl Element) {
|
||||||
|
debug_assert!(is_ancestor_or_self(&position.parent(), &self.root));
|
||||||
self.changes.push(Change::Insert(position, element.syntax_element()))
|
self.changes.push(Change::Insert(position, element.syntax_element()))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn insert_all(&mut self, position: Position, elements: Vec<SyntaxElement>) {
|
pub fn insert_all(&mut self, position: Position, elements: Vec<SyntaxElement>) {
|
||||||
|
debug_assert!(is_ancestor_or_self(&position.parent(), &self.root));
|
||||||
self.changes.push(Change::InsertAll(position, elements))
|
self.changes.push(Change::InsertAll(position, elements))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn delete(&mut self, element: impl Element) {
|
pub fn delete(&mut self, element: impl Element) {
|
||||||
|
let element = element.syntax_element();
|
||||||
|
debug_assert!(is_ancestor_or_self_of_element(&element, &self.root));
|
||||||
|
debug_assert!(
|
||||||
|
!matches!(&element, SyntaxElement::Node(node) if node == &self.root),
|
||||||
|
"should not delete root node"
|
||||||
|
);
|
||||||
self.changes.push(Change::Replace(element.syntax_element(), None));
|
self.changes.push(Change::Replace(element.syntax_element(), None));
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn replace(&mut self, old: impl Element, new: impl Element) {
|
pub fn replace(&mut self, old: impl Element, new: impl Element) {
|
||||||
|
let old = old.syntax_element();
|
||||||
|
debug_assert!(is_ancestor_or_self_of_element(&old, &self.root));
|
||||||
self.changes.push(Change::Replace(old.syntax_element(), Some(new.syntax_element())));
|
self.changes.push(Change::Replace(old.syntax_element(), Some(new.syntax_element())));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -199,7 +209,10 @@ impl Change {
|
||||||
fn target_parent(&self) -> SyntaxNode {
|
fn target_parent(&self) -> SyntaxNode {
|
||||||
match self {
|
match self {
|
||||||
Change::Insert(target, _) | Change::InsertAll(target, _) => target.parent(),
|
Change::Insert(target, _) | Change::InsertAll(target, _) => target.parent(),
|
||||||
Change::Replace(target, _) => target.parent().unwrap(),
|
Change::Replace(SyntaxElement::Node(target), _) => {
|
||||||
|
target.parent().unwrap_or_else(|| target.clone())
|
||||||
|
}
|
||||||
|
Change::Replace(SyntaxElement::Token(target), _) => target.parent().unwrap(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -248,6 +261,15 @@ impl Element for SyntaxToken {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn is_ancestor_or_self(node: &SyntaxNode, ancestor: &SyntaxNode) -> bool {
|
||||||
|
node == ancestor || node.ancestors().any(|it| &it == ancestor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_ancestor_or_self_of_element(node: &SyntaxElement, ancestor: &SyntaxNode) -> bool {
|
||||||
|
matches!(node, SyntaxElement::Node(node) if node == ancestor)
|
||||||
|
|| node.ancestors().any(|it| &it == ancestor)
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use expect_test::expect;
|
use expect_test::expect;
|
||||||
|
@ -370,15 +392,11 @@ mod tests {
|
||||||
Some(to_wrap.clone().into()),
|
Some(to_wrap.clone().into()),
|
||||||
);
|
);
|
||||||
|
|
||||||
// should die:
|
|
||||||
editor.replace(to_replace.syntax(), name_ref.syntax());
|
editor.replace(to_replace.syntax(), name_ref.syntax());
|
||||||
editor.replace(to_wrap.syntax(), new_block.syntax());
|
editor.replace(to_wrap.syntax(), new_block.syntax());
|
||||||
// editor.replace(to_replace.syntax(), name_ref.syntax());
|
|
||||||
|
|
||||||
let edit = editor.finish();
|
let edit = editor.finish();
|
||||||
|
|
||||||
dbg!(&edit.annotations);
|
|
||||||
|
|
||||||
let expect = expect![[r#"
|
let expect = expect![[r#"
|
||||||
_ => {
|
_ => {
|
||||||
let var_name = 2 + 2;
|
let var_name = 2 + 2;
|
||||||
|
@ -393,4 +411,173 @@ mod tests {
|
||||||
.flat_map(|(_, elements)| elements)
|
.flat_map(|(_, elements)| elements)
|
||||||
.all(|element| element.ancestors().any(|it| &it == edit.root())))
|
.all(|element| element.ancestors().any(|it| &it == edit.root())))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic = "some replace change ranges intersect: [Replace(Node(TUPLE_EXPR@5..7), Some(Node(NAME_REF@0..8))), Replace(Node(TUPLE_EXPR@5..7), Some(Node(NAME_REF@0..8)))]"]
|
||||||
|
fn fail_on_non_disjoint_single_replace() {
|
||||||
|
let root = make::match_arm([make::wildcard_pat().into()], None, make::expr_tuple([]));
|
||||||
|
|
||||||
|
let to_wrap = root.syntax().descendants().find_map(ast::TupleExpr::cast).unwrap();
|
||||||
|
|
||||||
|
let mut editor = SyntaxEditor::new(root.syntax().clone());
|
||||||
|
|
||||||
|
let name_ref = make::name_ref("var_name").clone_for_update();
|
||||||
|
|
||||||
|
// should die, ranges are not disjoint
|
||||||
|
editor.replace(to_wrap.syntax(), name_ref.syntax());
|
||||||
|
editor.replace(to_wrap.syntax(), name_ref.syntax());
|
||||||
|
|
||||||
|
let _ = editor.finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_insert_independent() {
|
||||||
|
let root = make::block_expr(
|
||||||
|
[make::let_stmt(
|
||||||
|
make::ext::simple_ident_pat(make::name("second")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("2").into()),
|
||||||
|
)
|
||||||
|
.into()],
|
||||||
|
None,
|
||||||
|
);
|
||||||
|
|
||||||
|
let second_let = root.syntax().descendants().find_map(ast::LetStmt::cast).unwrap();
|
||||||
|
|
||||||
|
let mut editor = SyntaxEditor::new(root.syntax().clone());
|
||||||
|
|
||||||
|
editor.insert(
|
||||||
|
Position::first_child_of(root.stmt_list().unwrap().syntax()),
|
||||||
|
make_let_stmt(
|
||||||
|
None,
|
||||||
|
make::ext::simple_ident_pat(make::name("first")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("1").into()),
|
||||||
|
)
|
||||||
|
.syntax(),
|
||||||
|
);
|
||||||
|
|
||||||
|
editor.insert(
|
||||||
|
Position::after(second_let.syntax()),
|
||||||
|
make_let_stmt(
|
||||||
|
None,
|
||||||
|
make::ext::simple_ident_pat(make::name("third")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("3").into()),
|
||||||
|
)
|
||||||
|
.syntax(),
|
||||||
|
);
|
||||||
|
|
||||||
|
let edit = editor.finish();
|
||||||
|
|
||||||
|
let expect = expect![[r#"
|
||||||
|
let first = 1;{
|
||||||
|
let second = 2;let third = 3;
|
||||||
|
}"#]];
|
||||||
|
expect.assert_eq(&edit.root.to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_insert_dependent() {
|
||||||
|
let root = make::block_expr(
|
||||||
|
[],
|
||||||
|
Some(
|
||||||
|
make::block_expr(
|
||||||
|
[make::let_stmt(
|
||||||
|
make::ext::simple_ident_pat(make::name("second")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("2").into()),
|
||||||
|
)
|
||||||
|
.into()],
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
.into(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
let inner_block =
|
||||||
|
root.syntax().descendants().flat_map(ast::BlockExpr::cast).nth(1).unwrap();
|
||||||
|
let second_let = root.syntax().descendants().find_map(ast::LetStmt::cast).unwrap();
|
||||||
|
|
||||||
|
let mut editor = SyntaxEditor::new(root.syntax().clone());
|
||||||
|
|
||||||
|
let new_block_expr =
|
||||||
|
make_block_expr(Some(&mut editor), [], Some(ast::Expr::BlockExpr(inner_block.clone())));
|
||||||
|
|
||||||
|
let first_let = make_let_stmt(
|
||||||
|
Some(&mut editor),
|
||||||
|
make::ext::simple_ident_pat(make::name("first")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("1").into()),
|
||||||
|
);
|
||||||
|
|
||||||
|
let third_let = make_let_stmt(
|
||||||
|
Some(&mut editor),
|
||||||
|
make::ext::simple_ident_pat(make::name("third")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("3").into()),
|
||||||
|
);
|
||||||
|
|
||||||
|
editor.insert(
|
||||||
|
Position::first_child_of(inner_block.stmt_list().unwrap().syntax()),
|
||||||
|
first_let.syntax(),
|
||||||
|
);
|
||||||
|
editor.insert(Position::after(second_let.syntax()), third_let.syntax());
|
||||||
|
editor.replace(inner_block.syntax(), new_block_expr.syntax());
|
||||||
|
|
||||||
|
let edit = editor.finish();
|
||||||
|
|
||||||
|
let expect = expect![[r#"
|
||||||
|
{
|
||||||
|
{
|
||||||
|
let first = 1;{
|
||||||
|
let second = 2;let third = 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}"#]];
|
||||||
|
expect.assert_eq(&edit.root.to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_replace_root_with_dependent() {
|
||||||
|
let root = make::block_expr(
|
||||||
|
[make::let_stmt(
|
||||||
|
make::ext::simple_ident_pat(make::name("second")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("2").into()),
|
||||||
|
)
|
||||||
|
.into()],
|
||||||
|
None,
|
||||||
|
);
|
||||||
|
|
||||||
|
let inner_block = root.clone();
|
||||||
|
|
||||||
|
let mut editor = SyntaxEditor::new(root.syntax().clone());
|
||||||
|
|
||||||
|
let new_block_expr =
|
||||||
|
make_block_expr(Some(&mut editor), [], Some(ast::Expr::BlockExpr(inner_block.clone())));
|
||||||
|
|
||||||
|
let first_let = make_let_stmt(
|
||||||
|
Some(&mut editor),
|
||||||
|
make::ext::simple_ident_pat(make::name("first")).into(),
|
||||||
|
None,
|
||||||
|
Some(make::expr_literal("1").into()),
|
||||||
|
);
|
||||||
|
|
||||||
|
editor.insert(
|
||||||
|
Position::first_child_of(inner_block.stmt_list().unwrap().syntax()),
|
||||||
|
first_let.syntax(),
|
||||||
|
);
|
||||||
|
editor.replace(inner_block.syntax(), new_block_expr.syntax());
|
||||||
|
|
||||||
|
let edit = editor.finish();
|
||||||
|
|
||||||
|
let expect = expect![[r#"
|
||||||
|
{
|
||||||
|
let first = 1;{
|
||||||
|
let second = 2;
|
||||||
|
}
|
||||||
|
}"#]];
|
||||||
|
expect.assert_eq(&edit.root.to_string());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -192,6 +192,8 @@ pub(super) fn apply_edits(editor: SyntaxEditor) -> SyntaxEdit {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Apply changes
|
// Apply changes
|
||||||
|
let mut root = tree_mutator.mutable_clone;
|
||||||
|
|
||||||
for change in changes {
|
for change in changes {
|
||||||
match change {
|
match change {
|
||||||
Change::Insert(position, element) => {
|
Change::Insert(position, element) => {
|
||||||
|
@ -205,6 +207,9 @@ pub(super) fn apply_edits(editor: SyntaxEditor) -> SyntaxEdit {
|
||||||
Change::Replace(target, None) => {
|
Change::Replace(target, None) => {
|
||||||
target.detach();
|
target.detach();
|
||||||
}
|
}
|
||||||
|
Change::Replace(SyntaxElement::Node(target), Some(new_target)) if &target == &root => {
|
||||||
|
root = new_target.into_node().expect("root node replacement should be a node");
|
||||||
|
}
|
||||||
Change::Replace(target, Some(new_target)) => {
|
Change::Replace(target, Some(new_target)) => {
|
||||||
let parent = target.parent().unwrap();
|
let parent = target.parent().unwrap();
|
||||||
parent.splice_children(target.index()..target.index() + 1, vec![new_target]);
|
parent.splice_children(target.index()..target.index() + 1, vec![new_target]);
|
||||||
|
@ -214,7 +219,7 @@ pub(super) fn apply_edits(editor: SyntaxEditor) -> SyntaxEdit {
|
||||||
|
|
||||||
// Propagate annotations
|
// Propagate annotations
|
||||||
let annotations = annotations.into_iter().filter_map(|(element, annotation)| {
|
let annotations = annotations.into_iter().filter_map(|(element, annotation)| {
|
||||||
match mappings.upmap_element(&element, &tree_mutator.mutable_clone) {
|
match mappings.upmap_element(&element, &root) {
|
||||||
// Needed to follow the new tree to find the resulting element
|
// Needed to follow the new tree to find the resulting element
|
||||||
Some(Ok(mapped)) => Some((mapped, annotation)),
|
Some(Ok(mapped)) => Some((mapped, annotation)),
|
||||||
// Element did not need to be mapped
|
// Element did not need to be mapped
|
||||||
|
@ -230,11 +235,7 @@ pub(super) fn apply_edits(editor: SyntaxEditor) -> SyntaxEdit {
|
||||||
annotation_groups.entry(annotation).or_insert(vec![]).push(element);
|
annotation_groups.entry(annotation).or_insert(vec![]).push(element);
|
||||||
}
|
}
|
||||||
|
|
||||||
SyntaxEdit {
|
SyntaxEdit { root, changed_elements, annotations: annotation_groups }
|
||||||
root: tree_mutator.mutable_clone,
|
|
||||||
changed_elements,
|
|
||||||
annotations: annotation_groups,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn to_owning_node(element: &SyntaxElement) -> SyntaxNode {
|
fn to_owning_node(element: &SyntaxElement) -> SyntaxNode {
|
||||||
|
@ -278,7 +279,6 @@ impl ChangedAncestor {
|
||||||
}
|
}
|
||||||
|
|
||||||
struct TreeMutator {
|
struct TreeMutator {
|
||||||
immutable: SyntaxNode,
|
|
||||||
mutable_clone: SyntaxNode,
|
mutable_clone: SyntaxNode,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -286,7 +286,7 @@ impl TreeMutator {
|
||||||
fn new(immutable: &SyntaxNode) -> TreeMutator {
|
fn new(immutable: &SyntaxNode) -> TreeMutator {
|
||||||
let immutable = immutable.clone();
|
let immutable = immutable.clone();
|
||||||
let mutable_clone = immutable.clone_for_update();
|
let mutable_clone = immutable.clone_for_update();
|
||||||
TreeMutator { immutable, mutable_clone }
|
TreeMutator { mutable_clone }
|
||||||
}
|
}
|
||||||
|
|
||||||
fn make_element_mut(&self, element: &SyntaxElement) -> SyntaxElement {
|
fn make_element_mut(&self, element: &SyntaxElement) -> SyntaxElement {
|
||||||
|
|
|
@ -108,7 +108,6 @@ impl SyntaxMapping {
|
||||||
input_ancestor: &SyntaxNode,
|
input_ancestor: &SyntaxNode,
|
||||||
output_ancestor: &SyntaxNode,
|
output_ancestor: &SyntaxNode,
|
||||||
) -> Result<Vec<usize>, MissingMapping> {
|
) -> Result<Vec<usize>, MissingMapping> {
|
||||||
eprintln!("mapping ancestor {input_ancestor:#?} to {output_ancestor:#?}");
|
|
||||||
let mut current =
|
let mut current =
|
||||||
self.upmap_node_single(input_ancestor).unwrap_or_else(|| input_ancestor.clone());
|
self.upmap_node_single(input_ancestor).unwrap_or_else(|| input_ancestor.clone());
|
||||||
let mut upmap_chain = vec![current.index()];
|
let mut upmap_chain = vec![current.index()];
|
||||||
|
|
Loading…
Reference in a new issue