From 2a8eb2297634c4d5b0a7567192f92a8ef4593c9f Mon Sep 17 00:00:00 2001 From: Nikolaj Frey Date: Fri, 7 Jul 2023 17:42:14 +1000 Subject: [PATCH 1/3] feat: feature/rainbow-component Fixed a few things --- remix/app/Providers/Chakra/colors.tsx | 59 ++++----- remix/app/Providers/ThingtimeProvider.tsx | 3 +- remix/app/components/Commander/Commander.tsx | 128 +++++++++++-------- remix/app/components/Nav/Nav.tsx | 54 ++++---- remix/app/components/Rainbow/Rainbow.tsx | 6 +- remix/app/components/Thingtime/Thingtime.tsx | 123 +++++++++--------- remix/app/functions/sanitise.tsx | 35 +++-- 7 files changed, 228 insertions(+), 180 deletions(-) diff --git a/remix/app/Providers/Chakra/colors.tsx b/remix/app/Providers/Chakra/colors.tsx index b613073..54a47da 100644 --- a/remix/app/Providers/Chakra/colors.tsx +++ b/remix/app/Providers/Chakra/colors.tsx @@ -1,45 +1,46 @@ const g = { - grey: '#F1F1F3', + grey: "#F1F1F3", greys: { - light: '#F1F1F3', - medium: '#E0E0E0', - dark: '#BDBDBD' - } + light: "#F1F1F3", + lightt: "#E7E6E8", + medium: "#E0E0E0", + dark: "#BDBDBD", + }, } g.gray = g.grey g.grays = g.greys export const colors = { - white: '#FFFFFF', + white: "#FFFFFF", ...g, - black: '#000000', + black: "#000000", // all colors of the chakras chakras: { - root: '#C62828', - sacral: '#FF7043', - solarPlexus: '#FFEE58', - heart: '#66BB6A', - throat: '#42A5F5', - thirdEye: '#5C6BC0', - crown: '#AB47BC', + root: "#C62828", + sacral: "#FF7043", + solarPlexus: "#FFEE58", + heart: "#66BB6A", + throat: "#42A5F5", + thirdEye: "#5C6BC0", + crown: "#AB47BC", - red: '#C62828', - orange: '#FF7043', - yellow: '#FFEE58', - green: '#66BB6A', - blue: '#42A5F5', - indigo: '#5C6BC0', - violet: '#AB47BC' + red: "#C62828", + orange: "#FF7043", + yellow: "#FFEE58", + green: "#66BB6A", + blue: "#42A5F5", + indigo: "#5C6BC0", + violet: "#AB47BC", }, // all colors of the rainbow rainbow: { - red: '#FF0000', - orange: '#FF7F00', - yellow: '#FFFF00', - green: '#00FF00', - blue: '#0000FF', - indigo: '#4B0082', - violet: '#8F00FF' - } + red: "#FF0000", + orange: "#FF7F00", + yellow: "#FFFF00", + green: "#00FF00", + blue: "#0000FF", + indigo: "#4B0082", + violet: "#8F00FF", + }, } diff --git a/remix/app/Providers/ThingtimeProvider.tsx b/remix/app/Providers/ThingtimeProvider.tsx index 650ba1c..7016657 100644 --- a/remix/app/Providers/ThingtimeProvider.tsx +++ b/remix/app/Providers/ThingtimeProvider.tsx @@ -22,7 +22,7 @@ try { const force = { settings: { - commanderActive: true, + commanderActive: false, }, version: 22, } @@ -89,6 +89,7 @@ export const ThingtimeProvider = (props: any): JSX.Element => { thingtimeRef.current = thingtime try { + console.log("Setting thingtime to localStorage") window.localStorage.setItem("thingtime", JSON.stringify(thingtime)) } catch (err) { console.error("There was an error saving thingtime to localStorage") diff --git a/remix/app/components/Commander/Commander.tsx b/remix/app/components/Commander/Commander.tsx index 14516df..e2e6f44 100644 --- a/remix/app/components/Commander/Commander.tsx +++ b/remix/app/components/Commander/Commander.tsx @@ -45,14 +45,29 @@ export const Commander = (props) => { if (commanderActive) { inputRef?.current?.focus?.() } else { + document.activeElement.blur() + if (thingtimeRef?.current?.settings?.clearCommanderOnToggle) { setValue("") } if (thingtimeRef?.current?.settings?.clearCommanderContextOnToggle) { setShowContext(false, "commanderActive useEffect") } + if (contextPath !== undefined) { + setContextPath(undefined) + } + if (showContext !== false) { + setShowContext(false) + } } - }, [commanderActive, thingtimeRef, setShowContext]) + }, [ + commanderActive, + thingtimeRef, + setShowContext, + value, + contextPath, + showContext, + ]) const onChange = React.useCallback((e) => { setValue(e.target.value) @@ -92,16 +107,6 @@ export const Commander = (props) => { return command?.[1] }, [command]) - const commandContainsPath = React.useMemo(() => { - console.log("nik command", commandPath) - // const commandIncludesSuggestion = suggestions?.find(suggestion => { - // return commandPath?.includes(suggestion) - // }) - // console.log('nik commandIncludesSuggestion', commandIncludesSuggestion) - return false - // return commandIncludesSuggestion - }, [commandPath]) - const validQuotations = React.useMemo(() => { return ['"', "'"] }, []) @@ -148,34 +153,66 @@ export const Commander = (props) => { // console.log('nik suggestions', suggestions) - console.log("nik useEffect suggestions commandPath", commandPath) + const removeSuggestions = [contextPath] + + let ret = [] if (commandPath) { const filteredSuggestions = suggestions.filter((suggestion, i) => { - return suggestion?.toLowerCase()?.includes(commandPath?.toLowerCase()) + return ( + suggestion?.toLowerCase()?.includes(commandPath?.toLowerCase()) || + suggestion?.includes(commandPath) + ) }) if (!filteredSuggestions?.includes(commandPath)) { const adjustedSuggestions = [commandPath, ...filteredSuggestions] - return adjustedSuggestions + ret = adjustedSuggestions } else { - return filteredSuggestions + ret = filteredSuggestions } } else { - return suggestions + ret = suggestions } + const filtered = ret.filter((suggestion) => { + return !removeSuggestions?.includes(suggestion) + }) + + console.log("nik useEffect suggestions commandPath", commandPath) + console.log("nik useEffect suggestions ret", ret) + console.log("nik useEffect suggestions suggestions", suggestions) + console.log("nik useEffect suggestions filtered", filtered) + + return filtered + // if (value) { // setShowContext(true, 'Thingtime changes update suggestions') // } } - }, [value, thingtime, commandPath]) + }, [value, thingtime, commandPath, contextPath]) + + const commandContainsPath = React.useMemo(() => { + console.log("nik command", commandPath) + console.log("nik suggestions", suggestions) + const commandIncludesSuggestion = suggestions?.find((suggestion) => { + return commandPath?.includes(suggestion) + }) + console.log("nik commandIncludesSuggestion", commandIncludesSuggestion) + // return false + return commandIncludesSuggestion + }, [commandPath, suggestions]) const onEnter = React.useCallback( (props) => { // if first characters of value equal tt. then run command // or if first character is a dot then run command try { - console.log("Commander onEnter") + console.log("nik Commander onEnter") + console.log("nik commandIsAction", commandIsAction) + console.log("nik commandContainsPath", commandContainsPath) + + console.log("nik onEnter commandPath", commandPath) + if (commandIsAction) { // nothing try { @@ -193,6 +230,10 @@ export const Commander = (props) => { console.log("setThingtime errored in Commander", err) } } else if (commandContainsPath) { + // const prevValue = getThingtime(commandPath) + + // const newValue = setThingtime(commandPath, prevValue) + console.log("Setting context path", commandPath) setContextPath(commandPath) setShowContext(true, "commandContainsPath check") @@ -233,31 +274,16 @@ export const Commander = (props) => { setThingtime("settings.commanderActive", true) }, [setThingtime]) - const closeCommander = React.useCallback(() => { - if (thingtime?.settings?.commanderActive) { - console.log("nik commander closing commander") - setThingtime("settings.commanderActive", false) - } - - document.activeElement.blur() - - if (value !== "") { - setValue("") - } - if (contextPath !== undefined) { - setContextPath(undefined) - } - if (showContext !== false) { - setShowContext(false) - } - }, [ - setThingtime, - setShowContext, - value, - contextPath, - showContext, - thingtime?.settings?.commanderActive, - ]) + const closeCommander = React.useCallback( + (e) => { + console.log("nik e", e) + if (thingtime?.settings?.commanderActive) { + console.log("nik commander closing commander") + setThingtime("settings.commanderActive", false) + } + }, + [setThingtime, thingtime?.settings?.commanderActive] + ) const toggleCommander = React.useCallback(() => { if (thingtime?.settings?.commanderActive) { @@ -311,7 +337,7 @@ export const Commander = (props) => { return "calc(100vw - 45px)" }, []) - const rainbowRepeats = 1 + const rainbowRepeats = 2 return ( @@ -326,11 +352,10 @@ export const Commander = (props) => { justifyContent={["flex-start", "center"]} // display={["flex", commanderActive ? "flex" : "none"]} maxWidth="100%" - height="100%" + height={12} + // height="100%" pointerEvents="none" id="commander" - paddingX={1} - paddingY={1} > { selectSuggestion(suggestion)} @@ -397,6 +422,7 @@ export const Commander = (props) => { opacity={commanderActive ? 0.25 : 0} repeats={rainbowRepeats} thickness={8} + opacityTransition="all 1000ms ease" overflow="visible" >
{ outline="none" > { onChange={onChange} onFocus={openCommander} onKeyDown={onKeyDown} - placeholder={"What's on your mind?"} + placeholder="Imagine.." value={value} >
diff --git a/remix/app/components/Nav/Nav.tsx b/remix/app/components/Nav/Nav.tsx index 3d1e44f..2317771 100644 --- a/remix/app/components/Nav/Nav.tsx +++ b/remix/app/components/Nav/Nav.tsx @@ -1,10 +1,11 @@ -import React from 'react' -import { Box, Flex } from '@chakra-ui/react' -import { RainbowSkeleton } from '../Skeleton/RainbowSkeleton' -import { ProfileDrawer } from './ProfileDrawer' -import { Commander } from '../Commander/Commander' +import React from "react" +import { Box, Flex } from "@chakra-ui/react" -export const Nav = props => { +import { Commander } from "../Commander/Commander" +import { RainbowSkeleton } from "../Skeleton/RainbowSkeleton" +import { ProfileDrawer } from "./ProfileDrawer" + +export const Nav = (props) => { const [profileDrawerOpen, setProfileDrawerOpen] = React.useState(false) const toggleProfileDrawer = React.useCallback(() => { @@ -14,36 +15,37 @@ export const Nav = props => { return ( <> {/* */} diff --git a/remix/app/components/Rainbow/Rainbow.tsx b/remix/app/components/Rainbow/Rainbow.tsx index f8498e9..821b484 100644 --- a/remix/app/components/Rainbow/Rainbow.tsx +++ b/remix/app/components/Rainbow/Rainbow.tsx @@ -7,6 +7,8 @@ import { useTrace } from "~/hooks/useTrace" import { useUuid } from "~/hooks/useUuid" export const Rainbow = (allProps: any): JSX.Element => { + // return allProps.children + const rainbow = ["#f34a4a", "#ffbc48", "#58ca70", "#47b5e6", "#a555e8"] const props = useProps(allProps) @@ -71,6 +73,7 @@ export const Rainbow = (allProps: any): JSX.Element => { const keyframe = `${(i / (repeatedColours.length - 1)) * 100}%` ret[keyframe] = { fill: colour, + "animation-timing-function": "ease-out", stroke: colour, } }) @@ -153,7 +156,7 @@ export const Rainbow = (allProps: any): JSX.Element => { width: pathWidth || 1, animation: { name: `rainbow-${uuid}`, - duration: 5, + duration: props?.duration || 7, }, }) @@ -169,6 +172,7 @@ export const Rainbow = (allProps: any): JSX.Element => { } }, [ uuid, + props?.duration, props?.segments, props?.samples, props?.precision, diff --git a/remix/app/components/Thingtime/Thingtime.tsx b/remix/app/components/Thingtime/Thingtime.tsx index a5f9953..a280b77 100644 --- a/remix/app/components/Thingtime/Thingtime.tsx +++ b/remix/app/components/Thingtime/Thingtime.tsx @@ -1,9 +1,10 @@ -import React from 'react' -import { Box, Flex } from '@chakra-ui/react' -import { Safe } from '../Safety/Safe' -import { useThingtime } from './useThingtime' +import React from "react" +import { Box, Flex } from "@chakra-ui/react" -export const Thingtime = props => { +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 @@ -34,11 +35,11 @@ export const Thingtime = props => { }, [props.thing]) const mode = React.useMemo(() => { - return 'view' + return "view" }, []) const validKeyTypes = React.useMemo(() => { - return ['object', 'array'] + return ["object", "array"] }, []) const keys = React.useMemo(() => { @@ -55,42 +56,42 @@ export const Thingtime = props => { }, [thing]) const valuePl = React.useMemo(() => { - if (typeof props?.valuePl === 'number') { + 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') { + if (type === "string") { if (!thing) { - return 'Empty string' + return "Empty string" } return thing - } else if (type === 'number') { + } else if (type === "number") { return thing - } else if (type === 'boolean') { - return thing ? 'true' : 'false' - } else if (type === 'object') { + } else if (type === "boolean") { + return thing ? "true" : "false" + } else if (type === "object") { if (thing === null) { - return 'null' + return "null" } if (!keys?.length) { - return 'Empty object' + return "Something!" } try { return JSON.stringify(thing, null, 2) } catch (err) { console.error( - 'Caught error making renderableValue of thing', + "Caught error making renderableValue of thing", err, thing ) - return 'Circular reference in object.' + return "Circular reference in object." } } else { - return 'Undefined' + return "Something!" } }, [thing, type]) @@ -108,24 +109,24 @@ export const Thingtime = props => { window.thingtime.tmp[randId] = 0 const recurse = (obj, prefix) => { - Object.keys(obj).forEach(key => { - if (typeof obj[key] === 'object') { + Object.keys(obj).forEach((key) => { + if (typeof obj[key] === "object") { if (window?.thingtime?.tmp[randId] < 1000) { window.thingtime.tmp[randId]++ - recurse(obj[key], `${prefix}${prefix && '.'}${key}`) + recurse(obj[key], `${prefix}${prefix && "."}${key}`) } else { - console.error('Recursion limit reached in Thingtime.tsx') + console.error("Recursion limit reached in Thingtime.tsx") } } else { ret.push({ - key: `${prefix}${prefix && '.'}${key}`, - human: `${prefix}${prefix && ' '}${key}` + key: `${prefix}${prefix && "."}${key}`, + human: `${prefix}${prefix && " "}${key}`, }) } }) } - recurse(thing, '') + recurse(thing, "") } catch (err) { // console.error('Error in Thingtime.tsx creating flattenedKeys', err) } @@ -139,27 +140,27 @@ export const Thingtime = props => { const keysToUse = keys // const keysToUse = flattenedKeys - const template1Modes = ['view', 'edit'] + const template1Modes = ["view", "edit"] if (template1Modes?.includes(mode)) { if (keys?.length) { value = ( {keysToUse?.length && keysToUse.map((key, idx) => { if (!key?.human) { key = { human: key, - key: key + key: key, } } @@ -183,13 +184,13 @@ export const Thingtime = props => { } else { editableValue = ( {renderableValue} @@ -199,7 +200,13 @@ export const Thingtime = props => { } const contextMenu = ( - + Settings ) @@ -207,26 +214,26 @@ export const Thingtime = props => { const [showContextMenu, setShowContextMenu] = React.useState(false) const humanPath = React.useMemo(() => { - if (typeof props?.path === 'string') { + if (typeof props?.path === "string") { return props?.path } - return props?.path?.human || '' + return props?.path?.human || "" }, [props?.path]) const path = React.useMemo(() => { - if (humanPath?.includes?.('hidden')) { + if (humanPath?.includes?.("hidden")) { return null } - if (humanPath?.includes?.('unique')) { + if (humanPath?.includes?.("unique")) { // take only path from before the string unique - return humanPath.split?.('unique')?.[0] + return humanPath.split?.("unique")?.[0] } return ( {humanPath} @@ -234,12 +241,12 @@ export const Thingtime = props => { }, [humanPath, pl, props?.pathPl]) const handleMouseEvent = React.useCallback( - e => { + (e) => { const target = e?.target // extract uuid from className const className = target?.className if (className?.includes(uuid?.current)) { - setShowContextMenu(e?.type === 'mouseenter') + setShowContextMenu(e?.type === "mouseenter") } }, [uuid] @@ -248,15 +255,15 @@ export const Thingtime = props => { return ( diff --git a/remix/app/functions/sanitise.tsx b/remix/app/functions/sanitise.tsx index 0314f3a..7ccdb65 100644 --- a/remix/app/functions/sanitise.tsx +++ b/remix/app/functions/sanitise.tsx @@ -1,19 +1,26 @@ -export const sanitise = str => { - const isTT = str?.slice(0, 3) === 'tt.' - const isThingtime = str?.slice(0, 10) === 'thingtime.' - const isDot = str?.slice(0, 1) === '.' +export const sanitise = (str) => { + let ret = "" - console.log('nik thingtime sanitis 1', str, isTT, isThingtime, isDot) + const value = str?.trim() - if (isTT) { - str = str?.slice(3) - } else if (isThingtime) { - str = str?.slice(9) - } else if (isDot) { - str = str?.slice(1) + const sanitised = ["tt", "thingtime"] + + const suffixes = [".", " "] + + // find combination of sanitised + suffixes which str starts with + + const match = sanitised.find((s) => value?.startsWith(s)) + const withSuffix = suffixes.find( + (s) => match?.length && value[match?.length] === s + ) + const noSuffix = match && value?.length === match?.length + + if (match && withSuffix) { + ret = value?.slice(match.length + 1) + } else if (!noSuffix) { + ret = value } + console.log("nik thingtime sanitis 1", match, withSuffix, ret) - console.log('nik thingtime sanitise 2', str, isTT, isThingtime, isDot) - - return str + return ret } From cca6f44b8e49b1439710f3e1e9cbff8a7a0f6c94 Mon Sep 17 00:00:00 2001 From: Nikolaj Frey Date: Mon, 10 Jul 2023 10:37:50 +1000 Subject: [PATCH 2/3] feat: feature/rainbow-component Added Ode --- remix/app/components/Thingtime/Thingtime.tsx | 9 ++-- remix/app/routes/index.tsx | 1 + remix/app/routes/ode.tsx | 47 ++++++++++++++++++++ 3 files changed, 54 insertions(+), 3 deletions(-) create mode 100644 remix/app/routes/ode.tsx diff --git a/remix/app/components/Thingtime/Thingtime.tsx b/remix/app/components/Thingtime/Thingtime.tsx index a280b77..0447cd7 100644 --- a/remix/app/components/Thingtime/Thingtime.tsx +++ b/remix/app/components/Thingtime/Thingtime.tsx @@ -64,10 +64,12 @@ export const Thingtime = (props) => { const renderableValue = React.useMemo(() => { if (type === "string") { - if (!thing) { + const trimmed = thing.trim() + + if (!trimmed) { return "Empty string" } - return thing + return trimmed } else if (type === "number") { return thing } else if (type === "boolean") { @@ -257,7 +259,8 @@ export const Thingtime = (props) => { + {/* */} + {/* */} +
+ +
+
+ ) +} From 5305e407c9624b1533466952ac02f12165ea0182 Mon Sep 17 00:00:00 2001 From: Nikolaj Frey Date: Mon, 10 Jul 2023 10:41:08 +1000 Subject: [PATCH 3/3] feat: feature/rainbow-component --- remix/app/routes/ode.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/remix/app/routes/ode.tsx b/remix/app/routes/ode.tsx index 60a1932..34c5e53 100644 --- a/remix/app/routes/ode.tsx +++ b/remix/app/routes/ode.tsx @@ -39,8 +39,8 @@ export default function Index() { > {/* */} {/* */} -
- +
+
)