Index: lib/plugin_transformer.dart |
diff --git a/lib/plugin_transformer.dart b/lib/plugin_transformer.dart |
new file mode 100644 |
index 0000000000000000000000000000000000000000..a7ecb8931dd3e2caa5c22e3d2dfe0fe2ceaaa5a4 |
--- /dev/null |
+++ b/lib/plugin_transformer.dart |
@@ -0,0 +1,87 @@ |
+// Copyright (c) 2015, 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. |
+library initialize.plugin_transformer; |
+ |
+import 'dart:async'; |
+import 'package:analyzer/analyzer.dart'; |
+import 'package:barback/barback.dart'; |
+import 'package:source_maps/refactor.dart'; |
+import 'package:source_span/source_span.dart'; |
+ |
+/// Abstract transformer which will call [initEntry] for every [InitEntry] |
Jennifer Messerly
2015/01/30 20:34:26
could we do the initEntry here instead? it would s
jakemac
2015/01/30 22:16:17
Yes, especially with transforms since they are jus
|
+/// expression found in the bootstrap file. This is used to centralize logic |
+/// for initializers which want to do something special at transform time. |
+abstract class InitializePluginTransformer extends AggregateTransformer { |
+ // Path to the bootstrap file created by the initialize transformer. |
+ final String _bootstrapFile; |
+ |
+ // All the extra assets that were found, if child classes override |
+ // classifyPrimary this lets them get at the other assets easily. |
+ final allAssets = <Asset>[]; |
+ |
+ TransformLogger _logger; |
+ |
+ InitializePluginTransformer(this._bootstrapFile); |
+ |
+ classifyPrimary(AssetId id) { |
+ if (_bootstrapFile == id.path) return _bootstrapFile; |
Jennifer Messerly
2015/01/30 20:34:26
silly, but I'd usually format this as:
=> _boots
jakemac
2015/01/30 22:16:18
Done.
|
+ return null; |
+ } |
+ |
+ Future apply(AggregateTransform transform) { |
+ _logger = transform.logger; |
+ var done = new Completer(); |
+ var listener = transform.primaryInputs.listen((Asset asset) { |
+ allAssets.add(asset); |
+ var id = asset.id; |
+ if (id.path != _bootstrapFile) return; |
+ // Calls initEntry for each InitEntry expression. |
+ asset.readAsString().then((dartCode) { |
+ var url = id.path.startsWith('lib/') |
+ ? 'package:${id.package}/${id.path.substring(4)}' |
+ : id.path; |
+ var source = new SourceFile(dartCode, url: url); |
+ var transaction = new TextEditTransaction(dartCode, source); |
+ (parseCompilationUnit(dartCode, |
+ suppressErrors: true) as dynamic).declarations.firstWhere( |
Jennifer Messerly
2015/01/30 20:34:26
is there a reason for `as dynamic`?
jakemac
2015/01/30 22:16:18
This is just to get rid of a bunch of casts and ty
|
+ (d) => d.name.toString() == |
Jennifer Messerly
2015/01/30 20:34:26
in general not all CompilationUnitMembers have nam
jakemac
2015/01/30 22:16:17
Related to the previous comment, I did start going
|
+ 'main').functionExpression.body.block.statements.firstWhere( |
+ (statement) { |
+ return statement.expression.target.toString() == 'initializers' && |
+ statement.expression.methodName.toString() == 'addAll'; |
+ }).expression.argumentList.arguments[0].elements |
+ .where((arg) => arg is InstanceCreationExpression) |
+ .forEach((e) => initEntry(e, transaction)); |
+ |
+ // Apply any transformations. |
+ if (!transaction.hasEdits) { |
+ transform.addOutput(asset); |
+ } else { |
+ var printer = transaction.commit(); |
+ printer.build(url); |
+ transform.addOutput(new Asset.fromString(id, printer.text)); |
+ } |
+ done.complete(); |
+ }); |
+ }); |
+ |
+ // Make sure all the assets are read before returning. |
+ return Future.wait([done.future, listener.asFuture()]); |
Jennifer Messerly
2015/01/30 20:34:26
if you're returning listener.asFuture(), do you ne
jakemac
2015/01/30 22:16:17
Looks like you are right, removed it.
|
+ } |
+ |
+ /// Gets called once for each generated [InitEntry] expression in the |
+ /// bootstrap file. A [TextEditTransaction] is supplied so that the user can |
+ /// modify the expression however they see fit. |
+ void initEntry( |
+ InstanceCreationExpression expression, TextEditTransaction transaction); |
+ |
+ /// Convenience method to delete an Initializer expression completely. |
+ void removeInitializer( |
+ InstanceCreationExpression expression, TextEditTransaction transaction) { |
+ // Delete the entire line. |
+ var line = transaction.file.getLine(expression.offset); |
+ transaction.edit(transaction.file.getOffset(line), |
+ transaction.file.getOffset(line + 1), ''); |
+ } |
+} |