2015-03-23 14:10:55 -07:00
|
|
|
import {isPresent} from 'angular2/src/facade/lang';
|
|
|
|
import {Promise} from 'angular2/src/facade/async';
|
|
|
|
import {List, Map} from 'angular2/src/facade/collection';
|
|
|
|
import {ASTWithSource} from 'angular2/change_detection';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* General notes:
|
2015-04-15 21:51:30 -07:00
|
|
|
*
|
|
|
|
* The methods for creating / destroying views in this API are used in the AppViewHydrator
|
|
|
|
* and RenderViewHydrator as well.
|
|
|
|
*
|
2015-03-23 14:10:55 -07:00
|
|
|
* We are already parsing expressions on the render side:
|
|
|
|
* - this makes the ElementBinders more compact
|
|
|
|
* (e.g. no need to distinguish interpolations from regular expressions from literals)
|
|
|
|
* - allows to retrieve which properties should be accessed from the event
|
|
|
|
* by looking at the expression
|
|
|
|
* - we need the parse at least for the `template` attribute to match
|
|
|
|
* directives in it
|
|
|
|
* - render compiler is not on the critical path as
|
|
|
|
* its output will be stored in precompiled templates.
|
|
|
|
*/
|
2015-04-02 15:56:58 +02:00
|
|
|
export class EventBinding {
|
2015-05-18 11:57:20 -07:00
|
|
|
fullName: string; // name/target:name, e.g "click", "window:resize"
|
2015-04-15 21:51:30 -07:00
|
|
|
source: ASTWithSource;
|
2015-04-02 15:56:58 +02:00
|
|
|
|
2015-05-18 11:57:20 -07:00
|
|
|
constructor(fullName: string, source: ASTWithSource) {
|
2015-04-02 15:56:58 +02:00
|
|
|
this.fullName = fullName;
|
|
|
|
this.source = source;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-23 14:10:55 -07:00
|
|
|
export class ElementBinder {
|
2015-05-18 11:57:20 -07:00
|
|
|
index: number;
|
|
|
|
parentIndex: number;
|
|
|
|
distanceToParent: number;
|
|
|
|
directives: List<DirectiveBinder>;
|
|
|
|
nestedProtoView: ProtoViewDto;
|
2015-03-23 14:10:55 -07:00
|
|
|
propertyBindings: Map<string, ASTWithSource>;
|
|
|
|
variableBindings: Map<string, ASTWithSource>;
|
|
|
|
// Note: this contains a preprocessed AST
|
|
|
|
// that replaced the values that should be extracted from the element
|
|
|
|
// with a local name
|
2015-04-02 15:56:58 +02:00
|
|
|
eventBindings: List<EventBinding>;
|
2015-03-23 14:10:55 -07:00
|
|
|
textBindings: List<ASTWithSource>;
|
2015-04-02 14:40:49 -07:00
|
|
|
readAttributes: Map<string, string>;
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
constructor({
|
2015-04-02 14:40:49 -07:00
|
|
|
index, parentIndex, distanceToParent,
|
|
|
|
directives, nestedProtoView,
|
2015-03-23 14:10:55 -07:00
|
|
|
propertyBindings, variableBindings,
|
2015-04-02 14:40:49 -07:00
|
|
|
eventBindings, textBindings,
|
|
|
|
readAttributes
|
2015-05-18 11:57:20 -07:00
|
|
|
}:{index?:number, parentIndex?:number, distanceToParent?:number,
|
|
|
|
directives?:List<DirectiveBinder>, nestedProtoView?:ProtoViewDto,
|
|
|
|
propertyBindings?:Map<string, ASTWithSource>, variableBindings?:Map<string, ASTWithSource>,
|
|
|
|
eventBindings?:List<EventBinding>, textBindings?:List<ASTWithSource>,
|
|
|
|
readAttributes?:Map<string, string>}={}) {
|
2015-03-23 14:10:55 -07:00
|
|
|
this.index = index;
|
|
|
|
this.parentIndex = parentIndex;
|
|
|
|
this.distanceToParent = distanceToParent;
|
|
|
|
this.directives = directives;
|
|
|
|
this.nestedProtoView = nestedProtoView;
|
|
|
|
this.propertyBindings = propertyBindings;
|
|
|
|
this.variableBindings = variableBindings;
|
|
|
|
this.eventBindings = eventBindings;
|
|
|
|
this.textBindings = textBindings;
|
2015-04-02 14:40:49 -07:00
|
|
|
this.readAttributes = readAttributes;
|
2015-03-23 14:10:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export class DirectiveBinder {
|
2015-04-09 21:20:11 +02:00
|
|
|
// Index into the array of directives in the View instance
|
2015-05-18 11:57:20 -07:00
|
|
|
directiveIndex: number;
|
2015-03-23 14:10:55 -07:00
|
|
|
propertyBindings: Map<string, ASTWithSource>;
|
|
|
|
// Note: this contains a preprocessed AST
|
|
|
|
// that replaced the values that should be extracted from the element
|
|
|
|
// with a local name
|
2015-04-02 15:56:58 +02:00
|
|
|
eventBindings: List<EventBinding>;
|
2015-04-21 11:47:53 -07:00
|
|
|
hostPropertyBindings: Map<string, ASTWithSource>;
|
2015-03-23 14:10:55 -07:00
|
|
|
constructor({
|
2015-04-21 11:47:53 -07:00
|
|
|
directiveIndex, propertyBindings, eventBindings, hostPropertyBindings
|
2015-05-18 11:57:20 -07:00
|
|
|
}:{
|
|
|
|
directiveIndex?:number, propertyBindings?:Map<string, ASTWithSource>,
|
|
|
|
eventBindings?:List<EventBinding>, hostPropertyBindings?:Map<string, ASTWithSource>
|
2015-03-23 14:10:55 -07:00
|
|
|
}) {
|
|
|
|
this.directiveIndex = directiveIndex;
|
|
|
|
this.propertyBindings = propertyBindings;
|
|
|
|
this.eventBindings = eventBindings;
|
2015-04-21 11:47:53 -07:00
|
|
|
this.hostPropertyBindings = hostPropertyBindings;
|
2015-03-23 14:10:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-09 21:20:11 +02:00
|
|
|
export class ProtoViewDto {
|
2015-04-15 21:51:30 -07:00
|
|
|
// A view that contains the host element with bound
|
|
|
|
// component directive.
|
|
|
|
// Contains a view of type #COMPONENT_VIEW_TYPE.
|
|
|
|
static get HOST_VIEW_TYPE() { return 0; }
|
|
|
|
// The view of the component
|
|
|
|
// Can contain 0 to n views of type #EMBEDDED_VIEW_TYPE
|
|
|
|
static get COMPONENT_VIEW_TYPE() { return 1; }
|
2015-04-29 15:07:55 -07:00
|
|
|
// A view that is embedded into another View via a <template> element
|
2015-04-15 21:51:30 -07:00
|
|
|
// inside of a component view
|
2015-05-11 17:59:39 -07:00
|
|
|
static get EMBEDDED_VIEW_TYPE() { return 2; }
|
2015-04-15 21:51:30 -07:00
|
|
|
|
2015-04-28 11:20:01 -07:00
|
|
|
render: RenderProtoViewRef;
|
2015-05-18 11:57:20 -07:00
|
|
|
elementBinders: List<ElementBinder>;
|
2015-03-23 14:10:55 -07:00
|
|
|
variableBindings: Map<string, string>;
|
2015-04-15 21:51:30 -07:00
|
|
|
type: number;
|
2015-03-23 14:10:55 -07:00
|
|
|
|
2015-05-18 11:57:20 -07:00
|
|
|
constructor({
|
|
|
|
render, elementBinders, variableBindings, type
|
|
|
|
}:{
|
|
|
|
render?:RenderProtoViewRef, elementBinders?:List<ElementBinder>,
|
|
|
|
variableBindings?:Map<string, string>, type?:number
|
|
|
|
}) {
|
2015-03-23 14:10:55 -07:00
|
|
|
this.render = render;
|
|
|
|
this.elementBinders = elementBinders;
|
|
|
|
this.variableBindings = variableBindings;
|
2015-04-15 21:51:30 -07:00
|
|
|
this.type = type;
|
2015-03-23 14:10:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export class DirectiveMetadata {
|
2015-04-30 13:38:40 -07:00
|
|
|
static get DIRECTIVE_TYPE() { return 0; }
|
2015-03-23 14:10:55 -07:00
|
|
|
static get COMPONENT_TYPE() { return 1; }
|
2015-05-18 11:57:20 -07:00
|
|
|
id: any;
|
|
|
|
selector: string;
|
|
|
|
compileChildren: boolean;
|
|
|
|
events: List<string>;
|
|
|
|
hostListeners: Map<string, string>;
|
|
|
|
hostProperties: Map<string, string>;
|
|
|
|
hostAttributes: Map<string, string>;
|
|
|
|
hostActions: Map<string, string>;
|
|
|
|
properties: Map<string, string>;
|
|
|
|
readAttributes: List<string>;
|
|
|
|
type: number;
|
|
|
|
callOnDestroy: boolean;
|
|
|
|
callOnChange: boolean;
|
|
|
|
callOnAllChangesDone: boolean;
|
2015-05-11 17:59:39 -07:00
|
|
|
changeDetection: string;
|
2015-05-18 11:57:20 -07:00
|
|
|
constructor({
|
|
|
|
id, selector, compileChildren, events, hostListeners, hostProperties,
|
2015-05-11 17:59:39 -07:00
|
|
|
hostAttributes, hostActions, properties, readAttributes, type,
|
|
|
|
callOnDestroy, callOnChange, callOnAllChangesDone,
|
|
|
|
changeDetection
|
2015-05-18 11:57:20 -07:00
|
|
|
}:{
|
|
|
|
id?:string, selector?:string, compileChildren?:boolean, events?:List<string>, hostListeners?:Map<string, string>, hostProperties?:Map<string, string>,
|
|
|
|
hostAttributes?:Map<string, string>, hostActions?:Map<string, string>, properties?:Map<string, string>, readAttributes?:List<string>, type?:number,
|
|
|
|
callOnDestroy?:boolean, callOnChange?:boolean, callOnAllChangesDone?:boolean,
|
|
|
|
changeDetection?:string
|
2015-05-11 17:59:39 -07:00
|
|
|
}) {
|
2015-03-23 14:10:55 -07:00
|
|
|
this.id = id;
|
|
|
|
this.selector = selector;
|
|
|
|
this.compileChildren = isPresent(compileChildren) ? compileChildren : true;
|
2015-05-11 17:59:39 -07:00
|
|
|
this.events = events;
|
2015-04-09 21:20:11 +02:00
|
|
|
this.hostListeners = hostListeners;
|
2015-04-21 11:47:53 -07:00
|
|
|
this.hostProperties = hostProperties;
|
2015-05-01 13:41:56 +02:00
|
|
|
this.hostAttributes = hostAttributes;
|
2015-05-11 12:31:16 -07:00
|
|
|
this.hostActions = hostActions;
|
2015-04-09 21:20:11 +02:00
|
|
|
this.properties = properties;
|
2015-04-02 14:40:49 -07:00
|
|
|
this.readAttributes = readAttributes;
|
2015-03-23 14:10:55 -07:00
|
|
|
this.type = type;
|
2015-05-11 17:59:39 -07:00
|
|
|
this.callOnDestroy = callOnDestroy;
|
|
|
|
this.callOnChange = callOnChange;
|
|
|
|
this.callOnAllChangesDone = callOnAllChangesDone;
|
|
|
|
this.changeDetection = changeDetection;
|
2015-03-23 14:10:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-06 11:22:28 -07:00
|
|
|
// An opaque reference to a DomProtoView
|
2015-05-18 11:57:20 -07:00
|
|
|
export class RenderProtoViewRef {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
2015-05-06 11:22:28 -07:00
|
|
|
// An opaque reference to a DomView
|
2015-05-18 11:57:20 -07:00
|
|
|
export class RenderViewRef {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
2015-04-09 21:20:11 +02:00
|
|
|
export class ViewDefinition {
|
2015-03-23 14:10:55 -07:00
|
|
|
componentId: string;
|
|
|
|
absUrl: string;
|
2015-04-09 21:20:11 +02:00
|
|
|
template: string;
|
2015-03-23 14:10:55 -07:00
|
|
|
directives: List<DirectiveMetadata>;
|
|
|
|
|
2015-05-18 11:57:20 -07:00
|
|
|
constructor({
|
|
|
|
componentId, absUrl, template, directives
|
|
|
|
}:{
|
|
|
|
componentId?:string, absUrl?:string, template?:string, directives?:List<DirectiveMetadata>
|
|
|
|
}) {
|
2015-03-23 14:10:55 -07:00
|
|
|
this.componentId = componentId;
|
|
|
|
this.absUrl = absUrl;
|
2015-04-09 21:20:11 +02:00
|
|
|
this.template = template;
|
2015-03-23 14:10:55 -07:00
|
|
|
this.directives = directives;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-06 10:17:38 -07:00
|
|
|
export class RenderCompiler {
|
2015-04-15 21:51:30 -07:00
|
|
|
/**
|
|
|
|
* Creats a ProtoViewDto that contains a single nested component with the given componentId.
|
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
compileHost(directiveMetadata: DirectiveMetadata): Promise<ProtoViewDto> { return null; }
|
2015-04-15 21:51:30 -07:00
|
|
|
|
2015-03-23 14:10:55 -07:00
|
|
|
/**
|
2015-05-06 11:22:28 -07:00
|
|
|
* Compiles a single DomProtoView. Non recursive so that
|
2015-03-23 14:10:55 -07:00
|
|
|
* we don't need to serialize all possible components over the wire,
|
|
|
|
* but only the needed ones based on previous calls.
|
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
compile(template: ViewDefinition): Promise<ProtoViewDto> { return null; }
|
2015-05-06 10:17:38 -07:00
|
|
|
}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
2015-05-06 10:17:38 -07:00
|
|
|
export class Renderer {
|
2015-03-23 14:10:55 -07:00
|
|
|
/**
|
2015-05-15 09:55:43 -07:00
|
|
|
* Creates a root host view that includes the given element.
|
2015-05-18 11:57:20 -07:00
|
|
|
* @param {RenderProtoViewRef} hostProtoViewRef a RenderProtoViewRef of type
|
|
|
|
* ProtoViewDto.HOST_VIEW_TYPE
|
|
|
|
* @param {any} hostElementSelector css selector for the host element (will be queried against the
|
|
|
|
* main document)
|
2015-05-06 10:49:42 -07:00
|
|
|
* @return {RenderViewRef} the created view
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
createRootHostView(hostProtoViewRef: RenderProtoViewRef,
|
|
|
|
hostElementSelector: string): RenderViewRef {
|
2015-05-06 10:49:42 -07:00
|
|
|
return null;
|
|
|
|
}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-15 09:55:43 -07:00
|
|
|
* Detaches a free host view's element from the DOM.
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
detachFreeHostView(parentHostViewRef: RenderViewRef, hostViewRef: RenderViewRef) {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-06 10:49:42 -07:00
|
|
|
* Creates a regular view out of the given ProtoView
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
createView(protoViewRef: RenderProtoViewRef): RenderViewRef { return null; }
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-06 10:49:42 -07:00
|
|
|
* Destroys the given view after it has been dehydrated and detached
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
destroyView(viewRef: RenderViewRef) {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
2015-04-15 21:51:30 -07:00
|
|
|
/**
|
2015-05-06 10:49:42 -07:00
|
|
|
* Attaches a componentView into the given hostView at the given element
|
2015-04-15 21:51:30 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
attachComponentView(hostViewRef: RenderViewRef, elementIndex: number,
|
|
|
|
componentViewRef: RenderViewRef) {}
|
2015-04-15 21:51:30 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-06 10:49:42 -07:00
|
|
|
* Detaches a componentView into the given hostView at the given element
|
2015-04-15 21:51:30 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
detachComponentView(hostViewRef: RenderViewRef, boundElementIndex: number,
|
|
|
|
componentViewRef: RenderViewRef) {}
|
2015-04-15 21:51:30 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-18 11:57:20 -07:00
|
|
|
* Attaches a view into a ViewContainer (in the given parentView at the given element) at the
|
|
|
|
* given index.
|
2015-04-15 21:51:30 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
attachViewInContainer(parentViewRef: RenderViewRef, boundElementIndex: number, atIndex: number,
|
|
|
|
viewRef: RenderViewRef) {}
|
2015-04-15 21:51:30 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-18 11:57:20 -07:00
|
|
|
* Detaches a view into a ViewContainer (in the given parentView at the given element) at the
|
|
|
|
* given index.
|
2015-04-15 21:51:30 -07:00
|
|
|
*/
|
2015-05-06 10:49:42 -07:00
|
|
|
// TODO(tbosch): this should return a promise as it can be animated!
|
2015-05-18 11:57:20 -07:00
|
|
|
detachViewInContainer(parentViewRef: RenderViewRef, boundElementIndex: number, atIndex: number,
|
|
|
|
viewRef: RenderViewRef) {}
|
2015-04-15 21:51:30 -07:00
|
|
|
|
2015-03-23 14:10:55 -07:00
|
|
|
/**
|
2015-05-18 11:57:20 -07:00
|
|
|
* Hydrates a view after it has been attached. Hydration/dehydration is used for reusing views
|
|
|
|
* inside of the view pool.
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
hydrateView(viewRef: RenderViewRef) {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-18 11:57:20 -07:00
|
|
|
* Dehydrates a view after it has been attached. Hydration/dehydration is used for reusing views
|
|
|
|
* inside of the view pool.
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
dehydrateView(viewRef: RenderViewRef) {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-11 12:31:16 -07:00
|
|
|
* Sets a property on an element.
|
2015-05-06 10:49:42 -07:00
|
|
|
* Note: This will fail if the property was not mentioned previously as a host property
|
|
|
|
* in the ProtoView
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
setElementProperty(viewRef: RenderViewRef, elementIndex: number, propertyName: string,
|
|
|
|
propertyValue: any) {}
|
2015-05-06 10:49:42 -07:00
|
|
|
|
2015-05-11 12:31:16 -07:00
|
|
|
/**
|
|
|
|
* Calls an action.
|
|
|
|
* Note: This will fail if the action was not mentioned previously as a host action
|
|
|
|
* in the ProtoView
|
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
callAction(viewRef: RenderViewRef, elementIndex: number, actionExpression: string,
|
|
|
|
actionArgs: any) {}
|
2015-05-11 12:31:16 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-06 10:49:42 -07:00
|
|
|
* Sets the value of a text node.
|
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
setText(viewRef: RenderViewRef, textNodeIndex: number, text: string) {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
2015-05-06 10:49:42 -07:00
|
|
|
* Sets the dispatcher for all events of the given view
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
setEventDispatcher(viewRef: RenderViewRef, dispatcher: EventDispatcher) {}
|
2015-03-23 14:10:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A dispatcher for all events happening in a view.
|
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
export interface EventDispatcher {
|
2015-03-23 14:10:55 -07:00
|
|
|
/**
|
|
|
|
* Called when an event was triggered for a on-* attribute on an element.
|
2015-04-07 20:54:20 -07:00
|
|
|
* @param {Map<string, any>} locals Locals to be used to evaluate the
|
2015-03-23 14:10:55 -07:00
|
|
|
* event expressions
|
|
|
|
*/
|
2015-05-18 11:57:20 -07:00
|
|
|
dispatchEvent(elementIndex: number, eventName: string, locals: Map<string, any>);
|
2015-03-23 14:10:55 -07:00
|
|
|
}
|