References
edit this pageWithin this documentation, and within Ractive's code, a reference is a string that refers to a piece of data - in other words, within a {{name}}
mustache, name
is the reference.
By themselves, references are useless - they must resolve to a keypath before we can do anything with them (like render their value). If the reference exists with a section mustache, it may need to be resolved in the context of that section. In fact, because sections can be nested, we have to resolve each reference within its context stack.
The resolution algorithm looks like this:
- If the context stack is empty, skip to (6).
- Find the innermost context on the current context stack.
- See if the keypath which is
context + '.' + reference
points to a value - If so, resolve with that keypath
- Otherwise, remove the innermost context from the stack. Repeat (1-5).
- See if
reference
is a valid keypath by itself. If so, resolve. - If we couldn't resolve the reference, add it to the 'pending resolution' pile. Each time the data changes, repeat steps (1-7).
Huh?
That's a little bit abstract. The following example may help explain:
{{#user}}
<p>Welcome back, {{name}}!
{{#messages}}
You have {{unread}} unread of {{total}} total messages.
You last logged in on {{lastLogin}}.
{{/messages}}
</p>
{{/user}}
ractive = new Ractive({
el: container,
template: myTemplate,
data: {
user: {
name: 'Jim',
messages: {
total: 10,
unread: 3
},
lastLogin: 'Wednesday'
}
}
});
We start with an empty context stack, so to resolve the user
in {{#user}}
we skip ahead to step 6 of the algorithm. Is user
a valid keypath - i.e. does ractive.data
have a user
property? Why yes, it does.
Within the {{#user}}
section, user
is a context. So we now have a non-empty context stack - ['user']
. So when we come to resolve the name
in {{name}}
, we go to step 2. The innermost (and only) context is user
, so we test the keypath user.name
. Is it valid? Why yes, it is. So the name
resolves to user.name
.
Next up, {{#messages}}
. The innermost context is still user
, so we test user.messages
- bingo. Because it's a section mustache, anything inside it now has a two-level context stack (['user', 'user.messages']
).
We take {{unread}}
and {{total}}
and apply the same algorithm - sure enough, they resolve to user.messages.unread
and user.messages.total
.
What about {{lastLogin}}
? Again, we take the innermost context from the stack - user.messages
. Is user.messages.lastLogin
a valid keypath? No, it isn't. So we take the next innermost context - user
. As it turns out, user.lastLogin
is a valid keypath, so the reference resolves.
Most of the time you don't need to be aware that this is going on, especially if you're already familiar with Mustache, but it's useful to have a background understanding.
Lists
This is also how list sections work. Consider the following:
{{#items}}
{{content}}
{{/items}}
ractive = new Ractive({
el: container,
template: myTemplate,
data: {
items: [{ content: 'zero' }, { content: 'one' }, { content: 'two' }]
}
});
The contents of the {{#items}}
section are rendered once for each member of items
. Each time round, the context changes - the first time it is items.0
, then it is items.1
, then it is items.2
.
With a context of items.0
, the content
reference resolves to items.0.content
, and so on.
The implicit iterator - {{.}}
Sometimes you will have lists of primitives (i.e. strings and numbers) rather than objects. To refer to the primitives, we must use the implicit iterator:
{{#items}}
{{.}}
{{/items}}
ractive = new Ractive({
el: container,
template: myTemplate,
data: {
items: [ 'zero', 'one', 'two' ]
}
});
Whenever Ractive sees {{.}}
, it simply resolves it to the innermost context - items.0
, items.1
, items.2
.
In Ractive (but not Mustache), you can use {{this}}
in place of {{.}}
. This is because it looks better in expressions - {{this.toFixed(1)}}
looks much nicer than {{..toFixed(1)}}
(for example).
Restricted references
Sometimes you may want to reference a property within the current context regardless of whether that property currently exists. For those situations you can use what's called a restricted reference (note that this feature does not exist in vanilla Mustache):
{{#options}}
<label><input type='checkbox' value='{{.selected}}'> {{description}}</label>
{{/options}}
This feature is particularly useful with two-way binding as these references have to be resolved immediately.