实现非常简单的js双向数据绑定
双向数据绑定指的就是,绑定对象属性的改变到用户界面的变化的能力,反之亦然。换种说法,如果我们有一个user对象和一个name属性,一旦我们赋了一个新值给user.name,在UI上就会显示新的姓名了。同样地,如果UI包含了一个输入用户姓名的输入框,输入一个新值就应该会使user对象的name属性做出相应的改变。
很多热门的JS框架客户端如Ember.js,Angular.js或者KnockoutJS,都在最新特性上刊登了双向数据绑定。这并不意味着从零实现它很难,也不是说需要这些功能的时候,采用这些框架是唯一的选择。下面的想法实际上很基础,可以被认为是3步走计划:
我们需要一个UI元素和属性相互绑定的方法
我们需要监视属性和UI元素的变化
我们需要让所有绑定的对象和元素都能感知到变化
还是有很多方法能够实现上面的想法,有一个简单有效的方法就是使用PubSub模式。这个思路很简单:我们使用数据特性来为HTML代码进行绑定,所有被绑定在一起的JavaScript对象和DOM元素都会订阅一个PubSub对象。只要JavaScript对象或者一个HTML输入元素监听到数据的变化时,就会触发绑定到PubSub对象上的事件,从而其他绑定的对象和元素都会做出相应的变化。
用jQuery做一个简单的实现
对于DOM事件的订阅和发布,用jQuery实现起来是非常简单的,接下来我们就是用Jquery比如下面:
functionDataBinder(object_id){
//UseajQueryobjectassimplePubSub
varpubSub=jQuery({});
//Weexpecta`data`elementspecifyingthebinding
//intheform:data-bind-<object_id>="<property_name>"
vardata_attr="bind-"+object_id,
message=object_id+":change";
//Listentochangeeventsonelementswiththedata-bindingattributeandproxy
//themtothePubSub,sothatthechangeis"broadcasted"toallconnectedobjects
jQuery(document).on("change","[data-"+data_attr+"]",function(evt){
var$input=jQuery(this);
pubSub.trigger(message,[$input.data(data_attr),$input.val()]);
});
//PubSubpropagateschangestoallboundelements,settingvalueof
//inputtagsorHTMLcontentofothertags
pubSub.on(message,function(evt,prop_name,new_val){
jQuery("[data-"+data_attr+"="+prop_name+"]").each(function(){
var$bound=jQuery(this);
if($bound.is("input,textarea,select")){
$bound.val(new_val);
}else{
$bound.html(new_val);
}
});
});
returnpubSub;
}
对于上面这个实现来说,下面是一个User模型的最简单的实现方法:
functionUser(uid){
varbinder=newDataBinder(uid),
user={
attributes:{},
//TheattributesetterpublishchangesusingtheDataBinderPubSub
set:function(attr_name,val){
this.attributes[attr_name]=val;
binder.trigger(uid+":change",[attr_name,val,this]);
},
get:function(attr_name){
returnthis.attributes[attr_name];
},
_binder:binder
};
//SubscribetothePubSub
binder.on(uid+":change",function(evt,attr_name,new_val,initiator){
if(initiator!==user){
user.set(attr_name,new_val);
}
});
returnuser;
}
现在我们如果想要将User模型属性绑定到UI上,我们只需要将适合的数据特性绑定到对应的HTML元素上。
//javascript
varuser=newUser(123);
user.set("name","Wolfgang");
//html
<inputtype="number"data-bind-123="name"/>
这样输入值会自动映射到user对象的name属性,反之
亦然。到此这个简单实现就完成啦!
不需要jQuery的实现
在如今的大多数项目里,可能已经使用了jQuery,因此上面的例子完全可以接受。不过,如果我们需要试着向另一个极端做,并且还删除对jQuery的依赖,那么怎么做呢?好,证实一下这么做并不难(尤其是在我们限制只支持IE8及以上版本的情况下)。最终,我们必须使用一般的javascript实现一个定制的PubSub并且保留了DOM事件:
functionDataBinder(object_id){
//CreateasimplePubSubobject
varpubSub={
callbacks:{},
on:function(msg,callback){
this.callbacks[msg]=this.callbacks[msg]||[];
this.callbacks[msg].push(callback);
},
publish:function(msg){
this.callbacks[msg]=this.callbacks[msg]||[]
for(vari=,len=this.callbacks[msg].length;i<len;i++){
this.callbacks[msg][i].apply(this,arguments);
}
}
},
data_attr="data-bind-"+object_id,
message=object_id+":change",
changeHandler=function(evt){
vartarget=evt.target||evt.srcElement,//IEcompatibility
prop_name=target.getAttribute(data_attr);
if(prop_name&&prop_name!==""){
pubSub.publish(message,prop_name,target.value);
}
};
//ListentochangeeventsandproxytoPubSub
if(document.addEventListener){
document.addEventListener("change",changeHandler,false);
}else{
//IEusesattachEventinsteadofaddEventListener
document.attachEvent("onchange",changeHandler);
}
//PubSubpropagateschangestoallboundelements
pubSub.on(message,function(evt,prop_name,new_val){
varelements=document.querySelectorAll("["+data_attr+"="+prop_name+"]"),
tag_name;
for(vari=,len=elements.length;i<len;i++){
tag_name=elements[i].tagName.toLowerCase();
if(tag_name==="input"||tag_name==="textarea"||tag_name==="select"){
elements[i].value=new_val;
}else{
elements[i].innerHTML=new_val;
}
}
});
returnpubSub;
}
除了设置器里调用jQuery的trigger方法外,模型可以保持一样。调用trigger方法将替代为调用我们定制的具有不同特征的PubSub的publish方法:
//Inthemodel'ssetter:
functionUser(uid){
//...
user={
//...
set:function(attr_name,val){
this.attributes[attr_name]=val;
//Usethe`publish`method
binder.publish(uid+":change",attr_name,val,this);
}
}
//...
}
我们又一次通过一百行不到,又可维护的纯javascript完成了我们想要的结果。
以上内容就是关于js双向数据绑定的相关教程,希望对大家学习有所帮助。