admin管理员组文章数量:1357592
I am trying to covert a JSON file to HTML using vanilla JavaScript. The way I approached it appears to be working but the code is not ideal.
What I am doing is checking to see if the value of the key is an object. If so, this is a new HTML tag. If not then this is a attribute for the tag.
The problem that I am having is that I don't know how to properly loop through an object and test these conditions without duplicating my code.
Can someone help me identify how I can better write this?
function createComponent(ponent, defaults, map) {
Object.keys(defaults).forEach(function (level1) {
if (typeof defaults[level1] === "object") {
var tag = document.createElement(level1);
Object.keys(defaults[level1]).forEach(function (level2) {
if (typeof defaults[level1][level2] === "object") {
tag.appendChild(document.createElement(level2));
Object.keys(defaults[level1][level2]).forEach(function (level3) {
if (typeof defaults[level1][level2][level3] === "object") {
console.log(defaults[level1][level2][level3]);
tag.appendChild(document.createElement(level3));
}
});
}
});
}
});
}
createComponent("textBox", textBoxDefaults, map);
Pseudo code:
Check defaults object, is there an object at this level? If so, create an HTML tag with the same Key name. Is there values that are not objects? If so, add attributes to the created tag using the key value pair. Stop diving deeper into the JSON when you don't find anymore objects.
Sample JSON
textbox: {
id: 1,
name: "Text Box",
tmprops: {
cur: 0,
min: 5000,
visible: true,
},
tmctxlst: {
version: "2",
txttmctx: {
alwysshw: false,
name: "default"
}
},
Desired Output
<textbox id="1" name="text box">
<tmprops cur="0" min="5000" visible="true" />
<tmctxlst version="2">
<txttmctx alwysshow="false" name="default">
</tmctxlst>
</textbox>
I am trying to covert a JSON file to HTML using vanilla JavaScript. The way I approached it appears to be working but the code is not ideal.
What I am doing is checking to see if the value of the key is an object. If so, this is a new HTML tag. If not then this is a attribute for the tag.
The problem that I am having is that I don't know how to properly loop through an object and test these conditions without duplicating my code.
Can someone help me identify how I can better write this?
function createComponent(ponent, defaults, map) {
Object.keys(defaults).forEach(function (level1) {
if (typeof defaults[level1] === "object") {
var tag = document.createElement(level1);
Object.keys(defaults[level1]).forEach(function (level2) {
if (typeof defaults[level1][level2] === "object") {
tag.appendChild(document.createElement(level2));
Object.keys(defaults[level1][level2]).forEach(function (level3) {
if (typeof defaults[level1][level2][level3] === "object") {
console.log(defaults[level1][level2][level3]);
tag.appendChild(document.createElement(level3));
}
});
}
});
}
});
}
createComponent("textBox", textBoxDefaults, map);
Pseudo code:
Check defaults object, is there an object at this level? If so, create an HTML tag with the same Key name. Is there values that are not objects? If so, add attributes to the created tag using the key value pair. Stop diving deeper into the JSON when you don't find anymore objects.
Sample JSON
textbox: {
id: 1,
name: "Text Box",
tmprops: {
cur: 0,
min: 5000,
visible: true,
},
tmctxlst: {
version: "2",
txttmctx: {
alwysshw: false,
name: "default"
}
},
Desired Output
<textbox id="1" name="text box">
<tmprops cur="0" min="5000" visible="true" />
<tmctxlst version="2">
<txttmctx alwysshow="false" name="default">
</tmctxlst>
</textbox>
Share
Improve this question
edited Jun 20, 2020 at 9:12
CommunityBot
11 silver badge
asked May 18, 2020 at 22:02
BromoxBromox
7072 gold badges13 silver badges35 bronze badges
2
- Sounds like you could do this with a recursive function... see if you can re-implement it using recursion. Also, if you could edit your post to include some example JSON and example desired output would be awesome! – marno11 Commented May 18, 2020 at 22:37
- 1 Added sample JSON and Desired HTML output – Bromox Commented May 18, 2020 at 23:02
2 Answers
Reset to default 5I believe the easiest way to avoid code duplication here is with recursion, and it can basically look like this:
function createObjectComponent(json) {
const ponent = document.createElement("div");
for(const entry of Object.entries(json)) {
if(Array.isArray(value)) {
ponent.appendChild(createArrayComponent(value));
} else if(typeof value === "object") {
ponent.appendChild(createObjectComponent(value));
} else {
ponent.appendChild(createSimpleValueComponent(value));
}
}
return ponent
}
here's a working example, with some minor css, it includes your own example and also an example of an array handling ponent:
const appDiv = document.getElementById('app');
const example1 = {
foo: 1,
bar: {
id: 2,
obj: {
test: 2
}
},
a: ["a", "b", 1, {
a: 1
}]
}
const example2 = {
id: 1,
name: "Text Box",
tmprops: {
cur: 0,
min: 5000,
visible: true,
},
tmctxlst: {
version: "2",
txttmctx: {
alwysshw: false,
name: "default"
}
}
}
appDiv.appendChild(createObjectComponent(example2));
appDiv.appendChild(createObjectComponent(example1));
appDiv.appendChild(createArrayComponent(example1.a));
function createObjectComponent(json) {
const ponent = document.createElement("div");
ponent.className = "obj-ponent";
ponent.innerHTML += "{";
for (const entry of Object.entries(json)) {
ponent.appendChild(getComponentForEntry(entry));
}
ponent.innerHTML += "}";
return ponent
}
function getComponentForEntry([key, value]) {
const entryDiv = document.createElement("div");
const keySpan = document.createElement("span");
keySpan.className = "key-span";
keySpan.innerText = key + ":";
entryDiv.appendChild(keySpan);
if (Array.isArray(value)) {
entryDiv.appendChild(createArrayComponent(value));
} else if (typeof value === "object") {
entryDiv.appendChild(createObjectComponent(value));
} else {
entryDiv.appendChild(createSimpleValueComponent(value));
}
return entryDiv;
}
function createArrayComponent(array) {
const list = document.createElement("ul");
list.className = "array-ponent";
list.innerHTML += "[";
for (let i = 0; i < array.length; i++) {
const item = array[i];
const listItem = document.createElement("li");
listItem.appendChild(getComponentForEntry([i, item]))
list.appendChild(listItem);
}
list.innerHTML += "]";
return list;
}
function createSimpleValueComponent(value) {
const span = document.createElement("span");
span.innerText = value;
return span;
}
#app {
font-family: monospace;
}
div.obj-ponent {
margin: 8px;
background-color: #e4e4e4;
border-radius: 6px;
max-width: 300px;
padding: 4px 16px;
}
ul.array-ponent {
margin: 8px;
background-color: #e4e4e4;
list-style-type: none;
padding: 4px 16px;
border-radius: 6px;
max-width: 300px;
}
span.key-span {
padding-left: 16px;
margin-right: 8px;
color: blueviolet
}
span {
color: green
}
<div id="app"></div>
You'll need a recursion to solve this,
Basically we take current object & it's key as string, then we traverse the object in dfs
fashion & make applicable children, we're ending by returning the tag to the outer element.
window.onload = function () {
console.log(solve(item.textbox, "textbox"));
}
const item = {
textbox: {
id: 1,
name: "Text Box",
tmprops: {
cur: 0,
min: 5000,
visible: true,
},
tmctxlst: {
version: "2",
txttmctx: {
alwysshw: false,
name: "default"
}
},
}
}
function solve(obj, tagName) {
const tag = document.createElement(tagName);
const currentKeys = Object.keys(obj)
currentKeys.forEach((attribute => {
if (typeof obj[attribute] === "object") {
tag.appendChild(solve(obj[attribute], attribute))
} else {
tag.setAttribute(attribute, obj[attribute]);
}
}))
return tag;
}
Output:
<textbox id="1" name="Text Box">
<tmprops cur="0" min="5000" visible="true"></tmprops>
<tmctxlst version="2">
<txttmctx alwysshw="false" name="default"></txttmctx>
</tmctxlst>
</textbox>
本文标签: Convert JSON to HTML in JavascriptStack Overflow
版权声明:本文标题:Convert JSON to HTML in Javascript - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1744077482a2587014.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论