Index: lib/src/transform/script_compactor.dart |
diff --git a/lib/src/transform/script_compactor.dart b/lib/src/transform/script_compactor.dart |
new file mode 100644 |
index 0000000000000000000000000000000000000000..84d2032e61893d5dabc3aecf5b674e3a2f27da52 |
--- /dev/null |
+++ b/lib/src/transform/script_compactor.dart |
@@ -0,0 +1,122 @@ |
+// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
+// for details. All rights reserved. Use of this source code is governed by a |
+// BSD-style license that can be found in the LICENSE file. |
+ |
+/** Transfomer that combines multiple dart script tags into a single one. */ |
+library polymer.src.transform.script_compactor; |
+ |
+import 'dart:async'; |
+ |
+import 'package:barback/barback.dart'; |
+import 'package:html5lib/parser.dart' show parseFragment; |
+import 'package:path/path.dart' as path; |
+ |
+import 'code_extractor.dart'; // import just for documentation. |
terry
2013/08/20 17:03:24
If this import isn't used the editor will flag a w
Siggi Cherem (dart-lang)
2013/08/21 20:35:42
the good news is that they also recognize that the
|
+import 'common.dart'; |
+ |
+/** |
+ * Combines Dart script tags into a single script tag, and creates a new Dart |
+ * file that calls the main function of each of the original script tags. |
+ * |
+ * This transformer assumes that all script tags point to external files. To |
+ * support script tags with inlined code, use this transformer after running |
+ * [InlineCodeExtractor] on an earlier phase. |
+ * |
+ * Internally, this transformer will convert each script tag into an import |
+ * statement to a library, and then uses `initPolymer` (see polymer.dart) to |
+ * invoke the main method on each of these libraries and register any polymer |
+ * elements annotated with `@CustomTag`. |
+ */ |
+class ScriptCompactor extends Transformer { |
+ Future<bool> isPrimary(Asset input) => |
+ new Future.value(input.id.extension == ".html"); |
+ |
+ Future apply(Transform transform) { |
+ var id = transform.primaryId; |
+ var logger = transform.logger; |
+ return getPrimaryContent(transform).then((content) { |
+ var document = parseHtml(content, id.path, logger); |
+ var libraries = []; |
+ bool changed = false; |
+ for (var tag in document.queryAll('script')) { |
+ if (tag.attributes['type'] != 'application/dart') continue; |
+ tag.remove(); |
+ changed = true; |
+ var src = tag.attributes['src']; |
+ if (src == null) { |
+ logger.warning('unexpected script without a src url', tag.sourceSpan); |
Jennifer Messerly
2013/08/19 22:57:26
Hmm, why the warning here? Are we assuming the scr
Siggi Cherem (dart-lang)
2013/08/21 20:35:42
Good question. The transformers should always be r
|
+ continue; |
+ } |
+ var libraryId = resolve(id, src, logger, tag.sourceSpan); |
+ |
+ // TODO(sigmund): should we detect/remove duplicates? |
Jennifer Messerly
2013/08/19 22:57:26
i think so. don't want to generate erroneous code
Siggi Cherem (dart-lang)
2013/08/21 20:35:42
But then, wouldn't it be ok to import it twice? (w
Jennifer Messerly
2013/08/21 23:35:29
good point, importing twice would be fine assuming
|
+ if (libraryId == null) continue; |
+ libraries.add(libraryId); |
+ } |
+ |
+ if (!changed) { |
+ transform.addOutput(new Asset.fromString(id, content)); |
+ return; |
+ } |
+ |
+ var bootstrapId = id.addExtension('_bootstrap.dart'); |
+ var filename = path.basename(bootstrapId.path); |
+ document.body.nodes.add(parseFragment( |
+ '<script type="application/dart" src="$filename"></script>')); |
+ |
+ var urls = libraries.map((id) => importUrlFor(id, bootstrapId, logger)) |
+ .where((url) => url != null).toList(); |
+ var buffer = new StringBuffer()..write(_header); |
+ for (int i = 0; i < urls.length; i++) { |
+ buffer.writeln("import '${urls[i]}' as i$i;"); |
+ } |
+ buffer..write(_mainPrefix) |
+ ..writeAll(urls.map((url) => " '$url',\n")) |
+ ..write(_mainSuffix); |
+ |
+ transform.addOutput(new Asset.fromString(bootstrapId, buffer.toString())); |
+ transform.addOutput(new Asset.fromString(id, document.outerHtml)); |
+ }); |
+ } |
+ |
+ /** |
+ * Generate the import url for a file described by [id], referenced by a file |
+ * with [sourceId]. |
+ */ |
+ String importUrlFor(AssetId id, AssetId sourceId, TransformLogger logger) { |
+ // use package: urls if possible |
+ if (id.path.startsWith('lib/')) { |
+ return 'package:${id.package}/${id.path.substring(4)}'; |
+ } |
+ |
+ // Use relative urls only if it's possible. |
+ if (id.package != sourceId.package) { |
+ logger.error("don't know how to import $id from $sourceId"); |
+ return null; |
+ } |
+ |
+ var upPath = path.joinAll(path.split(sourceId.path).map((_) => '..')); |
+ return path.normalize(path.join(sourceId.path, upPath, id.path)); |
+ } |
+} |
+ |
+const _header = """ |
+library app_bootstrap; |
+ |
+import 'package:polymer/polymer.dart'; |
+import 'dart:mirrors' show currentMirrorSystem; |
+ |
+"""; |
+ |
+const _mainPrefix = """ |
+ |
+void main() { |
+ initPolymer([ |
+"""; |
+ |
+const _mainSuffix = """ |
+ ], |
+ currentMirrorSystem().findLibrary(const Symbol('app_bootstrap')) |
Jennifer Messerly
2013/08/19 22:57:26
seems unfortunate to need to use mirrors for this.
Siggi Cherem (dart-lang)
2013/08/21 20:35:42
good question. I opened a bug to investigate.
Jennifer Messerly
2013/08/21 23:35:29
btw, I discovered another pattern using mirrors th
|
+ .first.uri.toString()); |
+} |
+"""; |