js實(shí)現(xiàn)一個(gè)簡(jiǎn)單的MVVM框架示例
以前都是默默地看園子里的文章,猥瑣的點(diǎn)贊,今天也分享一下自己用js實(shí)現(xiàn)的一個(gè)簡(jiǎn)單mvvm框架。
最初只做了自動(dòng)綁定事件,后面又參考學(xué)習(xí)了vue,knouckout以及argular實(shí)現(xiàn)方式,以及結(jié)合自己做WPF的一些經(jīng)驗(yàn),增加了屬性綁定,今天又稍微整理了下,完善了部分功能,把代碼提交到了碼云:https://gitee.com/zlj_fy/Simple-MVVM
先簡(jiǎn)單介紹下用法:
<form class="form-horizontal" role="form" data-context="TestController">
<div class="form-group">
<legend>Form title</legend>
</div>
<div class="form-group">
<div class="col-sm-6 col-sm-offset-2">
<input type="text" class="form-control" bind-val="age,format=format" style="margin:5px 0" />
<input type="text" class="form-control" bind-val="desc" style="margin:5px 0" />
<input type="range" min="10" max="300" bind-val="age" step="10" class="form-control" style="margin:5px 0" />
<input type="button" class="btn btn-primary" value="更新" style="margin:5px 0" on-click="update" />
</div>
</div>
</form>
<script>
var TestController = {
data: {
name: 'xiaoming',
age: 3,
desc: function() {
return this.name + ' likes looking little movie. he should take care of his body'
}
},
format: function(val) {
return val + '歲'
},
update: function() {
this.name = 'this is a test'
this.age = 18
}
}
$('body').controller()
</script>
首先定義一個(gè)控制器,可以是json對(duì)象,也可一是一個(gè)function,然后在頂層的元素定義data-context=“[控制器名稱(chēng)]”就可以將該控制器綁定到該節(jié)點(diǎn)底下所有元素。如果元素后代存在嵌套Controller,則其所在的元素以下子元素作用域指向子控制器。
1.監(jiān)控屬性以及復(fù)雜屬性
所有屬性必須定義在data節(jié)點(diǎn)下,如果里面的屬性定義成function則認(rèn)為是復(fù)雜屬性(例如desc),復(fù)雜屬性是只讀的,重新賦值的話會(huì)提示錯(cuò)誤。
綁定到html元素上的格式:"{屬性名,fomat=[控制器方法]}",屬性名支持嵌套屬性,例如(a.b);屬性名不支持表達(dá)式,考慮了覺(jué)得不是很有必要,完全可以使用復(fù)雜屬性去代替,當(dāng)前缺點(diǎn)是業(yè)務(wù)復(fù)雜的話可能造成大量復(fù)雜屬性;屬性名右邊是可選參數(shù),目前只有format,也就是屬性顯示在html上的轉(zhuǎn)換方法。
2.指令
綁定指令語(yǔ)法是 bind-{指令}的形式,目前只實(shí)現(xiàn)了val,attr,text,html,template,其實(shí)可以看出,前面4個(gè)都只是簡(jiǎn)單封裝了jqeury方法,template是用到了jquery-tmpl插件實(shí)現(xiàn)的,如果你需要更多的指令,你可以自己去擴(kuò)展,只需要實(shí)現(xiàn)init初始加載方法(接收當(dāng)前的observer參數(shù)),以及update方法(參數(shù)說(shuō)明:對(duì)應(yīng)的jquery元素,最新的值,當(dāng)前控制器實(shí)例);如果是擴(kuò)展已有的指令,默認(rèn)會(huì)覆蓋原有的。如下:
$.controller.addDirective("val", {
init: function (observer) {
if (observer.$ele.is('input,select')) {
//監(jiān)聽(tīng)onchange事件
observer.$ele.on('input propertychange', function () {
var newVal = $(this).val()
observer.writeValue(newVal)
})
}
},
update: function ($ele, newVal, controller) {
$ele.val && $ele.val(newVal)
}
})
3.事件
綁定事件語(yǔ)法:on-{事件}=“{控制器方法},type=on/one”,控制器方法右邊是可選參數(shù),目前只有綁定類(lèi)型on/one,默認(rèn)是on;控制器方法接收兩個(gè)參數(shù),一個(gè)是可在對(duì)應(yīng)事件的元素上設(shè)置初始參數(shù),一個(gè)是event事件參數(shù);
<button type="button" class="btn btn-primary" data-page="1" on-click="refesh">查詢</button>
4.方法
直接使用this.屬性名,就可以直接訪問(wèn)對(duì)應(yīng)data節(jié)點(diǎn)下的屬性。
5.鉤子
init以及created,init是在監(jiān)聽(tīng)所有屬性之后編譯dom之前,可以在這方法上初始化參數(shù);created是編譯dom元素之后。
其中控制器默認(rèn)實(shí)現(xiàn)了extend繼承方法,可以繼承另一個(gè)控制器,必須在init方法中使用。當(dāng)前你也可以自己使用原型繼承的方式去實(shí)現(xiàn)。
init: function () {
this.extend(PageController)
},
created: function () {
//TODO
},
6.擴(kuò)展
相信大家在做項(xiàng)目的時(shí)候肯定都會(huì)有一套公用的組件,那么可以像下面那樣擴(kuò)展,默認(rèn)對(duì)應(yīng)的組件掛載到所有的控制器示例下面,就可以之間在對(duì)應(yīng)的方法下直接調(diào)用了: this.http.post();
不過(guò)有一個(gè)建議,就是盡量統(tǒng)一將回調(diào)方法的作用域指向控制器,這樣開(kāi)發(fā)不至于老是出現(xiàn)作用域的問(wèn)題。
$.controller.extend({
utils: utils,
notify: $.notify,
modal: $.modal,
http: $.http,
alert: $.alert
})
7.原理以及代碼分析(待續(xù)...)
整個(gè)js代碼量只有300多行,所以實(shí)現(xiàn)的比較簡(jiǎn)單,有很多方面是沒(méi)有考慮到的,還有一些功能是想實(shí)現(xiàn)卻沒(méi)有去做的,目前不支持?jǐn)?shù)組變化檢測(cè),以及局部更新相關(guān)dom。
以上這篇js實(shí)現(xiàn)一個(gè)簡(jiǎn)單的MVVM框架示例就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
JS獲取瀏覽器語(yǔ)言動(dòng)態(tài)加載JS文件示例代碼
通過(guò)獲取瀏覽器語(yǔ)言版本,來(lái)相對(duì)的加載easyui語(yǔ)言包就是動(dòng)態(tài)加載JS文件,下面有個(gè)不錯(cuò)的實(shí)例,大家可以看看2014-10-10
js 倒計(jì)時(shí)(高效率服務(wù)器時(shí)間同步)
首先說(shuō)一下,為什么要服務(wù)器時(shí)間同步, 因?yàn)榉?wù)器時(shí)間和本地電腦時(shí)間存在一定的時(shí)間差。有些對(duì)時(shí)效性要求非常高的應(yīng)用,例如時(shí)時(shí)彩開(kāi)獎(jiǎng),是不能容忍這種時(shí)間差存在的2017-09-09
js實(shí)現(xiàn)iframe框架取值的方法(兼容IE,firefox,chrome等)
這篇文章主要介紹了js實(shí)現(xiàn)iframe框架取值的方法,可兼容IE,firefox,chrome等瀏覽器.涉及JavaScript針對(duì)框架元素取值的相關(guān)技巧,需要的朋友可以參考下2015-11-11
將Datatable轉(zhuǎn)化成json發(fā)送前臺(tái)實(shí)現(xiàn)思路
將Datatable轉(zhuǎn)化成json可以將dt序列化成json,放到前臺(tái)的隱藏控件hidBoundary中,具體的實(shí)現(xiàn)如下,有類(lèi)似需求的朋有可以參考下2013-09-09
JS控制阿拉伯?dāng)?shù)字轉(zhuǎn)為中文大寫(xiě)示例代碼
阿拉伯?dāng)?shù)字如何轉(zhuǎn)為中文大寫(xiě)這個(gè)實(shí)現(xiàn)的方法有很多,在本文將為大家介紹下js中時(shí)如何實(shí)現(xiàn)的,感興趣的朋友可以參考下2013-09-09

