国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

輕松實現javascript數據雙向綁定

2019-11-20 11:18:11
字體:
來源:轉載
供稿:網友

雙向數據綁定指的是當對象的屬性發生變化時能夠同時改變對應的UI,反之亦然。換句話說,如果我們有一個user對象,這個對象有一個name屬性,無論何時你對user.name設置了一個新值,UI也會展示這個新的值。同樣的,如果UI包含一個用于數據用戶名字的輸入框,輸入一個新值也會導致user對象的name屬性發生相應的改變。

許多流行的javascript框架,像Ember.js,Angular.js或者KnockoutJS都會把雙向數據綁定作為其中的主要特性來宣傳。這并不意味著從頭開始實現它很難,也不意味著當我們需要這種功能的時候,使用這些框架是我們唯一的選擇。內部的潛在思想事實上是相當基礎的,實現它可以歸納為以下三點:

  • 我們需要一種方式確定哪個UI元素綁定在哪個屬性上。
  • 我們需要監控屬性和UI的變化
  • 我們需要把所有綁定的對象和UI元素的變化傳播出去。

盡管有好多種方式去實現這幾點,一種簡單高效的方法是我們通過發布訂閱者模式來實現。方法很簡單:我們可以使用定制的data屬性作為HTML代碼中需要綁定的屬性。所有的綁定在一起的Javascript對象和DOM元素將會訂閱這個發布訂閱對象。任何時候我們檢測到無論是Javascript對象亦或是HTML的input元素的變化,我們都是把事件代理傳遞給發布訂閱對象,然后通過它把所有發生在綁定的對象和元素的的變化傳遞和廣播出去。

一個用jQuery實現的簡單例子

通過jQuery實現我們上面討論的東西是相當簡單明了的,因為作為一個流行的庫,它讓我們很簡單的實現訂閱和發布DOM事件,同時我們也可以定制一個:

function DataBinder(object_id){  // Use a jQuery object as simple PubSub  var pubSub=jQuery({});  // We expect a `data` element specifying the binding  // in the form:data-bind-<object_id>="<property_name>"  var data_attr="bind-"+object_id,    message=object_id+":change";  // Listen to chagne events on elements with data-binding attribute and proxy  // then to the PubSub, so that the change is "broadcasted" to all connected objects  jQuery(document).on("change","[data-]"+data_attr+"]",function(eve){    var $input=jQuery(this);    pubSub.trigger(message,[$input.data(data_attr),$input.val()]);  });  // PubSub propagates chagnes to all bound elemetns,setting value of  // input tags or HTML content of other tags  pubSub.on(message,function(evt,prop_name,new_val){    jQuery("[data-"+data_attr+"="+prop_name+"]").each(function(){      var $bound=jQuery(this);      if($bound.is("")){        $bound.val(new_val);      }else{        $bound.html(new_val);      }    });  });  return pubSub;}

至于javascript對象,下面是最小化的user數據模型實現的例子:

function User(uid){  var binder=new DataBinder(uid),        user={      attributes:{},      // The attribute setter publish changes using the DataBinder PubSub      set:function(attr_name,val){        this.attributes[attr_name]=val;        binder.trigger(uid+":change",[attr_name,val,this]);      },      get:function(attr_name){        return this.attributes[attr_name];      },          _binder:binder    };  // Subscribe to PubSub  binder.on(uid+":change",function(evt,attr_name,new_val,initiator){    if(initiator!==user){      user.set(attr_name,new_val);    }  });  return user;}

現在,無論何時我們想要綁定一個對象的屬性到UI上,我們只要在對應的HTML元素上設置合適的data屬性。

// javascript var user=new User(123);user.set("name","Wolfgang");// html<input type="number" data-bind-123="name" />

input輸入框上值得變化會自動的映射到user的name屬性,反之亦然。大功告成!

不需要jQuery的實現方式

現在的大部分項目一般jQuery都已經在使用啦,所以上面的例子是完全可以接受的。但是如果我們需要完全不依賴jQuery,那么該怎么實現呢?好吧,事實上其實也不難辦到(特別是當我們把對IE的支持只提供IE8以上的支持)。最后,我們只是要通過發布訂閱者模式來觀察DOM事件而已。

function DataBinder( object_id ) { // Create a simple PubSub object var pubSub = {  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 ( var i = 0, 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 ) {  var target = evt.target || evt.srcElement, // IE8 compatibility    prop_name = target.getAttribute( data_attr );  if ( prop_name && prop_name !== "" ) {   pubSub.publish( message, prop_name, target.value );  } }; // Listen to change events and proxy to PubSub if ( document.addEventListener ) {  document.addEventListener( "change", changeHandler, false ); } else {  // IE8 uses attachEvent instead of addEventListener  document.attachEvent( "onchange", changeHandler ); } // PubSub propagates changes to all bound elements pubSub.on( message, function( evt, prop_name, new_val ) {var elements = document.querySelectorAll("[" + data_attr + "=" + prop_name + "]"),  tag_name;for ( var i = 0, 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; }} }); return pubSub;}

數據模型可以保持不變,除了在setter中對jQuery中trigger方法的調用,我們可以通過我們在PubSub中自定義的publish方法來代替。

// In the model's setter:function User( uid ) { // ... user = {  // ...  set: function( attr_name, val ) {     this.attributes[ attr_name ] = val;     // Use the `publish` method     binder.publish( uid + ":change", attr_name, val, this );  } } // ...}

通過實例講解,并又一次通過一百行不到,又可維護的純javascript完成了我們想要的結果,希望對大家實現javascript數據雙向綁定有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 青田县| 昆明市| 沂水县| 志丹县| 余江县| 皮山县| 龙岩市| 蒲江县| 黄平县| 盐池县| 乌拉特中旗| 济宁市| 葫芦岛市| 库尔勒市| 宜良县| 普兰店市| 额敏县| 英山县| 定远县| 青海省| 万宁市| 芒康县| 通化县| 林西县| 昆山市| 赣榆县| 读书| 汉沽区| 新竹县| 绥阳县| 巴中市| 泌阳县| 故城县| 平昌县| 容城县| 澳门| 香港 | 大荔县| 青州市| 那坡县| 布尔津县|