import React from "react"
import {
Box,
Flex,
Input,
NumberDecrementStepper,
NumberIncrementStepper,
NumberInput,
NumberInputField,
NumberInputStepper,
Select,
Switch,
} from "@chakra-ui/react"
import { Icon } from "../Icon/Icon"
import { Safe } from "../Safety/Safe"
import { useThingtime } from "./useThingtime"
export const Thingtime = (props) => {
// TODO: Add a circular reference seen prop check
// and add button to expand circular reference
// up to 1 level deep
const { thingtime, setThingtime } = useThingtime()
const [uuid, setUuid] = React.useState()
const [circular, setCircular] = React.useState(props?.circular)
const depth = React.useMemo(() => {
return props?.depth || 1
}, [props?.depth])
const pl = React.useMemo(() => {
return props?.pl || [4, 6]
}, [props?.pl])
const pr = React.useMemo(() => {
return props?.pr || (depth === 1 ? [4, 6] : 0)
}, [props?.pr, depth])
// will only run on the client
React.useEffect(() => {
setUuid(Math.random().toString(36).substring(7))
}, [])
const thing = React.useMemo(() => {
return props.thing
}, [props.thing])
const seen = React.useMemo(() => {
if (props?.seen instanceof Array) {
if (props?.seen?.includes(thing)) {
return props?.seen
} else if (typeof thing === "object") {
return [...props.seen, thing]
}
return props?.seen || []
}
if (typeof thing === "object") {
return [thing]
}
return []
}, [props?.seen, thing])
const mode = React.useMemo(() => {
return "view"
}, [])
const validKeyTypes = React.useMemo(() => {
return ["object", "array"]
}, [])
const keys = React.useMemo(() => {
if (validKeyTypes?.includes(typeof thing)) {
try {
const keysRet = Object.keys(thing)
return keysRet
} catch {
// nothing
}
} else {
return []
}
}, [thing, validKeyTypes])
const type = React.useMemo(() => {
return typeof thing
}, [thing])
const typeIcon = React.useMemo(() => {
const size = 7
if (thing instanceof Array) {
return
} else if (type === "object") {
return
} else if (type === "string") {
return
} else if (type === "number") {
return
} else if (type === "boolean") {
return
} else {
return
}
}, [type, thing])
const valuePl = React.useMemo(() => {
if (typeof props?.valuePl === "number") {
return props?.valuePl
}
return props?.path ? [4, 6] : [0, 0]
}, [props?.valuePl, props?.path])
const renderableValue = React.useMemo(() => {
if (type === "string") {
const trimmed = thing.trim()
if (!trimmed) {
return ""
}
return trimmed
} else if (type === "number") {
return thing
} else if (type === "boolean") {
return thing ? "true" : "false"
} else if (type === "object") {
if (thing === null) {
return "null"
}
if (!keys?.length) {
return "Something!"
}
try {
return JSON.stringify(thing, null, 2)
} catch (err) {
// console.error(
// "Caught error making renderableValue of thing",
// err,
// thing
// )
return (
setCircular(false)}>
Click to Expand
)
}
} else {
return "Something!"
}
}, [thing, type, keys])
const keysToUse = React.useMemo(() => {
return keys
}, [keys])
// const keysToUse = flattenedKeys
const template1Modes = React.useMemo(() => {
return ["view", "edit"]
}, [])
const thingtimeChildren = React.useMemo(() => {
if (template1Modes?.includes(mode)) {
if (keys?.length && !circular) {
return (
{keysToUse?.length &&
keysToUse.map((key, idx) => {
if (!key?.human) {
key = {
human: key,
key: key,
}
}
const nextThing = thing[key?.key]
const nextSeen = [...seen]
if (typeof nextThing === "object") {
nextSeen.push(nextThing)
}
const fullPath = props?.fullPath || props?.path
return (
)
})}
)
}
}
}, [
keysToUse,
mode,
circular,
seen,
depth,
thing,
props,
valuePl,
pl,
keys,
template1Modes,
])
const AtomicWrapper = React.useCallback(
(props) => {
return (
{props?.children}
)
},
[pl]
)
const updateValue = React.useCallback(
(args) => {
const { value } = args
console.log("nik value", value)
console.log("nik props?.fullPath", props?.fullPath)
setThingtime(props?.fullPath, value)
},
[props?.fullPath, setThingtime]
)
const atomicValue = React.useMemo(() => {
if (props?.edit) {
if (type === "boolean") {
return (
{
e?.preventDefault?.()
e?.stopPropagation?.()
// cancel bubble
e?.nativeEvent?.stopImmediatePropagation?.()
console.log("nik 123123 clicked", !thing)
setTimeout(() => {
updateValue({ value: !thing })
console.log("nik 123123 changed", e)
}, 1)
}}
>
)
}
if (type === "number") {
const numberPxLength = thing?.toString()?.length * 13 + 30
return (
{
setTimeout(() => {
try {
const number = Number(value)
console.log("typeof number", typeof number)
updateValue({ value: number })
} catch {
// something went wrong casting to number
}
}, 1)
}}
value={thing}
>
)
}
}
return {renderableValue}
}, [renderableValue, AtomicWrapper, type, props?.edit, thing, updateValue])
const contextMenu = (
Settings
)
const [showContextMenu, setShowContextMenu] = React.useState(false)
const humanPath = React.useMemo(() => {
if (typeof props?.path === "string") {
return props?.path
}
return props?.path?.human || ""
}, [props?.path])
const renderedPath = React.useMemo(() => {
if (props?.edit) {
return humanPath
}
if (humanPath?.includes?.("hidden")) {
return null
}
if (humanPath?.includes?.("unique")) {
// take only path from before the string unique
return humanPath.split?.("unique")?.[0]
}
return humanPath
}, [humanPath, props?.edit])
const pathDom = React.useMemo(() => {
if (renderedPath) {
return (
{renderedPath}
)
}
}, [renderedPath, pl, props?.pathPl])
const handleMouseEvent = React.useCallback(
(e) => {
const target = e?.target
// extract uuid from className
const className = target?.className
if (className?.includes(uuid?.current)) {
setShowContextMenu(e?.type === "mouseenter")
}
},
[uuid]
)
const [showContextIcon, setShowContextIcon] = React.useState(false)
return (
{/* {uuid?.current} */}
setShowContextIcon(true)}
onMouseLeave={() => setShowContextIcon(false)}
>
{pathDom}
{props?.edit && (
{typeIcon}
)}
{pathDom && (
)}
{/* {showContextMenu && contextMenu} */}
{!thingtimeChildren && atomicValue}
{thingtimeChildren}
)
}