Onca (art-template fork)
Onca is a simple and superfast templating engine that optimizes template rendering speed by scope pre-declared technique, hence achieving runtime performance which is close to the limits of JavaScript. Worked only in NodeJS.
- Features
- Introduction
- Installation
- Syntax
- Debug
- Template variable
- Parsing rules
- Minimize
- Options
- Examples
- License
Features
- Performance is close to the JavaScript rendering limits
- Debugging friendly. Syntax errors or runtime errors will be positioned accurately at which line of template
- Support Express.js, Koa.js, NestJS
- Support template inheritance and sub template
Introduction
Template
Onca simultaneously supports two syntax of template. Standard syntax allows templates to be easier to read and write. While original syntax has powerful logical processing ability.
standard syntax
{{if user}}
<h2>{{user.name}}</h2>
{{/if}}
original syntax
<% if (user) { %>
<h2><%= user.name %></h2>
<% } %>
Original syntax is compatible with EJS, Underscore, LoDash templates.
Render template
const template = require('onca');
const html = template(`${__dirname}/user.art`, {
user: {
name: 'Jack Doe'
}
});
Core method
// render template basing on template name
template(filename, data);
// compile source of template as function
template.compile(source, options);
// compile source of template as function and immediately invoke it
template.render(source, data, options);
Installation
Working only inside NodeJS engine.
NPM
npm i onca
YARN
yarn add onca
Syntax
Standard syntax supports basic templating syntax and JavaScript expression. Original syntax supports arbitrary JavaScript statement, the same as EJS.
Output
standard syntax
{{value}}
{{data.key}}
{{data['key']}}
{{a ? b : c}}
{{a || b}}
{{a + b}}
original syntax
<%= value %>
<%= data.key %>
<%= data['key'] %>
<%= a ? b : c %>
<%= a || b %>
<%= a + b %>
You can use $data
with bracket notation to access a first-class variable of templates (such as keyword, reserved word, etc):
{{$data['user list']}}
Raw output
standard syntax
{{@ value }}
original syntax
<%- value %>
raw output will not escape the content of HTML, so there may be security problems. Please be careful.
Condition
standard syntax
{{if value}} ... {{/if}}
{{if v1}} ... {{else if v2}} ... {{/if}}
original syntax
<% if (value) { %> ... <% } %>
<% if (v1) { %> ... <% } else if (v2) { %> ... <% } %>
Loop
standard syntax
{{each target}}
{{$index}} {{$value}}
{{/each}}
original syntax
<% for(var i = 0; i < target.length; i++){ %>
<%= i %> <%= target[i] %>
<% } %>
target
supports iteration ofarray
andobject
, and its default value is$data
.$value
and$index
can be customized: {{each target val key}}.
Variable
standard syntax
{{set temp = data.sub.content}}
original syntax
<% var temp = data.sub.content; %>
Template inheritance
standard syntax
{{extend './layout.art'}}
{{block 'head'}}
...
{{/block}}
original syntax
<% extend('./layout.art') %>
<% block('head', function(){ %> ... <% }) %>
Template inheritance allows you to build a basic templating “skeleton” that contains common parts of your site. Example:
<!--layout.art-->
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<title>{{block 'title'}}My Site{{/block}}</title>
{{block 'head'}}
<link rel="stylesheet" href="main.css" />
{{/block}}
</head>
<body>
{{block 'content'}}{{/block}}
</body>
</html>
<!--index.art-->
{{extend './layout.art'}} {{block 'title'}}{{title}}{{/block}} {{block 'head'}}
<link rel="stylesheet" href="custom.css" />
{{/block}} {{block 'content'}}
<p>This is just an awesome page.</p>
{{/block}}
After rendering index.art, layout skeleton will be automatically applied.
Sub template
standard syntax
{{include './header.art'}}
{{include './header.art' data}}
original syntax
<% include('./header.art') %>
<% include('./header.art', data) %>
data
value is$data
by default. Standard syntax doesn't support declaration ofobject
andarray
but reference variable. However, original syntax is no limits.- Onca has built-in HTML minifier and please avoid writing abnormal
closing tag in sub templates. Otherwise, tags may be unexpectedly
“optimized” when
minimize
option is open.
Filters
register filters
template.defaults.imports.dateFormat = function (date, format) {
/*[code..]*/
};
template.defaults.imports.timestamp = function (value) {
return value * 1000;
};
The first parameter of filter function accepts target value.
standard syntax
{{date | timestamp | dateFormat 'yyyy-MM-dd hh:mm:ss'}}
{{value | filter}}
filter syntax is similar to pipe operator. Its last output will be the next input.
original syntax
<%= $imports.dateFormat($imports.timestamp(date), 'yyyy-MM-dd hh:mm:ss') %>
Debug
template.defaults.debug
Onca has a built-in debugger. It can catch syntax and runtime errors.
In addition, it supports custom syntax.
In NodeJS, debugging mode will be automatically opened according to the environment variable: process.env.NODE_ENV !== 'production'
Setting template.defaults.debug=true
is equivalent to:
{
"cache": false,
"minimize": false,
"compileDebug": true
}
Template variable
template.defaults.imports
Template can access global variable outside it and imported variable through $imports
.
Import variable
template.defaults.imports.log = console.log;
<% $imports.log('hello world') %>
Built-in variables
- $data the data passed into template
- $imports variable imported from outside and global variable
- print string-output function
- include sub-template loading function
- extend template-import function in template inheritance
- block template-block declaration function
Parsing rules
template.defaults.rules
You can customize template parsing rules in art-template. Standard syntax and original syntax is configured by default.
Modify delimiters
// delimiter rules of original syntax
template.defaults.rules[0].test = /<%(#?)((?:==|=#|[=-])?)[ \t]*([\w\W]*?)[ \t]*(-?)%>/;
// delimiter rules of standard syntax
template.defaults.rules[1].test = /{{([@#]?)[ \t]*(\/?)([\w\W]*?)[ \t]*}}/;
They are regular expressions, and you can only modify the delimiter part.
For example, modify <% %>
to <? ?>
:
const rule = template.defaults.rules[0];
rule.test = new RegExp(rule.test.source.replace('<%', '<\\?').replace('%>', '\\?>'));
Add syntax
Let’s start with a simple example that make template engine support parse
of template strings ${name}
of ES6:
template.defaults.rules.push({
test: /\${([\w\W]*?)}/,
use: function (match, code) {
return {
code: code,
output: 'escape'
};
}
});
test
is a regular expression which matches strings and use
is a callback function
after matching. About use
function:
- parameters: first parameter is the matching string,
and others are content of capturing group of
test
regular expression - return value: MUST return an object containing code and output properties:
- code transformed JavaScript statements
- output describe type of code, optional value:
- 'escape' output after encoding
- 'raw' output raw content
- false output nothing
It’s worth mentioning that syntax rules have no effect on rendering speed and template parser will help you optimize rendering performance.
Minimize
template.defaults.minimize
A built-in minifier of art-template can minimize HTML, JS and CSS, which happens in compilation phase. So it totally has no effect on rendering speed and moreover speeds up network transmission.
Minimize mode
template.defaults.minimize = true;
Configuration
Refer to: https://github.com/kangax/html-minifier
default configuration
template.defaults.htmlMinifierOptions = {
collapseWhitespace: true,
minifyCSS: true,
minifyJS: true,
// automatically merged at runtime: rules.map(rule => rule.test)
ignoreCustomFragments: []
};
Options
template.defaults
({
// template name
filename: null,
// an array of rules of template syntax
rules: [nativeRule, artRule],
// whether to automatically encode output statements of template. Setting false will close that functionality
// escape can prevent XSS attacks
escape: true,
// enable debug mode. If true: {cache:false, minimize:false, compileDebug:true}
debug: detectNode ? process.env.NODE_ENV !== 'production' : false,
// if bail is set true, compilation errors and runtime errors will throw exception
bail: true,
// whether to enable caching
cache: true,
// whether to enable minimization. It will execute htmlMinifier and minimize HTML, CSS, JS
// if template contains unclosing tags, please don't open minimize. Otherwise unclosing tags will be restored or filtered
minimize: true,
// whether to compile in debug mode
compileDebug: false,
// resolve template path
resolveFilename: resolveFilename,
// sub template compilation adapter
include: include,
// HTML minifier. Work only in NodeJS environment
htmlMinifier: htmlMinifier,
// HTML minifier configuration. Refer to: https://github.com/kangax/html-minifier
htmlMinifierOptions: {
collapseWhitespace: true,
minifyCSS: true,
minifyJS: true,
// automatically merged at runtime: rules.map(rule => rule.test)
ignoreCustomFragments: []
},
// error events. Work only if bail is false
onerror: onerror,
// template file loader
loader: loader,
// cache center adapter (depend on filename field)
caches: caches,
// root directory of template. If filename field is not a local path, template will be found in root directory
root: '/',
// default extension. If no extensions, extname will be automatically added
extname: '.art',
// ignored variables. An array of template variables ignored by template compiler
ignore: [],
// imported template variables
imports: runtime
});
Examples
Express.js
index.js
const express = require('express');
const path = require('path');
const { expressEngine } = require('onca');
const app = express();
// view engine setup
app.engine('art', expressEngine);
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'art');
// routes
app.get('/', function (req, res) {
res.render('index.art', {
user: {
name: 'aui',
tags: ['art', 'template', 'nodejs']
}
});
});
app.listen(3030, () => {
console.log('server');
});
/views/index.art
<div>
<h2>Onca template engine</h2>
</div>
Koa.js
index.js
const Koa = require('koa');
const { koaEngine } = require('onca');
const path = require('path');
const app = new Koa();
koaEngine(app, {
root: path.join(__dirname, 'views'),
extname: '.art',
debug: process.env.NODE_ENV !== 'production'
});
app.use(async function (ctx) {
await ctx.render('index');
});
app.listen(3030);
/views/index.art
<div>
<h2>Onca template engine</h2>
</div>
NestJS
main.ts
import { expressEngine } from 'onca';
// <NestJS application initialization>
// ...
app.useStaticAssets(join(__dirname, '..', 'public', 'assets'));
app.setBaseViewsDir(join(__dirname, 'views'));
app.setViewEngine('art');
app.engine('art', expressEngine);
app.set('view options', { debug: process.env.NODE_ENV !== 'production' });
mvc.controller.ts
export class MVCController {
@Get('/')
@Render('main-page')
public async mainPage() {
return '';
}
}
License
Onca is MIT licensed.