Index: lib/src/codegen/module_builder.dart |
diff --git a/lib/src/codegen/module_builder.dart b/lib/src/codegen/module_builder.dart |
index 42d8367ae0aae0060dc53c2d74d6e12b0962b79f..a32695ea0170636b201f050f7c105083ddb35b11 100644 |
--- a/lib/src/codegen/module_builder.dart |
+++ b/lib/src/codegen/module_builder.dart |
@@ -12,8 +12,8 @@ import '../options.dart' show ModuleFormat; |
/// Helper that builds JS modules in a given [ModuleFormat]. |
abstract class ModuleBuilder { |
- final List<String> _exports = <String>[]; |
- final List<_ModuleImport> _imports = <_ModuleImport>[]; |
+ final _exports = <String, String>{}; |
+ final _imports = <_ModuleImport>[]; |
ModuleBuilder._(); |
@@ -30,17 +30,21 @@ abstract class ModuleBuilder { |
return new LegacyModuleBuilder(); |
case ModuleFormat.es6: |
return new ES6ModuleBuilder(); |
+ case ModuleFormat.node: |
+ return new NodeModuleBuilder(); |
} |
} |
/// Adds [name] to the list of names to be exported from the module. |
- void addExport(String name) { |
- _exports.add(name); |
+ void addExport(String name, String exportName) { |
+ _exports[name] = exportName; |
} |
/// Adds an import from a module named [name] and locally aliased as [libVar]. |
- /// When [isLazy] is true, the import should be lazy (i.e. there is some |
+ /// When [isLazy] is `true`, the import should be lazy (i.e. there is some |
/// cyclic dependency of imports). |
+ /// When [libVar] is `null`, the import is there just to force the import |
+ /// order. |
void addImport(String name, JS.Identifier libVar, {bool isLazy: false}) { |
_imports.add(new _ModuleImport(name, libVar, isLazy)); |
} |
@@ -65,7 +69,7 @@ class LegacyModuleBuilder extends ModuleBuilder { |
LegacyModuleBuilder() : super._(); |
JS.Program build(String jsPath, String jsModuleValue, |
- JS.Identifier exportsVar, List<JS.ModuleItem> moduleItems) { |
+ JS.Identifier exportsVar, Iterable<JS.ModuleItem> moduleItems) { |
// TODO(jmesserly): it would be great to run the renamer on the body, |
// then figure out if we really need each of these parameters. |
// See ES6 modules: https://github.com/dart-lang/dev_compiler/issues/34 |
@@ -77,6 +81,8 @@ class LegacyModuleBuilder extends ModuleBuilder { |
var moduleStatements = <JS.Statement>[]; |
for (var i in _imports) { |
+ // No need to force the import order for the legacy library mechanism. |
+ if (i.libVar == null) continue; |
(i.isLazy ? lazyImports : imports).add(js.string(i.name, "'")); |
(i.isLazy ? lazyParams : params).add(i.libVar); |
} |
@@ -87,10 +93,10 @@ class LegacyModuleBuilder extends ModuleBuilder { |
if (_exports.isNotEmpty) { |
moduleStatements.add(js.comment('Exports:')); |
// TODO(jmesserly): make these immutable in JS? |
- for (var name in _exports) { |
+ _exports.forEach((name, exportName) { |
moduleStatements |
- .add(js.statement('#.# = #;', [exportsVar, name, name])); |
- } |
+ .add(js.statement('#.# = #;', [exportsVar, exportName, name])); |
+ }); |
} |
var module = |
@@ -123,7 +129,6 @@ class ES6ModuleBuilder extends ModuleBuilder { |
JS.Program build(String jsPath, String jsModuleValue, |
JS.Identifier exportsVar, Iterable<JS.ModuleItem> moduleItems) { |
var moduleStatements = <JS.ModuleItem>[ |
- // Lazy declarations may reference exports. |
js.statement("const # = {};", [exportsVar]) |
]; |
@@ -133,8 +138,13 @@ class ES6ModuleBuilder extends ModuleBuilder { |
for (var i in _imports) { |
var moduleName = js.string(_relativeModuleName(i.name, from: jsPath)); |
// TODO(ochafik): laziness, late binding, etc, to support Closure... |
- moduleStatements.add( |
- new JS.ImportDeclaration(defaultBinding: i.libVar, from: moduleName)); |
+ if (i.libVar == null) { |
+ moduleStatements |
+ .add(new JS.ImportDeclaration(namedImports: [], from: moduleName)); |
+ } else { |
+ moduleStatements.add(new JS.ImportDeclaration( |
+ defaultBinding: i.libVar, from: moduleName)); |
+ } |
} |
moduleStatements.addAll(_flattenBlocks(moduleItems)); |
@@ -142,10 +152,10 @@ class ES6ModuleBuilder extends ModuleBuilder { |
if (_exports.isNotEmpty) { |
moduleStatements.add(js.comment('Exports:')); |
// TODO(jmesserly): make these immutable in JS? |
- for (var name in _exports) { |
+ _exports.forEach((name, exportName) { |
moduleStatements |
- .add(js.statement('#.# = #;', [exportsVar, name, name])); |
- } |
+ .add(js.statement('#.# = #;', [exportsVar, exportName, name])); |
+ }); |
moduleStatements |
.add(new JS.ExportDeclaration(exportsVar, isDefault: true)); |
} |
@@ -154,6 +164,37 @@ class ES6ModuleBuilder extends ModuleBuilder { |
} |
} |
+/// Generates node modules. |
+class NodeModuleBuilder extends ModuleBuilder { |
+ NodeModuleBuilder() : super._(); |
+ |
+ JS.Program build(String jsPath, String jsModuleValue, |
+ JS.Identifier exportsVar, Iterable<JS.ModuleItem> moduleItems) { |
+ var moduleStatements = <JS.ModuleItem>[js.statement("'use strict';"),]; |
+ |
+ for (var i in _imports) { |
+ if (i.libVar == null) { |
+ moduleStatements.add(js.statement('require(#);', [js.string(i.name)])); |
+ } else { |
+ moduleStatements.add( |
+ js.statement('let # = require(#);', [i.libVar, js.string(i.name)])); |
+ } |
+ } |
+ |
+ moduleStatements.addAll(_flattenBlocks(moduleItems)); |
+ |
+ if (_exports.isNotEmpty) { |
+ moduleStatements.add(js.comment('Exports:')); |
+ _exports.forEach((name, exportName) { |
+ moduleStatements |
+ .add(js.statement('#.# = #;', [exportsVar, exportName, name])); |
+ }); |
+ } |
+ // TODO(ochafik): What to do of jsModuleValue? |
Jennifer Messerly
2016/01/29 00:32:31
Wording suggestion: "what to do with jsModuleValue
ochafik
2016/01/29 09:38:17
Thanks for the explanation! Upgraded the comment
|
+ return new JS.Program(moduleStatements); |
+ } |
+} |
+ |
/// Flattens blocks in [stats] to a single list of module items. |
/// Note that in general, blocks should not be flattened, because it can |
/// mess up with block-level scoping (let, const). |