2015-04-15 18:17:22 -07:00
|
|
|
library angular2.transform.directive_processor.rewriter;
|
2015-02-27 14:42:21 -08:00
|
|
|
|
|
|
|
import 'package:analyzer/analyzer.dart';
|
|
|
|
import 'package:analyzer/src/generated/java_core.dart';
|
2015-05-08 15:03:08 -07:00
|
|
|
import 'package:angular2/src/transform/common/annotation_matcher.dart';
|
2015-02-27 14:42:21 -08:00
|
|
|
import 'package:angular2/src/transform/common/logging.dart';
|
|
|
|
import 'package:angular2/src/transform/common/names.dart';
|
2015-03-19 15:47:10 -07:00
|
|
|
import 'package:barback/barback.dart' show AssetId;
|
2015-02-27 14:42:21 -08:00
|
|
|
import 'package:path/path.dart' as path;
|
|
|
|
|
|
|
|
import 'visitors.dart';
|
|
|
|
|
2015-05-08 15:03:08 -07:00
|
|
|
/// Generates a file registering all Angular 2 `Directive`s found in `code` in
|
|
|
|
/// ngDeps format [TODO(kegluneq): documentation reference needed]. `assetId` is
|
|
|
|
/// the id of the asset containing `code`.
|
2015-02-27 14:42:21 -08:00
|
|
|
///
|
2015-05-08 15:03:08 -07:00
|
|
|
/// If no Angular 2 `Directive`s are found in `code`, returns the empty
|
|
|
|
/// string unless `forceGenerate` is true, in which case an empty ngDeps
|
2015-02-27 14:42:21 -08:00
|
|
|
/// file is created.
|
2015-05-08 15:03:08 -07:00
|
|
|
String createNgDeps(
|
|
|
|
String code, AssetId assetId, AnnotationMatcher annotationMatcher) {
|
2015-02-27 14:42:21 -08:00
|
|
|
// TODO(kegluneq): Shortcut if we can determine that there are no
|
2015-03-19 14:10:28 -07:00
|
|
|
// [Directive]s present, taking into account `export`s.
|
2015-02-27 14:42:21 -08:00
|
|
|
var writer = new PrintStringWriter();
|
2015-05-08 15:03:08 -07:00
|
|
|
var visitor = new CreateNgDepsVisitor(writer, assetId, annotationMatcher);
|
|
|
|
parseCompilationUnit(code, name: assetId.toString()).accept(visitor);
|
2015-03-19 14:10:28 -07:00
|
|
|
return '$writer';
|
2015-02-27 14:42:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Visitor responsible for processing [CompilationUnit] and creating an
|
2015-03-16 13:32:29 -07:00
|
|
|
/// associated .ng_deps.dart file.
|
2015-04-13 16:54:09 -07:00
|
|
|
class CreateNgDepsVisitor extends Object with SimpleAstVisitor<Object> {
|
2015-02-27 14:42:21 -08:00
|
|
|
final PrintWriter writer;
|
2015-03-19 14:10:28 -07:00
|
|
|
bool _foundNgDirectives = false;
|
|
|
|
bool _wroteImport = false;
|
2015-02-27 14:42:21 -08:00
|
|
|
final ToSourceVisitor _copyVisitor;
|
|
|
|
final FactoryTransformVisitor _factoryVisitor;
|
|
|
|
final ParameterTransformVisitor _paramsVisitor;
|
|
|
|
final AnnotationsTransformVisitor _metaVisitor;
|
2015-05-08 15:03:08 -07:00
|
|
|
final AnnotationMatcher _annotationMatcher;
|
2015-02-27 14:42:21 -08:00
|
|
|
|
2015-05-08 15:03:08 -07:00
|
|
|
/// The assetId for the file which we are parsing.
|
|
|
|
final AssetId assetId;
|
2015-02-27 14:42:21 -08:00
|
|
|
|
2015-05-08 15:03:08 -07:00
|
|
|
CreateNgDepsVisitor(PrintWriter writer, this.assetId, this._annotationMatcher)
|
2015-02-27 14:42:21 -08:00
|
|
|
: writer = writer,
|
|
|
|
_copyVisitor = new ToSourceVisitor(writer),
|
|
|
|
_factoryVisitor = new FactoryTransformVisitor(writer),
|
|
|
|
_paramsVisitor = new ParameterTransformVisitor(writer),
|
2015-05-08 15:03:08 -07:00
|
|
|
_metaVisitor = new AnnotationsTransformVisitor(writer);
|
2015-02-27 14:42:21 -08:00
|
|
|
|
2015-04-13 16:54:09 -07:00
|
|
|
void _visitNodeListWithSeparator(NodeList<AstNode> list, String separator) {
|
|
|
|
if (list == null) return;
|
|
|
|
for (var i = 0, iLen = list.length; i < iLen; ++i) {
|
|
|
|
if (i != 0) {
|
|
|
|
writer.print(separator);
|
|
|
|
}
|
|
|
|
list[i].accept(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-27 14:42:21 -08:00
|
|
|
@override
|
2015-04-13 16:54:09 -07:00
|
|
|
Object visitCompilationUnit(CompilationUnit node) {
|
|
|
|
_visitNodeListWithSeparator(node.directives, " ");
|
2015-02-27 14:42:21 -08:00
|
|
|
_openFunctionWrapper();
|
2015-04-13 16:54:09 -07:00
|
|
|
_visitNodeListWithSeparator(node.declarations, " ");
|
2015-02-27 14:42:21 -08:00
|
|
|
_closeFunctionWrapper();
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2015-03-19 14:10:28 -07:00
|
|
|
/// Write the import to the file the .ng_deps.dart file is based on if it
|
|
|
|
/// has not yet been written.
|
|
|
|
void _maybeWriteImport() {
|
|
|
|
if (_wroteImport) return;
|
|
|
|
_wroteImport = true;
|
2015-05-08 15:03:08 -07:00
|
|
|
writer.print('''import '${path.basename(assetId.path)}';''');
|
2015-02-27 14:42:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Object visitImportDirective(ImportDirective node) {
|
2015-03-19 14:10:28 -07:00
|
|
|
_maybeWriteImport();
|
2015-02-27 14:42:21 -08:00
|
|
|
return node.accept(_copyVisitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Object visitExportDirective(ExportDirective node) {
|
2015-03-19 14:10:28 -07:00
|
|
|
_maybeWriteImport();
|
2015-02-27 14:42:21 -08:00
|
|
|
return node.accept(_copyVisitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _openFunctionWrapper() {
|
2015-03-19 14:10:28 -07:00
|
|
|
_maybeWriteImport();
|
2015-04-24 12:05:15 -07:00
|
|
|
writer.print('var _visited = false;'
|
2015-02-27 14:42:21 -08:00
|
|
|
'void ${SETUP_METHOD_NAME}(${REFLECTOR_VAR_NAME}) {'
|
|
|
|
'if (_visited) return; _visited = true;');
|
|
|
|
}
|
|
|
|
|
|
|
|
void _closeFunctionWrapper() {
|
2015-03-19 14:10:28 -07:00
|
|
|
if (_foundNgDirectives) {
|
2015-02-27 14:42:21 -08:00
|
|
|
writer.print(';');
|
|
|
|
}
|
|
|
|
writer.print('}');
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstructorDeclaration _getCtor(ClassDeclaration node) {
|
|
|
|
int numCtorsFound = 0;
|
|
|
|
var ctor = null;
|
|
|
|
|
|
|
|
for (ClassMember classMember in node.members) {
|
|
|
|
if (classMember is ConstructorDeclaration) {
|
|
|
|
numCtorsFound++;
|
|
|
|
ConstructorDeclaration constructor = classMember;
|
|
|
|
|
|
|
|
// Use the unnnamed constructor if it is present.
|
|
|
|
// Otherwise, use the first encountered.
|
|
|
|
if (ctor == null) {
|
|
|
|
ctor = constructor;
|
|
|
|
} else if (constructor.name == null) {
|
|
|
|
ctor = constructor;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (numCtorsFound > 1) {
|
|
|
|
var ctorName = ctor.name;
|
|
|
|
ctorName = ctorName == null
|
|
|
|
? 'the unnamed constructor'
|
|
|
|
: 'constructor "${ctorName}"';
|
|
|
|
logger.warning('Found ${numCtorsFound} ctors for class ${node.name},'
|
|
|
|
'Using ${ctorName}.');
|
|
|
|
}
|
|
|
|
return ctor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _generateEmptyFactory(String typeName) {
|
|
|
|
writer.print('() => new ${typeName}()');
|
|
|
|
}
|
|
|
|
|
|
|
|
void _generateEmptyParams() => writer.print('const []');
|
|
|
|
|
|
|
|
@override
|
|
|
|
Object visitClassDeclaration(ClassDeclaration node) {
|
2015-05-08 15:03:08 -07:00
|
|
|
if (!node.metadata.any((a) => _annotationMatcher.hasMatch(a, assetId))) {
|
|
|
|
return null;
|
|
|
|
}
|
2015-02-27 14:42:21 -08:00
|
|
|
|
2015-05-08 15:03:08 -07:00
|
|
|
var ctor = _getCtor(node);
|
2015-02-27 14:42:21 -08:00
|
|
|
|
2015-05-08 15:03:08 -07:00
|
|
|
if (!_foundNgDirectives) {
|
|
|
|
// The receiver for cascaded calls.
|
|
|
|
writer.print(REFLECTOR_VAR_NAME);
|
|
|
|
_foundNgDirectives = true;
|
|
|
|
}
|
|
|
|
writer.print('..registerType(');
|
|
|
|
node.name.accept(this);
|
|
|
|
writer.print(''', {'factory': ''');
|
|
|
|
if (ctor == null) {
|
|
|
|
_generateEmptyFactory(node.name.toString());
|
|
|
|
} else {
|
|
|
|
ctor.accept(_factoryVisitor);
|
2015-02-27 14:42:21 -08:00
|
|
|
}
|
2015-05-08 15:03:08 -07:00
|
|
|
writer.print(''', 'parameters': ''');
|
|
|
|
if (ctor == null) {
|
|
|
|
_generateEmptyParams();
|
|
|
|
} else {
|
|
|
|
ctor.accept(_paramsVisitor);
|
|
|
|
}
|
|
|
|
writer.print(''', 'annotations': ''');
|
|
|
|
node.accept(_metaVisitor);
|
|
|
|
writer.print('})');
|
2015-04-13 16:54:09 -07:00
|
|
|
return null;
|
2015-02-27 14:42:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
Object _nodeToSource(AstNode node) {
|
|
|
|
if (node == null) return null;
|
|
|
|
return node.accept(_copyVisitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2015-03-19 14:10:28 -07:00
|
|
|
Object visitLibraryDirective(LibraryDirective node) {
|
|
|
|
if (node != null && node.name != null) {
|
|
|
|
writer.print('library ');
|
|
|
|
_nodeToSource(node.name);
|
|
|
|
writer.print('$DEPS_EXTENSION;');
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
2015-02-27 14:42:21 -08:00
|
|
|
|
|
|
|
@override
|
|
|
|
Object visitPartOfDirective(PartOfDirective node) {
|
|
|
|
// TODO(kegluneq): Consider importing [node.libraryName].
|
2015-05-08 15:03:08 -07:00
|
|
|
logger.warning('[${assetId}]: '
|
2015-02-27 14:42:21 -08:00
|
|
|
'Found `part of` directive while generating ${DEPS_EXTENSION} file, '
|
|
|
|
'Transform may fail due to missing imports in generated file.');
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Object visitPrefixedIdentifier(PrefixedIdentifier node) =>
|
|
|
|
_nodeToSource(node);
|
|
|
|
|
|
|
|
@override
|
|
|
|
Object visitSimpleIdentifier(SimpleIdentifier node) => _nodeToSource(node);
|
|
|
|
}
|