打造属于自己的MVVM框架: 2.模版渲染引擎

上一篇介绍了MVVM的基本知识,本篇讲针对MVVM的模版渲染引擎进行介绍,不但从原理上对模版引擎的渲染原理进行剖析,而且有会相应的实现代码。

源码请戳
原文请戳

什么是模版渲染引擎

还是先来看一下上一篇有关knockoutjs的Demo:

<p>First name: <strong data-bind="text: firstName"></strong></p>
<p>Last name: <strong data-bind="text: lastName"></strong></p>
var viewModel = {
    firstName: "Bert",
    lastName: "Bertington"
};
ko.applyBindings(viewModel);

页面效果:

First name: Bert

Last name: Bertington

在HTML里,我们用data-bind: "text: firstName"作为Binding Instruction,而在JS里的viewModel相当于一个$scope,当Dom加载时,首先会检查HTML标签,发现有Binding Instruction后会对DOM进行解析,此时根据具体的指令在viewModel中进行解析,将解析后的值渲染到已经生成的DOM树上,就完成了整个指令渲染工作,而这个流程,就是前端模版渲染引擎的主要任务。

怎么做一个简单的渲染引擎

其实称为引擎还真有点夸张,充其量它只不过是一个解析的逻辑流程,在整个过程中有三个部分:

  • 模版,即Html
  • 渲染源,即viewModel
  • 所谓的引擎,一段解析流程的,由knockoutjs负责

现在我们来自己试着实现一下这个模版引擎。

1.模版

为了在渲染是保留原模版,我采用template标签去画Html模版,因为:

  • template标签可以放在任意位置
  • template标签默认display: none

基于以上优点,个人觉得template标签太适合做模版了,难怪会称为template。

<template id="test">
    <p>First name: <strong data-bind="text: firstName"></strong></p>
    <p>Last name: <strong data-bind="text: lastName"></strong></p>
</template>

将我们要渲染的Html包裹在<template>中,加上id是为了能够确保唯一。

2.解析template

利用id我们可以唯一找到template,首先将template里的内容取出来,

var clone = document.importNode(document.querySelector('#' + id).content, true);

分离子节点

var fragmentContent = splitSubRealDoms(clone);
function splitSubRealDoms(fatherDom) {
    var subRealDoms = [];
    while(fatherDom.firstElementChild) {
        var firstElementChild = fatherDom.removeChild(fatherDom.firstElementChild);
        subRealDoms.push(firstElementChild);
    }
    return subRealDoms;
}

3.根据父节点的Binding Instruction去渲染子节点

for(var i = 0;i < fragmentContent.length;i++) {
    var result = renderTemplate(fragmentContent[i], viewModel);
}

renderTemplate的方法较为复杂,首先会渲染父节点,然后将所有的子节点当作父节点再次递归,直到没有子节点为止。递归后的子节点集合渲染完后,需要加入到重新加入到父节点中。递归中途需要对data-bind = instruction: value进行解析,将得到的value值在viewModel的$scope中,利用eval进行解析后绑定到DOM上。(详细代码略长,就不在这里贴了,可以去我的repo里去查看)

4.渲染完成

渲染完成后,将最终的结果插入到body上。

    $('body').append($(result));

5.总结

这一节主要介绍了前端模版引擎的工作原理,同时也分享了我自己的代码。但模版引擎仅仅只起到了单向绑定的效果(即viewModl->view),要想体现MVVM的优势,那就必须得实现双向绑定,那就必须的介绍MVVM中的核心对象observable了,下一篇会介绍如何实现observable。

源码请戳
原文请戳

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • 这篇笔记主要包含 Vue 2 不同于 Vue 1 或者特有的内容,还有我对于 Vue 1.0 印象不深的内容。关于...
    云之外阅读 4,989评论 0 29
  • MVVM(Model View ViewModel)是一种基于MVC的设计,开发人员在HTML上写一些Bindin...
    Pursue阅读 3,066评论 0 10
  • MVVM中对Bingding的解析只能算viewModel->view的单项绑定,但MVVM绝不仅仅只有单向绑定,...
    Pursue阅读 698评论 4 6
  • 每天一百多人的门诊工作量,对于出门诊的医生就是折磨,看病要麻利,询问病史要简要击中要害,不让病人集堆,下班时...
    慈桉阅读 338评论 0 1
  • 看清自己的我,很悲伤…… 在别人感情世界里面指点、谈论,闲聊的我,一个人的时候,充满了落寞....
    堂堂工阅读 133评论 0 0