2023-11-02 03:57:20 +00:00
|
|
|
import * as _ from './is';
|
2023-11-16 02:14:14 +00:00
|
|
|
import { Json2htmlOptions } from '../types';
|
2023-11-02 03:38:29 +00:00
|
|
|
|
2023-11-02 08:10:41 +00:00
|
|
|
function toggleJS(options?: Json2htmlOptions): string {
|
2023-11-02 03:38:29 +00:00
|
|
|
if (options?.plainHtml) {
|
2023-11-02 03:57:20 +00:00
|
|
|
return '';
|
2023-11-02 03:38:29 +00:00
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
return 'onclick="json-to-html.toggleVisibility(this);return false"';
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-02 08:10:41 +00:00
|
|
|
function makeLabelDiv(options: any, level: number, keyName: string | number, datatype?: string): string {
|
|
|
|
if (typeof keyName === 'number') {
|
|
|
|
return `<div class='index'><span class='json-to-html-label'>${keyName} </span></div>`;
|
|
|
|
} else if (typeof keyName === 'string') {
|
2023-11-02 03:57:20 +00:00
|
|
|
if (datatype === 'array') {
|
2024-08-26 22:56:04 +00:00
|
|
|
return `<div class='collapsible level${level}' ${toggleJS(options)}><span class='json-to-html-label'>${keyName}</span></div>`;
|
2023-11-02 03:57:20 +00:00
|
|
|
} else if (datatype === 'object') {
|
2024-08-26 22:56:04 +00:00
|
|
|
return `<div class='attribute collapsible level${level}' ${toggleJS(options)}><span class='json-to-html-label'>${keyName}:</span></div>`;
|
2023-11-02 03:38:29 +00:00
|
|
|
} else {
|
2023-11-02 08:10:41 +00:00
|
|
|
return `<div class='leaf level${level}'><span class='json-to-html-label'>${keyName}:</span></div>`;
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
return '';
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-02 08:10:41 +00:00
|
|
|
function getContentClass(keyName: string | number): string {
|
|
|
|
if (typeof keyName === 'string') {
|
2023-11-02 03:57:20 +00:00
|
|
|
return 'content';
|
2023-11-02 03:38:29 +00:00
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
return '';
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function isPlainObject(val: any): boolean {
|
|
|
|
let lastKey: string | undefined;
|
|
|
|
let lastOwnKey: string | undefined;
|
|
|
|
for (const key in val) {
|
|
|
|
if (val.hasOwnProperty(key)) {
|
|
|
|
lastOwnKey = key;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (const key in val) {
|
|
|
|
lastKey = key;
|
|
|
|
}
|
|
|
|
return lastOwnKey === lastKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isLeafValue(val: any): boolean {
|
|
|
|
return (
|
|
|
|
_.isNumber(val) ||
|
|
|
|
_.isString(val) ||
|
|
|
|
_.isBoolean(val) ||
|
|
|
|
_.isDate(val) ||
|
|
|
|
_.isNull(val) ||
|
|
|
|
_.isUndefined(val) ||
|
2023-11-02 03:57:20 +00:00
|
|
|
isNaN(val) ||
|
2023-11-02 03:38:29 +00:00
|
|
|
_.isFunction(val) ||
|
|
|
|
!isPlainObject(val)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isLeafObject(obj: any): boolean {
|
|
|
|
if (!_.isObject(obj)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (const key in obj) {
|
|
|
|
const val = obj[key];
|
|
|
|
if (!isLeafValue(val)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isTable(arr: any[]): boolean {
|
|
|
|
if (!_.isArray(arr)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (arr.length === 0 || !_.isObject(arr[0])) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
let nonCompliant = arr.find(row => !isLeafObject(row));
|
2023-11-02 03:38:29 +00:00
|
|
|
if (nonCompliant) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
const cols = Object.keys(arr[0]);
|
|
|
|
nonCompliant = arr.find((row: object) => !_.isEqual(cols, Object.keys(row)));
|
2023-11-02 03:38:29 +00:00
|
|
|
if (nonCompliant) {
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function drawTable(arr: any[]): string {
|
|
|
|
function drawRow(headers: string[], rowObj: any): string {
|
2023-11-02 03:57:20 +00:00
|
|
|
return '<td>' + headers.map(header => rowObj[header]).join('</td><td>') + '</td>';
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
2023-11-02 13:38:10 +00:00
|
|
|
|
2023-11-02 03:57:20 +00:00
|
|
|
const cols = Object.keys(arr[0]);
|
|
|
|
const content = arr.map(rowObj => drawRow(cols, rowObj));
|
|
|
|
const headingHtml = '<tr><th>' + cols.join('</th><th>') + '</th></tr>';
|
|
|
|
const contentHtml = '<tr>' + content.join('</tr><tr>') + '</tr>';
|
2023-11-06 03:01:32 +00:00
|
|
|
return '<table style="display: table; width:100%; table-layout: fixed;">' + headingHtml + contentHtml + '</table>';
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
|
|
|
|
2023-11-02 08:10:41 +00:00
|
|
|
function _render(name: string, data: any, options: Json2htmlOptions, level: number, altRow: number): string {
|
2023-11-02 03:38:29 +00:00
|
|
|
const contentClass = getContentClass(name);
|
|
|
|
if (_.isArray(data)) {
|
2023-11-02 08:10:41 +00:00
|
|
|
const title = makeLabelDiv(options, level, `${name}`, 'array');
|
2023-11-02 03:38:29 +00:00
|
|
|
let subs: string;
|
|
|
|
if (isTable(data)) {
|
|
|
|
subs = drawTable(data);
|
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
subs =
|
|
|
|
"<div class='altRows'>" +
|
2023-11-16 02:14:14 +00:00
|
|
|
data
|
|
|
|
.map((val: any, idx: number) => _render(idx.toString(), val, options, level + 1, idx % 2))
|
|
|
|
.join("</div><div class='altRows'>") +
|
2023-11-02 03:57:20 +00:00
|
|
|
'</div>';
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
2023-11-02 08:10:41 +00:00
|
|
|
return `<div class="json-to-html-collapse clearfix ${altRow}">
|
2023-11-02 03:38:29 +00:00
|
|
|
${title}
|
|
|
|
<div class="${contentClass}">${subs}</div>
|
|
|
|
</div>`;
|
|
|
|
} else if (isLeafValue(data)) {
|
|
|
|
const title = makeLabelDiv(options, level, name);
|
|
|
|
if (_.isFunction(data)) {
|
2023-11-02 03:57:20 +00:00
|
|
|
return `${title}<span class='json-to-html-value'> -function() can't _render-</span>`;
|
2023-11-02 03:38:29 +00:00
|
|
|
} else if (!isPlainObject(data)) {
|
|
|
|
if (_.isFunction(data.toString)) {
|
2023-11-02 03:57:20 +00:00
|
|
|
return `${title}<span class='json-to-html-value'> ${data.toString()}</span>`;
|
2023-11-02 03:38:29 +00:00
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
return `${title}<span class='json-to-html-value'> -instance object, can't render-</span>`;
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
return `${title}<span class='json-to-html-value'> ${data}</span>`;
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|
|
|
|
} else {
|
2023-11-02 03:57:20 +00:00
|
|
|
const title = makeLabelDiv(options, level, name, 'object');
|
2023-11-02 03:38:29 +00:00
|
|
|
let count = 0;
|
2023-11-02 03:57:20 +00:00
|
|
|
const subs =
|
|
|
|
'<div>' +
|
|
|
|
Object.entries(data)
|
|
|
|
.map(([key, val]) => _render(key, val, options, level + 1, count++ % 2))
|
|
|
|
.join('</div><div>') +
|
|
|
|
'</div>';
|
2023-11-02 08:10:41 +00:00
|
|
|
const inner = `<div class="json-to-html-expand clearfix ${altRow}">
|
2023-11-02 03:38:29 +00:00
|
|
|
${title}
|
|
|
|
<div class="${contentClass}">${subs}</div>
|
|
|
|
</div>`;
|
2023-11-02 03:57:20 +00:00
|
|
|
return `${level === 0 ? "<div id='json-to-html'>" : ''}
|
2023-11-02 03:38:29 +00:00
|
|
|
${inner}
|
|
|
|
${level === 0 ? '</div>' : ''}`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-02 08:10:41 +00:00
|
|
|
export function render(name: string, json: any, options: Json2htmlOptions): string {
|
|
|
|
return `${_render(name, json, options, 0, 0)}`;
|
2023-11-02 03:38:29 +00:00
|
|
|
}
|