将表单数据序列化为JSON

时间:2012-07-05 06:21:31

标签: jquery backbone.js underscore.js

我想在Backbone.js 模型中对表单进行一些服务器前验证。为此,我需要将表单中的用户输入转换为可用数据。 我发现了三种方法:

  1. var input = $("#inputId").val();
  2. var input = $("form.login").serialize();
  3. var input = $("form.login").serializeArray();
  4. 不幸的是,没有一个提供我需要的良好的可重用和可开发的JSON对象。我已经查看了Stack Overflow上的几个问题,但我发现只有一些额外的库。

    不是Underscore.js,当前的jQuery或Backbone.js是否提供了辅助方法?

    我无法想象没有要求这样的功能。

    HTML

    <form class="login">
        <label for="_user_name">username:</label>
        <input type="text" id="_user_name" name="user[name]" value="dev.pus" />
        <label for="_user_pass">password:</label>
        <input type="password" id="_user_pass" name="user[pass]" value="1234" />
        <button type="submit">login</button>
    </form>
    

    的JavaScript

    var formData = $("form.login").serializeObject();
    console.log(formData);
    

    输出

    {
        "name": "dev.pus",
        "pass": "1234"
    }
    

    Backbone.js模型

    var user = new User(formData);
    user.save();
    

15 个答案:

答案 0 :(得分:177)

以下是此用例的函数:

function getFormData($form){
    var unindexed_array = $form.serializeArray();
    var indexed_array = {};

    $.map(unindexed_array, function(n, i){
        indexed_array[n['name']] = n['value'];
    });

    return indexed_array;
}

用法:

var $form = $("#form_data");
var data = getFormData($form);

答案 1 :(得分:123)

你可以这样做:

function onSubmit( form ){
  var data = JSON.stringify( $(form).serializeArray() ); //  <-----------

  console.log( data );
  return false; //don't submit
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>

<form onsubmit='return onSubmit(this)'>
  <input name='user' placeholder='user'><br>
  <input name='password' type='password' placeholder='password'><br>
  <button type='submit'>Try</button>
</form>

请参阅:http://www.json.org/js.html

答案 2 :(得分:51)

以下代码可以帮助您。 :)

 //The function is based on http://css-tricks.com/snippets/jquery/serialize-form-to-json/
 <script src="//code.jquery.com/jquery-2.1.0.min.js"></script>

<script>
    $.fn.serializeObject = function() {
        var o = {};
        var a = this.serializeArray();
        $.each(a, function() {
            if (o[this.name]) {
                if (!o[this.name].push) {
                    o[this.name] = [o[this.name]];
                }
                o[this.name].push(this.value || '');
            } else {
                o[this.name] = this.value || '';
            }
        });
        return o;
    };

    $(function() {
        $('form.login').on('submit', function(e) {
          e.preventDefault();

          var formData = $(this).serializeObject();
          console.log(formData);
          $('.datahere').html(formData);
        });
    });
</script>

答案 3 :(得分:41)

使用:

var config = {};
jQuery(form).serializeArray().map(function(item) {
    if ( config[item.name] ) {
        if ( typeof(config[item.name]) === "string" ) {
            config[item.name] = [config[item.name]];
        }
        config[item.name].push(item.value);
    } else {
        config[item.name] = item.value;
    }
});

答案 4 :(得分:25)

我知道这不符合辅助函数的要求,但我这样做的方法是使用jQuery的$ .each()方法

var loginForm = $('.login').serializeArray();
var loginFormObject = {};
$.each(loginForm,
    function(i, v) {
        loginFormObject[v.name] = v.value;
    });

然后我可以将loginFormObject传递给我的后端,或者你可以创建一个userobject并将它保存在主干中。

答案 5 :(得分:10)

我无法找到解决此问题的答案:

[{name:"Vehicle.Make", value: "Honda"}, {name:"Vehicle.VIN", value: "123"}]

这需要这个对象:

{Vehicle: {Make: "Honda", "VIN": "123"}}

所以我必须编写一个我自己的序列化器来解决这个问题:

function(formArray){
        var obj = {};
        $.each(formArray, function(i, pair){
            var cObj = obj, pObj, cpName;
            $.each(pair.name.split("."), function(i, pName){
                pObj = cObj;
                cpName = pName;
                cObj = cObj[pName] ? cObj[pName] : (cObj[pName] = {});
            });
            pObj[cpName] = pair.value;
        });
        return obj;
    }

也许它会帮助某人。

答案 6 :(得分:9)

如果您不关心具有相同名称的重复表单元素,则可以执行以下操作:

var data = $("form.login").serializeArray();
var formData = _.object(_.pluck(data, 'name'), _.pluck(data, 'value'));

我在这里使用Underscore.js

答案 7 :(得分:8)

尝试解决同样的问题(验证而不进入复杂的插件和库),我创建了jQuery.serializeJSON,它改进了serializeArray以支持任何类型的嵌套对象。

这个插件很受欢迎,但在另一个项目中我使用Backbone.js,我想在Backbone.js模型中编写验证逻辑。然后我创建了Backbone.Formwell,它允许您直接在表单中显示验证方法返回的错误。

答案 8 :(得分:5)

如果您要发送带有JSON的表单,则必须删除发送字符串中的[]。您可以使用jQuery函数serializeObject():

来实现
var frm = $(document.myform);
var data = JSON.stringify(frm.serializeObject());

$.fn.serializeObject = function() {
    var o = {};
    //    var a = this.serializeArray();
    $(this).find('input[type="hidden"], input[type="text"], input[type="password"], input[type="checkbox"]:checked, input[type="radio"]:checked, select').each(function() {
        if ($(this).attr('type') == 'hidden') { //if checkbox is checked do not take the hidden field
            var $parent = $(this).parent();
            var $chb = $parent.find('input[type="checkbox"][name="' + this.name.replace(/\[/g, '\[').replace(/\]/g, '\]') + '"]');
            if ($chb != null) {
                if ($chb.prop('checked')) return;
            }
        }
        if (this.name === null || this.name === undefined || this.name === '')
            return;
        var elemValue = null;
        if ($(this).is('select'))
            elemValue = $(this).find('option:selected').val();
        else elemValue = this.value;
        if (o[this.name] !== undefined) {
            if (!o[this.name].push) {
                o[this.name] = [o[this.name]];
            }
            o[this.name].push(elemValue || '');
        } else {
            o[this.name] = elemValue || '';
        }
    });
    return o;
}

答案 9 :(得分:5)

以下是我作为模块使用的情况(在我的formhelper.js中):

define(function(){
    FormHelper = {};

    FormHelper.parseForm = function($form){
        var serialized = $form.serializeArray();
        var s = '';
        var data = {};
        for(s in serialized){
            data[serialized[s]['name']] = serialized[s]['value']
        }
        return JSON.stringify(data);
    }

    return FormHelper;
});

有点糟透的是,我似乎无法找到另一种方法去做我想做的事情。

这确实为我返回了这个JSON:

{"first_name":"John","last_name":"Smith","age":"30"}

答案 10 :(得分:5)

使用Underscore.js

function serializeForm($form){
    return _.object(_.map($form.serializeArray(), function(item){return [item.name, item.value]; }));
}

答案 11 :(得分:3)

使用jQuery并避免serializeArray,以下代码序列化并以JSON格式发送表单数据:

$("#commentsForm").submit(function(event){
    var formJqObj = $("#commentsForm");
    var formDataObj = {};
    (function(){
        formJqObj.find(":input").not("[type='submit']").not("[type='reset']").each(function(){
            var thisInput = $(this);
            formDataObj[thisInput.attr("name")] = thisInput.val();
        });
    })();
    $.ajax({
        type: "POST",
        url: YOUR_URL_HERE,
        data: JSON.stringify(formDataObj),
        contentType: "application/json"
    })
    .done(function(data, textStatus, jqXHR){
        console.log("Ajax completed: " + data);
    })
    .fail(function(jqXHR, textStatus, errorThrown){
        console.log("Ajax problem: " + textStatus + ". " + errorThrown);
    });
    event.preventDefault();
});

答案 12 :(得分:2)

我的贡献:

function serializeToJson(serializer){
    var _string = '{';
    for(var ix in serializer)
    {
        var row = serializer[ix];
        _string += '"' + row.name + '":"' + row.value + '",';
    }
    var end =_string.length - 1;
    _string = _string.substr(0, end);
    _string += '}';
    console.log('_string: ', _string);
    return JSON.parse(_string);
}

var params = $('#frmPreguntas input').serializeArray();
params = serializeToJson(params);

答案 13 :(得分:1)

嗯,这是一个方便的插件:https://github.com/macek/jquery-serialize-object

它的问题是:

  

继续前进,除了核心序列化之外,.serializeObject将支持对布尔值和数值进行正确的序列化,从而为这两种情况生成有效类型。

     

期待这些在&gt; = 2.1.0

答案 14 :(得分:-1)

找到一个可能的帮手:

https://github.com/theironcook/Backbone.ModelBinder

对于那些根本不想接触表格的人: https://github.com/powmedia/backbone-forms

我将仔细研究第一个链接并提供一些反馈:)