rust-analyzer/crates/ra_syntax/src/ast/expr_extensions.rs

399 lines
12 KiB
Rust
Raw Normal View History

2019-04-02 10:02:23 +00:00
//! Various extension methods to ast Expr Nodes, which are hard to code-generate.
2019-04-02 09:47:39 +00:00
use crate::{
ast::{self, child_opt, children, AstChildren, AstNode},
2019-07-20 17:04:34 +00:00
SmolStr,
2019-05-15 12:35:47 +00:00
SyntaxKind::*,
SyntaxToken, T,
2019-04-02 09:47:39 +00:00
};
impl ast::Expr {
pub fn is_block_like(&self) -> bool {
match self {
ast::Expr::IfExpr(_)
| ast::Expr::LoopExpr(_)
| ast::Expr::ForExpr(_)
| ast::Expr::WhileExpr(_)
| ast::Expr::BlockExpr(_)
| ast::Expr::MatchExpr(_)
| ast::Expr::TryBlockExpr(_) => true,
_ => false,
}
}
}
2019-04-02 09:47:39 +00:00
#[derive(Debug, Clone, PartialEq, Eq)]
2019-07-18 16:23:05 +00:00
pub enum ElseBranch {
2019-09-02 18:23:19 +00:00
Block(ast::BlockExpr),
2019-07-18 16:23:05 +00:00
IfExpr(ast::IfExpr),
2019-04-02 09:47:39 +00:00
}
impl ast::IfExpr {
2019-09-02 18:23:19 +00:00
pub fn then_branch(&self) -> Option<ast::BlockExpr> {
self.blocks().next()
2019-04-02 09:47:39 +00:00
}
pub fn else_branch(&self) -> Option<ElseBranch> {
let res = match self.blocks().nth(1) {
Some(block) => ElseBranch::Block(block),
None => {
2019-07-18 16:23:05 +00:00
let elif: ast::IfExpr = child_opt(self)?;
2019-04-02 09:47:39 +00:00
ElseBranch::IfExpr(elif)
}
};
Some(res)
}
2019-09-02 18:23:19 +00:00
fn blocks(&self) -> AstChildren<ast::BlockExpr> {
2019-04-02 09:47:39 +00:00
children(self)
}
}
impl ast::RefExpr {
pub fn is_mut(&self) -> bool {
2019-05-15 12:35:47 +00:00
self.syntax().children_with_tokens().any(|n| n.kind() == T![mut])
2019-04-02 09:47:39 +00:00
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum PrefixOp {
/// The `*` operator for dereferencing
Deref,
/// The `!` operator for logical inversion
Not,
/// The `-` operator for negation
Neg,
}
impl ast::PrefixExpr {
pub fn op_kind(&self) -> Option<PrefixOp> {
match self.op_token()?.kind() {
2019-05-15 12:35:47 +00:00
T![*] => Some(PrefixOp::Deref),
T![!] => Some(PrefixOp::Not),
T![-] => Some(PrefixOp::Neg),
2019-04-02 09:47:39 +00:00
_ => None,
}
}
pub fn op_token(&self) -> Option<SyntaxToken> {
2019-07-19 16:05:34 +00:00
self.syntax().first_child_or_token()?.into_token()
2019-04-02 09:47:39 +00:00
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum BinOp {
/// The `||` operator for boolean OR
BooleanOr,
/// The `&&` operator for boolean AND
BooleanAnd,
/// The `==` operator for equality testing
EqualityTest,
/// The `!=` operator for equality testing
NegatedEqualityTest,
/// The `<=` operator for lesser-equal testing
LesserEqualTest,
/// The `>=` operator for greater-equal testing
GreaterEqualTest,
/// The `<` operator for comparison
LesserTest,
/// The `>` operator for comparison
GreaterTest,
/// The `+` operator for addition
Addition,
/// The `*` operator for multiplication
Multiplication,
/// The `-` operator for subtraction
Subtraction,
/// The `/` operator for division
Division,
/// The `%` operator for remainder after division
Remainder,
/// The `<<` operator for left shift
LeftShift,
/// The `>>` operator for right shift
RightShift,
/// The `^` operator for bitwise XOR
BitwiseXor,
/// The `|` operator for bitwise OR
BitwiseOr,
/// The `&` operator for bitwise AND
BitwiseAnd,
/// The `=` operator for assignment
Assignment,
/// The `+=` operator for assignment after addition
AddAssign,
/// The `/=` operator for assignment after division
DivAssign,
/// The `*=` operator for assignment after multiplication
MulAssign,
/// The `%=` operator for assignment after remainders
RemAssign,
/// The `>>=` operator for assignment after shifting right
ShrAssign,
/// The `<<=` operator for assignment after shifting left
ShlAssign,
/// The `-=` operator for assignment after subtraction
SubAssign,
/// The `|=` operator for assignment after bitwise OR
BitOrAssign,
/// The `&=` operator for assignment after bitwise AND
BitAndAssign,
/// The `^=` operator for assignment after bitwise XOR
BitXorAssign,
}
2020-01-04 20:54:31 +00:00
impl BinOp {
2020-01-13 16:27:06 +00:00
pub fn is_assignment(self) -> bool {
match self {
2020-01-04 20:54:31 +00:00
BinOp::Assignment
| BinOp::AddAssign
| BinOp::DivAssign
| BinOp::MulAssign
| BinOp::RemAssign
| BinOp::ShrAssign
| BinOp::ShlAssign
| BinOp::SubAssign
| BinOp::BitOrAssign
| BinOp::BitAndAssign
| BinOp::BitXorAssign => true,
_ => false,
}
}
}
2019-04-02 09:47:39 +00:00
impl ast::BinExpr {
2019-11-24 05:14:57 +00:00
pub fn op_details(&self) -> Option<(SyntaxToken, BinOp)> {
2019-08-17 14:14:22 +00:00
self.syntax().children_with_tokens().filter_map(|it| it.into_token()).find_map(|c| {
let bin_op = match c.kind() {
T![||] => BinOp::BooleanOr,
T![&&] => BinOp::BooleanAnd,
T![==] => BinOp::EqualityTest,
T![!=] => BinOp::NegatedEqualityTest,
T![<=] => BinOp::LesserEqualTest,
T![>=] => BinOp::GreaterEqualTest,
T![<] => BinOp::LesserTest,
T![>] => BinOp::GreaterTest,
T![+] => BinOp::Addition,
T![*] => BinOp::Multiplication,
T![-] => BinOp::Subtraction,
T![/] => BinOp::Division,
T![%] => BinOp::Remainder,
T![<<] => BinOp::LeftShift,
T![>>] => BinOp::RightShift,
T![^] => BinOp::BitwiseXor,
T![|] => BinOp::BitwiseOr,
T![&] => BinOp::BitwiseAnd,
T![=] => BinOp::Assignment,
T![+=] => BinOp::AddAssign,
T![/=] => BinOp::DivAssign,
T![*=] => BinOp::MulAssign,
T![%=] => BinOp::RemAssign,
T![>>=] => BinOp::ShrAssign,
T![<<=] => BinOp::ShlAssign,
T![-=] => BinOp::SubAssign,
T![|=] => BinOp::BitOrAssign,
T![&=] => BinOp::BitAndAssign,
T![^=] => BinOp::BitXorAssign,
_ => return None,
};
Some((c, bin_op))
2019-04-02 09:47:39 +00:00
})
}
pub fn op_kind(&self) -> Option<BinOp> {
self.op_details().map(|t| t.1)
}
pub fn op_token(&self) -> Option<SyntaxToken> {
self.op_details().map(|t| t.0)
}
2019-07-18 16:23:05 +00:00
pub fn lhs(&self) -> Option<ast::Expr> {
children(self).next()
2019-04-02 09:47:39 +00:00
}
2019-07-18 16:23:05 +00:00
pub fn rhs(&self) -> Option<ast::Expr> {
2019-04-02 09:47:39 +00:00
children(self).nth(1)
}
2019-07-18 16:23:05 +00:00
pub fn sub_exprs(&self) -> (Option<ast::Expr>, Option<ast::Expr>) {
2019-04-02 09:47:39 +00:00
let mut children = children(self);
let first = children.next();
let second = children.next();
(first, second)
}
}
2019-11-15 20:05:29 +00:00
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum RangeOp {
/// `..`
Exclusive,
/// `..=`
Inclusive,
}
impl ast::RangeExpr {
fn op_details(&self) -> Option<(usize, SyntaxToken, RangeOp)> {
self.syntax().children_with_tokens().enumerate().find_map(|(ix, child)| {
let token = child.into_token()?;
let bin_op = match token.kind() {
T![..] => RangeOp::Exclusive,
T![..=] => RangeOp::Inclusive,
_ => return None,
};
Some((ix, token, bin_op))
})
}
pub fn op_kind(&self) -> Option<RangeOp> {
self.op_details().map(|t| t.2)
}
pub fn op_token(&self) -> Option<SyntaxToken> {
self.op_details().map(|t| t.1)
}
pub fn start(&self) -> Option<ast::Expr> {
let op_ix = self.op_details()?.0;
self.syntax()
.children_with_tokens()
.take(op_ix)
.find_map(|it| ast::Expr::cast(it.into_node()?))
}
pub fn end(&self) -> Option<ast::Expr> {
let op_ix = self.op_details()?.0;
self.syntax()
.children_with_tokens()
.skip(op_ix + 1)
.find_map(|it| ast::Expr::cast(it.into_node()?))
}
}
2019-08-17 14:17:01 +00:00
impl ast::IndexExpr {
pub fn base(&self) -> Option<ast::Expr> {
children(self).next()
2019-08-17 14:17:01 +00:00
}
pub fn index(&self) -> Option<ast::Expr> {
children(self).nth(1)
}
}
2019-07-18 16:23:05 +00:00
pub enum ArrayExprKind {
Repeat { initializer: Option<ast::Expr>, repeat: Option<ast::Expr> },
ElementList(AstChildren<ast::Expr>),
}
impl ast::ArrayExpr {
pub fn kind(&self) -> ArrayExprKind {
if self.is_repeat() {
ArrayExprKind::Repeat {
initializer: children(self).next(),
2019-04-05 10:19:25 +00:00
repeat: children(self).nth(1),
}
} else {
ArrayExprKind::ElementList(children(self))
}
}
fn is_repeat(&self) -> bool {
2019-05-15 12:35:47 +00:00
self.syntax().children_with_tokens().any(|it| it.kind() == T![;])
}
}
2019-04-02 09:47:39 +00:00
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
2019-04-02 09:48:14 +00:00
pub enum LiteralKind {
2019-04-02 09:47:39 +00:00
String,
ByteString,
Char,
Byte,
IntNumber { suffix: Option<SmolStr> },
FloatNumber { suffix: Option<SmolStr> },
Bool,
}
impl ast::Literal {
pub fn token(&self) -> SyntaxToken {
2019-07-20 17:04:34 +00:00
self.syntax()
2019-06-15 13:22:31 +00:00
.children_with_tokens()
2019-07-20 17:04:34 +00:00
.find(|e| e.kind() != ATTR && !e.kind().is_trivia())
.and_then(|e| e.into_token())
.unwrap()
2019-04-02 09:47:39 +00:00
}
fn find_suffix(text: &str, possible_suffixes: &[&str]) -> Option<SmolStr> {
possible_suffixes
.iter()
.find(|&suffix| text.ends_with(suffix))
.map(|&suffix| SmolStr::new(suffix))
}
pub fn kind(&self) -> LiteralKind {
const INT_SUFFIXES: [&str; 12] = [
"u64", "u32", "u16", "u8", "usize", "isize", "i64", "i32", "i16", "i8", "u128", "i128",
];
const FLOAT_SUFFIXES: [&str; 2] = ["f32", "f64"];
let token = self.token();
match token.kind() {
INT_NUMBER => {
2020-01-22 10:39:56 +00:00
// FYI: there was a bug here previously, thus an if statement bellow is necessary.
2020-01-22 10:39:32 +00:00
// The lexer treats e.g. `1f64` as an integer literal. See
// https://github.com/rust-analyzer/rust-analyzer/issues/1592
// and the comments on the linked PR.
let text = token.text();
if let suffix @ Some(_) = Self::find_suffix(&text, &FLOAT_SUFFIXES) {
LiteralKind::FloatNumber { suffix }
} else {
LiteralKind::IntNumber { suffix: Self::find_suffix(&text, &INT_SUFFIXES) }
}
2019-04-02 09:47:39 +00:00
}
FLOAT_NUMBER => {
let text = token.text();
LiteralKind::FloatNumber { suffix: Self::find_suffix(&text, &FLOAT_SUFFIXES) }
2019-04-02 09:47:39 +00:00
}
2019-04-02 09:48:14 +00:00
STRING | RAW_STRING => LiteralKind::String,
2019-05-15 12:35:47 +00:00
T![true] | T![false] => LiteralKind::Bool,
2019-04-02 09:48:14 +00:00
BYTE_STRING | RAW_BYTE_STRING => LiteralKind::ByteString,
CHAR => LiteralKind::Char,
BYTE => LiteralKind::Byte,
2019-04-02 09:47:39 +00:00
_ => unreachable!(),
}
}
}
2019-09-02 18:41:50 +00:00
impl ast::BlockExpr {
/// false if the block is an intrinsic part of the syntax and can't be
/// replaced with arbitrary expression.
///
/// ```not_rust
/// fn foo() { not_stand_alone }
/// const FOO: () = { stand_alone };
/// ```
pub fn is_standalone(&self) -> bool {
let kind = match self.syntax().parent() {
None => return true,
Some(it) => it.kind(),
};
match kind {
FN_DEF | MATCH_ARM | IF_EXPR | WHILE_EXPR | LOOP_EXPR | TRY_BLOCK_EXPR => false,
_ => true,
}
}
}
2019-06-15 13:22:31 +00:00
#[test]
fn test_literal_with_attr() {
let parse = ast::SourceFile::parse(r#"const _: &str = { #[attr] "Hello" };"#);
2019-07-18 16:23:05 +00:00
let lit = parse.tree().syntax().descendants().find_map(ast::Literal::cast).unwrap();
2019-06-15 13:22:31 +00:00
assert_eq!(lit.token().text(), r#""Hello""#);
}
2019-08-23 12:55:21 +00:00
impl ast::RecordField {
pub fn parent_record_lit(&self) -> ast::RecordLit {
self.syntax().ancestors().find_map(ast::RecordLit::cast).unwrap()
2019-04-02 09:47:39 +00:00
}
}