rust/crates/ra_ide_api/src/completion/complete_scope.rs

238 lines
5.7 KiB
Rust
Raw Normal View History

use ra_text_edit::TextEditBuilder;
use ra_syntax::SmolStr;
use ra_assists::auto_import;
use crate::completion::{CompletionItem, Completions, CompletionKind, CompletionContext};
2019-01-08 22:33:36 +03:00
pub(super) fn complete_scope(acc: &mut Completions, ctx: &CompletionContext) {
if ctx.is_trivial_path {
let names = ctx.analyzer.all_names(ctx.db);
names.into_iter().for_each(|(name, res)| acc.add_resolution(ctx, name.to_string(), &res));
2019-01-08 22:33:36 +03:00
}
if let Some(name) = ctx.path_ident.as_ref() {
let import_names = ctx.analyzer.all_import_names(ctx.db, name);
import_names.into_iter().for_each(|(name, path)| {
let edit = {
let mut builder = TextEditBuilder::default();
builder.replace(ctx.source_range(), name.to_string());
auto_import::auto_import_text_edit(
ctx.token.parent(),
ctx.token.parent(),
&path,
&mut builder,
);
builder.finish()
};
// Hack: copied this check form conv.rs beacause auto import can produce edits
// that invalidate assert in conv_with.
if edit
.as_atoms()
.iter()
.filter(|atom| !ctx.source_range().is_subrange(&atom.delete))
.all(|atom| ctx.source_range().intersection(&atom.delete).is_none())
{
CompletionItem::new(
CompletionKind::Reference,
ctx.source_range(),
build_import_label(&name, &path),
)
.text_edit(edit)
.add_to(acc);
}
});
}
}
fn build_import_label(name: &str, path: &Vec<SmolStr>) -> String {
let mut buf = String::with_capacity(64);
buf.push_str(name);
buf.push_str(" (");
fmt_import_path(path, &mut buf);
buf.push_str(")");
buf
}
fn fmt_import_path(path: &Vec<SmolStr>, buf: &mut String) {
let mut segments = path.iter();
if let Some(s) = segments.next() {
buf.push_str(&s);
}
for s in segments {
buf.push_str("::");
buf.push_str(&s);
}
2019-01-08 22:33:36 +03:00
}
#[cfg(test)]
mod tests {
2019-02-24 21:54:13 +03:00
use crate::completion::{CompletionKind, check_completion};
2019-01-08 22:33:36 +03:00
fn check_reference_completion(name: &str, code: &str) {
check_completion(name, code, CompletionKind::Reference);
2019-01-08 22:33:36 +03:00
}
#[test]
fn completes_bindings_from_let() {
check_reference_completion(
"bindings_from_let",
2019-01-08 22:33:36 +03:00
r"
fn quux(x: i32) {
let y = 92;
1 + <|>;
let z = ();
}
",
);
}
#[test]
fn completes_bindings_from_if_let() {
check_reference_completion(
"bindings_from_if_let",
2019-01-08 22:33:36 +03:00
r"
fn quux() {
if let Some(x) = foo() {
let y = 92;
};
if let Some(a) = bar() {
let b = 62;
1 + <|>
}
}
",
);
}
#[test]
fn completes_bindings_from_for() {
check_reference_completion(
"bindings_from_for",
2019-01-08 22:33:36 +03:00
r"
fn quux() {
for x in &[1, 2, 3] {
<|>
}
}
",
);
}
2019-02-01 23:06:57 +01:00
#[test]
fn completes_generic_params() {
check_reference_completion(
"generic_params",
r"
fn quux<T>() {
<|>
}
",
);
}
#[test]
fn completes_generic_params_in_struct() {
check_reference_completion(
"generic_params_in_struct",
r"
struct X<T> {
x: <|>
}
",
);
}
2019-01-08 22:33:36 +03:00
#[test]
fn completes_module_items() {
check_reference_completion(
"module_items",
2019-01-08 22:33:36 +03:00
r"
struct Foo;
enum Baz {}
fn quux() {
<|>
}
",
);
}
#[test]
2019-02-04 22:09:56 +01:00
fn completes_extern_prelude() {
check_reference_completion(
"extern_prelude",
r"
//- /lib.rs
use <|>;
//- /other_crate/lib.rs
// nothing here
",
);
}
#[test]
2019-01-08 22:33:36 +03:00
fn completes_module_items_in_nested_modules() {
check_reference_completion(
"module_items_in_nested_modules",
2019-01-08 22:33:36 +03:00
r"
struct Foo;
mod m {
struct Bar;
fn quux() { <|> }
}
",
);
}
#[test]
fn completes_return_type() {
check_reference_completion(
"return_type",
2019-01-08 22:33:36 +03:00
r"
struct Foo;
fn x() -> <|>
",
)
}
#[test]
fn dont_show_both_completions_for_shadowing() {
check_reference_completion(
"dont_show_both_completions_for_shadowing",
2019-01-08 22:33:36 +03:00
r"
fn foo() {
2019-01-08 22:33:36 +03:00
let bar = 92;
{
let bar = 62;
<|>
}
}
",
)
}
#[test]
fn completes_self_in_methods() {
check_reference_completion("self_in_methods", r"impl S { fn foo(&self) { <|> } }")
2019-01-08 22:33:36 +03:00
}
2019-02-13 20:53:42 +01:00
#[test]
fn completes_prelude() {
check_reference_completion(
"completes_prelude",
"
//- /main.rs
fn foo() { let x: <|> }
//- /std/lib.rs
#[prelude_import]
use prelude::*;
mod prelude {
struct Option;
}
",
);
}
2019-01-08 22:33:36 +03:00
}