From ba187de334e840128dc6a087267156272b4aa8f1 Mon Sep 17 00:00:00 2001 From: Michael Brade Date: Sat, 19 Sep 2015 21:26:55 -0700 Subject: [PATCH 1/3] add support for method calls on the result of a model expression --- lib/expressions.js | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/lib/expressions.js b/lib/expressions.js index c22441a..465837f 100644 --- a/lib/expressions.js +++ b/lib/expressions.js @@ -515,10 +515,13 @@ FnExpression.prototype.apply = function(context, extraInputs) { return out; }; FnExpression.prototype._lookupParent = function(context) { + // Lookup function in the model + var segments = this.parentSegments; + var parent = (segments) ? lookup(segments, context.controller.model.data) : undefined; + if (parent && parent[this.lastSegment]) return parent; // Lookup function on current controller var controller = context.controller; - var segments = this.parentSegments; - var parent = (segments) ? lookup(segments, controller) : controller; + parent = (segments) ? lookup(segments, controller) : controller; if (parent && parent[this.lastSegment]) return parent; // Otherwise lookup function on page var page = controller.page; From c682619cbb9620dfb8f268941dd4111fb23d69d1 Mon Sep 17 00:00:00 2001 From: Michael Brade Date: Tue, 29 Sep 2015 18:34:31 -0700 Subject: [PATCH 2/3] FnExpressions: actually check that it is a function we are about to call --- lib/expressions.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/lib/expressions.js b/lib/expressions.js index 465837f..bfcce17 100644 --- a/lib/expressions.js +++ b/lib/expressions.js @@ -518,20 +518,20 @@ FnExpression.prototype._lookupParent = function(context) { // Lookup function in the model var segments = this.parentSegments; var parent = (segments) ? lookup(segments, context.controller.model.data) : undefined; - if (parent && parent[this.lastSegment]) return parent; - // Lookup function on current controller + if (parent && typeof parent[this.lastSegment] == 'function') return parent; + // Otherwise lookup function on current controller var controller = context.controller; parent = (segments) ? lookup(segments, controller) : controller; - if (parent && parent[this.lastSegment]) return parent; + if (parent && typeof parent[this.lastSegment] == 'function') return parent; // Otherwise lookup function on page var page = controller.page; if (controller !== page) { parent = (segments) ? lookup(segments, page) : page; - if (parent && parent[this.lastSegment]) return parent; + if (parent && typeof parent[this.lastSegment] == 'function') return parent; } // Otherwise lookup function on global parent = (segments) ? lookup(segments, global) : global; - if (parent && parent[this.lastSegment]) return parent; + if (parent && typeof parent[this.lastSegment] == 'function') return parent; // Throw if not found throw new Error('Function not found for: ' + this.segments.join('.')); }; From 601164a4f20af74ba636f24ca97c152e2315a5a9 Mon Sep 17 00:00:00 2001 From: Michael Brade Date: Fri, 15 Jan 2016 17:55:40 -0800 Subject: [PATCH 3/3] README - I like documentation, so I started writing.... --- README.md | 298 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 298 insertions(+) diff --git a/README.md b/README.md index e69de29..8fe6e4a 100644 --- a/README.md +++ b/README.md @@ -0,0 +1,298 @@ + +# API + +`derby-templates` exports four members: + + * a context api (`contexts`) + * an expression api (`expressions`) + * operator functions (`operatorFns`) + * and a template api (`templates`) + +Each of those are explained in the following sections. + + +## Contexts + +The `contexts` member has two prototypes: + +* `ContextMeta` +* `Context` + +#### Context + +A context used for TODO + +A context always has two properties: + +* `meta`: properties which are globally inherited for the entire page +* `controller`: The page or component. Must have a `model` property with a `data` property. + + +Optionally, the following properties may be defined: + +* `parent`: containing context +* `unbound`: boolean set to true when bindings should be ignored +* `expression`: the expression for a block +* `alias`: alias name for the given expression (`ExpressionMeta::as`) +* `keyAlias`: alias name for the index or iterated key (`ExpressionMeta::keyAs`) +* `item`: for `Context::eachChild`: the index of the `each` at render time + +* for `Context::viewChild`: + + - `view`: reference to the current view + - `attributes`: attribute values passed to the view instance + - `hooks`: MarkupHooks to be called after insert into DOM of component + - `initHooks`: MarkupHooks to be called immediately before init of component + +* `_id`: used in `EventModel` + + + +## Expressions + +The `expressions` member has four functions: + +* lookup +* templateTruthy +* pathSegments +* renderValue + +and all possible expression prototypes: + +* literals +* paths +* relative paths +* aliases +* attribute paths +* bracket expressions +* ... + +See below for the details. + + +#### lookup +#### templateTruthy + + +#### pathSegments + +Iterate through an array of segments and return a new array with each `segment`, or `segment.item` if available. + +#### renderValue + + + +#### `_outerDependency` + + +#### `ExpressionMeta` + +In Derby, templates can contain expressions introduced by `{{ }}`. `ExpressionMeta` holds everything but the path of the +expression. + +* `source`: the full original expression between `{{ }}` +* `blockType`: if the expression starts a block, one of: `if`, `else if`, `else`, `unless`, `each`, `bound`, `unbound`, `with`, `on` +* `isEnd`: `true` if it was an end tag (`{{/}}` or `{{/if}}`), `false` otherwise +* `as`: the name of the alias to be created; starts with `#` (e.g. `#alias` if expression ended with `as #alias`) +* `keyAs`: in case of `each ... as #v, #i`, this is name for the index key or iterated key, `#i` in this case +* `unescaped`: +* `bindType`: +* `valueType`: + + +#### Expression + +Every expression inherits the `Expression` prototype. + +TODO: serialize() + + +#### LiteralExpression + +Represents any valid JavaScript literal expression. Examples: +```js +56.8e-4 // floating-point +"string" // String +["arr", "ay"] // Array +{ key: "value" } // Object +true // boolean +/a+b/ // RegExp +``` + + + +#### PathExpression + +Represents a simple JSON path expression to lookup a value in an object hierarchy. Steps in the hierarchy are called +"segment" and are separated by a dot. Example: +``` +_page.color +``` + +```js +new PathExpression(['_page', 'color']) +``` + + + +#### RelativePathExpression + +Relative view paths begin with `this`. They refer to the expression in the containing block. + +``` +this.color +``` + +```js +new RelativePathExpression(['color']) +``` + + +#### AliasPathExpression + +Aliases label path expressions. They must begin with a hash (`#`) character to make it more obvious whether a path is an +alias or a model value. Each of the block types support defining aliases with the `as` keyword. + +Aliases are more explicit than relative paths and make it possible to refer to the scope of a parent block. + +``` +#page.color +``` + +```js +new AliasPathExpression('#page', ['color']) +``` + +#### AttributePathExpression + +Views can be passed values as attributes. These attributes are accessed via paths that start with an at sign (`@`). +Example: + +``` +@style.width +``` + +creates the following `AttributePathExpression`: + +```js +new AttributePathExpression('style', ['width']) +``` + + +#### BracketsExpression + +Represents an expression that contains member access through brackets. The top-level `BracketsExpression` represents the +last set of brackets. + +CTOR: TODO + +Example: + +```js +obj[style].en[v] +``` + +would be represented by: + +```js +new BracketsExpression( + new BracketsExpression( + new PathExpression(['obj']), + new PathExpression(['style']), + ['en']), + new expressions.PathExpression(['v'])) +``` + + + +#### FnExpression + +#### OperatorExpression + +Inherits `FnExpression`. + +#### NewExpression + +Inherits `FnExpression`. + +#### SequenceExpression + +Inherits `OperatorExpression`. + + +#### ScopedModelExpression + + + + + + +## operatorFns + +Exports two members: `get` and `set`. + +get: +* !U -U +U ~U typeofU +* || && | ^ & == != === !== < > <= >= instanceof in << >> >>> + - * / % ? , [] {} + +set: +* !U -U +* == === in + - * / + + + + + + + +## templates + +Template objects can directly create HTML for server rendering or DOM nodes for client rendering. + +The `templates` member exposes the following properties. + +* all `saddle` properties + +#### View (inherits saddle.Template) + +* `View(views, name, source, options)`: CTOR, called by `Views::register()`, for instance. + * `views`: the `Views` object, there is only one per App. + * `name`: + * `source`: + * `options`: +* `type`: constant string: `'View'` +* `get(context, unescaped)`: return the html of the view in the given `context` +* `getFragment(context, binding)`: return the view as a DocumentFragment in the given `context`. `binding`: TODO +* `appendTo`: +* `attachTo`: +* `dependencies`: +* `parse`: implemented only if `derby-parsing` is loaded + + + + +#### ViewInstance +#### DynamicViewInstance +#### ParentWrapper + +#### Views + +Manages all views for a Derby app. + +* find +* `register(name, source, options)`: register a new view. See `View` CTOR for explanation of arguments. +* `serialize()` +* findErrorMessage + + +#### MarkupHook +#### ElementOn +#### ComponentOn +#### ComponentMarker +#### AsProperty +#### AsObject +#### AsObjectComponent +#### AsArray +#### AsArrayComponent + +#### emptyTemplate