mirror of
https://github.com/zrwusa/data-structure-typed.git
synced 2024-11-14 01:24:03 +00:00
323 lines
9.3 KiB
TypeScript
323 lines
9.3 KiB
TypeScript
import * as _ from './is';
|
|
import { Json2htmlOptions } from '../types';
|
|
|
|
function toggleJS(options?: Json2htmlOptions): string {
|
|
if (options?.plainHtml) {
|
|
return '';
|
|
} else {
|
|
return 'onclick="json-to-html.toggleVisibility(this);return false"';
|
|
}
|
|
}
|
|
|
|
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') {
|
|
if (datatype === 'array') {
|
|
return `<div class='collapsible level${level}' ${toggleJS(
|
|
options
|
|
)}><span class='json-to-html-label'>${keyName}</span></div>`;
|
|
} else if (datatype === 'object') {
|
|
return `<div class='attribute collapsible level${level}' ${toggleJS(
|
|
options
|
|
)}><span class='json-to-html-label'>${keyName}:</span></div>`;
|
|
} else {
|
|
return `<div class='leaf level${level}'><span class='json-to-html-label'>${keyName}:</span></div>`;
|
|
}
|
|
} else {
|
|
return '';
|
|
}
|
|
}
|
|
|
|
function getContentClass(keyName: string | number): string {
|
|
if (typeof keyName === 'string') {
|
|
return 'content';
|
|
} else {
|
|
return '';
|
|
}
|
|
}
|
|
|
|
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) ||
|
|
isNaN(val) ||
|
|
_.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 {
|
|
let nonCompliant = arr.find(row => !isLeafObject(row));
|
|
if (nonCompliant) {
|
|
return false;
|
|
} else {
|
|
const cols = Object.keys(arr[0]);
|
|
nonCompliant = arr.find((row: object) => !_.isEqual(cols, Object.keys(row)));
|
|
if (nonCompliant) {
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function drawTable(arr: any[]): string {
|
|
function drawRow(headers: string[], rowObj: any): string {
|
|
return '<td>' + headers.map(header => rowObj[header]).join('</td><td>') + '</td>';
|
|
}
|
|
|
|
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>';
|
|
return '<table style="display: table; width:100%; table-layout: fixed;">' + headingHtml + contentHtml + '</table>';
|
|
}
|
|
|
|
function _render(name: string, data: any, options: Json2htmlOptions, level: number, altRow: number): string {
|
|
const contentClass = getContentClass(name);
|
|
if (_.isArray(data)) {
|
|
const title = makeLabelDiv(options, level, `${name}`, 'array');
|
|
let subs: string;
|
|
if (isTable(data)) {
|
|
subs = drawTable(data);
|
|
} else {
|
|
subs =
|
|
"<div class='altRows'>" +
|
|
data
|
|
.map((val: any, idx: number) => _render(idx.toString(), val, options, level + 1, idx % 2))
|
|
.join("</div><div class='altRows'>") +
|
|
'</div>';
|
|
}
|
|
return `<div class="json-to-html-collapse clearfix ${altRow}">
|
|
${title}
|
|
<div class="${contentClass}">${subs}</div>
|
|
</div>`;
|
|
} else if (isLeafValue(data)) {
|
|
const title = makeLabelDiv(options, level, name);
|
|
if (_.isFunction(data)) {
|
|
return `${title}<span class='json-to-html-value'> -function() can't _render-</span>`;
|
|
} else if (!isPlainObject(data)) {
|
|
if (_.isFunction(data.toString)) {
|
|
return `${title}<span class='json-to-html-value'> ${data.toString()}</span>`;
|
|
} else {
|
|
return `${title}<span class='json-to-html-value'> -instance object, can't render-</span>`;
|
|
}
|
|
} else {
|
|
return `${title}<span class='json-to-html-value'> ${data}</span>`;
|
|
}
|
|
} else {
|
|
const title = makeLabelDiv(options, level, name, 'object');
|
|
let count = 0;
|
|
const subs =
|
|
'<div>' +
|
|
Object.entries(data)
|
|
.map(([key, val]) => _render(key, val, options, level + 1, count++ % 2))
|
|
.join('</div><div>') +
|
|
'</div>';
|
|
const inner = `<div class="json-to-html-expand clearfix ${altRow}">
|
|
${title}
|
|
<div class="${contentClass}">${subs}</div>
|
|
</div>`;
|
|
return `${level === 0 ? "<div id='json-to-html'>" : ''}
|
|
${inner}
|
|
${level === 0 ? '</div>' : ''}`;
|
|
}
|
|
}
|
|
|
|
export function render(name: string, json: any, options: Json2htmlOptions): string {
|
|
// return `${head}${_render('', json, options, 0, 0)}`;
|
|
return `${_render(name, json, options, 0, 0)}`;
|
|
}
|
|
|
|
// const head = `<style>
|
|
// #json-to-html table {
|
|
// border-collapse: collapse;
|
|
// }
|
|
// #json-to-html th {
|
|
// color: #888;
|
|
// }
|
|
// #json-to-html table,th, td {
|
|
// border: 1px solid #DDD;
|
|
// padding: 10px 5px;
|
|
// }
|
|
// #json-to-html th, td {
|
|
// text-align: center;
|
|
// }
|
|
// #json-to-html .content {
|
|
// padding-left: 30px;
|
|
// font-family: Arial;
|
|
// }
|
|
//
|
|
// #json-to-html .index {
|
|
// font-size: 100%;
|
|
// color: #999;
|
|
// float: left;
|
|
// }
|
|
// #json-to-html .clearfix:after {
|
|
// content: ".";
|
|
// display: block;
|
|
// height: 0;
|
|
// clear: both;
|
|
// visibility: hidden;
|
|
// }
|
|
// #json-to-html .json-to-html-label {
|
|
// font-family: Helvetica Neue;
|
|
// color: #333;
|
|
// }
|
|
// #json-to-html .json-to-html-value {
|
|
// font-family: Arial;
|
|
// color: #777;
|
|
// }
|
|
// #json-to-html .collapsible > .json-to-html-label:hover {
|
|
// text-decoration: underline;
|
|
// }
|
|
// #json-to-html .collapsible > .json-to-html-label {
|
|
// color: #15C;
|
|
// }
|
|
// #json-to-html .json-to-html-collapse > div.content {
|
|
// display: none;
|
|
// }
|
|
// #json-to-html .json-to-html-collapse > .json-to-html-label {
|
|
// font-weight: bold;
|
|
// }
|
|
//
|
|
// #json-to-html .json-to-html-expand > div > .json-to-html-label, #json-to-html .json-to-html-collapse > div > .json-to-html-label {
|
|
// background-repeat: no-repeat;
|
|
// background-position: left;
|
|
// padding-left: 25px;
|
|
// margin: 5px 0px 5px 15px;
|
|
// display: inline-block;
|
|
// }
|
|
//
|
|
// #json-to-html .json-to-html-expand > div > .json-to-html-label {
|
|
// width: 30px;
|
|
// height: 30px;
|
|
// background-image: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="100" height="100"><polygon points="50,10 10,90 90,90" fill="blue" /></svg>');
|
|
// background-size: cover;
|
|
// background-position: center;
|
|
// }
|
|
//
|
|
// #json-to-html .json-to-html-collapse > div > .json-to-html-label {
|
|
// width: 30px;
|
|
// height: 30px;
|
|
// background-image: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" width="100" height="100"><polygon points="50,10 10,90 90,90" fill="blue" /></svg>');
|
|
// background-size: cover;
|
|
// background-position: center;
|
|
// }
|
|
//
|
|
// #json-to-html .json-to-html-collapse > span.collapsible:before {
|
|
// border-radius: 2px;
|
|
// border-color: #A44;
|
|
// border-style: solid;
|
|
// border-width: 1px;
|
|
// color: #A44;
|
|
// content: '+';
|
|
// display: inline-block;
|
|
// line-height: 7px;
|
|
// margin: 0 2px;
|
|
// overflow: hidden;
|
|
// padding: 1px;
|
|
// font-size: 11px;
|
|
// }
|
|
//
|
|
// #json-to-html .json-to-html-expand > span.collapsible:before {
|
|
// border: none;
|
|
// color: #A44;
|
|
// content: '-';
|
|
// display: inline-block;
|
|
// line-height: 7px;
|
|
// margin: 4px;
|
|
// overflow: hidden;
|
|
// padding: 1px;
|
|
// font-size: 11px;
|
|
// }
|
|
//
|
|
// #json-to-html.level0 {
|
|
// font-size: 25px;
|
|
// }
|
|
// #json-to-html .level1 {
|
|
// font-size: 22px;
|
|
// }
|
|
//
|
|
// #json-to-html .leaf {
|
|
// color: #666;
|
|
// display: inline;
|
|
// }
|
|
//
|
|
// #json-to-html .altRows:nth-child(odd) { background-color:#ddd; }
|
|
// #json-to-html .altRows:nth-child(even) { background-color:#fff; }
|
|
//
|
|
// #json-to-html tr:nth-child(odd) { background-color:#eee; }
|
|
// #json-to-html tr:nth-child(even) { background-color:#fff; }
|
|
// </style>
|
|
// <script type="text/javascript">
|
|
// json-to-html = {
|
|
// toggleVisibility: function(el, name) {
|
|
// json-to-html.toggleClass(el.parentElement,'json-to-html-collapse json-to-html-expand');
|
|
// },
|
|
// classRe: function(name) {
|
|
// return new RegExp('(?:^|\\s)'+name+'(?!\\S)');
|
|
// },
|
|
// addClass: function(el, name) {
|
|
// el.className += " "+name;
|
|
// },
|
|
// removeClass: function(el, name) {
|
|
// var re = json-to-html.classRe(name);
|
|
// el.className = el.className.replace(json-to-html.classRe(name) , '' )
|
|
// },
|
|
// hasClass: function(el, name) {
|
|
// var re = json-to-html.classRe(name);
|
|
// return json-to-html.classRe(name).exec(el.className);
|
|
// },
|
|
// toggleClass: function(el, name) {
|
|
// var names = name.split(/\s+/);
|
|
// for (n in names) {
|
|
// if (json-to-html.hasClass(el, names[n])) {
|
|
// json-to-html.removeClass(el, names[n]);
|
|
// } else {
|
|
// json-to-html.addClass(el, names[n]);
|
|
// }
|
|
// }
|
|
// }
|
|
// };
|
|
// </script>`;
|