2015-02-18 12:51:12 -08:00
|
|
|
library angular2.src.transform;
|
2015-02-17 08:38:54 -08:00
|
|
|
|
|
|
|
import 'dart:async';
|
2015-02-19 12:00:09 -08:00
|
|
|
import 'package:analyzer/src/generated/ast.dart';
|
|
|
|
import 'package:analyzer/src/generated/element.dart';
|
2015-02-17 08:38:54 -08:00
|
|
|
import 'package:barback/barback.dart';
|
|
|
|
import 'package:code_transformers/resolver.dart';
|
|
|
|
|
|
|
|
import 'annotation_processor.dart';
|
|
|
|
import 'codegen.dart' as codegen;
|
2015-02-19 12:00:09 -08:00
|
|
|
import 'find_bootstrap.dart';
|
2015-02-17 08:38:54 -08:00
|
|
|
import 'html_transform.dart';
|
2015-02-19 12:00:09 -08:00
|
|
|
import 'logging.dart' as log;
|
2015-02-17 08:38:54 -08:00
|
|
|
import 'options.dart';
|
|
|
|
import 'resolvers.dart';
|
|
|
|
import 'traversal.dart';
|
|
|
|
|
|
|
|
export 'options.dart';
|
|
|
|
|
|
|
|
/// Removes the mirror-based initialization logic and replaces it with static
|
|
|
|
/// logic.
|
|
|
|
class AngularTransformer extends Transformer {
|
|
|
|
final Resolvers _resolvers;
|
|
|
|
final TransformerOptions options;
|
|
|
|
|
|
|
|
AngularTransformer(this.options) : _resolvers = createResolvers();
|
|
|
|
|
2015-02-19 12:00:09 -08:00
|
|
|
static const _bootstrapEntryPointParam = 'bootstrap_entry_point';
|
2015-02-17 08:38:54 -08:00
|
|
|
static const _entryPointParam = 'entry_point';
|
|
|
|
static const _newEntryPointParam = 'new_entry_point';
|
|
|
|
static const _htmlEntryPointParam = 'html_entry_point';
|
|
|
|
|
|
|
|
factory AngularTransformer.asPlugin(BarbackSettings settings) {
|
2015-02-19 12:00:09 -08:00
|
|
|
var bootstrapEntryPoint = settings.configuration[_bootstrapEntryPointParam];
|
2015-02-17 08:38:54 -08:00
|
|
|
var entryPoint = settings.configuration[_entryPointParam];
|
|
|
|
var newEntryPoint = settings.configuration[_newEntryPointParam];
|
|
|
|
if (newEntryPoint == null) {
|
|
|
|
newEntryPoint = entryPoint.replaceFirst('.dart', '.bootstrap.dart');
|
|
|
|
}
|
|
|
|
var htmlEntryPoint = settings.configuration[_htmlEntryPointParam];
|
2015-02-19 12:00:09 -08:00
|
|
|
return new AngularTransformer(new TransformerOptions(
|
|
|
|
bootstrapEntryPoint, entryPoint, newEntryPoint, htmlEntryPoint));
|
2015-02-17 08:38:54 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool isPrimary(AssetId id) =>
|
|
|
|
options.entryPoint == id.path || options.htmlEntryPoint == id.path;
|
|
|
|
|
|
|
|
Future apply(Transform transform) {
|
2015-02-19 12:00:09 -08:00
|
|
|
log.init(transform);
|
|
|
|
|
2015-02-17 08:38:54 -08:00
|
|
|
if (transform.primaryInput.id.path == options.entryPoint) {
|
|
|
|
return _buildBootstrapFile(transform);
|
|
|
|
} else if (transform.primaryInput.id.path == options.htmlEntryPoint) {
|
|
|
|
return transformHtmlEntryPoint(options, transform);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future _buildBootstrapFile(Transform transform) {
|
2015-02-19 12:00:09 -08:00
|
|
|
var bootstrapEntryPointId = new AssetId(
|
|
|
|
transform.primaryInput.id.package, options.bootstrapEntryPoint);
|
2015-02-17 08:38:54 -08:00
|
|
|
var newEntryPointId =
|
|
|
|
new AssetId(transform.primaryInput.id.package, options.newEntryPoint);
|
|
|
|
return transform.hasInput(newEntryPointId).then((exists) {
|
|
|
|
if (exists) {
|
2015-02-19 12:00:09 -08:00
|
|
|
log.logger
|
2015-02-17 08:38:54 -08:00
|
|
|
.error('New entry point file $newEntryPointId already exists.');
|
|
|
|
} else {
|
|
|
|
return _resolvers.get(transform).then((resolver) {
|
2015-02-18 12:51:12 -08:00
|
|
|
try {
|
|
|
|
new _BootstrapFileBuilder(resolver, transform,
|
2015-02-19 12:00:09 -08:00
|
|
|
transform.primaryInput.id, bootstrapEntryPointId,
|
|
|
|
newEntryPointId).run();
|
|
|
|
} catch (err, stackTrace) {
|
|
|
|
log.logger.error('${err}: ${stackTrace}',
|
|
|
|
asset: bootstrapEntryPointId);
|
|
|
|
rethrow;
|
2015-02-18 12:51:12 -08:00
|
|
|
} finally {
|
|
|
|
resolver.release();
|
|
|
|
}
|
2015-02-17 08:38:54 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _BootstrapFileBuilder {
|
|
|
|
final Resolver _resolver;
|
|
|
|
final Transform _transform;
|
2015-02-19 12:00:09 -08:00
|
|
|
final AssetId _bootstrapEntryPoint;
|
2015-02-17 08:38:54 -08:00
|
|
|
final AssetId _entryPoint;
|
|
|
|
final AssetId _newEntryPoint;
|
|
|
|
|
|
|
|
_BootstrapFileBuilder(Resolver resolver, Transform transform,
|
2015-02-19 12:00:09 -08:00
|
|
|
this._entryPoint, this._bootstrapEntryPoint, this._newEntryPoint)
|
2015-02-17 08:38:54 -08:00
|
|
|
: _resolver = resolver,
|
2015-02-19 12:00:09 -08:00
|
|
|
_transform = transform;
|
2015-02-17 08:38:54 -08:00
|
|
|
|
|
|
|
/// Adds the new entry point file to the transform. Should only be ran once.
|
|
|
|
void run() {
|
|
|
|
var entryLib = _resolver.getLibrary(_entryPoint);
|
|
|
|
|
2015-02-19 12:00:09 -08:00
|
|
|
Set<BootstrapCallInfo> bootstrapCalls = findBootstrapCalls(
|
|
|
|
_resolver, _resolver.getLibrary(_bootstrapEntryPoint));
|
|
|
|
|
|
|
|
log.logger.info('found ${bootstrapCalls.length} call(s) to `bootstrap`');
|
|
|
|
bootstrapCalls.forEach((BootstrapCallInfo info) {
|
|
|
|
log.logger.info('Arg1: ${info.bootstrapType}');
|
|
|
|
});
|
|
|
|
|
|
|
|
var types = new Angular2Types(_resolver);
|
|
|
|
// TODO(kegluneq): Also match [Inject].
|
|
|
|
var matcher = new AnnotationMatcher(new Set.from([
|
|
|
|
types.componentAnnotation,
|
|
|
|
types.decoratorAnnotation,
|
|
|
|
types.templateAnnotation
|
|
|
|
]));
|
|
|
|
|
|
|
|
var traversal = new AngularVisibleTraversal(types, matcher);
|
|
|
|
bootstrapCalls.forEach((call) => traversal.traverse(call.bootstrapType));
|
2015-02-17 08:38:54 -08:00
|
|
|
|
|
|
|
var context = new codegen.Context(logger: _transform.logger);
|
2015-02-19 12:00:09 -08:00
|
|
|
matcher.matchQueue
|
2015-02-17 08:38:54 -08:00
|
|
|
.forEach((entry) => context.directiveRegistry.register(entry));
|
|
|
|
|
|
|
|
_transform.addOutput(new Asset.fromString(_newEntryPoint, codegen
|
|
|
|
.codegenEntryPoint(context,
|
|
|
|
entryPoint: entryLib, newEntryPoint: _newEntryPoint)));
|
|
|
|
}
|
|
|
|
}
|