mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-14 14:13:58 +00:00
Merge pull request #18695 from roife/improve-tuple-destruction
minor: improve name suggestion for destructure_tuple_binding
This commit is contained in:
commit
0189a8937b
3 changed files with 72 additions and 72 deletions
|
@ -3,6 +3,7 @@
|
||||||
use hir::{FileRange, Semantics};
|
use hir::{FileRange, Semantics};
|
||||||
use ide_db::EditionedFileId;
|
use ide_db::EditionedFileId;
|
||||||
use ide_db::{label::Label, FileId, RootDatabase};
|
use ide_db::{label::Label, FileId, RootDatabase};
|
||||||
|
use syntax::Edition;
|
||||||
use syntax::{
|
use syntax::{
|
||||||
algo::{self, find_node_at_offset, find_node_at_range},
|
algo::{self, find_node_at_offset, find_node_at_range},
|
||||||
AstNode, AstToken, Direction, SourceFile, SyntaxElement, SyntaxKind, SyntaxToken, TextRange,
|
AstNode, AstToken, Direction, SourceFile, SyntaxElement, SyntaxKind, SyntaxToken, TextRange,
|
||||||
|
@ -94,6 +95,10 @@ impl<'a> AssistContext<'a> {
|
||||||
self.frange.file_id
|
self.frange.file_id
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn edition(&self) -> Edition {
|
||||||
|
self.frange.file_id.edition()
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) fn has_empty_selection(&self) -> bool {
|
pub(crate) fn has_empty_selection(&self) -> bool {
|
||||||
self.trimmed_range.is_empty()
|
self.trimmed_range.is_empty()
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
use ide_db::text_edit::TextRange;
|
|
||||||
use ide_db::{
|
use ide_db::{
|
||||||
assists::{AssistId, AssistKind},
|
assists::{AssistId, AssistKind},
|
||||||
defs::Definition,
|
defs::Definition,
|
||||||
search::{FileReference, SearchScope, UsageSearchResult},
|
search::{FileReference, SearchScope},
|
||||||
|
syntax_helpers::suggest_name,
|
||||||
|
text_edit::TextRange,
|
||||||
};
|
};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
|
use syntax::SmolStr;
|
||||||
use syntax::{
|
use syntax::{
|
||||||
ast::{self, make, AstNode, FieldExpr, HasName, IdentPat},
|
ast::{self, make, AstNode, FieldExpr, HasName, IdentPat},
|
||||||
ted,
|
ted,
|
||||||
|
@ -122,33 +124,43 @@ fn collect_data(ident_pat: IdentPat, ctx: &AssistContext<'_>) -> Option<TupleDat
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
let name = ident_pat.name()?.to_string();
|
let usages = ctx.sema.to_def(&ident_pat).and_then(|def| {
|
||||||
|
|
||||||
let usages = ctx.sema.to_def(&ident_pat).map(|def| {
|
|
||||||
Definition::Local(def)
|
Definition::Local(def)
|
||||||
.usages(&ctx.sema)
|
.usages(&ctx.sema)
|
||||||
.in_scope(&SearchScope::single_file(ctx.file_id()))
|
.in_scope(&SearchScope::single_file(ctx.file_id()))
|
||||||
.all()
|
.all()
|
||||||
|
.iter()
|
||||||
|
.next()
|
||||||
|
.map(|(_, refs)| refs.to_vec())
|
||||||
});
|
});
|
||||||
|
|
||||||
let field_names = (0..field_types.len())
|
let mut name_generator = {
|
||||||
.map(|i| generate_name(ctx, i, &name, &ident_pat, &usages))
|
let mut names = vec![];
|
||||||
|
if let Some(scope) = ctx.sema.scope(ident_pat.syntax()) {
|
||||||
|
scope.process_all_names(&mut |name, scope| {
|
||||||
|
if let hir::ScopeDef::Local(_) = scope {
|
||||||
|
names.push(name.as_str().into())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
suggest_name::NameGenerator::new_with_names(names.iter().map(|s: &SmolStr| s.as_str()))
|
||||||
|
};
|
||||||
|
|
||||||
|
let field_names = field_types
|
||||||
|
.into_iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(|(id, ty)| {
|
||||||
|
match name_generator.for_type(&ty, ctx.db(), ctx.edition()) {
|
||||||
|
Some(name) => name,
|
||||||
|
None => name_generator.suggest_name(&format!("_{}", id)),
|
||||||
|
}
|
||||||
|
.to_string()
|
||||||
|
})
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
Some(TupleData { ident_pat, ref_type, field_names, usages })
|
Some(TupleData { ident_pat, ref_type, field_names, usages })
|
||||||
}
|
}
|
||||||
|
|
||||||
fn generate_name(
|
|
||||||
_ctx: &AssistContext<'_>,
|
|
||||||
index: usize,
|
|
||||||
_tuple_name: &str,
|
|
||||||
_ident_pat: &IdentPat,
|
|
||||||
_usages: &Option<UsageSearchResult>,
|
|
||||||
) -> String {
|
|
||||||
// FIXME: detect if name already used
|
|
||||||
format!("_{index}")
|
|
||||||
}
|
|
||||||
|
|
||||||
enum RefType {
|
enum RefType {
|
||||||
ReadOnly,
|
ReadOnly,
|
||||||
Mutable,
|
Mutable,
|
||||||
|
@ -157,7 +169,7 @@ struct TupleData {
|
||||||
ident_pat: IdentPat,
|
ident_pat: IdentPat,
|
||||||
ref_type: Option<RefType>,
|
ref_type: Option<RefType>,
|
||||||
field_names: Vec<String>,
|
field_names: Vec<String>,
|
||||||
usages: Option<UsageSearchResult>,
|
usages: Option<Vec<FileReference>>,
|
||||||
}
|
}
|
||||||
fn edit_tuple_assignment(
|
fn edit_tuple_assignment(
|
||||||
ctx: &AssistContext<'_>,
|
ctx: &AssistContext<'_>,
|
||||||
|
@ -213,9 +225,6 @@ fn edit_tuple_usages(
|
||||||
ctx: &AssistContext<'_>,
|
ctx: &AssistContext<'_>,
|
||||||
in_sub_pattern: bool,
|
in_sub_pattern: bool,
|
||||||
) -> Option<Vec<EditTupleUsage>> {
|
) -> Option<Vec<EditTupleUsage>> {
|
||||||
let mut current_file_usages = None;
|
|
||||||
|
|
||||||
if let Some(usages) = data.usages.as_ref() {
|
|
||||||
// We need to collect edits first before actually applying them
|
// We need to collect edits first before actually applying them
|
||||||
// as mapping nodes to their mutable node versions requires an
|
// as mapping nodes to their mutable node versions requires an
|
||||||
// unmodified syntax tree.
|
// unmodified syntax tree.
|
||||||
|
@ -224,31 +233,15 @@ fn edit_tuple_usages(
|
||||||
// tree mutation in the same file breaks when `builder.edit_file`
|
// tree mutation in the same file breaks when `builder.edit_file`
|
||||||
// is called
|
// is called
|
||||||
|
|
||||||
if let Some((_, refs)) = usages.iter().find(|(file_id, _)| *file_id == ctx.file_id()) {
|
let edits = data
|
||||||
current_file_usages = Some(
|
.usages
|
||||||
refs.iter()
|
.as_ref()?
|
||||||
.filter_map(|r| edit_tuple_usage(ctx, edit, r, data, in_sub_pattern))
|
.as_slice()
|
||||||
.collect_vec(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (file_id, refs) in usages.iter() {
|
|
||||||
if file_id == ctx.file_id() {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
edit.edit_file(file_id.file_id());
|
|
||||||
|
|
||||||
let tuple_edits = refs
|
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|r| edit_tuple_usage(ctx, edit, r, data, in_sub_pattern))
|
.filter_map(|r| edit_tuple_usage(ctx, edit, r, data, in_sub_pattern))
|
||||||
.collect_vec();
|
.collect_vec();
|
||||||
|
|
||||||
tuple_edits.into_iter().for_each(|tuple_edit| tuple_edit.apply(edit))
|
Some(edits)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
current_file_usages
|
|
||||||
}
|
}
|
||||||
fn edit_tuple_usage(
|
fn edit_tuple_usage(
|
||||||
ctx: &AssistContext<'_>,
|
ctx: &AssistContext<'_>,
|
||||||
|
@ -1769,14 +1762,14 @@ struct S4 {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn foo() -> Option<()> {
|
fn foo() -> Option<()> {
|
||||||
let ($0_0, _1, _2, _3, _4, _5) = &(0, (1,"1"), Some(2), [3;3], S4 { value: 4 }, &5);
|
let ($0_0, _1, _2, _3, s4, _5) = &(0, (1,"1"), Some(2), [3;3], S4 { value: 4 }, &5);
|
||||||
let v: i32 = *_0; // deref, no parens
|
let v: i32 = *_0; // deref, no parens
|
||||||
let v: &i32 = _0; // no deref, no parens, remove `&`
|
let v: &i32 = _0; // no deref, no parens, remove `&`
|
||||||
f1(*_0); // deref, no parens
|
f1(*_0); // deref, no parens
|
||||||
f2(_0); // `&*` -> cancel out -> no deref, no parens
|
f2(_0); // `&*` -> cancel out -> no deref, no parens
|
||||||
// https://github.com/rust-lang/rust-analyzer/issues/1109#issuecomment-658868639
|
// https://github.com/rust-lang/rust-analyzer/issues/1109#issuecomment-658868639
|
||||||
// let v: i32 = t.1.0; // no deref, no parens
|
// let v: i32 = t.1.0; // no deref, no parens
|
||||||
let v: i32 = _4.value; // no deref, no parens
|
let v: i32 = s4.value; // no deref, no parens
|
||||||
(*_0).do_stuff(); // deref, parens
|
(*_0).do_stuff(); // deref, parens
|
||||||
let v: i32 = (*_2)?; // deref, parens
|
let v: i32 = (*_2)?; // deref, parens
|
||||||
let v: i32 = _3[0]; // no deref, no parens
|
let v: i32 = _3[0]; // no deref, no parens
|
||||||
|
@ -1815,8 +1808,8 @@ impl S {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let ($0_0, _1) = &(S,2);
|
let ($0s, _1) = &(S,2);
|
||||||
let s = _0.f();
|
let s = s.f();
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
)
|
)
|
||||||
|
@ -1845,8 +1838,8 @@ impl S {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let ($0_0, _1) = &(S,2);
|
let ($0s, _1) = &(S,2);
|
||||||
let s = (*_0).f();
|
let s = (*s).f();
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
)
|
)
|
||||||
|
@ -1882,8 +1875,8 @@ impl T for &S {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let ($0_0, _1) = &(S,2);
|
let ($0s, _1) = &(S,2);
|
||||||
let s = (*_0).f();
|
let s = (*s).f();
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
)
|
)
|
||||||
|
@ -1923,8 +1916,8 @@ impl T for &S {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let ($0_0, _1) = &(S,2);
|
let ($0s, _1) = &(S,2);
|
||||||
let s = (*_0).f();
|
let s = (*s).f();
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
)
|
)
|
||||||
|
@ -1951,8 +1944,8 @@ impl S {
|
||||||
fn do_stuff(&self) -> i32 { 42 }
|
fn do_stuff(&self) -> i32 { 42 }
|
||||||
}
|
}
|
||||||
fn main() {
|
fn main() {
|
||||||
let ($0_0, _1) = &(S,&S);
|
let ($0s, s1) = &(S,&S);
|
||||||
let v = _0.do_stuff();
|
let v = s.do_stuff();
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
)
|
)
|
||||||
|
@ -1973,7 +1966,7 @@ fn main() {
|
||||||
// `t.0` gets auto-refed -> no deref needed -> no parens
|
// `t.0` gets auto-refed -> no deref needed -> no parens
|
||||||
let v = t.0.do_stuff(); // no deref, no parens
|
let v = t.0.do_stuff(); // no deref, no parens
|
||||||
let v = &t.0.do_stuff(); // `&` is for result -> no deref, no parens
|
let v = &t.0.do_stuff(); // `&` is for result -> no deref, no parens
|
||||||
// deref: `_1` is `&&S`, but method called is on `&S` -> there might be a method accepting `&&S`
|
// deref: `s1` is `&&S`, but method called is on `&S` -> there might be a method accepting `&&S`
|
||||||
let v = t.1.do_stuff(); // deref, parens
|
let v = t.1.do_stuff(); // deref, parens
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
|
@ -1984,13 +1977,13 @@ impl S {
|
||||||
fn do_stuff(&self) -> i32 { 42 }
|
fn do_stuff(&self) -> i32 { 42 }
|
||||||
}
|
}
|
||||||
fn main() {
|
fn main() {
|
||||||
let ($0_0, _1) = &(S,&S);
|
let ($0s, s1) = &(S,&S);
|
||||||
let v = _0.do_stuff(); // no deref, remove parens
|
let v = s.do_stuff(); // no deref, remove parens
|
||||||
// `t.0` gets auto-refed -> no deref needed -> no parens
|
// `t.0` gets auto-refed -> no deref needed -> no parens
|
||||||
let v = _0.do_stuff(); // no deref, no parens
|
let v = s.do_stuff(); // no deref, no parens
|
||||||
let v = &_0.do_stuff(); // `&` is for result -> no deref, no parens
|
let v = &s.do_stuff(); // `&` is for result -> no deref, no parens
|
||||||
// deref: `_1` is `&&S`, but method called is on `&S` -> there might be a method accepting `&&S`
|
// deref: `s1` is `&&S`, but method called is on `&S` -> there might be a method accepting `&&S`
|
||||||
let v = (*_1).do_stuff(); // deref, parens
|
let v = (*s1).do_stuff(); // deref, parens
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
)
|
)
|
||||||
|
|
|
@ -377,6 +377,8 @@ fn name_of_type(ty: &hir::Type, db: &RootDatabase, edition: Edition) -> Option<S
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
name
|
name
|
||||||
|
} else if let Some(inner_ty) = ty.remove_ref() {
|
||||||
|
return name_of_type(&inner_ty, db, edition);
|
||||||
} else {
|
} else {
|
||||||
return None;
|
return None;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in a new issue