From 31e95310ca80fe822fb299aaff108dc8b050024a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=D0=A0=D0=B0=D1=81=D1=83=D0=BB?= Date: Fri, 6 Dec 2024 13:18:21 +0300 Subject: [PATCH 1/2] feat: updated links --- .../ActionsToolbar/PlayerControl/index.tsx | 2 +- src/components/AppContainer/index.tsx | 36 +++++++++--- .../Graph/Connections/LineComponent.tsx | 18 +++--- .../Universe/Graph/Connections/index.tsx | 55 +------------------ .../Cubes/Text/hooks/useTexture/index.ts | 35 ++++++++---- src/components/Universe/constants.ts | 2 +- src/components/mindset/index.tsx | 2 +- 7 files changed, 67 insertions(+), 83 deletions(-) diff --git a/src/components/App/ActionsToolbar/PlayerControl/index.tsx b/src/components/App/ActionsToolbar/PlayerControl/index.tsx index ec9de2600..a4e65546b 100644 --- a/src/components/App/ActionsToolbar/PlayerControl/index.tsx +++ b/src/components/App/ActionsToolbar/PlayerControl/index.tsx @@ -54,7 +54,7 @@ export const PlayerControl = () => { const showPlayer = (sidebarIsOpen && selectedNode?.ref_id !== playingNode?.ref_id) || (playingNode && !sidebarIsOpen) - const isMindset = window.location?.hostname === 'graphmindset.sphinx.chat' + const isMindset = window.location?.hostname === 'graphmindset.sphinx.chat' || window.location.hostname === 'localhost' return miniPlayerIsVisible && playingNode && showPlayer && !isMindset ? ( diff --git a/src/components/AppContainer/index.tsx b/src/components/AppContainer/index.tsx index c9f8ea3d9..d6004c2da 100644 --- a/src/components/AppContainer/index.tsx +++ b/src/components/AppContainer/index.tsx @@ -4,23 +4,43 @@ import { E2ETests } from '~/utils' import { AppProviders } from '../App/Providers' import { AuthGuard } from '../Auth' +// Lazy-loaded components const LazyApp = lazy(() => import('../App').then(({ App }) => ({ default: App }))) const LazyMindSet = lazy(() => import('../mindset').then(({ MindSet }) => ({ default: MindSet }))) export const AppContainer = () => { - const App = - const MindSet = - - const path = window.location?.hostname === 'graphmindset.sphinx.chat' ? '/' : '/mindset' + const isMindSetHost = + window.location?.hostname === 'graphmindset.sphinx.chat' || window.location.hostname === 'localhost' return ( Loading...}> - - {App}} path="/" /> - {App}} path="/search" /> - {App}} path="*" /> + {isMindSetHost && } path="/" />} + + + + } + path="/" + /> + + + + } + path="/search" + /> + + + + } + path="*" + /> diff --git a/src/components/Universe/Graph/Connections/LineComponent.tsx b/src/components/Universe/Graph/Connections/LineComponent.tsx index a26270268..719e997d9 100644 --- a/src/components/Universe/Graph/Connections/LineComponent.tsx +++ b/src/components/Universe/Graph/Connections/LineComponent.tsx @@ -1,6 +1,6 @@ import { Line } from '@react-three/drei' import gsap from 'gsap' -import { forwardRef, memo, useEffect } from 'react' +import { memo, useEffect, useRef } from 'react' import { Vector3 } from 'three' import { Line2 } from 'three-stdlib' import { LinkPosition } from '..' @@ -12,31 +12,33 @@ type LineComponentProps = { } // eslint-disable-next-line no-underscore-dangle -const _LineComponent = forwardRef(({ isSelected, position }, ref) => { +const _LineComponent = ({ isSelected, position }: LineComponentProps) => { + const lineRef = useRef(null) + useEffect(() => { - if (ref && (ref as React.MutableRefObject).current) { - const line = (ref as React.MutableRefObject).current + if (lineRef.current) { + const line = lineRef.current gsap.fromTo( line.material, - { linewidth: LINE_WIDTH * 5 }, + { linewidth: LINE_WIDTH * 15 }, { linewidth: LINE_WIDTH, duration: 1, }, ) } - }, [isSelected, ref]) + }, [isSelected, lineRef]) return ( ) -}) +} _LineComponent.displayName = 'LineComponent' diff --git a/src/components/Universe/Graph/Connections/index.tsx b/src/components/Universe/Graph/Connections/index.tsx index a0553e3f6..5c80ff828 100644 --- a/src/components/Universe/Graph/Connections/index.tsx +++ b/src/components/Universe/Graph/Connections/index.tsx @@ -1,54 +1,18 @@ -import gsap from 'gsap' -import { memo, useEffect, useRef } from 'react' -import { Line2 } from 'three-stdlib' +import { memo } from 'react' import { useDataStore } from '~/stores/useDataStore' -import { useGraphStore, useHoveredNode, useSelectedNode } from '~/stores/useGraphStore' +import { useGraphStore, useSelectedNode } from '~/stores/useGraphStore' import { Link } from '~/types' import { LinkPosition } from '..' -import { LINE_WIDTH } from '../../constants' import { LineComponent } from './LineComponent' type Props = { linksPosition: Map } -const LINE_TRANSFORM_DURATION = 0.5 - export const Connections = memo(({ linksPosition }: Props) => { const data = useDataStore((s) => s.dataInitial) const { showSelectionGraph } = useGraphStore((s) => s) const selectedNode = useSelectedNode() - const hoveredNode = useHoveredNode() - const lineRefs = useRef(new Map()) - - useEffect(() => { - const activeNode = hoveredNode || selectedNode - - if (!activeNode) { - lineRefs.current.forEach((line) => { - if (line) { - gsap.to(line.material, { - linewidth: LINE_WIDTH, - duration: LINE_TRANSFORM_DURATION, - }) - } - }) - - return - } - - lineRefs.current.forEach((line, refId) => { - if (line) { - const link = data?.links.find((l) => l.ref_id === refId) - const isActive = link && (link.source === activeNode.ref_id || link.target === activeNode.ref_id) - - gsap.to(line.material, { - linewidth: isActive ? LINE_WIDTH * 2 : 0, - duration: LINE_TRANSFORM_DURATION, - }) - } - }) - }, [data?.links, hoveredNode, selectedNode]) return ( @@ -64,20 +28,7 @@ export const Connections = memo(({ linksPosition }: Props) => { tz: 0, } - return ( - { - if (el) { - lineRefs.current.set(l.ref_id, el as Line2) - } else { - lineRefs.current.delete(l.ref_id) - } - }} - isSelected={isSelected} - position={position} - /> - ) + return })} ) diff --git a/src/components/Universe/Graph/Cubes/Text/hooks/useTexture/index.ts b/src/components/Universe/Graph/Cubes/Text/hooks/useTexture/index.ts index 896732e3d..26dfffe87 100644 --- a/src/components/Universe/Graph/Cubes/Text/hooks/useTexture/index.ts +++ b/src/components/Universe/Graph/Cubes/Text/hooks/useTexture/index.ts @@ -1,16 +1,16 @@ import { useEffect, useState } from 'react' -import { Texture } from 'three' +import * as THREE from 'three' // Use * as THREE for consistency import { loader } from './constants' -type materialRecord = { +type MaterialRecord = { texture: THREE.Texture material: THREE.MeshStandardMaterial } -const cachedMaterials: Record = {} +const cachedMaterials: Record = {} export const useTexture = (url: string) => { - const [texture, setTexture] = useState(null) + const [texture, setTexture] = useState(null) useEffect(() => { if (!url) { @@ -19,33 +19,44 @@ export const useTexture = (url: string) => { return } - const cashPath = url + const cachePath = url - if (cachedMaterials[cashPath]) { - setTexture(cachedMaterials[cashPath].texture) + // Check if texture is already cached + if (cachedMaterials[cachePath]?.texture) { + setTexture(cachedMaterials[cachePath].texture) return } + // Load texture and cache it loader.load( url, (loadedTexture) => { + cachedMaterials[cachePath] = { + texture: loadedTexture, + material: new THREE.MeshStandardMaterial({ map: loadedTexture }), + } + setTexture(loadedTexture) }, undefined, () => { - setTexture(null) + setTexture(null) // Handle loading error }, ) }, [url]) useEffect( - () => - function cleanup() { - if (texture) { + () => () => { + if (texture) { + // Avoid disposing of cached textures + const isCached = Object.values(cachedMaterials).some((entry) => entry.texture === texture) + + if (!isCached) { texture.dispose() } - }, + } + }, [texture], ) diff --git a/src/components/Universe/constants.ts b/src/components/Universe/constants.ts index 9ead991d8..a6b41bd2b 100644 --- a/src/components/Universe/constants.ts +++ b/src/components/Universe/constants.ts @@ -3,7 +3,7 @@ import { Guests, NodeExtended } from '~/types' export const variableVector3 = new Vector3(0, 0, 0) -export const LINE_WIDTH = 0.5 +export const LINE_WIDTH = 1 export const outlineEffectColor = 0xffffff diff --git a/src/components/mindset/index.tsx b/src/components/mindset/index.tsx index fb007818f..2c1b24000 100644 --- a/src/components/mindset/index.tsx +++ b/src/components/mindset/index.tsx @@ -80,7 +80,7 @@ export const MindSet = () => { const [episodesAndClips, remainingNodes] = (data?.nodes || []).reduce<[Node[], Node[]]>( ([matches, remaining], node) => { - if (['Episode', 'Show'].includes(node.node_type)) { + if (['Episode', 'Show', 'Host', 'Guest'].includes(node.node_type)) { matches.push(node) } else { remaining.push(node) From d4e6eb7686ab81e9206669a881328020c7d95f6a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=D0=A0=D0=B0=D1=81=D1=83=D0=BB?= Date: Fri, 6 Dec 2024 13:26:24 +0300 Subject: [PATCH 2/2] feat: added image support --- src/components/App/ActionsToolbar/PlayerControl/index.tsx | 2 +- src/components/Universe/Graph/Cubes/Text/index.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/App/ActionsToolbar/PlayerControl/index.tsx b/src/components/App/ActionsToolbar/PlayerControl/index.tsx index a4e65546b..ec9de2600 100644 --- a/src/components/App/ActionsToolbar/PlayerControl/index.tsx +++ b/src/components/App/ActionsToolbar/PlayerControl/index.tsx @@ -54,7 +54,7 @@ export const PlayerControl = () => { const showPlayer = (sidebarIsOpen && selectedNode?.ref_id !== playingNode?.ref_id) || (playingNode && !sidebarIsOpen) - const isMindset = window.location?.hostname === 'graphmindset.sphinx.chat' || window.location.hostname === 'localhost' + const isMindset = window.location?.hostname === 'graphmindset.sphinx.chat' return miniPlayerIsVisible && playingNode && showPlayer && !isMindset ? ( diff --git a/src/components/Universe/Graph/Cubes/Text/index.tsx b/src/components/Universe/Graph/Cubes/Text/index.tsx index 31518e065..ce5743a77 100644 --- a/src/components/Universe/Graph/Cubes/Text/index.tsx +++ b/src/components/Universe/Graph/Cubes/Text/index.tsx @@ -166,7 +166,7 @@ export const TextNode = memo(({ node, hide, ignoreDistance }: Props) => { - {node.properties?.image_url && ['Person', 'Episode'].includes(node.node_type) && texture ? ( + {node.properties?.image_url && ['Person', 'Episode', 'Guest', 'Host'].includes(node.node_type) && texture ? (