Avoid turning completion objects into builders

This commit is contained in:
Kirill Bulatov 2020-11-16 23:16:41 +02:00
parent 4109968934
commit d4128beb3d
9 changed files with 117 additions and 84 deletions

View file

@ -15,7 +15,7 @@ use syntax::{
}; };
use test_utils::mark; use test_utils::mark;
#[derive(Debug)] #[derive(Debug, Clone)]
pub enum ImportScope { pub enum ImportScope {
File(ast::SourceFile), File(ast::SourceFile),
Module(ast::ItemList), Module(ast::ItemList),

View file

@ -90,7 +90,7 @@ impl Completions {
Some(it) => it, Some(it) => it,
None => return, None => return,
}; };
if let Some(item) = render_macro(RenderContext::new(ctx), name, macro_) { if let Some(item) = render_macro(RenderContext::new(ctx), None, name, macro_) {
self.add(item); self.add(item);
} }
} }
@ -101,7 +101,7 @@ impl Completions {
func: hir::Function, func: hir::Function,
local_name: Option<String>, local_name: Option<String>,
) { ) {
let item = render_fn(RenderContext::new(ctx), local_name, func); let item = render_fn(RenderContext::new(ctx), None, local_name, func);
self.add(item) self.add(item)
} }
@ -123,7 +123,7 @@ impl Completions {
variant: hir::EnumVariant, variant: hir::EnumVariant,
path: ModPath, path: ModPath,
) { ) {
let item = render_enum_variant(RenderContext::new(ctx), None, variant, Some(path)); let item = render_enum_variant(RenderContext::new(ctx), None, None, variant, Some(path));
self.add(item); self.add(item);
} }
@ -133,7 +133,7 @@ impl Completions {
variant: hir::EnumVariant, variant: hir::EnumVariant,
local_name: Option<String>, local_name: Option<String>,
) { ) {
let item = render_enum_variant(RenderContext::new(ctx), local_name, variant, None); let item = render_enum_variant(RenderContext::new(ctx), None, local_name, variant, None);
self.add(item); self.add(item);
} }
} }

View file

@ -1,14 +1,14 @@
//! Completion of names from the current scope, e.g. locals and imported items. //! Completion of names from the current scope, e.g. locals and imported items.
use assists::utils::{insert_use, mod_path_to_ast, ImportScope}; use assists::utils::ImportScope;
use either::Either; use either::Either;
use hir::{Adt, ModuleDef, ScopeDef, Type}; use hir::{Adt, ModuleDef, ScopeDef, Type};
use ide_db::imports_locator; use ide_db::imports_locator;
use syntax::{algo, AstNode}; use syntax::AstNode;
use test_utils::mark; use test_utils::mark;
use crate::{ use crate::{
render::{render_resolution, RenderContext}, render::{render_resolution_with_import, RenderContext},
CompletionContext, Completions, CompletionContext, Completions,
}; };
@ -95,35 +95,13 @@ fn fuzzy_completion(acc: &mut Completions, ctx: &CompletionContext) -> Option<()
)), )),
}) })
.filter(|(mod_path, _)| mod_path.len() > 1) .filter(|(mod_path, _)| mod_path.len() > 1)
.filter_map(|(mod_path, definition)| { .filter_map(|(import_path, definition)| {
let use_to_insert = mod_path_to_ast(&mod_path); render_resolution_with_import(
let mut mod_path_without_last_segment = mod_path; RenderContext::new(ctx),
let name_after_import = mod_path_without_last_segment.segments.pop()?.to_string(); import_path.clone(),
import_scope.clone(),
let resolution_with_missing_import = ctx.config.merge,
render_resolution(RenderContext::new(ctx), name_after_import, &definition)?; &definition,
let lookup_string = resolution_with_missing_import.lookup().to_owned();
let mut text_edits =
resolution_with_missing_import.text_edit().to_owned().into_builder();
let rewriter = insert_use(&import_scope, use_to_insert, ctx.config.merge);
let old_ast = rewriter.rewrite_root()?;
algo::diff(&old_ast, &rewriter.rewrite(&old_ast)).into_text_edit(&mut text_edits);
let qualifier_string = mod_path_without_last_segment.to_string();
let qualified_label = if qualifier_string.is_empty() {
resolution_with_missing_import.label().to_owned()
} else {
format!("{}::{}", qualifier_string, resolution_with_missing_import.label())
};
Some(
resolution_with_missing_import
.into_builder()
.text_edit(text_edits.finish())
.label(qualified_label)
.lookup_by(lookup_string)
.build(),
) )
}) })
.take(20); .take(20);

View file

@ -2,8 +2,9 @@
use std::fmt; use std::fmt;
use hir::{Documentation, Mutability}; use assists::utils::{insert_use, mod_path_to_ast, ImportScope, MergeBehaviour};
use syntax::TextRange; use hir::{Documentation, ModPath, Mutability};
use syntax::{algo, TextRange};
use text_edit::TextEdit; use text_edit::TextEdit;
use crate::config::SnippetCap; use crate::config::SnippetCap;
@ -200,25 +201,7 @@ impl CompletionItem {
trigger_call_info: None, trigger_call_info: None,
score: None, score: None,
ref_match: None, ref_match: None,
} import_data: None,
}
pub(crate) fn into_builder(self) -> Builder {
Builder {
source_range: self.source_range,
completion_kind: self.completion_kind,
label: self.label,
insert_text: None,
insert_text_format: self.insert_text_format,
detail: self.detail,
documentation: self.documentation,
lookup: self.lookup,
kind: self.kind,
text_edit: Some(self.text_edit),
deprecated: Some(self.deprecated),
trigger_call_info: Some(self.trigger_call_info),
score: self.score,
ref_match: self.ref_match,
} }
} }
@ -278,6 +261,7 @@ impl CompletionItem {
pub(crate) struct Builder { pub(crate) struct Builder {
source_range: TextRange, source_range: TextRange,
completion_kind: CompletionKind, completion_kind: CompletionKind,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
label: String, label: String,
insert_text: Option<String>, insert_text: Option<String>,
insert_text_format: InsertTextFormat, insert_text_format: InsertTextFormat,
@ -294,23 +278,50 @@ pub(crate) struct Builder {
impl Builder { impl Builder {
pub(crate) fn build(self) -> CompletionItem { pub(crate) fn build(self) -> CompletionItem {
let label = self.label; let mut label = self.label;
let text_edit = match self.text_edit { 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(&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, Some(it) => it,
None => TextEdit::replace( None => {
self.source_range, TextEdit::replace(self.source_range, insert_text.unwrap_or_else(|| label.clone()))
self.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 { CompletionItem {
source_range: self.source_range, source_range: self.source_range,
label, label,
insert_text_format: self.insert_text_format, insert_text_format: self.insert_text_format,
text_edit, text_edit: resulting_edit,
detail: self.detail, detail: self.detail,
documentation: self.documentation, documentation: self.documentation,
lookup: self.lookup, lookup,
kind: self.kind, kind: self.kind,
completion_kind: self.completion_kind, completion_kind: self.completion_kind,
deprecated: self.deprecated.unwrap_or(false), deprecated: self.deprecated.unwrap_or(false),
@ -379,6 +390,13 @@ impl Builder {
self.trigger_call_info = Some(true); self.trigger_call_info = Some(true);
self self
} }
pub(crate) fn import_data(
mut self,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
) -> Builder {
self.import_data = import_data;
self
}
pub(crate) fn set_ref_match( pub(crate) fn set_ref_match(
mut self, mut self,
ref_match: Option<(Mutability, CompletionScore)>, ref_match: Option<(Mutability, CompletionScore)>,

View file

@ -9,7 +9,8 @@ pub(crate) mod type_alias;
mod builder_ext; mod builder_ext;
use hir::{Documentation, HasAttrs, HirDisplay, Mutability, ScopeDef, Type}; use assists::utils::{ImportScope, MergeBehaviour};
use hir::{Documentation, HasAttrs, HirDisplay, ModPath, Mutability, ScopeDef, Type};
use ide_db::RootDatabase; use ide_db::RootDatabase;
use syntax::TextRange; use syntax::TextRange;
use test_utils::mark; use test_utils::mark;
@ -42,7 +43,22 @@ pub(crate) fn render_resolution<'a>(
local_name: String, local_name: String,
resolution: &ScopeDef, resolution: &ScopeDef,
) -> Option<CompletionItem> { ) -> Option<CompletionItem> {
Render::new(ctx).render_resolution(local_name, resolution) Render::new(ctx).render_resolution(local_name, None, resolution)
}
pub(crate) fn render_resolution_with_import<'a>(
ctx: RenderContext<'a>,
import: ModPath,
import_scope: ImportScope,
merge_behaviour: Option<MergeBehaviour>,
resolution: &ScopeDef,
) -> Option<CompletionItem> {
let local_name = import.segments.last()?.to_string();
Render::new(ctx).render_resolution(
local_name,
Some((import, import_scope, merge_behaviour)),
resolution,
)
} }
/// Interface for data and methods required for items rendering. /// Interface for data and methods required for items rendering.
@ -131,6 +147,7 @@ impl<'a> Render<'a> {
fn render_resolution( fn render_resolution(
self, self,
local_name: String, local_name: String,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
resolution: &ScopeDef, resolution: &ScopeDef,
) -> Option<CompletionItem> { ) -> Option<CompletionItem> {
use hir::ModuleDef::*; use hir::ModuleDef::*;
@ -142,15 +159,15 @@ impl<'a> Render<'a> {
let kind = match resolution { let kind = match resolution {
ScopeDef::ModuleDef(Function(func)) => { ScopeDef::ModuleDef(Function(func)) => {
let item = render_fn(self.ctx, Some(local_name), *func); let item = render_fn(self.ctx, import_data, Some(local_name), *func);
return Some(item); return Some(item);
} }
ScopeDef::ModuleDef(EnumVariant(var)) => { ScopeDef::ModuleDef(EnumVariant(var)) => {
let item = render_enum_variant(self.ctx, Some(local_name), *var, None); let item = render_enum_variant(self.ctx, import_data, Some(local_name), *var, None);
return Some(item); return Some(item);
} }
ScopeDef::MacroDef(mac) => { ScopeDef::MacroDef(mac) => {
let item = render_macro(self.ctx, local_name, *mac); let item = render_macro(self.ctx, import_data, local_name, *mac);
return item; return item;
} }
@ -175,6 +192,7 @@ impl<'a> Render<'a> {
local_name, local_name,
) )
.kind(CompletionItemKind::UnresolvedReference) .kind(CompletionItemKind::UnresolvedReference)
.import_data(import_data)
.build(); .build();
return Some(item); return Some(item);
} }
@ -227,7 +245,12 @@ impl<'a> Render<'a> {
} }
} }
let item = item.kind(kind).set_documentation(docs).set_ref_match(ref_match).build(); let item = item
.kind(kind)
.import_data(import_data)
.set_documentation(docs)
.set_ref_match(ref_match)
.build();
Some(item) Some(item)
} }

View file

@ -1,5 +1,6 @@
//! Renderer for `enum` variants. //! Renderer for `enum` variants.
use assists::utils::{ImportScope, MergeBehaviour};
use hir::{HasAttrs, HirDisplay, ModPath, StructKind}; use hir::{HasAttrs, HirDisplay, ModPath, StructKind};
use itertools::Itertools; use itertools::Itertools;
use test_utils::mark; use test_utils::mark;
@ -11,11 +12,12 @@ use crate::{
pub(crate) fn render_enum_variant<'a>( pub(crate) fn render_enum_variant<'a>(
ctx: RenderContext<'a>, ctx: RenderContext<'a>,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
local_name: Option<String>, local_name: Option<String>,
variant: hir::EnumVariant, variant: hir::EnumVariant,
path: Option<ModPath>, path: Option<ModPath>,
) -> CompletionItem { ) -> CompletionItem {
EnumVariantRender::new(ctx, local_name, variant, path).render() EnumVariantRender::new(ctx, local_name, variant, path).render(import_data)
} }
#[derive(Debug)] #[derive(Debug)]
@ -60,7 +62,10 @@ impl<'a> EnumVariantRender<'a> {
} }
} }
fn render(self) -> CompletionItem { fn render(
self,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
) -> CompletionItem {
let mut builder = CompletionItem::new( let mut builder = CompletionItem::new(
CompletionKind::Reference, CompletionKind::Reference,
self.ctx.source_range(), self.ctx.source_range(),
@ -69,6 +74,7 @@ impl<'a> EnumVariantRender<'a> {
.kind(CompletionItemKind::EnumVariant) .kind(CompletionItemKind::EnumVariant)
.set_documentation(self.variant.docs(self.ctx.db())) .set_documentation(self.variant.docs(self.ctx.db()))
.set_deprecated(self.ctx.is_deprecated(self.variant)) .set_deprecated(self.ctx.is_deprecated(self.variant))
.import_data(import_data)
.detail(self.detail()); .detail(self.detail());
if self.variant_kind == StructKind::Tuple { if self.variant_kind == StructKind::Tuple {

View file

@ -1,6 +1,7 @@
//! Renderer for function calls. //! Renderer for function calls.
use hir::{HasSource, Type}; use assists::utils::{ImportScope, MergeBehaviour};
use hir::{HasSource, ModPath, Type};
use syntax::{ast::Fn, display::function_declaration}; use syntax::{ast::Fn, display::function_declaration};
use crate::{ use crate::{
@ -10,10 +11,11 @@ use crate::{
pub(crate) fn render_fn<'a>( pub(crate) fn render_fn<'a>(
ctx: RenderContext<'a>, ctx: RenderContext<'a>,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
local_name: Option<String>, local_name: Option<String>,
fn_: hir::Function, fn_: hir::Function,
) -> CompletionItem { ) -> CompletionItem {
FunctionRender::new(ctx, local_name, fn_).render() FunctionRender::new(ctx, local_name, fn_).render(import_data)
} }
#[derive(Debug)] #[derive(Debug)]
@ -36,7 +38,10 @@ impl<'a> FunctionRender<'a> {
FunctionRender { ctx, name, fn_, ast_node } FunctionRender { ctx, name, fn_, ast_node }
} }
fn render(self) -> CompletionItem { fn render(
self,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
) -> CompletionItem {
let params = self.params(); let params = self.params();
CompletionItem::new(CompletionKind::Reference, self.ctx.source_range(), self.name.clone()) CompletionItem::new(CompletionKind::Reference, self.ctx.source_range(), self.name.clone())
.kind(self.kind()) .kind(self.kind())
@ -44,6 +49,7 @@ impl<'a> FunctionRender<'a> {
.set_deprecated(self.ctx.is_deprecated(self.fn_)) .set_deprecated(self.ctx.is_deprecated(self.fn_))
.detail(self.detail()) .detail(self.detail())
.add_call_parens(self.ctx.completion, self.name, params) .add_call_parens(self.ctx.completion, self.name, params)
.import_data(import_data)
.build() .build()
} }

View file

@ -1,6 +1,7 @@
//! Renderer for macro invocations. //! Renderer for macro invocations.
use hir::{Documentation, HasSource}; use assists::utils::{ImportScope, MergeBehaviour};
use hir::{Documentation, HasSource, ModPath};
use syntax::display::macro_label; use syntax::display::macro_label;
use test_utils::mark; use test_utils::mark;
@ -11,10 +12,11 @@ use crate::{
pub(crate) fn render_macro<'a>( pub(crate) fn render_macro<'a>(
ctx: RenderContext<'a>, ctx: RenderContext<'a>,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
name: String, name: String,
macro_: hir::MacroDef, macro_: hir::MacroDef,
) -> Option<CompletionItem> { ) -> Option<CompletionItem> {
MacroRender::new(ctx, name, macro_).render() MacroRender::new(ctx, name, macro_).render(import_data)
} }
#[derive(Debug)] #[derive(Debug)]
@ -36,7 +38,10 @@ impl<'a> MacroRender<'a> {
MacroRender { ctx, name, macro_, docs, bra, ket } MacroRender { ctx, name, macro_, docs, bra, ket }
} }
fn render(&self) -> Option<CompletionItem> { fn render(
&self,
import_data: Option<(ModPath, ImportScope, Option<MergeBehaviour>)>,
) -> Option<CompletionItem> {
// FIXME: Currently proc-macro do not have ast-node, // FIXME: Currently proc-macro do not have ast-node,
// such that it does not have source // such that it does not have source
if self.macro_.is_proc_macro() { if self.macro_.is_proc_macro() {
@ -48,6 +53,7 @@ impl<'a> MacroRender<'a> {
.kind(CompletionItemKind::Macro) .kind(CompletionItemKind::Macro)
.set_documentation(self.docs.clone()) .set_documentation(self.docs.clone())
.set_deprecated(self.ctx.is_deprecated(self.macro_)) .set_deprecated(self.ctx.is_deprecated(self.macro_))
.import_data(import_data)
.detail(self.detail()); .detail(self.detail());
let needs_bang = self.needs_bang(); let needs_bang = self.needs_bang();

View file

@ -48,10 +48,6 @@ impl TextEdit {
TextEditBuilder::default() TextEditBuilder::default()
} }
pub fn into_builder(self) -> TextEditBuilder {
TextEditBuilder { indels: self.indels }
}
pub fn insert(offset: TextSize, text: String) -> TextEdit { pub fn insert(offset: TextSize, text: String) -> TextEdit {
let mut builder = TextEdit::builder(); let mut builder = TextEdit::builder();
builder.insert(offset, text); builder.insert(offset, text);