1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
use crate::{pass::Pass, util::IdentExt};
use ast::*;
use swc_common::{Fold, DUMMY_SP};
#[cfg(test)]
mod tests;
pub fn export() -> impl Pass {
ExportDefaultFrom
}
#[derive(Clone)]
struct ExportDefaultFrom;
impl Fold<Vec<ModuleItem>> for ExportDefaultFrom {
fn fold(&mut self, items: Vec<ModuleItem>) -> Vec<ModuleItem> {
let mut stmts = Vec::with_capacity(items.len() + 4);
let mut extra_stmts = Vec::with_capacity(items.len() + 4);
for item in items {
match item {
ModuleItem::ModuleDecl(ModuleDecl::ExportNamed(mut export)) => {
if export.specifiers.iter().all(|s| match *s {
ExportSpecifier::Named(..) => true,
_ => false,
}) {
extra_stmts.push(ModuleItem::ModuleDecl(ModuleDecl::ExportNamed(export)));
continue;
}
match export.specifiers.remove(0) {
ExportSpecifier::Default(DefaultExportSpecifier { exported: default }) => {
let local = default.prefix("_").private();
stmts.push(ModuleItem::ModuleDecl(ModuleDecl::Import(ImportDecl {
span: DUMMY_SP,
specifiers: vec![ImportSpecifier::Default(ImportDefault {
span: DUMMY_SP,
local: local.clone(),
})],
src: export
.src
.clone()
.expect("`export default from` requires source"),
})));
extra_stmts.push(ModuleItem::ModuleDecl(ModuleDecl::ExportNamed(
NamedExport {
span: DUMMY_SP,
specifiers: vec![ExportSpecifier::Named(
NamedExportSpecifier {
span: DUMMY_SP,
orig: local,
exported: Some(default),
},
)],
src: None,
},
)));
}
ExportSpecifier::Namespace(ns) => {
let local = ns.name.prefix("_").private();
stmts.push(ModuleItem::ModuleDecl(ModuleDecl::Import(ImportDecl {
span: DUMMY_SP,
specifiers: vec![ImportSpecifier::Namespace(ImportStarAs {
span: DUMMY_SP,
local: local.clone(),
})],
src: export
.src
.clone()
.expect("`export default from` requires source"),
})));
extra_stmts.push(ModuleItem::ModuleDecl(ModuleDecl::ExportNamed(
NamedExport {
span: DUMMY_SP,
specifiers: vec![ExportSpecifier::Named(
NamedExportSpecifier {
span: DUMMY_SP,
orig: local,
exported: Some(ns.name),
},
)],
src: None,
},
)));
}
_ => unreachable!(),
};
if !export.specifiers.is_empty() {
extra_stmts.push(ModuleItem::ModuleDecl(ModuleDecl::ExportNamed(export)));
}
}
ModuleItem::ModuleDecl(ModuleDecl::Import(..)) => stmts.push(item),
_ => extra_stmts.push(item),
}
}
stmts.append(&mut extra_stmts);
stmts.shrink_to_fit();
stmts
}
}