2023-08-13 05:53:47 +00:00
|
|
|
import React from "react"
|
|
|
|
import { Box } from "@chakra-ui/react"
|
|
|
|
|
|
|
|
import { useThingtime } from "../Thingtime/useThingtime"
|
|
|
|
|
2023-08-17 14:21:38 +00:00
|
|
|
export const MagicInput = React.forwardRef((props, ref) => {
|
2023-08-13 05:53:47 +00:00
|
|
|
const { thingtime, setThingtime, loading } = useThingtime()
|
|
|
|
|
|
|
|
const [inputValue, setInputValue] = React.useState()
|
|
|
|
|
2023-08-14 05:10:46 +00:00
|
|
|
const [isClientSide, setIsClientSide] = React.useState(false)
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
setIsClientSide(true)
|
|
|
|
}, [])
|
|
|
|
|
2023-08-13 05:53:47 +00:00
|
|
|
const contentEditableRef = React.useRef(null)
|
|
|
|
const editValueRef = React.useRef({})
|
|
|
|
|
|
|
|
const fullPath = React.useMemo(() => {
|
|
|
|
const ret = props?.fullPath || props?.path
|
|
|
|
|
|
|
|
// store this thing in the global db
|
|
|
|
try {
|
|
|
|
// window.meta.things[ret] = props?.value
|
|
|
|
} catch {
|
|
|
|
// nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret
|
|
|
|
}, [props?.fullPath, props?.path, props?.value])
|
|
|
|
|
|
|
|
const [contentEditableValue, setContentEditableValue] = React.useState(
|
|
|
|
props?.value || props?.placeholder
|
|
|
|
)
|
|
|
|
|
|
|
|
const updateContentEditableValue = React.useCallback((value) => {
|
|
|
|
// replace all new line occurences in value with <div><br></div>
|
|
|
|
|
|
|
|
console.log("MagicInput updating contentEditableValue with value", value)
|
|
|
|
|
|
|
|
// extract all series of new lines
|
|
|
|
const newlines = value?.split?.(/[^\n]/)?.filter((v) => v !== "")
|
|
|
|
|
|
|
|
let newValue = value
|
|
|
|
|
|
|
|
// replace all new lines groups with <div><br></div>
|
|
|
|
newlines?.forEach?.((newline) => {
|
|
|
|
const baseLength = "\n"?.length
|
|
|
|
|
|
|
|
const newlineClone = newline
|
|
|
|
|
|
|
|
const newlineClonePart1 = newlineClone?.replace(
|
|
|
|
"\n\n\n",
|
|
|
|
"<div><br /></div>"
|
|
|
|
)
|
|
|
|
const newlineClonePart2 = newlineClonePart1?.replace(
|
|
|
|
/\n\n/g,
|
|
|
|
"<div><br /></div>"
|
|
|
|
)
|
|
|
|
const newlineClonePart3 = newlineClonePart2?.replace(/\n/g, "<br />")
|
|
|
|
|
|
|
|
newValue = newValue?.replace(newline, newlineClonePart3)
|
|
|
|
})
|
|
|
|
|
|
|
|
setContentEditableValue(newValue)
|
|
|
|
}, [])
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
setInputValue(contentEditableValue)
|
|
|
|
}, [contentEditableValue])
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
const entries = Object.entries(editValueRef.current)
|
|
|
|
const propsValueInEntries = entries?.find?.(
|
|
|
|
(entry) => entry[1] === props?.value
|
|
|
|
)
|
|
|
|
if (!propsValueInEntries) {
|
|
|
|
// const valueToUse = props?.value || props?.placeholder
|
|
|
|
const valueToUse = props?.value
|
|
|
|
updateContentEditableValue(valueToUse)
|
|
|
|
// setContentEditableValue(props?.value)
|
|
|
|
} else {
|
|
|
|
const [time, value] = propsValueInEntries
|
|
|
|
if (time && value) {
|
|
|
|
delete editValueRef.current[time]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, [props?.value, props?.placeholder, updateContentEditableValue])
|
|
|
|
|
|
|
|
const onValueChange = React.useCallback(
|
|
|
|
(args) => {
|
|
|
|
const { value } = args
|
|
|
|
props?.onValueChange?.({ value })
|
2023-08-17 14:21:38 +00:00
|
|
|
props?.onChange?.({ value })
|
2023-08-13 05:53:47 +00:00
|
|
|
// updateContentEditableValue(value)
|
|
|
|
},
|
2023-08-17 14:21:38 +00:00
|
|
|
[props?.onValueChange, props?.onChange]
|
2023-08-13 05:53:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const updateValue = React.useCallback(
|
|
|
|
(args) => {
|
|
|
|
const { value } = args
|
|
|
|
|
|
|
|
onValueChange({ value })
|
|
|
|
setInputValue(value)
|
|
|
|
// setThingtime(fullPath, value)
|
|
|
|
},
|
|
|
|
[fullPath, setThingtime, onValueChange]
|
|
|
|
)
|
|
|
|
|
|
|
|
const onFocus = React.useCallback(
|
|
|
|
(e) => {
|
|
|
|
props?.onFocus?.(e)
|
|
|
|
},
|
|
|
|
[props?.onFocus]
|
|
|
|
)
|
|
|
|
|
2023-08-17 14:21:38 +00:00
|
|
|
const maybeUpdateValue = React.useCallback(
|
|
|
|
(e) => {
|
|
|
|
const { key } = e
|
|
|
|
if (key === "Enter") {
|
|
|
|
if (props?.onEnter) {
|
|
|
|
e?.preventDefault?.()
|
|
|
|
}
|
|
|
|
props?.onEnter?.({ value: inputValue })
|
|
|
|
}
|
|
|
|
},
|
|
|
|
[inputValue, props?.onEnter]
|
|
|
|
)
|
|
|
|
|
2023-08-13 06:14:40 +00:00
|
|
|
const dangerouslySetInnerHTML = React.useMemo(() => {
|
|
|
|
if (!props?.readonly) {
|
|
|
|
return { __html: contentEditableValue }
|
|
|
|
}
|
|
|
|
// return { __html: contentEditableValue }
|
|
|
|
}, [contentEditableValue, props?.readonly])
|
2023-08-17 14:21:38 +00:00
|
|
|
|
2023-08-13 05:53:47 +00:00
|
|
|
return (
|
|
|
|
<Box
|
|
|
|
position="relative"
|
|
|
|
width="100%"
|
|
|
|
transition={props?.transition}
|
|
|
|
{...(props?.chakras || {})}
|
2023-08-17 14:21:38 +00:00
|
|
|
ref={ref}
|
2023-08-13 05:53:47 +00:00
|
|
|
>
|
|
|
|
<Box
|
2023-08-17 14:21:38 +00:00
|
|
|
className="magic-input-focusable"
|
2023-08-13 05:53:47 +00:00
|
|
|
ref={contentEditableRef}
|
|
|
|
width="100%"
|
|
|
|
border="none"
|
|
|
|
outline="none"
|
2023-08-13 06:14:40 +00:00
|
|
|
contentEditable={!props?.readonly ? true : false}
|
|
|
|
dangerouslySetInnerHTML={dangerouslySetInnerHTML}
|
2023-08-13 05:53:47 +00:00
|
|
|
onFocus={onFocus}
|
|
|
|
onInput={(value) => {
|
|
|
|
const innerText = value?.target?.innerText
|
|
|
|
console.log("MagicInput got onInput event value", value)
|
|
|
|
console.log("MagicInput got onInput event innerText", innerText)
|
|
|
|
if (typeof innerText === "string") {
|
|
|
|
const time = Date.now()
|
|
|
|
editValueRef.current[time] = innerText
|
|
|
|
updateValue({ value: innerText })
|
|
|
|
}
|
|
|
|
}}
|
2023-08-17 14:21:38 +00:00
|
|
|
onKeyDown={maybeUpdateValue}
|
2023-08-13 06:14:40 +00:00
|
|
|
>
|
|
|
|
{props?.readonly ? props?.value : null}
|
|
|
|
</Box>
|
2023-08-14 05:15:12 +00:00
|
|
|
<Box
|
|
|
|
display={
|
|
|
|
!props.readonly || inputValue || !isClientSide ? "none" : "block"
|
|
|
|
}
|
|
|
|
opacity={0}
|
|
|
|
>
|
|
|
|
Imagine..
|
|
|
|
</Box>
|
2023-08-13 05:53:47 +00:00
|
|
|
<Box
|
|
|
|
position="absolute"
|
|
|
|
top={0}
|
|
|
|
left={0}
|
2023-08-14 05:10:46 +00:00
|
|
|
display={inputValue || !isClientSide ? "none" : "block"}
|
2023-08-14 00:51:00 +00:00
|
|
|
width="100%"
|
|
|
|
maxWidth="100%"
|
2023-08-13 05:53:47 +00:00
|
|
|
color="greys.dark"
|
|
|
|
pointerEvents="none"
|
|
|
|
>
|
|
|
|
{props?.placeholder || "Imagine.."}
|
|
|
|
</Box>
|
|
|
|
</Box>
|
|
|
|
)
|
2023-08-17 14:21:38 +00:00
|
|
|
})
|