rust-analyzer/crates/ide_completion/src/item.rs

614 lines
20 KiB
Rust
Raw Normal View History

//! See `CompletionItem` structure.
use std::fmt;
2021-02-24 23:06:31 +00:00
use hir::{Documentation, Mutability};
use ide_db::{
helpers::{
2021-02-24 23:06:31 +00:00
import_assets::LocatedImport,
2021-03-06 11:02:26 +00:00
insert_use::{self, ImportScope, InsertUseConfig},
mod_path_to_ast, SnippetCap,
},
SymbolKind,
2020-12-02 21:55:35 +00:00
};
2021-03-03 21:55:21 +00:00
use stdx::{format_to, impl_from, never};
2020-12-04 08:02:22 +00:00
use syntax::{algo, TextRange};
2020-08-12 15:03:06 +00:00
use text_edit::TextEdit;
2019-01-08 19:33:36 +00:00
/// `CompletionItem` describes a single completion variant in the editor pop-up.
/// It is basically a POD with various properties. To construct a
/// `CompletionItem`, use `new` method and the `Builder` struct.
#[derive(Clone)]
2019-01-08 19:33:36 +00:00
pub struct CompletionItem {
/// Used only internally in tests, to check only specific kind of
/// completion (postfix, keyword, reference, etc).
#[allow(unused)]
2020-03-11 09:46:43 +00:00
pub(crate) completion_kind: CompletionKind,
/// Label in the completion pop up which identifies completion.
2019-01-08 19:33:36 +00:00
label: String,
/// Range of identifier that is being completed.
///
/// It should be used primarily for UI, but we also use this to convert
/// generic TextEdit into LSP's completion edit (see conv.rs).
///
/// `source_range` must contain the completion offset. `insert_text` should
/// start with what `source_range` points to, or VSCode will filter out the
/// completion silently.
source_range: TextRange,
/// What happens when user selects this item.
///
/// Typically, replaces `source_range` with new identifier.
text_edit: TextEdit,
2020-09-18 20:40:11 +00:00
insert_text_format: InsertTextFormat,
/// What item (struct, function, etc) are we completing.
kind: Option<CompletionItemKind>,
/// Lookup is used to check if completion item indeed can complete current
/// ident.
///
2021-01-06 20:15:48 +00:00
/// That is, in `foo.bar$0` lookup of `abracadabra` will be accepted (it
/// contains `bar` sub sequence), and `quux` will rejected.
lookup: Option<String>,
/// Additional info to show in the UI pop up.
2019-01-09 15:09:49 +00:00
detail: Option<String>,
documentation: Option<Documentation>,
/// Whether this item is marked as deprecated
deprecated: bool,
/// If completing a function call, ask the editor to show parameter popup
/// after completion.
trigger_call_info: bool,
2021-03-09 17:24:09 +00:00
/// We use this to sort completion. Relevance records facts like "do the
/// types align precisely?". We can't sort by relevances directly, they are
/// only partially ordered.
///
/// Note that Relevance ignores fuzzy match score. We compute Relevance for
/// all possible items, and then separately build an ordered completion list
/// based on relevance and fuzzy matching with the already typed identifier.
2021-03-12 03:11:14 +00:00
relevance: CompletionRelevance,
2020-09-29 20:24:56 +00:00
/// Indicates that a reference or mutable reference to this variable is a
/// possible match.
2021-03-09 15:06:08 +00:00
ref_match: Option<Mutability>,
2020-11-28 14:26:30 +00:00
2020-12-02 22:13:32 +00:00
/// The import data to add to completion's edits.
2020-12-03 09:13:28 +00:00
import_to_add: Option<ImportEdit>,
2019-01-08 19:33:36 +00:00
}
// We use custom debug for CompletionItem to make snapshot tests more readable.
impl fmt::Debug for CompletionItem {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let mut s = f.debug_struct("CompletionItem");
2019-02-18 09:23:31 +00:00
s.field("label", &self.label()).field("source_range", &self.source_range());
2020-05-21 13:56:18 +00:00
if self.text_edit().len() == 1 {
let atom = &self.text_edit().iter().next().unwrap();
2019-02-18 09:23:31 +00:00
s.field("delete", &atom.delete);
s.field("insert", &atom.insert);
} else {
s.field("text_edit", &self.text_edit);
}
if let Some(kind) = self.kind().as_ref() {
s.field("kind", kind);
}
if self.lookup() != self.label() {
s.field("lookup", &self.lookup());
}
if let Some(detail) = self.detail() {
s.field("detail", &detail);
}
if let Some(documentation) = self.documentation() {
s.field("documentation", &documentation);
}
if self.deprecated {
s.field("deprecated", &true);
}
2021-03-12 03:11:14 +00:00
if self.relevance != CompletionRelevance::default() {
2021-03-09 17:24:09 +00:00
s.field("relevance", &self.relevance);
}
2021-03-12 03:11:14 +00:00
2021-03-09 15:06:08 +00:00
if let Some(mutability) = &self.ref_match {
s.field("ref_match", &format!("&{}", mutability.as_keyword_for_ref()));
}
2020-03-12 18:02:55 +00:00
if self.trigger_call_info {
s.field("trigger_call_info", &true);
}
s.finish()
}
}
#[derive(Debug, Clone, Copy, Eq, PartialEq, Default)]
2021-03-12 03:11:14 +00:00
pub struct CompletionRelevance {
2021-03-09 17:24:09 +00:00
/// This is set in cases like these:
///
/// ```
/// fn f(spam: String) {}
/// fn main {
/// let spam = 92;
/// f($0) // name of local matches the name of param
/// }
/// ```
pub exact_name_match: bool,
/// See CompletionRelevanceTypeMatch doc comments for cases where this is set.
pub type_match: Option<CompletionRelevanceTypeMatch>,
2021-03-09 17:24:09 +00:00
/// This is set in cases like these:
///
/// ```
/// fn foo(a: u32) {
/// let b = 0;
/// $0 // `a` and `b` are local
2021-03-09 17:24:09 +00:00
/// }
/// ```
pub is_local: bool,
}
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
pub enum CompletionRelevanceTypeMatch {
/// This is set in cases like these:
///
/// ```
/// enum Option<T> { Some(T), None }
/// fn f(a: Option<u32>) {}
/// fn main {
/// f(Option::N$0) // type `Option<T>` could unify with `Option<u32>`
/// }
/// ```
CouldUnify,
/// This is set in cases like these:
///
/// ```
/// fn f(spam: String) {}
/// fn main {
/// let foo = String::new();
/// f($0) // type of local matches the type of param
/// }
/// ```
Exact,
2021-03-09 17:24:09 +00:00
}
2021-03-12 03:11:14 +00:00
impl CompletionRelevance {
/// Provides a relevance score. Higher values are more relevant.
///
/// The absolute value of the relevance score is not meaningful, for
/// example a value of 0 doesn't mean "not relevant", rather
/// it means "least relevant". The score value should only be used
/// for relative ordering.
///
/// See is_relevant if you need to make some judgement about score
/// in an absolute sense.
2021-03-12 14:08:07 +00:00
pub fn score(&self) -> u32 {
2021-03-12 03:11:14 +00:00
let mut score = 0;
if self.exact_name_match {
score += 1;
}
score += match self.type_match {
Some(CompletionRelevanceTypeMatch::Exact) => 4,
Some(CompletionRelevanceTypeMatch::CouldUnify) => 3,
None => 0,
};
if self.is_local {
2021-03-12 03:11:14 +00:00
score += 1;
}
score
}
/// Returns true when the score for this threshold is above
/// some threshold such that we think it is especially likely
/// to be relevant.
2021-03-09 17:24:09 +00:00
pub fn is_relevant(&self) -> bool {
2021-03-12 03:11:14 +00:00
self.score() > 0
2021-03-09 17:24:09 +00:00
}
}
2019-01-08 19:33:36 +00:00
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum CompletionItemKind {
SymbolKind(SymbolKind),
Attribute,
Binding,
2021-01-20 17:46:14 +00:00
BuiltinType,
Keyword,
Method,
Snippet,
UnresolvedReference,
2019-01-08 19:33:36 +00:00
}
impl_from!(SymbolKind for CompletionItemKind);
2020-06-12 18:30:57 +00:00
impl CompletionItemKind {
2020-06-13 11:47:30 +00:00
#[cfg(test)]
pub(crate) fn tag(&self) -> &'static str {
match self {
CompletionItemKind::SymbolKind(kind) => match kind {
SymbolKind::Const => "ct",
SymbolKind::ConstParam => "cp",
SymbolKind::Enum => "en",
SymbolKind::Field => "fd",
SymbolKind::Function => "fn",
SymbolKind::Impl => "im",
SymbolKind::Label => "lb",
SymbolKind::LifetimeParam => "lt",
SymbolKind::Local => "lc",
SymbolKind::Macro => "ma",
SymbolKind::Module => "md",
SymbolKind::SelfParam => "sp",
SymbolKind::Static => "sc",
SymbolKind::Struct => "st",
SymbolKind::Trait => "tt",
SymbolKind::TypeAlias => "ta",
SymbolKind::TypeParam => "tp",
SymbolKind::Union => "un",
SymbolKind::ValueParam => "vp",
SymbolKind::Variant => "ev",
},
2020-07-03 09:48:48 +00:00
CompletionItemKind::Attribute => "at",
CompletionItemKind::Binding => "bn",
2020-06-12 18:30:57 +00:00
CompletionItemKind::BuiltinType => "bt",
2020-07-03 09:48:48 +00:00
CompletionItemKind::Keyword => "kw",
CompletionItemKind::Method => "me",
CompletionItemKind::Snippet => "sn",
CompletionItemKind::UnresolvedReference => "??",
2020-06-13 11:47:30 +00:00
}
2020-06-12 18:30:57 +00:00
}
}
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
2019-01-08 19:33:36 +00:00
pub(crate) enum CompletionKind {
/// Parser-based keyword completion.
Keyword,
/// Your usual "complete all valid identifiers".
Reference,
/// "Secret sauce" completions.
Magic,
Snippet,
2019-01-21 05:19:51 +00:00
Postfix,
2019-05-30 13:10:07 +00:00
BuiltinType,
2020-04-23 16:22:33 +00:00
Attribute,
2019-01-08 19:33:36 +00:00
}
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
pub enum InsertTextFormat {
PlainText,
Snippet,
}
2019-01-08 19:33:36 +00:00
impl CompletionItem {
pub(crate) fn new(
completion_kind: CompletionKind,
source_range: TextRange,
label: impl Into<String>,
) -> Builder {
2019-01-08 19:33:36 +00:00
let label = label.into();
Builder {
source_range,
2019-01-08 19:33:36 +00:00
completion_kind,
label,
insert_text: None,
insert_text_format: InsertTextFormat::PlainText,
2019-01-09 15:09:49 +00:00
detail: None,
documentation: None,
2019-01-08 19:33:36 +00:00
lookup: None,
kind: None,
text_edit: None,
2021-03-09 14:44:27 +00:00
deprecated: false,
trigger_call_info: None,
2021-03-12 03:11:14 +00:00
relevance: CompletionRelevance::default(),
2020-09-29 20:24:56 +00:00
ref_match: None,
2020-11-27 10:22:10 +00:00
import_to_add: None,
}
}
2019-01-08 19:33:36 +00:00
/// What user sees in pop-up in the UI.
pub fn label(&self) -> &str {
&self.label
}
pub fn source_range(&self) -> TextRange {
self.source_range
}
pub fn insert_text_format(&self) -> InsertTextFormat {
self.insert_text_format
}
pub fn text_edit(&self) -> &TextEdit {
&self.text_edit
}
2019-01-09 15:09:49 +00:00
/// Short one-line additional information, like a type
pub fn detail(&self) -> Option<&str> {
2020-02-18 13:32:19 +00:00
self.detail.as_deref()
2019-01-09 15:09:49 +00:00
}
/// A doc-comment
pub fn documentation(&self) -> Option<Documentation> {
self.documentation.clone()
}
2019-01-08 19:33:36 +00:00
/// What string is used for filtering.
pub fn lookup(&self) -> &str {
2020-04-23 23:17:33 +00:00
self.lookup.as_deref().unwrap_or(&self.label)
2019-01-08 19:33:36 +00:00
}
2019-01-08 19:33:36 +00:00
pub fn kind(&self) -> Option<CompletionItemKind> {
self.kind
}
pub fn deprecated(&self) -> bool {
self.deprecated
}
2021-03-12 03:11:14 +00:00
pub fn relevance(&self) -> CompletionRelevance {
2021-03-09 17:24:09 +00:00
self.relevance
}
pub fn trigger_call_info(&self) -> bool {
self.trigger_call_info
}
2020-09-29 20:24:56 +00:00
2021-03-12 03:11:14 +00:00
pub fn ref_match(&self) -> Option<(Mutability, CompletionRelevance)> {
// Relevance of the ref match should be the same as the original
// match, but with exact type match set because self.ref_match
// is only set if there is an exact type match.
let mut relevance = self.relevance;
relevance.type_match = Some(CompletionRelevanceTypeMatch::Exact);
2021-03-12 03:11:14 +00:00
self.ref_match.map(|mutability| (mutability, relevance))
2020-09-29 20:24:56 +00:00
}
2020-12-03 09:13:28 +00:00
pub fn import_to_add(&self) -> Option<&ImportEdit> {
self.import_to_add.as_ref()
}
2019-01-08 19:33:36 +00:00
}
2020-11-27 10:22:10 +00:00
/// An extra import to add after the completion is applied.
#[derive(Debug, Clone)]
2020-12-03 09:13:28 +00:00
pub struct ImportEdit {
2021-02-24 23:06:31 +00:00
pub import: LocatedImport,
2021-03-03 21:55:21 +00:00
pub scope: ImportScope,
2020-11-27 10:22:10 +00:00
}
2020-12-03 09:13:28 +00:00
impl ImportEdit {
/// Attempts to insert the import to the given scope, producing a text edit.
/// May return no edit in edge cases, such as scope already containing the import.
2021-03-06 11:02:26 +00:00
pub fn to_text_edit(&self, cfg: InsertUseConfig) -> Option<TextEdit> {
2020-12-04 18:02:42 +00:00
let _p = profile::span("ImportEdit::to_text_edit");
2020-12-03 09:13:28 +00:00
let new_ast = self.scope.clone_for_update();
insert_use::insert_use(&new_ast, mod_path_to_ast(&self.import.import_path), &cfg);
2020-12-03 09:13:28 +00:00
let mut import_insert = TextEdit::builder();
algo::diff(self.scope.as_syntax_node(), new_ast.as_syntax_node())
.into_text_edit(&mut import_insert);
2020-12-03 09:13:28 +00:00
Some(import_insert.finish())
}
}
2019-01-08 19:33:36 +00:00
/// A helper to make `CompletionItem`s.
#[must_use]
2020-09-29 20:24:56 +00:00
#[derive(Clone)]
pub(crate) struct Builder {
source_range: TextRange,
2019-01-08 19:33:36 +00:00
completion_kind: CompletionKind,
2020-12-03 09:13:28 +00:00
import_to_add: Option<ImportEdit>,
2019-01-08 19:33:36 +00:00
label: String,
insert_text: Option<String>,
insert_text_format: InsertTextFormat,
2019-01-09 15:09:49 +00:00
detail: Option<String>,
documentation: Option<Documentation>,
2019-01-08 19:33:36 +00:00
lookup: Option<String>,
kind: Option<CompletionItemKind>,
text_edit: Option<TextEdit>,
2021-03-09 14:44:27 +00:00
deprecated: bool,
trigger_call_info: Option<bool>,
2021-03-12 03:11:14 +00:00
relevance: CompletionRelevance,
2021-03-09 15:06:08 +00:00
ref_match: Option<Mutability>,
2019-01-08 19:33:36 +00:00
}
impl Builder {
2019-01-08 19:33:36 +00:00
pub(crate) fn build(self) -> CompletionItem {
2020-11-27 16:00:03 +00:00
let _p = profile::span("item::Builder::build");
2021-03-03 21:55:21 +00:00
let mut label = self.label;
let mut lookup = self.lookup;
let mut insert_text = self.insert_text;
if let Some(original_path) = self
.import_to_add
.as_ref()
.and_then(|import_edit| import_edit.import.original_path.as_ref())
{
lookup = lookup.or_else(|| Some(label.clone()));
insert_text = insert_text.or_else(|| Some(label.clone()));
let original_path_label = original_path.to_string();
if original_path_label.ends_with(&label) {
label = original_path_label;
} else {
format_to!(label, " ({})", original_path)
}
}
let text_edit = match self.text_edit {
Some(it) => it,
None => {
TextEdit::replace(self.source_range, insert_text.unwrap_or_else(|| label.clone()))
}
};
2019-01-08 19:33:36 +00:00
CompletionItem {
source_range: self.source_range,
label,
insert_text_format: self.insert_text_format,
2020-12-03 09:13:28 +00:00
text_edit,
2019-01-09 15:09:49 +00:00
detail: self.detail,
documentation: self.documentation,
lookup,
2019-01-08 19:33:36 +00:00
kind: self.kind,
completion_kind: self.completion_kind,
2021-03-09 14:44:27 +00:00
deprecated: self.deprecated,
trigger_call_info: self.trigger_call_info.unwrap_or(false),
2021-03-09 17:24:09 +00:00
relevance: self.relevance,
2020-09-29 20:24:56 +00:00
ref_match: self.ref_match,
import_to_add: self.import_to_add,
2019-01-08 19:33:36 +00:00
}
}
pub(crate) fn lookup_by(&mut self, lookup: impl Into<String>) -> &mut Builder {
2019-01-08 19:33:36 +00:00
self.lookup = Some(lookup.into());
self
}
pub(crate) fn label(&mut self, label: impl Into<String>) -> &mut Builder {
self.label = label.into();
self
}
pub(crate) fn insert_text(&mut self, insert_text: impl Into<String>) -> &mut Builder {
self.insert_text = Some(insert_text.into());
self
}
pub(crate) fn insert_snippet(
&mut self,
_cap: SnippetCap,
snippet: impl Into<String>,
) -> &mut Builder {
self.insert_text_format = InsertTextFormat::Snippet;
self.insert_text(snippet)
}
pub(crate) fn kind(&mut self, kind: impl Into<CompletionItemKind>) -> &mut Builder {
self.kind = Some(kind.into());
2019-01-08 19:33:36 +00:00
self
}
pub(crate) fn text_edit(&mut self, edit: TextEdit) -> &mut Builder {
self.text_edit = Some(edit);
self
}
pub(crate) fn snippet_edit(&mut self, _cap: SnippetCap, edit: TextEdit) -> &mut Builder {
self.insert_text_format = InsertTextFormat::Snippet;
self.text_edit(edit)
}
pub(crate) fn detail(&mut self, detail: impl Into<String>) -> &mut Builder {
2019-01-09 15:46:02 +00:00
self.set_detail(Some(detail))
}
pub(crate) fn set_detail(&mut self, detail: Option<impl Into<String>>) -> &mut Builder {
2019-01-09 15:46:02 +00:00
self.detail = detail.map(Into::into);
if let Some(detail) = &self.detail {
2021-01-26 19:11:12 +00:00
if never!(detail.contains('\n'), "multiline detail:\n{}", detail) {
2021-01-18 12:58:10 +00:00
self.detail = Some(detail.splitn(2, '\n').next().unwrap().to_string());
}
}
2019-01-09 15:09:49 +00:00
self
}
#[allow(unused)]
pub(crate) fn documentation(&mut self, docs: Documentation) -> &mut Builder {
self.set_documentation(Some(docs))
}
pub(crate) fn set_documentation(&mut self, docs: Option<Documentation>) -> &mut Builder {
self.documentation = docs.map(Into::into);
self
}
pub(crate) fn set_deprecated(&mut self, deprecated: bool) -> &mut Builder {
2021-03-09 14:44:27 +00:00
self.deprecated = deprecated;
self
}
2021-03-12 03:11:14 +00:00
pub(crate) fn set_relevance(&mut self, relevance: CompletionRelevance) -> &mut Builder {
2021-03-09 17:24:09 +00:00
self.relevance = relevance;
self
}
pub(crate) fn trigger_call_info(&mut self) -> &mut Builder {
self.trigger_call_info = Some(true);
self
}
pub(crate) fn add_import(&mut self, import_to_add: Option<ImportEdit>) -> &mut Builder {
2020-11-27 10:22:10 +00:00
self.import_to_add = import_to_add;
self
}
pub(crate) fn ref_match(&mut self, mutability: Mutability) -> &mut Builder {
2021-03-09 15:06:08 +00:00
self.ref_match = Some(mutability);
2020-09-29 20:24:56 +00:00
self
}
2019-01-08 19:33:36 +00:00
}
2021-03-12 04:39:25 +00:00
#[cfg(test)]
mod tests {
use itertools::Itertools;
use test_utils::assert_eq_text;
use super::{CompletionRelevance, CompletionRelevanceTypeMatch};
2021-03-12 04:39:25 +00:00
/// Check that these are CompletionRelevance are sorted in ascending order
/// by their relevance score.
///
/// We want to avoid making assertions about the absolute score of any
/// item, but we do want to assert whether each is >, <, or == to the
/// others.
///
/// If provided vec![vec![a], vec![b, c], vec![d]], then this will assert:
/// a.score < b.score == c.score < d.score
fn check_relevance_score_ordered(expected_relevance_order: Vec<Vec<CompletionRelevance>>) {
let expected = format!("{:#?}", &expected_relevance_order);
let actual_relevance_order = expected_relevance_order
.into_iter()
.flatten()
.map(|r| (r.score(), r))
.sorted_by_key(|(score, _r)| *score)
.fold(
2021-03-12 14:08:07 +00:00
(u32::MIN, vec![vec![]]),
2021-03-12 04:39:25 +00:00
|(mut currently_collecting_score, mut out), (score, r)| {
if currently_collecting_score == score {
out.last_mut().unwrap().push(r);
} else {
currently_collecting_score = score;
out.push(vec![r]);
}
(currently_collecting_score, out)
},
)
.1;
let actual = format!("{:#?}", &actual_relevance_order);
assert_eq_text!(&expected, &actual);
}
#[test]
fn relevance_score() {
// This test asserts that the relevance score for these items is ascending, and
// that any items in the same vec have the same score.
let expected_relevance_order = vec![
vec![CompletionRelevance::default()],
vec![
CompletionRelevance { exact_name_match: true, ..CompletionRelevance::default() },
CompletionRelevance { is_local: true, ..CompletionRelevance::default() },
2021-03-12 04:39:25 +00:00
],
vec![CompletionRelevance {
exact_name_match: true,
is_local: true,
..CompletionRelevance::default()
}],
vec![CompletionRelevance {
type_match: Some(CompletionRelevanceTypeMatch::CouldUnify),
..CompletionRelevance::default()
}],
vec![CompletionRelevance {
type_match: Some(CompletionRelevanceTypeMatch::Exact),
..CompletionRelevance::default()
}],
vec![CompletionRelevance {
exact_name_match: true,
type_match: Some(CompletionRelevanceTypeMatch::Exact),
..CompletionRelevance::default()
}],
vec![CompletionRelevance {
exact_name_match: true,
type_match: Some(CompletionRelevanceTypeMatch::Exact),
is_local: true,
}],
2021-03-12 04:39:25 +00:00
];
check_relevance_score_ordered(expected_relevance_order);
}
}