279 lines
6.7 KiB
TypeScript
Raw Normal View History

import React from "react"
import { Box, Flex } from "@chakra-ui/react"
2023-06-28 08:25:17 +00:00
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 [uuid, setUuid] = React.useState()
2023-06-30 01:46:42 +00:00
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])
// will only run on the client
2023-06-30 01:46:42 +00:00
React.useEffect(() => {
setUuid(Math.random().toString(36).substring(7))
}, [])
const { thingtime } = useThingtime()
2023-06-28 08:25:17 +00:00
const thing = React.useMemo(() => {
return props.thing
}, [props.thing])
2023-06-29 11:06:58 +00:00
const mode = React.useMemo(() => {
return "view"
2023-06-29 11:06:58 +00:00
}, [])
const validKeyTypes = React.useMemo(() => {
return ["object", "array"]
2023-06-29 11:06:58 +00:00
}, [])
const keys = React.useMemo(() => {
if (validKeyTypes?.includes(typeof thing)) {
const keysRet = Object.keys(thing)
return keysRet
2023-06-29 11:06:58 +00:00
} else {
return []
}
}, [thing, validKeyTypes])
const type = React.useMemo(() => {
return typeof thing
}, [thing])
const valuePl = React.useMemo(() => {
if (typeof props?.valuePl === "number") {
return props?.valuePl
}
return props?.path ? [4, 6] : [0, 0]
}, [props?.valuePl, props?.path])
2023-06-29 11:06:58 +00:00
const renderableValue = React.useMemo(() => {
if (type === "string") {
if (!thing) {
return "Empty string"
}
2023-06-29 11:06:58 +00:00
return thing
} else if (type === "number") {
2023-06-29 11:06:58 +00:00
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 "Circular reference in object."
}
2023-06-29 11:06:58 +00:00
} else {
return "Something!"
2023-06-29 11:06:58 +00:00
}
}, [thing, type])
const flattenedKeys = React.useMemo(() => {
// create an array of all keys on object so that if object is
// { my: { child: {} } }
// the array looks like
// ['my', 'my.child']
const ret = []
try {
const randId = Math.random().toString(36).substring(7)
window.thingtime.tmp[randId] = 0
const recurse = (obj, prefix) => {
Object.keys(obj).forEach((key) => {
if (typeof obj[key] === "object") {
2023-06-29 11:06:58 +00:00
if (window?.thingtime?.tmp[randId] < 1000) {
window.thingtime.tmp[randId]++
recurse(obj[key], `${prefix}${prefix && "."}${key}`)
2023-06-29 11:06:58 +00:00
} else {
console.error("Recursion limit reached in Thingtime.tsx")
2023-06-29 11:06:58 +00:00
}
} else {
ret.push({
key: `${prefix}${prefix && "."}${key}`,
human: `${prefix}${prefix && " "}${key}`,
2023-06-29 11:06:58 +00:00
})
}
})
}
recurse(thing, "")
2023-06-29 11:06:58 +00:00
} catch (err) {
// console.error('Error in Thingtime.tsx creating flattenedKeys', err)
}
return ret
}, [thing])
let value = null
let editableValue = null
2023-06-29 11:06:58 +00:00
const keysToUse = keys
// const keysToUse = flattenedKeys
const template1Modes = ["view", "edit"]
2023-06-29 11:06:58 +00:00
if (template1Modes?.includes(mode)) {
if (keys?.length) {
value = (
<Safe {...props}>
<Flex
position="relative"
flexDirection="column"
// w={'500px'}
// w={['200px', '500px']}
maxWidth="100%"
paddingLeft={valuePl}
paddingY={props?.path ? 3 : 0}
>
{keysToUse?.length &&
keysToUse.map((key, idx) => {
if (!key?.human) {
key = {
human: key,
key: key,
}
2023-06-29 11:06:58 +00:00
}
const nextThing = thing[key?.key]
return (
<Thingtime
key={idx}
depth={depth + 1}
parent={thing}
path={key}
thing={nextThing}
// thing={{ infinite: { yes: true } }}
valuePl={pl}
></Thingtime>
)
})}
</Flex>
</Safe>
2023-06-29 11:06:58 +00:00
)
} else {
editableValue = (
<Box
paddingLeft={pl}
fontSize="20px"
border="none"
whiteSpace="pre-line"
outline="none"
contentEditable={mode === "edit"}
paddingY={2}
// dangerouslySetInnerHTML={{ __html: renderableValue }}
>
{renderableValue}
</Box>
2023-06-29 11:06:58 +00:00
)
}
}
const contextMenu = (
<Flex
position="absolute"
top={0}
right={0}
paddingRight={4}
userSelect="none"
>
2023-06-29 11:06:58 +00:00
Settings
</Flex>
)
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 path = React.useMemo(() => {
if (humanPath?.includes?.("hidden")) {
return null
}
if (humanPath?.includes?.("unique")) {
// take only path from before the string unique
return humanPath.split?.("unique")?.[0]
}
return (
<Flex
maxWidth="100%"
paddingLeft={props?.pathPl || pl}
fontSize="12px"
wordBreak="break-all"
>
{humanPath}
</Flex>
)
}, [humanPath, 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]
)
return (
<Safe {...props} depth={depth} uuid={uuid?.current}>
<Flex
position="relative"
flexDirection="column"
width="500px"
maxWidth="100%"
paddingRight={pr}
onMouseEnter={handleMouseEvent}
onMouseLeave={handleMouseEvent}
// minW={depth === 1 ? '120px' : null}
paddingY={3}
{...props}
className={`thing-${uuid?.current}`}
>
{/* {uuid?.current} */}
{path}
{/* {showContextMenu && contextMenu} */}
{editableValue}
{value}
</Flex>
</Safe>
2023-06-29 11:06:58 +00:00
)
}