JavascriptでネストされたJSONをHTMLのネストされたリストに変換する

StackOverflow https://stackoverflow.com/questions/1211764

  •  06-07-2019
  •  | 
  •  

質問

(XMLとは対照的に)JSONを使用するのはかなり新しいので、現在は返されたJSONデータをダイジェスト、解析、表示するためにJavascriptのみを使用しています。

JSON2.jsライブラリを使用しており、かなり単純なネストされたリストを表す有効なJSONデータを取得しています:

{
  "node":{
    "class":"folder",
    "title":"Test Framework",
    "node":{
      "class":"folder",
      "title":"Item 1",
      "node":{
        "class":"folder",
        "title":"Item 1.1",
        "node":{
          "class":"file",
          "title":"Item 1.1.a"
        }
      },
      "node":{
        "class":"folder",
        "title":"Item 1.2",
        "node":{
          "class":"file",
          "title":"Item 1.2.a"
        },
        "node":{
          "class":"file",
          "title":"Item 1.2.b"
        },
        "node":{
          "class":"file",
          "title":"Item 1.2.c"
        }
      },
      "node":{
        "class":"folder",
        "title":"Item 1.3",
        "node":{
          "class":"folder",
          "title":"Item 1.3.a",
          "node":{
            "class":"file",
            "title":"Item 1.3.a.i"
          },
          "node":{
            "class":"file",
            "title":"Item 1.3.a.ii"
          }
        }
      }
    },
    "node":{
      "class":"folder",
      "title":"Item 2",
      "node":{
        "class":"file",
        "title":"item 2.a"
      },
      "node":{
        "class":"file",
        "title":"Item 2.b"
      }
    }
  }
}

すべてのネストされた要素を使用して、そのロットをULにすばやく変換するためのポインタがありますか? <!> quot; class <!> quot; JSONにある要素は、各LIのクラスとして使用できます。

ご協力いただければ幸いです。

ありがとう、

デイブ。

役に立ちましたか?

解決

あなたのjsonはあなたのタスクに適していない。一部のオブジェクトには、同じ名前(<!> quot; node <!> quot;)のプロパティがいくつかあるため、互いにオーバーライドしています。代わりにノードの配列を使用する必要があります。作業データ構造と、それをネストされたリストに変換できる関数を次に示します。

<!DOCTYPE HTML PUBLIC"-//W3C//DTD HTML 4.01//EN"
    "http://www.w3.org/TR/html4/strict.dtd">
<html>
<head>
<title></title>
<script type="text/javascript">
function parseNodes(nodes) { // takes a nodes array and turns it into a <ol>
    var ol = document.createElement("OL");
    for(var i=0; i<nodes.length; i++) {
        ol.appendChild(parseNode(nodes[i]));
    }
    return ol;
}

function parseNode(node) { // takes a node object and turns it into a <li>
    var li = document.createElement("LI");
    li.innerHTML = node.title;
    li.className = node.class;
    if(node.nodes) li.appendChild(parseNodes(node.nodes));
    return li;
}

window.jsonData = [{
    "class": "folder",
    "title": "Test Framework",
    "nodes": [{
        "class": "folder",
        "title": "Item 1",
        "nodes": [{
            "class": "folder",
            "title": "Item 1.1",
            "nodes": [{
                "class": "file",
                "title": "Item 1.1.a"
            }]
        },
        {
            "class": "folder",
            "title": "Item 1.2",
            "nodes": [{
                "class": "file",
                "title": "Item 1.2.a"
            },
            {
                "class": "file",
                "title": "Item 1.2.b"
            },
            {
                "class": "file",
                "title": "Item 1.2.c"
            }]
        },
        {
            "class": "folder",
            "title": "Item 1.3",
            "nodes": [{
                "class": "folder",
                "title": "Item 1.3.a",
                "nodes": [{
                    "class": "file",
                    "title": "Item 1.3.a.i"
                },
                {
                    "class": "file",
                    "title": "Item 1.3.a.ii"
                }]
            }]
        }]
    },
    {
        "class": "folder",
        "title": "Item 2",
        "nodes": [{
            "class": "file",
            "title": "item 2.a"
        },
        {
            "class": "file",
            "title": "Item 2.b"
        }]
    }]
}];

</script>
</head>
<body>
    <input type="button" 
    onclick="document.body.appendChild(parseNodes(jsonData))"
    value="go" />
</body>
</html>

このcssを追加して、アイテムの番号付けをノードのタイトルに一致させることができます:)

<style type="text/css">
ol { list-style-type: none }
ol ol { list-style-type: decimal }
ol ol ol { list-style-type: decimal }
ol ol ol ol { list-style-type: lower-alpha }
ol ol ol ol ol { list-style-type: lower-roman }
</style>

実際の動作を見る

他のヒント

各オブジェクトに<ul>要素を作成し、各部分に<li>要素を作成するかなり単純なコードを次に示します。特別なケースを追加する場合は、クイックチェックを追加して、childなどの要素に対して"class"変数をテストできます。

function jsonToHtmlList(json) {
    return objToHtmlList(JSON.parse(json));
}

function objToHtmlList(obj) {
    if (obj instanceof Array) {
        var ol = document.createElement('ol');
        for (var child in obj) {
            var li = document.createElement('li');
            li.appendChild(objToHtmlList(obj[child]));
            ol.appendChild(li);
        }
        return ol;
    }
    else if (obj instanceof Object && !(obj instanceof String)) {
        var ul = document.createElement('ul');
        for (var child in obj) {
            var li = document.createElement('li');
            li.appendChild(document.createTextNode(child + ": "));
            li.appendChild(objToHtmlList(obj[child]));
            ul.appendChild(li);
        }
        return ul;
    }
    else {
        return document.createTextNode(obj);
    }
}

これは、JSONが意味をなさないため、望みどおりの動作をしません。 JavaScriptのオブジェクト、つまりJSONはマップであるため、同じ名前の子を複数持つことはできません。 C <!>#233; dricが指摘しているように、複数の<!> quot; node <!> quot;を配列に変換する必要があります。

json構造を自分で歩いて、必要なときに適切な<li>を作成できますが、Composite Patternを実装することをお勧めします。詳細については、リンクをご覧ください

ライセンス: CC-BY-SA帰属
所属していません StackOverflow
scroll top