2020-05-19 22:07:00 +00:00
|
|
|
use ra_ide_db::defs::{classify_name_ref, Definition, NameRefClass};
|
|
|
|
use ra_syntax::{ast, AstNode, SyntaxKind, T};
|
2020-05-19 22:19:59 +00:00
|
|
|
use test_utils::mark;
|
2020-05-19 22:07:00 +00:00
|
|
|
|
|
|
|
use crate::{
|
|
|
|
assist_context::{AssistContext, Assists},
|
|
|
|
AssistId,
|
|
|
|
};
|
|
|
|
|
|
|
|
// Assist: add_turbo_fish
|
|
|
|
//
|
|
|
|
// Adds `::<_>` to a call of a generic method or function.
|
|
|
|
//
|
|
|
|
// ```
|
|
|
|
// fn make<T>() -> T { todo!() }
|
|
|
|
// fn main() {
|
|
|
|
// let x = make<|>();
|
|
|
|
// }
|
|
|
|
// ```
|
|
|
|
// ->
|
|
|
|
// ```
|
|
|
|
// fn make<T>() -> T { todo!() }
|
|
|
|
// fn main() {
|
|
|
|
// let x = make::<${0:_}>();
|
|
|
|
// }
|
|
|
|
// ```
|
|
|
|
pub(crate) fn add_turbo_fish(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
|
|
|
|
let ident = ctx.find_token_at_offset(SyntaxKind::IDENT)?;
|
|
|
|
let next_token = ident.next_token()?;
|
|
|
|
if next_token.kind() == T![::] {
|
2020-05-19 22:19:59 +00:00
|
|
|
mark::hit!(add_turbo_fish_one_fish_is_enough);
|
2020-05-19 22:07:00 +00:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
let name_ref = ast::NameRef::cast(ident.parent())?;
|
|
|
|
let def = match classify_name_ref(&ctx.sema, &name_ref)? {
|
|
|
|
NameRefClass::Definition(def) => def,
|
|
|
|
NameRefClass::FieldShorthand { .. } => return None,
|
|
|
|
};
|
|
|
|
let fun = match def {
|
|
|
|
Definition::ModuleDef(hir::ModuleDef::Function(it)) => it,
|
|
|
|
_ => return None,
|
|
|
|
};
|
|
|
|
let generics = hir::GenericDef::Function(fun).params(ctx.sema.db);
|
|
|
|
if generics.is_empty() {
|
2020-05-19 22:19:59 +00:00
|
|
|
mark::hit!(add_turbo_fish_non_generic);
|
2020-05-19 22:07:00 +00:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
acc.add(AssistId("add_turbo_fish"), "Add `::<>`", ident.text_range(), |builder| {
|
|
|
|
match ctx.config.snippet_cap {
|
|
|
|
Some(cap) => builder.insert_snippet(cap, ident.text_range().end(), "::<${0:_}>"),
|
|
|
|
None => builder.insert(ident.text_range().end(), "::<_>"),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use crate::tests::{check_assist, check_assist_not_applicable};
|
|
|
|
|
|
|
|
use super::*;
|
2020-05-19 22:19:59 +00:00
|
|
|
use test_utils::mark;
|
2020-05-19 22:07:00 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn add_turbo_fish_function() {
|
|
|
|
check_assist(
|
|
|
|
add_turbo_fish,
|
|
|
|
r#"
|
|
|
|
fn make<T>() -> T {}
|
|
|
|
fn main() {
|
|
|
|
make<|>();
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
fn make<T>() -> T {}
|
|
|
|
fn main() {
|
|
|
|
make::<${0:_}>();
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn add_turbo_fish_method() {
|
|
|
|
check_assist(
|
|
|
|
add_turbo_fish,
|
|
|
|
r#"
|
|
|
|
struct S;
|
|
|
|
impl S {
|
|
|
|
fn make<T>(&self) -> T {}
|
|
|
|
}
|
|
|
|
fn main() {
|
|
|
|
S.make<|>();
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct S;
|
|
|
|
impl S {
|
|
|
|
fn make<T>(&self) -> T {}
|
|
|
|
}
|
|
|
|
fn main() {
|
|
|
|
S.make::<${0:_}>();
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn add_turbo_fish_one_fish_is_enough() {
|
2020-05-19 22:19:59 +00:00
|
|
|
mark::check!(add_turbo_fish_one_fish_is_enough);
|
2020-05-19 22:07:00 +00:00
|
|
|
check_assist_not_applicable(
|
|
|
|
add_turbo_fish,
|
|
|
|
r#"
|
|
|
|
fn make<T>() -> T {}
|
|
|
|
fn main() {
|
|
|
|
make<|>::<()>();
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn add_turbo_fish_non_generic() {
|
2020-05-19 22:19:59 +00:00
|
|
|
mark::check!(add_turbo_fish_non_generic);
|
2020-05-19 22:07:00 +00:00
|
|
|
check_assist_not_applicable(
|
|
|
|
add_turbo_fish,
|
|
|
|
r#"
|
|
|
|
fn make() -> () {}
|
|
|
|
fn main() {
|
|
|
|
make<|>();
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|