1 use ide_db::defs::{Definition, NameRefClass};
2 use syntax::{ast, AstNode, SyntaxKind, T};
5 assist_context::{AssistContext, Assists},
9 // Assist: add_turbo_fish
11 // Adds `::<_>` to a call of a generic method or function.
14 // fn make<T>() -> T { todo!() }
21 // fn make<T>() -> T { todo!() }
23 // let x = make::<${0:_}>();
26 pub(crate) fn add_turbo_fish(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
27 let ident = ctx.find_token_syntax_at_offset(SyntaxKind::IDENT).or_else(|| {
28 let arg_list = ctx.find_node_at_offset::<ast::ArgList>()?;
29 if arg_list.args().next().is_some() {
32 cov_mark::hit!(add_turbo_fish_after_call);
33 cov_mark::hit!(add_type_ascription_after_call);
34 arg_list.l_paren_token()?.prev_token().filter(|it| it.kind() == SyntaxKind::IDENT)
36 let next_token = ident.next_token()?;
37 if next_token.kind() == T![::] {
38 cov_mark::hit!(add_turbo_fish_one_fish_is_enough);
41 let name_ref = ast::NameRef::cast(ident.parent()?)?;
42 let def = match NameRefClass::classify(&ctx.sema, &name_ref)? {
43 NameRefClass::Definition(def) => def,
44 NameRefClass::FieldShorthand { .. } => return None,
47 Definition::Function(it) => it,
50 let generics = hir::GenericDef::Function(fun).params(ctx.sema.db);
51 if generics.is_empty() {
52 cov_mark::hit!(add_turbo_fish_non_generic);
56 if let Some(let_stmt) = ctx.find_node_at_offset::<ast::LetStmt>() {
57 if let_stmt.colon_token().is_none() {
58 let type_pos = let_stmt.pat()?.syntax().last_token()?.text_range().end();
59 let semi_pos = let_stmt.syntax().last_token()?.text_range().end();
62 AssistId("add_type_ascription", AssistKind::RefactorRewrite),
63 "Add `: _` before assignment operator",
66 if let_stmt.semicolon_token().is_none() {
67 builder.insert(semi_pos, ";");
69 match ctx.config.snippet_cap {
70 Some(cap) => builder.insert_snippet(cap, type_pos, ": ${0:_}"),
71 None => builder.insert(type_pos, ": _"),
76 cov_mark::hit!(add_type_ascription_already_typed);
80 let number_of_arguments = generics
83 matches!(param, hir::GenericParam::TypeParam(_) | hir::GenericParam::ConstParam(_))
86 let fish_head = std::iter::repeat("_").take(number_of_arguments).collect::<Vec<_>>().join(",");
89 AssistId("add_turbo_fish", AssistKind::RefactorRewrite),
92 |builder| match ctx.config.snippet_cap {
94 let snip = format!("::<${{0:{}}}>", fish_head);
95 builder.insert_snippet(cap, ident.text_range().end(), snip)
98 let snip = format!("::<{}>", fish_head);
99 builder.insert(ident.text_range().end(), snip);
107 use crate::tests::{check_assist, check_assist_by_label, check_assist_not_applicable};
112 fn add_turbo_fish_function() {
131 fn add_turbo_fish_function_multiple_generic_types() {
135 fn make<T, A>() -> T {}
141 fn make<T, A>() -> T {}
150 fn add_turbo_fish_function_many_generic_types() {
154 fn make<T, A, B, C, D, E, F>() -> T {}
160 fn make<T, A, B, C, D, E, F>() -> T {}
162 make::<${0:_,_,_,_,_,_,_}>();
169 fn add_turbo_fish_after_call() {
170 cov_mark::check!(add_turbo_fish_after_call);
189 fn add_turbo_fish_method() {
195 fn make<T>(&self) -> T {}
204 fn make<T>(&self) -> T {}
214 fn add_turbo_fish_one_fish_is_enough() {
215 cov_mark::check!(add_turbo_fish_one_fish_is_enough);
216 check_assist_not_applicable(
228 fn add_turbo_fish_non_generic() {
229 cov_mark::check!(add_turbo_fish_non_generic);
230 check_assist_not_applicable(
242 fn add_type_ascription_function() {
243 check_assist_by_label(
254 let x: ${0:_} = make();
257 "Add `: _` before assignment operator",
262 fn add_type_ascription_after_call() {
263 cov_mark::check!(add_type_ascription_after_call);
264 check_assist_by_label(
275 let x: ${0:_} = make();
278 "Add `: _` before assignment operator",
283 fn add_type_ascription_method() {
284 check_assist_by_label(
289 fn make<T>(&self) -> T {}
298 fn make<T>(&self) -> T {}
301 let x: ${0:_} = S.make();
304 "Add `: _` before assignment operator",
309 fn add_type_ascription_already_typed() {
310 cov_mark::check!(add_type_ascription_already_typed);
316 let x: () = make$0();
322 let x: () = make::<${0:_}>();
329 fn add_type_ascription_append_semicolon() {
330 check_assist_by_label(
341 let x: ${0:_} = make();
344 "Add `: _` before assignment operator",
349 fn add_turbo_fish_function_lifetime_parameter() {
353 fn make<'a, T, A>(t: T, a: A) {}
359 fn make<'a, T, A>(t: T, a: A) {}
361 make::<${0:_,_}>(5, 2);
368 fn add_turbo_fish_function_const_parameter() {
372 fn make<T, const N: usize>(t: T) {}
378 fn make<T, const N: usize>(t: T) {}