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:
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
export class ElementBinder {
|
|
|
|
index:number;
|
|
|
|
parentIndex:number;
|
|
|
|
distanceToParent:number;
|
|
|
|
directives:List<DirectiveBinder>;
|
|
|
|
nestedProtoView:ProtoView;
|
|
|
|
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
|
|
|
|
eventBindings: Map<string, ASTWithSource>;
|
|
|
|
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-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 {
|
|
|
|
// Index into the array of directives in the Template instance
|
|
|
|
directiveIndex:any;
|
|
|
|
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
|
|
|
|
eventBindings: Map<string, ASTWithSource>;
|
|
|
|
constructor({
|
|
|
|
directiveIndex, propertyBindings, eventBindings
|
|
|
|
}) {
|
|
|
|
this.directiveIndex = directiveIndex;
|
|
|
|
this.propertyBindings = propertyBindings;
|
|
|
|
this.eventBindings = eventBindings;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export class ProtoView {
|
|
|
|
render: ProtoViewRef;
|
|
|
|
elementBinders:List<ElementBinder>;
|
|
|
|
variableBindings: Map<string, string>;
|
|
|
|
|
2015-04-02 14:40:49 -07:00
|
|
|
constructor({render, elementBinders, variableBindings}={}) {
|
2015-03-23 14:10:55 -07:00
|
|
|
this.render = render;
|
|
|
|
this.elementBinders = elementBinders;
|
|
|
|
this.variableBindings = variableBindings;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export class DirectiveMetadata {
|
|
|
|
static get DECORATOR_TYPE() { return 0; }
|
|
|
|
static get COMPONENT_TYPE() { return 1; }
|
|
|
|
static get VIEWPORT_TYPE() { return 2; }
|
|
|
|
id:any;
|
|
|
|
selector:string;
|
|
|
|
compileChildren:boolean;
|
|
|
|
events:Map<string, string>;
|
|
|
|
bind:Map<string, string>;
|
|
|
|
setters:List<string>;
|
2015-04-02 14:40:49 -07:00
|
|
|
readAttributes:List<string>;
|
2015-03-23 14:10:55 -07:00
|
|
|
type:number;
|
2015-04-02 14:40:49 -07:00
|
|
|
constructor({id, selector, compileChildren, events, bind, setters, readAttributes, type}) {
|
2015-03-23 14:10:55 -07:00
|
|
|
this.id = id;
|
|
|
|
this.selector = selector;
|
|
|
|
this.compileChildren = isPresent(compileChildren) ? compileChildren : true;
|
|
|
|
this.events = events;
|
|
|
|
this.bind = bind;
|
|
|
|
this.setters = setters;
|
2015-04-02 14:40:49 -07:00
|
|
|
this.readAttributes = readAttributes;
|
2015-03-23 14:10:55 -07:00
|
|
|
this.type = type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// An opaque reference to a ProtoView
|
|
|
|
export class ProtoViewRef {}
|
|
|
|
|
|
|
|
// An opaque reference to a View
|
|
|
|
export class ViewRef {}
|
|
|
|
|
|
|
|
export class ViewContainerRef {
|
|
|
|
view:ViewRef;
|
|
|
|
elementIndex:number;
|
|
|
|
constructor(view:ViewRef, elementIndex: number) {
|
|
|
|
this.view = view;
|
|
|
|
this.elementIndex = elementIndex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export class Template {
|
|
|
|
componentId: string;
|
|
|
|
absUrl: string;
|
|
|
|
inline: string;
|
|
|
|
directives: List<DirectiveMetadata>;
|
|
|
|
|
|
|
|
constructor({componentId, absUrl, inline, directives}) {
|
|
|
|
this.componentId = componentId;
|
|
|
|
this.absUrl = absUrl;
|
|
|
|
this.inline = inline;
|
|
|
|
this.directives = directives;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export class Renderer {
|
|
|
|
/**
|
|
|
|
* Compiles a single ProtoView. Non recursive so that
|
|
|
|
* we don't need to serialize all possible components over the wire,
|
|
|
|
* but only the needed ones based on previous calls.
|
|
|
|
*/
|
|
|
|
compile(template:Template):Promise<ProtoView> { return null; }
|
|
|
|
|
|
|
|
/**
|
2015-04-07 17:24:09 -07:00
|
|
|
* Sets the preset nested components,
|
2015-03-23 14:10:55 -07:00
|
|
|
* which will be instantiated when this protoView is instantiated.
|
2015-04-07 17:24:09 -07:00
|
|
|
* Note: We can't create new ProtoViewRefs here as we need to support cycles / recursive components.
|
2015-03-23 14:10:55 -07:00
|
|
|
* @param {List<ProtoViewRef>} protoViewRefs
|
|
|
|
* ProtoView for every element with a component in this protoView or in a view container's protoView
|
|
|
|
*/
|
2015-04-07 17:24:09 -07:00
|
|
|
mergeChildComponentProtoViews(protoViewRef:ProtoViewRef, componentProtoViewRefs:List<ProtoViewRef>) { return null; }
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Creats a ProtoView that will create a root view for the given element,
|
|
|
|
* i.e. it will not clone the element but only attach other proto views to it.
|
2015-04-07 17:24:09 -07:00
|
|
|
* Contains a single nested component with the given componentId.
|
2015-03-23 14:10:55 -07:00
|
|
|
*/
|
2015-04-07 17:24:09 -07:00
|
|
|
createRootProtoView(selectorOrElement, componentId):Promise<ProtoView> { return null; }
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a view and all of its nested child components.
|
|
|
|
* @return {List<ViewRef>} depth first list of nested child components
|
|
|
|
*/
|
|
|
|
createView(protoView:ProtoViewRef):List<ViewRef> { return null; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Destroys a view and returns it back into the pool.
|
|
|
|
*/
|
|
|
|
destroyView(view:ViewRef):void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Inserts a detached view into a viewContainer.
|
|
|
|
*/
|
|
|
|
insertViewIntoContainer(vcRef:ViewContainerRef, view:ViewRef, atIndex):void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Detaches a view from a container so that it can be inserted later on
|
|
|
|
* Note: We are not return the ViewRef as this can't be done in sync,
|
|
|
|
* so we assume that the caller knows which view is in which spot...
|
|
|
|
*/
|
|
|
|
detachViewFromContainer(vcRef:ViewContainerRef, atIndex:number):void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets a property on an element.
|
|
|
|
* Note: This will fail if the property was not mentioned previously as a propertySetter
|
|
|
|
* in the Template.
|
|
|
|
*/
|
|
|
|
setElementProperty(view:ViewRef, elementIndex:number, propertyName:string, propertyValue:any):void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Installs a nested component in another view.
|
|
|
|
* Note: only allowed if there is a dynamic component directive
|
|
|
|
*/
|
|
|
|
setDynamicComponentView(view:ViewRef, elementIndex:number, nestedViewRef:ViewRef):void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This will set the value for a text node.
|
|
|
|
* Note: This needs to be separate from setElementProperty as we don't have ElementBinders
|
|
|
|
* for text nodes in the ProtoView either.
|
|
|
|
*/
|
|
|
|
setText(view:ViewRef, textNodeIndex:number, text:string):void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the dispatcher for all events that have been defined in the template or in directives
|
|
|
|
* in the given view.
|
|
|
|
*/
|
2015-04-07 20:54:20 -07:00
|
|
|
setEventDispatcher(viewRef:ViewRef, dispatcher:any/*EventDispatcher*/):void {}
|
2015-03-23 14:10:55 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* To be called at the end of the VmTurn so the API can buffer calls
|
|
|
|
*/
|
|
|
|
flush():void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A dispatcher for all events happening in a view.
|
|
|
|
*/
|
|
|
|
export class EventDispatcher {
|
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
dispatchEvent(
|
2015-04-07 20:54:20 -07:00
|
|
|
elementIndex:number, eventName:string, locals:Map<string, any>
|
2015-03-23 14:10:55 -07:00
|
|
|
):void {}
|
|
|
|
}
|