自用开源javascript模板引擎

这是一个小巧且强大的js模板引擎,是我在2015年设计编写,那时候我还是个前端萌新,还记得那天夕阳下的奔跑,那是我逝去的青春........

haole ,言归正传,这个模板引擎还算不错,已经用在了不少的项目中,主要优点就是小巧快捷,

(function () {

    var config = {
        open: '{{',
        close: '}}'
    }

    var regEach1 = /^\s*each\s+(?<name>.+?)(\s+as\s+(?<val>[\w|\$]+?)){0,1}(\s*,\s*(?<key>[\w|\$]+?)){0,1}(\s*,\s*(?<idx>[\w|\$]+?)){0,1}\s*$/
    var regEach2 = /^\s*\/each\s*$/
    var regIf1 = /^\s*if\s+(.+)\s*$/
    var regIf2 = /^\s*\/if\s*$/
    var regElse = /^\s*else\s*$/
    var regElseIf = /^\s*else\s*if\s+(.+)\s*$/
    var $sz = /^((?!if|for|else|switch|case|break|{|}|var\s+).)*$/; //不包含这些关键字

    var seq = 1; //序列值,用于记录each共处理多少次
    function uuid() {
        var s = [];
        var hexDigits = "0123456789abcdef";
        for (var i = 0; i < 8; i++) {
            s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1);
        }
        s[14] = "4";
        s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1);
        //s[8] = s[13] = s[18] = s[23] = "";

        var uuid = s.join("");
        return uuid;
    }

    /**
     * 模板渲染
     * @param {string} content 模板
     * @param {any} data 数据
     * @returns string
     */
    function render(content, data) {
        data = data || {};
        var list = ['this.tpl = "";'];

        for (var key in data){
            list.push('var '+key+'=data["'+key+'"]');
        }

        var codeArr = transform(content);  // 代码分割项数组

        for (var i = 0, len = codeArr.length; i < len; i++) {
            var item = codeArr[i]; // 当前分割项

            if (item.type === 1) {  // js逻辑

                //处理 each,if,else,elseif
                var match,block
                if (match = regEach1.exec(item.txt)) {
                    console.log(match)
                    var items='items'+uuid()+seq++;
                    var name = match.groups.name
                    var val=match.groups.val||'val'
                    var key=match.groups.key||'key'
                    var idx=match.groups.idx||'index'
                    block = `var ${idx}=-1;var ${items}=${name};for(var ${key} in ${items}){ var ${val}=${items}[${key}];${idx}++;`;
                    list.push(block)
                } else if (match = regEach2.exec(item.txt)) {
                    list.push('}');
                } else if (match = regIf1.exec(item.txt)) {
                    block = 'if(' + match[1] + '){'
                    list.push(block)
                } else if (match = regIf2.exec(item.txt)) {
                    list.push('}')
                } else if (match = regElse.exec(item.txt)) {
                    list.push('}else{')
                } else if (match = regElseIf.exec(item.txt)) {
                    block = '}else if(' + match[1] + '){'
                    list.push(block)
                } else if ($sz.exec(item.txt)){ //匹配输出表达式
                    list.push('this.tpl+='+item.txt);
                }else{
                    list.push(item.txt);
                }

            } else if (item.type === 2) {  // js占位
                var txt = 'this.tpl+=' + item.txt + ';';
                list.push(txt);
            } else {  //文本
                var txt = 'this.tpl+="' +
                    item.txt.replace(/"/g, '\\"') +
                    '";';
                list.push(txt);
            }
        }

        list.push('return this.tpl;');
        //console.log(list.join('\n'))

        var obj = {
            fun:new Function('data', list.join('\n'))
        }

        return obj.fun(data);
    }

    /**
     * 从原始模板中提取 文本/js 部分
     * @param {string} content
     * @returns {Array<{type:number,txt:string}>}
     */
    function transform(content) {
        var arr = [];                 //返回的数组,用于保存匹配结果
        //var reg = /\{\{\s*((.|\n)*?)\s*\}\}/g;  //用于匹配js代码的正则
        var str = config.open+'\\s*((.|\\n)*?)\\s*'+config.close
        var reg = new RegExp(str,'g')

        var match;   				  //当前匹配到的match
        var nowIndex = 0;			  //当前匹配到的索引

        while (match = reg.exec(content)) {
            // 保存当前匹配项之前的普通文本/占位
            appendTxt(arr, content.substring(nowIndex, match.index));
            //保存当前匹配项
            var item = {
                type: 1,      // 类型  1- js代码块 2- js表达式 null- 文本
                txt: match[1] // 内容
            };
            var first = match[1].substr(0, 1)
            if (first == ':' || first == '=') {  // 如果是js占位
                item.type = 2;
                item.txt = item.txt.substr(1);
            }
            arr.push(item);
            //更新当前匹配索引
            nowIndex = match.index + match[0].length;
        }
        console.log(content)
        //保存文本尾部
        appendTxt(arr, content.substr(nowIndex));
        return arr;
    }

    /**
     * 普通文本添加到数组,对换行部分进行转义
     *
     * @param {Array<{type:number,txt:string}>} list
     * @param {string} content
     */
    function appendTxt(list, content) {
        content = content.replace(/\r?\n/g, "\\n");
        list.push({txt: content});
    }

    window.cctpl={
        render:render,
        config:config,
    }

})();

编写模板

<script id="tpl1" type="text/tpl">
    {{ each arr as val,key,idx }}
       <li>{{val.name}}-{{key}}-{{idx}}</li>
    {{ /each }}
    <div>{{name}}</div>
</script>

数据合成,这里用了jquery辅助获取

$(function (){
    var tmp = $('#tpl1').html().trim();
    var ret = cctpl.render(tmp,{name:'天穹之光',arr:[{name:'关羽'},{name:'张飞'},{name:'赵云'}]})
    console.log(ret)
})

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
一、什么是 ChakraCore ChakraCore 是微软开源的 Microsoft Edge 浏览器 Chakra JavaScript 引擎的核心部分,主要用于 Microsoft Edge 和 Windows 中 HTML/CSS/JavaScript 编写的应用 ChakraCore 支持 x86/x64/ARM 架构 JavaScript 的 Just-in-time (JIT) 编译,垃圾收集和大量的最新 JavaScript 特性。ChakraCore 还支持 JavaScript Runtime (JSRT) APIs ,允许用户简单嵌入 ChakraCore 到应用中。 ChakraCore 是一个功能完整的、独立的 JavaScript 虚拟机,可嵌入到衍生产品中,驱动需要脚本功能的产品如 NoSQL 数据库、生产力工具和游戏引擎。 ChakraCore 现在已经跨平台支持:Windows、MacOS、 Ubuntu 详细参考微软开源地址: https://github.com/Microsoft/ChakraCore 二、ChakraCore架构 三、JS脚本支持有哪些优势? 在流行的脚本语言中,Lua的小巧高性能(性能指LuaJit的性能),Python的功能性一直受开发者青睐。有什么理由使用JS脚本呢? JS脚本有众多的库支持 JS脚本被用于HTML网页开发,开发者众多 JS有众多大公司的支持 JS有优秀的即时编译(JIT)性能 JS有无敌的开发工具Visual Studio的支持 JS语言特性更类似C/C++,相比Lua要舒服很多 Chakra的嵌入优势:相比Lua的堆栈式API,Chakra的API更容易写胶水代码。 很多游戏使用Lua的原因是比Python性能好,没有其他可选方案了。Chakra的开源,应该带动开发者去使用JS脚本。Chakra对于大型Windows游戏开发者更大的好处在于系统支持,Lua需要去下载编译,而Chakra只需要包含头文件,链接lib。 四、QA 可以列出全局对象或函数吗? 可以,除了Intl,这是个特例 JS可以使用引用(c++的,c#的ref)参数吗? 不可以,即使你为传入Native的函数参数修改值也是没用的。 如果一定要用,那只能传一个引用类型的对象,在函数内部修改此对象的成员。var arr=[];(function (v){v[1]=1;})(arr);//arr[1] == 1 Chakra的API支持多线程吗? 支持,据我当前的研究,不同线程必须有各自的runtime对象,每个runtime可以有多个环境(context),同一个runtime下的多个环境可以自由交换数据,但环境之间不共享数据。也就是说api级别可以把环境1的数据带到环境2,但是脚本里,环境2是看不到环境1的数据的。 Chakra支持ES6的Symbol吗? 完全支持。 Chakra如何在原生函数里支持JS的闭包? 函数(function)也是对象(object),可以有自定义属性,所以,在原生API级别操作Chakra时,可以把需要闭包的变量放在函数的属性里。如果希望在脚本中是只读的,那么可以设置属性描述。如果希望在脚本中是隐藏的,那么可以用符号属性。 五、关于世界4大js引擎的简介 google v8 ,目前为止,只能编译出静态链接库版本。静态库将近3G,运行链接一次需要2分钟,太庞大了 mozilla spider monkey,老牌的js引擎,编译后大小合适,很适合使用 ms chakra core,最新的js引擎,支持es6标准,编译后最小,速度很快,我个人蛮喜欢的 苹果的webkit中的javascript core引擎,没编译过,但是感觉不会太小 代码及相关例程截图 目前该组件已经在自身业务上稳定运用近一年,特以此机会开源给大家使用。 选择这个引擎是因为谷歌的V8太大了,而且API对易语言 也不友好,微软这个性能和V8差不太大,API封装友好。也一直在更新。而且体积也只有5M多一点。非常适合做嵌入开发。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

编程大玩家

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值