rust-analyzer/crates/ide-assists/src/handlers/flip_binexpr.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

169 lines
4.7 KiB
Rust
Raw Normal View History

use syntax::ast::{self, AstNode, BinExpr};
2020-06-28 22:36:05 +00:00
use crate::{AssistContext, AssistId, AssistKind, Assists};
2019-10-26 16:08:13 +00:00
// Assist: flip_binexpr
//
// Flips operands of a binary expression.
//
// ```
// fn main() {
2021-01-06 20:15:48 +00:00
// let _ = 90 +$0 2;
2019-10-26 16:08:13 +00:00
// }
// ```
// ->
// ```
// fn main() {
// let _ = 2 + 90;
// }
// ```
2022-07-20 13:02:08 +00:00
pub(crate) fn flip_binexpr(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<()> {
let expr = ctx.find_node_at_offset::<BinExpr>()?;
2019-07-19 08:24:41 +00:00
let rhs = expr.rhs()?.syntax().clone();
2023-08-25 23:14:35 +00:00
let lhs = expr.lhs()?.syntax().clone();
let lhs = if let Some(bin_expr) = BinExpr::cast(lhs.clone()) {
if bin_expr.op_kind() == expr.op_kind() {
bin_expr.rhs()?.syntax().clone()
} else {
lhs
}
} else {
lhs
};
2019-07-20 09:58:27 +00:00
let op_range = expr.op_token()?.text_range();
// The assist should be applied only if the cursor is on the operator
let cursor_in_range = op_range.contains_range(ctx.selection_trimmed());
if !cursor_in_range {
return None;
}
let action: FlipAction = expr.op_kind()?.into();
// The assist should not be applied for certain operators
if let FlipAction::DontFlip = action {
return None;
}
2020-06-28 22:36:05 +00:00
acc.add(
2020-07-02 21:48:35 +00:00
AssistId("flip_binexpr", AssistKind::RefactorRewrite),
2020-06-28 22:36:05 +00:00
"Flip binary expression",
op_range,
|edit| {
if let FlipAction::FlipAndReplaceOp(new_op) = action {
edit.replace(op_range, new_op);
}
edit.replace(lhs.text_range(), rhs.text());
edit.replace(rhs.text_range(), lhs.text());
},
)
}
enum FlipAction {
// Flip the expression
Flip,
// Flip the expression and replace the operator with this string
FlipAndReplaceOp(&'static str),
// Do not flip the expression
DontFlip,
}
impl From<ast::BinaryOp> for FlipAction {
fn from(op_kind: ast::BinaryOp) -> Self {
match op_kind {
ast::BinaryOp::Assignment { .. } => FlipAction::DontFlip,
ast::BinaryOp::CmpOp(ast::CmpOp::Ord { ordering, strict }) => {
let rev_op = match (ordering, strict) {
(ast::Ordering::Less, true) => ">",
(ast::Ordering::Less, false) => ">=",
(ast::Ordering::Greater, true) => "<",
(ast::Ordering::Greater, false) => "<=",
};
FlipAction::FlipAndReplaceOp(rev_op)
}
_ => FlipAction::Flip,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
2020-05-06 08:16:55 +00:00
use crate::tests::{check_assist, check_assist_not_applicable, check_assist_target};
#[test]
fn flip_binexpr_target_is_the_op() {
2021-01-06 20:15:48 +00:00
check_assist_target(flip_binexpr, "fn f() { let res = 1 ==$0 2; }", "==")
}
#[test]
fn flip_binexpr_not_applicable_for_assignment() {
2021-01-06 20:15:48 +00:00
check_assist_not_applicable(flip_binexpr, "fn f() { let mut _x = 1; _x +=$0 2 }")
}
#[test]
fn flip_binexpr_works_for_eq() {
2021-01-06 20:15:48 +00:00
check_assist(flip_binexpr, "fn f() { let res = 1 ==$0 2; }", "fn f() { let res = 2 == 1; }")
}
#[test]
fn flip_binexpr_works_for_gt() {
2021-01-06 20:15:48 +00:00
check_assist(flip_binexpr, "fn f() { let res = 1 >$0 2; }", "fn f() { let res = 2 < 1; }")
}
#[test]
fn flip_binexpr_works_for_lteq() {
2021-01-06 20:15:48 +00:00
check_assist(flip_binexpr, "fn f() { let res = 1 <=$0 2; }", "fn f() { let res = 2 >= 1; }")
}
#[test]
fn flip_binexpr_works_for_complex_expr() {
check_assist(
flip_binexpr,
2021-01-06 20:15:48 +00:00
"fn f() { let res = (1 + 1) ==$0 (2 + 2); }",
"fn f() { let res = (2 + 2) == (1 + 1); }",
)
}
2023-08-25 18:47:27 +00:00
#[test]
2023-08-25 23:14:35 +00:00
fn flip_binexpr_works_for_lhs_arith() {
2023-08-25 18:47:27 +00:00
check_assist(
flip_binexpr,
r"fn f() { let res = 1 + (2 - 3) +$0 4 + 5; }",
r"fn f() { let res = 1 + 4 + (2 - 3) + 5; }",
)
}
2023-08-25 23:14:35 +00:00
#[test]
fn flip_binexpr_works_for_lhs_cmp() {
check_assist(
flip_binexpr,
r"fn f() { let res = 1 + (2 - 3) >$0 4 + 5; }",
r"fn f() { let res = 4 + 5 < 1 + (2 - 3); }",
)
}
#[test]
fn flip_binexpr_works_inside_match() {
check_assist(
flip_binexpr,
r#"
fn dyn_eq(&self, other: &dyn Diagnostic) -> bool {
match other.downcast_ref::<Self>() {
None => false,
2021-01-06 20:15:48 +00:00
Some(it) => it ==$0 self,
}
}
"#,
r#"
fn dyn_eq(&self, other: &dyn Diagnostic) -> bool {
match other.downcast_ref::<Self>() {
None => false,
Some(it) => self == it,
}
}
"#,
)
}
}