Unnamed repository; edit this file 'description' to name the repository.
Fix overlap edit on tuple to record assist expr uses self
Example --- ```rust struct T$0(u8); fn test(t: T) { T(t.0); } ``` **Before this PR** Panic **After this PR** ```rust struct T { field1: u8 } fn test(t: T) { T { field1: t.field1 }; } ```
A4-Tacks 2 months ago
parent 30b8d5e · commit 34839f6
-rw-r--r--crates/ide-assists/src/handlers/convert_named_struct_to_tuple_struct.rs25
-rw-r--r--crates/ide-assists/src/handlers/convert_tuple_struct_to_named_struct.rs264
-rw-r--r--crates/ide-assists/src/utils.rs14
3 files changed, 198 insertions, 105 deletions
diff --git a/crates/ide-assists/src/handlers/convert_named_struct_to_tuple_struct.rs b/crates/ide-assists/src/handlers/convert_named_struct_to_tuple_struct.rs
index 8dfd4598df..42fceb8533 100644
--- a/crates/ide-assists/src/handlers/convert_named_struct_to_tuple_struct.rs
+++ b/crates/ide-assists/src/handlers/convert_named_struct_to_tuple_struct.rs
@@ -1,9 +1,7 @@
-use std::ops::RangeInclusive;
-
use either::Either;
use ide_db::{defs::Definition, search::FileReference};
use syntax::{
- NodeOrToken, SyntaxElement, SyntaxKind, SyntaxNode, T, TextRange,
+ NodeOrToken, SyntaxKind, SyntaxNode, T,
algo::next_non_trivia_token,
ast::{
self, AstNode, HasAttrs, HasGenericParams, HasVisibility, syntax_factory::SyntaxFactory,
@@ -12,7 +10,9 @@ use syntax::{
syntax_editor::{Element, Position, SyntaxEditor},
};
-use crate::{AssistContext, AssistId, Assists, assist_context::SourceChangeBuilder};
+use crate::{
+ AssistContext, AssistId, Assists, assist_context::SourceChangeBuilder, utils::cover_edit_range,
+};
// Assist: convert_named_struct_to_tuple_struct
//
@@ -242,7 +242,7 @@ where
{
let make = SyntaxFactory::without_mappings();
let orig = ctx.sema.original_range_opt(field_list.syntax())?;
- let list_range = cover_range(source, orig.range);
+ let list_range = cover_edit_range(source, orig.range);
let l_curly = match list_range.start() {
NodeOrToken::Node(node) => node.first_token()?,
@@ -265,7 +265,7 @@ where
for name_ref in fields(&field_list) {
let Some(orig) = ctx.sema.original_range_opt(name_ref.syntax()) else { continue };
- let name_range = cover_range(source, orig.range);
+ let name_range = cover_edit_range(source, orig.range);
if let Some(colon) = next_non_trivia_token(name_range.end().clone())
&& colon.kind() == T![:]
@@ -306,7 +306,7 @@ fn edit_field_references(
// Only edit the field reference if it's part of a `.field` access
if name_ref.syntax().parent().and_then(ast::FieldExpr::cast).is_some() {
edit.replace_all(
- cover_range(&source, r.range),
+ cover_edit_range(&source, r.range),
vec![make.name_ref(&index.to_string()).syntax().clone().into()],
);
}
@@ -318,17 +318,6 @@ fn edit_field_references(
}
}
-fn cover_range(source: &ast::SourceFile, range: TextRange) -> RangeInclusive<SyntaxElement> {
- let node = match source.syntax().covering_element(range) {
- NodeOrToken::Node(node) => node,
- NodeOrToken::Token(t) => t.parent().unwrap(),
- };
- let mut iter = node.children_with_tokens().filter(|it| range.contains_range(it.text_range()));
- let first = iter.next().unwrap_or(node.into());
- let last = iter.last().unwrap_or_else(|| first.clone());
- first..=last
-}
-
fn delete_whitespace(edit: &mut SyntaxEditor, whitespace: Option<impl Element>) {
let Some(whitespace) = whitespace else { return };
let NodeOrToken::Token(token) = whitespace.syntax_element() else { return };
diff --git a/crates/ide-assists/src/handlers/convert_tuple_struct_to_named_struct.rs b/crates/ide-assists/src/handlers/convert_tuple_struct_to_named_struct.rs
index ba22224b6b..f1eae83866 100644
--- a/crates/ide-assists/src/handlers/convert_tuple_struct_to_named_struct.rs
+++ b/crates/ide-assists/src/handlers/convert_tuple_struct_to_named_struct.rs
@@ -1,17 +1,21 @@
use either::Either;
-use hir::FileRangeWrapper;
-use ide_db::defs::{Definition, NameRefClass};
-use std::ops::RangeInclusive;
+use ide_db::{
+ defs::{Definition, NameRefClass},
+ search::FileReference,
+};
use syntax::{
- SyntaxElement, SyntaxKind, SyntaxNode, T, TextSize,
+ SyntaxKind, T,
ast::{
- self, AstNode, HasAttrs, HasGenericParams, HasVisibility, syntax_factory::SyntaxFactory,
+ self, AstNode, HasArgList, HasAttrs, HasGenericParams, HasVisibility,
+ syntax_factory::SyntaxFactory,
},
match_ast,
syntax_editor::{Element, Position, SyntaxEditor},
};
-use crate::{AssistContext, AssistId, Assists, assist_context::SourceChangeBuilder};
+use crate::{
+ AssistContext, AssistId, Assists, assist_context::SourceChangeBuilder, utils::cover_edit_range,
+};
// Assist: convert_tuple_struct_to_named_struct
//
@@ -147,93 +151,121 @@ fn edit_struct_references(
};
let usages = strukt_def.usages(&ctx.sema).include_self_refs().all();
- let edit_node = |node: SyntaxNode| -> Option<SyntaxNode> {
- let make = SyntaxFactory::without_mappings();
- match_ast! {
- match node {
- ast::TupleStructPat(tuple_struct_pat) => {
- Some(make.record_pat_with_fields(
- tuple_struct_pat.path()?,
- generate_record_pat_list(&tuple_struct_pat, names),
- ).syntax().clone())
- },
- // for tuple struct creations like Foo(42)
- ast::CallExpr(call_expr) => {
- let path = call_expr.syntax().descendants().find_map(ast::PathExpr::cast).and_then(|expr| expr.path())?;
-
- // this also includes method calls like Foo::new(42), we should skip them
- if let Some(name_ref) = path.segment().and_then(|s| s.name_ref()) {
- match NameRefClass::classify(&ctx.sema, &name_ref) {
- Some(NameRefClass::Definition(Definition::SelfType(_), _)) => {},
- Some(NameRefClass::Definition(def, _)) if def == strukt_def => {},
- _ => return None,
- };
- }
+ for (file_id, refs) in usages {
+ let source = ctx.sema.parse(file_id);
+ let mut editor = edit.make_editor(source.syntax());
- let arg_list = call_expr.syntax().descendants().find_map(ast::ArgList::cast)?;
- Some(
- make.record_expr(
- path,
- ast::make::record_expr_field_list(arg_list.args().zip(names).map(
- |(expr, name)| {
- ast::make::record_expr_field(
- ast::make::name_ref(&name.to_string()),
- Some(expr),
- )
- },
- )),
- ).syntax().clone()
- )
- },
- _ => None,
- }
+ for r in refs {
+ process_struct_name_reference(ctx, r, &mut editor, &source, &strukt_def, names);
}
- };
- for (file_id, refs) in usages {
- let source = ctx.sema.parse(file_id);
- let source = source.syntax();
-
- let mut editor = edit.make_editor(source);
- for r in refs.iter().rev() {
- if let Some((old_node, new_node)) = r
- .name
- .syntax()
- .ancestors()
- .find_map(|node| Some((node.clone(), edit_node(node.clone())?)))
- {
- if let Some(old_node) = ctx.sema.original_syntax_node_rooted(&old_node) {
- editor.replace(old_node, new_node);
- } else {
- let FileRangeWrapper { file_id: _, range } = ctx.sema.original_range(&old_node);
- let parent = source.covering_element(range);
- match parent {
- SyntaxElement::Token(token) => {
- editor.replace(token, new_node.syntax_element());
- }
- SyntaxElement::Node(parent_node) => {
- // replace the part of macro
- // ```
- // foo!(a, Test::A(0));
- // ^^^^^^^^^^^^^^^ // parent_node
- // ^^^^^^^^^^ // replace_range
- // ```
- let start = parent_node
- .children_with_tokens()
- .find(|t| t.text_range().contains(range.start()));
- let end = parent_node
- .children_with_tokens()
- .find(|t| t.text_range().contains(range.end() - TextSize::new(1)));
- if let (Some(start), Some(end)) = (start, end) {
- let replace_range = RangeInclusive::new(start, end);
- editor.replace_all(replace_range, vec![new_node.into()]);
- }
- }
+ edit.add_file_edits(file_id.file_id(ctx.db()), editor);
+ }
+}
+
+fn process_struct_name_reference(
+ ctx: &AssistContext<'_>,
+ r: FileReference,
+ editor: &mut SyntaxEditor,
+ source: &ast::SourceFile,
+ strukt_def: &Definition,
+ names: &[ast::Name],
+) -> Option<()> {
+ let make = SyntaxFactory::without_mappings();
+ let name_ref = r.name.as_name_ref()?;
+ let path_segment = name_ref.syntax().parent().and_then(ast::PathSegment::cast)?;
+ let full_path = path_segment.syntax().parent().and_then(ast::Path::cast)?.top_path();
+
+ if full_path.segment()?.name_ref()? != *name_ref {
+ // `name_ref` isn't the last segment of the path, so `full_path` doesn't point to the
+ // struct we want to edit.
+ return None;
+ }
+
+ let parent = full_path.syntax().parent()?;
+ match_ast! {
+ match parent {
+ ast::TupleStructPat(tuple_struct_pat) => {
+ let range = ctx.sema.original_range_opt(tuple_struct_pat.syntax())?.range;
+ let new = make.record_pat_with_fields(
+ full_path,
+ generate_record_pat_list(&tuple_struct_pat, names),
+ );
+ editor.replace_all(cover_edit_range(source, range), vec![new.syntax().clone().into()]);
+ },
+ ast::PathExpr(path_expr) => {
+ let call_expr = path_expr.syntax().parent().and_then(ast::CallExpr::cast)?;
+
+ // this also includes method calls like Foo::new(42), we should skip them
+ match NameRefClass::classify(&ctx.sema, name_ref) {
+ Some(NameRefClass::Definition(Definition::SelfType(_), _)) => {},
+ Some(NameRefClass::Definition(def, _)) if def == *strukt_def => {},
+ _ => return None,
+ }
+
+ let arg_list = call_expr.arg_list()?;
+ let mut first_insert = vec![];
+ for (expr, name) in arg_list.args().zip(names) {
+ let range = ctx.sema.original_range_opt(expr.syntax())?.range;
+ let place = cover_edit_range(source, range);
+ let elements = vec![
+ make.name_ref(&name.text()).syntax().clone().into(),
+ make.token(T![:]).into(),
+ make.whitespace(" ").into(),
+ ];
+ if first_insert.is_empty() {
+ // XXX: SyntaxEditor cannot insert after deleted element
+ first_insert = elements;
+ } else {
+ editor.insert_all(Position::before(place.start()), elements);
}
}
- }
+ process_delimiter(ctx, source, editor, &arg_list, first_insert);
+ },
+ _ => {}
}
- edit.add_file_edits(file_id.file_id(ctx.db()), editor);
+ }
+ Some(())
+}
+
+fn process_delimiter(
+ ctx: &AssistContext<'_>,
+ source: &ast::SourceFile,
+ editor: &mut SyntaxEditor,
+ list: &impl AstNode,
+ first_insert: Vec<syntax::SyntaxElement>,
+) {
+ let Some(range) = ctx.sema.original_range_opt(list.syntax()) else { return };
+ let place = cover_edit_range(source, range.range);
+
+ let l_paren = match place.start() {
+ syntax::NodeOrToken::Node(node) => node.first_token(),
+ syntax::NodeOrToken::Token(t) => Some(t.clone()),
+ };
+ let r_paren = match place.end() {
+ syntax::NodeOrToken::Node(node) => node.last_token(),
+ syntax::NodeOrToken::Token(t) => Some(t.clone()),
+ };
+
+ let make = SyntaxFactory::without_mappings();
+ if let Some(l_paren) = l_paren
+ && l_paren.kind() == T!['(']
+ {
+ let mut open_delim = vec![
+ make.whitespace(" ").into(),
+ make.token(T!['{']).into(),
+ make.whitespace(" ").into(),
+ ];
+ open_delim.extend(first_insert);
+ editor.replace_with_many(l_paren, open_delim);
+ }
+ if let Some(r_paren) = r_paren
+ && r_paren.kind() == T![')']
+ {
+ editor.replace_with_many(
+ r_paren,
+ vec![make.whitespace(" ").into(), make.token(T!['}']).into()],
+ );
}
}
@@ -741,6 +773,64 @@ where
"#,
);
}
+
+ #[test]
+ fn convert_expr_uses_self() {
+ check_assist(
+ convert_tuple_struct_to_named_struct,
+ r#"
+macro_rules! id {
+ ($($t:tt)*) => { $($t)* }
+}
+struct T$0(u8);
+fn test(t: T) {
+ T(t.0);
+ id!(T(t.0));
+}"#,
+ r#"
+macro_rules! id {
+ ($($t:tt)*) => { $($t)* }
+}
+struct T { field1: u8 }
+fn test(t: T) {
+ T { field1: t.field1 };
+ id!(T { field1: t.field1 });
+}"#,
+ );
+ }
+
+ #[test]
+ #[ignore = "FIXME overlap edits in nested uses self"]
+ fn convert_pat_uses_self() {
+ check_assist(
+ convert_tuple_struct_to_named_struct,
+ r#"
+macro_rules! id {
+ ($($t:tt)*) => { $($t)* }
+}
+enum T {
+ $0Value(&'static T),
+ Nil,
+}
+fn test(t: T) {
+ if let T::Value(T::Value(t)) = t {}
+ if let id!(T::Value(T::Value(t))) = t {}
+}"#,
+ r#"
+macro_rules! id {
+ ($($t:tt)*) => { $($t)* }
+}
+enum T {
+ Value { field1: &'static T },
+ Nil,
+}
+fn test(t: T) {
+ if let T::Value { field1: T::Value { field1: t } } = t {}
+ if let id!(T::Value { field1: T::Value { field1: t } }) = t {}
+}"#,
+ );
+ }
+
#[test]
fn not_applicable_other_than_tuple_variant() {
check_assist_not_applicable(
diff --git a/crates/ide-assists/src/utils.rs b/crates/ide-assists/src/utils.rs
index b4055e77cc..a3178aba12 100644
--- a/crates/ide-assists/src/utils.rs
+++ b/crates/ide-assists/src/utils.rs
@@ -1254,6 +1254,20 @@ pub(crate) fn cover_let_chain(mut expr: ast::Expr, range: TextRange) -> Option<a
}
}
+pub(crate) fn cover_edit_range(
+ source: &impl AstNode,
+ range: TextRange,
+) -> std::ops::RangeInclusive<syntax::SyntaxElement> {
+ let node = match source.syntax().covering_element(range) {
+ NodeOrToken::Node(node) => node,
+ NodeOrToken::Token(t) => t.parent().unwrap(),
+ };
+ let mut iter = node.children_with_tokens().filter(|it| range.contains_range(it.text_range()));
+ let first = iter.next().unwrap_or(node.into());
+ let last = iter.last().unwrap_or_else(|| first.clone());
+ first..=last
+}
+
pub(crate) fn is_selected(
it: &impl AstNode,
selection: syntax::TextRange,