From 91861a127ada4928da09b1c9d720107650921180 Mon Sep 17 00:00:00 2001 From: anarachnid Date: Sun, 20 Aug 2023 12:53:44 -0700 Subject: [PATCH 1/6] add derive impls for NameLike --- crates/syntax/src/ast/node_ext.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/syntax/src/ast/node_ext.rs b/crates/syntax/src/ast/node_ext.rs index f81dff8840cc..329505fd9d00 100644 --- a/crates/syntax/src/ast/node_ext.rs +++ b/crates/syntax/src/ast/node_ext.rs @@ -442,7 +442,7 @@ impl ast::RecordExprField { } } -#[derive(Debug, Clone)] +#[derive(Debug, Clone, PartialEq, Eq, Hash)] pub enum NameLike { NameRef(ast::NameRef), Name(ast::Name), From 61e7121b8031fce70d036e85c7dc16ce11455e55 Mon Sep 17 00:00:00 2001 From: anarachnid Date: Sun, 20 Aug 2023 12:59:35 -0700 Subject: [PATCH 2/6] impl GenericParamsOwnerEdit for some types Union, TypeAlias, TraitAlias --- crates/syntax/src/ast/edit_in_place.rs | 92 +++++++++++++++++++++++++- 1 file changed, 91 insertions(+), 1 deletion(-) diff --git a/crates/syntax/src/ast/edit_in_place.rs b/crates/syntax/src/ast/edit_in_place.rs index 37d8212042da..ef614f813128 100644 --- a/crates/syntax/src/ast/edit_in_place.rs +++ b/crates/syntax/src/ast/edit_in_place.rs @@ -178,10 +178,93 @@ impl GenericParamsOwnerEdit for ast::Enum { self.where_clause().unwrap() } } +impl GenericParamsOwnerEdit for ast::Union { + fn get_or_create_generic_param_list(&self) -> ast::GenericParamList { + self.generic_param_list().unwrap_or_else(|| { + let position = if let Some(name) = self.name() { + Position::after(name.syntax) + } else if let Some(union_token) = self.union_token() { + Position::after(union_token) + } else { + Position::last_child_of(self.syntax()) + }; + create_generic_param_list(position) + }) + } + + fn get_or_create_where_clause(&self) -> ast::WhereClause { + self.where_clause().unwrap_or_else(|| { + let position = if let Some(gpl) = self.generic_param_list() { + Position::after(gpl.syntax()) + } else if let Some(name) = self.name() { + Position::after(name.syntax()) + } else { + Position::last_child_of(self.syntax()) + }; + create_where_clause(position) + }) + } +} +impl GenericParamsOwnerEdit for ast::TypeAlias { + fn get_or_create_generic_param_list(&self) -> ast::GenericParamList { + self.generic_param_list().unwrap_or_else(|| { + let position = if let Some(name) = self.name() { + Position::after(name.syntax) + } else if let Some(type_token) = self.type_token() { + Position::after(type_token) + } else { + Position::last_child_of(self.syntax()) + }; + create_generic_param_list(position) + }) + } + + fn get_or_create_where_clause(&self) -> ast::WhereClause { + self.where_clause().unwrap_or_else(|| { + let position = if let Some(gpl) = self.generic_param_list() { + Position::after(gpl.syntax()) + } else if let Some(name) = self.name() { + Position::after(name.syntax()) + } else { + Position::last_child_of(self.syntax()) + }; + create_where_clause(position) + }) + } +} + +impl GenericParamsOwnerEdit for ast::TraitAlias { + fn get_or_create_generic_param_list(&self) -> ast::GenericParamList { + self.generic_param_list().unwrap_or_else(|| { + let position = if let Some(name) = self.name() { + Position::after(name.syntax) + } else if let Some(trait_token) = self.trait_token() { + Position::after(trait_token) + } else { + Position::last_child_of(self.syntax()) + }; + create_generic_param_list(position) + }) + } + + fn get_or_create_where_clause(&self) -> ast::WhereClause { + self.where_clause().unwrap_or_else(|| { + let position = if let Some(gpl) = self.generic_param_list() { + Position::after(gpl.syntax()) + } else if let Some(name) = self.name() { + Position::after(name.syntax()) + } else { + Position::last_child_of(self.syntax()) + }; + create_where_clause(position) + }) + } +} -fn create_where_clause(position: Position) { +fn create_where_clause(position: Position) -> ast::WhereClause { let where_clause = make::where_clause(empty()).clone_for_update(); ted::insert(position, where_clause.syntax()); + where_clause } fn create_generic_param_list(position: Position) -> ast::GenericParamList { @@ -977,6 +1060,13 @@ mod tests { check_create_gpl::("enum E", "enum E<>"); check_create_gpl::("enum E {", "enum E<> {"); + + check_create_gpl::("union A", "union A<>"); + check_create_gpl::("union A {}", "union A<> {}"); + + check_create_gpl::("type A = B;", "type A<> = B;"); + + check_create_gpl::("trait A = B;", "trait A<> = B;"); } #[test] From 7c076b47e0b028f117e8a792c0f8f006e82a6b07 Mon Sep 17 00:00:00 2001 From: anarachnid Date: Wed, 23 Aug 2023 12:09:29 -0700 Subject: [PATCH 3/6] initial implementation --- .../src/handlers/add_generic_parameter.rs | 1800 +++++++++++++++++ crates/ide-assists/src/lib.rs | 2 + crates/ide-assists/src/tests/generated.rs | 17 + crates/ide-db/src/search.rs | 70 +- 4 files changed, 1860 insertions(+), 29 deletions(-) create mode 100644 crates/ide-assists/src/handlers/add_generic_parameter.rs diff --git a/crates/ide-assists/src/handlers/add_generic_parameter.rs b/crates/ide-assists/src/handlers/add_generic_parameter.rs new file mode 100644 index 000000000000..904345057f58 --- /dev/null +++ b/crates/ide-assists/src/handlers/add_generic_parameter.rs @@ -0,0 +1,1800 @@ +use ide_db::{ + base_db::{FileId, FileLoader, FilePosition}, + defs::{Definition, NameClass, NameRefClass}, + rename, + source_change::SourceChangeBuilder, + FxHashMap, FxHashSet, RootDatabase, +}; +use syntax::{ + ast::{self, edit_in_place::GenericParamsOwnerEdit, HasName as _}, + match_ast, + ted::{self, Position}, + AstNode, SmolStr, SyntaxKind, SyntaxNode, +}; +use text_edit::{TextRange, TextSize}; + +use crate::{AssistContext, AssistId, AssistKind, Assists}; + +// Assist: add_generic_parameter +// +// Adds a new generic parameter virally to anything that accepts one, e.g. struct, enum, union, trait, impl. +// +// ``` +// struct Channel$0(u8); +// struct Color([Channel; 3]); +// struct Image(Vec); +// ``` +// -> +// ``` +// struct Channel(u8); +// struct Color([Channel; 3]); +// struct Image(Vec>); +// ``` +pub(crate) fn add_generic_parameter(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<()> { + let offset = ctx.offset(); + + let sema = &ctx.sema; + let file_id = ctx.file_id(); + let position = FilePosition { file_id, offset }; + let source_file = sema.parse(file_id); + let syntax = source_file.syntax(); + + let file_text = ctx.db().file_text(file_id); + let file_text = file_text.as_ref(); + + let definitions = match find_definitions(sema, syntax, position) { + Err(_) => { + return None; + } + Ok(x) => x, + }; + let definitions = definitions.collect::>(); + // FIXME: should this also check if the namelike is a ast::Name? + // this would make it only appear when the cursor is on the definition of a name, rather than on uses too. + let all_definitions_can_have_generic_params = + definitions.iter().all(|(_, x)| as_definition_with_generic_params(*x).is_some()); + if !all_definitions_can_have_generic_params { + return None; + } + let definitions = definitions.into_iter(); + // FIXME: generalize this to adding and removing + // - const generic + // - lifetime generic + // - function value parameters + acc.add( + AssistId(stringify!(add_generic_parameter), AssistKind::Generate), + "Add generic parameter", + /* + see ide_db::assists::Assist.target + This should be as large as possible because adding a generic could modify an entire crate + in a similar way that renaming does. + + should this be just the name's text_range()? maybe the definition's text_range()? + */ + TextRange::up_to(TextSize::of(file_text)), + |builder| { + let snippet_cap = ctx.config.snippet_cap; + + let config = { + use ast::make::*; + // FIXME: allow user to specify the name somehow? Automatically pick a name? + Configuration { + param: { + let x = type_param(name("T1"), None); + ast::GenericParam::from(x) + }, + arg: { + let x = type_arg(ext::ty_name(name("T1"))); + ast::GenericArg::from(x) + }, + } + }; + assist_impl(builder, config, snippet_cap, sema, definitions) + }, + ) +} + +fn assist_impl( + builder: &mut SourceChangeBuilder, + config: Configuration, + snippet_cap: Option, + sema: &hir::Semantics<'_, RootDatabase>, + definitions: impl Iterator, +) { + let mut state = AssistState { + builder: SourceChangeBuilder2 { inner: builder }, + config, + snippet_cap, + sema, + aux_state: Default::default(), + }; + let mut processing_queue = ProcessingQueue::default(); + let mut acc = ChangeAccumulator::default(); + // FIXME: possibly recompute the definitions? + // to make sure they haven't changed since the user last touched anything + let r = (|| -> R { + let acc = &mut acc; + let processing_queue = &mut processing_queue; + definitions.into_iter().try_for_each(|(name, definition)| -> R { + state.on_definition(acc, processing_queue, name, definition) + })?; + 'outer: loop { + // Pop a node that off the queue that needs to have generic params added, and add them + // This might create more things on the queue. + // loop until the queue is empty. + for (file_id, q) in processing_queue.files.iter_mut() { + for (_imp, queued_nodes) in q + .inner_usage_waiting_to_see_if_a_surrounding_impl_would_have_generics + .iter_mut() + { + if let Some(x) = queued_nodes.pop() { + let UsageWaitingOnSurroudingImpl { non_impl_generic_params_node } = x; + let file_id = *file_id; + state.on_non_impl_generic_params_ancestor( + acc, + processing_queue, + file_id, + non_impl_generic_params_node, + )?; + continue 'outer; + } + } + } + // queue is empty since nothing got popped, so we are done. + break; + } + + Ok(()) + })(); + + let r = (|| -> R { + let () = r?; + // Write, to the tree, all the changes that we accumulated. + let () = state.finish(acc)?; + Ok(()) + })(); + match r { + Ok(()) => (), + Err(_x) => { + // FIXME: report the error better? + } + } +} +struct Configuration { + param: ast::GenericParam, + arg: ast::GenericArg, +} +struct SourceChangeBuilder2<'a> { + inner: &'a mut SourceChangeBuilder, +} + +// Small utility wrapper for ast::AstNode's +// ensures the node is mutable before doing mutable operations. +// requires some due diligence, since mutable methods are accessible on regular +// ast::AstNode's +#[derive(Debug, Clone, PartialEq, Eq, Hash)] +struct MutableAst(T); + +impl From for MutableAst { + fn from(x: T) -> Self { + // owned == mutable + use std::borrow::Cow; + assert!(matches!(x.syntax().green(), Cow::Owned(_))); + Self::unchecked_from(x) + } +} + +impl MutableAst { + fn unchecked_from(x: T) -> Self { + Self(x) + } +} + +impl MutableAst { + fn map(self, f: impl FnOnce(T) -> U) -> MutableAst { + let MutableAst(x) = self; + let x = f(x); + MutableAst(x) + } +} +impl<'a> SourceChangeBuilder2<'a> { + fn make_mut(&mut self, x: T) -> MutableAst { + let x = self.inner.make_mut(x); + MutableAst::from(x) + } +} +struct AssistState<'b, 'sema_a, 'sema> { + builder: SourceChangeBuilder2<'b>, + config: Configuration, + // FIXME: support snippets, when enabled and only one file + #[allow(unused)] + snippet_cap: Option, + sema: &'sema hir::Semantics<'sema_a, RootDatabase>, + aux_state: AuxilliaryState, +} +#[derive(Debug, Clone, Default)] +struct AuxilliaryState { + definitions_already_handled: FxHashSet<(Definition, SmolStr)>, + usages_already_handled: FxHashSet, +} +#[derive(Debug, Clone, Default)] +struct ProcessingQueue { + // FIXME: use nohash_hasher::IntMap; + // as of now, it is not in this crate's Cargo.toml + files: FxHashMap, +} +#[derive(Debug, Clone)] +struct UsageWaitingOnSurroudingImpl { + non_impl_generic_params_node: AnyGenericParamsOwnerEdit, +} +#[derive(Debug, Clone, Default)] +struct FileProcessingQueue { + inner_usage_waiting_to_see_if_a_surrounding_impl_would_have_generics: + FxHashMap>, +} + +#[derive(Debug, Clone, Default)] +struct ChangeAccumulator { + // FIXME: use nohash_hasher::IntMap; + // as of now, it is not in this crate's Cargo.toml + files: FxHashMap, +} +#[derive(Debug, Clone, Default)] +struct FileChangeAccumulator { + // This may never need to check for duplicates, as it hasn't encountered any yet + // so possibly change to Vec + path_segments: FxHashSet<(ast::PathSegment, Definition)>, + // encounters duplicates when multiple children need a parent to supply the generic param + // e.g. trait Stuff{ fn a(x: NeedsGeneric); fn b(x: NeedsGeneric); } + // the `trait Stuff` would be inserted twice. + defs_with_generic_params: FxHashSet, +} +#[derive(Debug, PartialEq, Eq, Copy, Clone, Hash)] +enum DefinitionWithGenericParams { + Struct(hir::Struct), + Union(hir::Union), + Enum(hir::Enum), + Trait(hir::Trait), + TraitAlias(hir::TraitAlias), + TypeAlias(hir::TypeAlias), + Function(hir::Function), +} +fn as_definition_with_generic_params( + definition: Definition, +) -> Option { + let x = match definition { + Definition::Adt(x) => match x { + hir::Adt::Struct(x) => DefinitionWithGenericParams::Struct(x), + hir::Adt::Union(x) => DefinitionWithGenericParams::Union(x), + hir::Adt::Enum(x) => DefinitionWithGenericParams::Enum(x), + }, + Definition::Trait(x) => DefinitionWithGenericParams::Trait(x), + Definition::TraitAlias(x) => DefinitionWithGenericParams::TraitAlias(x), + Definition::TypeAlias(x) => DefinitionWithGenericParams::TypeAlias(x), + Definition::Function(x) => DefinitionWithGenericParams::Function(x), + _ => return None, + }; + Some(x) +} +impl<'b, 'sema_a, 'sema> AssistState<'b, 'sema_a, 'sema> { + fn on_definition( + &mut self, + acc: &mut ChangeAccumulator, + processing_queue: &mut ProcessingQueue, + name: ast::NameLike, + definition: Definition, + ) -> R { + let tmp_text = name.text(); + let name_str = tmp_text.as_str(); + { + // cycle detection: (mutually) recursive types/traits/fns etc + // We have to key on the current name too because aliases + // have the same definition, but a different string name. + + // FIXME: don't allocate q for .contains(...) + // this is a limitation of std::borrow::Borrow, it can't return a tuple of references + let q = (definition, SmolStr::from(name_str)); + let was_already_handled = self.aux_state.definitions_already_handled.contains(&q); + if was_already_handled { + return Ok(()); + } + self.aux_state.definitions_already_handled.insert(q); + } + + (|| -> R { + macro_rules! on_def_variant { + ($ast_type: ident, $x: expr $(,)?) => {{ + let (file_id, x) = (|| -> Option<_> { + use hir::HasSource; + let src = $x.source(self.sema.db)?; + let hir_file_id = src.syntax().file_id; + // note: HirFileId could be a macro. error out + // since supporting macros is hard. + // FIXME: support macros + let file_id = hir_file_id.file_id()?; + let name = src.value.name()?; + let def_node = name.syntax().ancestors().find_map(ast::$ast_type::cast)?; + let def_node = AnyGenericParamsOwnerEdit::cast(def_node.syntax().clone())?; + Some((file_id, def_node)) + })() + .ok_or_else(|| Error::Str(format!("unable to find parent definition node")))?; + (file_id, x) + }}; + } + let def = as_definition_with_generic_params(definition); + let Some(def) = def else { + return Err(Error::Str(format!( + "was not a defintition that has generic params: {def:?}" + ))); + }; + let (file_id, x) = match def { + DefinitionWithGenericParams::Struct(x) => on_def_variant!(Struct, x), + DefinitionWithGenericParams::Union(x) => on_def_variant!(Union, x), + DefinitionWithGenericParams::Enum(x) => on_def_variant!(Enum, x), + DefinitionWithGenericParams::Trait(x) => on_def_variant!(Trait, x), + DefinitionWithGenericParams::TraitAlias(x) => on_def_variant!(TraitAlias, x), + DefinitionWithGenericParams::TypeAlias(x) => on_def_variant!(TypeAlias, x), + DefinitionWithGenericParams::Function(x) => on_def_variant!(Fn, x), + }; + let acc = acc.files.entry(file_id).or_default(); + self.run_generic_params_owner_edit_without_segment(acc, x)?; + Ok(()) + })()?; + + let usages = definition.usages(self.sema).with_override_name(name_str); + // usages will include name, so ignore name + let _ = name; + usages.search(&mut |file_id, file_ref| { + let r = self.on_usage(acc, processing_queue, file_id, file_ref.name, definition); + match r { + Ok(()) => false, + Err(_x) => { + // FIXME: on a non-continuable error, return true (signaling to stop) + // also, propagate the Err through `FindUsages::search` somehow. std::ops::ControlFlow would fit perfectly. + false + } + } + }); + Ok(()) + } + fn on_usage( + &mut self, + acc: &mut ChangeAccumulator, + processing_queue: &mut ProcessingQueue, + file_id: FileId, + name: ast::NameLike, + definition: Definition, + ) -> R { + { + // double visit detection: this seems to happen for type aliases in traits + let q = &name; + let was_already_handled = self.aux_state.usages_already_handled.contains(q); + if was_already_handled { + return Ok(()); + } + self.aux_state.usages_already_handled.insert(name.clone()); + } + + let r = match name.clone() { + // This is the usual case, when the symbol is used + // e.g. "Vec" in "Vec" is a NameRef + ast::NameLike::NameRef(name_node) => { + self.on_usage_name_ref(acc, processing_queue, name_node, file_id, definition) + } + // This is a somewhat unusual case, I have seen this when the same symbol is defined elsewhere. + // e.g. trait HasFood{ type Food; } + // ^~~~ this is a usage (ast::Name) + // impl HasFood for Cat{ type Food = Fish; } + // ^~~~ when this is a definition + ast::NameLike::Name(name) => { + let syn = name.syntax(); + let Some(x) = syn.ancestors().find_map(AnyGenericParamsOwnerEdit::cast) else { + return Ok(()); + }; + + self.run_generic_params_owner_edit_without_segment( + acc.files.entry(file_id).or_default(), + x, + )?; + let x = match find_definitions( + self.sema, + syn, + FilePosition { file_id, offset: syn.text_range().start() }, + ) { + Ok(x) => x, + Err(_) => return Ok(()), + }; + x.into_iter().try_for_each(|(name, definition)| { + self.on_definition(acc, processing_queue, name, definition) + })?; + Ok(()) + } + // FIXME: make this work for lifetimes + ast::NameLike::Lifetime(_lifetime) => Ok(()), + }; + r + } + + fn on_usage_name_ref( + &mut self, + acc: &mut ChangeAccumulator, + processing_queue: &mut ProcessingQueue, + name_node: ast::NameRef, + file_id: FileId, + definition: Definition, + ) -> R { + let name_node_syntax = name_node.syntax(); + + let mut ancestors = name_node_syntax.ancestors(); + let x = ancestors.next(); + // self == ancestors[0] + stdx::always!(x.is_some_and(|x| x == *name_node_syntax)); + let x = ancestors.next(); + // usages always are a path segment, usually in a single segment path + // e.g. "Vec" in "Vec", multi segment: "std::vec::Vec" + let Some(parent) = x else { + return Err(Error::Str(format!("node did not have a parent, node = {name_node:?}"))); + }; + + let Some(segment) = ast::PathSegment::cast(parent.clone()) else { + return Err(Error::Str(format!( + "node's parent was not a path segment, node = {name_node:?}, parent = {parent:?}" + ))); + }; + + #[derive(Debug, Clone)] + enum Either4 { + A(A), + B(B), + C(C), + D(D), + } + let x = ancestors.find_map(|x| { + // find an ancestor that is: + let x = match_ast! { + match x { + // e.g. impl items, fn (params, return, bounds), struct members, enum variant members + // impl Stuff{ fn do_stuff(x: Vec); } + // struct Stuff{ x: Vec } + AnyGenericParamsOwnerEdit(x) => Either4::A(x), + // usually a path expression, usually inferred, e.g. "Vec" in "Vec::new()" + // also patterns and struct literals + // Here I just assume it is always inferred, so I ignore this path segment. + ast::Expr(x) => Either4::B(x), + ast::Pat(x) => Either4::D(x), + // use Vec as MyVec; + ast::UseTree(x) =>Either4::C(x), + _ => return None, + } + }; + Some(x) + }); + let Some(meaningful_parent) = x else { + // broken ast, or possible top level ast that I haven't accounted for + return Ok(()); + }; + let ancestor_with_generic_params = match meaningful_parent { + Either4::B(x) => { + let _ = x; + let _ = segment; + return Ok(()); + } + Either4::D(x) => { + let _ = x; + let _ = segment; + return Ok(()); + } + Either4::C(x) => { + // a use alias needs to add generics to its usages + let Some(name) = (|| -> Option<_> { + let x = x.rename()?; + let x = x.name()?; + Some(x) + })() else { + // no right ident + // either means broken ast or it's an _ + // we could try to see which case it is, but it doesn't matter + return Ok(()); + }; + let defs = find_definitions( + self.sema, + name.syntax(), + FilePosition { file_id, offset: name.syntax().text_range().start() }, + ); + + let defs = defs + .map_err(|_| Error::Str(format!("no definitions found for generic thing")))?; + + let r = defs.into_iter().try_for_each(|(name, definition)| -> R { + self.on_definition(acc, processing_queue, name, definition) + }); + return r; + } + Either4::A(x) => x, + }; + let outer_acc = acc; + self.add_segment(outer_acc.files.entry(file_id).or_default(), segment, definition)?; + /* + We want to use an outer parameter if there is one already. + However, we cannot access generic parameters across certain boundaries. + The rules that I have figured out are probably incomplete. + I took some information from rustc_resolve, specifically RibKind, which specifies how upvars + can be accessed. + upvar just means e.g. + impl X{ type A = T; } + ^upvar, has to access outer("up") impl's scope + I also just tried various combinations to see which failed to compile. + Notably: + - anything can access upvar of impl, function signature included + - as soon as you see an expr or non-alias HasGenericParams ancestor, stop + */ + + let acc = outer_acc.files.entry(file_id).or_default(); + + let ancestor_with_generic_params = 'thing: { + let mut current_generic_params_node = ancestor_with_generic_params.clone(); + let mut candidate_generic_params_node = current_generic_params_node; + if ast::Impl::can_cast(candidate_generic_params_node.syntax().kind()) { + break 'thing candidate_generic_params_node; + } + let x = ancestor_with_generic_params.syntax(); + for x in x.ancestors() { + // this takes care of both function bodies and const generics + if let Some(_) = ast::Expr::cast(x.clone()) { + break; + } + let Some(x) = AnyGenericParamsOwnerEdit::cast(x.clone()) else { + continue; + }; + current_generic_params_node = x.clone(); + + // impl's are not allowed to use an outer generic param. + // This doesn't seem like it's necessary for rust to restrict like this + // but that's how it is right now. + let Some(x) = ast::Impl::cast(x.syntax().clone()) else { + // all other (non-Impl) HasGenericParams can freely define generic params. + candidate_generic_params_node = current_generic_params_node.clone(); + continue; + }; + // Only use this impl's generic param if it already would get one + // This is because doing the following is not allowed: + + #[rustfmt::skip] #[cfg(FALSE)] const _: () = { + struct A; + impl A { fn make_vec() -> Vec { vec![] } } + // ^ error: unconstrained type parameter + }; + // note: We only know if it would already get one in some cases, if the impl + // has already been visited. + // We can't easily control visit order, so we have to wait until that impl has been processed. + + if acc.defs_with_generic_params.contains(¤t_generic_params_node) { + // we worked on this impl already, so we're done + return Ok(()); + } + // haven't seen this impl yet, so wait to see if it needs generics, so we can just use those + // instead of making our own + let queue = processing_queue.files.entry(file_id).or_default(); + let work = queue + .inner_usage_waiting_to_see_if_a_surrounding_impl_would_have_generics + .entry(x) + .or_default(); + work.push(UsageWaitingOnSurroudingImpl { + non_impl_generic_params_node: candidate_generic_params_node, + }); + return Ok(()); + } + candidate_generic_params_node + }; + if let Some(x) = ast::Impl::cast(ancestor_with_generic_params.syntax().clone()) { + let queue = processing_queue.files.entry(file_id).or_default(); + // This impl does have surrounding generics, so we can tell all things waiting + // that they don't need to add any generics at their inner scope. + let _ = queue + .inner_usage_waiting_to_see_if_a_surrounding_impl_would_have_generics + .remove(&x); + } + + self.on_non_impl_generic_params_ancestor( + outer_acc, + processing_queue, + file_id, + ancestor_with_generic_params, + )?; + Ok(()) + } + + fn on_non_impl_generic_params_ancestor( + &mut self, + acc: &mut ChangeAccumulator, + processing_queue: &mut ProcessingQueue, + file_id: FileId, + ancestor_with_generic_params: AnyGenericParamsOwnerEdit, + ) -> R { + /* + this 'block thing functions as + if let ... + && let ... + since `&& let` (let_chains) isn't on stable. + */ + 'block: { + let Some(ancestor_with_generic_params) = + ast::AnyHasName::cast(ancestor_with_generic_params.syntax().clone()) + else { + // no name, should only be taken for "impl ..." + // so this should probably never be reached. + break 'block self.run_generic_params_owner_edit_without_segment( + acc.files.entry(file_id).or_default(), + ancestor_with_generic_params, + ); + }; + let Some(name_of_ancestor_with_generic_params) = ancestor_with_generic_params.name() + else { + break 'block Ok(()); + }; + let defs = find_definitions( + self.sema, + ancestor_with_generic_params.syntax(), + FilePosition { + file_id, + offset: name_of_ancestor_with_generic_params.syntax().text_range().start(), + }, + ); + + let defs = + defs.map_err(|_| Error::Str(format!("no definitions found for generic thing")))?; + + defs.into_iter().try_for_each(|(name, definition)| -> R { + self.on_definition(acc, processing_queue, name, definition) + }) + } + } +} + +#[derive(Debug, Clone)] +enum Error { + Str(String), +} +type MyResult = Result; +type R = MyResult<()>; + +impl Configuration { + fn add_generics_to_path_segment(&mut self, parent: MutableAst) -> R { + let generic_arg_list = parent.map(|x| x.get_or_create_generic_arg_list()); + add_generic_arg(generic_arg_list.0.clone(), self.arg.clone_subtree().clone_for_update()); + Ok(()) + } + + fn run_generic_params_owner_edit_without_segment_impl( + &mut self, + parent: MutableAst, + ) -> R { + let gpl = parent.clone().map(|x| x.get_or_create_generic_param_list()); + // FIXME: this seems inefficient having to clone twice + gpl.0.add_generic_param(self.param.clone_subtree().clone_for_update()); + Ok(()) + } +} + +impl<'b, 'sema_a, 'sema> AssistState<'b, 'sema_a, 'sema> { + fn add_segment( + &mut self, + acc: &mut FileChangeAccumulator, + segment: ast::PathSegment, + definition: Definition, + ) -> R { + let x = (segment, definition); + acc.path_segments.insert(x); + Ok(()) + } + + fn run_generic_params_owner_edit_without_segment( + &mut self, + acc: &mut FileChangeAccumulator, + parent: T, + ) -> R + where + T: GenericParamsOwnerEdit, + AnyGenericParamsOwnerEdit: From, + { + let x = parent.into(); + acc.defs_with_generic_params.insert(x); + Ok(()) + } + + fn finish(self, acc: ChangeAccumulator) -> R { + let Self { mut builder, snippet_cap: _, config: mut thingy, sema: _, aux_state: _ } = self; + + let ChangeAccumulator { files } = acc; + + /* + FIXME: add a way to check for duplicate generic parameter names + e.g. for: + struct Thing$0(T1); + current implementation: + -> struct Thing(T1); + ^^ + desired implementation: + -> struct Thing(T1); + ^^ + It should also not duplicate names of upvars. + e.g. + trait DoStuff{ fn stuff(hello: Thing$0, other: T1); } + trait DoStuff{ fn stuff(hello: Thing, other: T1); } + added: ^^ ^^ + accidentally refered to ^^ when looking at^ + desired: + trait DoStuff{ fn stuff(hello: Thing, other: T1); } + */ + for (file_id, acc) in files { + let FileChangeAccumulator { path_segments, defs_with_generic_params } = acc; + builder.inner.edit_file(file_id); + /* + note: make sure to make_mut for both of these before modifying any of them + */ + let path_segments = path_segments + .into_iter() + .map(|(x, def)| (builder.make_mut(x), def)) + .collect::>(); + + let defs_with_generic_params = defs_with_generic_params + .into_iter() + .map(|x| builder.make_mut(x)) + .collect::>(); + path_segments.into_iter().try_for_each(|(x, _def)| -> R { + thingy.add_generics_to_path_segment(x)?; + Ok(()) + })?; + defs_with_generic_params.into_iter().try_for_each(|x| -> R { + thingy.run_generic_params_owner_edit_without_segment_impl(x)?; + Ok(()) + })?; + } + Ok(()) + } +} + +// copied (and modified) from ide_db::rename (it was a private function) +// FIXME: share code with the previously mentioned module +// note: in practice, the return value seems to always be length 1 +// [(node_of_name_reference, e.g. NAME_REF @ 7..10, hir of definition of that name, e.g. StructId(0))] +fn find_definitions( + sema: &hir::Semantics<'_, RootDatabase>, + syntax: &SyntaxNode, + position: FilePosition, +) -> rename::Result> { + use ide_db::rename::{bail, format_err, RenameError}; + let symbols = sema.find_nodes_at_offset_with_descend::(syntax, position.offset); + let symbols = symbols.map(|name_like| { + let res = match &name_like { + ast::NameLike::Name(name) + if name.syntax().parent().map_or(false, |it| ast::Rename::can_cast(it.kind())) + // FIXME: uncomment this once we resolve to usages to extern crate declarations + // && name + // .syntax() + // .ancestors() + // .nth(2) + // .map_or(true, |it| !ast::ExternCrate::can_cast(it.kind())) + => + { + let x = NameClass::classify(sema, name); + let name_class = x.ok_or_else(|| format_err!("No references found at position"))?; + let def = match name_class { + NameClass::Definition(it) => it, + NameClass::ConstReference(it) => it, + NameClass::PatFieldShorthand { local_def, field_ref: _ } => { + Definition::Local(local_def) + } + }; + Ok((name_like, def)) + } + ast::NameLike::Name(name) => NameClass::classify(sema, name) + .map(|class| match class { + NameClass::Definition(it) | NameClass::ConstReference(it) => it, + NameClass::PatFieldShorthand { local_def, field_ref: _ } => { + Definition::Local(local_def) + } + }) + .map(|def| (name_like.clone(), def)) + .ok_or_else(|| format_err!("No references found at position")), + ast::NameLike::NameRef(name_ref) => { + NameRefClass::classify(sema, name_ref) + .map(|class| match class { + NameRefClass::Definition(def) => def, + NameRefClass::FieldShorthand { local_ref, field_ref: _ } => { + Definition::Local(local_ref) + } + NameRefClass::ExternCrateShorthand { decl, .. } => { + Definition::ExternCrateDecl(decl) + } + }) + // FIXME: uncomment this once we resolve to usages to extern crate declarations + .filter(|def| !matches!(def, Definition::ExternCrateDecl(..))) + .ok_or_else(|| format_err!("No references found at position")) + .and_then(|def| { + // if the name differs from the definitions name it has to be an alias + // if def + // .name(sema.db) + // .map_or(false, |it| it.to_smol_str() != name_ref.text().as_str()) + // { + // return Err(format_err!("Renaming aliases is currently unsupported")) + // } + + // I do support adding generic parameters to aliases. + Ok((name_like.clone(), def)) + }) + } + ast::NameLike::Lifetime(lifetime) => NameRefClass::classify_lifetime(sema, lifetime) + .and_then(|class| match class { + NameRefClass::Definition(def) => Some(def), + _ => None, + }) + .or_else(|| { + NameClass::classify_lifetime(sema, lifetime).and_then(|it| match it { + NameClass::Definition(it) => Some(it), + _ => None, + }) + }) + .map(|def| (name_like, def)) + .ok_or_else(|| format_err!("No references found at position")), + }; + + res + }); + + let res: rename::Result> = symbols.collect(); + match res { + Ok(v) => { + if v.is_empty() { + // FIXME: some semantic duplication between "empty vec" and "Err()" + bail!("No references found at position") + } else { + // remove duplicates, comparing `Definition`s + use itertools::Itertools; + Ok(v.into_iter().unique_by(|t| t.1)) + } + } + Err(e) => Err(e), + } +} +// FIXME: add this to crates/syntax/src/ast/node_ext.rs +#[derive(Debug, Clone, PartialEq, Eq, Hash)] +enum AnyGenericParamsOwnerEdit { + Fn(ast::Fn), + Impl(ast::Impl), + Trait(ast::Trait), + Struct(ast::Struct), + Enum(ast::Enum), + Union(ast::Union), + TypeAlias(ast::TypeAlias), + TraitAlias(ast::TraitAlias), +} + +const _: () = { + macro_rules! for_each_member { + ($next: ident) => { + $next!( + (FN, Fn), + (IMPL, Impl), + (TRAIT, Trait), + (STRUCT, Struct), + (ENUM, Enum), + (UNION, Union), + (TYPE_ALIAS, TypeAlias), + (TRAIT_ALIAS, TraitAlias), + ) + }; + } + macro_rules! impl_from { + ($(($kind: ident, $variant: ident),)*) => { $( + impl From for AnyGenericParamsOwnerEdit { + fn from(x: ast::$variant) -> Self { + Self::$variant(x) + } + } + )* }; + } + for_each_member!(impl_from); + impl AstNode for AnyGenericParamsOwnerEdit { + fn can_cast(kind: SyntaxKind) -> bool { + macro_rules! next { + ($(($kind: ident, $variant: ident),)*) => { + match kind { + $(SyntaxKind::$kind => true,)* + _ => false, + } + }; + } + for_each_member!(next) + } + + fn cast(syntax: SyntaxNode) -> Option { + macro_rules! next { + ($(($kind: ident, $variant: ident),)*) => { + match syntax.kind() { + $(SyntaxKind::$kind => Some(Self::$variant(ast::$variant::cast(syntax)?)),)* + _ => return None, + } + }; + } + for_each_member!(next) + } + + fn syntax(&self) -> &SyntaxNode { + macro_rules! next { + ($(($kind: ident, $variant: ident),)*) => { + match self { + $(Self::$variant(x) => x.syntax(),)* + } + }; + } + for_each_member!(next) + } + } + impl ast::HasGenericParams for AnyGenericParamsOwnerEdit {} + impl GenericParamsOwnerEdit for AnyGenericParamsOwnerEdit { + fn get_or_create_generic_param_list(&self) -> ast::GenericParamList { + macro_rules! next { + ($(($kind: ident, $variant: ident),)*) => { + match self { + $(Self::$variant(x) => x.get_or_create_generic_param_list(),)* + } + }; + } + for_each_member!(next) + } + fn get_or_create_where_clause(&self) -> ast::WhereClause { + macro_rules! next { + ($(($kind: ident, $variant: ident),)*) => { + match self { + $(Self::$variant(x) => x.get_or_create_where_clause(),)* + } + }; + } + for_each_member!(next) + } + } +}; + +// I copied this from edit_in_place.rs::add_generic_param +// I'm not sure why this didnt exist already +// FIXME: add to crates/syntax/src/ast/edit_in_place.rs +pub fn add_generic_arg(self_: ast::GenericArgList, generic_arg: ast::GenericArg) { + match self_.generic_args().last() { + Some(last) => { + let position = Position::after(last.syntax()); + use ast::make; + let elements = vec![ + make::token(syntax::T![,]).into(), + make::tokens::single_space().into(), + generic_arg.syntax().clone().into(), + ]; + ted::insert_all(position, elements); + } + None => { + let after_l_angle = Position::after(self_.l_angle_token().unwrap()); + ted::insert(after_l_angle, generic_arg.syntax()); + } + } +} + +#[cfg(test)] +mod tests { + use crate::tests::{check_assist, check_assist_not_applicable}; + + use super::*; + // FIXME: add tests to make sure it doesn't panic with strange input + // FIXME: add cov_mark::{hit, check} in some tests + /* + note: Tests should cover all of + AnyGenericParamsOwnerEdit's variants. + */ + + #[test] + fn tuple_struct_basic() { + check_assist( + add_generic_parameter, + r#"struct Color$0(u32);"#, + r#"struct Color(u32);"#, + ); + } + #[test] + fn unit_struct_basic() { + check_assist(add_generic_parameter, r#"struct Color$0;"#, r#"struct Color;"#); + check_assist(add_generic_parameter, r#"struct Color$0;"#, r#"struct Color;"#); + } + #[test] + fn struct_basic() { + check_assist( + add_generic_parameter, + r#"struct Color$0 { x: u32 }"#, + r#"struct Color { x: u32 }"#, + ); + check_assist( + add_generic_parameter, + r#"struct Color$0 { x: u32 }"#, + r#"struct Color { x: u32 }"#, + ); + } + #[test] + fn union_basic() { + check_assist( + add_generic_parameter, + r#"union Color$0 { x: u32 }"#, + r#"union Color { x: u32 }"#, + ); + check_assist( + add_generic_parameter, + r#"union Color$0 { x: u32 }"#, + r#"union Color { x: u32 }"#, + ); + } + + #[test] + fn enum_basic() { + check_assist( + add_generic_parameter, + r#"enum Color$0 { A{x: u32} }"#, + r#"enum Color { A{x: u32} }"#, + ); + check_assist( + add_generic_parameter, + r#"enum Color$0 { A{x: u32} }"#, + r#"enum Color { A{x: u32} }"#, + ); + } + #[test] + fn nest_struct_direct() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0 { x: u8 } +struct Color{ channel: Channel } +"#, + r#" +struct Channel { x: u8 } +struct Color{ channel: Channel } +"#, + ); + } + #[test] + fn nest_tuple_struct_direct() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0 { x: u8 } +struct Color(Channel); +"#, + r#" +struct Channel { x: u8 } +struct Color(Channel); +"#, + ); + } + #[test] + fn nest_tuple_struct_direct_multi() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0 { x: u8 } +struct Color1(Channel); +struct Color2(Channel); +"#, + r#" +struct Channel { x: u8 } +struct Color1(Channel); +struct Color2(Channel); +"#, + ); + } + #[test] + fn nest_enum_tuple_struct_direct() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0 { x: u8 } +enum Color{ A(Channel) } +"#, + r#" +struct Channel { x: u8 } +enum Color{ A(Channel) } +"#, + ); + } + #[test] + fn nest_enum_record_struct_direct() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0 { x: u8 } +enum Color{ A{c: Channel} } +"#, + r#" +struct Channel { x: u8 } +enum Color{ A{c: Channel} } +"#, + ); + } + + #[test] + fn multi_layered_struct() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0(u8); +struct Color([Channel; 3]); +struct Image(Vec); +"#, + r#" +struct Channel(u8); +struct Color([Channel; 3]); +struct Image(Vec>); +"#, + ); + } + #[test] + fn cyclic_structs() { + check_assist( + add_generic_parameter, + r#" +struct Cons$0(i32, List); +struct List(Option>); +"#, + r#" +struct Cons(i32, List); +struct List(Option>>); +"#, + ); + } + #[test] + fn cyclic_structs_impl() { + check_assist( + add_generic_parameter, + r#" +struct Cons$0(i32, List); +impl Cons{ fn len(&self){42} } +struct List(Option>); +impl List{ fn len(&self){42} } +"#, + r#" +struct Cons(i32, List); +impl Cons{ fn len(&self){42} } +struct List(Option>>); +impl List{ fn len(&self){42} } +"#, + ); + } + #[test] + fn type_alias_basic() { + check_assist( + add_generic_parameter, + r#" +type Color$0 = u32; +"#, + r#" +type Color = u32; +"#, + ); + } + #[test] + fn type_alias_into_struct() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0(u8); +type Color = [Channel; 3]; +struct Image(Vec); +"#, + r#" +struct Channel(u8); +type Color = [Channel; 3]; +struct Image(Vec>); +"#, + ); + } + + #[test] + fn trait_basic() { + check_assist( + add_generic_parameter, + r#" +trait HasFood$0 {} +"#, + r#" +trait HasFood {} +"#, + ); + } + #[test] + fn trait_into_trait_bound() { + check_assist( + add_generic_parameter, + r#" +trait HasFood$0 {} +trait HasSupplies: HasFood {} +"#, + r#" +trait HasFood {} +trait HasSupplies: HasFood {} +"#, + ); + } + + #[test] + fn trait_into_struct() { + check_assist( + add_generic_parameter, + r#" +trait HasFood$0 {} +struct Supplies(Box); +"#, + r#" +trait HasFood {} +struct Supplies(Box>); +"#, + ); + } + /* + this is where some ambiguity comes in: + it is unclear where the should go in this case. + it may go after make: make + or after CanMakeFood: CanMakeFood + + one thing reason for doing make is that one can just use this assist + on CanMakeFood to add to it. + another alternative is to temporarily add a bound ":HasFood" + to CanMakeFood so that it adds to CanMakeFood + */ + #[test] + fn trait_into_trait_fn() { + check_assist( + add_generic_parameter, + r#" +trait HasFood$0 {} +trait CanMakeFood{ fn make() -> impl HasFood; } +"#, + r#" +trait HasFood {} +trait CanMakeFood{ fn make() -> impl HasFood; } +"#, + ); + } + + // here, T1 looks like it is being used before it is defined + // but rust allows it, so no need to special case this + #[test] + fn trait_into_fn_bound() { + check_assist( + add_generic_parameter, + r#" +trait HasFood$0 {} +fn make(){} +"#, + r#" +trait HasFood {} +fn make, T1>(){} +"#, + ); + } + #[test] + fn trait_into_trait_fn_bound() { + check_assist( + add_generic_parameter, + r#" +trait HasFood$0 {} +trait CanMakeFood{ fn make(); } +"#, + r#" +trait HasFood {} +trait CanMakeFood{ fn make>(); } +"#, + ); + } + + #[test] + fn struct_into_trait_fn_into_impl() { + check_assist( + add_generic_parameter, + r#" +struct Food$0; +trait MakeFood{ fn make() -> Food; } +impl MakeFood for Farm{ fn make() -> Food { Food }} +"#, + r#" +struct Food; +trait MakeFood{ fn make() -> Food; } +impl MakeFood for Farm{ fn make() -> Food { Food }} +"#, + ); + } + + #[test] + fn trait_alias_basic() { + check_assist( + add_generic_parameter, + r#" +trait Debug2$0 = Debug; +"#, + r#" +trait Debug2 = Debug; +"#, + ); + } + #[test] + fn trait_alias_into_trait_bound() { + check_assist( + add_generic_parameter, + r#" +trait Debug2$0 = Debug; +trait SuperDebug: Debug2 {} +"#, + r#" +trait Debug2 = Debug; +trait SuperDebug: Debug2 {} +"#, + ); + } + + // not sure if dyn TraitAlias is allowed, but it works here + #[test] + fn trait_alias_into_struct() { + check_assist( + add_generic_parameter, + r#" +trait Debug2$0 = Debug; +struct Debug2Thing(Box); +"#, + r#" +trait Debug2 = Debug; +struct Debug2Thing(Box>); +"#, + ); + } + + #[test] + fn struct_into_use_into_struct() { + check_assist( + add_generic_parameter, + r#" +struct ColorImpl$0(u8); +use ColorImpl as Color; +struct Image(Vec); +"#, + r#" +struct ColorImpl(u8); +use ColorImpl as Color; +struct Image(Vec>); +"#, + ); + } + + #[test] + fn struct_into_impl_with_self_type() { + check_assist( + add_generic_parameter, + r#" +struct Color$0(u8); +impl Color { fn new() -> Self { Color(0) }} +impl Color { fn replace1(self: &Self) -> Self { Color(1) }} +"#, + r#" +struct Color(u8); +impl Color { fn new() -> Self { Color(0) }} +impl Color { fn replace1(self: &Self) -> Self { Color(1) }} +"#, + ); + } + + /* + Traits don't play super nicely + There are a few options here: + 1. used only in Color's impl + trait HasChannel{ type ChannelTy; fn channel(&self) -> &Self::ChannelTy; } + 2. used in both fn, type alias + trait HasChannel{ type ChannelTy; fn channel(&self) -> &Self::ChannelTy; } + 3. used in both trait, type alias + trait HasChannel{ type ChannelTy; fn channel(&self) -> &Self::ChannelTy; } + + For the non-trait analog, we prefer adding to the inner HasGenericParams. + In this case, this analogs as preferring 2. over 3. + + 1. seems to be the most likely case because adt's use their parts inside + of impls and trait impls, and if their part adds a generic, + then that generic gets added inside of the trait body, which means + that it can access the outer trait generic param. + + Maybe when usages are detected, it should try to see if there is a surrounding + definition to take the generic params from. + */ + #[allow(unused)] + fn struct_into_trait_impl_with_type_alias_option_1() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0(u8); +struct Color(Channel); +trait HasChannel{ type ChannelTy; fn channel(&self) -> &Self::ChannelTy; } +impl HasChannel for Color { type ChannelTy = Channel; + fn channel(&self) -> &Self::ChannelTy { &self.0 } } +"#, + r#" +struct Channel(u8); +struct Color(Channel); +trait HasChannel{ type ChannelTy; fn channel(&self) -> &Self::ChannelTy; } +impl HasChannel for Color { type ChannelTy = Channel; + fn channel(&self) -> &Self::ChannelTy { &self.0 } } +"#, + ); + } + #[test] + fn struct_into_trait_impl_with_type_alias_option_2() { + check_assist( + add_generic_parameter, + r#" +struct Channel$0(u8); +struct Color(Channel); +trait HasChannel{ type ChannelTy; fn channel(&self) -> &Self::ChannelTy; } +impl HasChannel for Color { type ChannelTy = Channel; + fn channel(&self) -> &Self::ChannelTy { &self.0 } } +"#, + r#" +struct Channel(u8); +struct Color(Channel); +trait HasChannel{ type ChannelTy; fn channel(&self) -> &Self::ChannelTy; } +impl HasChannel for Color { type ChannelTy = Channel; + fn channel(&self) -> &Self::ChannelTy { &self.0 } } +"#, + ); + } + + #[test] + fn function_basic() { + check_assist(add_generic_parameter, r#"fn func$0(){}"#, r#"fn func(){}"#); + check_assist(add_generic_parameter, r#"fn func$0(){}"#, r#"fn func(){}"#); + } + + #[test] + fn struct_into_function_return() { + check_assist( + add_generic_parameter, + r#" +struct Color$0 {} +fn func() -> Color {42} +"#, + r#" +struct Color {} +fn func() -> Color {42} +"#, + ); + } + #[test] + fn struct_into_function_param() { + check_assist( + add_generic_parameter, + r#" +struct Color$0 {} +fn func(c: Color) {42} +"#, + r#" +struct Color {} +fn func(c: Color) {42} +"#, + ); + } + + /* + Inside an expression, do not add any generic parameters. + This is because they are usually inferred. + This is helped along if the type is mentioned in the function signature + so that type inference can work from the params/return type. + + Maybe in the future this could be changed. + */ + #[test] + fn struct_in_expr() { + check_assist( + add_generic_parameter, + r#" +struct Color$0 { x: u32 } +fn func() { Color{x: 42} } +fn func() { let Color{x: y} = 42; } +fn func() { Color::red(42) } +fn func() { const_func::<{Color{x: 9}}> } +"#, + r#" +struct Color { x: u32 } +fn func() { Color{x: 42} } +fn func() { let Color{x: y} = 42; } +fn func() { Color::red(42) } +fn func() { const_func::<{Color{x: 9}}> } +"#, + ); + } + + #[test] + fn struct_in_const_type() { + check_assist( + add_generic_parameter, + r#" +struct Color$0 { x: u32 } +fn func() {} +"#, + r#" +struct Color { x: u32 } +fn func, T1>() {} +"#, + ); + } + + // FIXME: make these type ascription cases work, possibly adding <_> as a generic param + #[test] + fn type_ascription_in_fn() { + check_assist( + add_generic_parameter, + r#" +struct Color$0 { x: u32 } +fn func() { let c: Color; } +"#, + r#" +struct Color { x: u32 } +fn func() { let c: Color; } +"#, + ); + } + #[test] + fn type_ascription_in_fn_in_trait() { + check_assist( + add_generic_parameter, + r#" +struct Color$0 { x: u32 } +trait Thing{ fn func() { let c: Color; } } +"#, + r#" +struct Color { x: u32 } +trait Thing{ fn func() { let c: Color; } } +"#, + ); + } + #[test] + fn duplicate_parameter_name() { + // FIXME: Make this use a name other than T1 when there exists a parameter with the same name already. + check_assist( + add_generic_parameter, + r#" +struct Color$0 {} +"#, + r#" +struct Color {} +"#, + ); + } + + #[test] + fn duplicate_parameter_name_upvar() { + // FIXME: Make this use a name other than T1 when there exists a (possibly upvar) parameter with the same name already. + check_assist( + add_generic_parameter, + r#" +struct Thing; +trait DoStuff{ fn stuff(hello: Thing$0); } +"#, + r#" +struct Thing; +trait DoStuff{ fn stuff(hello: Thing); } +"#, + ); + } + #[test] + fn use_alias_in_different_modules() { + check_assist( + add_generic_parameter, + r#" +struct ColorImpl$0(u8); +use ColorImpl as Color; +struct Image(Vec); +mod tmp{ use super::ColorImpl as Color2; struct Image2(Vec); } +"#, + r#" +struct ColorImpl(u8); +use ColorImpl as Color; +struct Image(Vec>); +mod tmp{ use super::ColorImpl as Color2; struct Image2(Vec>); } +"#, + ); + } + #[test] + fn non_generic_impl_has_fn_returns_generic() { + check_assist( + add_generic_parameter, + r#" +struct Color$0(u8); +struct MakeColor; +impl MakeColor { fn make() -> Color { Color(0) }} +"#, + r#" +struct Color(u8); +struct MakeColor; +impl MakeColor { fn make() -> Color { Color(0) }} +"#, + ); + } + #[test] + fn generic_impl_has_fn_returns_generic() { + check_assist( + add_generic_parameter, + r#" +struct Color$0(u8); +struct MakeColor2(Color); +impl MakeColor2 { fn make() -> Color { Color(0) }} +"#, + r#" +struct Color(u8); +struct MakeColor2(Color); +impl MakeColor2 { fn make() -> Color { Color(0) }} +"#, + ); + } + mod realer_cases { + use super::*; + #[test] + + fn impl_from() { + check_assist( + add_generic_parameter, + r#" +struct Color { channels: [Channel; 3] } +struct Channel$0(u8); +impl From for Color { + fn from(x: Channel){ Color{ channels: [x.clone(); 3] } } } +"#, + r#" +struct Color { channels: [Channel; 3] } +struct Channel(u8); +impl From> for Color { + fn from(x: Channel){ Color{ channels: [x.clone(); 3] } } } +"#, + ); + } + #[test] + fn impl_deref() { + // example from crates/hir/src/semantics.rs + check_assist( + add_generic_parameter, + r#" +struct TraitId$0; +pub struct VisibleTraits(pub FxHashSet); +impl ops::Deref for VisibleTraits { + type Target = FxHashSet; + fn deref(&self) -> &Self::Target { &self.0 } } +"#, + r#" +struct TraitId; +pub struct VisibleTraits(pub FxHashSet>); +impl ops::Deref for VisibleTraits { + type Target = FxHashSet>; + fn deref(&self) -> &Self::Target { &self.0 } } +"#, + ); + } + /* + This tests specifically against an implementation that orders adding the generics + by the depth of the ast. This is one implementation that I tried. + Box is nested deep enough that it would get visited after + fn new(x: Expr) + + which means that new(x: Expr) does not see the surrounding impl with generics + so it adds generics to the `new` function. + + A correct implementation would order checks by waiting for the impl to possibly be analyzed. + */ + #[test] + fn differently_nested_things() { + check_assist( + add_generic_parameter, + r#" +struct Expr$0; +impl ExprParen { fn new(x: Expr){} } +struct ExprParen { x: Box } +"#, + r#" +struct Expr; +impl ExprParen { fn new(x: Expr){} } +struct ExprParen { x: Box> } +"#, + ); + } + /* + with the above in mind, is it possible to create a cyclic dependency? + I wasn't able to, but these should hopefully be weird cases + */ + + #[test] + fn strange_impl_dependencies() { + check_assist( + add_generic_parameter, + r#" +struct Thing$0; +impl HasFood for Wrap { type Food = Thing; } +impl Lunch { fn new(_: Thing){} } +struct Lunch(::Food); +struct Wrap(Thing); +"#, + r#" +struct Thing; +impl HasFood for Wrap { type Food = Thing; } +impl Lunch { fn new(_: Thing){} } +struct Lunch( as HasFood>::Food); +struct Wrap(Thing); +"#, + ); + } + #[test] + fn strange_impl_dependencies2() { + check_assist( + add_generic_parameter, + r#" +struct Thing$0; +impl HasFood for Wrap { type Food = Thing; } +impl Lunch { fn new(_: Thing){} } +struct Lunch; +struct Wrap(Thing); +"#, + r#" +struct Thing; +impl HasFood for Wrap { type Food = Thing; } +impl Lunch { fn new(_: Thing){} } +struct Lunch; +struct Wrap(Thing); +"#, + ); + } + + #[test] + fn strange_impl_dependencies3() { + check_assist( + add_generic_parameter, + r#" +struct Blue$0; +trait HasColor{ type Color; } +struct Sky; impl HasColor for Sky{ + type Color = Blue; } +struct Ocean; impl HasColor for Ocean { + type Color = ::Color; } +"#, + r#" +struct Blue; +trait HasColor{ type Color; } +struct Sky; impl HasColor for Sky{ + type Color = Blue; } +struct Ocean; impl HasColor for Ocean { + type Color = ::Color; } +"#, + ); + } + } + // FIXME: add more not_applicable cases + #[test] + fn na_fn_part_0() { + check_assist_not_applicable(add_generic_parameter, r#"fn f(){$0}"#); + } + #[test] + fn na_fn_part_1() { + check_assist_not_applicable(add_generic_parameter, r#"fn f()$0{}"#); + } + #[test] + fn na_fn_part_2() { + check_assist_not_applicable(add_generic_parameter, r#"fn f($0){}"#); + } + #[test] + fn na_fn_part_3() { + check_assist_not_applicable(add_generic_parameter, r#"fn$0 f(){}"#); + } + #[test] + fn na_fn_part_4() { + check_assist_not_applicable(add_generic_parameter, r#"f$0n f(){}"#); + } + #[test] + fn na_fn_part_5() { + check_assist_not_applicable(add_generic_parameter, r#"$0fn f(){}"#); + } +} diff --git a/crates/ide-assists/src/lib.rs b/crates/ide-assists/src/lib.rs index 1e4d1c94f5be..637c2a75f66d 100644 --- a/crates/ide-assists/src/lib.rs +++ b/crates/ide-assists/src/lib.rs @@ -112,6 +112,7 @@ mod handlers { mod add_lifetime_to_type; mod add_missing_impl_members; mod add_turbo_fish; + mod add_generic_parameter; mod apply_demorgan; mod auto_import; mod bind_unused_param; @@ -228,6 +229,7 @@ mod handlers { add_lifetime_to_type::add_lifetime_to_type, add_return_type::add_return_type, add_turbo_fish::add_turbo_fish, + add_generic_parameter::add_generic_parameter, apply_demorgan::apply_demorgan, apply_demorgan::apply_demorgan_iterator, auto_import::auto_import, diff --git a/crates/ide-assists/src/tests/generated.rs b/crates/ide-assists/src/tests/generated.rs index da5822bba9c8..e790aa33de01 100644 --- a/crates/ide-assists/src/tests/generated.rs +++ b/crates/ide-assists/src/tests/generated.rs @@ -44,6 +44,23 @@ fn main() { ) } +#[test] +fn doctest_add_generic_parameter() { + check_doc_test( + "add_generic_parameter", + r#####" +struct Channel$0(u8); +struct Color([Channel; 3]); +struct Image(Vec); +"#####, + r#####" +struct Channel(u8); +struct Color([Channel; 3]); +struct Image(Vec>); +"#####, + ) +} + #[test] fn doctest_add_hash() { check_doc_test( diff --git a/crates/ide-db/src/search.rs b/crates/ide-db/src/search.rs index dbef36026822..e7da8a138fdf 100644 --- a/crates/ide-db/src/search.rs +++ b/crates/ide-db/src/search.rs @@ -376,6 +376,7 @@ impl Definition { def: self, assoc_item_container: self.as_assoc_item(sema.db).map(|a| a.container(sema.db)), sema, + override_name: None, scope: None, include_self_kw_refs: None, search_self_mod: false, @@ -394,9 +395,16 @@ pub struct FindUsages<'a> { include_self_kw_refs: Option, /// whether to search for the `self` module search_self_mod: bool, + /// whether to use an override name, e.g. find usages through a `use` + override_name: Option<&'a str>, } impl<'a> FindUsages<'a> { + /// whether to use an override name, e.g. find usages through a `use` + pub fn with_override_name(mut self, override_name: &'a str) -> Self { + self.override_name = Some(override_name); + self + } /// Enable searching for `Self` when the definition is a type or `self` for modules. pub fn include_self_refs(mut self) -> Self { self.include_self_kw_refs = def_to_ty(self.sema, &self.def); @@ -437,7 +445,6 @@ impl<'a> FindUsages<'a> { pub fn search(&self, sink: &mut dyn FnMut(FileId, FileReference) -> bool) { let _p = profile::span("FindUsages:search"); let sema = self.sema; - let search_scope = { // FIXME: Is the trait scope needed for trait impl assoc items? let base = @@ -447,36 +454,41 @@ impl<'a> FindUsages<'a> { Some(scope) => base.intersection(scope), } }; - - let name = match self.def { - // special case crate modules as these do not have a proper name - Definition::Module(module) if module.is_crate_root() => { - // FIXME: This assumes the crate name is always equal to its display name when it - // really isn't - // we should instead look at the dependency edge name and recursively search our way - // up the ancestors - module - .krate() - .display_name(self.sema.db) - .map(|crate_name| crate_name.crate_name().as_smol_str().clone()) - } - _ => { - let self_kw_refs = || { - self.include_self_kw_refs.as_ref().and_then(|ty| { - ty.as_adt() - .map(|adt| adt.name(self.sema.db)) - .or_else(|| ty.as_builtin().map(|builtin| builtin.name())) - }) - }; - // We need to unescape the name in case it is written without "r#" in earlier - // editions of Rust where it isn't a keyword. - self.def.name(sema.db).or_else(self_kw_refs).map(|it| it.unescaped().to_smol_str()) + let name = self.override_name.map(either::Left).or_else(|| { + match self.def { + // special case crate modules as these do not have a proper name + Definition::Module(module) if module.is_crate_root() => { + // FIXME: This assumes the crate name is always equal to its display name when it really isn't + let name = module + .krate() + .display_name(self.sema.db) + .map(|crate_name| crate_name.crate_name().as_smol_str().clone())?; + Some(either::Right(name)) + } + _ => { + let self_kw_refs = || { + self.include_self_kw_refs.as_ref().and_then(|ty| { + ty.as_adt() + .map(|adt| adt.name(self.sema.db)) + .or_else(|| ty.as_builtin().map(|builtin| builtin.name())) + }) + }; + // We need to unescape the name in case it is written without "r#" in earlier + // editions of Rust where it isn't a keyword. + + let name = self + .def + .name(sema.db) + .or_else(self_kw_refs) + .map(|it| it.unescaped().to_smol_str())?; + Some(either::Right(name)) + } } + }); + let Some(name) = name else { + return; }; - let name = match &name { - Some(s) => s.as_str(), - None => return, - }; + let name = either::for_both!(&name, x => x.as_ref()); let finder = &Finder::new(name); let include_self_kw_refs = self.include_self_kw_refs.as_ref().map(|ty| (ty, Finder::new("Self"))); From 0bf1cd54a99f8413bd7372651b7f1c31db4120ff Mon Sep 17 00:00:00 2001 From: anarachnid Date: Wed, 30 Aug 2023 14:53:17 -0700 Subject: [PATCH 4/6] fix unreachable-pub warning --- crates/ide-assists/src/handlers/add_generic_parameter.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/ide-assists/src/handlers/add_generic_parameter.rs b/crates/ide-assists/src/handlers/add_generic_parameter.rs index 904345057f58..fb16feeca054 100644 --- a/crates/ide-assists/src/handlers/add_generic_parameter.rs +++ b/crates/ide-assists/src/handlers/add_generic_parameter.rs @@ -959,7 +959,7 @@ const _: () = { // I copied this from edit_in_place.rs::add_generic_param // I'm not sure why this didnt exist already // FIXME: add to crates/syntax/src/ast/edit_in_place.rs -pub fn add_generic_arg(self_: ast::GenericArgList, generic_arg: ast::GenericArg) { +fn add_generic_arg(self_: ast::GenericArgList, generic_arg: ast::GenericArg) { match self_.generic_args().last() { Some(last) => { let position = Position::after(last.syntax()); From 10e308af8a1c29ed272056966362d185ea79dc69 Mon Sep 17 00:00:00 2001 From: anarachnid Date: Tue, 12 Dec 2023 10:01:58 -0800 Subject: [PATCH 5/6] update an api usage --- crates/ide-assists/src/handlers/add_generic_parameter.rs | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/crates/ide-assists/src/handlers/add_generic_parameter.rs b/crates/ide-assists/src/handlers/add_generic_parameter.rs index fb16feeca054..58a23cc33453 100644 --- a/crates/ide-assists/src/handlers/add_generic_parameter.rs +++ b/crates/ide-assists/src/handlers/add_generic_parameter.rs @@ -110,8 +110,8 @@ fn assist_impl( }; let mut processing_queue = ProcessingQueue::default(); let mut acc = ChangeAccumulator::default(); - // FIXME: possibly recompute the definitions? - // to make sure they haven't changed since the user last touched anything + // FIXME: Possibly recompute the definitions? + // (to make sure they haven't changed since the user last touched anything) let r = (|| -> R { let acc = &mut acc; let processing_queue = &mut processing_queue; @@ -345,7 +345,10 @@ impl<'b, 'sema_a, 'sema> AssistState<'b, 'sema_a, 'sema> { // usages will include name, so ignore name let _ = name; usages.search(&mut |file_id, file_ref| { - let r = self.on_usage(acc, processing_queue, file_id, file_ref.name, definition); + let Some(name) = file_ref.name.into_name_like() else { + return false; + }; + let r = self.on_usage(acc, processing_queue, file_id, name, definition); match r { Ok(()) => false, Err(_x) => { From e4c4f9cd70833cc0d0fea9a5069cdac6f9bc91f2 Mon Sep 17 00:00:00 2001 From: anarachnid Date: Tue, 12 Dec 2023 11:04:57 -0800 Subject: [PATCH 6/6] add multi file test --- .../src/handlers/add_generic_parameter.rs | 27 +++++++++++++++++++ 1 file changed, 27 insertions(+) diff --git a/crates/ide-assists/src/handlers/add_generic_parameter.rs b/crates/ide-assists/src/handlers/add_generic_parameter.rs index 58a23cc33453..15389e325f50 100644 --- a/crates/ide-assists/src/handlers/add_generic_parameter.rs +++ b/crates/ide-assists/src/handlers/add_generic_parameter.rs @@ -1605,6 +1605,33 @@ mod tmp{ use super::ColorImpl as Color2; struct Image2(Vec>); } "#, ); } + #[test] + fn use_alias_in_different_modules_multi_file() { + check_assist( + add_generic_parameter, + r#" +//- /main.rs +pub mod foo; +use foo::ColorImpl; +use ColorImpl as Color; +struct Image(Vec); +mod tmp{ use super::ColorImpl as Color2; struct Image2(Vec); } +//- /foo.rs +pub(crate) struct ColorImpl$0(u8); +"#, + r#" +//- /main.rs +pub mod foo; +use foo::ColorImpl; +use ColorImpl as Color; +struct Image(Vec>); +mod tmp{ use super::ColorImpl as Color2; struct Image2(Vec>); } +//- /foo.rs +pub(crate) struct ColorImpl(u8); +"#, + ); + } + #[test] fn non_generic_impl_has_fn_returns_generic() { check_assist(