impl PartialOrd codegen for tuple enum

This commit is contained in:
Yoshua Wuyts 2021-10-12 15:07:57 +02:00
parent 77b5fe6c52
commit 5f72bd81a9
2 changed files with 56 additions and 12 deletions

View file

@ -821,6 +821,47 @@ impl PartialOrd for Foo {
) )
} }
#[test]
fn add_custom_impl_partial_ord_tuple_enum() {
check_assist(
replace_derive_with_manual_impl,
r#"
//- minicore: ord
#[derive(Partial$0Ord)]
enum Foo {
Bar(String),
Baz(String, String),
Qux(),
Bin,
}
"#,
r#"
enum Foo {
Bar(String),
Baz(String, String),
Qux(),
Bin,
}
impl PartialOrd for Foo {
$0fn partial_cmp(&self, other: &Self) -> Option<core::cmp::Ordering> {
match (self, other) {
(Self::Bar(l0), Self::Bar(r0)) => l0.partial_cmp(r0),
(Self::Baz(l0, l1), Self::Baz(r0, r1)) => {
match l0.partial_cmp(r0) {
Some(core::cmp::Ordering::Eq) => {}
ord => return ord,
}
l1.partial_cmp(r1)
}
_ => core::mem::discriminant(self).partial_cmp(core::mem::discriminant(other)),
}
}
}
"#,
)
}
#[test] #[test]
fn add_custom_impl_partial_eq_record_struct() { fn add_custom_impl_partial_eq_record_struct() {
check_assist( check_assist(

View file

@ -574,13 +574,6 @@ fn gen_partial_eq(adt: &ast::Adt, func: &ast::Fn) -> Option<()> {
} }
fn gen_partial_ord(adt: &ast::Adt, func: &ast::Fn) -> Option<()> { fn gen_partial_ord(adt: &ast::Adt, func: &ast::Fn) -> Option<()> {
fn gen_eq_chain(expr: Option<ast::Expr>, cmp: ast::Expr) -> Option<ast::Expr> {
match expr {
Some(expr) => Some(make::expr_op(ast::BinOp::BooleanAnd, expr, cmp)),
None => Some(cmp),
}
}
fn gen_partial_eq_match(match_target: ast::Expr) -> Option<ast::Stmt> { fn gen_partial_eq_match(match_target: ast::Expr) -> Option<ast::Stmt> {
let mut arms = vec![]; let mut arms = vec![];
@ -683,7 +676,7 @@ fn gen_partial_ord(adt: &ast::Adt, func: &ast::Fn) -> Option<()> {
} }
Some(ast::FieldList::TupleFieldList(list)) => { Some(ast::FieldList::TupleFieldList(list)) => {
let mut expr = None; let mut exprs = vec![];
let mut l_fields = vec![]; let mut l_fields = vec![];
let mut r_fields = vec![]; let mut r_fields = vec![];
@ -698,16 +691,26 @@ fn gen_partial_ord(adt: &ast::Adt, func: &ast::Fn) -> Option<()> {
let lhs = make::expr_path(make::ext::ident_path(&l_name)); let lhs = make::expr_path(make::ext::ident_path(&l_name));
let rhs = make::expr_path(make::ext::ident_path(&r_name)); let rhs = make::expr_path(make::ext::ident_path(&r_name));
let cmp = make::expr_op(ast::BinOp::EqualityTest, lhs, rhs); let ord = gen_partial_cmp_call(lhs, rhs);
expr = gen_eq_chain(expr, cmp); exprs.push(ord);
} }
let left = make::tuple_struct_pat(gen_variant_path(&variant)?, l_fields); let left = make::tuple_struct_pat(gen_variant_path(&variant)?, l_fields);
let right = make::tuple_struct_pat(gen_variant_path(&variant)?, r_fields); let right = make::tuple_struct_pat(gen_variant_path(&variant)?, r_fields);
let tuple = make::tuple_pat(vec![left.into(), right.into()]); let tuple = make::tuple_pat(vec![left.into(), right.into()]);
if let Some(expr) = expr { if let Some(tail) = exprs.pop() {
arms.push(make::match_arm(Some(tuple.into()), None, expr)); let stmts = exprs
.into_iter()
.map(gen_partial_eq_match)
.collect::<Option<Vec<ast::Stmt>>>()?;
let expr = match stmts.len() {
0 => tail,
_ => make::block_expr(stmts.into_iter(), Some(tail))
.indent(ast::edit::IndentLevel(1))
.into(),
};
arms.push(make::match_arm(Some(tuple.into()), None, expr.into()));
} }
} }
None => continue, None => continue,