import { useState } from 'react'

type Node = any

function isObject(value: any) {
  return value !== null && typeof value === 'object'
}

function JsonNode({ name, data, depth = 0, defaultExpandDepth = 1 }: { name?: string; data: Node; depth?: number; defaultExpandDepth?: number }) {
  const [open, setOpen] = useState(depth < (defaultExpandDepth ?? 1)) // collapse deeper levels by default

  if (!isObject(data)) {
    return (
      <div style={{ paddingLeft: depth * 12 }}>
        {name !== undefined ? <strong>{name}: </strong> : null}
        <span style={{ color: '#0b6' }}>{String(data)}</span>
      </div>
    )
  }

  const entries: [any, any][] = Array.isArray(data)
    ? (data.map((v: any, i: number) => [i, v]) as [any, any][])
    : (Object.entries(data) as [any, any][])

  return (
    <div style={{ paddingLeft: depth * 12 }}>
      <div style={{ cursor: 'pointer', userSelect: 'none' }} onClick={() => setOpen(!open)}>
        {name !== undefined ? <strong>{name} </strong> : null}
        <span style={{ color: '#888' }}>{Array.isArray(data) ? `[Array(${entries.length})]` : `{Object(${entries.length})}`}</span>
        <span style={{ marginLeft: 8 }}>{open ? '▾' : '▸'}</span>
      </div>
      {open && (
        <div>
          {entries.map(([k, v]: [any, any]) => (
            <JsonNode key={String(k)} name={String(k)} data={v} depth={depth + 1} defaultExpandDepth={defaultExpandDepth} />
          ))}
        </div>
      )}
    </div>
  )
}

export default function JsonTree({ data, defaultExpandDepth = 1 }: { data: any; defaultExpandDepth?: number }) {
  return (
    <div style={{ fontFamily: 'monospace', fontSize: 13 }}>
      <JsonNode data={data} defaultExpandDepth={defaultExpandDepth} />
    </div>
  )
}
