Cleanup auto-ref in completion

This commit is contained in:
Aleksey Kladov 2021-03-09 18:06:08 +03:00
parent abc0ed36bd
commit 12fe301a0c
4 changed files with 87 additions and 30 deletions

View file

@ -1614,10 +1614,9 @@ impl Type {
} }
pub fn remove_ref(&self) -> Option<Type> { pub fn remove_ref(&self) -> Option<Type> {
if let Ty::Ref(.., substs) = &self.ty.value { match &self.ty.value {
Some(self.derived(substs[0].clone())) Ty::Ref(.., substs) => Some(self.derived(substs[0].clone())),
} else { _ => None,
None
} }
} }

View file

@ -68,7 +68,7 @@ pub struct CompletionItem {
/// Indicates that a reference or mutable reference to this variable is a /// Indicates that a reference or mutable reference to this variable is a
/// possible match. /// possible match.
ref_match: Option<(Mutability, CompletionScore)>, ref_match: Option<Mutability>,
/// The import data to add to completion's edits. /// The import data to add to completion's edits.
import_to_add: Option<ImportEdit>, import_to_add: Option<ImportEdit>,
@ -104,6 +104,9 @@ impl fmt::Debug for CompletionItem {
if let Some(score) = &self.score { if let Some(score) = &self.score {
s.field("score", score); s.field("score", score);
} }
if let Some(mutability) = &self.ref_match {
s.field("ref_match", &format!("&{}", mutability.as_keyword_for_ref()));
}
if self.trigger_call_info { if self.trigger_call_info {
s.field("trigger_call_info", &true); s.field("trigger_call_info", &true);
} }
@ -261,7 +264,7 @@ impl CompletionItem {
self.trigger_call_info self.trigger_call_info
} }
pub fn ref_match(&self) -> Option<(Mutability, CompletionScore)> { pub fn ref_match(&self) -> Option<Mutability> {
self.ref_match self.ref_match
} }
@ -311,7 +314,7 @@ pub(crate) struct Builder {
deprecated: bool, deprecated: bool,
trigger_call_info: Option<bool>, trigger_call_info: Option<bool>,
score: Option<CompletionScore>, score: Option<CompletionScore>,
ref_match: Option<(Mutability, CompletionScore)>, ref_match: Option<Mutability>,
} }
impl Builder { impl Builder {
@ -430,8 +433,8 @@ impl Builder {
self.import_to_add = import_to_add; self.import_to_add = import_to_add;
self self
} }
pub(crate) fn ref_match(mut self, ref_match: (Mutability, CompletionScore)) -> Builder { pub(crate) fn ref_match(mut self, mutability: Mutability) -> Builder {
self.ref_match = Some(ref_match); self.ref_match = Some(mutability);
self self
} }
} }

View file

@ -254,10 +254,17 @@ impl<'a> Render<'a> {
{ {
item = item.set_score(score); item = item.set_score(score);
} }
if let Some(ref_match) =
refed_type_matches(&active_type, &active_name, &ty, &local_name) if let Some(ty_without_ref) = active_type.remove_ref() {
{ if ty_without_ref == ty {
item = item.ref_match(ref_match); cov_mark::hit!(suggest_ref);
let mutability = if active_type.is_mutable_reference() {
Mutability::Mut
} else {
Mutability::Shared
};
item = item.ref_match(mutability)
}
} }
} }
} }
@ -340,19 +347,6 @@ fn compute_score_from_active(
Some(res) Some(res)
} }
fn refed_type_matches(
active_type: &Type,
active_name: &str,
ty: &Type,
name: &str,
) -> Option<(Mutability, CompletionScore)> {
let derefed_active = active_type.remove_ref()?;
let score = compute_score_from_active(&derefed_active, &active_name, &ty, &name)?;
Some((
if active_type.is_mutable_reference() { Mutability::Mut } else { Mutability::Shared },
score,
))
}
fn compute_score(ctx: &RenderContext, ty: &Type, name: &str) -> Option<CompletionScore> { fn compute_score(ctx: &RenderContext, ty: &Type, name: &str) -> Option<CompletionScore> {
let (active_name, active_type) = ctx.active_name_and_type()?; let (active_name, active_type) = ctx.active_name_and_type()?;
@ -947,4 +941,66 @@ fn f(foo: &Foo) { f(foo, w$0) }
"#]], "#]],
); );
} }
#[test]
fn suggest_ref_mut() {
cov_mark::check!(suggest_ref);
check(
r#"
struct S;
fn foo(s: &mut S) {}
fn main() {
let mut s = S;
foo($0);
}
"#,
expect![[r#"
[
CompletionItem {
label: "S",
source_range: 70..70,
delete: 70..70,
insert: "S",
kind: SymbolKind(
Struct,
),
},
CompletionItem {
label: "foo(…)",
source_range: 70..70,
delete: 70..70,
insert: "foo(${1:&mut s})$0",
kind: SymbolKind(
Function,
),
lookup: "foo",
detail: "-> ()",
trigger_call_info: true,
},
CompletionItem {
label: "main()",
source_range: 70..70,
delete: 70..70,
insert: "main()$0",
kind: SymbolKind(
Function,
),
lookup: "main",
detail: "-> ()",
},
CompletionItem {
label: "s",
source_range: 70..70,
delete: 70..70,
insert: "s",
kind: SymbolKind(
Local,
),
detail: "S",
ref_match: "&mut ",
},
]
"#]],
)
}
} }

View file

@ -232,9 +232,8 @@ pub(crate) fn completion_item(
} }
let mut res = match item.ref_match() { let mut res = match item.ref_match() {
Some(ref_match) => { Some(mutability) => {
let mut refed = lsp_item.clone(); let mut refed = lsp_item.clone();
let (mutability, _score) = ref_match;
let label = format!("&{}{}", mutability.as_keyword_for_ref(), refed.label); let label = format!("&{}{}", mutability.as_keyword_for_ref(), refed.label);
set_score(&mut refed, &label); set_score(&mut refed, &label);
refed.label = label; refed.label = label;
@ -243,8 +242,8 @@ pub(crate) fn completion_item(
None => vec![lsp_item], None => vec![lsp_item],
}; };
for mut r in res.iter_mut() { for lsp_item in res.iter_mut() {
r.insert_text_format = Some(insert_text_format(item.insert_text_format())); lsp_item.insert_text_format = Some(insert_text_format(item.insert_text_format()));
} }
res res
} }