Lightweight templates for React.
- No runtime libraries. No magic. Simply precompile your way to clear React code.
- Easy syntax that's similar to HTML, supported by most IDEs.
- Clear separation of presentation and logic - almost zero HTML in component files.
- Declarative coding ensures that the HTML that you write and the HTML you inspect look nearly identical.
- Supports AMD, CommonJS, ES6, Typescript and globals.
React Templates compiles an *.rt file (react template file - an extended HTML format) into a JavaScript file. This file, which uses AMD syntax, returns a function. When invoked, this function returns a virtual React DOM based on React.DOM elements and custom user components.
A common use case would be that a regular React component would require a JavaScript file generated from a template, and then perform `func.apply(this)`, causing the template to have that component as its context.
http://wix.github.io/react-templates/
https://github.com/wix/generator-react-templates
Here's a sample Hello project:
https://github.com/wix/hello-react-templates
http://plugins.jetbrains.com/plugin/7648
- Any valid HTML (including comments) is a template
- {} to identify JS expression
- Built-in directives:
- styles
- event handlers
Some love JSX, some don't. We don't. More specifically, it seems to us that JSX is only a good fit for components with very little HTML inside. And this can be accomplished by creating DOM elements in code. Also, we like to separate code and HTML because it just feels right.
You can install react-templates using npm:
npm install react-templates -g
rt [file.rt|dir]* [options]
See more on CLI usage here.
In most cases, this package will be wrapped in a build task, so CLI will not be used explicitly:
- Grunt: grunt-react-templates
- Gulp: gulp-react-templates
- Broccoli: broccoli-react-templates
- Browserify plugin: react-templatify
- Webpack loader : react-templates-loader
Any HTML that you write is a valid template, except for inline event handlers ("on" attributes). See the "event handlers" section below for more information.
To embed JavaScript expressions in both attribute values and tag content, encapsulate them in {}. If this is done inside an attribute value, the value still needs to be wrapped in quotes. For directives (see below), {} are not used.
This syntax unfortunately means that it's not possible to write object literals as the first thing in the value of a directive attribute:
<div rt-state="{first: 'Jack', last: 'Sparrow'} as user">
does not work. While this is probably a rare use case, one can hack around it by doing something like
<div rt-state="[{first: 'Jack', last: 'Sparrow'}][0] as user">
<a href="{this.state.linkRef}">{this.state.linkText}</a>
define([
'react',
'lodash'
], function (React, _) {
'use strict';
return function () {
return React.DOM.a({ 'href': this.state.linkRef }, this.state.linkText);
};
});
This lets you add conditions to a subtree of HTML. If the condition evaluates to true, the subtree will be returned; otherwise, it will not be calculated. It is implemented as a ternary expression.
<div rt-if="this.state.resultCode === 200">Success!</div>
define([
'react',
'lodash'
], function (React, _) {
'use strict';
return function () {
return this.state.resultCode === 200 ? React.DOM.div({}, 'Success!') : null;
};
});
The negated version of rt-if
, to aid readability. Simply wraps the condition in !().
Repeats a DOM node with its subtree for each item in an array. The syntax is rt-repeat="itemVar in arrayExpr"
, where the element, itemVar
, will be available in JavaScript context, and an itemVarIndex
will be created to represent the index of the item. By using this naming scheme, repeated expressions have access to all levels of nesting.
<div rt-repeat="myNum in this.getMyNumbers()">{myNumIndex}. {myNum}</div>
define([
'react',
'lodash'
], function (React, _) {
'use strict';
function repeatMyNum1(myNum, myNumIndex) {
return React.DOM.div({}, myNumIndex + '. ' + myNum);
}
return function () {
return _.map(this.getMyNumbers(), repeatMyNum1.bind(this));
};
});
Repeats the children of a DOM node for each item in an array, discarding the wrapper node with the rt-repeat-children
attribute.
The syntax is rt-repeat-children="itemVar in arrayExpr"
, where the element, itemVar
, will be available in JavaScript context,
and an itemVarIndex
will be created to represent the index of the item.
By using this naming scheme, repeated expressions have access to all levels of nesting.
<div rt-scope="[1, 4, 9] as squares">
<div rt-repeat-children="square in squares">
<span>{squareIndex+1}</span>
<span>{square}</span>
</div>
</div>
<div>
<span>1</span>
<span>1</span>
<span>2</span>
<span>4</span>
<span>3</span>
<span>9</span>
</div>
This directive is deprecated in favour of the more semantically useful rt-inner-scope. It is kept around for backwards compatibility.
This directive creates a new JavaScript scope by creating a new method and invoking it with its current context. The syntax is rt-scope="expr1 as var1; expr2 as var2
. This allows for a convenient shorthand to make the code more readable. It also helps to execute an expression only once per scope.
When used together with rt-if
, the rt-scope
mappings are evaluated irrespective of whether the rt-if
condition's value.
This is often inconvenient, so use rt-inner-scope
instead.
When used together with rt-repeat
, the rt-scope
mappings are evaluated outside of the repeat loop,
which means you are often forced to have a nested element just to facilitate an inner rt-scope
.
This is often inconvenient, so use rt-inner-scope
instead.
Because of the underying implementation, it is not possible to reference var1
inside of expr2
,
which is often inconvenient. Use rt-inner-scope
if you want this behaviour.
<div rt-repeat="rpt in array">
<div rt-scope="')' as separator; rpt.val as val">{rptIndex}{separator} {val}</div>
<div>'rpt' exists here, but not 'separator' and 'val'</div>
</div>
define([
'react',
'lodash'
], function (React, _) {
'use strict';
function scopeSeparatorVal1(rpt, rptIndex, separator, val) {
return React.DOM.div({}, rptIndex + separator + ' ' + val);
}
function repeatRpt2(rpt, rptIndex) {
return React.DOM.div({}, scopeSeparatorVal1.apply(this, [
rpt,
rptIndex,
')',
rpt.val
]), React.DOM.div({}, '\'rpt\' exists here, but not \'separator\' and \'val\''));
}
return function () {
return _.map(array, repeatRpt2.bind(this));
};
});
This directive creates a new JavaScript scope by creating a new method and invoking it with its current context.
The syntax is rt-scope="expr1 as var1; expr2 as var2
.
This allows for a convenient shorthand to make the code more readable.
It also helps to execute an expression only once per scope.
Subsequent expressions may reference preceding variables, since generated code declares each alias as a var
(as opposed to a function parameter, which get bound to formal parameter names only after evaluation), so you can do stuff like
<div rt-inner-scope="users[userId] as user; user.profile as profile; profile.avatar as avatar;">
When used with rt-if
, the rt-if
condition is evaluated first, and only if it is truthy, the rt-inner-scope
mappings are processed. This means you can write things like
<div rt-if="user.profile" rt-inner-scope="user.profile.image as image">
without risking accessing a field on an undefined
, or doing something ugly like user.profile && user.profile.image as image
.
When used with rt-repeat
, the rt-inner-scope
is evaluated for every iteration, so that iteration's item
and itemIndex
are in scope.
rt-props is used to inject properties into an element programmatically. It will merge the properties with the properties received in the template. This option allows you to build properties based on external logic and pass them to the template. It is also useful when passing properties set on the component to an element within the template. The expected value of this attribute is an expression returning an object. The keys will be the property name, and the values will be the property values.
<input style="height:10px;width:3px;" rt-props="{style:{width:'5px'},type:'text'}"/>
define([
'react',
'lodash'
], function (React, _) {
'use strict';
return function () {
return React.DOM.input(_.merge({}, {
'style': {
height: '10px',
width: '3px'
}
}, {
style: { width: '5px' },
type: 'text'
}));
};
});
To reduce the boilerplate code when setting class names programatically, you can use the rt-class directive. It expects a JSON object with keys as class names, and a Boolean as the value. If the value is true, the class name will be included.
Note the following:
1. In React templates, you can use the "class" attribute as you would in HTML.
2. If you use both class and rt-class on the same HTML element, they get merged as expected.
<div rt-scope="{blue: true, selected: this.isSelected()} as classes">
These are logically equivalent
<div rt-class="classes">Reference</div>
<div rt-class="{blue: true, selected: this.isSelected()}">Inline</div>
<div class="blue{this.isSelected() ? ' selected' : ''}">Using the class attribute</div>
<div class="blue" rt-class="{selected: this.isSelected()}">Using the combination of class and rt-class</div>
</div>
define([
'react/addons',
'lodash'
], function (React, _) {
'use strict';
function scopeClasses1(classes) {
return React.createElement('div', {}, '\n These are logically equivalent\n ',
React.createElement('div', { 'className': _.keys(_.pick(classes, _.identity)).join(' ') }, 'Reference'),
React.createElement('div', {
'className': _.keys(_.pick({
blue: true,
selected: this.isSelected()
}, _.identity)).join(' ')
}, 'Inline'),
React.createElement('div', { 'className': 'blue' + (this.isSelected() ? ' selected' : '') }, 'Using the class attribute'),
React.createElement('div', { 'className': 'blue' + ' ' + _.keys(_.pick({ selected: this.isSelected() }, _.identity)).join(' ') }, 'Using the combination of class and rt-class'));
}
return function () {
return scopeClasses1.apply(this, [[{
blue: true,
selected: this.isSelected()
}][0]]);
};
});
React templates allow the settings of styles inline in HTML, optionally returning an object from the evaluation context. By default, style names will be converted from hyphen-style to camelCase-style naming.
To embed JavaScript inside a style attribute, single curly braces are used. To embed an entire object, double curly braces are used. Note: When embedding objects, styles must conform to camelCase-style naming.
<div>
These are really equivalent
<div style="color:white; line-height:{this.state.lineHeight}px">Inline</div>
<div style="{{'color': 'white', 'lineHeight': this.state.lineHeight + 'px'}}">Inline</div>
</div>
define([
'react',
'lodash'
], function (React, _) {
'use strict';
return function () {
return React.DOM.div({}, 'These are really equivalent', React.DOM.div({
'style': {
color: 'white',
lineHeight: this.state.lineHeight + 'px'
}
}, 'Inline'), React.DOM.div({
'style': {
'color': 'white',
'lineHeight': this.state.lineHeight + 'px'
}
}, 'Inline'));
};
});
React event handlers accept function references inside of {}, such as onClick="{this.myClickHandler}"
. When functions are not needed, lambda notation can be used, which will create a React template that creates a function for the included code. There is no performance impact, as the function created is bound to the context instead of being recreated.
The lambda notation has the form: `onClick="(evt) => console.log(evt)"`. In this example, **evt** is the name of the first argument passed into the inline function. With browser events, this will most likely be the React synthetic event. However, if you expect a property that starts with **on**Something, then React templates will treat it as an event handler. If you have an event handler called **onBoxSelected** that triggers an event with row and column params, you can write `onBoxSelected="(row, col)=>this.doSomething(row,col)"`. A no-param version is supported as well: `onClick="()=>console.log('just wanted to know it clicked')"`.
<div rt-repeat="item in items">
<div onClick="()=>this.itemSelected(item)" onMouseDown="{this.mouseDownHandler}">
</div>
define([
'react',
'lodash'
], function (React, _) {
'use strict';
function onClick1(item, itemIndex) {
this.itemSelected(item);
}
function repeatItem2(item, itemIndex) {
return React.DOM.div({}, React.DOM.div({
'onClick': onClick1.bind(this, item, itemIndex),
'onMouseDown': this.mouseDownHandler
}));
}
return function () {
return _.map(items, repeatItem2.bind(this));
};
});
In many cases, you'd like to use either external code or other components within your template. An rt-require tag lets you include dependencies: <rt-require dependency="depVarPath" as="depVarName"/>
. Once included, depVarName will be in scope. You can only use rt-require tags at the beginning of your template. When including React components, they can be referred to by their tag name inside a template. For example, <MySlider prop1="val1" onMyChange="{this.onSliderMoved}">
. Nesting is also supported: <MyContainer><div>child</div><div>another</div></MyContainer>
. Children are accessible from this.props.children.
<rt-require dependency="comps/myComp" as="MyComp"/>
<rt-require dependency="utils/utils" as="utils"/>
<MyComp rt-repeat="item in items">
<div>{utils.toLower(item.name)}</div>
</MyComp>
define([
'react/addons',
'lodash',
'comps/myComp',
'utils/utils'
], function (React, _, MyComp, utils) {
'use strict';
function repeatItem1(item, itemIndex) {
return React.createElement(MyComp, {}, React.createElement('div', {}, utils.toLower(item.name)));
}
return function () {
return _.map(items, repeatItem1.bind(this));
};
});
var React = require('react/addons');
var _ = require('lodash');
var MyComp = require('comps/myComp');
var utils = require('utils/utils');
'use strict';
function repeatItem1(item, itemIndex) {
return React.createElement(MyComp, {}, React.createElement('div', {}, utils.toLower(item.name)));
}
module.exports = function () {
return _.map(items, repeatItem1.bind(this));
};
import React from 'react/addons';
import _ from 'lodash';
import MyComp from 'comps/myComp';
import utils from 'utils/utils';
function repeatItem1(item, itemIndex) {
return React.createElement(MyComp, {}, React.createElement('div', {}, utils.toLower(item.name)));
}
export default function () {
return _.map(items, repeatItem1.bind(this));
};
Copyright (c) 2014 Wix. Licensed under the MIT license.