mirror of
				https://github.com/rust-lang/rust-analyzer.git
				synced 2025-11-03 13:13:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			3012 lines
		
	
	
		
			83 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			3012 lines
		
	
	
		
			83 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
//! `render` module provides utilities for rendering completion suggestions
 | 
						|
//! into code pieces that will be presented to user.
 | 
						|
 | 
						|
pub(crate) mod const_;
 | 
						|
pub(crate) mod function;
 | 
						|
pub(crate) mod literal;
 | 
						|
pub(crate) mod macro_;
 | 
						|
pub(crate) mod pattern;
 | 
						|
pub(crate) mod type_alias;
 | 
						|
pub(crate) mod union_literal;
 | 
						|
pub(crate) mod variant;
 | 
						|
 | 
						|
use hir::{sym, AsAssocItem, HasAttrs, HirDisplay, ImportPathConfig, ModuleDef, ScopeDef, Type};
 | 
						|
use ide_db::{
 | 
						|
    documentation::{Documentation, HasDocs},
 | 
						|
    helpers::item_name,
 | 
						|
    imports::import_assets::LocatedImport,
 | 
						|
    RootDatabase, SnippetCap, SymbolKind,
 | 
						|
};
 | 
						|
use syntax::{ast, format_smolstr, AstNode, SmolStr, SyntaxKind, TextRange};
 | 
						|
use text_edit::TextEdit;
 | 
						|
 | 
						|
use crate::{
 | 
						|
    context::{DotAccess, DotAccessKind, PathCompletionCtx, PathKind, PatternContext},
 | 
						|
    item::{Builder, CompletionRelevanceTypeMatch},
 | 
						|
    render::{
 | 
						|
        function::render_fn,
 | 
						|
        literal::render_variant_lit,
 | 
						|
        macro_::{render_macro, render_macro_pat},
 | 
						|
    },
 | 
						|
    CompletionContext, CompletionItem, CompletionItemKind, CompletionRelevance,
 | 
						|
};
 | 
						|
/// Interface for data and methods required for items rendering.
 | 
						|
#[derive(Debug, Clone)]
 | 
						|
pub(crate) struct RenderContext<'a> {
 | 
						|
    completion: &'a CompletionContext<'a>,
 | 
						|
    is_private_editable: bool,
 | 
						|
    import_to_add: Option<LocatedImport>,
 | 
						|
    doc_aliases: Vec<SmolStr>,
 | 
						|
}
 | 
						|
 | 
						|
impl<'a> RenderContext<'a> {
 | 
						|
    pub(crate) fn new(completion: &'a CompletionContext<'a>) -> RenderContext<'a> {
 | 
						|
        RenderContext {
 | 
						|
            completion,
 | 
						|
            is_private_editable: false,
 | 
						|
            import_to_add: None,
 | 
						|
            doc_aliases: vec![],
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    pub(crate) fn private_editable(mut self, private_editable: bool) -> Self {
 | 
						|
        self.is_private_editable = private_editable;
 | 
						|
        self
 | 
						|
    }
 | 
						|
 | 
						|
    pub(crate) fn import_to_add(mut self, import_to_add: Option<LocatedImport>) -> Self {
 | 
						|
        self.import_to_add = import_to_add;
 | 
						|
        self
 | 
						|
    }
 | 
						|
 | 
						|
    pub(crate) fn doc_aliases(mut self, doc_aliases: Vec<SmolStr>) -> Self {
 | 
						|
        self.doc_aliases = doc_aliases;
 | 
						|
        self
 | 
						|
    }
 | 
						|
 | 
						|
    fn snippet_cap(&self) -> Option<SnippetCap> {
 | 
						|
        self.completion.config.snippet_cap
 | 
						|
    }
 | 
						|
 | 
						|
    fn db(&self) -> &'a RootDatabase {
 | 
						|
        self.completion.db
 | 
						|
    }
 | 
						|
 | 
						|
    fn source_range(&self) -> TextRange {
 | 
						|
        self.completion.source_range()
 | 
						|
    }
 | 
						|
 | 
						|
    fn completion_relevance(&self) -> CompletionRelevance {
 | 
						|
        CompletionRelevance {
 | 
						|
            is_private_editable: self.is_private_editable,
 | 
						|
            requires_import: self.import_to_add.is_some(),
 | 
						|
            ..Default::default()
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    fn is_immediately_after_macro_bang(&self) -> bool {
 | 
						|
        self.completion.token.kind() == SyntaxKind::BANG
 | 
						|
            && self
 | 
						|
                .completion
 | 
						|
                .token
 | 
						|
                .parent()
 | 
						|
                .map_or(false, |it| it.kind() == SyntaxKind::MACRO_CALL)
 | 
						|
    }
 | 
						|
 | 
						|
    fn is_deprecated(&self, def: impl HasAttrs) -> bool {
 | 
						|
        let attrs = def.attrs(self.db());
 | 
						|
        attrs.by_key(&sym::deprecated).exists()
 | 
						|
    }
 | 
						|
 | 
						|
    fn is_deprecated_assoc_item(&self, as_assoc_item: impl AsAssocItem) -> bool {
 | 
						|
        let db = self.db();
 | 
						|
        let assoc = match as_assoc_item.as_assoc_item(db) {
 | 
						|
            Some(assoc) => assoc,
 | 
						|
            None => return false,
 | 
						|
        };
 | 
						|
 | 
						|
        let is_assoc_deprecated = match assoc {
 | 
						|
            hir::AssocItem::Function(it) => self.is_deprecated(it),
 | 
						|
            hir::AssocItem::Const(it) => self.is_deprecated(it),
 | 
						|
            hir::AssocItem::TypeAlias(it) => self.is_deprecated(it),
 | 
						|
        };
 | 
						|
        is_assoc_deprecated
 | 
						|
            || assoc
 | 
						|
                .container_or_implemented_trait(db)
 | 
						|
                .map(|trait_| self.is_deprecated(trait_))
 | 
						|
                .unwrap_or(false)
 | 
						|
    }
 | 
						|
 | 
						|
    // FIXME: remove this
 | 
						|
    fn docs(&self, def: impl HasDocs) -> Option<Documentation> {
 | 
						|
        def.docs(self.db())
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_field(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    dot_access: &DotAccess,
 | 
						|
    receiver: Option<hir::Name>,
 | 
						|
    field: hir::Field,
 | 
						|
    ty: &hir::Type,
 | 
						|
) -> CompletionItem {
 | 
						|
    let db = ctx.db();
 | 
						|
    let is_deprecated = ctx.is_deprecated(field);
 | 
						|
    let name = field.name(db);
 | 
						|
    let (name, escaped_name) = (name.unescaped().to_smol_str(), name.to_smol_str());
 | 
						|
    let mut item = CompletionItem::new(
 | 
						|
        SymbolKind::Field,
 | 
						|
        ctx.source_range(),
 | 
						|
        field_with_receiver(db, receiver.as_ref(), &name),
 | 
						|
    );
 | 
						|
    item.set_relevance(CompletionRelevance {
 | 
						|
        type_match: compute_type_match(ctx.completion, ty),
 | 
						|
        exact_name_match: compute_exact_name_match(ctx.completion, name.as_str()),
 | 
						|
        ..CompletionRelevance::default()
 | 
						|
    });
 | 
						|
    item.detail(ty.display(db).to_string())
 | 
						|
        .set_documentation(field.docs(db))
 | 
						|
        .set_deprecated(is_deprecated)
 | 
						|
        .lookup_by(name);
 | 
						|
 | 
						|
    let is_field_access = matches!(dot_access.kind, DotAccessKind::Field { .. });
 | 
						|
    if !is_field_access || ty.is_fn() || ty.is_closure() {
 | 
						|
        let mut builder = TextEdit::builder();
 | 
						|
        // Using TextEdit, insert '(' before the struct name and ')' before the
 | 
						|
        // dot access, then comes the field name and optionally insert function
 | 
						|
        // call parens.
 | 
						|
 | 
						|
        builder.replace(
 | 
						|
            ctx.source_range(),
 | 
						|
            field_with_receiver(db, receiver.as_ref(), &escaped_name).into(),
 | 
						|
        );
 | 
						|
 | 
						|
        let expected_fn_type =
 | 
						|
            ctx.completion.expected_type.as_ref().is_some_and(|ty| ty.is_fn() || ty.is_closure());
 | 
						|
 | 
						|
        if !expected_fn_type {
 | 
						|
            if let Some(receiver) = &dot_access.receiver {
 | 
						|
                if let Some(receiver) = ctx.completion.sema.original_ast_node(receiver.clone()) {
 | 
						|
                    builder.insert(receiver.syntax().text_range().start(), "(".to_owned());
 | 
						|
                    builder.insert(ctx.source_range().end(), ")".to_owned());
 | 
						|
 | 
						|
                    let is_parens_needed =
 | 
						|
                        !matches!(dot_access.kind, DotAccessKind::Method { has_parens: true });
 | 
						|
 | 
						|
                    if is_parens_needed {
 | 
						|
                        builder.insert(ctx.source_range().end(), "()".to_owned());
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        item.text_edit(builder.finish());
 | 
						|
    } else {
 | 
						|
        item.insert_text(field_with_receiver(db, receiver.as_ref(), &escaped_name));
 | 
						|
    }
 | 
						|
    if let Some(receiver) = &dot_access.receiver {
 | 
						|
        if let Some(original) = ctx.completion.sema.original_ast_node(receiver.clone()) {
 | 
						|
            if let Some(ref_match) = compute_ref_match(ctx.completion, ty) {
 | 
						|
                item.ref_match(ref_match, original.syntax().text_range().start());
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    item.doc_aliases(ctx.doc_aliases);
 | 
						|
    item.build(db)
 | 
						|
}
 | 
						|
 | 
						|
fn field_with_receiver(
 | 
						|
    db: &RootDatabase,
 | 
						|
    receiver: Option<&hir::Name>,
 | 
						|
    field_name: &str,
 | 
						|
) -> SmolStr {
 | 
						|
    receiver.map_or_else(
 | 
						|
        || field_name.into(),
 | 
						|
        |receiver| format_smolstr!("{}.{field_name}", receiver.display(db)),
 | 
						|
    )
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_tuple_field(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    receiver: Option<hir::Name>,
 | 
						|
    field: usize,
 | 
						|
    ty: &hir::Type,
 | 
						|
) -> CompletionItem {
 | 
						|
    let mut item = CompletionItem::new(
 | 
						|
        SymbolKind::Field,
 | 
						|
        ctx.source_range(),
 | 
						|
        field_with_receiver(ctx.db(), receiver.as_ref(), &field.to_string()),
 | 
						|
    );
 | 
						|
    item.detail(ty.display(ctx.db()).to_string()).lookup_by(field.to_string());
 | 
						|
    item.build(ctx.db())
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_type_inference(
 | 
						|
    ty_string: String,
 | 
						|
    ctx: &CompletionContext<'_>,
 | 
						|
) -> CompletionItem {
 | 
						|
    let mut builder =
 | 
						|
        CompletionItem::new(CompletionItemKind::InferredType, ctx.source_range(), ty_string);
 | 
						|
    builder.set_relevance(CompletionRelevance { is_definite: true, ..Default::default() });
 | 
						|
    builder.build(ctx.db)
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_path_resolution(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    path_ctx: &PathCompletionCtx,
 | 
						|
    local_name: hir::Name,
 | 
						|
    resolution: ScopeDef,
 | 
						|
) -> Builder {
 | 
						|
    render_resolution_path(ctx, path_ctx, local_name, None, resolution)
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_pattern_resolution(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    pattern_ctx: &PatternContext,
 | 
						|
    local_name: hir::Name,
 | 
						|
    resolution: ScopeDef,
 | 
						|
) -> Builder {
 | 
						|
    render_resolution_pat(ctx, pattern_ctx, local_name, None, resolution)
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_resolution_with_import(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    path_ctx: &PathCompletionCtx,
 | 
						|
    import_edit: LocatedImport,
 | 
						|
) -> Option<Builder> {
 | 
						|
    let resolution = ScopeDef::from(import_edit.original_item);
 | 
						|
    let local_name = scope_def_to_name(resolution, &ctx, &import_edit)?;
 | 
						|
    //this now just renders the alias text, but we need to find the aliases earlier and call this with the alias instead
 | 
						|
    let doc_aliases = ctx.completion.doc_aliases_in_scope(resolution);
 | 
						|
    let ctx = ctx.doc_aliases(doc_aliases);
 | 
						|
    Some(render_resolution_path(ctx, path_ctx, local_name, Some(import_edit), resolution))
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_resolution_with_import_pat(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    pattern_ctx: &PatternContext,
 | 
						|
    import_edit: LocatedImport,
 | 
						|
) -> Option<Builder> {
 | 
						|
    let resolution = ScopeDef::from(import_edit.original_item);
 | 
						|
    let local_name = scope_def_to_name(resolution, &ctx, &import_edit)?;
 | 
						|
    Some(render_resolution_pat(ctx, pattern_ctx, local_name, Some(import_edit), resolution))
 | 
						|
}
 | 
						|
 | 
						|
pub(crate) fn render_expr(
 | 
						|
    ctx: &CompletionContext<'_>,
 | 
						|
    expr: &hir::term_search::Expr,
 | 
						|
) -> Option<Builder> {
 | 
						|
    let mut i = 1;
 | 
						|
    let mut snippet_formatter = |ty: &hir::Type| {
 | 
						|
        let arg_name = ty
 | 
						|
            .as_adt()
 | 
						|
            .map(|adt| stdx::to_lower_snake_case(adt.name(ctx.db).as_str()))
 | 
						|
            .unwrap_or_else(|| String::from("_"));
 | 
						|
        let res = format!("${{{i}:{arg_name}}}");
 | 
						|
        i += 1;
 | 
						|
        res
 | 
						|
    };
 | 
						|
 | 
						|
    let mut label_formatter = |ty: &hir::Type| {
 | 
						|
        ty.as_adt()
 | 
						|
            .map(|adt| stdx::to_lower_snake_case(adt.name(ctx.db).as_str()))
 | 
						|
            .unwrap_or_else(|| String::from("..."))
 | 
						|
    };
 | 
						|
 | 
						|
    let cfg = ImportPathConfig {
 | 
						|
        prefer_no_std: ctx.config.prefer_no_std,
 | 
						|
        prefer_prelude: ctx.config.prefer_prelude,
 | 
						|
        prefer_absolute: ctx.config.prefer_absolute,
 | 
						|
    };
 | 
						|
 | 
						|
    let label = expr.gen_source_code(&ctx.scope, &mut label_formatter, cfg).ok()?;
 | 
						|
 | 
						|
    let source_range = match ctx.original_token.parent() {
 | 
						|
        Some(node) => match node.ancestors().find_map(ast::Path::cast) {
 | 
						|
            Some(path) => path.syntax().text_range(),
 | 
						|
            None => node.text_range(),
 | 
						|
        },
 | 
						|
        None => ctx.source_range(),
 | 
						|
    };
 | 
						|
 | 
						|
    let mut item = CompletionItem::new(CompletionItemKind::Expression, source_range, label);
 | 
						|
 | 
						|
    let snippet =
 | 
						|
        format!("{}$0", expr.gen_source_code(&ctx.scope, &mut snippet_formatter, cfg).ok()?);
 | 
						|
    let edit = TextEdit::replace(source_range, snippet);
 | 
						|
    item.snippet_edit(ctx.config.snippet_cap?, edit);
 | 
						|
    item.documentation(Documentation::new(String::from("Autogenerated expression by term search")));
 | 
						|
    item.set_relevance(crate::CompletionRelevance {
 | 
						|
        type_match: compute_type_match(ctx, &expr.ty(ctx.db)),
 | 
						|
        ..Default::default()
 | 
						|
    });
 | 
						|
    for trait_ in expr.traits_used(ctx.db) {
 | 
						|
        let trait_item = hir::ItemInNs::from(hir::ModuleDef::from(trait_));
 | 
						|
        let Some(path) = ctx.module.find_path(ctx.db, trait_item, cfg) else {
 | 
						|
            continue;
 | 
						|
        };
 | 
						|
 | 
						|
        item.add_import(LocatedImport::new(path, trait_item, trait_item));
 | 
						|
    }
 | 
						|
 | 
						|
    Some(item)
 | 
						|
}
 | 
						|
 | 
						|
fn scope_def_to_name(
 | 
						|
    resolution: ScopeDef,
 | 
						|
    ctx: &RenderContext<'_>,
 | 
						|
    import_edit: &LocatedImport,
 | 
						|
) -> Option<hir::Name> {
 | 
						|
    Some(match resolution {
 | 
						|
        ScopeDef::ModuleDef(hir::ModuleDef::Function(f)) => f.name(ctx.completion.db),
 | 
						|
        ScopeDef::ModuleDef(hir::ModuleDef::Const(c)) => c.name(ctx.completion.db)?,
 | 
						|
        ScopeDef::ModuleDef(hir::ModuleDef::TypeAlias(t)) => t.name(ctx.completion.db),
 | 
						|
        _ => item_name(ctx.db(), import_edit.original_item)?,
 | 
						|
    })
 | 
						|
}
 | 
						|
 | 
						|
fn render_resolution_pat(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    pattern_ctx: &PatternContext,
 | 
						|
    local_name: hir::Name,
 | 
						|
    import_to_add: Option<LocatedImport>,
 | 
						|
    resolution: ScopeDef,
 | 
						|
) -> Builder {
 | 
						|
    let _p = tracing::info_span!("render_resolution_pat").entered();
 | 
						|
    use hir::ModuleDef::*;
 | 
						|
 | 
						|
    if let ScopeDef::ModuleDef(Macro(mac)) = resolution {
 | 
						|
        let ctx = ctx.import_to_add(import_to_add);
 | 
						|
        render_macro_pat(ctx, pattern_ctx, local_name, mac)
 | 
						|
    } else {
 | 
						|
        render_resolution_simple_(ctx, &local_name, import_to_add, resolution)
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn render_resolution_path(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    path_ctx: &PathCompletionCtx,
 | 
						|
    local_name: hir::Name,
 | 
						|
    import_to_add: Option<LocatedImport>,
 | 
						|
    resolution: ScopeDef,
 | 
						|
) -> Builder {
 | 
						|
    let _p = tracing::info_span!("render_resolution_path").entered();
 | 
						|
    use hir::ModuleDef::*;
 | 
						|
 | 
						|
    match resolution {
 | 
						|
        ScopeDef::ModuleDef(Macro(mac)) => {
 | 
						|
            let ctx = ctx.import_to_add(import_to_add);
 | 
						|
            return render_macro(ctx, path_ctx, local_name, mac);
 | 
						|
        }
 | 
						|
        ScopeDef::ModuleDef(Function(func)) => {
 | 
						|
            let ctx = ctx.import_to_add(import_to_add);
 | 
						|
            return render_fn(ctx, path_ctx, Some(local_name), func);
 | 
						|
        }
 | 
						|
        ScopeDef::ModuleDef(Variant(var)) => {
 | 
						|
            let ctx = ctx.clone().import_to_add(import_to_add.clone());
 | 
						|
            if let Some(item) =
 | 
						|
                render_variant_lit(ctx, path_ctx, Some(local_name.clone()), var, None)
 | 
						|
            {
 | 
						|
                return item;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        _ => (),
 | 
						|
    }
 | 
						|
 | 
						|
    let completion = ctx.completion;
 | 
						|
    let cap = ctx.snippet_cap();
 | 
						|
    let db = completion.db;
 | 
						|
    let config = completion.config;
 | 
						|
    let requires_import = import_to_add.is_some();
 | 
						|
 | 
						|
    let name = local_name.to_smol_str();
 | 
						|
    let mut item = render_resolution_simple_(ctx, &local_name, import_to_add, resolution);
 | 
						|
    if local_name.is_escaped() {
 | 
						|
        item.insert_text(local_name.to_smol_str());
 | 
						|
    }
 | 
						|
    // Add `<>` for generic types
 | 
						|
    let type_path_no_ty_args = matches!(
 | 
						|
        path_ctx,
 | 
						|
        PathCompletionCtx { kind: PathKind::Type { .. }, has_type_args: false, .. }
 | 
						|
    ) && config.callable.is_some();
 | 
						|
    if type_path_no_ty_args {
 | 
						|
        if let Some(cap) = cap {
 | 
						|
            let has_non_default_type_params = match resolution {
 | 
						|
                ScopeDef::ModuleDef(hir::ModuleDef::Adt(it)) => it.has_non_default_type_params(db),
 | 
						|
                ScopeDef::ModuleDef(hir::ModuleDef::TypeAlias(it)) => {
 | 
						|
                    it.has_non_default_type_params(db)
 | 
						|
                }
 | 
						|
                _ => false,
 | 
						|
            };
 | 
						|
 | 
						|
            if has_non_default_type_params {
 | 
						|
                cov_mark::hit!(inserts_angle_brackets_for_generics);
 | 
						|
                item.lookup_by(name.clone())
 | 
						|
                    .label(SmolStr::from_iter([&name, "<…>"]))
 | 
						|
                    .trigger_call_info()
 | 
						|
                    .insert_snippet(cap, format!("{}<$0>", local_name.display(db)));
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    let mut set_item_relevance = |ty: Type| {
 | 
						|
        if !ty.is_unknown() {
 | 
						|
            item.detail(ty.display(db).to_string());
 | 
						|
        }
 | 
						|
 | 
						|
        item.set_relevance(CompletionRelevance {
 | 
						|
            type_match: compute_type_match(completion, &ty),
 | 
						|
            exact_name_match: compute_exact_name_match(completion, &name),
 | 
						|
            is_local: matches!(resolution, ScopeDef::Local(_)),
 | 
						|
            requires_import,
 | 
						|
            ..CompletionRelevance::default()
 | 
						|
        });
 | 
						|
 | 
						|
        path_ref_match(completion, path_ctx, &ty, &mut item);
 | 
						|
    };
 | 
						|
 | 
						|
    match resolution {
 | 
						|
        ScopeDef::Local(local) => set_item_relevance(local.ty(db)),
 | 
						|
        ScopeDef::ModuleDef(ModuleDef::Adt(adt)) | ScopeDef::AdtSelfType(adt) => {
 | 
						|
            set_item_relevance(adt.ty(db))
 | 
						|
        }
 | 
						|
        // Filtered out above
 | 
						|
        ScopeDef::ModuleDef(
 | 
						|
            ModuleDef::Function(_) | ModuleDef::Variant(_) | ModuleDef::Macro(_),
 | 
						|
        ) => (),
 | 
						|
        ScopeDef::ModuleDef(ModuleDef::Const(konst)) => set_item_relevance(konst.ty(db)),
 | 
						|
        ScopeDef::ModuleDef(ModuleDef::Static(stat)) => set_item_relevance(stat.ty(db)),
 | 
						|
        ScopeDef::ModuleDef(ModuleDef::BuiltinType(bt)) => set_item_relevance(bt.ty(db)),
 | 
						|
        ScopeDef::ImplSelfType(imp) => set_item_relevance(imp.self_ty(db)),
 | 
						|
        ScopeDef::GenericParam(_)
 | 
						|
        | ScopeDef::Label(_)
 | 
						|
        | ScopeDef::Unknown
 | 
						|
        | ScopeDef::ModuleDef(
 | 
						|
            ModuleDef::Trait(_)
 | 
						|
            | ModuleDef::TraitAlias(_)
 | 
						|
            | ModuleDef::Module(_)
 | 
						|
            | ModuleDef::TypeAlias(_),
 | 
						|
        ) => (),
 | 
						|
    };
 | 
						|
 | 
						|
    item
 | 
						|
}
 | 
						|
 | 
						|
fn render_resolution_simple_(
 | 
						|
    ctx: RenderContext<'_>,
 | 
						|
    local_name: &hir::Name,
 | 
						|
    import_to_add: Option<LocatedImport>,
 | 
						|
    resolution: ScopeDef,
 | 
						|
) -> Builder {
 | 
						|
    let _p = tracing::info_span!("render_resolution_simple_").entered();
 | 
						|
 | 
						|
    let db = ctx.db();
 | 
						|
    let ctx = ctx.import_to_add(import_to_add);
 | 
						|
    let kind = res_to_kind(resolution);
 | 
						|
 | 
						|
    let mut item =
 | 
						|
        CompletionItem::new(kind, ctx.source_range(), local_name.unescaped().to_smol_str());
 | 
						|
    item.set_relevance(ctx.completion_relevance())
 | 
						|
        .set_documentation(scope_def_docs(db, resolution))
 | 
						|
        .set_deprecated(scope_def_is_deprecated(&ctx, resolution));
 | 
						|
 | 
						|
    if let Some(import_to_add) = ctx.import_to_add {
 | 
						|
        item.add_import(import_to_add);
 | 
						|
    }
 | 
						|
 | 
						|
    item.doc_aliases(ctx.doc_aliases);
 | 
						|
    item
 | 
						|
}
 | 
						|
 | 
						|
fn res_to_kind(resolution: ScopeDef) -> CompletionItemKind {
 | 
						|
    use hir::ModuleDef::*;
 | 
						|
    match resolution {
 | 
						|
        ScopeDef::Unknown => CompletionItemKind::UnresolvedReference,
 | 
						|
        ScopeDef::ModuleDef(Function(_)) => CompletionItemKind::SymbolKind(SymbolKind::Function),
 | 
						|
        ScopeDef::ModuleDef(Variant(_)) => CompletionItemKind::SymbolKind(SymbolKind::Variant),
 | 
						|
        ScopeDef::ModuleDef(Macro(_)) => CompletionItemKind::SymbolKind(SymbolKind::Macro),
 | 
						|
        ScopeDef::ModuleDef(Module(..)) => CompletionItemKind::SymbolKind(SymbolKind::Module),
 | 
						|
        ScopeDef::ModuleDef(Adt(adt)) => CompletionItemKind::SymbolKind(match adt {
 | 
						|
            hir::Adt::Struct(_) => SymbolKind::Struct,
 | 
						|
            hir::Adt::Union(_) => SymbolKind::Union,
 | 
						|
            hir::Adt::Enum(_) => SymbolKind::Enum,
 | 
						|
        }),
 | 
						|
        ScopeDef::ModuleDef(Const(..)) => CompletionItemKind::SymbolKind(SymbolKind::Const),
 | 
						|
        ScopeDef::ModuleDef(Static(..)) => CompletionItemKind::SymbolKind(SymbolKind::Static),
 | 
						|
        ScopeDef::ModuleDef(Trait(..)) => CompletionItemKind::SymbolKind(SymbolKind::Trait),
 | 
						|
        ScopeDef::ModuleDef(TraitAlias(..)) => {
 | 
						|
            CompletionItemKind::SymbolKind(SymbolKind::TraitAlias)
 | 
						|
        }
 | 
						|
        ScopeDef::ModuleDef(TypeAlias(..)) => CompletionItemKind::SymbolKind(SymbolKind::TypeAlias),
 | 
						|
        ScopeDef::ModuleDef(BuiltinType(..)) => CompletionItemKind::BuiltinType,
 | 
						|
        ScopeDef::GenericParam(param) => CompletionItemKind::SymbolKind(match param {
 | 
						|
            hir::GenericParam::TypeParam(_) => SymbolKind::TypeParam,
 | 
						|
            hir::GenericParam::ConstParam(_) => SymbolKind::ConstParam,
 | 
						|
            hir::GenericParam::LifetimeParam(_) => SymbolKind::LifetimeParam,
 | 
						|
        }),
 | 
						|
        ScopeDef::Local(..) => CompletionItemKind::SymbolKind(SymbolKind::Local),
 | 
						|
        ScopeDef::Label(..) => CompletionItemKind::SymbolKind(SymbolKind::Label),
 | 
						|
        ScopeDef::AdtSelfType(..) | ScopeDef::ImplSelfType(..) => {
 | 
						|
            CompletionItemKind::SymbolKind(SymbolKind::SelfParam)
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn scope_def_docs(db: &RootDatabase, resolution: ScopeDef) -> Option<Documentation> {
 | 
						|
    use hir::ModuleDef::*;
 | 
						|
    match resolution {
 | 
						|
        ScopeDef::ModuleDef(Module(it)) => it.docs(db),
 | 
						|
        ScopeDef::ModuleDef(Adt(it)) => it.docs(db),
 | 
						|
        ScopeDef::ModuleDef(Variant(it)) => it.docs(db),
 | 
						|
        ScopeDef::ModuleDef(Const(it)) => it.docs(db),
 | 
						|
        ScopeDef::ModuleDef(Static(it)) => it.docs(db),
 | 
						|
        ScopeDef::ModuleDef(Trait(it)) => it.docs(db),
 | 
						|
        ScopeDef::ModuleDef(TypeAlias(it)) => it.docs(db),
 | 
						|
        _ => None,
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn scope_def_is_deprecated(ctx: &RenderContext<'_>, resolution: ScopeDef) -> bool {
 | 
						|
    match resolution {
 | 
						|
        ScopeDef::ModuleDef(it) => ctx.is_deprecated_assoc_item(it),
 | 
						|
        ScopeDef::GenericParam(it) => ctx.is_deprecated(it),
 | 
						|
        ScopeDef::AdtSelfType(it) => ctx.is_deprecated(it),
 | 
						|
        _ => false,
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
// FIXME: This checks types without possible coercions which some completions might want to do
 | 
						|
fn match_types(
 | 
						|
    ctx: &CompletionContext<'_>,
 | 
						|
    ty1: &hir::Type,
 | 
						|
    ty2: &hir::Type,
 | 
						|
) -> Option<CompletionRelevanceTypeMatch> {
 | 
						|
    if ty1 == ty2 {
 | 
						|
        Some(CompletionRelevanceTypeMatch::Exact)
 | 
						|
    } else if ty1.could_unify_with(ctx.db, ty2) {
 | 
						|
        Some(CompletionRelevanceTypeMatch::CouldUnify)
 | 
						|
    } else {
 | 
						|
        None
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn compute_type_match(
 | 
						|
    ctx: &CompletionContext<'_>,
 | 
						|
    completion_ty: &hir::Type,
 | 
						|
) -> Option<CompletionRelevanceTypeMatch> {
 | 
						|
    let expected_type = ctx.expected_type.as_ref()?;
 | 
						|
 | 
						|
    // We don't ever consider unit type to be an exact type match, since
 | 
						|
    // nearly always this is not meaningful to the user.
 | 
						|
    if expected_type.is_unit() {
 | 
						|
        return None;
 | 
						|
    }
 | 
						|
 | 
						|
    match_types(ctx, expected_type, completion_ty)
 | 
						|
}
 | 
						|
 | 
						|
fn compute_exact_name_match(ctx: &CompletionContext<'_>, completion_name: &str) -> bool {
 | 
						|
    ctx.expected_name.as_ref().map_or(false, |name| name.text() == completion_name)
 | 
						|
}
 | 
						|
 | 
						|
fn compute_ref_match(
 | 
						|
    ctx: &CompletionContext<'_>,
 | 
						|
    completion_ty: &hir::Type,
 | 
						|
) -> Option<hir::Mutability> {
 | 
						|
    let expected_type = ctx.expected_type.as_ref()?;
 | 
						|
    if completion_ty != expected_type {
 | 
						|
        let expected_type_without_ref = expected_type.remove_ref()?;
 | 
						|
        if completion_ty.autoderef(ctx.db).any(|deref_ty| deref_ty == expected_type_without_ref) {
 | 
						|
            cov_mark::hit!(suggest_ref);
 | 
						|
            let mutability = if expected_type.is_mutable_reference() {
 | 
						|
                hir::Mutability::Mut
 | 
						|
            } else {
 | 
						|
                hir::Mutability::Shared
 | 
						|
            };
 | 
						|
            return Some(mutability);
 | 
						|
        };
 | 
						|
    }
 | 
						|
    None
 | 
						|
}
 | 
						|
 | 
						|
fn path_ref_match(
 | 
						|
    completion: &CompletionContext<'_>,
 | 
						|
    path_ctx: &PathCompletionCtx,
 | 
						|
    ty: &hir::Type,
 | 
						|
    item: &mut Builder,
 | 
						|
) {
 | 
						|
    if let Some(original_path) = &path_ctx.original_path {
 | 
						|
        // At least one char was typed by the user already, in that case look for the original path
 | 
						|
        if let Some(original_path) = completion.sema.original_ast_node(original_path.clone()) {
 | 
						|
            if let Some(ref_match) = compute_ref_match(completion, ty) {
 | 
						|
                item.ref_match(ref_match, original_path.syntax().text_range().start());
 | 
						|
            }
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        // completion requested on an empty identifier, there is no path here yet.
 | 
						|
        // FIXME: This might create inconsistent completions where we show a ref match in macro inputs
 | 
						|
        // as long as nothing was typed yet
 | 
						|
        if let Some(ref_match) = compute_ref_match(completion, ty) {
 | 
						|
            item.ref_match(ref_match, completion.position.offset);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
#[cfg(test)]
 | 
						|
mod tests {
 | 
						|
    use std::cmp;
 | 
						|
 | 
						|
    use expect_test::{expect, Expect};
 | 
						|
    use ide_db::SymbolKind;
 | 
						|
    use itertools::Itertools;
 | 
						|
 | 
						|
    use crate::{
 | 
						|
        item::CompletionRelevanceTypeMatch,
 | 
						|
        tests::{check_edit, do_completion, get_all_items, TEST_CONFIG},
 | 
						|
        CompletionItem, CompletionItemKind, CompletionRelevance, CompletionRelevancePostfixMatch,
 | 
						|
    };
 | 
						|
 | 
						|
    #[track_caller]
 | 
						|
    fn check(ra_fixture: &str, kind: impl Into<CompletionItemKind>, expect: Expect) {
 | 
						|
        let actual = do_completion(ra_fixture, kind.into());
 | 
						|
        expect.assert_debug_eq(&actual);
 | 
						|
    }
 | 
						|
 | 
						|
    #[track_caller]
 | 
						|
    fn check_kinds(ra_fixture: &str, kinds: &[CompletionItemKind], expect: Expect) {
 | 
						|
        let actual: Vec<_> =
 | 
						|
            kinds.iter().flat_map(|&kind| do_completion(ra_fixture, kind)).collect();
 | 
						|
        expect.assert_debug_eq(&actual);
 | 
						|
    }
 | 
						|
 | 
						|
    #[track_caller]
 | 
						|
    fn check_function_relevance(ra_fixture: &str, expect: Expect) {
 | 
						|
        let actual: Vec<_> =
 | 
						|
            do_completion(ra_fixture, CompletionItemKind::SymbolKind(SymbolKind::Method))
 | 
						|
                .into_iter()
 | 
						|
                .map(|item| (item.detail.unwrap_or_default(), item.relevance.function))
 | 
						|
                .collect();
 | 
						|
 | 
						|
        expect.assert_debug_eq(&actual);
 | 
						|
    }
 | 
						|
 | 
						|
    #[track_caller]
 | 
						|
    fn check_relevance_for_kinds(ra_fixture: &str, kinds: &[CompletionItemKind], expect: Expect) {
 | 
						|
        let mut actual = get_all_items(TEST_CONFIG, ra_fixture, None);
 | 
						|
        actual.retain(|it| kinds.contains(&it.kind));
 | 
						|
        actual.sort_by_key(|it| cmp::Reverse(it.relevance.score()));
 | 
						|
        check_relevance_(actual, expect);
 | 
						|
    }
 | 
						|
 | 
						|
    #[track_caller]
 | 
						|
    fn check_relevance(ra_fixture: &str, expect: Expect) {
 | 
						|
        let mut actual = get_all_items(TEST_CONFIG, ra_fixture, None);
 | 
						|
        actual.retain(|it| it.kind != CompletionItemKind::Snippet);
 | 
						|
        actual.retain(|it| it.kind != CompletionItemKind::Keyword);
 | 
						|
        actual.retain(|it| it.kind != CompletionItemKind::BuiltinType);
 | 
						|
        actual.sort_by_key(|it| cmp::Reverse(it.relevance.score()));
 | 
						|
        check_relevance_(actual, expect);
 | 
						|
    }
 | 
						|
 | 
						|
    #[track_caller]
 | 
						|
    fn check_relevance_(actual: Vec<CompletionItem>, expect: Expect) {
 | 
						|
        let actual = actual
 | 
						|
            .into_iter()
 | 
						|
            .flat_map(|it| {
 | 
						|
                let mut items = vec![];
 | 
						|
 | 
						|
                let tag = it.kind.tag();
 | 
						|
                let relevance = display_relevance(it.relevance);
 | 
						|
                items.push(format!(
 | 
						|
                    "{tag} {}{} {relevance}\n",
 | 
						|
                    it.label,
 | 
						|
                    it.label_detail.clone().unwrap_or_default(),
 | 
						|
                ));
 | 
						|
 | 
						|
                if let Some((label, _indel, relevance)) = it.ref_match() {
 | 
						|
                    let relevance = display_relevance(relevance);
 | 
						|
 | 
						|
                    items.push(format!("{tag} {label} {relevance}\n"));
 | 
						|
                }
 | 
						|
 | 
						|
                items
 | 
						|
            })
 | 
						|
            .collect::<String>();
 | 
						|
 | 
						|
        expect.assert_eq(&actual);
 | 
						|
 | 
						|
        fn display_relevance(relevance: CompletionRelevance) -> String {
 | 
						|
            let relevance_factors = vec![
 | 
						|
                (relevance.type_match == Some(CompletionRelevanceTypeMatch::Exact), "type"),
 | 
						|
                (
 | 
						|
                    relevance.type_match == Some(CompletionRelevanceTypeMatch::CouldUnify),
 | 
						|
                    "type_could_unify",
 | 
						|
                ),
 | 
						|
                (relevance.exact_name_match, "name"),
 | 
						|
                (relevance.is_local, "local"),
 | 
						|
                (
 | 
						|
                    relevance.postfix_match == Some(CompletionRelevancePostfixMatch::Exact),
 | 
						|
                    "snippet",
 | 
						|
                ),
 | 
						|
                (relevance.is_op_method, "op_method"),
 | 
						|
                (relevance.requires_import, "requires_import"),
 | 
						|
            ]
 | 
						|
            .into_iter()
 | 
						|
            .filter_map(|(cond, desc)| if cond { Some(desc) } else { None })
 | 
						|
            .join("+");
 | 
						|
 | 
						|
            format!("[{relevance_factors}]")
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_struct_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
 | 
						|
pub mod test_mod_b {
 | 
						|
    pub struct Struct {}
 | 
						|
}
 | 
						|
 | 
						|
pub mod test_mod_a {
 | 
						|
    pub struct Struct {}
 | 
						|
}
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
fn test(input: dep::test_mod_b::Struct) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(Struct$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                st dep::test_mod_b::Struct {…} [type_could_unify]
 | 
						|
                ex dep::test_mod_b::Struct {  } [type_could_unify]
 | 
						|
                st Struct (use dep::test_mod_b::Struct) [type_could_unify+requires_import]
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
                md dep []
 | 
						|
                st Struct (use dep::test_mod_a::Struct) [requires_import]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_union_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
 | 
						|
pub mod test_mod_b {
 | 
						|
    pub union Union {
 | 
						|
        a: i32,
 | 
						|
        b: i32
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
pub mod test_mod_a {
 | 
						|
    pub enum Union {
 | 
						|
        a: i32,
 | 
						|
        b: i32
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
fn test(input: dep::test_mod_b::Union) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(Union$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                un Union (use dep::test_mod_b::Union) [type_could_unify+requires_import]
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
                md dep []
 | 
						|
                en Union (use dep::test_mod_a::Union) [requires_import]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_enum_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
 | 
						|
pub mod test_mod_b {
 | 
						|
    pub enum Enum {
 | 
						|
        variant
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
pub mod test_mod_a {
 | 
						|
    pub enum Enum {
 | 
						|
        variant
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
fn test(input: dep::test_mod_b::Enum) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(Enum$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                ev dep::test_mod_b::Enum::variant [type_could_unify]
 | 
						|
                ex dep::test_mod_b::Enum::variant [type_could_unify]
 | 
						|
                en Enum (use dep::test_mod_b::Enum) [type_could_unify+requires_import]
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
                md dep []
 | 
						|
                en Enum (use dep::test_mod_a::Enum) [requires_import]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_enum_variant_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
 | 
						|
pub mod test_mod_b {
 | 
						|
    pub enum Enum {
 | 
						|
        Variant
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
pub mod test_mod_a {
 | 
						|
    pub enum Enum {
 | 
						|
        Variant
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
fn test(input: dep::test_mod_b::Enum) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(Variant$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                ev dep::test_mod_b::Enum::Variant [type_could_unify]
 | 
						|
                ex dep::test_mod_b::Enum::Variant [type_could_unify]
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
                md dep []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_fn_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
 | 
						|
pub mod test_mod_b {
 | 
						|
    pub fn function(j: isize) -> i32 {}
 | 
						|
}
 | 
						|
 | 
						|
pub mod test_mod_a {
 | 
						|
    pub fn function(i: usize) -> i32 {}
 | 
						|
}
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
fn test(input: fn(usize) -> i32) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(function$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
                md dep []
 | 
						|
                fn function (use dep::test_mod_a::function) [requires_import]
 | 
						|
                fn function(…) (use dep::test_mod_b::function) [requires_import]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_const_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
 | 
						|
pub mod test_mod_b {
 | 
						|
    pub const CONST: i32 = 1;
 | 
						|
}
 | 
						|
 | 
						|
pub mod test_mod_a {
 | 
						|
    pub const CONST: i64 = 2;
 | 
						|
}
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
fn test(input: i32) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(CONST$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                ct CONST (use dep::test_mod_b::CONST) [type_could_unify+requires_import]
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
                md dep []
 | 
						|
                ct CONST (use dep::test_mod_a::CONST) [requires_import]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_static_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
 | 
						|
pub mod test_mod_b {
 | 
						|
    pub static STATIC: i32 = 5;
 | 
						|
}
 | 
						|
 | 
						|
pub mod test_mod_a {
 | 
						|
    pub static STATIC: i64 = 5;
 | 
						|
}
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
fn test(input: i32) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(STATIC$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                sc STATIC (use dep::test_mod_b::STATIC) [type_could_unify+requires_import]
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
                md dep []
 | 
						|
                sc STATIC (use dep::test_mod_a::STATIC) [requires_import]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_self_type_completion_info_with_params() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /lib.rs crate:dep
 | 
						|
pub struct Struct;
 | 
						|
 | 
						|
impl Struct {
 | 
						|
    pub fn Function(&self, input: i32) -> bool {
 | 
						|
                false
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
//- /main.rs crate:main deps:dep
 | 
						|
 | 
						|
use dep::Struct;
 | 
						|
 | 
						|
 | 
						|
fn test(input: fn(&dep::Struct, i32) -> bool) { }
 | 
						|
 | 
						|
fn main() {
 | 
						|
    test(Struct::Function$0);
 | 
						|
}
 | 
						|
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                me Function []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_self_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /main.rs crate:main
 | 
						|
 | 
						|
struct Struct;
 | 
						|
 | 
						|
impl Struct {
 | 
						|
fn test(&self) {
 | 
						|
        func(Self$0);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn func(input: Struct) { }
 | 
						|
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                st Struct [type]
 | 
						|
                st Self [type]
 | 
						|
                sp Self [type]
 | 
						|
                st Struct [type]
 | 
						|
                ex Struct [type]
 | 
						|
                lc self [local]
 | 
						|
                fn func(…) []
 | 
						|
                me self.test() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn set_builtin_type_completion_info() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- /main.rs crate:main
 | 
						|
 | 
						|
fn test(input: bool) { }
 | 
						|
    pub Input: bool = false;
 | 
						|
 | 
						|
fn main() {
 | 
						|
    let input = false;
 | 
						|
    let inputbad = 3;
 | 
						|
    test(inp$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                lc input [type+name+local]
 | 
						|
                ex input [type]
 | 
						|
                ex true [type]
 | 
						|
                ex false [type]
 | 
						|
                lc inputbad [local]
 | 
						|
                fn main() []
 | 
						|
                fn test(…) []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn enum_detail_includes_record_fields() {
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
enum Foo { Foo { x: i32, y: i32 } }
 | 
						|
 | 
						|
fn main() { Foo::Fo$0 }
 | 
						|
"#,
 | 
						|
            SymbolKind::Variant,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "Foo {…}",
 | 
						|
                        source_range: 54..56,
 | 
						|
                        delete: 54..56,
 | 
						|
                        insert: "Foo { x: ${1:()}, y: ${2:()} }$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Variant,
 | 
						|
                        ),
 | 
						|
                        lookup: "Foo{}",
 | 
						|
                        detail: "Foo { x: i32, y: i32 }",
 | 
						|
                        trigger_call_info: true,
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn enum_detail_includes_tuple_fields() {
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
enum Foo { Foo (i32, i32) }
 | 
						|
 | 
						|
fn main() { Foo::Fo$0 }
 | 
						|
"#,
 | 
						|
            SymbolKind::Variant,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "Foo(…)",
 | 
						|
                        source_range: 46..48,
 | 
						|
                        delete: 46..48,
 | 
						|
                        insert: "Foo(${1:()}, ${2:()})$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Variant,
 | 
						|
                        ),
 | 
						|
                        lookup: "Foo()",
 | 
						|
                        detail: "Foo(i32, i32)",
 | 
						|
                        trigger_call_info: true,
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn fn_detail_includes_args_and_return_type() {
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
fn foo<T>(a: u32, b: u32, t: T) -> (u32, T) { (a, t) }
 | 
						|
 | 
						|
fn main() { fo$0 }
 | 
						|
"#,
 | 
						|
            SymbolKind::Function,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "foo(…)",
 | 
						|
                        source_range: 68..70,
 | 
						|
                        delete: 68..70,
 | 
						|
                        insert: "foo(${1:a}, ${2:b}, ${3:t})$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Function,
 | 
						|
                        ),
 | 
						|
                        lookup: "foo",
 | 
						|
                        detail: "fn(u32, u32, T) -> (u32, T)",
 | 
						|
                        trigger_call_info: true,
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "main()",
 | 
						|
                        source_range: 68..70,
 | 
						|
                        delete: 68..70,
 | 
						|
                        insert: "main()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Function,
 | 
						|
                        ),
 | 
						|
                        lookup: "main",
 | 
						|
                        detail: "fn()",
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn enum_detail_just_name_for_unit() {
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
enum Foo { Foo }
 | 
						|
 | 
						|
fn main() { Foo::Fo$0 }
 | 
						|
"#,
 | 
						|
            SymbolKind::Variant,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "Foo",
 | 
						|
                        source_range: 35..37,
 | 
						|
                        delete: 35..37,
 | 
						|
                        insert: "Foo$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Variant,
 | 
						|
                        ),
 | 
						|
                        detail: "Foo",
 | 
						|
                        trigger_call_info: true,
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn lookup_enums_by_two_qualifiers() {
 | 
						|
        check_kinds(
 | 
						|
            r#"
 | 
						|
mod m {
 | 
						|
    pub enum Spam { Foo, Bar(i32) }
 | 
						|
}
 | 
						|
fn main() { let _: m::Spam = S$0 }
 | 
						|
"#,
 | 
						|
            &[
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Function),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Module),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Variant),
 | 
						|
            ],
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "main()",
 | 
						|
                        source_range: 75..76,
 | 
						|
                        delete: 75..76,
 | 
						|
                        insert: "main()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Function,
 | 
						|
                        ),
 | 
						|
                        lookup: "main",
 | 
						|
                        detail: "fn()",
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "m",
 | 
						|
                        source_range: 75..76,
 | 
						|
                        delete: 75..76,
 | 
						|
                        insert: "m",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Module,
 | 
						|
                        ),
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "m::Spam::Bar(…)",
 | 
						|
                        source_range: 75..76,
 | 
						|
                        delete: 75..76,
 | 
						|
                        insert: "m::Spam::Bar(${1:()})$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Variant,
 | 
						|
                        ),
 | 
						|
                        lookup: "Spam::Bar()",
 | 
						|
                        detail: "m::Spam::Bar(i32)",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: Some(
 | 
						|
                                Exact,
 | 
						|
                            ),
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: None,
 | 
						|
                        },
 | 
						|
                        trigger_call_info: true,
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "m::Spam::Foo",
 | 
						|
                        source_range: 75..76,
 | 
						|
                        delete: 75..76,
 | 
						|
                        insert: "m::Spam::Foo$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Variant,
 | 
						|
                        ),
 | 
						|
                        lookup: "Spam::Foo",
 | 
						|
                        detail: "m::Spam::Foo",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: Some(
 | 
						|
                                Exact,
 | 
						|
                            ),
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: None,
 | 
						|
                        },
 | 
						|
                        trigger_call_info: true,
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn sets_deprecated_flag_in_items() {
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
#[deprecated]
 | 
						|
fn something_deprecated() {}
 | 
						|
 | 
						|
fn main() { som$0 }
 | 
						|
"#,
 | 
						|
            SymbolKind::Function,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "main()",
 | 
						|
                        source_range: 56..59,
 | 
						|
                        delete: 56..59,
 | 
						|
                        insert: "main()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Function,
 | 
						|
                        ),
 | 
						|
                        lookup: "main",
 | 
						|
                        detail: "fn()",
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "something_deprecated()",
 | 
						|
                        source_range: 56..59,
 | 
						|
                        delete: 56..59,
 | 
						|
                        insert: "something_deprecated()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Function,
 | 
						|
                        ),
 | 
						|
                        lookup: "something_deprecated",
 | 
						|
                        detail: "fn()",
 | 
						|
                        deprecated: true,
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
struct A { #[deprecated] the_field: u32 }
 | 
						|
fn foo() { A { the$0 } }
 | 
						|
"#,
 | 
						|
            SymbolKind::Field,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "the_field",
 | 
						|
                        source_range: 57..60,
 | 
						|
                        delete: 57..60,
 | 
						|
                        insert: "the_field",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Field,
 | 
						|
                        ),
 | 
						|
                        detail: "u32",
 | 
						|
                        deprecated: true,
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: Some(
 | 
						|
                                CouldUnify,
 | 
						|
                            ),
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: None,
 | 
						|
                        },
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn renders_docs() {
 | 
						|
        check_kinds(
 | 
						|
            r#"
 | 
						|
struct S {
 | 
						|
    /// Field docs
 | 
						|
    foo:
 | 
						|
}
 | 
						|
impl S {
 | 
						|
    /// Method docs
 | 
						|
    fn bar(self) { self.$0 }
 | 
						|
}"#,
 | 
						|
            &[
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Method),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Field),
 | 
						|
            ],
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "bar()",
 | 
						|
                        source_range: 94..94,
 | 
						|
                        delete: 94..94,
 | 
						|
                        insert: "bar()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Method,
 | 
						|
                        ),
 | 
						|
                        lookup: "bar",
 | 
						|
                        detail: "fn(self)",
 | 
						|
                        documentation: Documentation(
 | 
						|
                            "Method docs",
 | 
						|
                        ),
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: None,
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: Some(
 | 
						|
                                CompletionRelevanceFn {
 | 
						|
                                    has_params: true,
 | 
						|
                                    has_self_param: true,
 | 
						|
                                    return_type: Other,
 | 
						|
                                },
 | 
						|
                            ),
 | 
						|
                        },
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "foo",
 | 
						|
                        source_range: 94..94,
 | 
						|
                        delete: 94..94,
 | 
						|
                        insert: "foo",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Field,
 | 
						|
                        ),
 | 
						|
                        detail: "{unknown}",
 | 
						|
                        documentation: Documentation(
 | 
						|
                            "Field docs",
 | 
						|
                        ),
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
 | 
						|
        check_kinds(
 | 
						|
            r#"
 | 
						|
use self::my$0;
 | 
						|
 | 
						|
/// mod docs
 | 
						|
mod my { }
 | 
						|
 | 
						|
/// enum docs
 | 
						|
enum E {
 | 
						|
    /// variant docs
 | 
						|
    V
 | 
						|
}
 | 
						|
use self::E::*;
 | 
						|
"#,
 | 
						|
            &[
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Module),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Variant),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Enum),
 | 
						|
            ],
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "my",
 | 
						|
                        source_range: 10..12,
 | 
						|
                        delete: 10..12,
 | 
						|
                        insert: "my",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Module,
 | 
						|
                        ),
 | 
						|
                        documentation: Documentation(
 | 
						|
                            "mod docs",
 | 
						|
                        ),
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "V",
 | 
						|
                        source_range: 10..12,
 | 
						|
                        delete: 10..12,
 | 
						|
                        insert: "V$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Variant,
 | 
						|
                        ),
 | 
						|
                        detail: "V",
 | 
						|
                        documentation: Documentation(
 | 
						|
                            "variant docs",
 | 
						|
                        ),
 | 
						|
                        trigger_call_info: true,
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "E",
 | 
						|
                        source_range: 10..12,
 | 
						|
                        delete: 10..12,
 | 
						|
                        insert: "E",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Enum,
 | 
						|
                        ),
 | 
						|
                        detail: "E",
 | 
						|
                        documentation: Documentation(
 | 
						|
                            "enum docs",
 | 
						|
                        ),
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn dont_render_attrs() {
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
struct S;
 | 
						|
impl S {
 | 
						|
    #[inline]
 | 
						|
    fn the_method(&self) { }
 | 
						|
}
 | 
						|
fn foo(s: S) { s.$0 }
 | 
						|
"#,
 | 
						|
            CompletionItemKind::SymbolKind(SymbolKind::Method),
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "the_method()",
 | 
						|
                        source_range: 81..81,
 | 
						|
                        delete: 81..81,
 | 
						|
                        insert: "the_method()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Method,
 | 
						|
                        ),
 | 
						|
                        lookup: "the_method",
 | 
						|
                        detail: "fn(&self)",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: None,
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: Some(
 | 
						|
                                CompletionRelevanceFn {
 | 
						|
                                    has_params: true,
 | 
						|
                                    has_self_param: true,
 | 
						|
                                    return_type: Other,
 | 
						|
                                },
 | 
						|
                            ),
 | 
						|
                        },
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn no_call_parens_if_fn_ptr_needed() {
 | 
						|
        cov_mark::check!(no_call_parens_if_fn_ptr_needed);
 | 
						|
        check_edit(
 | 
						|
            "foo",
 | 
						|
            r#"
 | 
						|
fn foo(foo: u8, bar: u8) {}
 | 
						|
struct ManualVtable { f: fn(u8, u8) }
 | 
						|
 | 
						|
fn main() -> ManualVtable {
 | 
						|
    ManualVtable { f: f$0 }
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
fn foo(foo: u8, bar: u8) {}
 | 
						|
struct ManualVtable { f: fn(u8, u8) }
 | 
						|
 | 
						|
fn main() -> ManualVtable {
 | 
						|
    ManualVtable { f: foo }
 | 
						|
}
 | 
						|
"#,
 | 
						|
        );
 | 
						|
        check_edit(
 | 
						|
            "type",
 | 
						|
            r#"
 | 
						|
struct RawIdentTable { r#type: u32 }
 | 
						|
 | 
						|
fn main() -> RawIdentTable {
 | 
						|
    RawIdentTable { t$0: 42 }
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct RawIdentTable { r#type: u32 }
 | 
						|
 | 
						|
fn main() -> RawIdentTable {
 | 
						|
    RawIdentTable { r#type: 42 }
 | 
						|
}
 | 
						|
"#,
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn no_parens_in_use_item() {
 | 
						|
        check_edit(
 | 
						|
            "foo",
 | 
						|
            r#"
 | 
						|
mod m { pub fn foo() {} }
 | 
						|
use crate::m::f$0;
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
mod m { pub fn foo() {} }
 | 
						|
use crate::m::foo;
 | 
						|
"#,
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn no_parens_in_call() {
 | 
						|
        check_edit(
 | 
						|
            "foo",
 | 
						|
            r#"
 | 
						|
fn foo(x: i32) {}
 | 
						|
fn main() { f$0(); }
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
fn foo(x: i32) {}
 | 
						|
fn main() { foo(); }
 | 
						|
"#,
 | 
						|
        );
 | 
						|
        check_edit(
 | 
						|
            "foo",
 | 
						|
            r#"
 | 
						|
struct Foo;
 | 
						|
impl Foo { fn foo(&self){} }
 | 
						|
fn f(foo: &Foo) { foo.f$0(); }
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct Foo;
 | 
						|
impl Foo { fn foo(&self){} }
 | 
						|
fn f(foo: &Foo) { foo.foo(); }
 | 
						|
"#,
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn inserts_angle_brackets_for_generics() {
 | 
						|
        cov_mark::check!(inserts_angle_brackets_for_generics);
 | 
						|
        check_edit(
 | 
						|
            "Vec",
 | 
						|
            r#"
 | 
						|
struct Vec<T> {}
 | 
						|
fn foo(xs: Ve$0)
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct Vec<T> {}
 | 
						|
fn foo(xs: Vec<$0>)
 | 
						|
"#,
 | 
						|
        );
 | 
						|
        check_edit(
 | 
						|
            "Vec",
 | 
						|
            r#"
 | 
						|
type Vec<T> = (T,);
 | 
						|
fn foo(xs: Ve$0)
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
type Vec<T> = (T,);
 | 
						|
fn foo(xs: Vec<$0>)
 | 
						|
"#,
 | 
						|
        );
 | 
						|
        check_edit(
 | 
						|
            "Vec",
 | 
						|
            r#"
 | 
						|
struct Vec<T = i128> {}
 | 
						|
fn foo(xs: Ve$0)
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct Vec<T = i128> {}
 | 
						|
fn foo(xs: Vec)
 | 
						|
"#,
 | 
						|
        );
 | 
						|
        check_edit(
 | 
						|
            "Vec",
 | 
						|
            r#"
 | 
						|
struct Vec<T> {}
 | 
						|
fn foo(xs: Ve$0<i128>)
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct Vec<T> {}
 | 
						|
fn foo(xs: Vec<i128>)
 | 
						|
"#,
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn active_param_relevance() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct S { foo: i64, bar: u32, baz: u32 }
 | 
						|
fn test(bar: u32) { }
 | 
						|
fn foo(s: S) { test(s.$0) }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fd bar [type+name]
 | 
						|
                fd baz [type]
 | 
						|
                fd foo []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn record_field_relevances() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct A { foo: i64, bar: u32, baz: u32 }
 | 
						|
struct B { x: (), y: f32, bar: u32 }
 | 
						|
fn foo(a: A) { B { bar: a.$0 }; }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fd bar [type+name]
 | 
						|
                fd baz [type]
 | 
						|
                fd foo []
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn tuple_field_detail() {
 | 
						|
        check(
 | 
						|
            r#"
 | 
						|
struct S(i32);
 | 
						|
 | 
						|
fn f() -> i32 {
 | 
						|
    let s = S(0);
 | 
						|
    s.0$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            SymbolKind::Field,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "0",
 | 
						|
                        source_range: 56..57,
 | 
						|
                        delete: 56..57,
 | 
						|
                        insert: "0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Field,
 | 
						|
                        ),
 | 
						|
                        detail: "i32",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: Some(
 | 
						|
                                Exact,
 | 
						|
                            ),
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: None,
 | 
						|
                        },
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn record_field_and_call_relevances() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct A { foo: i64, bar: u32, baz: u32 }
 | 
						|
struct B { x: (), y: f32, bar: u32 }
 | 
						|
fn f(foo: i64) {  }
 | 
						|
fn foo(a: A) { B { bar: f(a.$0) }; }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fd foo [type+name]
 | 
						|
                fd bar []
 | 
						|
                fd baz []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct A { foo: i64, bar: u32, baz: u32 }
 | 
						|
struct B { x: (), y: f32, bar: u32 }
 | 
						|
fn f(foo: i64) {  }
 | 
						|
fn foo(a: A) { f(B { bar: a.$0 }); }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fd bar [type+name]
 | 
						|
                fd baz [type]
 | 
						|
                fd foo []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn prioritize_exact_ref_match() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct WorldSnapshot { _f: () };
 | 
						|
fn go(world: &WorldSnapshot) { go(w$0) }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                lc world [type+name+local]
 | 
						|
                ex world [type]
 | 
						|
                st WorldSnapshot {…} []
 | 
						|
                st &WorldSnapshot {…} [type]
 | 
						|
                st WorldSnapshot []
 | 
						|
                st &WorldSnapshot [type]
 | 
						|
                fn go(…) []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn too_many_arguments() {
 | 
						|
        cov_mark::check!(too_many_arguments);
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct Foo;
 | 
						|
fn f(foo: &Foo) { f(foo, w$0) }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                lc foo [local]
 | 
						|
                st Foo []
 | 
						|
                fn f(…) []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn score_fn_type_and_name_match() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct A { bar: u8 }
 | 
						|
fn baz() -> u8 { 0 }
 | 
						|
fn bar() -> u8 { 0 }
 | 
						|
fn f() { A { bar: b$0 }; }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fn bar() [type+name]
 | 
						|
                fn baz() [type]
 | 
						|
                ex baz() [type]
 | 
						|
                ex bar() [type]
 | 
						|
                st A []
 | 
						|
                fn f() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn score_method_type_and_name_match() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
fn baz(aaa: u32){}
 | 
						|
struct Foo;
 | 
						|
impl Foo {
 | 
						|
fn aaa(&self) -> u32 { 0 }
 | 
						|
fn bbb(&self) -> u32 { 0 }
 | 
						|
fn ccc(&self) -> u64 { 0 }
 | 
						|
}
 | 
						|
fn f() {
 | 
						|
    baz(Foo.$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                me aaa() [type+name]
 | 
						|
                me bbb() [type]
 | 
						|
                me ccc() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn score_method_name_match_only() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
fn baz(aaa: u32){}
 | 
						|
struct Foo;
 | 
						|
impl Foo {
 | 
						|
fn aaa(&self) -> u64 { 0 }
 | 
						|
}
 | 
						|
fn f() {
 | 
						|
    baz(Foo.$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                me aaa() [name]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn suggest_ref_mut() {
 | 
						|
        cov_mark::check!(suggest_ref);
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct S;
 | 
						|
fn foo(s: &mut S) {}
 | 
						|
fn main() {
 | 
						|
    let mut s = S;
 | 
						|
    foo($0);
 | 
						|
}
 | 
						|
            "#,
 | 
						|
            expect![[r#"
 | 
						|
                lc s [name+local]
 | 
						|
                lc &mut s [type+name+local]
 | 
						|
                st S []
 | 
						|
                st &mut S [type]
 | 
						|
                st S []
 | 
						|
                st &mut S [type]
 | 
						|
                fn foo(…) []
 | 
						|
                fn main() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct S;
 | 
						|
fn foo(s: &mut S) {}
 | 
						|
fn main() {
 | 
						|
    let mut s = S;
 | 
						|
    foo(&mut $0);
 | 
						|
}
 | 
						|
            "#,
 | 
						|
            expect![[r#"
 | 
						|
                lc s [type+name+local]
 | 
						|
                st S [type]
 | 
						|
                st S [type]
 | 
						|
                ex s [type]
 | 
						|
                ex S [type]
 | 
						|
                fn foo(…) []
 | 
						|
                fn main() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct S;
 | 
						|
fn foo(s: &mut S) {}
 | 
						|
fn main() {
 | 
						|
    let mut ssss = S;
 | 
						|
    foo(&mut s$0);
 | 
						|
}
 | 
						|
            "#,
 | 
						|
            expect![[r#"
 | 
						|
                lc ssss [type+local]
 | 
						|
                st S [type]
 | 
						|
                st S [type]
 | 
						|
                ex ssss [type]
 | 
						|
                ex S [type]
 | 
						|
                fn foo(…) []
 | 
						|
                fn main() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn suggest_deref() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- minicore: deref
 | 
						|
struct S;
 | 
						|
struct T(S);
 | 
						|
 | 
						|
impl core::ops::Deref for T {
 | 
						|
    type Target = S;
 | 
						|
 | 
						|
    fn deref(&self) -> &Self::Target {
 | 
						|
        &self.0
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn foo(s: &S) {}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    let t = T(S);
 | 
						|
    let m = 123;
 | 
						|
 | 
						|
    foo($0);
 | 
						|
}
 | 
						|
            "#,
 | 
						|
            expect![[r#"
 | 
						|
                ex core::ops::Deref::deref(&t) (use core::ops::Deref) [type_could_unify]
 | 
						|
                lc m [local]
 | 
						|
                lc t [local]
 | 
						|
                lc &t [type+local]
 | 
						|
                st S []
 | 
						|
                st &S [type]
 | 
						|
                st S []
 | 
						|
                st &S [type]
 | 
						|
                st T []
 | 
						|
                st &T [type]
 | 
						|
                fn foo(…) []
 | 
						|
                fn main() []
 | 
						|
                md core []
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn suggest_deref_mut() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- minicore: deref_mut
 | 
						|
struct S;
 | 
						|
struct T(S);
 | 
						|
 | 
						|
impl core::ops::Deref for T {
 | 
						|
    type Target = S;
 | 
						|
 | 
						|
    fn deref(&self) -> &Self::Target {
 | 
						|
        &self.0
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
impl core::ops::DerefMut for T {
 | 
						|
    fn deref_mut(&mut self) -> &mut Self::Target {
 | 
						|
        &mut self.0
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn foo(s: &mut S) {}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    let t = T(S);
 | 
						|
    let m = 123;
 | 
						|
 | 
						|
    foo($0);
 | 
						|
}
 | 
						|
            "#,
 | 
						|
            expect![[r#"
 | 
						|
                ex core::ops::DerefMut::deref_mut(&mut t) (use core::ops::DerefMut) [type_could_unify]
 | 
						|
                lc m [local]
 | 
						|
                lc t [local]
 | 
						|
                lc &mut t [type+local]
 | 
						|
                st S []
 | 
						|
                st &mut S [type]
 | 
						|
                st S []
 | 
						|
                st &mut S [type]
 | 
						|
                st T []
 | 
						|
                st &mut T [type]
 | 
						|
                fn foo(…) []
 | 
						|
                fn main() []
 | 
						|
                md core []
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn locals() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
fn foo(bar: u32) {
 | 
						|
    let baz = 0;
 | 
						|
 | 
						|
    f$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                lc baz [local]
 | 
						|
                lc bar [local]
 | 
						|
                fn foo(…) []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn enum_owned() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
enum Foo { A, B }
 | 
						|
fn foo() {
 | 
						|
    bar($0);
 | 
						|
}
 | 
						|
fn bar(t: Foo) {}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                ev Foo::A [type]
 | 
						|
                ev Foo::B [type]
 | 
						|
                en Foo [type]
 | 
						|
                ex Foo::A [type]
 | 
						|
                ex Foo::B [type]
 | 
						|
                fn bar(…) []
 | 
						|
                fn foo() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn enum_ref() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
enum Foo { A, B }
 | 
						|
fn foo() {
 | 
						|
    bar($0);
 | 
						|
}
 | 
						|
fn bar(t: &Foo) {}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                ev Foo::A []
 | 
						|
                ev &Foo::A [type]
 | 
						|
                ev Foo::B []
 | 
						|
                ev &Foo::B [type]
 | 
						|
                en Foo []
 | 
						|
                en &Foo [type]
 | 
						|
                fn bar(…) []
 | 
						|
                fn foo() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn suggest_deref_fn_ret() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
//- minicore: deref
 | 
						|
struct S;
 | 
						|
struct T(S);
 | 
						|
 | 
						|
impl core::ops::Deref for T {
 | 
						|
    type Target = S;
 | 
						|
 | 
						|
    fn deref(&self) -> &Self::Target {
 | 
						|
        &self.0
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn foo(s: &S) {}
 | 
						|
fn bar() -> T {}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    foo($0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                ex core::ops::Deref::deref(&bar()) (use core::ops::Deref) [type_could_unify]
 | 
						|
                st S []
 | 
						|
                st &S [type]
 | 
						|
                st S []
 | 
						|
                st &S [type]
 | 
						|
                st T []
 | 
						|
                st &T [type]
 | 
						|
                fn bar() []
 | 
						|
                fn &bar() [type]
 | 
						|
                fn foo(…) []
 | 
						|
                fn main() []
 | 
						|
                md core []
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn op_function_relevances() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
#[lang = "sub"]
 | 
						|
trait Sub {
 | 
						|
    fn sub(self, other: Self) -> Self { self }
 | 
						|
}
 | 
						|
impl Sub for u32 {}
 | 
						|
fn foo(a: u32) { a.$0 }
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                me sub(…) (as Sub) [op_method]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct Foo;
 | 
						|
impl Foo {
 | 
						|
    fn new() -> Self {}
 | 
						|
}
 | 
						|
#[lang = "eq"]
 | 
						|
pub trait PartialEq<Rhs: ?Sized = Self> {
 | 
						|
    fn eq(&self, other: &Rhs) -> bool;
 | 
						|
    fn ne(&self, other: &Rhs) -> bool;
 | 
						|
}
 | 
						|
 | 
						|
impl PartialEq for Foo {}
 | 
						|
fn main() {
 | 
						|
    Foo::$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fn new() []
 | 
						|
                me eq(…) (as PartialEq) [op_method]
 | 
						|
                me ne(…) (as PartialEq) [op_method]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn constructor_order_simple() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct Foo;
 | 
						|
struct Other;
 | 
						|
struct Option<T>(T);
 | 
						|
 | 
						|
impl Foo {
 | 
						|
    fn fn_ctr() -> Foo { unimplemented!() }
 | 
						|
    fn fn_another(n: u32) -> Other { unimplemented!() }
 | 
						|
    fn fn_ctr_self() -> Option<Self> { unimplemented!() }
 | 
						|
}
 | 
						|
 | 
						|
fn test() {
 | 
						|
    let a = Foo::$0;
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                fn fn_ctr() [type_could_unify]
 | 
						|
                fn fn_ctr_self() [type_could_unify]
 | 
						|
                fn fn_another(…) [type_could_unify]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn constructor_order_kind() {
 | 
						|
        check_function_relevance(
 | 
						|
            r#"
 | 
						|
struct Foo;
 | 
						|
struct Bar;
 | 
						|
struct Option<T>(T);
 | 
						|
enum Result<T, E> { Ok(T), Err(E) };
 | 
						|
 | 
						|
impl Foo {
 | 
						|
    fn fn_ctr(&self) -> Foo { unimplemented!() }
 | 
						|
    fn fn_ctr_with_args(&self, n: u32) -> Foo { unimplemented!() }
 | 
						|
    fn fn_another(&self, n: u32) -> Bar { unimplemented!() }
 | 
						|
    fn fn_ctr_wrapped(&self, ) -> Option<Self> { unimplemented!() }
 | 
						|
    fn fn_ctr_wrapped_2(&self, ) -> Result<Self, Bar> { unimplemented!() }
 | 
						|
    fn fn_ctr_wrapped_3(&self, ) -> Result<Bar, Self> { unimplemented!() } // Self is not the first type
 | 
						|
    fn fn_ctr_wrapped_with_args(&self, m: u32) -> Option<Self> { unimplemented!() }
 | 
						|
    fn fn_another_unit(&self) { unimplemented!() }
 | 
						|
}
 | 
						|
 | 
						|
fn test() {
 | 
						|
    let a = self::Foo::$0;
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    (
 | 
						|
                        "fn(&self, u32) -> Bar",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: Other,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                    (
 | 
						|
                        "fn(&self)",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: Other,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                    (
 | 
						|
                        "fn(&self) -> Foo",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: DirectConstructor,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                    (
 | 
						|
                        "fn(&self, u32) -> Foo",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: DirectConstructor,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                    (
 | 
						|
                        "fn(&self) -> Option<Foo>",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: Constructor,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                    (
 | 
						|
                        "fn(&self) -> Result<Foo, Bar>",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: Constructor,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                    (
 | 
						|
                        "fn(&self) -> Result<Bar, Foo>",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: Constructor,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                    (
 | 
						|
                        "fn(&self, u32) -> Option<Foo>",
 | 
						|
                        Some(
 | 
						|
                            CompletionRelevanceFn {
 | 
						|
                                has_params: true,
 | 
						|
                                has_self_param: true,
 | 
						|
                                return_type: Constructor,
 | 
						|
                            },
 | 
						|
                        ),
 | 
						|
                    ),
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn constructor_order_relevance() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct Foo;
 | 
						|
struct FooBuilder;
 | 
						|
struct Result<T>(T);
 | 
						|
 | 
						|
impl Foo {
 | 
						|
    fn fn_no_ret(&self) {}
 | 
						|
    fn fn_ctr_with_args(input: u32) -> Foo { unimplemented!() }
 | 
						|
    fn fn_direct_ctr() -> Self { unimplemented!() }
 | 
						|
    fn fn_ctr() -> Result<Self> { unimplemented!() }
 | 
						|
    fn fn_other() -> Result<u32> { unimplemented!() }
 | 
						|
    fn fn_builder() -> FooBuilder { unimplemented!() }
 | 
						|
}
 | 
						|
 | 
						|
fn test() {
 | 
						|
    let a = self::Foo::$0;
 | 
						|
}
 | 
						|
"#,
 | 
						|
            // preference:
 | 
						|
            // Direct Constructor
 | 
						|
            // Direct Constructor with args
 | 
						|
            // Builder
 | 
						|
            // Constructor
 | 
						|
            // Others
 | 
						|
            expect![[r#"
 | 
						|
                fn fn_direct_ctr() [type_could_unify]
 | 
						|
                fn fn_ctr_with_args(…) [type_could_unify]
 | 
						|
                fn fn_builder() [type_could_unify]
 | 
						|
                fn fn_ctr() [type_could_unify]
 | 
						|
                me fn_no_ret(…) [type_could_unify]
 | 
						|
                fn fn_other() [type_could_unify]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
 | 
						|
        //
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn function_relevance_generic_1() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct Foo<T: Default>(T);
 | 
						|
struct FooBuilder;
 | 
						|
struct Option<T>(T);
 | 
						|
enum Result<T, E>{Ok(T), Err(E)};
 | 
						|
 | 
						|
impl<T: Default> Foo<T> {
 | 
						|
    fn fn_returns_unit(&self) {}
 | 
						|
    fn fn_ctr_with_args(input: T) -> Foo<T> { unimplemented!() }
 | 
						|
    fn fn_direct_ctr() -> Self { unimplemented!() }
 | 
						|
    fn fn_ctr_wrapped() -> Option<Self> { unimplemented!() }
 | 
						|
    fn fn_ctr_wrapped_2() -> Result<Self, u32> { unimplemented!() }
 | 
						|
    fn fn_other() -> Option<u32> { unimplemented!() }
 | 
						|
    fn fn_builder() -> FooBuilder { unimplemented!() }
 | 
						|
}
 | 
						|
 | 
						|
fn test() {
 | 
						|
    let a = self::Foo::<u32>::$0;
 | 
						|
}
 | 
						|
                "#,
 | 
						|
            expect![[r#"
 | 
						|
                        fn fn_direct_ctr() [type_could_unify]
 | 
						|
                        fn fn_ctr_with_args(…) [type_could_unify]
 | 
						|
                        fn fn_builder() [type_could_unify]
 | 
						|
                        fn fn_ctr_wrapped() [type_could_unify]
 | 
						|
                        fn fn_ctr_wrapped_2() [type_could_unify]
 | 
						|
                        me fn_returns_unit(…) [type_could_unify]
 | 
						|
                        fn fn_other() [type_could_unify]
 | 
						|
                    "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn function_relevance_generic_2() {
 | 
						|
        // Generic 2
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
struct Foo<T: Default>(T);
 | 
						|
struct FooBuilder;
 | 
						|
struct Option<T>(T);
 | 
						|
enum Result<T, E>{Ok(T), Err(E)};
 | 
						|
 | 
						|
impl<T: Default> Foo<T> {
 | 
						|
    fn fn_no_ret(&self) {}
 | 
						|
    fn fn_ctr_with_args(input: T) -> Foo<T> { unimplemented!() }
 | 
						|
    fn fn_direct_ctr() -> Self { unimplemented!() }
 | 
						|
    fn fn_ctr() -> Option<Self> { unimplemented!() }
 | 
						|
    fn fn_ctr2() -> Result<Self, u32> { unimplemented!() }
 | 
						|
    fn fn_other() -> Option<u32> { unimplemented!() }
 | 
						|
    fn fn_builder() -> FooBuilder { unimplemented!() }
 | 
						|
}
 | 
						|
 | 
						|
fn test() {
 | 
						|
    let a : Res<Foo<u32>> = Foo::$0;
 | 
						|
}
 | 
						|
                "#,
 | 
						|
            expect![[r#"
 | 
						|
                fn fn_direct_ctr() [type_could_unify]
 | 
						|
                fn fn_ctr_with_args(…) [type_could_unify]
 | 
						|
                fn fn_builder() [type_could_unify]
 | 
						|
                fn fn_ctr() [type_could_unify]
 | 
						|
                fn fn_ctr2() [type_could_unify]
 | 
						|
                me fn_no_ret(…) [type_could_unify]
 | 
						|
                fn fn_other() [type_could_unify]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn struct_field_method_ref() {
 | 
						|
        check_kinds(
 | 
						|
            r#"
 | 
						|
struct Foo { bar: u32, qux: fn() }
 | 
						|
impl Foo { fn baz(&self) -> u32 { 0 } }
 | 
						|
 | 
						|
fn foo(f: Foo) { let _: &u32 = f.b$0 }
 | 
						|
"#,
 | 
						|
            &[
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Method),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Field),
 | 
						|
            ],
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "baz()",
 | 
						|
                        source_range: 109..110,
 | 
						|
                        delete: 109..110,
 | 
						|
                        insert: "baz()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Method,
 | 
						|
                        ),
 | 
						|
                        lookup: "baz",
 | 
						|
                        detail: "fn(&self) -> u32",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: None,
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: Some(
 | 
						|
                                CompletionRelevanceFn {
 | 
						|
                                    has_params: true,
 | 
						|
                                    has_self_param: true,
 | 
						|
                                    return_type: Other,
 | 
						|
                                },
 | 
						|
                            ),
 | 
						|
                        },
 | 
						|
                        ref_match: "&@107",
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "bar",
 | 
						|
                        source_range: 109..110,
 | 
						|
                        delete: 109..110,
 | 
						|
                        insert: "bar",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Field,
 | 
						|
                        ),
 | 
						|
                        detail: "u32",
 | 
						|
                        ref_match: "&@107",
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "qux",
 | 
						|
                        source_range: 109..110,
 | 
						|
                        text_edit: TextEdit {
 | 
						|
                            indels: [
 | 
						|
                                Indel {
 | 
						|
                                    insert: "(",
 | 
						|
                                    delete: 107..107,
 | 
						|
                                },
 | 
						|
                                Indel {
 | 
						|
                                    insert: "qux)()",
 | 
						|
                                    delete: 109..110,
 | 
						|
                                },
 | 
						|
                            ],
 | 
						|
                        },
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Field,
 | 
						|
                        ),
 | 
						|
                        detail: "fn()",
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn expected_fn_type_ref() {
 | 
						|
        check_kinds(
 | 
						|
            r#"
 | 
						|
struct S { field: fn() }
 | 
						|
 | 
						|
fn foo() {
 | 
						|
    let foo: fn() = S { fields: || {}}.fi$0;
 | 
						|
}
 | 
						|
"#,
 | 
						|
            &[CompletionItemKind::SymbolKind(SymbolKind::Field)],
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "field",
 | 
						|
                        source_range: 76..78,
 | 
						|
                        delete: 76..78,
 | 
						|
                        insert: "field",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Field,
 | 
						|
                        ),
 | 
						|
                        detail: "fn()",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: Some(
 | 
						|
                                Exact,
 | 
						|
                            ),
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: None,
 | 
						|
                        },
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn qualified_path_ref() {
 | 
						|
        check_kinds(
 | 
						|
            r#"
 | 
						|
struct S;
 | 
						|
 | 
						|
struct T;
 | 
						|
impl T {
 | 
						|
    fn foo() -> S {}
 | 
						|
}
 | 
						|
 | 
						|
fn bar(s: &S) {}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    bar(T::$0);
 | 
						|
}
 | 
						|
"#,
 | 
						|
            &[CompletionItemKind::SymbolKind(SymbolKind::Function)],
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "foo()",
 | 
						|
                        source_range: 95..95,
 | 
						|
                        delete: 95..95,
 | 
						|
                        insert: "foo()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Function,
 | 
						|
                        ),
 | 
						|
                        lookup: "foo",
 | 
						|
                        detail: "fn() -> S",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: None,
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: false,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: Some(
 | 
						|
                                CompletionRelevanceFn {
 | 
						|
                                    has_params: false,
 | 
						|
                                    has_self_param: false,
 | 
						|
                                    return_type: Other,
 | 
						|
                                },
 | 
						|
                            ),
 | 
						|
                        },
 | 
						|
                        ref_match: "&@92",
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn generic_enum() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
enum Foo<T> { A(T), B }
 | 
						|
// bar() should not be an exact type match
 | 
						|
// because the generic parameters are different
 | 
						|
fn bar() -> Foo<u8> { Foo::B }
 | 
						|
// FIXME baz() should be an exact type match
 | 
						|
// because the types could unify, but it currently
 | 
						|
// is not. This is due to the T here being
 | 
						|
// TyKind::Placeholder rather than TyKind::Missing.
 | 
						|
fn baz<T>() -> Foo<T> { Foo::B }
 | 
						|
fn foo() {
 | 
						|
    let foo: Foo<u32> = Foo::B;
 | 
						|
    let _: Foo<u32> = f$0;
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                lc foo [type+local]
 | 
						|
                ex foo [type]
 | 
						|
                ex Foo::B [type]
 | 
						|
                ev Foo::A(…) [type_could_unify]
 | 
						|
                ev Foo::B [type_could_unify]
 | 
						|
                en Foo [type_could_unify]
 | 
						|
                fn foo() []
 | 
						|
                fn bar() []
 | 
						|
                fn baz() []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn postfix_exact_match_is_high_priority() {
 | 
						|
        cov_mark::check!(postfix_exact_match_is_high_priority);
 | 
						|
        check_relevance_for_kinds(
 | 
						|
            r#"
 | 
						|
mod ops {
 | 
						|
    pub trait Not {
 | 
						|
        type Output;
 | 
						|
        fn not(self) -> Self::Output;
 | 
						|
    }
 | 
						|
 | 
						|
    impl Not for bool {
 | 
						|
        type Output = bool;
 | 
						|
        fn not(self) -> bool { if self { false } else { true }}
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    let _: bool = (9 > 2).not$0;
 | 
						|
}
 | 
						|
    "#,
 | 
						|
            &[CompletionItemKind::Snippet, CompletionItemKind::SymbolKind(SymbolKind::Method)],
 | 
						|
            expect![[r#"
 | 
						|
                sn not [snippet]
 | 
						|
                me not() (use ops::Not) [type_could_unify+requires_import]
 | 
						|
                sn if []
 | 
						|
                sn while []
 | 
						|
                sn ref []
 | 
						|
                sn refm []
 | 
						|
                sn deref []
 | 
						|
                sn unsafe []
 | 
						|
                sn match []
 | 
						|
                sn box []
 | 
						|
                sn dbg []
 | 
						|
                sn dbgr []
 | 
						|
                sn call []
 | 
						|
                sn return []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn postfix_inexact_match_is_low_priority() {
 | 
						|
        cov_mark::check!(postfix_inexact_match_is_low_priority);
 | 
						|
        check_relevance_for_kinds(
 | 
						|
            r#"
 | 
						|
struct S;
 | 
						|
impl S {
 | 
						|
    fn f(&self) {}
 | 
						|
}
 | 
						|
fn main() {
 | 
						|
    S.$0
 | 
						|
}
 | 
						|
    "#,
 | 
						|
            &[CompletionItemKind::Snippet, CompletionItemKind::SymbolKind(SymbolKind::Method)],
 | 
						|
            expect![[r#"
 | 
						|
                me f() []
 | 
						|
                sn ref []
 | 
						|
                sn refm []
 | 
						|
                sn deref []
 | 
						|
                sn unsafe []
 | 
						|
                sn match []
 | 
						|
                sn box []
 | 
						|
                sn dbg []
 | 
						|
                sn dbgr []
 | 
						|
                sn call []
 | 
						|
                sn let []
 | 
						|
                sn letm []
 | 
						|
                sn return []
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn flyimport_reduced_relevance() {
 | 
						|
        check_relevance(
 | 
						|
            r#"
 | 
						|
mod std {
 | 
						|
    pub mod io {
 | 
						|
        pub trait BufRead {}
 | 
						|
        pub struct BufReader;
 | 
						|
        pub struct BufWriter;
 | 
						|
    }
 | 
						|
}
 | 
						|
struct Buffer;
 | 
						|
 | 
						|
fn f() {
 | 
						|
    Buf$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            expect![[r#"
 | 
						|
                st Buffer []
 | 
						|
                fn f() []
 | 
						|
                md std []
 | 
						|
                tt BufRead (use std::io::BufRead) [requires_import]
 | 
						|
                st BufReader (use std::io::BufReader) [requires_import]
 | 
						|
                st BufWriter (use std::io::BufWriter) [requires_import]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn completes_struct_with_raw_identifier() {
 | 
						|
        check_edit(
 | 
						|
            "type",
 | 
						|
            r#"
 | 
						|
mod m { pub struct r#type {} }
 | 
						|
fn main() {
 | 
						|
    let r#type = m::t$0;
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
mod m { pub struct r#type {} }
 | 
						|
fn main() {
 | 
						|
    let r#type = m::r#type;
 | 
						|
}
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn completes_fn_with_raw_identifier() {
 | 
						|
        check_edit(
 | 
						|
            "type",
 | 
						|
            r#"
 | 
						|
mod m { pub fn r#type {} }
 | 
						|
fn main() {
 | 
						|
    m::t$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
mod m { pub fn r#type {} }
 | 
						|
fn main() {
 | 
						|
    m::r#type()$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn completes_macro_with_raw_identifier() {
 | 
						|
        check_edit(
 | 
						|
            "let!",
 | 
						|
            r#"
 | 
						|
macro_rules! r#let { () => {} }
 | 
						|
fn main() {
 | 
						|
    $0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
macro_rules! r#let { () => {} }
 | 
						|
fn main() {
 | 
						|
    r#let!($0)
 | 
						|
}
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn completes_variant_with_raw_identifier() {
 | 
						|
        check_edit(
 | 
						|
            "type",
 | 
						|
            r#"
 | 
						|
enum A { r#type }
 | 
						|
fn main() {
 | 
						|
    let a = A::t$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
enum A { r#type }
 | 
						|
fn main() {
 | 
						|
    let a = A::r#type$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn completes_field_with_raw_identifier() {
 | 
						|
        check_edit(
 | 
						|
            "fn",
 | 
						|
            r#"
 | 
						|
mod r#type {
 | 
						|
    pub struct r#struct {
 | 
						|
        pub r#fn: u32
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    let a = r#type::r#struct {};
 | 
						|
    a.$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
mod r#type {
 | 
						|
    pub struct r#struct {
 | 
						|
        pub r#fn: u32
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    let a = r#type::r#struct {};
 | 
						|
    a.r#fn
 | 
						|
}
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn completes_const_with_raw_identifier() {
 | 
						|
        check_edit(
 | 
						|
            "type",
 | 
						|
            r#"
 | 
						|
struct r#struct {}
 | 
						|
impl r#struct { pub const r#type: u8 = 1; }
 | 
						|
fn main() {
 | 
						|
    r#struct::t$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct r#struct {}
 | 
						|
impl r#struct { pub const r#type: u8 = 1; }
 | 
						|
fn main() {
 | 
						|
    r#struct::r#type
 | 
						|
}
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn completes_type_alias_with_raw_identifier() {
 | 
						|
        check_edit(
 | 
						|
            "type type",
 | 
						|
            r#"
 | 
						|
struct r#struct {}
 | 
						|
trait r#trait { type r#type; }
 | 
						|
impl r#trait for r#struct { type t$0 }
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct r#struct {}
 | 
						|
trait r#trait { type r#type; }
 | 
						|
impl r#trait for r#struct { type r#type = $0; }
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn field_access_includes_self() {
 | 
						|
        check_edit(
 | 
						|
            "length",
 | 
						|
            r#"
 | 
						|
struct S {
 | 
						|
    length: i32
 | 
						|
}
 | 
						|
 | 
						|
impl S {
 | 
						|
    fn some_fn(&self) {
 | 
						|
        let l = len$0
 | 
						|
    }
 | 
						|
}
 | 
						|
"#,
 | 
						|
            r#"
 | 
						|
struct S {
 | 
						|
    length: i32
 | 
						|
}
 | 
						|
 | 
						|
impl S {
 | 
						|
    fn some_fn(&self) {
 | 
						|
        let l = self.length
 | 
						|
    }
 | 
						|
}
 | 
						|
"#,
 | 
						|
        )
 | 
						|
    }
 | 
						|
 | 
						|
    #[test]
 | 
						|
    fn notable_traits_method_relevance() {
 | 
						|
        check_kinds(
 | 
						|
            r#"
 | 
						|
#[doc(notable_trait)]
 | 
						|
trait Write {
 | 
						|
    fn write(&self);
 | 
						|
    fn flush(&self);
 | 
						|
}
 | 
						|
 | 
						|
struct Writer;
 | 
						|
 | 
						|
impl Write for Writer {
 | 
						|
    fn write(&self) {}
 | 
						|
    fn flush(&self) {}
 | 
						|
}
 | 
						|
 | 
						|
fn main() {
 | 
						|
    Writer.$0
 | 
						|
}
 | 
						|
"#,
 | 
						|
            &[
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Method),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Field),
 | 
						|
                CompletionItemKind::SymbolKind(SymbolKind::Function),
 | 
						|
            ],
 | 
						|
            expect![[r#"
 | 
						|
                [
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "flush()",
 | 
						|
                        source_range: 193..193,
 | 
						|
                        delete: 193..193,
 | 
						|
                        insert: "flush()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Method,
 | 
						|
                        ),
 | 
						|
                        lookup: "flush",
 | 
						|
                        detail: "fn(&self)",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: None,
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: true,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: None,
 | 
						|
                        },
 | 
						|
                    },
 | 
						|
                    CompletionItem {
 | 
						|
                        label: "write()",
 | 
						|
                        source_range: 193..193,
 | 
						|
                        delete: 193..193,
 | 
						|
                        insert: "write()$0",
 | 
						|
                        kind: SymbolKind(
 | 
						|
                            Method,
 | 
						|
                        ),
 | 
						|
                        lookup: "write",
 | 
						|
                        detail: "fn(&self)",
 | 
						|
                        relevance: CompletionRelevance {
 | 
						|
                            exact_name_match: false,
 | 
						|
                            type_match: None,
 | 
						|
                            is_local: false,
 | 
						|
                            is_item_from_trait: false,
 | 
						|
                            is_item_from_notable_trait: true,
 | 
						|
                            is_name_already_imported: false,
 | 
						|
                            requires_import: false,
 | 
						|
                            is_op_method: false,
 | 
						|
                            is_private_editable: false,
 | 
						|
                            postfix_match: None,
 | 
						|
                            is_definite: false,
 | 
						|
                            function: None,
 | 
						|
                        },
 | 
						|
                    },
 | 
						|
                ]
 | 
						|
            "#]],
 | 
						|
        );
 | 
						|
    }
 | 
						|
}
 |