2015-05-20 09:48:15 -07:00
|
|
|
import {Binding, resolveForwardRef, Injectable} from 'angular2/di';
|
|
|
|
import {
|
|
|
|
Type,
|
|
|
|
isBlank,
|
2015-06-25 13:19:51 -07:00
|
|
|
isType,
|
2015-05-20 09:48:15 -07:00
|
|
|
isPresent,
|
|
|
|
BaseException,
|
|
|
|
normalizeBlank,
|
2015-06-11 19:32:55 +02:00
|
|
|
stringify,
|
|
|
|
isArray,
|
|
|
|
isPromise
|
2015-05-20 09:48:15 -07:00
|
|
|
} from 'angular2/src/facade/lang';
|
|
|
|
import {Promise, PromiseWrapper} from 'angular2/src/facade/async';
|
|
|
|
import {List, ListWrapper, Map, MapWrapper} from 'angular2/src/facade/collection';
|
|
|
|
|
|
|
|
import {DirectiveResolver} from './directive_resolver';
|
|
|
|
|
2015-06-24 13:46:39 -07:00
|
|
|
import {AppProtoView, AppProtoViewMergeMapping} from './view';
|
2015-05-20 09:48:15 -07:00
|
|
|
import {ProtoViewRef} from './view_ref';
|
|
|
|
import {DirectiveBinding} from './element_injector';
|
2015-06-24 10:54:04 +02:00
|
|
|
import {ViewResolver} from './view_resolver';
|
2015-05-20 09:48:15 -07:00
|
|
|
import {View} from '../annotations_impl/view';
|
|
|
|
import {ComponentUrlMapper} from './component_url_mapper';
|
|
|
|
import {ProtoViewFactory} from './proto_view_factory';
|
|
|
|
import {UrlResolver} from 'angular2/src/services/url_resolver';
|
2015-06-23 11:28:06 +02:00
|
|
|
import {AppRootUrl} from 'angular2/src/services/app_root_url';
|
2015-05-20 09:48:15 -07:00
|
|
|
|
|
|
|
import * as renderApi from 'angular2/src/render/api';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cache that stores the AppProtoView of the template of a component.
|
|
|
|
* Used to prevent duplicate work and resolve cyclic dependencies.
|
|
|
|
*/
|
|
|
|
@Injectable()
|
|
|
|
export class CompilerCache {
|
2015-06-17 16:21:40 -07:00
|
|
|
_cache: Map<Type, AppProtoView> = new Map();
|
|
|
|
_hostCache: Map<Type, AppProtoView> = new Map();
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-06-17 16:21:40 -07:00
|
|
|
set(component: Type, protoView: AppProtoView): void { this._cache.set(component, protoView); }
|
2015-05-20 09:48:15 -07:00
|
|
|
|
|
|
|
get(component: Type): AppProtoView {
|
2015-06-17 16:21:40 -07:00
|
|
|
var result = this._cache.get(component);
|
2015-05-20 09:48:15 -07:00
|
|
|
return normalizeBlank(result);
|
|
|
|
}
|
|
|
|
|
2015-06-16 09:45:03 -07:00
|
|
|
setHost(component: Type, protoView: AppProtoView): void {
|
2015-06-17 16:21:40 -07:00
|
|
|
this._hostCache.set(component, protoView);
|
2015-06-16 09:45:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
getHost(component: Type): AppProtoView {
|
2015-06-17 16:21:40 -07:00
|
|
|
var result = this._hostCache.get(component);
|
2015-06-16 09:45:03 -07:00
|
|
|
return normalizeBlank(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
clear(): void {
|
2015-06-18 09:59:51 -07:00
|
|
|
this._cache.clear();
|
|
|
|
this._hostCache.clear();
|
2015-06-16 09:45:03 -07:00
|
|
|
}
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-07-07 08:15:58 +02:00
|
|
|
*
|
|
|
|
* ## URL Resolution
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* var appRootUrl: AppRootUrl = ...;
|
|
|
|
* var componentUrlMapper: ComponentUrlMapper = ...;
|
|
|
|
* var urlResolver: UrlResolver = ...;
|
|
|
|
*
|
|
|
|
* var componentType: Type = ...;
|
|
|
|
* var componentAnnotation: ComponentAnnotation = ...;
|
|
|
|
* var viewAnnotation: ViewAnnotation = ...;
|
|
|
|
*
|
|
|
|
* // Resolving a URL
|
|
|
|
*
|
|
|
|
* var url = viewAnnotation.templateUrl;
|
|
|
|
* var componentUrl = componentUrlMapper.getUrl(componentType);
|
|
|
|
* var componentResolvedUrl = urlResolver.resolve(appRootUrl.value, componentUrl);
|
|
|
|
* var templateResolvedUrl = urlResolver.resolve(componetResolvedUrl, url);
|
|
|
|
* ```
|
2015-05-20 09:48:15 -07:00
|
|
|
*/
|
|
|
|
@Injectable()
|
|
|
|
export class Compiler {
|
|
|
|
private _reader: DirectiveResolver;
|
|
|
|
private _compilerCache: CompilerCache;
|
|
|
|
private _compiling: Map<Type, Promise<AppProtoView>>;
|
2015-06-24 10:54:04 +02:00
|
|
|
private _viewResolver: ViewResolver;
|
2015-05-20 09:48:15 -07:00
|
|
|
private _componentUrlMapper: ComponentUrlMapper;
|
|
|
|
private _urlResolver: UrlResolver;
|
|
|
|
private _appUrl: string;
|
|
|
|
private _render: renderApi.RenderCompiler;
|
|
|
|
private _protoViewFactory: ProtoViewFactory;
|
2015-06-24 13:46:39 -07:00
|
|
|
private _unmergedCyclicEmbeddedProtoViews: RecursiveEmbeddedProtoView[] = [];
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-07-07 08:15:58 +02:00
|
|
|
/**
|
|
|
|
* @private
|
|
|
|
*/
|
2015-06-24 10:54:04 +02:00
|
|
|
constructor(reader: DirectiveResolver, cache: CompilerCache, viewResolver: ViewResolver,
|
2015-05-20 09:48:15 -07:00
|
|
|
componentUrlMapper: ComponentUrlMapper, urlResolver: UrlResolver,
|
2015-06-23 11:28:06 +02:00
|
|
|
render: renderApi.RenderCompiler, protoViewFactory: ProtoViewFactory,
|
|
|
|
appUrl: AppRootUrl) {
|
2015-05-20 09:48:15 -07:00
|
|
|
this._reader = reader;
|
|
|
|
this._compilerCache = cache;
|
2015-06-17 16:21:40 -07:00
|
|
|
this._compiling = new Map();
|
2015-06-24 10:54:04 +02:00
|
|
|
this._viewResolver = viewResolver;
|
2015-05-20 09:48:15 -07:00
|
|
|
this._componentUrlMapper = componentUrlMapper;
|
|
|
|
this._urlResolver = urlResolver;
|
2015-06-23 11:28:06 +02:00
|
|
|
this._appUrl = appUrl.value;
|
2015-05-20 09:48:15 -07:00
|
|
|
this._render = render;
|
|
|
|
this._protoViewFactory = protoViewFactory;
|
|
|
|
}
|
|
|
|
|
|
|
|
private _bindDirective(directiveTypeOrBinding): DirectiveBinding {
|
|
|
|
if (directiveTypeOrBinding instanceof DirectiveBinding) {
|
|
|
|
return directiveTypeOrBinding;
|
|
|
|
} else if (directiveTypeOrBinding instanceof Binding) {
|
|
|
|
let annotation = this._reader.resolve(directiveTypeOrBinding.token);
|
|
|
|
return DirectiveBinding.createFromBinding(directiveTypeOrBinding, annotation);
|
|
|
|
} else {
|
|
|
|
let annotation = this._reader.resolve(directiveTypeOrBinding);
|
|
|
|
return DirectiveBinding.createFromType(directiveTypeOrBinding, annotation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a hostView as if the compiler encountered <hostcmp></hostcmp>.
|
|
|
|
// Used for bootstrapping.
|
|
|
|
compileInHost(componentTypeOrBinding: Type | Binding): Promise<ProtoViewRef> {
|
2015-06-25 13:19:51 -07:00
|
|
|
var componentType = isType(componentTypeOrBinding) ? componentTypeOrBinding :
|
|
|
|
(<Binding>componentTypeOrBinding).token;
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-06-25 13:19:51 -07:00
|
|
|
var hostAppProtoView = this._compilerCache.getHost(componentType);
|
2015-06-16 09:45:03 -07:00
|
|
|
var hostPvPromise;
|
|
|
|
if (isPresent(hostAppProtoView)) {
|
|
|
|
hostPvPromise = PromiseWrapper.resolve(hostAppProtoView);
|
|
|
|
} else {
|
2015-07-07 08:15:58 +02:00
|
|
|
var componentBinding: DirectiveBinding = this._bindDirective(componentTypeOrBinding);
|
2015-06-25 13:19:51 -07:00
|
|
|
Compiler._assertTypeIsComponent(componentBinding);
|
|
|
|
|
|
|
|
var directiveMetadata = componentBinding.metadata;
|
2015-06-24 13:46:39 -07:00
|
|
|
hostPvPromise =
|
|
|
|
this._render.compileHost(directiveMetadata)
|
|
|
|
.then((hostRenderPv) => {
|
|
|
|
var protoView = this._protoViewFactory.createAppProtoViews(
|
|
|
|
componentBinding, hostRenderPv, [componentBinding]);
|
|
|
|
this._compilerCache.setHost(componentType, protoView);
|
|
|
|
return this._compileNestedProtoViews(hostRenderPv, protoView, componentType);
|
|
|
|
});
|
2015-06-16 09:45:03 -07:00
|
|
|
}
|
2015-07-17 08:03:40 -07:00
|
|
|
return hostPvPromise.then(
|
|
|
|
hostAppProtoView => this._mergeCyclicEmbeddedProtoViews().then(_ => hostAppProtoView.ref));
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _compile(componentBinding: DirectiveBinding): Promise<AppProtoView>| AppProtoView {
|
|
|
|
var component = <Type>componentBinding.key.token;
|
|
|
|
var protoView = this._compilerCache.get(component);
|
|
|
|
if (isPresent(protoView)) {
|
|
|
|
// The component has already been compiled into an AppProtoView,
|
|
|
|
// returns a plain AppProtoView, not wrapped inside of a Promise.
|
|
|
|
// Needed for recursive components.
|
|
|
|
return protoView;
|
|
|
|
}
|
|
|
|
|
2015-06-24 13:46:39 -07:00
|
|
|
var resultPromise = this._compiling.get(component);
|
|
|
|
if (isPresent(resultPromise)) {
|
2015-05-20 09:48:15 -07:00
|
|
|
// The component is already being compiled, attach to the existing Promise
|
|
|
|
// instead of re-compiling the component.
|
|
|
|
// It happens when a template references a component multiple times.
|
2015-06-24 13:46:39 -07:00
|
|
|
return resultPromise;
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
2015-06-24 11:10:29 +02:00
|
|
|
var view = this._viewResolver.resolve(component);
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-06-24 11:10:29 +02:00
|
|
|
var directives = this._flattenDirectives(view);
|
2015-05-20 09:48:15 -07:00
|
|
|
|
|
|
|
for (var i = 0; i < directives.length; i++) {
|
|
|
|
if (!Compiler._isValidDirective(directives[i])) {
|
|
|
|
throw new BaseException(
|
|
|
|
`Unexpected directive value '${stringify(directives[i])}' on the View of component '${stringify(component)}'`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-16 20:16:08 +02:00
|
|
|
var boundDirectives = this._removeDuplicatedDirectives(
|
|
|
|
ListWrapper.map(directives, (directive) => this._bindDirective(directive)));
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-06-24 11:10:29 +02:00
|
|
|
var renderTemplate = this._buildRenderTemplate(component, view, boundDirectives);
|
2015-06-24 13:46:39 -07:00
|
|
|
resultPromise = this._render.compile(renderTemplate)
|
|
|
|
.then((renderPv) => {
|
|
|
|
var protoView = this._protoViewFactory.createAppProtoViews(
|
|
|
|
componentBinding, renderPv, boundDirectives);
|
|
|
|
// Populate the cache before compiling the nested components,
|
|
|
|
// so that components can reference themselves in their template.
|
|
|
|
this._compilerCache.set(component, protoView);
|
|
|
|
MapWrapper.delete(this._compiling, component);
|
|
|
|
|
|
|
|
return this._compileNestedProtoViews(renderPv, protoView, component);
|
|
|
|
});
|
|
|
|
this._compiling.set(component, resultPromise);
|
|
|
|
return resultPromise;
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
|
|
|
|
2015-06-16 20:16:08 +02:00
|
|
|
private _removeDuplicatedDirectives(directives: List<DirectiveBinding>): List<DirectiveBinding> {
|
|
|
|
var directivesMap: Map<number, DirectiveBinding> = new Map();
|
|
|
|
directives.forEach((dirBinding) => { directivesMap.set(dirBinding.key.id, dirBinding); });
|
|
|
|
return MapWrapper.values(directivesMap);
|
|
|
|
}
|
|
|
|
|
2015-06-24 13:46:39 -07:00
|
|
|
private _compileNestedProtoViews(renderProtoView: renderApi.ProtoViewDto,
|
|
|
|
appProtoView: AppProtoView,
|
|
|
|
componentType: Type): Promise<AppProtoView> {
|
2015-05-20 09:48:15 -07:00
|
|
|
var nestedPVPromises = [];
|
2015-06-24 13:46:39 -07:00
|
|
|
this._loopComponentElementBinders(appProtoView, (parentPv, elementBinder) => {
|
2015-05-20 09:48:15 -07:00
|
|
|
var nestedComponent = elementBinder.componentDirective;
|
2015-05-21 16:42:19 +02:00
|
|
|
var elementBinderDone =
|
|
|
|
(nestedPv: AppProtoView) => { elementBinder.nestedProtoView = nestedPv; };
|
2015-05-20 09:48:15 -07:00
|
|
|
var nestedCall = this._compile(nestedComponent);
|
2015-06-11 19:32:55 +02:00
|
|
|
if (isPromise(nestedCall)) {
|
2015-06-17 11:17:21 -07:00
|
|
|
nestedPVPromises.push((<Promise<AppProtoView>>nestedCall).then(elementBinderDone));
|
2015-06-16 09:45:03 -07:00
|
|
|
} else {
|
2015-05-21 16:42:19 +02:00
|
|
|
elementBinderDone(<AppProtoView>nestedCall);
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
2015-05-21 16:42:19 +02:00
|
|
|
});
|
2015-06-24 13:46:39 -07:00
|
|
|
return PromiseWrapper.all(nestedPVPromises)
|
|
|
|
.then((_) => {
|
|
|
|
var appProtoViewsToMergeInto = [];
|
|
|
|
var mergeRenderProtoViews = this._collectMergeRenderProtoViewsRecurse(
|
|
|
|
renderProtoView, appProtoView, appProtoViewsToMergeInto);
|
|
|
|
if (isBlank(mergeRenderProtoViews)) {
|
|
|
|
throw new BaseException(`Unconditional component cycle in ${stringify(componentType)}`);
|
|
|
|
}
|
|
|
|
return this._mergeProtoViews(appProtoViewsToMergeInto, mergeRenderProtoViews);
|
|
|
|
});
|
|
|
|
}
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-06-24 13:46:39 -07:00
|
|
|
private _mergeProtoViews(
|
|
|
|
appProtoViewsToMergeInto: AppProtoView[],
|
|
|
|
mergeRenderProtoViews:
|
|
|
|
List<renderApi.RenderProtoViewRef | List<any>>): Promise<AppProtoView> {
|
|
|
|
return this._render.mergeProtoViewsRecursively(mergeRenderProtoViews)
|
|
|
|
.then((mergeResults: List<renderApi.RenderProtoViewMergeMapping>) => {
|
|
|
|
// Note: We don't need to check for nulls here as we filtered them out before!
|
|
|
|
// (in RenderCompiler.mergeProtoViewsRecursively and
|
|
|
|
// _collectMergeRenderProtoViewsRecurse).
|
|
|
|
for (var i = 0; i < mergeResults.length; i++) {
|
|
|
|
appProtoViewsToMergeInto[i].mergeMapping =
|
|
|
|
new AppProtoViewMergeMapping(mergeResults[i]);
|
|
|
|
}
|
|
|
|
return appProtoViewsToMergeInto[0];
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private _loopComponentElementBinders(appProtoView: AppProtoView, callback: Function) {
|
|
|
|
appProtoView.elementBinders.forEach((elementBinder) => {
|
|
|
|
if (isPresent(elementBinder.componentDirective)) {
|
|
|
|
callback(appProtoView, elementBinder);
|
|
|
|
} else if (isPresent(elementBinder.nestedProtoView)) {
|
|
|
|
this._loopComponentElementBinders(elementBinder.nestedProtoView, callback);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private _collectMergeRenderProtoViewsRecurse(
|
|
|
|
renderProtoView: renderApi.ProtoViewDto, appProtoView: AppProtoView,
|
|
|
|
targetAppProtoViews: AppProtoView[]): List<renderApi.RenderProtoViewRef | List<any>> {
|
|
|
|
targetAppProtoViews.push(appProtoView);
|
|
|
|
var result = [renderProtoView.render];
|
|
|
|
for (var i = 0; i < appProtoView.elementBinders.length; i++) {
|
|
|
|
var binder = appProtoView.elementBinders[i];
|
|
|
|
if (binder.hasStaticComponent()) {
|
|
|
|
if (isBlank(binder.nestedProtoView.mergeMapping)) {
|
|
|
|
// cycle via an embedded ProtoView. store the AppProtoView and ProtoViewDto for later.
|
|
|
|
this._unmergedCyclicEmbeddedProtoViews.push(
|
|
|
|
new RecursiveEmbeddedProtoView(appProtoView, renderProtoView));
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
result.push(binder.nestedProtoView.mergeMapping.renderProtoViewRef);
|
|
|
|
} else if (binder.hasEmbeddedProtoView()) {
|
|
|
|
result.push(this._collectMergeRenderProtoViewsRecurse(
|
|
|
|
renderProtoView.elementBinders[i].nestedProtoView, binder.nestedProtoView,
|
|
|
|
targetAppProtoViews));
|
|
|
|
}
|
2015-05-21 16:42:19 +02:00
|
|
|
}
|
2015-06-24 13:46:39 -07:00
|
|
|
return result;
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
|
|
|
|
2015-06-24 13:46:39 -07:00
|
|
|
private _mergeCyclicEmbeddedProtoViews() {
|
|
|
|
var pvs = this._unmergedCyclicEmbeddedProtoViews;
|
|
|
|
this._unmergedCyclicEmbeddedProtoViews = [];
|
|
|
|
var promises = pvs.map(entry => {
|
|
|
|
var appProtoView = entry.appProtoView;
|
|
|
|
var mergeRenderProtoViews = [entry.renderProtoView.render];
|
|
|
|
appProtoView.elementBinders.forEach((binder) => {
|
|
|
|
if (binder.hasStaticComponent()) {
|
|
|
|
mergeRenderProtoViews.push(binder.nestedProtoView.mergeMapping.renderProtoViewRef);
|
|
|
|
} else if (binder.hasEmbeddedProtoView()) {
|
|
|
|
mergeRenderProtoViews.push(null);
|
2015-05-21 16:42:19 +02:00
|
|
|
}
|
|
|
|
});
|
2015-06-24 13:46:39 -07:00
|
|
|
return this._mergeProtoViews([appProtoView], mergeRenderProtoViews);
|
2015-05-20 09:48:15 -07:00
|
|
|
});
|
2015-06-24 13:46:39 -07:00
|
|
|
return PromiseWrapper.all(promises);
|
2015-05-21 16:42:19 +02:00
|
|
|
}
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-05-21 16:42:19 +02:00
|
|
|
private _buildRenderTemplate(component, view, directives): renderApi.ViewDefinition {
|
|
|
|
var componentUrl =
|
|
|
|
this._urlResolver.resolve(this._appUrl, this._componentUrlMapper.getUrl(component));
|
|
|
|
var templateAbsUrl = null;
|
2015-06-10 14:40:24 +02:00
|
|
|
var styleAbsUrls = null;
|
2015-05-21 16:42:19 +02:00
|
|
|
if (isPresent(view.templateUrl)) {
|
|
|
|
templateAbsUrl = this._urlResolver.resolve(componentUrl, view.templateUrl);
|
|
|
|
} else if (isPresent(view.template)) {
|
|
|
|
// Note: If we have an inline template, we also need to send
|
|
|
|
// the url for the component to the render so that it
|
|
|
|
// is able to resolve urls in stylesheets.
|
|
|
|
templateAbsUrl = componentUrl;
|
|
|
|
}
|
2015-06-10 14:40:24 +02:00
|
|
|
if (isPresent(view.styleUrls)) {
|
|
|
|
styleAbsUrls =
|
|
|
|
ListWrapper.map(view.styleUrls, url => this._urlResolver.resolve(componentUrl, url));
|
|
|
|
}
|
2015-05-21 16:42:19 +02:00
|
|
|
return new renderApi.ViewDefinition({
|
|
|
|
componentId: stringify(component),
|
2015-06-10 14:40:24 +02:00
|
|
|
templateAbsUrl: templateAbsUrl, template: view.template,
|
|
|
|
styleAbsUrls: styleAbsUrls,
|
|
|
|
styles: view.styles,
|
2015-05-21 16:42:19 +02:00
|
|
|
directives: ListWrapper.map(directives, directiveBinding => directiveBinding.metadata)
|
|
|
|
});
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
|
|
|
|
2015-05-21 16:42:19 +02:00
|
|
|
private _flattenDirectives(template: View): List<Type> {
|
|
|
|
if (isBlank(template.directives)) return [];
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-05-21 16:42:19 +02:00
|
|
|
var directives = [];
|
|
|
|
this._flattenList(template.directives, directives);
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-05-21 16:42:19 +02:00
|
|
|
return directives;
|
|
|
|
}
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-05-21 16:42:19 +02:00
|
|
|
private _flattenList(tree: List<any>, out: List<Type | Binding | List<any>>): void {
|
|
|
|
for (var i = 0; i < tree.length; i++) {
|
|
|
|
var item = resolveForwardRef(tree[i]);
|
2015-06-11 19:32:55 +02:00
|
|
|
if (isArray(item)) {
|
2015-05-21 16:42:19 +02:00
|
|
|
this._flattenList(item, out);
|
|
|
|
} else {
|
2015-06-17 11:17:21 -07:00
|
|
|
out.push(item);
|
2015-05-21 16:42:19 +02:00
|
|
|
}
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-21 16:42:19 +02:00
|
|
|
private static _isValidDirective(value: Type | Binding): boolean {
|
|
|
|
return isPresent(value) && (value instanceof Type || value instanceof Binding);
|
|
|
|
}
|
2015-05-20 09:48:15 -07:00
|
|
|
|
2015-05-21 16:42:19 +02:00
|
|
|
private static _assertTypeIsComponent(directiveBinding: DirectiveBinding): void {
|
|
|
|
if (directiveBinding.metadata.type !== renderApi.DirectiveMetadata.COMPONENT_TYPE) {
|
|
|
|
throw new BaseException(
|
|
|
|
`Could not load '${stringify(directiveBinding.key.token)}' because it is not a component.`);
|
|
|
|
}
|
2015-05-20 09:48:15 -07:00
|
|
|
}
|
|
|
|
}
|
2015-06-24 13:46:39 -07:00
|
|
|
|
|
|
|
class RecursiveEmbeddedProtoView {
|
|
|
|
constructor(public appProtoView: AppProtoView, public renderProtoView: renderApi.ProtoViewDto) {}
|
|
|
|
}
|