mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 21:54:42 +00:00
add new editing API, suitable for modifying several nodes at once
This commit is contained in:
parent
a452e50e0e
commit
a525e830a6
7 changed files with 119 additions and 29 deletions
2
Cargo.lock
generated
2
Cargo.lock
generated
|
@ -903,6 +903,7 @@ dependencies = [
|
||||||
"ra_hir 0.1.0",
|
"ra_hir 0.1.0",
|
||||||
"ra_syntax 0.1.0",
|
"ra_syntax 0.1.0",
|
||||||
"ra_text_edit 0.1.0",
|
"ra_text_edit 0.1.0",
|
||||||
|
"rustc-hash 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"test_utils 0.1.0",
|
"test_utils 0.1.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -1088,6 +1089,7 @@ dependencies = [
|
||||||
"ra_parser 0.1.0",
|
"ra_parser 0.1.0",
|
||||||
"ra_text_edit 0.1.0",
|
"ra_text_edit 0.1.0",
|
||||||
"rowan 0.6.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
"rowan 0.6.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"rustc-hash 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"rustc_lexer 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
"rustc_lexer 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"smol_str 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)",
|
"smol_str 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"test_utils 0.1.0",
|
"test_utils 0.1.0",
|
||||||
|
|
|
@ -10,6 +10,7 @@ once_cell = "1.0.1"
|
||||||
join_to_string = "0.1.3"
|
join_to_string = "0.1.3"
|
||||||
itertools = "0.8.0"
|
itertools = "0.8.0"
|
||||||
arrayvec = "0.4.10"
|
arrayvec = "0.4.10"
|
||||||
|
rustc-hash = "1.0.1"
|
||||||
|
|
||||||
ra_syntax = { path = "../ra_syntax" }
|
ra_syntax = { path = "../ra_syntax" }
|
||||||
ra_text_edit = { path = "../ra_text_edit" }
|
ra_text_edit = { path = "../ra_text_edit" }
|
||||||
|
|
|
@ -3,10 +3,9 @@ use ra_syntax::{
|
||||||
ast::{self, AstNode, NameOwner, TypeBoundsOwner},
|
ast::{self, AstNode, NameOwner, TypeBoundsOwner},
|
||||||
SyntaxElement,
|
SyntaxElement,
|
||||||
SyntaxKind::*,
|
SyntaxKind::*,
|
||||||
TextRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{ast_builder::Make, Assist, AssistCtx, AssistId};
|
use crate::{ast_builder::Make, ast_editor::AstEditor, Assist, AssistCtx, AssistId};
|
||||||
|
|
||||||
pub(crate) fn move_bounds_to_where_clause(mut ctx: AssistCtx<impl HirDatabase>) -> Option<Assist> {
|
pub(crate) fn move_bounds_to_where_clause(mut ctx: AssistCtx<impl HirDatabase>) -> Option<Assist> {
|
||||||
let type_param_list = ctx.node_at_offset::<ast::TypeParamList>()?;
|
let type_param_list = ctx.node_at_offset::<ast::TypeParamList>()?;
|
||||||
|
@ -36,23 +35,23 @@ pub(crate) fn move_bounds_to_where_clause(mut ctx: AssistCtx<impl HirDatabase>)
|
||||||
AssistId("move_bounds_to_where_clause"),
|
AssistId("move_bounds_to_where_clause"),
|
||||||
"move_bounds_to_where_clause",
|
"move_bounds_to_where_clause",
|
||||||
|edit| {
|
|edit| {
|
||||||
let type_params = type_param_list.type_params().collect::<Vec<_>>();
|
let new_params = type_param_list
|
||||||
|
.type_params()
|
||||||
|
.filter(|it| it.type_bound_list().is_some())
|
||||||
|
.map(|type_param| {
|
||||||
|
let without_bounds =
|
||||||
|
AstEditor::new(type_param.clone()).remove_bounds().ast().clone();
|
||||||
|
(type_param, without_bounds)
|
||||||
|
});
|
||||||
|
|
||||||
for param in &type_params {
|
let mut ast_editor = AstEditor::new(type_param_list.clone());
|
||||||
if let Some(bounds) = param.type_bound_list() {
|
ast_editor.replace_descendants(new_params);
|
||||||
let colon = param
|
ast_editor.into_text_edit(edit.text_edit_builder());
|
||||||
.syntax()
|
|
||||||
.children_with_tokens()
|
|
||||||
.find(|it| it.kind() == COLON)
|
|
||||||
.unwrap();
|
|
||||||
let start = colon.text_range().start();
|
|
||||||
let end = bounds.syntax().text_range().end();
|
|
||||||
edit.delete(TextRange::from_to(start, end));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let predicates = type_params.iter().filter_map(build_predicate);
|
let where_clause = {
|
||||||
let where_clause = Make::<ast::WhereClause>::from_predicates(predicates);
|
let predicates = type_param_list.type_params().filter_map(build_predicate);
|
||||||
|
Make::<ast::WhereClause>::from_predicates(predicates)
|
||||||
|
};
|
||||||
|
|
||||||
let to_insert = match anchor.prev_sibling_or_token() {
|
let to_insert = match anchor.prev_sibling_or_token() {
|
||||||
Some(ref elem) if elem.kind() == WHITESPACE => {
|
Some(ref elem) if elem.kind() == WHITESPACE => {
|
||||||
|
@ -68,7 +67,7 @@ pub(crate) fn move_bounds_to_where_clause(mut ctx: AssistCtx<impl HirDatabase>)
|
||||||
ctx.build()
|
ctx.build()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build_predicate(param: &ast::TypeParam) -> Option<ast::WherePred> {
|
fn build_predicate(param: ast::TypeParam) -> Option<ast::WherePred> {
|
||||||
let path = Make::<ast::Path>::from_name(param.name()?);
|
let path = Make::<ast::Path>::from_name(param.name()?);
|
||||||
let predicate = Make::<ast::WherePred>::from(path, param.type_bound_list()?.bounds());
|
let predicate = Make::<ast::WherePred>::from(path, param.type_bound_list()?.bounds());
|
||||||
Some(predicate)
|
Some(predicate)
|
||||||
|
|
|
@ -1,11 +1,13 @@
|
||||||
use std::{iter, ops::RangeInclusive};
|
use std::{iter, ops::RangeInclusive};
|
||||||
|
|
||||||
use arrayvec::ArrayVec;
|
use arrayvec::ArrayVec;
|
||||||
|
use rustc_hash::FxHashMap;
|
||||||
|
|
||||||
use ra_fmt::leading_indent;
|
use ra_fmt::leading_indent;
|
||||||
use ra_syntax::{
|
use ra_syntax::{
|
||||||
algo::{insert_children, replace_children},
|
algo,
|
||||||
ast, AstNode, Direction, InsertPosition, SyntaxElement,
|
ast::{self, TypeBoundsOwner},
|
||||||
|
AstNode, Direction, InsertPosition, NodeOrToken, SyntaxElement,
|
||||||
SyntaxKind::*,
|
SyntaxKind::*,
|
||||||
T,
|
T,
|
||||||
};
|
};
|
||||||
|
@ -27,26 +29,55 @@ impl<N: AstNode> AstEditor<N> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn into_text_edit(self, builder: &mut TextEditBuilder) {
|
pub fn into_text_edit(self, builder: &mut TextEditBuilder) {
|
||||||
// FIXME: compute a more fine-grained diff here.
|
// FIXME: this is both horrible inefficient and gives larger than
|
||||||
// If *you* know a nice algorithm to compute diff between two syntax
|
// necessary diff. I bet there's a cool algorithm to diff trees properly.
|
||||||
// tree, tell me about it!
|
go(builder, self.original_ast.syntax().clone().into(), self.ast().syntax().clone().into());
|
||||||
builder.replace(
|
|
||||||
self.original_ast.syntax().text_range(),
|
fn go(buf: &mut TextEditBuilder, lhs: SyntaxElement, rhs: SyntaxElement) {
|
||||||
self.ast().syntax().text().to_string(),
|
if lhs.kind() == rhs.kind() && lhs.text_range().len() == rhs.text_range().len() {
|
||||||
);
|
if match (&lhs, &rhs) {
|
||||||
|
(NodeOrToken::Node(lhs), NodeOrToken::Node(rhs)) => lhs.text() == rhs.text(),
|
||||||
|
(NodeOrToken::Token(lhs), NodeOrToken::Token(rhs)) => lhs.text() == rhs.text(),
|
||||||
|
_ => false,
|
||||||
|
} {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if let (Some(lhs), Some(rhs)) = (lhs.as_node(), rhs.as_node()) {
|
||||||
|
if lhs.children_with_tokens().count() == rhs.children_with_tokens().count() {
|
||||||
|
for (lhs, rhs) in lhs.children_with_tokens().zip(rhs.children_with_tokens()) {
|
||||||
|
go(buf, lhs, rhs)
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
buf.replace(lhs.text_range(), rhs.to_string())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn ast(&self) -> &N {
|
pub fn ast(&self) -> &N {
|
||||||
&self.ast
|
&self.ast
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn replace_descendants<T: AstNode>(
|
||||||
|
&mut self,
|
||||||
|
replacement_map: impl Iterator<Item = (T, T)>,
|
||||||
|
) -> &mut Self {
|
||||||
|
let map = replacement_map
|
||||||
|
.map(|(from, to)| (from.syntax().clone().into(), to.syntax().clone().into()))
|
||||||
|
.collect::<FxHashMap<_, _>>();
|
||||||
|
let new_syntax = algo::replace_descendants(self.ast.syntax(), &map);
|
||||||
|
self.ast = N::cast(new_syntax).unwrap();
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
fn insert_children(
|
fn insert_children(
|
||||||
&self,
|
&self,
|
||||||
position: InsertPosition<SyntaxElement>,
|
position: InsertPosition<SyntaxElement>,
|
||||||
mut to_insert: impl Iterator<Item = SyntaxElement>,
|
mut to_insert: impl Iterator<Item = SyntaxElement>,
|
||||||
) -> N {
|
) -> N {
|
||||||
let new_syntax = insert_children(self.ast().syntax(), position, &mut to_insert);
|
let new_syntax = algo::insert_children(self.ast().syntax(), position, &mut to_insert);
|
||||||
N::cast(new_syntax).unwrap()
|
N::cast(new_syntax).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -56,7 +87,7 @@ impl<N: AstNode> AstEditor<N> {
|
||||||
to_delete: RangeInclusive<SyntaxElement>,
|
to_delete: RangeInclusive<SyntaxElement>,
|
||||||
mut to_insert: impl Iterator<Item = SyntaxElement>,
|
mut to_insert: impl Iterator<Item = SyntaxElement>,
|
||||||
) -> N {
|
) -> N {
|
||||||
let new_syntax = replace_children(self.ast().syntax(), to_delete, &mut to_insert);
|
let new_syntax = algo::replace_children(self.ast().syntax(), to_delete, &mut to_insert);
|
||||||
N::cast(new_syntax).unwrap()
|
N::cast(new_syntax).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -240,3 +271,18 @@ impl AstEditor<ast::FnDef> {
|
||||||
self.ast = self.replace_children(replace_range, to_insert.into_iter())
|
self.ast = self.replace_children(replace_range, to_insert.into_iter())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl AstEditor<ast::TypeParam> {
|
||||||
|
pub fn remove_bounds(&mut self) -> &mut Self {
|
||||||
|
let colon = match self.ast.colon_token() {
|
||||||
|
Some(it) => it,
|
||||||
|
None => return self,
|
||||||
|
};
|
||||||
|
let end = match self.ast.type_bound_list() {
|
||||||
|
Some(it) => it.syntax().clone().into(),
|
||||||
|
None => colon.clone().into(),
|
||||||
|
};
|
||||||
|
self.ast = self.replace_children(RangeInclusive::new(colon.into(), end), iter::empty());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ repository = "https://github.com/rust-analyzer/rust-analyzer"
|
||||||
itertools = "0.8.0"
|
itertools = "0.8.0"
|
||||||
rowan = "0.6.1"
|
rowan = "0.6.1"
|
||||||
rustc_lexer = "0.1.0"
|
rustc_lexer = "0.1.0"
|
||||||
|
rustc-hash = "1.0.1"
|
||||||
|
|
||||||
# ideally, `serde` should be enabled by `ra_lsp_server`, but we enable it here
|
# ideally, `serde` should be enabled by `ra_lsp_server`, but we enable it here
|
||||||
# to reduce number of compilations
|
# to reduce number of compilations
|
||||||
|
|
|
@ -3,6 +3,7 @@ pub mod visit;
|
||||||
use std::ops::RangeInclusive;
|
use std::ops::RangeInclusive;
|
||||||
|
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
|
use rustc_hash::FxHashMap;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
AstNode, Direction, NodeOrToken, SyntaxElement, SyntaxNode, SyntaxNodePtr, TextRange, TextUnit,
|
AstNode, Direction, NodeOrToken, SyntaxElement, SyntaxNode, SyntaxNodePtr, TextRange, TextUnit,
|
||||||
|
@ -123,6 +124,37 @@ pub fn replace_children(
|
||||||
with_children(parent, new_children)
|
with_children(parent, new_children)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Replaces descendants in the node, according to the mapping.
|
||||||
|
///
|
||||||
|
/// This is a type-unsafe low-level editing API, if you need to use it, prefer
|
||||||
|
/// to create a type-safe abstraction on top of it instead.
|
||||||
|
pub fn replace_descendants(
|
||||||
|
parent: &SyntaxNode,
|
||||||
|
map: &FxHashMap<SyntaxElement, SyntaxElement>,
|
||||||
|
) -> SyntaxNode {
|
||||||
|
// FIXME: this could be made much faster.
|
||||||
|
let new_children = parent.children_with_tokens().map(|it| go(map, it)).collect::<Box<[_]>>();
|
||||||
|
return with_children(parent, new_children);
|
||||||
|
|
||||||
|
fn go(
|
||||||
|
map: &FxHashMap<SyntaxElement, SyntaxElement>,
|
||||||
|
element: SyntaxElement,
|
||||||
|
) -> NodeOrToken<rowan::GreenNode, rowan::GreenToken> {
|
||||||
|
if let Some(replacement) = map.get(&element) {
|
||||||
|
return match replacement {
|
||||||
|
NodeOrToken::Node(it) => NodeOrToken::Node(it.green().clone()),
|
||||||
|
NodeOrToken::Token(it) => NodeOrToken::Token(it.green().clone()),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
match element {
|
||||||
|
NodeOrToken::Token(it) => NodeOrToken::Token(it.green().clone()),
|
||||||
|
NodeOrToken::Node(it) => {
|
||||||
|
NodeOrToken::Node(replace_descendants(&it, map).green().clone())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn with_children(
|
fn with_children(
|
||||||
parent: &SyntaxNode,
|
parent: &SyntaxNode,
|
||||||
new_children: Box<[NodeOrToken<rowan::GreenNode, rowan::GreenToken>]>,
|
new_children: Box<[NodeOrToken<rowan::GreenNode, rowan::GreenToken>]>,
|
||||||
|
|
|
@ -373,6 +373,15 @@ impl ast::LifetimeParam {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ast::TypeParam {
|
||||||
|
pub fn colon_token(&self) -> Option<SyntaxToken> {
|
||||||
|
self.syntax()
|
||||||
|
.children_with_tokens()
|
||||||
|
.filter_map(|it| it.into_token())
|
||||||
|
.find(|it| it.kind() == T![:])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl ast::WherePred {
|
impl ast::WherePred {
|
||||||
pub fn lifetime_token(&self) -> Option<SyntaxToken> {
|
pub fn lifetime_token(&self) -> Option<SyntaxToken> {
|
||||||
self.syntax()
|
self.syntax()
|
||||||
|
|
Loading…
Reference in a new issue