//! See `CompletionItem` structure. use std::fmt; use hir::{Documentation, ModPath, Mutability}; use ide_helpers::{ insert_use::{self, ImportScope, MergeBehaviour}, mod_path_to_ast, }; use syntax::{algo, TextRange}; use text_edit::TextEdit; use crate::config::SnippetCap; /// `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. pub struct CompletionItem { /// Used only internally in tests, to check only specific kind of /// completion (postfix, keyword, reference, etc). #[allow(unused)] pub(crate) completion_kind: CompletionKind, /// Label in the completion pop up which identifies completion. label: String, /// Range of identifier that is being completed. /// /// It should be used primarily for UI, but we also use this to convert /// genetic 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, insert_text_format: InsertTextFormat, /// What item (struct, function, etc) are we completing. kind: Option, /// Lookup is used to check if completion item indeed can complete current /// ident. /// /// That is, in `foo.bar<|>` lookup of `abracadabra` will be accepted (it /// contains `bar` sub sequence), and `quux` will rejected. lookup: Option, /// Additional info to show in the UI pop up. detail: Option, documentation: Option, /// 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, /// Score is useful to pre select or display in better order completion items score: Option, /// Indicates that a reference or mutable reference to this variable is a /// possible match. ref_match: Option<(Mutability, CompletionScore)>, } // 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"); s.field("label", &self.label()).field("source_range", &self.source_range()); if self.text_edit().len() == 1 { let atom = &self.text_edit().iter().next().unwrap(); 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); } if let Some(score) = &self.score { s.field("score", score); } if self.trigger_call_info { s.field("trigger_call_info", &true); } s.finish() } } #[derive(Debug, Clone, Copy, Ord, PartialOrd, Eq, PartialEq)] pub enum CompletionScore { /// If only type match TypeMatch, /// If type and name match TypeAndNameMatch, } #[derive(Debug, Clone, Copy, PartialEq, Eq)] pub enum CompletionItemKind { Snippet, Keyword, Module, Function, BuiltinType, Struct, Enum, EnumVariant, Binding, Field, Static, Const, Trait, TypeAlias, Method, TypeParam, Macro, Attribute, UnresolvedReference, } impl CompletionItemKind { #[cfg(test)] pub(crate) fn tag(&self) -> &'static str { match self { CompletionItemKind::Attribute => "at", CompletionItemKind::Binding => "bn", CompletionItemKind::BuiltinType => "bt", CompletionItemKind::Const => "ct", CompletionItemKind::Enum => "en", CompletionItemKind::EnumVariant => "ev", CompletionItemKind::Field => "fd", CompletionItemKind::Function => "fn", CompletionItemKind::Keyword => "kw", CompletionItemKind::Macro => "ma", CompletionItemKind::Method => "me", CompletionItemKind::Module => "md", CompletionItemKind::Snippet => "sn", CompletionItemKind::Static => "sc", CompletionItemKind::Struct => "st", CompletionItemKind::Trait => "tt", CompletionItemKind::TypeAlias => "ta", CompletionItemKind::TypeParam => "tp", CompletionItemKind::UnresolvedReference => "??", } } } #[derive(Debug, PartialEq, Eq, Copy, Clone)] pub(crate) enum CompletionKind { /// Parser-based keyword completion. Keyword, /// Your usual "complete all valid identifiers". Reference, /// "Secret sauce" completions. Magic, Snippet, Postfix, BuiltinType, Attribute, } #[derive(Debug, PartialEq, Eq, Copy, Clone)] pub enum InsertTextFormat { PlainText, Snippet, } impl CompletionItem { pub(crate) fn new( completion_kind: CompletionKind, source_range: TextRange, label: impl Into, ) -> Builder { let label = label.into(); Builder { source_range, completion_kind, label, insert_text: None, insert_text_format: InsertTextFormat::PlainText, detail: None, documentation: None, lookup: None, kind: None, text_edit: None, deprecated: None, trigger_call_info: None, score: None, ref_match: None, import_data: None, } } /// 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 } /// Short one-line additional information, like a type pub fn detail(&self) -> Option<&str> { self.detail.as_deref() } /// A doc-comment pub fn documentation(&self) -> Option { self.documentation.clone() } /// What string is used for filtering. pub fn lookup(&self) -> &str { self.lookup.as_deref().unwrap_or(&self.label) } pub fn kind(&self) -> Option { self.kind } pub fn deprecated(&self) -> bool { self.deprecated } pub fn score(&self) -> Option { self.score } pub fn trigger_call_info(&self) -> bool { self.trigger_call_info } pub fn ref_match(&self) -> Option<(Mutability, CompletionScore)> { self.ref_match } } /// A helper to make `CompletionItem`s. #[must_use] #[derive(Clone)] pub(crate) struct Builder { source_range: TextRange, completion_kind: CompletionKind, import_data: Option<(ModPath, ImportScope, Option)>, label: String, insert_text: Option, insert_text_format: InsertTextFormat, detail: Option, documentation: Option, lookup: Option, kind: Option, text_edit: Option, deprecated: Option, trigger_call_info: Option, score: Option, ref_match: Option<(Mutability, CompletionScore)>, } impl Builder { pub(crate) fn build(self) -> CompletionItem { let _p = profile::span("item::Builder::build"); let mut label = self.label; let mut lookup = self.lookup; let mut insert_text = self.insert_text; let mut text_edits = TextEdit::builder(); if let Some((import_path, import_scope, merge_behaviour)) = self.import_data { let import = mod_path_to_ast(&import_path); let mut import_path_without_last_segment = import_path; let _ = import_path_without_last_segment.segments.pop(); if !import_path_without_last_segment.segments.is_empty() { if lookup.is_none() { lookup = Some(label.clone()); } if insert_text.is_none() { insert_text = Some(label.clone()); } label = format!("{}::{}", import_path_without_last_segment, label); } let rewriter = insert_use::insert_use(&import_scope, import, merge_behaviour); if let Some(old_ast) = rewriter.rewrite_root() { algo::diff(&old_ast, &rewriter.rewrite(&old_ast)).into_text_edit(&mut text_edits); } } let original_edit = match self.text_edit { Some(it) => it, None => { TextEdit::replace(self.source_range, insert_text.unwrap_or_else(|| label.clone())) } }; let mut resulting_edit = text_edits.finish(); resulting_edit.union(original_edit).expect("Failed to unite text edits"); CompletionItem { source_range: self.source_range, label, insert_text_format: self.insert_text_format, text_edit: resulting_edit, detail: self.detail, documentation: self.documentation, lookup, kind: self.kind, completion_kind: self.completion_kind, deprecated: self.deprecated.unwrap_or(false), trigger_call_info: self.trigger_call_info.unwrap_or(false), score: self.score, ref_match: self.ref_match, } } pub(crate) fn lookup_by(mut self, lookup: impl Into) -> Builder { self.lookup = Some(lookup.into()); self } pub(crate) fn label(mut self, label: impl Into) -> Builder { self.label = label.into(); self } pub(crate) fn insert_text(mut self, insert_text: impl Into) -> Builder { self.insert_text = Some(insert_text.into()); self } pub(crate) fn insert_snippet( mut self, _cap: SnippetCap, snippet: impl Into, ) -> Builder { self.insert_text_format = InsertTextFormat::Snippet; self.insert_text(snippet) } pub(crate) fn kind(mut self, kind: CompletionItemKind) -> Builder { self.kind = Some(kind); self } pub(crate) fn text_edit(mut self, edit: TextEdit) -> Builder { self.text_edit = Some(edit); self } pub(crate) fn snippet_edit(mut self, _cap: SnippetCap, edit: TextEdit) -> Builder { self.insert_text_format = InsertTextFormat::Snippet; self.text_edit(edit) } #[allow(unused)] pub(crate) fn detail(self, detail: impl Into) -> Builder { self.set_detail(Some(detail)) } pub(crate) fn set_detail(mut self, detail: Option>) -> Builder { self.detail = detail.map(Into::into); self } #[allow(unused)] pub(crate) fn documentation(self, docs: Documentation) -> Builder { self.set_documentation(Some(docs)) } pub(crate) fn set_documentation(mut self, docs: Option) -> Builder { self.documentation = docs.map(Into::into); self } pub(crate) fn set_deprecated(mut self, deprecated: bool) -> Builder { self.deprecated = Some(deprecated); self } pub(crate) fn set_score(mut self, score: CompletionScore) -> Builder { self.score = Some(score); self } pub(crate) fn trigger_call_info(mut self) -> Builder { self.trigger_call_info = Some(true); self } pub(crate) fn import_data( mut self, import_data: Option<(ModPath, ImportScope, Option)>, ) -> Builder { self.import_data = import_data; self } pub(crate) fn set_ref_match( mut self, ref_match: Option<(Mutability, CompletionScore)>, ) -> Builder { self.ref_match = ref_match; self } } impl<'a> Into for Builder { fn into(self) -> CompletionItem { self.build() } }