feat: add expandable functionality to JSON tree nodes for better data visualization

This commit is contained in:
2026-01-15 16:17:09 +08:00
parent d8a7b3c578
commit 65c0ae8ada
+28 -14
View File
@@ -1,4 +1,4 @@
import { useMemo } from "react" import { useMemo, useState } from "react"
import jp, { PathComponent } from "jsonpath" import jp, { PathComponent } from "jsonpath"
import _ from "lodash" import _ from "lodash"
@@ -6,11 +6,13 @@ export interface JsonTreeNodeProps {
data: unknown data: unknown
path: PathComponent[] path: PathComponent[]
matchedPaths: string[] matchedPaths: string[]
defaultExpanded?: boolean
} }
export default function JsonTreeNode({ data, path, matchedPaths }: JsonTreeNodeProps) { export default function JsonTreeNode({ data, path, matchedPaths, defaultExpanded = true }: JsonTreeNodeProps) {
const currentPathString = useMemo(() => jp.stringify(path), [path]) const currentPathString = useMemo(() => jp.stringify(path), [path])
const isMatched = matchedPaths.includes(currentPathString) const isMatched = matchedPaths.includes(currentPathString)
const [expanded, setExpanded] = useState(defaultExpanded)
const highlightClass = isMatched ? "bg-yellow-200 ring-2 ring-yellow-400 rounded-sm" : "" const highlightClass = isMatched ? "bg-yellow-200 ring-2 ring-yellow-400 rounded-sm" : ""
@@ -20,19 +22,31 @@ export default function JsonTreeNode({ data, path, matchedPaths }: JsonTreeNodeP
return ( return (
<div className="ml-6 border-l border-slate-200 pl-3 transition-all"> <div className="ml-6 border-l border-slate-200 pl-3 transition-all">
<span className="text-slate-500">{isArray ? "[" : "{"}</span> <span
{entries.map(([key, value], index) => { className="text-slate-500 cursor-pointer select-none hover:text-slate-700"
const nextPath = [...path, isArray ? Number(key) : key] onClick={() => setExpanded(!expanded)}
>
<span className="inline-block w-4 text-xs">{expanded ? "▼" : "▶"}</span>
{isArray ? "[" : "{"}
{!expanded && <span className="text-slate-400">{entries.length} items</span>}
{!expanded && <span>{isArray ? "]" : "}"}</span>}
</span>
{expanded && (
<>
{entries.map(([key, value], index) => {
const nextPath = [...path, isArray ? Number(key) : key]
return ( return (
<div key={key} className="my-1"> <div key={key} className="my-1">
{!isArray && <span className="text-indigo-600 font-medium">"{key}": </span>} {!isArray && <span className="text-indigo-600 font-medium">"{key}": </span>}
<JsonTreeNode data={value} path={nextPath} matchedPaths={matchedPaths} /> <JsonTreeNode data={value} path={nextPath} matchedPaths={matchedPaths} />
{index < entries.length - 1 && <span className="text-slate-400">,</span>} {index < entries.length - 1 && <span className="text-slate-400">,</span>}
</div> </div>
) )
})} })}
<span className="text-slate-500">{isArray ? "]" : "}"}</span> <span className="text-slate-500">{isArray ? "]" : "}"}</span>
</>
)}
</div> </div>
) )
} }