Auto merge of #12539 - soruh:instanciate_empty_structs, r=Veykril

Automatically instaciate trivially instaciable structs in "Generate new" and "Fill struct fields"

As proposed in #12535 this PR changes the "Generate new" and "Fill struct fields" assist/diagnostic to instanciate structs with no fields and enums with a single empty variant.

For example:
```rust
pub enum Bar {
    Bar {},
}
struct Foo<T> {
    a: usize,
    bar: Bar,
    _phantom: std::marker::PhantomData<T>,
}
impl<T> Foo<T> {
    /* generate new */

    fn random() -> Self {
        Self { /* Fill struct fields */ }
    }
}
```

was previously:
```rust
impl<T> Foo<T> {
    fn new(a: usize, bar: Bar, _phantom: std::marker::PhantomData<T>) -> Self {
        Self { a, bar, _phantom }
    }

    fn random() -> Self {
        Self {
            a: todo!(),
            bar: todo!(),
            _phantom: todo!(),
        }
    }
}
```

and is now:
```rust
impl<T> Foo<T> {
  fn new(a: usize) -> Self {
      Self {
          a,
          bar: Bar::Bar {},
          _phantom: std::marker::PhantomData
      }
  }

  fn random() -> Self {
      Self {
          a: todo!(),
          bar: Bar::Bar {},
          _phantom: std::marker::PhantomData,
      }
  }
}
```

I'd be happy about any suggestions.

## TODO
   - [x]  deduplicate `use_trivial_constructor` (unclear how to do as it's used in two separate crates)
   - [x]  write tests

Closes #12535
This commit is contained in:
bors 2022-07-16 16:36:57 +00:00
commit 01d251789f
4 changed files with 287 additions and 4 deletions

View file

@ -1,3 +1,6 @@
use ide_db::{
imports::import_assets::item_for_path_search, use_trivial_contructor::use_trivial_constructor,
};
use itertools::Itertools; use itertools::Itertools;
use stdx::format_to; use stdx::format_to;
use syntax::ast::{self, AstNode, HasName, HasVisibility, StructKind}; use syntax::ast::{self, AstNode, HasName, HasVisibility, StructKind};
@ -38,6 +41,8 @@ pub(crate) fn generate_new(acc: &mut Assists, ctx: &AssistContext) -> Option<()>
// Return early if we've found an existing new fn // Return early if we've found an existing new fn
let impl_def = find_struct_impl(ctx, &ast::Adt::Struct(strukt.clone()), "new")?; let impl_def = find_struct_impl(ctx, &ast::Adt::Struct(strukt.clone()), "new")?;
let current_module = ctx.sema.scope(strukt.syntax())?.module();
let target = strukt.syntax().text_range(); let target = strukt.syntax().text_range();
acc.add(AssistId("generate_new", AssistKind::Generate), "Generate `new`", target, |builder| { acc.add(AssistId("generate_new", AssistKind::Generate), "Generate `new`", target, |builder| {
let mut buf = String::with_capacity(512); let mut buf = String::with_capacity(512);
@ -48,11 +53,50 @@ pub(crate) fn generate_new(acc: &mut Assists, ctx: &AssistContext) -> Option<()>
let vis = strukt.visibility().map_or(String::new(), |v| format!("{} ", v)); let vis = strukt.visibility().map_or(String::new(), |v| format!("{} ", v));
let trivial_constructors = field_list
.fields()
.map(|f| {
let ty = ctx.sema.resolve_type(&f.ty()?)?;
let item_in_ns = hir::ItemInNs::from(hir::ModuleDef::from(ty.as_adt()?));
let type_path = current_module
.find_use_path(ctx.sema.db, item_for_path_search(ctx.sema.db, item_in_ns)?)?;
let expr = use_trivial_constructor(
&ctx.sema.db,
ide_db::helpers::mod_path_to_ast(&type_path),
&ty,
)?;
Some(format!("{}: {}", f.name()?.syntax(), expr))
})
.collect::<Vec<_>>();
let params = field_list let params = field_list
.fields() .fields()
.filter_map(|f| Some(format!("{}: {}", f.name()?.syntax(), f.ty()?.syntax()))) .enumerate()
.filter_map(|(i, f)| {
if trivial_constructors[i].is_none() {
Some(format!("{}: {}", f.name()?.syntax(), f.ty()?.syntax()))
} else {
None
}
})
.format(", ");
let fields = field_list
.fields()
.enumerate()
.filter_map(|(i, f)| {
let contructor = trivial_constructors[i].clone();
if contructor.is_some() {
contructor
} else {
Some(f.name()?.to_string())
}
})
.format(", "); .format(", ");
let fields = field_list.fields().filter_map(|f| f.name()).format(", ");
format_to!(buf, " {}fn new({}) -> Self {{ Self {{ {} }} }}", vis, params, fields); format_to!(buf, " {}fn new({}) -> Self {{ Self {{ {} }} }}", vis, params, fields);
@ -79,6 +123,97 @@ mod tests {
use super::*; use super::*;
#[test]
fn test_generate_new_with_zst_fields() {
check_assist(
generate_new,
r#"
struct Empty;
struct Foo { empty: Empty $0}
"#,
r#"
struct Empty;
struct Foo { empty: Empty }
impl Foo {
fn $0new() -> Self { Self { empty: Empty } }
}
"#,
);
check_assist(
generate_new,
r#"
struct Empty;
struct Foo { baz: String, empty: Empty $0}
"#,
r#"
struct Empty;
struct Foo { baz: String, empty: Empty }
impl Foo {
fn $0new(baz: String) -> Self { Self { baz, empty: Empty } }
}
"#,
);
check_assist(
generate_new,
r#"
enum Empty { Bar }
struct Foo { empty: Empty $0}
"#,
r#"
enum Empty { Bar }
struct Foo { empty: Empty }
impl Foo {
fn $0new() -> Self { Self { empty: Empty::Bar } }
}
"#,
);
// make sure the assist only works on unit variants
check_assist(
generate_new,
r#"
struct Empty {}
struct Foo { empty: Empty $0}
"#,
r#"
struct Empty {}
struct Foo { empty: Empty }
impl Foo {
fn $0new(empty: Empty) -> Self { Self { empty } }
}
"#,
);
check_assist(
generate_new,
r#"
enum Empty { Bar {} }
struct Foo { empty: Empty $0}
"#,
r#"
enum Empty { Bar {} }
struct Foo { empty: Empty }
impl Foo {
fn $0new(empty: Empty) -> Self { Self { empty } }
}
"#,
);
}
#[test] #[test]
fn test_generate_new() { fn test_generate_new() {
check_assist( check_assist(

View file

@ -20,6 +20,7 @@ pub mod source_change;
pub mod symbol_index; pub mod symbol_index;
pub mod traits; pub mod traits;
pub mod ty_filter; pub mod ty_filter;
pub mod use_trivial_contructor;
pub mod imports { pub mod imports {
pub mod import_assets; pub mod import_assets;

View file

@ -0,0 +1,34 @@
//! Functionality for generating trivial contructors
use hir::StructKind;
use syntax::ast;
/// given a type return the trivial contructor (if one exists)
pub fn use_trivial_constructor(
db: &crate::RootDatabase,
path: ast::Path,
ty: &hir::Type,
) -> Option<ast::Expr> {
match ty.as_adt() {
Some(hir::Adt::Enum(x)) => {
if let &[variant] = &*x.variants(db) {
if variant.kind(db) == hir::StructKind::Unit {
let path = ast::make::path_qualified(
path,
syntax::ast::make::path_segment(ast::make::name_ref(
&variant.name(db).to_smol_str(),
)),
);
return Some(syntax::ast::make::expr_path(path));
}
}
}
Some(hir::Adt::Struct(x)) if x.kind(db) == StructKind::Unit => {
return Some(syntax::ast::make::expr_path(path));
}
_ => {}
}
None
}

View file

@ -3,7 +3,10 @@ use hir::{
db::{AstDatabase, HirDatabase}, db::{AstDatabase, HirDatabase},
known, AssocItem, HirDisplay, InFile, Type, known, AssocItem, HirDisplay, InFile, Type,
}; };
use ide_db::{assists::Assist, famous_defs::FamousDefs, source_change::SourceChange, FxHashMap}; use ide_db::{
assists::Assist, famous_defs::FamousDefs, imports::import_assets::item_for_path_search,
source_change::SourceChange, use_trivial_contructor::use_trivial_constructor, FxHashMap,
};
use stdx::format_to; use stdx::format_to;
use syntax::{ use syntax::{
algo, algo,
@ -55,6 +58,11 @@ fn fixes(ctx: &DiagnosticsContext<'_>, d: &hir::MissingFields) -> Option<Vec<Ass
let root = ctx.sema.db.parse_or_expand(d.file)?; let root = ctx.sema.db.parse_or_expand(d.file)?;
let current_module = match &d.field_list_parent {
Either::Left(ptr) => ctx.sema.scope(ptr.to_node(&root).syntax()).map(|it| it.module()),
Either::Right(ptr) => ctx.sema.scope(ptr.to_node(&root).syntax()).map(|it| it.module()),
};
let build_text_edit = |parent_syntax, new_syntax: &SyntaxNode, old_syntax| { let build_text_edit = |parent_syntax, new_syntax: &SyntaxNode, old_syntax| {
let edit = { let edit = {
let mut builder = TextEdit::builder(); let mut builder = TextEdit::builder();
@ -110,7 +118,26 @@ fn fixes(ctx: &DiagnosticsContext<'_>, d: &hir::MissingFields) -> Option<Vec<Ass
Some(generate_fill_expr(ty)) Some(generate_fill_expr(ty))
} }
} else { } else {
Some(generate_fill_expr(ty)) let expr = (|| -> Option<ast::Expr> {
let item_in_ns = hir::ItemInNs::from(hir::ModuleDef::from(ty.as_adt()?));
let type_path = current_module?.find_use_path(
ctx.sema.db,
item_for_path_search(ctx.sema.db, item_in_ns)?,
)?;
use_trivial_constructor(
&ctx.sema.db,
ide_db::helpers::mod_path_to_ast(&type_path),
&ty,
)
})();
if expr.is_some() {
expr
} else {
Some(generate_fill_expr(ty))
}
}; };
let field = make::record_expr_field( let field = make::record_expr_field(
make::name_ref(&f.name(ctx.sema.db).to_smol_str()), make::name_ref(&f.name(ctx.sema.db).to_smol_str()),
@ -318,6 +345,92 @@ fn test_fn() {
); );
} }
#[test]
fn test_fill_struct_zst_fields() {
check_fix(
r#"
struct Empty;
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct {$0};
}
"#,
r#"
struct Empty;
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct { one: 0, two: Empty };
}
"#,
);
check_fix(
r#"
enum Empty { Foo };
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct {$0};
}
"#,
r#"
enum Empty { Foo };
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct { one: 0, two: Empty::Foo };
}
"#,
);
// make sure the assist doesn't fill non Unit variants
check_fix(
r#"
struct Empty {};
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct {$0};
}
"#,
r#"
struct Empty {};
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct { one: 0, two: todo!() };
}
"#,
);
check_fix(
r#"
enum Empty { Foo {} };
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct {$0};
}
"#,
r#"
enum Empty { Foo {} };
struct TestStruct { one: i32, two: Empty }
fn test_fn() {
let s = TestStruct { one: 0, two: todo!() };
}
"#,
);
}
#[test] #[test]
fn test_fill_struct_fields_self() { fn test_fill_struct_fields_self() {
check_fix( check_fix(