Components
edit this pageIn many situations, you want to encapsulate behaviour and markup into a single reusable component, which can be dropped in to Ractive applications.
First, you need to create the component with Ractive.extend(), which allows you to define a template, initialisation behaviour, default data, and so on. (If you're not familiar with Ractive.extend() you should read that page first.)
- Initiliasation Options
- Binding
- Events
- The {{>content}} Directive
- The {{yield}} Directive
- Pseudo-Dynamic Templating
Initialisation Options
beforeInit Function
beforeInit will be passed all the data that will initialize the particular component. This is where you should modify the component with dynamic logic. This could include setting/changing partials, adding decorators, adding other subcomponents, or anything you would use to initialise a normal Ractive instance. This is great for dynamic templating, an example can be found below.
init Function
Init is called after the component has been initialised. This means that this
will reference that component instance. This allows you to observe on, or listen on events/keypaths for this particular component.
isolated Boolean
Isolated is defaulted to false. This applies to your template scope being isolated to only the data that is passed in. For example if you wanted to access a particular variable on your parent data as if it existed on your component then setting isolated false would allow this. However if you wanted your template scoped to only data you give it so that everything was very module then set isolated to true.
Setting isolated to false
var Component = Ractive.extend({
isolated: false,
template: '#component'
});
var ractive = new Ractive({
el: 'body',
template: '#template',
data: {
nonIsolatedSetting: 'This is a non isolated setting, get it anywhere'
}
});
<script id="template" type="text/ractive">
<div>
<Component>
</div>
</script>
<script id="component" type="text/ractive">
{{nonIsolatedString}}
</script>
The result of this would be
<div>
This is a non isolated setting, get it anywhere
</div>
But if we used the same example and set isolated to true
var Component = Ractive.extend({
isolated: true,
template: '#template'
})
Then the resulting output would be
<div>
</div>
Very complex example for such a simple idea but this could drastically effect you.
init is called after the extended component has been initialized
var MyWidget = Ractive.extend({
template: '<div on-click="activate">{{message}}</div>',
init: function () {
this.on( 'activate', function () {
alert( 'Activating!' );
});
},
data: {
message: 'No message specified, using the default'
}
});
Then, you need to register the component by adding it to Ractive.components
:
Ractive.components.widget = MyWidget;
// or, if you don't want to make it globally available,
// you can use it with only a single Ractive instance
var ractive = new Ractive({
el: 'body',
template: mainTemplate,
components: {
widget: MyWidget
}
});
Finally, you use the component by referring to it within another template:
<div class='application'>
<h1>I've got a widget</h1>
<widget message='Click to activate!'/>
</div>
Using the 'widget' component like this is more or less equivalent to doing the following:
new MyWidget({
el: document.querySelector( '.application' ),
append: true, // so the component doesn't nuke the <h1>
data: {
message: 'Click to activate!'
}
});
Binding
Instead of passing a static message ('Click to activate!') through, we could have passed a dynamic one:
<widget message='{{foo}}'/>
In this case, the value of message
within the component would be bound to the value of foo
in the parent Ractive instance. When the value of parent-foo
changes, the value of child-message
also changes, and vice-versa.
Data Context
A new data context gets created for the component, so any parameters don't pollute the primary data (bound values still update):
var data = {
name: 'Colors',
colors: ['red', 'blue', 'yellow'],
style: 'tint'
}
Ractive.components.widget = Ractive.extend({})
var ractive = new Ractive({
template: '<widget items='{{colors}}' option1='A' option2='{{style}}'/>',
data: data
})
function logData(){
console.log('main', JSON.stringify(ractive.data))
console.log('widget', JSON.stringify(ractive.findComponent('widget').data))
}
logData()
// main {"name":"Colors","colors":["red","blue","yellow"],"style":"tint"}
// widget {"items":["red","blue","yellow"],"option1":"A","option2":"tint"}
ractive.set('colors.1', 'green')
logData()
// main {"name":"Colors","colors":["red","green","yellow"],"style":"tint"}
// widget {"items":["red","green","yellow"],"option1":"A","option2":"tint"}
Events
You can subscribe to events that emanate from a component in the normal way. For example, our widget template contains a <div>
element with an on-click="activate"
directive. We're listening for that event inside the MyWidget init
method, but we can also listen for it like so:
<widget on-activate="doSomething"/>
ractive.on( 'doSomething', function () {
alert( 'Widget activated!' );
});
(The same goes for any events that are fired programmatically by the component, rather than as a result of user interaction.)
Bubbling
Events fired from within components will also bubble up the component hierarchy with their component name attached as a namespace. For instance, if a component referenced in a template as Foo
fires an event named myFoo
, then all of the Ractive instances in the hierarchy above the Foo
will also have a Foo.myFoo
event fired. This can be used to avoid having to reproxy events at each level in a deeply nested component hierarchy.
Namespaced events may be subscribed in a two ways:
- Explicitly
ractive.on('Foo.myFoo', function() { // ... });
- As a wildcard
ractive.on('*.myFoo', function() { // ... });
Using wildcards will subscribe to any events with the given name, regardless of their namespace. Multiple instances of the same component can be namespaced separately in the same hierarchy by adding the same component with multiple names.
var component = Ractive.extend({});
Ractive.components.Foo = component;
Ractive.components.Bar = component;
Here, <Foo />
and <Bar />
will both be instances of component
, but any events bubbled out of <Foo />
will have Foo.
as their namespace and events from <Bar />
will have Bar.
as their namespace.
Stop bubbling
If you don't want the event to bubble any further up the component hierarchy, simply return false from the event handler function. This will also stopPropagation()
and preventDefault()
on the original
event if there is one.
Additionally, events proxied within a component will stop events from internal components from bubbling further upwards. Within components, proxy event attributes may use wildcards.
<Foo on-Bar.foo="barfooed" on-Baz.*="baz-event" on-*.bippy="any-bippy">
<Bar /><Baz /><Baz />
</Foo>
Here, Bar.foo
, Baz.
, and *.bippy
events will not bubble beyond the Foo
component. At this point though, Foo
events will be bubbled for barfooed
, baz-event
, and any-bippy
. To completely stop events from bubbling, you can pass an empty handler.
<Foo on-*.*="">
<Bar /><Baz /><Baz />
</Foo>
Here, no internal component events will bubble above Foo
.
The {{>content}}
Directive
Any content in the template calling the component:
<list items='{{gunas}}''>
{{name}} ({{qualities}})
</list>
<list items='{{fates}}'>
"{{.}}"
</list>
is exposed in the component as a partial named "content":
<!-- template for "list" component -->
<ul>
{{#items}}
<li>{{>content}}</li>
{{/}}
</ul>
A partial, or component, can be used as well:
<list items='{{gunas}}''>
{{>partialA}})
</list>
<list items='{{fates}}''>
<widget value='{{.}}'/>
</list>
Currently, the content markup resolves in the component's context. In the second example above the value
parameter for <widget>
is the individual list item of {{fates}}
. And in the first case, the partial would need to be defined globally (Ractive.partials
) or on the component, as it won't be found if it is defined, for example, inline in the outer context:
<list items="{{fates}}">
{{>partialB}}
</list>
<!-- {{>partialB}} -->
//won't be found by list component,
//because this markup is handed off
//and rendered inside component
<!-- {{/partial}} -->
Be aware that the context scope for resolution may change or be expanded in a future release to control whether it resolves in the calling or component context.
The {{yield}}
Directive
The {{>content}}
directive, as a special partial, renders everthing in the context of the component. The {{yield}}
directive renders everything in the context of the parent, which is typically what the component user expects to happen. Any event handlers or data references will be registered on or refer the parent context instead of the component context.
Any inline templates defined in the template passed to the component, however, will be rendered in the context of the component.
<Foo>
{{ myData }}
<button on-click="clicked">Click</button>
</Foo>
var Foo = Ractive.extend({
template: '<div>{{ yield }}</div>',
data: { myData: 2 }
});
var ractive = new Ractive({
components: { Foo: Foo },
data: { myData: 1 }
});
ractive.on('clicked', function(ev) {
console.log('clicked');
});
Here, 1
will be displayed next to the button, and the button, when clicked, will log a message to the console.
Pseudo Dynamic Templating
In some cases you want to render a different template depending on the data that is passed in.
Form Element Component
var formElement = Ractive.extend({
template: '{{>element}}',
setTemplate: function(options) {
options.partials.element = options.data.template;
},
beforeInit: function(options) {
this.setTemplate(options)
}
});
Form Template
{{#items}}
<formElement value="{{value}}" template="{{template}}"/>
{{/items}}
Ractive Form
var ractive = new Ractive({
el: 'body',
template: formTemplate,
components: {
formElement: formElement
}
data: {
items: [
{
template: '<h1>{{value}}</h1>',
value: 'This is a title'
},
{
template: '<input type="text" value="{{value}}" style="display:block; clear: both;" />',
value: 'Input Value'
},
{
template: '<textarea value="{{value}}"></textarea>',
value: 'Textarea Value'
}
]
}
});
ractive.observe('items.*.value', function(newValue, oldValue, keypath) {
console.log(keypath + ' ' + newValue);
});
Caveat
Components are a stable feature, but still relatively new - certain aspects have yet to be implemented. Pardon the dust! There is a long-running GitHub issue, which may be of interest.