Adds the method .serializeJSON()
to jQuery (or Zepto) that serializes a form into a JavaScript Object with the same format as the default Ruby on Rails request params hash.
HTML form (input, textarea and select tags supported):
<form id="my-profile">
<!-- simple attribute -->
<input type="text" name="fullName" value="Mario Izquierdo" />
<!-- nested attributes -->
<input type="text" name="address[city]" value="San Francisco" />
<input type="text" name="address[state][name]" value="California" />
<input type="text" name="address[state][abbr]" value="CA" />
<!-- array -->
<input type="text" name="jobbies[]" value="code" />
<input type="text" name="jobbies[]" value="climbing" />
<!-- and more ... -->
<textarea name="projects[0][name]">serializeJSON</textarea>
<textarea name="projects[0][language]">javascript</textarea>
<input type="hidden" name="projects[0][popular]" value="0" />
<input type="checkbox" name="projects[0][popular]" value="1" checked="checked"/>
<textarea name="projects[1][name]">tinytest.js</textarea>
<textarea name="projects[1][language]">javascript</textarea>
<input type="hidden" name="projects[1][popular]" value="0" />
<input type="checkbox" name="projects[1][popular]" value="1"/>
</form>
JavaScript:
$('#my-profile').serializeJSON();
// returns =>
{
fullName: "Mario Izquierdo",
address: {
city: "San Francisco",
state: {
name: "California",
abbr: "CA"
}
},
jobbies: ["code", "climbing"],
projects: {
'0': { name: "serializeJSON", language: "javascript", popular: "1" },
'1': { name: "tinytest.js", language: "javascript", popular: "0" }
}
}
The serializeJSON
function returns a JavaScript object, not a JSON String. It should probably have been called serializeObject
, or something like that, but those names were already taken.
To serialize into JSON, use the JSON.stringify
method, that is available on all major new browsers.
To support old browsers, just include the json2.js polyfill (as described on stackoverfow).
var json = JSON.stringify(object);
Values are strings by default. But you can force values to be parsed with specific types by appending the type with a colon.
<form>
<input type="text" name="notype" value="default type is :string"/>
<input type="text" name="string:string" value=":string type overrides parsing options"/>
<input type="text" name="excluded:skip" value="Use :skip to not include this field in the result"/>
<input type="text" name="number[1]:number" value="1"/>
<input type="text" name="number[1.1]:number" value="1.1"/>
<input type="text" name="number[other stuff]:number" value="other stuff"/>
<input type="text" name="boolean[true]:boolean" value="true"/>
<input type="text" name="boolean[false]:boolean" value="false"/>
<input type="text" name="boolean[0]:boolean" value="0"/>
<input type="text" name="null[null]:null" value="null"/>
<input type="text" name="null[other stuff]:null" value="other stuff"/>
<input type="text" name="auto[string]:auto" value="text with stuff"/>
<input type="text" name="auto[0]:auto" value="0"/>
<input type="text" name="auto[1]:auto" value="1"/>
<input type="text" name="auto[true]:auto" value="true"/>
<input type="text" name="auto[false]:auto" value="false"/>
<input type="text" name="auto[null]:auto" value="null"/>
<input type="text" name="auto[list]:auto" value="[1, 2, 3]"/>
<input type="text" name="array[empty]:array" value="[]"/>
<input type="text" name="array[list]:array" value="[1, 2, 3]"/>
<input type="text" name="object[empty]:object" value="{}"/>
<input type="text" name="object[dict]:object" value='{"my": "stuff"}'/>
</form>
$('form').serializeJSON();
// returns =>
{
"notype": "default type is :string",
"string": ":string type overrides parsing options",
// :skip type removes the field from the output
"number": {
"1": 1,
"1.1": 1.1,
"other stuff": NaN, // <-- Other stuff parses as NaN (Not a Number)
},
"boolean": {
"true": true,
"false": false,
"0": false, // <-- "false", "null", "undefined", "", "0" parse as false
},
"null": {
"null": null, // <-- "false", "null", "undefined", "", "0" parse as null
"other stuff": "other stuff"
},
"auto": { // works as the parseAll option
"string": "text with stuff",
"0": 0, // <-- parsed as number
"1": 1, // <-- parsed as number
"true": true, // <-- parsed as boolean
"false": false, // <-- parsed as boolean
"null": null, // <-- parsed as null
"list": "[1, 2, 3]" // <-- array and object types are not auto-parsed
},
"array": { // <-- works using JSON.parse
"empty": [],
"not empty": [1,2,3]
},
"object": { // <-- works using JSON.parse
"empty": {},
"not empty": {"my": "stuff"}
}
}
By default:
- Values are always strings (unless using :types in the input names)
- Keys (names) are always strings (no auto-array detection by default)
- Unchecked checkboxes are ignored (as defined in the W3C rules for successful controls).
- Disabled elements are ignored (W3C rules)
This is because serializeJSON
is designed to return exactly the same as a regular HTML form submission when serialized as Rack/Rails params.
To change the default behavior you have the following options:
- parseBooleans: true, convert strings
"true"
and"false"
to booleanstrue
andfalse
. - parseNumbers: true, convert strings like
"1"
,"33.33"
,"-44"
to numbers like1
,33.33
,-44
. - parseNulls: true, convert the string
"null"
to the null valuenull
. - parseAll: true, all of the above. Makes default type to be
:auto
instead of:string
. - parseWithFunction: function, define your own parse function(inputValue, inputName) { return parsedValue }
- checkboxUncheckedValue: string, Use this value for unchecked checkboxes, instead of ignoring them. Make sure to use a String. If the value needs to be parsed (i.e. to a Boolean) use a parse option (i.e.
parseBooleans: true
). - useIntKeysAsArrayIndex: true, when using integers as keys, serialize as an array.
More details about options usage in the sections below.
If no :types are specified in the name, the values are always parsed as strings:
<form>
<input type="text" name="bool[true]" value="true"/>
<input type="text" name="bool[false]" value="false"/>
<input type="text" name="number[0]" value="0"/>
<input type="text" name="number[1]" value="1"/>
<input type="text" name="number[2.2]" value="2.2"/>
<input type="text" name="number[-2.25]" value="-2.25"/>
<input type="text" name="null" value="null"/>
<input type="text" name="string" value="text is always string"/>
<input type="text" name="empty" value=""/>
</form>
$('form').serializeJSON();
// returns =>
{
"bool": {
"true": "true",
"false": "false",
}
"number": {
"0": "0",
"1": "1",
"2.2": "2.2",
"-2.25": "-2.25",
}
"null": "null",
"string": "text is always string",
"empty": ""
}
Note that all values are strings, even if they look like booleans, numbers or nulls.
To auto-detect types, use the parse options. For example, to parse nulls and numbers:
$('form').serializeJSON({parseNulls: true, parseNumbers: true});
// returns =>
{
"bool": {
"true": "true", // booleans are still strings, because parseBooleans was not set
"false": "false",
}
"number": {
"0": 0, // numbers are parsed because parseNumbers: true
"1": 1,
"2.2": 2.2,
"-2.25": -2.25,
}
"null": null, // "null" strings are converted to null becase parseNulls: true
"string": "text is always string",
"empty": ""
}
For rare cases, a custom parser can be defined with a function:
var emptyStringsAndZerosToNulls = function(val, inputName) {
if (val === "") return null; // parse empty strings as nulls
if (val === 0) return null; // parse 0 as null
return val;
}
$('form').serializeJSON({parseWithFunction: emptyStringsAndZerosToNulls, parseNumbers: true});
// returns =>
{
"bool": {
"true": "true",
"false": "false",
}
"number": {
"0": null, // <-- parsed with custom function
"1": 1,
"2.2": 2.2,
"-2.25": -2.25,
}
"null": "null",
"string": "text is always string",
"empty": null // <-- parsed with custom function
}
In my opinion, the most confusing detail when serializing a form is the input type checkbox, that will include the value if checked, and nothing if unchecked.
To deal with this, it is a common practice to use hidden fields for the "unchecked" values:
<!-- Only one booleanAttr will be serialized, being "true" or "false" depending if the checkbox is selected or not -->
<input type="hidden" name="booleanAttr" value="false" />
<input type="checkbox" name="booleanAttr" value="true" />
This solution is somehow verbose, but it is unobtrusive and ensures progressive enhancement, because it works without JavaScript as well.
But, to make things easier, serializeJSON
includes the option checkboxUncheckedValue
and the possibility to add the attribute data-unchecked-value
to the checkboxes.
For example:
<form>
<input type="checkbox" name="check1" value="true" checked/>
<input type="checkbox" name="check2" value="true"/>
<input type="checkbox" name="check3" value="true"/>
</form>
Serializes like this by default:
$('form').serializeJSON();
// returns =>
{'check1': 'true'} // Note that check2 and check3 are not included because they are not checked
Which ignores any unchecked checkboxes.
To include all checkboxes, use the checkboxUncheckedValue
option like this:
$('form').serializeJSON({checkboxUncheckedValue: "false"});
// returns =>
{'check1': 'true', check2: 'false', check3: 'false'}
The "unchecked" value can also be specified via the HTML attribute data-unchecked-value
(Note this attribute is only recognized by the plugin):
<form id="checkboxes">
<input type="checkbox" name="checked[bool]" value="true" data-unchecked-value="false" checked/>
<input type="checkbox" name="checked[bin]" value="1" data-unchecked-value="0" checked/>
<input type="checkbox" name="checked[cool]" value="YUP" checked/>
<input type="checkbox" name="unchecked[bool]" value="true" data-unchecked-value="false" />
<input type="checkbox" name="unchecked[bin]" value="1" data-unchecked-value="0" />
<input type="checkbox" name="unchecked[cool]" value="YUP" /> <!-- No unchecked value specified -->
</form>
Serializes like this by default:
$('form#checkboxes').serializeJSON(); // Note no option is used
// returns =>
{
'checked': {
'bool': 'true',
'bin': '1',
'cool': 'YUP'
},
'unchecked': {
'bool': 'false',
'bin': '0'
// Note that unchecked cool does not appear, because it doesn't use data-unchecked-value
}
}
You can use both the option checkboxUncheckedValue
and the attribute data-unchecked-value
at the same time, in which case the attribute has precedence over the option.
And remember that you can combine it with other options to parse values as well.
$('form#checkboxes').serializeJSON({checkboxUncheckedValue: 'NOPE', parseBooleans: true, parseNumbers: true});
// returns =>
{
'checked': {
'bool': true,
'bin': 1,
'cool': 'YUP'
},
'unchecked': {
'bool': false, // value from data-unchecked-value attribute, and parsed with parseBooleans
'bin': 0, // value from data-unchecked-value attribute, and parsed with parseNumbers
'cool': 'NOPE' // value from checkboxUncheckedValue option
}
}
Using the option useIntKeysAsArrayIndex
.
For example:
<form>
<input type="text" name="arr[0]" value="foo"/>
<input type="text" name="arr[1]" value="var"/>
<input type="text" name="arr[5]" value="inn"/>
</form>
Serializes like this by default:
$('form').serializeJSON();
// returns =>
{'arr': {'0': 'foo', '1': 'var', '5': 'inn' }}
Which is how the Rack parse_nested_query method behaves (remember that serializeJSON input name format is inspired by Rails parameters, that are parsed using this Rack method).
But to interpret integers as array indexes, use the option useIntKeysAsArrayIndex
:
$('form').serializeJSON({useIntKeysAsArrayIndex: true});
// returns =>
{'arr': ['foo', 'var', undefined, undefined, undefined, 'inn']}
Note: that this was the default behavior of serializeJSON before version 2. Use this option for backwards compatibility.
All options defaults are defined in $.serializeJSON.defaultOptions
. You can just modify it to avoid setting the option on every call to serializeJSON
.
For example:
$.serializeJSON.defaultOptions.parseAll = true; // parse booleans, numbers and nulls by default
$('form').serializeJSON(); // No options => then use $.serializeJSON.defaultOptions
// returns =>
{
"bool": {
"true": true,
"false": false,
}
"number": {
"0": 0,
"1": 1,
"2.2": 2.2,
"-2.25": -2.25,
}
"null": null,
"string": "text is always string",
"empty": ""
}
Install it like any other jQuery plugin. For example, download the jquery.serializejson.min.js script and include it in your page after jQuery:
<script type="text/javascript" src="jquery.min.js"></script>
<script type="text/javascript" src="jquery.serializejson.min.js"></script>
I found others solving the same problem:
- https://github.com/macek/jquery-serialize-object
- https://github.com/hongymagic/jQuery.serializeObject
- https://github.com/danheberden/jquery-serializeForm
- https://github.com/maxatwork/form2js (plain js, no jQuery)
- https://github.com/serbanghita/formToObject.js (plain js, no jQuery)
- https://gist.github.com/shiawuen/2634143 (simpler but small)
But I still think this one is better because:
- It is built on top of jQuery (or Zepto)
serializeArray
, that creates a JavaScript array of objects, ready to be encoded as a JSON string. It takes in account the W3C rules for successful controls, makingserializeJSON
as standard, stable and crossbrowser as it can be. - The format suggested for the form field names is the same used by Rails, that is standard and well tested.
- The spec suite makes sure we don't break functionality on future versions.
- Compatible with bower.
- Compatible with zepto.js and pretty much every version of jQuery.
- The source code is as small as it can be. The minified version is 1Kb.
Probably to submit via AJAX, or to handle user input in your JavaScript application.
To submit a form using AJAX, the jQuery .serialize() function should work just fine. Most backend frameworks will understand the form attribute names and convert them to accessible values that can be easily assigned to your backend models.
Actually, the input name format used by .serializeJSON()
is borrowed from Rails Parameter Naming Conventions.
But if you want to handle user input in the frontend JavaScript application, then .serialize()
is not very useful because it just creates a params string. Another jQuery function is .serializeArray
, but it doesn't handle nested objects.
The current implementation of .serializeJSON()
relies on jQuery's .serializeArray() to grab the form attributes and then create the object using the names.
It means, it will serialize the inputs that are supported by .serializeArray()
, that uses the standard W3C rules for successful controls to determine which elements it should include; in particular the element cannot be disabled and must contain a name attribute. No submit button value is serialized since the form was not submitted using a button. Data from file select elements is not serialized.
Contributions are awesome. Feature branch pull requests are the preferred method. Just make sure to add tests for it. To run the jasmine specs, open spec/SpecRunner.html
in your browser.
- 2.4.1 (Oct 12, 2014): Add
:auto
type, that works like theparseAll
option, but targeted to a single input. - 2.4.0 (Oct 12, 2014): Implement :types. Types allow to easily specify how to parse each input.
- 2.3.2 (Oct 11, 2014): Bugfix #27 (parsing error on nested keys like name="foo[inn[bar]]"). Thanks to danlo for finding the issue.
- 2.3.1 (Oct 06, 2014): Bugfix #22 (ignore checkboxes with no name when doing
checkboxUncheckedValue
). Thanks to KATT for finding and fixing the issue. - 2.3.0 (Sep 25, 2014): Properly spell "data-unckecked-value", change for "data-unchecked-value"
- 2.2.0 (Sep 17, 2014): Add option
checkboxUncheckedValue
and attributedata-unckecked-value
to allow parsing unchecked checkboxes. - 2.1.0 (Jun 08, 2014): Add option
parseWithFunction
to allow custom parsers. And fix issue #14: empty strings were parsed as a zero whenparseNumbers
option was true. - 2.0.0 (May 04, 2014): Nested keys are always object attributes by default (discussed on issue #12). Set option
$.serializeJSON.defaultOptions.useIntKeysAsArrayIndex = true;
for backwards compatibility (see Options section). Thanks to joshuajabbour for finding the issue. - 1.3.0 (May 03, 2014): Accept options {parseBooleans, parseNumbers, parseNulls, parseAll} to modify what type to values are interpreted from the strings. Thanks to diaswrd for finding the issue.
- 1.2.3 (Apr 12, 2014): Lowercase filenames.
- 1.2.2 (Apr 03, 2014): Now also works with Zepto.js.
- 1.2.1 (Mar 17, 2014): Refactor, cleanup, lint code and improve test coverage.
- 1.2.0 (Mar 11, 2014): Arrays with empty index and objects with empty values are added and not overriden. Thanks to kotas.
- 1.1.1 (Feb 16, 2014): Only unsigned integers are used to create arrays. Alphanumeric keys are always for objects. Thanks to Nicocin.
- 1.0.2 (Jan 07, 2014): Tag to be on the jQuery plugin registry.
- 1.0.1 (Aug 20, 2012): Bugfix: ensure that generated arrays are being displayed when parsed with JSON.stringify
- 1.0.0 (Aug 20, 2012): Initial release
Written by Mario Izquierdo