高性能的前端模板引擎art-template.js学习教程
简介
art-template 是一个简约、超快的模板引擎。它采用作用域预声明的技术来优化模板渲染速度,从而获得接近 JavaScript 极限的运行性能,并且同时支持 NodeJS 和浏览器。
你可以进行在线速度测试,对比现有的一些模板引擎的速度:在线速度测试
特性
- 拥有接近 JavaScript 渲染极限的的性能
 - 调试友好:语法、运行时错误日志精确到模板所在行;支持在模板文件上打断点(Webpack Loader)
 - 支持 Express、Koa、Webpack
 - 支持模板继承与子模板
 - 浏览器版本仅 6KB 大小
 
安装
NPM
npm install art-template --save
浏览器
如果要兼容到IE8的话,需要补丁,请参考示例:示例
Webpack
npm install art-template
npm install art-template-loader --save-dev
Express
npm install --save art-template
npm install --save express-art-template
koa
npm install --save art-template
npm install --save koa-art-template
语法
art-template支持标准语法与原始语法。标准语法可以让模板易读写,而原始语法拥有强大的逻辑表达能力。
标准语法支持基本模板语法以及基本 JavaScript 表达式;原始语法支持任意 JavaScript 语句,这和 EJS 一样。
输出
标准语法
{{value}}
{{data.key}}
{{data['key']}}
{{a ? b : c}}
{{a || b}}
{{a + b}}
原始语法
<%= value %>
<%= data.key %>
<%= data['key'] %>
<%= a ? b : c %>
<%= a || b %>
<%= a + b %>
模板一级特殊变量可以使用 $data 加下标的方式访问:
{{$data['user list']}}
原文输出
标准语法
{{@ value }}
原始语法
<%- value %>
条件
标准语法
{{if value}} ... {{/if}}
{{if v1}} ... {{else if v2}} ... {{/if}}
原始语法
<% if (value) { %> ... <% } %>
<% if (v1) { %> ... <% } else if (v2) { %> ... <% } %>
循环
标准语法
{{each target}}
    {{$index}} {{$value}}
{{/each}}
原始语法
<% for(var i = 0; i < target.length; i++){ %>
    <%= i %> <%= target[i] %>
<% } %>
target支持array与object的迭代,其默认值为$data。$value与$index可以自定义:{{each target val key}}。
变量
标准语法
{{set temp = data.sub.content}}
原始语法
<% var temp = data.sub.content; %>
模板继承
标准语法
{{extend './layout.art'}}
{{block 'head'}} ... {{/block}}
原始语法
<% extend('./layout.art') %>
<% block('head', function(){ %> ... <% }) %>
子模板
标准语法
{{include './header.art'}}
{{include './header.art' data}}
原始语法
<% include('./header.art') %>
<% include('./header.art', data) %>
data数默认值为$data;标准语法不支持声明object与array,只支持引用变量,而原始语法不受限制。art-template内建 HTML 压缩器,请避免书写 HTML 非正常闭合的子模板,否则开启压缩后标签可能会被意外“优化。
过滤器
注册过滤器
template.defaults.imports.dateFormat = function(date, format){/*[code..]*/};
template.defaults.imports.timestamp = function(value){return value * 1000};
过滤器函数第一个参数接受目标值。
标准语法
{{date | timestamp | dateFormat 'yyyy-MM-dd hh:mm:ss'}}
{{value | filter}} 过滤器语法类似管道操作符,它的上一个输出作为下一个输入。
原始语法
<%= $imports.dateFormat($imports.timestamp(date), 'yyyy-MM-dd hh:mm:ss') %>
模板变量
art-template内置了如下这些变量:
$data传入模板的数据$imports外部导入的变量以及全局变量print字符串输出函数include子模板载入函数extend模板继承模板导入函数block模板块声明函数
你也可以使用template.defaults.imports导入外部变量
template.defaults.imports.log = console.log;
<% $imports.log('hello world') %>
解析规则
art-template使用template.defaults.rules定义了内部的解析规则,你也可以通过这个修改或添加你自己的规则。
修改界定符
// 原始语法的界定符规则
template.defaults.rules[0].test = /<%(#?)((?:==|=#|[=-])?)[ \t]*([\w\W]*?)[ \t]*(-?)%>/;
// 标准语法的界定符规则
template.defaults.rules[1].test = /{{([@#]?)[ \t]*(\/?)([\w\W]*?)[ \t]*}}/;
它们是一个正则表达式,你可以只修改界定符部分。例如修改 <% %>为 <? ?>:
var rule = template.defaults.rules[0];
rule.test = new RegExp(rule.test.source.replace('<%', '<\\\?').replace('%>', '\\\?>'));
添加语法
从一个简单的例子说起,让模板引擎支持 ES6 ${name} 模板字符串的解析:
template.defaults.rules.push({
    test: /\${([\w\W]*?)}/,
    use: function(match, code) {
        return {
            code: code,
            output: 'escape'
        }
    }
});
其中 test 是匹配字符串正则,use 是匹配后的调用函数。关于 use 函数:
- 参数:一个参数为匹配到的字符串,其余的参数依次接收 
test正则的分组匹配内容 - 返回值:必须返回一个对象,包含 
code与output两个字段:code转换后的 JavaScript 语句output描述code的类型,可选值:'escape'编码后进行输出'raw'输出原始内容false不输出任何内容
 
值得一提的是,语法规则对渲染速度没有影响,模板引擎编译器会帮你优化渲染性能。
压缩页面
art-template内置了html-minifier,可以压缩 HTML、JS、CSS,它在编译阶段运行,因此完全不影响渲染速度,并且能够加快网络传输。
它的默认配置如下:
template.defaults.htmlMinifierOptions = {
    collapseWhitespace: true,
    minifyCSS: true,
    minifyJS: true,
    // 运行时自动合并:rules.map(rule => rule.test)
    ignoreCustomFragments: []
};
选项
template.defaults有如下这些选项:
// 模板名
filename: null,
// 模板语法规则列表
rules: [nativeRule, artRule],
// 是否开启对模板输出语句自动编码功能。为 false 则关闭编码输出功能
// escape 可以防范 XSS 攻击
escape: true,
// 启动模板引擎调试模式。如果为 true: {cache:false, minimize:false, compileDebug:true}
debug: detectNode ? process.env.NODE_ENV !== 'production' : false,
// bail 如果为 true,编译错误与运行时错误都会抛出异常
bail: true,
// 是否开启缓存
cache: true,
// 是否开启压缩。它会运行 htmlMinifier,将页面 HTML、CSS、CSS 进行压缩输出
// 如果模板包含没有闭合的 HTML 标签,请不要打开 minimize,否则可能被 htmlMinifier 修复或过滤
minimize: true,
// 是否编译调试版
compileDebug: false,
// 模板路径转换器
resolveFilename: resolveFilename,
// 子模板编译适配器
include: include,
// HTML 压缩器。仅在 NodeJS 环境下有效
htmlMinifier: htmlMinifier,
// HTML 压缩器配置。参见 https://github.com/kangax/html-minifier
htmlMinifierOptions: {
    collapseWhitespace: true,
    minifyCSS: true,
    minifyJS: true,
    // 运行时自动合并:rules.map(rule => rule.test)
    ignoreCustomFragments: []
},
// 错误事件。仅在 bail 为 false 时生效
onerror: onerror,
// 模板文件加载器
loader: loader,
// 缓存中心适配器(依赖 filename 字段)
caches: caches,
// 模板根目录。如果 filename 字段不是本地路径,则在 root 查找模板
root: '/',
// 默认后缀名。如果没有后缀名,则会自动添加 extname
extname: '.art',
// 忽略的变量。被模板编译器忽略的模板变量列表
ignore: [],
// 导入的模板变量
imports: runtime
API
template(filename, content)
根据模板名渲染模板。
- 参数:
{string} filename{Object,string} content
 - 返回值:
- 如果 
content为Object,则渲染模板并返回string - 如果 
content为string,则编译模板并返回function 
 - 如果 
 
浏览器版本无法加载外部文件,filename 为存放模板的元素 id
.compile(source, options)
编译模板并返回一个渲染函数。
- 参数:
{string} source{Object} options
 - 返回值:
{function} 
.render(source, data, options)
编译并返回渲染结果。
- 参数:
{string} source{Object} options
 - 返回值:
{string} 
.defaults
模板引擎默认配置。参考选项。
- 类型:
{Object} 
.extension
给 NodeJS require.extensions 注册的模板渲染函数。
- 类型:
{function} 
加载 .ejs 模板:
var template = require('art-template');
require.extensions['.ejs'] = template.extension;
var view = require('./index.ejs');
var html = view(data);
实例
webpack
module.exports = {
    // ...
    module: {
        rules: [{
            test: /\.art$/,
            loader: "art-template-loader",
            options: {
                // art-template options (if necessary)
                // @see https://github.com/aui/art-template
            }
        }]
    },
    // ...
}
Express
var express = require('express');
var app = express();
app.engine('art', require('express-art-template'));
app.set('view options', {
    debug: process.env.NODE_ENV !== 'production'
});
app.get('/', function (req, res) {
    res.render('index.art', {
        user: {
            name: 'aui',
            tags: ['art', 'template', 'nodejs']
        }
    });
});
Koa
const Koa = require('koa');
const render = require('koa-art-template');
const app = new Koa();
render(app, {
  root: path.join(__dirname, 'view'),
  extname: '.art',
  debug: process.env.NODE_ENV !== 'production'
});
app.use(async function (ctx) {
  await ctx.render('user');
});
app.listen(8080);
浏览器
<!DOCTYPE HTML>
<html>
<head>
<meta charset="UTF-8">
<title>basic-demo</title>
<script src="../../lib/template-web.js"></script>
</head>
<body>
<div id="content"></div>
<script id="test" type="text/html">
{{if isAdmin}}
<h1>{{title}}</h1>
<ul>
    {{each list value i}}
        <li>索引 {{i + 1}} :{{value}}</li>
    {{/each}}
</ul>
{{/if}}
{{$data}}
</script>
<script>
var data = {
	title: '基本例子',
	isAdmin: true,
	list: ['文艺', '博客', '摄影', '电影', '民谣', '旅行', '吉他']
};
var html = template('test', data);
document.getElementById('content').innerHTML = html;
</script>
</body>
</html>
浏览器
<!DOCTYPE HTML>
<html>
<head>
<meta charset="UTF-8">
<title>compile-demo</title>
<script src="../../lib/template-web.js"></script>
</head>
<body>
<h1>在javascript中存放模板</h1>
<div id="content"></div>
<script>
var source = '<ul>'
+    '{{each list value i}}'
+        '<li>索引 {{i + 1}} :{{value}}</li>'
+    '{{/each}}'
+ '</ul>';
var render = template.compile(source);
var html = render({
    list: ['摄影', '电影', '民谣', '旅行', '吉他']
});
document.getElementById('content').innerHTML = html;
</script>
</body>
</html>