mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-19 00:23:58 +00:00
Auto merge of #16114 - roife:fix-inline-with-self-type, r=Veykril
fix: self type replacement in inline-function Fix #16113, fix #16091 The problem described in this issue actually involves three bugs. Firstly, when using `ted` to modify the syntax tree, the offset of nodes on the tree changes, which causes the syntax range information from `hir` to become invalid. Therefore, we need to edit the AST after the last usage for `usages_for_locals`. The second issue is that when inserting nodes, it's necessary to use `clone_subtree` for duplication because the `ted::replace` operation essentially moves a node. The third issue is that we should use `ancestors_with_macros` instead of `ancestors` to handle impl definition in macros. I have fixed the three bugs mentioned above and added unit tests.
This commit is contained in:
commit
306defaef4
1 changed files with 123 additions and 11 deletions
|
@ -315,17 +315,6 @@ fn inline(
|
||||||
} else {
|
} else {
|
||||||
fn_body.clone_for_update()
|
fn_body.clone_for_update()
|
||||||
};
|
};
|
||||||
if let Some(imp) = body.syntax().ancestors().find_map(ast::Impl::cast) {
|
|
||||||
if !node.syntax().ancestors().any(|anc| &anc == imp.syntax()) {
|
|
||||||
if let Some(t) = imp.self_ty() {
|
|
||||||
body.syntax()
|
|
||||||
.descendants_with_tokens()
|
|
||||||
.filter_map(NodeOrToken::into_token)
|
|
||||||
.filter(|tok| tok.kind() == SyntaxKind::SELF_TYPE_KW)
|
|
||||||
.for_each(|tok| ted::replace(tok, t.syntax()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let usages_for_locals = |local| {
|
let usages_for_locals = |local| {
|
||||||
Definition::Local(local)
|
Definition::Local(local)
|
||||||
.usages(sema)
|
.usages(sema)
|
||||||
|
@ -381,6 +370,27 @@ fn inline(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We should place the following code after last usage of `usages_for_locals`
|
||||||
|
// because `ted::replace` will change the offset in syntax tree, which makes
|
||||||
|
// `FileReference` incorrect
|
||||||
|
if let Some(imp) =
|
||||||
|
sema.ancestors_with_macros(fn_body.syntax().clone()).find_map(ast::Impl::cast)
|
||||||
|
{
|
||||||
|
if !node.syntax().ancestors().any(|anc| &anc == imp.syntax()) {
|
||||||
|
if let Some(t) = imp.self_ty() {
|
||||||
|
while let Some(self_tok) = body
|
||||||
|
.syntax()
|
||||||
|
.descendants_with_tokens()
|
||||||
|
.filter_map(NodeOrToken::into_token)
|
||||||
|
.find(|tok| tok.kind() == SyntaxKind::SELF_TYPE_KW)
|
||||||
|
{
|
||||||
|
let replace_with = t.clone_subtree().syntax().clone_for_update();
|
||||||
|
ted::replace(self_tok, replace_with);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let mut func_let_vars: BTreeSet<String> = BTreeSet::new();
|
let mut func_let_vars: BTreeSet<String> = BTreeSet::new();
|
||||||
|
|
||||||
// grab all of the local variable declarations in the function
|
// grab all of the local variable declarations in the function
|
||||||
|
@ -1510,4 +1520,106 @@ fn main() {
|
||||||
"#,
|
"#,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn inline_call_with_multiple_self_types_eq() {
|
||||||
|
check_assist(
|
||||||
|
inline_call,
|
||||||
|
r#"
|
||||||
|
#[derive(PartialEq, Eq)]
|
||||||
|
enum Enum {
|
||||||
|
A,
|
||||||
|
B,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Enum {
|
||||||
|
fn a_or_b_eq(&self) -> bool {
|
||||||
|
self == &Self::A || self == &Self::B
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn a() -> bool {
|
||||||
|
Enum::A.$0a_or_b_eq()
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
r#"
|
||||||
|
#[derive(PartialEq, Eq)]
|
||||||
|
enum Enum {
|
||||||
|
A,
|
||||||
|
B,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Enum {
|
||||||
|
fn a_or_b_eq(&self) -> bool {
|
||||||
|
self == &Self::A || self == &Self::B
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn a() -> bool {
|
||||||
|
{
|
||||||
|
let ref this = Enum::A;
|
||||||
|
this == &Enum::A || this == &Enum::B
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn inline_call_with_self_type_in_macros() {
|
||||||
|
check_assist(
|
||||||
|
inline_call,
|
||||||
|
r#"
|
||||||
|
trait Trait<T1> {
|
||||||
|
fn f(a: T1) -> Self;
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! impl_from {
|
||||||
|
($t: ty) => {
|
||||||
|
impl Trait<$t> for $t {
|
||||||
|
fn f(a: $t) -> Self {
|
||||||
|
a as Self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
struct A {}
|
||||||
|
|
||||||
|
impl_from!(A);
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let a: A = A{};
|
||||||
|
let b = <A as Trait<A>>::$0f(a);
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
r#"
|
||||||
|
trait Trait<T1> {
|
||||||
|
fn f(a: T1) -> Self;
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! impl_from {
|
||||||
|
($t: ty) => {
|
||||||
|
impl Trait<$t> for $t {
|
||||||
|
fn f(a: $t) -> Self {
|
||||||
|
a as Self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
struct A {}
|
||||||
|
|
||||||
|
impl_from!(A);
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let a: A = A{};
|
||||||
|
let b = {
|
||||||
|
let a = a;
|
||||||
|
a as A
|
||||||
|
};
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue