refactor(compiler): remove `Viewport` directives, use `Decorator` instead

BREAKING_CHANGE:
- The special type of `Viewport` directives is removed
  in favor of a more general `Decorator` directive
- `ViewContainerRef` now no more has a default `ProtoViewRef`
  but requires an explicit one when creating views.

Closes #1536
This commit is contained in:
Tobias Bosch 2015-04-29 15:07:55 -07:00
parent fb67e37339
commit 3aac2fefd7
35 changed files with 280 additions and 366 deletions

View File

@ -5,7 +5,7 @@ projected to DOM as well as which DOM events should invoke which methods on the
syntax which is core to Angular and allows for data-binding, event-binding, template-instantiation. syntax which is core to Angular and allows for data-binding, event-binding, template-instantiation.
The design of the template syntax has these properties: The design of the template syntax has these properties:
* All data-binding expressions are easily identifiable. (i.e. there is never an ambiguity whether the value should be * All data-binding expressions are easily identifiable. (i.e. there is never an ambiguity whether the value should be
interpreted as string literal or as an expression.) interpreted as string literal or as an expression.)
@ -372,8 +372,8 @@ Where:
inserted. The template can be defined implicitly with `template` attribute, which turns the current element into inserted. The template can be defined implicitly with `template` attribute, which turns the current element into
a template, or explicitly with `<template>` element. Explicit declaration is longer, but it allows for having a template, or explicitly with `<template>` element. Explicit declaration is longer, but it allows for having
templates which have more than one root DOM node. templates which have more than one root DOM node.
* `viewport` is required for templates. The Viewport directive is responsible for deciding when * `viewport` is required for templates. The directive is responsible for deciding when
and in which order should child views be inserted into this location. An Viewport directive usually has one or and in which order should child views be inserted into this location. Such a directive usually has one or
more bindings and can be represented as either `viewport-directive-bindings` or more bindings and can be represented as either `viewport-directive-bindings` or
`viewport-directive-microsyntax` on `template` element or attribute. See template microsyntax for more details. `viewport-directive-microsyntax` on `template` element or attribute. See template microsyntax for more details.
@ -387,7 +387,7 @@ Hello {{user}}!
</div> </div>
``` ```
In the above example the `if` Viewport determines whether the child view (an instance of the child template) should be In the above example the `if` directive determines whether the child view (an instance of the child template) should be
inserted into the root view. The `if` makes this decision based on if the `isAdministrator` binding is true. inserted into the root view. The `if` makes this decision based on if the `isAdministrator` binding is true.
The above example is in the short form, for better clarity let's rewrite it in the canonical form, which is functionally The above example is in the short form, for better clarity let's rewrite it in the canonical form, which is functionally
@ -402,8 +402,6 @@ Hello {{user}}!
</template> </template>
``` ```
NOTE: Only Viewport directives can be placed on the template element. (Decorators and Components are not allowed.)
### Template Microsyntax ### Template Microsyntax
@ -514,7 +512,7 @@ Where:
* `some-element` Any element which can generate DOM events (or has an angular directive which generates the event). * `some-element` Any element which can generate DOM events (or has an angular directive which generates the event).
* `some-event` (escaped with `()` or `on-`) is the name of the event `some-event`. In this case the * `some-event` (escaped with `()` or `on-`) is the name of the event `some-event`. In this case the
dash-case is converted into camel-case `someEvent`. dash-case is converted into camel-case `someEvent`.
* `statement` is a valid statement (as defined in section below). * `statement` is a valid statement (as defined in section below).
If the execution of the statement returns `false`, then `preventDefault`is applied on the DOM event. If the execution of the statement returns `false`, then `preventDefault`is applied on the DOM event.
By default, angular only listens to the element on the event, and ignores events which bubble. To listen to bubbled By default, angular only listens to the element on the event, and ignores events which bubble. To listen to bubbled

View File

@ -10,7 +10,6 @@ There are three different kinds of directives (described in more detail in later
1. *Decorators*: can be placed on any DOM element and can be combined with other directives. 1. *Decorators*: can be placed on any DOM element and can be combined with other directives.
2. *Components*: Components have an encapsulated view and can configure injectors. 2. *Components*: Components have an encapsulated view and can configure injectors.
3. *Viewport*: is responsible for adding or removing child views in a parent view. (i.e. for, if)
@ -163,21 +162,19 @@ Example of usage:
## Viewport ## Directives that use a ViewContainer
Viewport is a directive which can control instantiation of child views which are then inserted into the DOM. (Examples are `if` and `for`.) Directives that use a ViewContainer can control instantiation of child views which are then inserted into the DOM. (Examples are `if` and `for`.)
* Viewports can only be placed on `<template>` elements (or the short hand version which uses `<element template>` attribute.) * Every `template` element creates a `ProtoView` which can be used to create Views via the ViewContainer.
* Only one viewport can be present per DOM template element. * The child views show up as siblings of the directive in the DOM.
* The viewport is created over the `template` element. This is known as the `ViewContainerRef`.
* Viewport can insert child views into the `ViewContainerRef`. The child views show up as siblings of the `Viewport` in the DOM.
>> TODO(misko): Relationship with Injection >> TODO(misko): Relationship with Injection
>> TODO(misko): Instantiator can not be injected into child Views >> TODO(misko): Instantiator can not be injected into child Views
``` ```
@Viewport({ @Directive({
selector: '[if]', selector: '[if]',
properties: { properties: {
'condition': 'if' 'condition': 'if'
@ -185,17 +182,19 @@ Viewport is a directive which can control instantiation of child views which are
}) })
export class If { export class If {
viewContainer: ViewContainerRef; viewContainer: ViewContainerRef;
protoViewRef: ProtoViewRef;
view: View; view: View;
constructor(viewContainer: ViewContainerRef) { constructor(viewContainer: ViewContainerRef, protoViewRef: ProtoViewRef) {
this.viewContainer = viewContainer; this.viewContainer = viewContainer;
this.protoViewRef = protoViewRef;
this.view = null; this.view = null;
} }
set condition(value) { set condition(value) {
if (value) { if (value) {
if (this.view === null) { if (this.view === null) {
this.view = this.viewContainer.create(); this.view = this.viewContainer.create(protoViewRef);
} }
} else { } else {
if (this.view !== null) { if (this.view !== null) {
@ -342,7 +341,7 @@ Shadow DOM provides an encapsulation for components, so as a general rule it doe
}) })
class Kid { class Kid {
constructor( constructor(
@Parent() dad:Dad, @Parent() dad:Dad,
@Optional() grandpa:Grandpa @Optional() grandpa:Grandpa
) { ) {
this.name = 'Billy'; this.name = 'Billy';
@ -365,7 +364,7 @@ class Dad {
this.dad = dad.name; this.dad = dad.name;
console.log(dad) console.log(dad)
} }
} }
@Component({ @Component({
selector: '[grandpa]', selector: '[grandpa]',
@ -379,17 +378,17 @@ class Grandpa {
constructor() { constructor() {
this.name = 'Joe'; this.name = 'Joe';
} }
} }
``` ```
Assume the following DOM structure for `grandpa.html`: The Dad has access to the Grandpa. Assume the following DOM structure for `grandpa.html`: The Dad has access to the Grandpa.
``` ```
Name: {{name}}: <br> Children: <div dad></div> Name: {{name}}: <br> Children: <div dad></div>
``` ```
Assume the following DOM structure for `dad.html`: Here the rendered Kid will also have access to Grandpa. Assume the following DOM structure for `dad.html`: Here the rendered Kid will also have access to Grandpa.
``` ```
Name: {{name}}: <br> Dad: {{dad}} <br> Children: <div kid></div> Name: {{name}}: <br> Dad: {{dad}} <br> Children: <div kid></div>
``` ```
## Further Reading ## Further Reading

View File

@ -4,8 +4,8 @@
This document explains the concept of a View. This document explains the concept of a View.
A View is a core primitive used by angular to render the DOM tree. A View is a core primitive used by angular to render the DOM tree.
A ViewPort is location in a View which can accept child Views. A ViewContainer is location in a View which can accept child Views.
Every ViewPort has an associated ViewContainerRef than can contain any number of child Views. Every ViewContainer has an associated ViewContainerRef than can contain any number of child Views.
Views form a tree structure which mimics the DOM tree. Views form a tree structure which mimics the DOM tree.
* View is a core rendering construct. A running application is just a collection of Views which are * View is a core rendering construct. A running application is just a collection of Views which are
@ -15,7 +15,7 @@ Views form a tree structure which mimics the DOM tree.
* Views represent a running instance of a DOM View. This implies that while elements in a View * Views represent a running instance of a DOM View. This implies that while elements in a View
can change properties, they can not change structurally. (Structural changes such as, adding or can change properties, they can not change structurally. (Structural changes such as, adding or
removing elements requires adding or removing child Views into ViewContainers). removing elements requires adding or removing child Views into ViewContainers).
* View can have zero or more ViewPorts. A ViewPort is a marker in the DOM which allows * View can have zero or more ViewContainers. A ViewContainer is a marker in the DOM which allows
the insertion of child Views. the insertion of child Views.
* Views are created from a ProtoView. A ProtoView is a compiled DOM View which is efficient at * Views are created from a ProtoView. A ProtoView is a compiled DOM View which is efficient at
creating Views. creating Views.
@ -88,7 +88,7 @@ Note:
## Composed View ## Composed View
An important part of an application is to be able to change the DOM structure to render data for the An important part of an application is to be able to change the DOM structure to render data for the
user. In Angular this is done by inserting child views into the ViewPort. user. In Angular this is done by inserting child views into the ViewContainer.
Let's start with a View such as: Let's start with a View such as:
@ -108,7 +108,7 @@ and
``` ```
<ul> | protoViewA(someContext) <ul> | protoViewA(someContext)
<template></template> | protoViewA(someContext): new ProtoViewPort(protoViewB) <template></template> | protoViewA(someContext): protoViewB
</ul> | protoViewA(someContext) </ul> | protoViewA(someContext)
``` ```
@ -119,7 +119,7 @@ The next step is to compose these two ProtoViews into an actual view which is re
``` ```
<ul> | viewA(someContext) <ul> | viewA(someContext)
<template></template> | viewA(someContext): new Foreach(new ViewPort(protoViewB)) <template></template> | viewA(someContext): new Foreach(new ViewContainer(protoViewB))
</ul> | viewA(someContext) </ul> | viewA(someContext)
``` ```
@ -128,11 +128,11 @@ has a reference to `protoViewA`).
*Step3:* As the `Foreach` directive unrolls it asks the `ViewContainerRef` to instantiate `protoViewB` and insert *Step3:* As the `Foreach` directive unrolls it asks the `ViewContainerRef` to instantiate `protoViewB` and insert
it after the `ViewPort` anchor. This is repeated for each `person` in `people`. Notice that it after the `ViewContainer` anchor. This is repeated for each `person` in `people`. Notice that
``` ```
<ul> | viewA(someContext) <ul> | viewA(someContext)
<template></template> | viewA(someContext): new Foreach(new ViewPort(protoViewB)) <template></template> | viewA(someContext): new Foreach(new ViewContainer(protoViewB))
<li>{{person}}</li> | viewB0(locals0(someContext)) <li>{{person}}</li> | viewB0(locals0(someContext))
<li>{{person}}</li> | viewB1(locals0(someContext)) <li>{{person}}</li> | viewB1(locals0(someContext))
</ul> | viewA(someContext) </ul> | viewA(someContext)
@ -145,13 +145,13 @@ delegate any unknown references to the parent context.
``` ```
<ul> | viewA <ul> | viewA
<template></template> | viewA: new Foreach(new ViewPort(protoViewB)) <template></template> | viewA: new Foreach(new ViewContainer(protoViewB))
<li>Alice</li> | viewB0 <li>Alice</li> | viewB0
<li>Bob</li> | viewB1 <li>Bob</li> | viewB1
</ul> | viewA </ul> | viewA
``` ```
Each View can have zero or more ViewPorts. By inserting and removing child Views to and from the Each View can have zero or more ViewContainers. By inserting and removing child Views to and from the
ViewContainers, the application can mutate the DOM structure to any desirable state. A View may contain ViewContainers, the application can mutate the DOM structure to any desirable state. A View may contain
individual nodes or a complex DOM structure. The insertion points for the child Views, known as individual nodes or a complex DOM structure. The insertion points for the child Views, known as
ViewContainers, contain a DOM element which acts as an anchor. The anchor is either a `template` or ViewContainers, contain a DOM element which acts as an anchor. The anchor is either a `template` or
@ -161,7 +161,7 @@ inserted.
## Component Views ## Component Views
A View can also contain Components. Components contain Shadow DOM for encapsulating their internal A View can also contain Components. Components contain Shadow DOM for encapsulating their internal
rendering state. Unlike ViewPorts which can contain zero or more Views, the Component always contains rendering state. Unlike ViewContainers which can contain zero or more Views, the Component always contains
exactly one Shadow View. exactly one Shadow View.
``` ```

View File

@ -8,8 +8,7 @@ import {DEFAULT} from 'angular2/change_detection';
/** /**
* Directives allow you to attach behavior to elements in the DOM. * Directives allow you to attach behavior to elements in the DOM.
* *
* Directive is an abstract concept, instead use concrete directives: {@link Component}, {@link DynamicComponent}, {@link Decorator} * Directive is an abstract concept, instead use concrete directives: {@link Component}, {@link DynamicComponent}, {@link Decorator}.
* or {@link Viewport}.
* *
* A directive consists of a single directive annotation and a controller class. When the directive's `selector` matches * A directive consists of a single directive annotation and a controller class. When the directive's `selector` matches
* elements in the DOM, the following steps occur: * elements in the DOM, the following steps occur:
@ -55,7 +54,7 @@ import {DEFAULT} from 'angular2/change_detection';
* *
* To inject element-specific special objects, declare the constructor parameter as: * To inject element-specific special objects, declare the constructor parameter as:
* - `element: ElementRef` to obtain a reference to logical element in the view. * - `element: ElementRef` to obtain a reference to logical element in the view.
* - `viewContainer: ViewContainerRef` to control child template instantiation, for {@link Viewport} directives only * - `viewContainer: ViewContainerRef` to control child template instantiation, for {@link Decorator} directives only
* - `bindingPropagation: BindingPropagation` to control change detection in a more granular way. * - `bindingPropagation: BindingPropagation` to control change detection in a more granular way.
* *
* ## Example * ## Example
@ -188,8 +187,8 @@ import {DEFAULT} from 'angular2/change_detection';
* *
* A directive can also query for other child directives. Since parent directives are instantiated before child * A directive can also query for other child directives. Since parent directives are instantiated before child
* directives, a directive can't simply inject the list of child directives. Instead, the directive * directives, a directive can't simply inject the list of child directives. Instead, the directive
* injects a {@link QueryList}, which updates its contents as children are added, removed, or moved by any * injects a {@link QueryList}, which updates its contents as children are added, removed, or moved by a directive
* {@link Viewport} directive such as a `for`, an `if`, or a `switch`. * that uses a {@link ViewContainerRef} such as a `for`, an `if`, or a `switch`.
* *
* ``` * ```
* @Decorator({ selector: '[my-directive]' }) * @Decorator({ selector: '[my-directive]' })
@ -783,6 +782,98 @@ export class DynamicComponent extends Directive {
* <div tooltip="some text here"></div> * <div tooltip="some text here"></div>
* ``` * ```
* *
* Decorators can also control the instantiation, destruction, and positioning of inline template elements:
*
* A directive uses a {@link ViewContainerRef} to instantiate, insert, move, and destroy views at runtime.
* The {@link ViewContainerRef} is created as a result of `<template>` element, and represents a location in the current view
* where these actions are performed.
*
* Views are always created as children of the current {@link View}, and as siblings of the `<template>` element. Thus a
* directive in a child view cannot inject the directive that created it.
*
* Since directives that create views via ViewContainers are common in Angular, and using the full `<template>` element syntax is wordy, Angular
* also supports a shorthand notation: `<li *foo="bar">` and `<li template="foo: bar">` are equivalent.
*
* Thus,
*
* ```
* <ul>
* <li *foo="bar" title="text"></li>
* </ul>
* ```
*
* Expands in use to:
*
* ```
* <ul>
* <template [foo]="bar">
* <li title="text"></li>
* </template>
* </ul>
* ```
*
* Notice that although the shorthand places `*foo="bar"` within the `<li>` element, the binding for the directive
* controller is correctly instantiated on the `<template>` element rather than the `<li>` element.
*
*
* ## Example
*
* Let's suppose we want to implement the `unless` behavior, to conditionally include a template.
*
* Here is a simple directive that triggers on an `unless` selector:
*
* ```
* @Directive({
* selector: '[unless]',
* properties: {
* 'unless': 'unless'
* }
* })
* export class Unless {
* viewContainer: ViewContainerRef;
* protoViewRef: ProtoViewRef;
* prevCondition: boolean;
*
* constructor(viewContainer: ViewContainerRef, protoViewRef: ProtoViewRef) {
* this.viewContainer = viewContainer;
* this.protoViewRef = protoViewRef;
* this.prevCondition = null;
* }
*
* set unless(newCondition) {
* if (newCondition && (isBlank(this.prevCondition) || !this.prevCondition)) {
* this.prevCondition = true;
* this.viewContainer.clear();
* } else if (!newCondition && (isBlank(this.prevCondition) || this.prevCondition)) {
* this.prevCondition = false;
* this.viewContainer.create(this.protoViewRef);
* }
* }
* }
* ```
*
* We can then use this `unless` selector in a template:
* ```
* <ul>
* <li *unless="expr"></li>
* </ul>
* ```
*
* Once the directive instantiates the child view, the shorthand notation for the template expands and the result is:
*
* ```
* <ul>
* <template [unless]="exp">
* <li></li>
* </template>
* <li></li>
* </ul>
* ```
*
* Note also that although the `<li></li>` template still exists inside the `<template></template>`, the instantiated
* view occurs on the second `<li></li>` which is a sibling to the `<template>` element.
*
*
* @exportedAs angular2/annotations * @exportedAs angular2/annotations
*/ */
export class Decorator extends Directive { export class Decorator extends Directive {
@ -824,127 +915,6 @@ export class Decorator extends Directive {
} }
} }
/**
* Directive that controls the instantiation, destruction, and positioning of inline template elements.
*
* A viewport directive uses a {@link ViewContainerRef} to instantiate, insert, move, and destroy views at runtime.
* The {@link ViewContainerRef} is created as a result of `<template>` element, and represents a location in the current view
* where these actions are performed.
*
* Views are always created as children of the current {@link View}, and as siblings of the `<template>` element. Thus a
* directive in a child view cannot inject the viewport directive that created it.
*
* Since viewport directives are common in Angular, and using the full `<template>` element syntax is wordy, Angular
* also supports a shorthand notation: `<li *foo="bar">` and `<li template="foo: bar">` are equivalent.
*
* Thus,
*
* ```
* <ul>
* <li *foo="bar" title="text"></li>
* </ul>
* ```
*
* Expands in use to:
*
* ```
* <ul>
* <template [foo]="bar">
* <li title="text"></li>
* </template>
* </ul>
* ```
*
* Notice that although the shorthand places `*foo="bar"` within the `<li>` element, the binding for the `Viewport`
* controller is correctly instantiated on the `<template>` element rather than the `<li>` element.
*
*
* ## Example
*
* Let's suppose we want to implement the `unless` behavior, to conditionally include a template.
*
* Here is a simple viewport directive that triggers on an `unless` selector:
*
* ```
* @Viewport({
* selector: '[unless]',
* properties: {
* 'unless': 'unless'
* }
* })
* export class Unless {
* viewContainer: ViewContainerRef;
* prevCondition: boolean;
*
* constructor(viewContainer: ViewContainerRef) {
* this.viewContainer = viewContainer;
* this.prevCondition = null;
* }
*
* set unless(newCondition) {
* if (newCondition && (isBlank(this.prevCondition) || !this.prevCondition)) {
* this.prevCondition = true;
* this.viewContainer.clear();
* } else if (!newCondition && (isBlank(this.prevCondition) || this.prevCondition)) {
* this.prevCondition = false;
* this.viewContainer.create();
* }
* }
* }
* ```
*
* We can then use this `unless` selector in a template:
* ```
* <ul>
* <li *unless="expr"></li>
* </ul>
* ```
*
* Once the viewport instantiates the child view, the shorthand notation for the template expands and the result is:
*
* ```
* <ul>
* <template [unless]="exp">
* <li></li>
* </template>
* <li></li>
* </ul>
* ```
*
* Note also that although the `<li></li>` template still exists inside the `<template></template>`, the instantiated
* view occurs on the second `<li></li>` which is a sibling to the `<template>` element.
*
*
* @exportedAs angular2/annotations
*/
export class Viewport extends Directive {
@CONST()
constructor({
selector,
properties,
events,
hostListeners,
hostProperties,
lifecycle
}:{
selector:string,
properties:any,
hostListeners:any,
hostProperties:any,
events:List,
lifecycle:List
}={})
{
super({
selector: selector,
properties: properties,
events: events,
hostListeners: hostListeners,
hostProperties: hostProperties,
lifecycle: lifecycle
});
}
}
//TODO(misko): turn into LifecycleEvent class once we switch to TypeScript; //TODO(misko): turn into LifecycleEvent class once we switch to TypeScript;

View File

@ -4,7 +4,7 @@ import {Promise, PromiseWrapper} from 'angular2/src/facade/async';
import {List, ListWrapper, Map, MapWrapper} from 'angular2/src/facade/collection'; import {List, ListWrapper, Map, MapWrapper} from 'angular2/src/facade/collection';
import {DirectiveMetadataReader} from './directive_metadata_reader'; import {DirectiveMetadataReader} from './directive_metadata_reader';
import {Component, Viewport, DynamicComponent, Decorator} from '../annotations_impl/annotations'; import {Component, DynamicComponent, Decorator} from '../annotations_impl/annotations';
import {AppProtoView} from './view'; import {AppProtoView} from './view';
import {ProtoViewRef} from './view_ref'; import {ProtoViewRef} from './view_ref';
import {DirectiveBinding} from './element_injector'; import {DirectiveBinding} from './element_injector';
@ -233,8 +233,6 @@ export class Compiler {
var compileChildren = true; var compileChildren = true;
if ((ann instanceof Component) || (ann instanceof DynamicComponent)) { if ((ann instanceof Component) || (ann instanceof DynamicComponent)) {
renderType = renderApi.DirectiveMetadata.COMPONENT_TYPE; renderType = renderApi.DirectiveMetadata.COMPONENT_TYPE;
} else if (ann instanceof Viewport) {
renderType = renderApi.DirectiveMetadata.VIEWPORT_TYPE;
} else if (ann instanceof Decorator) { } else if (ann instanceof Decorator) {
renderType = renderApi.DirectiveMetadata.DECORATOR_TYPE; renderType = renderApi.DirectiveMetadata.DECORATOR_TYPE;
compileChildren = ann.compileChildren; compileChildren = ann.compileChildren;

View File

@ -87,7 +87,7 @@ export class DynamicComponentLoader {
var binding = this._getBinding(typeOrBinding); var binding = this._getBinding(typeOrBinding);
return this._compiler.compileInHost(binding).then(hostProtoViewRef => { return this._compiler.compileInHost(binding).then(hostProtoViewRef => {
var viewContainer = this._viewManager.getViewContainer(location); var viewContainer = this._viewManager.getViewContainer(location);
var hostViewRef = viewContainer.create(-1, hostProtoViewRef, injector); var hostViewRef = viewContainer.create(hostProtoViewRef, viewContainer.length, injector);
var newLocation = new ElementRef(hostViewRef, 0); var newLocation = new ElementRef(hostViewRef, 0);
var component = this._viewManager.getComponent(newLocation); var component = this._viewManager.getComponent(newLocation);

View File

@ -7,7 +7,6 @@ import * as viewModule from './view';
export class ElementBinder { export class ElementBinder {
protoElementInjector:eiModule.ProtoElementInjector; protoElementInjector:eiModule.ProtoElementInjector;
componentDirective:DirectiveBinding; componentDirective:DirectiveBinding;
viewportDirective:DirectiveBinding;
nestedProtoView: viewModule.AppProtoView; nestedProtoView: viewModule.AppProtoView;
hostListeners:StringMap; hostListeners:StringMap;
parent:ElementBinder; parent:ElementBinder;
@ -15,15 +14,13 @@ export class ElementBinder {
distanceToParent:int; distanceToParent:int;
constructor( constructor(
index:int, parent:ElementBinder, distanceToParent: int, index:int, parent:ElementBinder, distanceToParent: int,
protoElementInjector: eiModule.ProtoElementInjector, componentDirective:DirectiveBinding, protoElementInjector: eiModule.ProtoElementInjector, componentDirective:DirectiveBinding) {
viewportDirective:DirectiveBinding) {
if (isBlank(index)) { if (isBlank(index)) {
throw new BaseException('null index not allowed.'); throw new BaseException('null index not allowed.');
} }
this.protoElementInjector = protoElementInjector; this.protoElementInjector = protoElementInjector;
this.componentDirective = componentDirective; this.componentDirective = componentDirective;
this.viewportDirective = viewportDirective;
this.parent = parent; this.parent = parent;
this.index = index; this.index = index;
this.distanceToParent = distanceToParent; this.distanceToParent = distanceToParent;
@ -40,4 +37,8 @@ export class ElementBinder {
hasDynamicComponent() { hasDynamicComponent() {
return isPresent(this.componentDirective) && isBlank(this.nestedProtoView); return isPresent(this.componentDirective) && isBlank(this.nestedProtoView);
} }
hasEmbeddedProtoView() {
return !isPresent(this.componentDirective) && isPresent(this.nestedProtoView);
}
} }

View File

@ -638,7 +638,7 @@ export class ElementInjector extends TreeNode {
} }
getViewContainerRef() { getViewContainerRef() {
return new ViewContainerRef(this._preBuiltObjects.viewManager, this.getElementRef(), new ProtoViewRef(this._preBuiltObjects.protoView)); return new ViewContainerRef(this._preBuiltObjects.viewManager, this.getElementRef());
} }
getDynamicallyLoadedComponent() { getDynamicallyLoadedComponent() {
@ -719,6 +719,9 @@ export class ElementInjector extends TreeNode {
return this.getViewContainerRef(); return this.getViewContainerRef();
} }
if (dep.key.id === StaticKeys.instance().protoViewId) { if (dep.key.id === StaticKeys.instance().protoViewId) {
if (isBlank(this._preBuiltObjects.protoView)) {
throw new NoBindingError(dep.key);
}
return new ProtoViewRef(this._preBuiltObjects.protoView); return new ProtoViewRef(this._preBuiltObjects.protoView);
} }
return this._getByKey(dep.key, dep.depth, dep.optional, requestor); return this._getByKey(dep.key, dep.depth, dep.optional, requestor);

View File

@ -4,7 +4,7 @@ import {isPresent, isBlank} from 'angular2/src/facade/lang';
import {reflector} from 'angular2/src/reflection/reflection'; import {reflector} from 'angular2/src/reflection/reflection';
import {ChangeDetection, DirectiveIndex} from 'angular2/change_detection'; import {ChangeDetection, DirectiveIndex} from 'angular2/change_detection';
import {Component, Viewport, DynamicComponent} from '../annotations_impl/annotations'; import {Component, DynamicComponent} from '../annotations_impl/annotations';
import * as renderApi from 'angular2/src/render/api'; import * as renderApi from 'angular2/src/render/api';
import {AppProtoView} from './view'; import {AppProtoView} from './view';
@ -81,8 +81,7 @@ export class ProtoViewFactory {
isPresent(sortedDirectives.componentDirective), parentPeiWithDistance.distance isPresent(sortedDirectives.componentDirective), parentPeiWithDistance.distance
); );
protoElementInjector.attributes = renderElementBinder.readAttributes; protoElementInjector.attributes = renderElementBinder.readAttributes;
// Viewport directives are treated differently than other element with var- definitions. if (hasVariables) {
if (hasVariables && !isPresent(sortedDirectives.viewportDirective)) {
protoElementInjector.exportComponent = isPresent(sortedDirectives.componentDirective); protoElementInjector.exportComponent = isPresent(sortedDirectives.componentDirective);
protoElementInjector.exportElement = isBlank(sortedDirectives.componentDirective); protoElementInjector.exportElement = isBlank(sortedDirectives.componentDirective);
@ -105,8 +104,7 @@ export class ProtoViewFactory {
parent, parent,
renderElementBinder.distanceToParent, renderElementBinder.distanceToParent,
protoElementInjector, protoElementInjector,
sortedDirectives.componentDirective, sortedDirectives.componentDirective
sortedDirectives.viewportDirective
); );
// text nodes // text nodes
for (var i=0; i<renderElementBinder.textBindings.length; i++) { for (var i=0; i<renderElementBinder.textBindings.length; i++) {
@ -155,14 +153,12 @@ export class ProtoViewFactory {
class SortedDirectives { class SortedDirectives {
componentDirective: DirectiveBinding; componentDirective: DirectiveBinding;
viewportDirective: DirectiveBinding;
renderDirectives: List<renderApi.DirectiveBinder>; renderDirectives: List<renderApi.DirectiveBinder>;
directives: List<DirectiveBinding>; directives: List<DirectiveBinding>;
constructor(renderDirectives, allDirectives) { constructor(renderDirectives, allDirectives) {
this.renderDirectives = []; this.renderDirectives = [];
this.directives = []; this.directives = [];
this.viewportDirective = null;
this.componentDirective = null; this.componentDirective = null;
ListWrapper.forEach(renderDirectives, (renderDirectiveBinder) => { ListWrapper.forEach(renderDirectives, (renderDirectiveBinder) => {
var directiveBinding = allDirectives[renderDirectiveBinder.directiveIndex]; var directiveBinding = allDirectives[renderDirectiveBinder.directiveIndex];
@ -172,9 +168,6 @@ class SortedDirectives {
ListWrapper.insert(this.renderDirectives, 0, renderDirectiveBinder); ListWrapper.insert(this.renderDirectives, 0, renderDirectiveBinder);
ListWrapper.insert(this.directives, 0, directiveBinding); ListWrapper.insert(this.directives, 0, directiveBinding);
} else { } else {
if (directiveBinding.annotation instanceof Viewport) {
this.viewportDirective = directiveBinding;
}
ListWrapper.push(this.renderDirectives, renderDirectiveBinder); ListWrapper.push(this.renderDirectives, renderDirectiveBinder);
ListWrapper.push(this.directives, directiveBinding); ListWrapper.push(this.directives, directiveBinding);
} }

View File

@ -7,7 +7,7 @@ import {BaseQueryList} from './base_query_list';
* The directives are kept in depth-first pre-order traversal of the DOM. * The directives are kept in depth-first pre-order traversal of the DOM.
* *
* The `QueryList` is iterable, therefore it can be used in both javascript code with `for..of` loop as well as in * The `QueryList` is iterable, therefore it can be used in both javascript code with `for..of` loop as well as in
* template with `*for="of"` viewport. * template with `*for="of"` directive.
* *
* NOTE: In the future this class will implement an `Observable` interface. For now it uses a plain list of observable * NOTE: In the future this class will implement an `Observable` interface. For now it uses a plain list of observable
* callbacks. * callbacks.

View File

@ -235,9 +235,9 @@ export class AppProtoView {
} }
bindElement(parent:ElementBinder, distanceToParent:int, protoElementInjector:ProtoElementInjector, bindElement(parent:ElementBinder, distanceToParent:int, protoElementInjector:ProtoElementInjector,
componentDirective:DirectiveBinding = null, viewportDirective:DirectiveBinding = null):ElementBinder { componentDirective:DirectiveBinding = null):ElementBinder {
var elBinder = new ElementBinder(this.elementBinders.length, parent, distanceToParent, var elBinder = new ElementBinder(this.elementBinders.length, parent, distanceToParent,
protoElementInjector, componentDirective, viewportDirective); protoElementInjector, componentDirective);
ListWrapper.push(this.elementBinders, elBinder); ListWrapper.push(this.elementBinders, elBinder);
return elBinder; return elBinder;
} }

View File

@ -10,14 +10,11 @@ import {ViewRef, ProtoViewRef, internalView} from './view_ref';
export class ViewContainerRef { export class ViewContainerRef {
_viewManager: avmModule.AppViewManager; _viewManager: avmModule.AppViewManager;
_element: ElementRef; _element: ElementRef;
_defaultProtoViewRef: ProtoViewRef;
constructor(viewManager: avmModule.AppViewManager, constructor(viewManager: avmModule.AppViewManager,
element: ElementRef, element: ElementRef) {
defaultProtoViewRef: ProtoViewRef) {
this._viewManager = viewManager; this._viewManager = viewManager;
this._element = element; this._element = element;
this._defaultProtoViewRef = defaultProtoViewRef;
} }
_getViews() { _getViews() {
@ -41,11 +38,8 @@ export class ViewContainerRef {
// TODO(rado): profile and decide whether bounds checks should be added // TODO(rado): profile and decide whether bounds checks should be added
// to the methods below. // to the methods below.
create(atIndex:number=-1, protoViewRef:ProtoViewRef = null, injector:Injector = null): ViewRef { create(protoViewRef:ProtoViewRef = null, atIndex:number=-1, injector:Injector = null): ViewRef {
if (atIndex == -1) atIndex = this.length; if (atIndex == -1) atIndex = this.length;
if (isBlank(protoViewRef)) {
protoViewRef = this._defaultProtoViewRef;
}
return this._viewManager.createViewInContainer(this._element, atIndex, protoViewRef, injector); return this._viewManager.createViewInContainer(this._element, atIndex, protoViewRef, injector);
} }

View File

@ -56,8 +56,8 @@ export class AppViewManagerUtils {
// preBuiltObjects // preBuiltObjects
if (isPresent(elementInjector)) { if (isPresent(elementInjector)) {
var defaultProtoView = isPresent(binder.viewportDirective) ? binder.nestedProtoView : null; var embeddedProtoView = binder.hasEmbeddedProtoView() ? binder.nestedProtoView : null;
preBuiltObjects[binderIdx] = new eli.PreBuiltObjects(viewManager, view, defaultProtoView); preBuiltObjects[binderIdx] = new eli.PreBuiltObjects(viewManager, view, embeddedProtoView);
} }
} }

View File

@ -1,6 +1,6 @@
import {Viewport} from 'angular2/src/core/annotations_impl/annotations'; import {Decorator} from 'angular2/src/core/annotations_impl/annotations';
import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref'; import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref';
import {ViewRef} from 'angular2/src/core/compiler/view_ref'; import {ViewRef, ProtoViewRef} from 'angular2/src/core/compiler/view_ref';
import {isPresent, isBlank} from 'angular2/src/facade/lang'; import {isPresent, isBlank} from 'angular2/src/facade/lang';
import {ListWrapper} from 'angular2/src/facade/collection'; import {ListWrapper} from 'angular2/src/facade/collection';
@ -36,7 +36,7 @@ import {ListWrapper} from 'angular2/src/facade/collection';
* *
* @exportedAs angular2/directives * @exportedAs angular2/directives
*/ */
@Viewport({ @Decorator({
selector: '[for][of]', selector: '[for][of]',
properties: { properties: {
'iterableChanges': 'of | iterableDiff' 'iterableChanges': 'of | iterableDiff'
@ -44,8 +44,10 @@ import {ListWrapper} from 'angular2/src/facade/collection';
}) })
export class For { export class For {
viewContainer: ViewContainerRef; viewContainer: ViewContainerRef;
constructor(viewContainer:ViewContainerRef) { protoViewRef: ProtoViewRef;
constructor(viewContainer:ViewContainerRef, protoViewRef: ProtoViewRef) {
this.viewContainer = viewContainer; this.viewContainer = viewContainer;
this.protoViewRef = protoViewRef;
} }
set iterableChanges(changes) { set iterableChanges(changes) {
@ -71,7 +73,7 @@ export class For {
(addedRecord) => ListWrapper.push(insertTuples, new RecordViewTuple(addedRecord, null)) (addedRecord) => ListWrapper.push(insertTuples, new RecordViewTuple(addedRecord, null))
); );
For.bulkInsert(insertTuples, this.viewContainer); For.bulkInsert(insertTuples, this.viewContainer, this.protoViewRef);
for (var i = 0; i < insertTuples.length; i++) { for (var i = 0; i < insertTuples.length; i++) {
this.perViewChange(insertTuples[i].view, insertTuples[i].record); this.perViewChange(insertTuples[i].view, insertTuples[i].record);
@ -99,14 +101,14 @@ export class For {
return movedTuples; return movedTuples;
} }
static bulkInsert(tuples, viewContainer) { static bulkInsert(tuples, viewContainer, protoViewRef) {
tuples.sort((a, b) => a.record.currentIndex - b.record.currentIndex); tuples.sort((a, b) => a.record.currentIndex - b.record.currentIndex);
for (var i = 0; i < tuples.length; i++) { for (var i = 0; i < tuples.length; i++) {
var tuple = tuples[i]; var tuple = tuples[i];
if (isPresent(tuple.view)) { if (isPresent(tuple.view)) {
viewContainer.insert(tuple.view, tuple.record.currentIndex); viewContainer.insert(tuple.view, tuple.record.currentIndex);
} else { } else {
tuple.view = viewContainer.create(tuple.record.currentIndex); tuple.view = viewContainer.create(protoViewRef, tuple.record.currentIndex);
} }
} }
return tuples; return tuples;

View File

@ -1,5 +1,6 @@
import {Viewport} from 'angular2/src/core/annotations_impl/annotations'; import {Decorator} from 'angular2/src/core/annotations_impl/annotations';
import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref'; import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref';
import {ProtoViewRef} from 'angular2/src/core/compiler/view_ref';
import {isBlank} from 'angular2/src/facade/lang'; import {isBlank} from 'angular2/src/facade/lang';
/** /**
@ -25,7 +26,7 @@ import {isBlank} from 'angular2/src/facade/lang';
* *
* @exportedAs angular2/directives * @exportedAs angular2/directives
*/ */
@Viewport({ @Decorator({
selector: '[if]', selector: '[if]',
properties: { properties: {
'condition': 'if' 'condition': 'if'
@ -33,17 +34,19 @@ import {isBlank} from 'angular2/src/facade/lang';
}) })
export class If { export class If {
viewContainer: ViewContainerRef; viewContainer: ViewContainerRef;
protoViewRef: ProtoViewRef;
prevCondition: boolean; prevCondition: boolean;
constructor(viewContainer: ViewContainerRef) { constructor(viewContainer: ViewContainerRef, protoViewRef:ProtoViewRef) {
this.viewContainer = viewContainer; this.viewContainer = viewContainer;
this.prevCondition = null; this.prevCondition = null;
this.protoViewRef = protoViewRef;
} }
set condition(newCondition /* boolean */) { set condition(newCondition /* boolean */) {
if (newCondition && (isBlank(this.prevCondition) || !this.prevCondition)) { if (newCondition && (isBlank(this.prevCondition) || !this.prevCondition)) {
this.prevCondition = true; this.prevCondition = true;
this.viewContainer.create(); this.viewContainer.create(this.protoViewRef);
} else if (!newCondition && (isBlank(this.prevCondition) || this.prevCondition)) { } else if (!newCondition && (isBlank(this.prevCondition) || this.prevCondition)) {
this.prevCondition = false; this.prevCondition = false;
this.viewContainer.clear(); this.viewContainer.clear();

View File

@ -1,9 +1,28 @@
import {Decorator, Viewport} from 'angular2/src/core/annotations_impl/annotations'; import {Decorator} from 'angular2/src/core/annotations_impl/annotations';
import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref'; import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref';
import {ProtoViewRef} from 'angular2/src/core/compiler/view_ref';
import {isPresent, isBlank, normalizeBlank} from 'angular2/src/facade/lang'; import {isPresent, isBlank, normalizeBlank} from 'angular2/src/facade/lang';
import {ListWrapper, List, MapWrapper, Map} from 'angular2/src/facade/collection'; import {ListWrapper, List, MapWrapper, Map} from 'angular2/src/facade/collection';
import {Parent} from 'angular2/src/core/annotations_impl/visibility'; import {Parent} from 'angular2/src/core/annotations_impl/visibility';
class SwitchView {
_viewContainerRef: ViewContainerRef;
_protoViewRef: ProtoViewRef;
constructor(viewContainerRef: ViewContainerRef, protoViewRef: ProtoViewRef) {
this._protoViewRef = protoViewRef;
this._viewContainerRef = viewContainerRef;
}
create() {
this._viewContainerRef.create(this._protoViewRef);
}
destroy() {
this._viewContainerRef.clear();
}
}
/** /**
* The `Switch` directive is used to conditionally swap DOM structure on your template based on a * The `Switch` directive is used to conditionally swap DOM structure on your template based on a
* scope expression. * scope expression.
@ -40,93 +59,94 @@ import {Parent} from 'angular2/src/core/annotations_impl/visibility';
export class Switch { export class Switch {
_switchValue: any; _switchValue: any;
_useDefault: boolean; _useDefault: boolean;
_valueViewContainers: Map; _valueViews: Map;
_activeViewContainers: List<ViewContainerRef>; _activeViews: List<SwitchView>;
constructor() { constructor() {
this._valueViewContainers = MapWrapper.create(); this._valueViews = MapWrapper.create();
this._activeViewContainers = ListWrapper.create(); this._activeViews = ListWrapper.create();
this._useDefault = false; this._useDefault = false;
} }
set value(value) { set value(value) {
// Empty the currently active ViewContainers // Empty the currently active ViewContainers
this._emptyAllActiveViewContainers(); this._emptyAllActiveViews();
// Add the ViewContainers matching the value (with a fallback to default) // Add the ViewContainers matching the value (with a fallback to default)
this._useDefault = false; this._useDefault = false;
var containers = MapWrapper.get(this._valueViewContainers, value); var views = MapWrapper.get(this._valueViews, value);
if (isBlank(containers)) { if (isBlank(views)) {
this._useDefault = true; this._useDefault = true;
containers = normalizeBlank(MapWrapper.get(this._valueViewContainers, _whenDefault)); views = normalizeBlank(MapWrapper.get(this._valueViews, _whenDefault));
} }
this._activateViewContainers(containers); this._activateViews(views);
this._switchValue = value; this._switchValue = value;
} }
_onWhenValueChanged(oldWhen, newWhen, viewContainer: ViewContainerRef):void { _onWhenValueChanged(oldWhen, newWhen, view: SwitchView):void {
this._deregisterViewContainer(oldWhen, viewContainer); this._deregisterView(oldWhen, view);
this._registerViewContainer(newWhen, viewContainer); this._registerView(newWhen, view);
if (oldWhen === this._switchValue) { if (oldWhen === this._switchValue) {
viewContainer.remove(); view.destroy();
ListWrapper.remove(this._activeViewContainers, viewContainer); ListWrapper.remove(this._activeViews, view);
} else if (newWhen === this._switchValue) { } else if (newWhen === this._switchValue) {
if (this._useDefault) { if (this._useDefault) {
this._useDefault = false; this._useDefault = false;
this._emptyAllActiveViewContainers(); this._emptyAllActiveViews();
} }
viewContainer.create(); view.create();
ListWrapper.push(this._activeViewContainers, viewContainer); ListWrapper.push(this._activeViews, view);
} }
// Switch to default when there is no more active ViewContainers // Switch to default when there is no more active ViewContainers
if (this._activeViewContainers.length === 0 && !this._useDefault) { if (this._activeViews.length === 0 && !this._useDefault) {
this._useDefault = true; this._useDefault = true;
this._activateViewContainers(MapWrapper.get(this._valueViewContainers, _whenDefault)); this._activateViews(MapWrapper.get(this._valueViews, _whenDefault));
} }
} }
_emptyAllActiveViewContainers():void { _emptyAllActiveViews():void {
var activeContainers = this._activeViewContainers; var activeContainers = this._activeViews;
for (var i = 0; i < activeContainers.length; i++) { for (var i = 0; i < activeContainers.length; i++) {
activeContainers[i].remove(); activeContainers[i].destroy();
} }
this._activeViewContainers = ListWrapper.create(); this._activeViews = ListWrapper.create();
} }
_activateViewContainers(containers: List<ViewContainerRef>):void { _activateViews(views: List<SwitchView>):void {
// TODO(vicb): assert(this._activeViewContainers.length === 0); // TODO(vicb): assert(this._activeViews.length === 0);
if (isPresent(containers)) { if (isPresent(views)) {
for (var i = 0; i < containers.length; i++) { for (var i = 0; i < views.length; i++) {
containers[i].create(); views[i].create();
} }
this._activeViewContainers = containers; this._activeViews = views;
} }
} }
_registerViewContainer(value, container: ViewContainerRef): void { _registerView(value, view: SwitchView): void {
var containers = MapWrapper.get(this._valueViewContainers, value); var views = MapWrapper.get(this._valueViews, value);
if (isBlank(containers)) { if (isBlank(views)) {
containers = ListWrapper.create(); views = ListWrapper.create();
MapWrapper.set(this._valueViewContainers, value, containers); MapWrapper.set(this._valueViews, value, views);
} }
ListWrapper.push(containers, container); ListWrapper.push(views, view);
} }
_deregisterViewContainer(value, container: ViewContainerRef):void { _deregisterView(value, view: SwitchView):void {
// `_whenDefault` is used a marker for non-registered whens // `_whenDefault` is used a marker for non-registered whens
if (value == _whenDefault) return; if (value == _whenDefault) return;
var containers = MapWrapper.get(this._valueViewContainers, value); var views = MapWrapper.get(this._valueViews, value);
if (containers.length == 1) { if (views.length == 1) {
MapWrapper.delete(this._valueViewContainers, value); MapWrapper.delete(this._valueViews, value);
} else { } else {
ListWrapper.remove(containers, container); ListWrapper.remove(views, view);
} }
} }
} }
/** /**
* Defines a case statement as an expression. * Defines a case statement as an expression.
* *
@ -144,7 +164,7 @@ export class Switch {
* *
* @exportedAs angular2/directives * @exportedAs angular2/directives
*/ */
@Viewport({ @Decorator({
selector: '[switch-when]', selector: '[switch-when]',
properties: { properties: {
'when' : 'switch-when' 'when' : 'switch-when'
@ -153,17 +173,21 @@ export class Switch {
export class SwitchWhen { export class SwitchWhen {
_value: any; _value: any;
_switch: Switch; _switch: Switch;
_viewContainer: ViewContainerRef; _view: SwitchView;
constructor(viewContainer: ViewContainerRef, @Parent() sswitch: Switch) { constructor(viewContainer: ViewContainerRef, protoViewRef: ProtoViewRef, @Parent() sswitch: Switch) {
// `_whenDefault` is used as a marker for a not yet initialized value // `_whenDefault` is used as a marker for a not yet initialized value
this._value = _whenDefault; this._value = _whenDefault;
this._switch = sswitch; this._switch = sswitch;
this._viewContainer = viewContainer; this._view = new SwitchView(viewContainer, protoViewRef);
}
onDestroy() {
this._switch
} }
set when(value) { set when(value) {
this._switch._onWhenValueChanged(this._value, value, this._viewContainer); this._switch._onWhenValueChanged(this._value, value, this._view);
this._value = value; this._value = value;
} }
} }
@ -182,12 +206,12 @@ export class SwitchWhen {
* *
* @exportedAs angular2/directives * @exportedAs angular2/directives
*/ */
@Viewport({ @Decorator({
selector: '[switch-default]' selector: '[switch-default]'
}) })
export class SwitchDefault { export class SwitchDefault {
constructor(viewContainer: ViewContainerRef, @Parent() sswitch: Switch) { constructor(viewContainer: ViewContainerRef, protoViewRef: ProtoViewRef, @Parent() sswitch: Switch) {
sswitch._registerViewContainer(_whenDefault, viewContainer); sswitch._registerView(_whenDefault, new SwitchView(viewContainer, protoViewRef));
} }
} }

View File

@ -91,7 +91,7 @@ export class ProtoViewDto {
// The view of the component // The view of the component
// Can contain 0 to n views of type #EMBEDDED_VIEW_TYPE // Can contain 0 to n views of type #EMBEDDED_VIEW_TYPE
static get COMPONENT_VIEW_TYPE() { return 1; } static get COMPONENT_VIEW_TYPE() { return 1; }
// A view that is included via a Viewport directive // A view that is embedded into another View via a <template> element
// inside of a component view // inside of a component view
static get EMBEDDED_VIEW_TYPE() { return 1; } static get EMBEDDED_VIEW_TYPE() { return 1; }
@ -111,7 +111,6 @@ export class ProtoViewDto {
export class DirectiveMetadata { export class DirectiveMetadata {
static get DECORATOR_TYPE() { return 0; } static get DECORATOR_TYPE() { return 0; }
static get COMPONENT_TYPE() { return 1; } static get COMPONENT_TYPE() { return 1; }
static get VIEWPORT_TYPE() { return 2; }
id:any; id:any;
selector:string; selector:string;
compileChildren:boolean; compileChildren:boolean;

View File

@ -56,11 +56,7 @@ export class DirectiveParser extends CompileStep {
cssSelector.addAttribute(attrName, attrValue); cssSelector.addAttribute(attrName, attrValue);
}); });
var viewportDirective;
var componentDirective; var componentDirective;
// Note: We assume that the ViewSplitter already did its work, i.e. template directive should
// only be present on <template> elements!
var isTemplateElement = DOM.isTemplateElement(current.element);
this._selectorMatcher.match(cssSelector, (selector, directiveIndex) => { this._selectorMatcher.match(cssSelector, (selector, directiveIndex) => {
var elementBinder = current.bindElement(); var elementBinder = current.bindElement();
@ -87,26 +83,12 @@ export class DirectiveParser extends CompileStep {
elementBinder.readAttribute(attrName); elementBinder.readAttribute(attrName);
}); });
} }
if (directive.type === DirectiveMetadata.VIEWPORT_TYPE) { if (directive.type === DirectiveMetadata.COMPONENT_TYPE) {
if (!isTemplateElement) { if (isPresent(componentDirective)) {
throw new BaseException(`Viewport directives need to be placed on <template> elements or elements ` + throw new BaseException(`Only one component directive is allowed per element - check ${current.elementDescription}`);
`with template attribute - check ${current.elementDescription}`);
}
if (isPresent(viewportDirective)) {
throw new BaseException(`Only one viewport directive is allowed per element - check ${current.elementDescription}`);
}
viewportDirective = directive;
} else {
if (isTemplateElement) {
throw new BaseException(`Only template directives are allowed on template elements - check ${current.elementDescription}`);
}
if (directive.type === DirectiveMetadata.COMPONENT_TYPE) {
if (isPresent(componentDirective)) {
throw new BaseException(`Only one component directive is allowed per element - check ${current.elementDescription}`);
}
componentDirective = directive;
elementBinder.setComponentId(directive.id);
} }
componentDirective = directive;
elementBinder.setComponentId(directive.id);
} }
}); });
} }

View File

@ -32,7 +32,7 @@ export class RouterOutlet {
]); ]);
this._viewContainer.clear(); this._viewContainer.clear();
this._viewContainer.create(0, pv, outletInjector); this._viewContainer.create(pv, 0, outletInjector);
}); });
} }

View File

@ -19,8 +19,6 @@ num _getDirectiveType(String annotationName) {
return DirectiveMetadata.DECORATOR_TYPE; return DirectiveMetadata.DECORATOR_TYPE;
case 'Component': case 'Component':
return DirectiveMetadata.COMPONENT_TYPE; return DirectiveMetadata.COMPONENT_TYPE;
case 'Viewport':
return DirectiveMetadata.VIEWPORT_TYPE;
default: default:
return -1; return -1;
} }

View File

@ -21,7 +21,7 @@ import {Compiler, CompilerCache} from 'angular2/src/core/compiler/compiler';
import {AppProtoView} from 'angular2/src/core/compiler/view'; import {AppProtoView} from 'angular2/src/core/compiler/view';
import {ElementBinder} from 'angular2/src/core/compiler/element_binder'; import {ElementBinder} from 'angular2/src/core/compiler/element_binder';
import {DirectiveMetadataReader} from 'angular2/src/core/compiler/directive_metadata_reader'; import {DirectiveMetadataReader} from 'angular2/src/core/compiler/directive_metadata_reader';
import {Component, DynamicComponent, Viewport, Decorator} from 'angular2/src/core/annotations_impl/annotations'; import {Component, DynamicComponent, Decorator} from 'angular2/src/core/annotations_impl/annotations';
import {Attribute} from 'angular2/src/core/annotations_impl/di'; import {Attribute} from 'angular2/src/core/annotations_impl/di';
import {View} from 'angular2/src/core/annotations_impl/view'; import {View} from 'angular2/src/core/annotations_impl/view';
import {internalProtoView} from 'angular2/src/core/compiler/view_ref'; import {internalProtoView} from 'angular2/src/core/compiler/view_ref';
@ -150,13 +150,6 @@ export function main() {
}); });
})); }));
it('should fill directive.type for viewport directives', inject([AsyncTestCompleter], (async) => {
captureDirective(SomeViewportDirective).then( (renderDir) => {
expect(renderDir.type).toEqual(renderApi.DirectiveMetadata.VIEWPORT_TYPE);
async.done();
});
}));
it('should fill directive.type for decorator directives', inject([AsyncTestCompleter], (async) => { it('should fill directive.type for decorator directives', inject([AsyncTestCompleter], (async) => {
captureDirective(SomeDecoratorDirective).then( (renderDir) => { captureDirective(SomeDecoratorDirective).then( (renderDir) => {
expect(renderDir.type).toEqual(renderApi.DirectiveMetadata.DECORATOR_TYPE); expect(renderDir.type).toEqual(renderApi.DirectiveMetadata.DECORATOR_TYPE);
@ -297,7 +290,7 @@ export function main() {
}); });
})); }));
it('should load nested components in viewport', inject([AsyncTestCompleter], (async) => { it('should load nested components in viewcontainers', inject([AsyncTestCompleter], (async) => {
tplResolver.setView(MainComponent, new View({template: '<div></div>'})); tplResolver.setView(MainComponent, new View({template: '<div></div>'}));
tplResolver.setView(NestedComponent, new View({template: '<div></div>'})); tplResolver.setView(NestedComponent, new View({template: '<div></div>'}));
var mainProtoView = createProtoView([ var mainProtoView = createProtoView([
@ -451,16 +444,14 @@ function createComponentElementBinder(reader, type) {
var binding = createDirectiveBinding(reader, type); var binding = createDirectiveBinding(reader, type);
return new ElementBinder( return new ElementBinder(
0, null, 0, 0, null, 0,
null, binding, null, binding
null
); );
} }
function createViewportElementBinder(nestedProtoView) { function createViewportElementBinder(nestedProtoView) {
var elBinder = new ElementBinder( var elBinder = new ElementBinder(
0, null, 0, 0, null, 0,
null, null, null, null
null
); );
elBinder.nestedProtoView = nestedProtoView; elBinder.nestedProtoView = nestedProtoView;
return elBinder; return elBinder;
@ -502,9 +493,6 @@ class RecursiveComponent {}
@DynamicComponent() @DynamicComponent()
class SomeDynamicComponentDirective {} class SomeDynamicComponentDirective {}
@Viewport()
class SomeViewportDirective {}
@Decorator() @Decorator()
class SomeDecoratorDirective {} class SomeDecoratorDirective {}

View File

@ -2,7 +2,7 @@ import {isPresent} from 'angular2/src/facade/lang';
import {ListWrapper} from 'angular2/src/facade/collection'; import {ListWrapper} from 'angular2/src/facade/collection';
import {ddescribe, describe, it, iit, expect, beforeEach} from 'angular2/test_lib'; import {ddescribe, describe, it, iit, expect, beforeEach} from 'angular2/test_lib';
import {DirectiveMetadataReader} from 'angular2/src/core/compiler/directive_metadata_reader'; import {DirectiveMetadataReader} from 'angular2/src/core/compiler/directive_metadata_reader';
import {Decorator, Component, Viewport} from 'angular2/src/core/annotations_impl/annotations'; import {Decorator, Component} from 'angular2/src/core/annotations_impl/annotations';
import {DirectiveMetadata} from 'angular2/src/core/compiler/directive_metadata'; import {DirectiveMetadata} from 'angular2/src/core/compiler/directive_metadata';
import {Injectable, Injector} from 'angular2/di'; import {Injectable, Injector} from 'angular2/di';
@ -15,9 +15,6 @@ class SomeDecorator {}
@Component({selector: 'someComponent', injectables: [SomeInjectable]}) @Component({selector: 'someComponent', injectables: [SomeInjectable]})
class SomeComponent {} class SomeComponent {}
@Viewport({selector: 'someViewport'})
class SomeViewport {}
class SomeDirectiveWithoutAnnotation { class SomeDirectiveWithoutAnnotation {
} }
@ -35,12 +32,6 @@ export function main() {
new DirectiveMetadata(SomeDecorator, new Decorator({selector: 'someDecorator'}), null)); new DirectiveMetadata(SomeDecorator, new Decorator({selector: 'someDecorator'}), null));
}); });
it('should read out the Viewport annotation', () => {
var directiveMetadata = reader.read(SomeViewport);
expect(directiveMetadata).toEqual(
new DirectiveMetadata(SomeViewport, new Viewport({selector: 'someViewport'}), null));
});
it('should read out the Component annotation', () => { it('should read out the Component annotation', () => {
var m = reader.read(SomeComponent); var m = reader.read(SomeComponent);
// For some reason `toEqual` fails to compare ResolvedBinding objects. // For some reason `toEqual` fails to compare ResolvedBinding objects.

View File

@ -16,7 +16,7 @@ import {
import {TestBed} from 'angular2/src/test_lib/test_bed'; import {TestBed} from 'angular2/src/test_lib/test_bed';
import {Decorator, Component, Viewport, DynamicComponent} from 'angular2/src/core/annotations_impl/annotations'; import {Decorator, Component, DynamicComponent} from 'angular2/src/core/annotations_impl/annotations';
import {View} from 'angular2/src/core/annotations_impl/view'; import {View} from 'angular2/src/core/annotations_impl/view';
import {DynamicComponentLoader} from 'angular2/src/core/compiler/dynamic_component_loader'; import {DynamicComponentLoader} from 'angular2/src/core/compiler/dynamic_component_loader';
import {ElementRef} from 'angular2/src/core/compiler/element_ref'; import {ElementRef} from 'angular2/src/core/compiler/element_ref';
@ -61,7 +61,7 @@ export function main() {
}); });
})); }));
it('should allow to destroy and create them via viewport directives', it('should allow to destroy and create them via viewcontainer directives',
inject([TestBed, AsyncTestCompleter], (tb, async) => { inject([TestBed, AsyncTestCompleter], (tb, async) => {
tb.overrideView(MyComp, new View({ tb.overrideView(MyComp, new View({
template: '<div><dynamic-comp #dynamic template="if: ctxBoolProp"></dynamic-comp></div>', template: '<div><dynamic-comp #dynamic template="if: ctxBoolProp"></dynamic-comp></div>',

View File

@ -705,6 +705,12 @@ export function main() {
expect(inj.get(NeedsProtoViewRef).protoViewRef).toEqual(new ProtoViewRef(protoView)); expect(inj.get(NeedsProtoViewRef).protoViewRef).toEqual(new ProtoViewRef(protoView));
}); });
it("should throw if there is no ProtoViewRef", function () {
expect(
() => injector([NeedsProtoViewRef])
).toThrowError('No provider for ProtoViewRef! (NeedsProtoViewRef -> ProtoViewRef)');
});
}); });
describe('directive queries', () => { describe('directive queries', () => {

View File

@ -24,7 +24,7 @@ import {Injector, bind} from 'angular2/di';
import {PipeRegistry, defaultPipeRegistry, import {PipeRegistry, defaultPipeRegistry,
ChangeDetection, DynamicChangeDetection, Pipe, ChangeDetectorRef, ON_PUSH} from 'angular2/change_detection'; ChangeDetection, DynamicChangeDetection, Pipe, ChangeDetectorRef, ON_PUSH} from 'angular2/change_detection';
import {Decorator, Component, Viewport, DynamicComponent} from 'angular2/src/core/annotations_impl/annotations'; import {Decorator, Component, DynamicComponent} from 'angular2/src/core/annotations_impl/annotations';
import {View} from 'angular2/src/core/annotations_impl/view'; import {View} from 'angular2/src/core/annotations_impl/view';
import {Parent, Ancestor} from 'angular2/src/core/annotations_impl/visibility'; import {Parent, Ancestor} from 'angular2/src/core/annotations_impl/visibility';
import {Attribute} from 'angular2/src/core/annotations_impl/di'; import {Attribute} from 'angular2/src/core/annotations_impl/di';
@ -32,6 +32,7 @@ import {Attribute} from 'angular2/src/core/annotations_impl/di';
import {If} from 'angular2/src/directives/if'; import {If} from 'angular2/src/directives/if';
import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref'; import {ViewContainerRef} from 'angular2/src/core/compiler/view_container_ref';
import {ProtoViewRef} from 'angular2/src/core/compiler/view_ref';
import {Compiler} from 'angular2/src/core/compiler/compiler'; import {Compiler} from 'angular2/src/core/compiler/compiler';
import {ElementRef} from 'angular2/src/core/compiler/element_ref'; import {ElementRef} from 'angular2/src/core/compiler/element_ref';
@ -522,7 +523,7 @@ export function main() {
}) })
})); }));
it('should create a component that injects an @Ancestor through viewport directive', inject([TestBed, AsyncTestCompleter], (tb, async) => { it('should create a component that injects an @Ancestor through viewcontainer directive', inject([TestBed, AsyncTestCompleter], (tb, async) => {
tb.overrideView(MyComp, new View({ tb.overrideView(MyComp, new View({
template: ` template: `
<some-directive> <some-directive>
@ -865,7 +866,7 @@ class DynamicViewport {
var myService = new MyService(); var myService = new MyService();
myService.greeting = 'dynamic greet'; myService.greeting = 'dynamic greet';
this.done = compiler.compileInHost(ChildCompUsingService).then( (hostPv) => { this.done = compiler.compileInHost(ChildCompUsingService).then( (hostPv) => {
vc.create(0, hostPv, inj.createChildFromResolved(Injector.resolve([bind(MyService).toValue(myService)]))) vc.create(hostPv, 0, inj.createChildFromResolved(Injector.resolve([bind(MyService).toValue(myService)])))
}); });
} }
} }
@ -1037,13 +1038,13 @@ class ChildComp2 {
} }
} }
@Viewport({ @Decorator({
selector: '[some-viewport]' selector: '[some-viewport]'
}) })
class SomeViewport { class SomeViewport {
constructor(container: ViewContainerRef) { constructor(container: ViewContainerRef, protoView:ProtoViewRef) {
container.create().setLocal('some-tmpl', 'hello'); container.create(protoView).setLocal('some-tmpl', 'hello');
container.create().setLocal('some-tmpl', 'again'); container.create(protoView).setLocal('some-tmpl', 'again');
} }
} }

View File

@ -44,8 +44,8 @@ export function main() {
return new AppView(null, createProtoView(), MapWrapper.create()); return new AppView(null, createProtoView(), MapWrapper.create());
} }
function createViewContainer(defaultProtoView = null) { function createViewContainer() {
return new ViewContainerRef(viewManager, location, defaultProtoView); return new ViewContainerRef(viewManager, location);
} }
beforeEach( () => { beforeEach( () => {

View File

@ -60,12 +60,12 @@ export function main() {
} }
function createEmptyElBinder() { function createEmptyElBinder() {
return new ElementBinder(0, null, 0, null, null, null); return new ElementBinder(0, null, 0, null, null);
} }
function createComponentElBinder(nestedProtoView = null) { function createComponentElBinder(nestedProtoView = null) {
var binding = createDirectiveBinding(SomeComponent); var binding = createDirectiveBinding(SomeComponent);
var binder = new ElementBinder(0, null, 0, null, binding, null); var binder = new ElementBinder(0, null, 0, null, binding);
binder.nestedProtoView = nestedProtoView; binder.nestedProtoView = nestedProtoView;
return binder; return binder;
} }

View File

@ -45,12 +45,12 @@ export function main() {
} }
function createEmptyElBinder() { function createEmptyElBinder() {
return new ElementBinder(0, null, 0, null, null, null); return new ElementBinder(0, null, 0, null, null);
} }
function createComponentElBinder(nestedProtoView = null) { function createComponentElBinder(nestedProtoView = null) {
var binding = createDirectiveBinding(SomeComponent); var binding = createDirectiveBinding(SomeComponent);
var binder = new ElementBinder(0, null, 0, null, binding, null); var binder = new ElementBinder(0, null, 0, null, binding);
binder.nestedProtoView = nestedProtoView; binder.nestedProtoView = nestedProtoView;
return binder; return binder;
} }

View File

@ -17,8 +17,6 @@ export function main() {
annotatedDirectives = [ annotatedDirectives = [
someComponent, someComponent,
someComponent2, someComponent2,
someViewport,
someViewport2,
someDecorator, someDecorator,
someDecoratorIgnoringChildren, someDecoratorIgnoringChildren,
decoratorWithMultipleAttrs, decoratorWithMultipleAttrs,
@ -153,27 +151,6 @@ export function main() {
expect(eventBinding.source.source).toEqual('doItGlobal()'); expect(eventBinding.source.source).toEqual('doItGlobal()');
}); });
//TODO: assertions should be enabled when running tests: https://github.com/angular/angular/issues/1340
describe('viewport directives', () => {
it('should not allow multiple viewport directives on the same element', () => {
expect( () => {
process(
el('<template some-vp some-vp2></template>')
);
}).toThrowError('Only one viewport directive is allowed per element - check '
+ (assertionsEnabled() ? '<template some-vp some-vp2>' : 'null'));
});
it('should not allow viewport directives on non <template> elements', () => {
expect( () => {
process(
el('<div some-vp></div>')
);
}).toThrowError('Viewport directives need to be placed on <template> elements or elements with template attribute - check '
+ (assertionsEnabled() ? '<div some-vp>' : 'null'));
});
});
//TODO: assertions should be enabled when running tests: https://github.com/angular/angular/issues/1340 //TODO: assertions should be enabled when running tests: https://github.com/angular/angular/issues/1340
describe('component directives', () => { describe('component directives', () => {
it('should save the component id', () => { it('should save the component id', () => {
@ -231,18 +208,6 @@ var someComponent2 = new DirectiveMetadata({
type: DirectiveMetadata.COMPONENT_TYPE type: DirectiveMetadata.COMPONENT_TYPE
}); });
var someViewport = new DirectiveMetadata({
selector: '[some-vp]',
id: 'someViewport',
type: DirectiveMetadata.VIEWPORT_TYPE
});
var someViewport2 = new DirectiveMetadata({
selector: '[some-vp2]',
id: 'someViewport2',
type: DirectiveMetadata.VIEWPORT_TYPE
});
var someDecorator = new DirectiveMetadata({ var someDecorator = new DirectiveMetadata({
selector: '[some-decor]', selector: '[some-decor]',
type: DirectiveMetadata.DECORATOR_TYPE type: DirectiveMetadata.DECORATOR_TYPE

View File

@ -43,7 +43,7 @@ export function main() {
['properties', MapWrapper.createFromPairs([['propKey', 'propVal']])], ['properties', MapWrapper.createFromPairs([['propKey', 'propVal']])],
['readAttributes', ['readTest1', 'readTest2']], ['readAttributes', ['readTest1', 'readTest2']],
['selector', 'testSelector'], ['selector', 'testSelector'],
['type', DirectiveMetadata.VIEWPORT_TYPE] ['type', DirectiveMetadata.DECORATOR_TYPE]
]); ]);
var meta = directiveMetadataFromMap(map); var meta = directiveMetadataFromMap(map);
expect(meta.compileChildren).toEqual(false); expect(meta.compileChildren).toEqual(false);
@ -56,7 +56,7 @@ export function main() {
MapWrapper.createFromPairs([['propKey', 'propVal']])); MapWrapper.createFromPairs([['propKey', 'propVal']]));
expect(meta.readAttributes).toEqual(['readTest1', 'readTest2']); expect(meta.readAttributes).toEqual(['readTest1', 'readTest2']);
expect(meta.selector).toEqual('testSelector'); expect(meta.selector).toEqual('testSelector');
expect(meta.type).toEqual(DirectiveMetadata.VIEWPORT_TYPE); expect(meta.type).toEqual(DirectiveMetadata.DECORATOR_TYPE);
}); });
}); });
} }

View File

@ -405,7 +405,7 @@ var multipleContentTagsComponent = new DirectiveMetadata({
var manualViewportDirective = new DirectiveMetadata({ var manualViewportDirective = new DirectiveMetadata({
selector: '[manual]', selector: '[manual]',
id: 'manual', id: 'manual',
type: DirectiveMetadata.VIEWPORT_TYPE type: DirectiveMetadata.DECORATOR_TYPE
}); });
var outerWithIndirectNestedComponent = new DirectiveMetadata({ var outerWithIndirectNestedComponent = new DirectiveMetadata({
@ -441,7 +441,7 @@ var conditionalContentComponent = new DirectiveMetadata({
var autoViewportDirective = new DirectiveMetadata({ var autoViewportDirective = new DirectiveMetadata({
selector: '[auto]', selector: '[auto]',
id: '[auto]', id: '[auto]',
type: DirectiveMetadata.VIEWPORT_TYPE type: DirectiveMetadata.DECORATOR_TYPE
}); });
var tabGroupComponent = new DirectiveMetadata({ var tabGroupComponent = new DirectiveMetadata({

View File

@ -16,7 +16,7 @@ import {
import {TestBed} from 'angular2/test'; import {TestBed} from 'angular2/test';
import {Injector, bind} from 'angular2/di'; import {Injector, bind} from 'angular2/di';
import {Component, Viewport} from 'angular2/src/core/annotations_impl/annotations'; import {Component} from 'angular2/src/core/annotations_impl/annotations';
import {View} from 'angular2/src/core/annotations_impl/view'; import {View} from 'angular2/src/core/annotations_impl/view';
import {RootRouter} from 'angular2/src/router/router'; import {RootRouter} from 'angular2/src/router/router';

View File

@ -9,7 +9,7 @@ import {
import {IMPLEMENTS} from 'angular2/src/facade/lang'; import {IMPLEMENTS} from 'angular2/src/facade/lang';
import {Promise, PromiseWrapper} from 'angular2/src/facade/async'; import {Promise, PromiseWrapper} from 'angular2/src/facade/async';
import {RootRouter, Viewport} from 'angular2/src/router/router'; import {RootRouter} from 'angular2/src/router/router';
import {Pipeline} from 'angular2/src/router/pipeline'; import {Pipeline} from 'angular2/src/router/pipeline';
import {RouterOutlet} from 'angular2/src/router/router_outlet'; import {RouterOutlet} from 'angular2/src/router/router_outlet';

View File

@ -1,4 +1,4 @@
import {bootstrap, Component, Viewport, View, ViewContainerRef, Compiler, Decorator} from 'angular2/angular2'; import {bootstrap, Component, View, ViewContainerRef, Compiler, Decorator} from 'angular2/angular2';
import {LifeCycle} from 'angular2/src/core/life_cycle/life_cycle'; import {LifeCycle} from 'angular2/src/core/life_cycle/life_cycle';
import {reflector} from 'angular2/src/reflection/reflection'; import {reflector} from 'angular2/src/reflection/reflection';

View File

@ -7,11 +7,10 @@ import {Component, Decorator} from 'angular2/src/core/annotations_impl/annotatio
import {View} from 'angular2/src/core/annotations_impl/view'; import {View} from 'angular2/src/core/annotations_impl/view';
// Angular 2.0 supports 3 basic types of directives: // Angular 2.0 supports 2 basic types of directives:
// - Component - the basic building blocks of Angular 2.0 apps. Backed by // - Component - the basic building blocks of Angular 2.0 apps. Backed by
// ShadowDom.(http://www.html5rocks.com/en/tutorials/webcomponents/shadowdom/) // ShadowDom.(http://www.html5rocks.com/en/tutorials/webcomponents/shadowdom/)
// - Decorator - add behavior to existing elements. // - Decorator - add behavior to existing elements.
// - Viewport - allow for stamping out of a html template (not in this demo).
// @Component is AtScript syntax to annotate the HelloCmp class as an Angular // @Component is AtScript syntax to annotate the HelloCmp class as an Angular
// 2.0 component. // 2.0 component.
@ -46,8 +45,8 @@ export class HelloCmp {
} }
} }
// Decorators are light-weight. They don't allow for templates, or new // Decorators are light-weight. They don't allow new
// expression contexts (use @Component or @Viewport for those needs). // expression contexts (use @Component for those needs).
@Decorator({ @Decorator({
selector: '[red]' selector: '[red]'
}) })