swc/ecmascript/transforms/src/ext.rs

248 lines
5.6 KiB
Rust
Raw Normal View History

2020-08-26 14:50:18 +03:00
use std::{mem::replace, ops::DerefMut};
use swc_common::DUMMY_SP;
2020-07-23 20:18:22 +03:00
use swc_ecma_ast::*;
use swc_ecma_utils::ExprExt;
2020-08-26 14:50:18 +03:00
/// Helper for migration from [Fold] to [VisitMut]
pub(crate) trait MapWithMut: Sized {
fn dummy() -> Self;
fn take(&mut self) -> Self {
replace(self, Self::dummy())
}
2020-08-26 14:50:18 +03:00
#[inline]
fn map_with_mut<F>(&mut self, op: F)
where
F: FnOnce(Self) -> Self,
{
let dummy = Self::dummy();
let v = replace(self, dummy);
let v = op(v);
let _dummy = replace(self, v);
}
}
impl MapWithMut for ModuleItem {
#[inline(always)]
fn dummy() -> Self {
ModuleItem::Stmt(Stmt::Empty(EmptyStmt { span: DUMMY_SP }))
}
}
impl MapWithMut for Stmt {
#[inline(always)]
fn dummy() -> Self {
Stmt::Empty(EmptyStmt { span: DUMMY_SP })
}
}
impl MapWithMut for Expr {
#[inline(always)]
fn dummy() -> Self {
Expr::Invalid(Invalid { span: DUMMY_SP })
}
}
impl MapWithMut for Pat {
#[inline(always)]
fn dummy() -> Self {
Pat::Invalid(Invalid { span: DUMMY_SP })
}
}
impl<T> MapWithMut for Option<T> {
#[inline(always)]
fn dummy() -> Self {
None
}
}
impl<T> MapWithMut for Vec<T> {
#[inline(always)]
fn dummy() -> Self {
Vec::new()
}
}
impl<T> MapWithMut for Box<T>
where
T: MapWithMut,
{
#[inline(always)]
fn dummy() -> Self {
Box::new(T::dummy())
}
}
impl MapWithMut for Ident {
fn dummy() -> Self {
Ident::new(js_word!(""), DUMMY_SP)
}
}
impl MapWithMut for ObjectPatProp {
fn dummy() -> Self {
ObjectPatProp::Assign(AssignPatProp {
span: DUMMY_SP,
key: Ident::dummy(),
value: None,
})
}
}
2020-07-23 20:18:22 +03:00
pub(crate) trait PatOrExprExt: AsOptExpr {
fn as_ref(&self) -> &PatOrExpr;
fn as_mut(&mut self) -> &mut PatOrExpr;
fn as_ident(&self) -> Option<&Ident> {
if let Some(expr) = self.as_expr() {
match expr {
Expr::Ident(i) => return Some(i),
_ => {}
}
}
match self.as_ref() {
PatOrExpr::Expr(e) => match &**e {
Expr::Ident(i) => Some(i),
_ => None,
},
PatOrExpr::Pat(p) => match &**p {
Pat::Ident(i) => Some(i),
_ => None,
},
}
}
fn as_ident_mut(&mut self) -> Option<&mut Ident> {
match self.as_mut() {
PatOrExpr::Pat(p) => match **p {
Pat::Ident(ref mut i) => Some(i),
Pat::Expr(ref mut e) => match e.deref_mut() {
Expr::Ident(i) => Some(i),
_ => None,
},
_ => None,
},
PatOrExpr::Expr(ref mut e) => match e.deref_mut() {
Expr::Ident(i) => Some(i),
_ => None,
},
}
}
fn normalize_expr(self) -> Self;
fn normalize_ident(self) -> Self;
}
impl PatOrExprExt for PatOrExpr {
fn as_ref(&self) -> &PatOrExpr {
self
}
fn as_mut(&mut self) -> &mut PatOrExpr {
self
}
fn normalize_expr(self) -> Self {
match self {
PatOrExpr::Pat(pat) => match *pat {
Pat::Expr(expr) => PatOrExpr::Expr(expr),
_ => return PatOrExpr::Pat(pat),
},
_ => self,
}
}
fn normalize_ident(self) -> Self {
match self {
PatOrExpr::Expr(expr) => match *expr {
Expr::Ident(i) => PatOrExpr::Pat(Box::new(Pat::Ident(i))),
_ => PatOrExpr::Expr(expr),
},
PatOrExpr::Pat(pat) => match *pat {
Pat::Expr(expr) => match *expr {
Expr::Ident(i) => PatOrExpr::Pat(Box::new(Pat::Ident(i))),
_ => PatOrExpr::Expr(expr),
},
_ => PatOrExpr::Pat(pat),
},
}
}
}
pub(crate) trait ExprRefExt: ExprExt {
fn as_ident(&self) -> Option<&Ident> {
match self.as_expr() {
Expr::Ident(ref i) => Some(i),
_ => None,
}
}
}
impl<T> ExprRefExt for T where T: ExprExt {}
pub(crate) trait AsOptExpr {
fn as_expr(&self) -> Option<&Expr>;
fn as_expr_mut(&mut self) -> Option<&mut Expr>;
}
impl AsOptExpr for PatOrExpr {
fn as_expr(&self) -> Option<&Expr> {
match self.as_ref() {
PatOrExpr::Expr(e) => Some(e),
PatOrExpr::Pat(p) => match &**p {
Pat::Expr(e) => Some(e),
_ => None,
},
}
}
fn as_expr_mut(&mut self) -> Option<&mut Expr> {
match self.as_mut() {
PatOrExpr::Expr(e) => Some(e.deref_mut()),
PatOrExpr::Pat(p) => match &mut **p {
Pat::Expr(e) => Some(e.deref_mut()),
_ => None,
},
}
}
}
impl AsOptExpr for ExprOrSuper {
fn as_expr(&self) -> Option<&Expr> {
match self {
ExprOrSuper::Super(_) => None,
ExprOrSuper::Expr(e) => Some(e),
}
}
fn as_expr_mut(&mut self) -> Option<&mut Expr> {
match self {
ExprOrSuper::Super(_) => None,
ExprOrSuper::Expr(e) => Some(e),
}
}
}
impl<N> AsOptExpr for Option<N>
where
N: AsOptExpr,
{
fn as_expr(&self) -> Option<&Expr> {
match self {
Some(n) => n.as_expr(),
None => None,
}
}
fn as_expr_mut(&mut self) -> Option<&mut Expr> {
match self {
None => None,
Some(n) => n.as_expr_mut(),
}
}
}