【问题标题】:Ajax data two-way data binding strategies?Ajax数据双向数据绑定策略?
【发布时间】:2011-01-19 04:36:39
【问题描述】:

我愿意 1) 绘制创建表单字段并使用来自 javascript 对象的数据填充它们 2) 只要​​表单字段的值发生变化,就更新这些支持对象

1 号很简单。我有几个 js 模板系统,我一直在使用,效果很好。

第 2 点可能需要一些思考。对“ajax 数据绑定”的快速谷歌搜索出现了一些看起来基本上是单向的系统。它们旨在基于支持 js 对象更新 UI,但似乎没有解决如何在对 UI 进行更改时更新这些支持对象的问题。任何人都可以推荐任何可以为我做这件事的图书馆吗?这是我自己写出来的东西,没有太多麻烦,但如果这个问题已经想清楚了,我宁愿不要重复这个工作。

//////////////////////////////////////////////////////////////////////////////////////////////////////////////

我已经创建了我自己的 jquery 插件来完成这个。这里是。请让我知道它是否有用,如果你认为它可能值得让它更“官方”。如果您有任何问题或疑问,也请告诉我。

/*

    Takes a jquery object and binds its form elements with a backing javascript object. Takes two arguments: the object 
    to be bound to, and an optional "changeListener", which must implement a "changeHappened" method.

    Example: 

    // ============================
    // =     backing object       =
    // ============================

    demoBean = {
        prop1 : "val",
        prop2 : [
            {nestedObjProp:"val"},
            {nestedObjProp:"val"}
        ],
        prop3 : [
            "stringVal1",
            "stringVal12"
        ]
    }

    // ===========================
    // =        FORM FIELD       =
    // ===========================

    <input class="bindable" name="prop2[1].nestedObjProp">


    // ===========================
    // =       INVOCATION        =
    // ===========================

    $jq(".bindable").bindData( 
        demoBean, 
        {changeHappened: function(){console.log("change")}}
    )


*/


(function($){


    // returns the value of the property found at the given path
    // plus a function you can use to set that property
    var navigateObject = function(parentObj, pathArg){
        var immediateParent = parentObj;
        var path = pathArg
            .replace("[", ".")
            .replace("]", "")
            .replace("].", ".")
            .split(".");
        for(var i=0; i< (path.length-1); i++){
            var currentPathKey = path[i];
            immediateParent = immediateParent[currentPathKey];
            if(immediateParent === null){
                throw new Error("bindData plugin encountered a null value at  " + path[i] + " in path" + path);
            }
        }

        return {
            value: immediateParent[path[path.length - 1]],
            set: function(val){
                immediateParent[path[path.length - 1]] = val
            },
            deleteObj: function(){
                if($.isArray(immediateParent)){
                    immediateParent.splice(path[path.length - 1], 1);
                }else{
                    delete  immediateParent[path[path.length - 1]];
                }
            } 
        }

    }

    var isEmpty = function(str){
        return str == null || str == "";
    }

    var bindData = function(parentObj, changeListener){

        var parentObj,
            radioButtons = [];
        var changeListener;
        var settings;
        var defaultSettings = {
            // if this flag is true, you can put a label in a field,
            // like <input value="Phone Number"/>, and the value
            // won't be replaced by a blank value in the parentObj
            // Additionally, if the user clicks on the field, the field will be cleared.
            allowLabelsInfields: true 
        };

        // allow two forms: 
        // function(parentObj, changeListener)
        // and function(settings). 
        if(arguments.length == 2){
            parentObj = arguments[0];
            changeListener = arguments[1]
            settings = defaultSettings;
        }else{  
            settings = $jq.extend(defaultSettings, arguments[0]);
            parentObj = settings.parentObj;
            changeListener = settings.changeListener;
        }

        var changeHappened = function(){};
        if(typeof changeListener != "undefined"){
            if(typeof changeListener.changeHappened == "function"){
                changeHappened = changeListener.changeHappened;
            }else{
                throw new Error("A changeListener must have a method called 'changeHappened'.");
            }
        };
        this.each(function(key,val){
            var formElem = $(val);
            var tagName = formElem.attr("tagName").toLowerCase();
            var fieldType;
          if(tagName == "input"){
            fieldType = formElem.attr("type").toLowerCase();
          }else{
            fieldType = tagName;
          }


            // Use the "name" attribute as the address of the property we want to bind to.
            // Except if it's a radio button, in which case, use the "value" because "name" is the name of the group
            // This should work for arbitrarily deeply nested data. 
            var navigationResult = navigateObject(parentObj, formElem.attr(fieldType === "radio"? "value" : "name"));

            // populate the field with the data in the backing object

            switch(fieldType){

        // is it a radio button? If so, check it or not based on the 
        // boolean value of navigationResult.value
        case "radio":
          radioButtons.push(formElem);
          formElem.data("bindDataPlugin", {navigationResult: navigationResult});
          formElem.attr("checked", navigationResult.value);
          formElem.change(function(){
            // Radio buttons only seem to update when _selected_, not 
            // when deselected. So if one is clicked, update the bound
            // object for all of them. I know it's a little ugly,
            // but it works.
            $jq.each(radioButtons, function(index, button){
              var butt = $jq(button);
              butt.data("bindDataPlugin").navigationResult.set(butt.attr("checked"));
            });
            navigationResult.set(formElem.attr("checked"));           
            changeHappened();
          });
          break;

        case "text":
          // if useFieldLabel is true, it means that the field is 
          // self-labeling. For example, an email field whose 
          // default value is "Enter Email".
          var useFieldLabel = isEmpty( navigationResult.value )
                   && !isEmpty( formElem.val() )  
                   && settings.allowLabelsInfields;
          if(useFieldLabel){
           var labelText = formElem.val();
           formElem.click(function(){
             if(formElem.val() === labelText){
               formElem.val("");
             }
           })
          }else{
           formElem.attr("value", navigationResult.value);
          }
          formElem.keyup(function(){
           navigationResult.set(formElem.attr("value"));
           changeHappened();
          });

          break;

        case "select":
          var domElem = formElem.get(0);
                $jq.each(domElem.options, function(index, option){
                    if(option.value === navigationResult.value){
                        domElem.selectedIndex = index;
                    }
                });
                formElem.change(function(){
                    navigationResult.set(formElem.val());
                })
          break;

        case "textarea":
          formElem.text(navigationResult.value);
          formElem.keyup(function(){
           changeHappened();
           navigationResult.set(formElem.val());
          });
          break;
      }


        });
        return this;
    };

    bindData.navigateObject = navigateObject;

    $.fn.bindData = bindData;

})(jQuery);

【问题讨论】:

  • 您能否将您的“jQuery 插件”编辑转换为答案,以便我为它投票?
  • @morgancodes:我从您的代码中创建了一个JsFiddle example,并将console.log 替换为与浏览器无关的内容。你能看看吗?我不明白你的代码应该做什么。

标签: javascript ajax data-binding


【解决方案1】:

有大量的库可以实现你想要的。

对于初学者,您可以使用DWR 来获取Ajax 功能。为表单字段的onChange 事件触发的方法应该对相应的支持对象进行 DWR 调用

希望这会有所帮助!

【讨论】:

  • 谢谢 Mihir,我在想一些更轻量级的东西,可能只是前端,或者与服务器无关。但我会看看 DWR。
【解决方案2】:

DataBind - 一个 jQuery 模板插件。该库有一个方法unbinddata(),它将数据收集回json-array。

不幸的是,该库仅适用于表单输入。

如果你解决了,我可以看看吗?

【讨论】:

  • 听起来可能与我创建的非常相似。贴在上面。
【解决方案3】:

Microsoft 提出了一项增强 jQuery 核心的提案,该核心实现了体面的到达数据绑定,包括两个绑定(源到目标,以及目标到源)。

他们称之为 Data Linking - 这很奇怪,因为这个概念在所有其他 MS 技术(WinForms、WPF、WebForms 等)中的名称是 DataBinding.

Link to the MS Proposal

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2013-02-24
    • 2015-08-08
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-11-18
    • 2014-01-30
    相关资源
    最近更新 更多