web-dev-qa-db-ja.com

FormData(HTML5オブジェクト)をJSONに変換する方法

HTML5 FormDataオブジェクトをJSONに変換する方法は? JQueryを使用せず、FormDataのネストされたプロパティをオブジェクトのように処理します。

48

forEachオブジェクトでFormDataを直接使用することもできます。

var object = {};
formData.forEach(function(value, key){
    object[key] = value;
});
var json = JSON.stringify(object);

更新:

そして、 ES6矢印関数 で同じソリューションを好む人のために:

var object = {};
formData.forEach((value, key) => {object[key] = value});
var json = JSON.stringify(object);

更新2:

そして、複数の選択リストまたは複数の値を持つ他のフォーム要素のサポートが必要な場合(この問題に関する回答の下に非常に多くのコメントがあるため、可能な解決策を追加します)

var object = {};
formData.forEach((value, key) => {
    if(!object.hasOwnProperty(key)){
        object[key] = value;
        return;
    }
    if(!Array.isArray(object[key])){
        object[key] = [object[key]];    
    }
    object[key].Push(value);
});
var json = JSON.stringify(object);

ここではフィドル単純な複数選択リストでこのメソッドを使用する方法を示しています。

更新3:

ここで終わる人への補足として、フォームデータをjsonに変換する目的がXML HTTPリクエストを介してサーバーに送信する場合、変換せずにFormDataオブジェクトを直接送信できます。これと同じくらい簡単:

var request = new XMLHttpRequest();
request.open("POST", "http://example.com/submitform.php");
request.send(formData);

またFormDataオブジェクトの使用をMDNで参照 を参照してください:

77
Wilt

2019年、この種のタスクは非常に簡単になりました。

JSON.stringify(Object.fromEntries(formData));

Object.fromEntries :Chrome 73 +、Firefox 63 +、Safari 12.1でサポート

24
hakatashi

ライブラリを使用せずに、より機能的なスタイルでそれを行う方法を次に示します。

Array.from(formData.entries()).reduce((memo, pair) => ({
  ...memo,
  [pair[0]]: pair[1],
}), {});

例:

document.getElementById('foobar').addEventListener('submit', (e) => {
  e.preventDefault();

  const formData = new FormData(e.target);
  const data = Array.from(formData.entries()).reduce((memo, pair) => ({
    ...memo,
    [pair[0]]: pair[1],
  }), {});
  document.getElementById('output').innerHTML = JSON.stringify(data);
});
<form id='foobar'>
  <input name='baz' />
  <input type='submit' />
</form>

<pre id='output'>Input some value and submit</pre>
18
dzuc

同じ名前のエントリが複数ある場合、たとえば<SELECT multiple>を使用する場合や、同じ名前の複数の<INPUT type="checkbox">がある場合は、そのことに注意して値の配列を作成する必要があります。それ以外の場合は、最後に選択した値のみを取得します。

最新のES6バリアントは次のとおりです。

function formToJSON( elem ) {
  let output = {};
  new FormData( elem ).forEach(
    ( value, key ) => {
      // Check if property already exist
      if ( Object.prototype.hasOwnProperty.call( output, key ) ) {
        let current = output[ key ];
        if ( !Array.isArray( current ) ) {
          // If it's not an array, convert it to an array.
          current = output[ key ] = [ current ];
        }
        current.Push( value ); // Add the new value to the array.
      } else {
        output[ key ] = value;
      }
    }
  );
  return JSON.stringify( output );
}

少し古いコード(ただし、ForEachまたはentriesFormDataをサポートしていないため、IE11ではまだサポートされていません)

function formToJSON( elem ) {
  var current, entries, item, key, output, value;
  output = {};
  entries = new FormData( elem ).entries();
  // Iterate over values, and assign to item.
  while ( item = entries.next().value )
    {
      // assign to variables to make the code more readable.
      key = item[0];
      value = item[1];
      // Check if key already exist
      if (Object.prototype.hasOwnProperty.call( output, key)) {
        current = output[ key ];
        if ( !Array.isArray( current ) ) {
          // If it's not an array, convert it to an array.
          current = output[ key ] = [ current ];
        }
        current.Push( value ); // Add the new value to the array.
      } else {
        output[ key ] = value;
      }
    }
    return JSON.stringify( output );
  }
7
some

使いやすい機能

私は作成しました これのための関数

function FormDataToJSON(FormElement){    
    var formData = new FormData(FormElement);
    var ConvertedJSON= {};
    for (const [key, value]  of formData.entries())
    {
        ConvertedJSON[key] = value;
    }

    return ConvertedJSON
}

使用例

var ReceivedJSON = FormDataToJSON(document.getElementById('FormId');)

このコードでは、forループを使用して空のJSON変数を作成しました。すべての反復でkeysをformDataオブジェクトからJSONキーに使用しました。

このコードはGitHubのJSライブラリにあります。改善が必要な場合はここにコードを配置しました https://github.com/alijamal14/Utilities/blob/master/Utilities.js

6
Ali Jamal

この投稿はすでに1年前です...しかし、私はES6の@dzucの回答が本当に好きです。ただし、複数の選択またはチェックボックスを処理できないため、不完全です。これはすでに指摘されており、コードソリューションが提供されています。それらは重く、最適化されていません。そこで、これらのケースを処理するために、@ dzucに基づいて2つのバージョンを作成しました。

  • ASPスタイルフォームの場合、複数のアイテム名を単純に繰り返すことができます。
let r=Array.from(fd).reduce(
  (o , [k,v]) => (
     (!o[k])
     ? {...o , [k] : v}
     : {...o , [k] : [...o[k] , v]}
   )
   ,{}
);
let obj=JSON.stringify(r);

1行のHotshotバージョン:

Array.from(fd).reduce((o,[k,v])=>((!o[k])?{...o,[k]:v}:{...o,[k]:[...o[k],v]}),{});
  • 複数のアイテム名に[]サフィックスが必要なPHPスタイルのフォームの場合。
let r=Array.from(fd).reduce(
  (o , [k,v]) => (
    (k.split('[').length>1)
    ? (k=k.split('[')[0]
      , (!o[k])
      ? {...o , [k] : [v]}
      : {...o , [k] : [...o[k] , v ]}
    )
    : {...o , [k] : v}
  )
  ,{}
);
let obj=JSON.stringify(r);

1行のHotshotバージョン:

Array.from(fd).reduce((o,[k,v])=>((k.split('[').length>1)?(k=k.split('[')[0],(!o[k])?{...o,[k]:[v]}:{...o,[k]:[...o[k],v]}):{...o,[k]:v}),{});
  • マルチレベル配列をサポートするPHPフォームの拡張。

前回私が前の2番目のケースを書いたときから、職場ではPHPフォームにマルチレベルのチェックボックスがあるというケースがありました。以前のケースとこれをサポートする新しいケースを作成しました。このケースをよりよく紹介するためにスニペットを作成しました。このデモの結果はコンソールに表示され、必要に応じて変更します。パフォーマンスを損なうことなく、できる限り最適化しようとしましたが、人間の可読性を損ないました。配列はオブジェクトであり、配列を指す変数は参照として保持されるという利点があります。これのホットショットはありません、私のゲストになります。

let nosubmit = (e) => {
  e.preventDefault();
  const f = Array.from(new FormData(e.target));
  const obj = f.reduce((o, [k, v]) => {
    let a = v,
      b, i,
      m = k.split('['),
      n = m[0],
      l = m.length;
    if (l > 1) {
      a = b = o[n] || [];
      for (i = 1; i < l; i++) {
        m[i] = (m[i].split(']')[0] || b.length) * 1;
        b = b[m[i]] = ((i + 1) == l) ? v : b[m[i]] || [];
      }
    }
    return { ...o, [n]: a };
  }, {});
  console.log(obj);
}
document.querySelector('#theform').addEventListener('submit', nosubmit, {capture: true});
<h1>Multilevel Form</h1>
<form action="#" method="POST" enctype="multipart/form-data" id="theform">
  <input type="hidden" name="_id" value="93242" />
  <input type="hidden" name="_fid" value="45c0ec96929bc0d39a904ab5c7af70ef" />
  <label>Select:
    <select name="uselect">
      <option value="A">A</option>
      <option value="B">B</option>
      <option value="C">C</option>
    </select>
  </label>
  <br /><br />
  <label>Checkboxes one level:<br/>
    <input name="c1[]" type="checkbox" checked value="1"/>v1 
    <input name="c1[]" type="checkbox" checked value="2"/>v2
    <input name="c1[]" type="checkbox" checked value="3"/>v3
  </label>
  <br /><br />
  <label>Checkboxes two levels:<br/>
    <input name="c2[0][]" type="checkbox" checked value="4"/>0 v4 
    <input name="c2[0][]" type="checkbox" checked value="5"/>0 v5
    <input name="c2[0][]" type="checkbox" checked value="6"/>0 v6
    <br/>
    <input name="c2[1][]" type="checkbox" checked value="7"/>1 v7 
    <input name="c2[1][]" type="checkbox" checked value="8"/>1 v8
    <input name="c2[1][]" type="checkbox" checked value="9"/>1 v9
  </label>
  <br /><br />
  <label>Radios:
    <input type="radio" name="uradio" value="yes">YES
    <input type="radio" name="uradio" checked value="no">NO
  </label>
  <br /><br />
  <input type="submit" value="Submit" />
</form>
5
CarlosH.

FormData() オブジェクトを使用してこれを実現できます。このFormDataオブジェクトには、キーの各要素のnameプロパティと値の送信値を使用して、フォームの現在のキー/値が入力されます。また、ファイル入力コンテンツをエンコードします。

例:

var myForm = document.getElementById('myForm');
myForm.addEventListener('submit', function(event)
{
    event.preventDefault();
    var formData = new FormData(myForm),
        result = {};

    for (var entry of formData.entries())
    {
        result[entry[0]] = entry[1];
    }
    result = JSON.stringify(result)
    console.log(result);

});
5
GiriB

FormDataメソッド.entriesおよびfor ofサポートされていません IE11およびSafari。

Safari、Chrome、Firefox、およびEdgeをサポートするための簡易バージョンを以下に示します

function formDataToJSON(formElement) {    
    var formData = new FormData(formElement),
        convertedJSON = {};

    formData.forEach(function(value, key) { 
        convertedJSON[key] = value;
    });

    return convertedJSON;
}

警告:この回答はIE11では機能しません。
FormDataにはIE11のforEachメソッドがありません。
私はまだすべての主要なブラウザをサポートする最終的なソリューションを探しています。

4
Tomas Prado

Lodashを使用している場合は、fromPairsを使用して簡潔に実行できます。

import {fromPairs} from 'lodash';

const object = fromPairs(Array.from(formData.entries()));
3
Erik van Velzen

@dzucからの回答はすでに非常に優れていますが、配列の構造化(最新のブラウザーまたはBabelで利用可能)を使用して、さらにエレガントにすることができます。

// original version from @dzuc
const data = Array.from(formData.entries())
  .reduce((memo, pair) => ({
    ...memo,
    [pair[0]: pair[1],
  }), {})

// with array destructuring
const data = Array.from(formData.entries())
  .reduce((memo,[key, value]) => ({
    ...memo,
    [key]: value,
  }), {})
2
Jeremias Erbs

次のアイテムがニーズを満たしている場合、あなたは幸運です:

  1. [['key','value1'], ['key2','value2']のような配列の配列(FormDataが提供するもののような)を{key1: 'value1', key2: 'value2'}のようなキー->値オブジェクトに変換し、それをJSONストリングに変換します。
  2. 最新のES6インタープリターでブラウザー/デバイスをターゲットにしている、またはbabelのようなものでコンパイルしている。
  3. これを実現する最も小さな方法が必要です。

必要なコードは次のとおりです。

const data = new FormData(document.querySelector('form'));
const json = JSON.stringify(Array.from(data).reduce((o,[k,v])=>(o[k]=v,o),{}));

これが誰かを助けることを願っています。

1
KyleFarris

これを試すことができます

formDataToJSON($('#form_example'));

# Create a function to convert the serialize and convert the form data
# to JSON
# @param : $('#form_example');
# @return a JSON Stringify
function formDataToJSON(form) {
    let obj = {};
    let formData = form.serialize();
    let formArray = formData.split("&");

    for (inputData of formArray){
        let dataTmp = inputData.split('=');
        obj[dataTmp[0]] = dataTmp[1];
    }
    return JSON.stringify(obj);
}
1
Ivan Fretes

虐待的なワンライナー!

Array.from(fd).reduce((obj, [k, v]) => ({...obj, [k]: v}), {});

今日、Firefoxにはオブジェクトスプレッドのサポートと配列の構造化があることがわかりました!

1
nackjicholson

私のために働いた

                var myForm = document.getElementById("form");
                var formData = new FormData(myForm),
                obj = {};
                for (var entry of formData.entries()){
                    obj[entry[0]] = entry[1];
                }
                console.log(obj);