From 9f9adf4665fc00ea07fb87054039af51a99c0a4a Mon Sep 17 00:00:00 2001 From: wheattoast11 Date: Sun, 1 Dec 2024 13:00:07 -0600 Subject: [PATCH] major style updates --- index.html | 20 +- src/App.jsx | 291 ++++- src/Video.jsx | 199 ++-- src/components/Canvas.jsx | 64 -- src/components/CardInteraction.jsx | 135 +++ src/components/ErrorBoundary.jsx | 172 +-- src/components/LayeredBackground.jsx | 168 +++ src/components/LoadingState.jsx | 117 +- src/components/MouseInteraction.jsx | 90 ++ src/components/Navigation.jsx | 50 + src/components/ParallaxContainer.jsx | 6 + src/components/PlatformMarquee.jsx | 76 ++ src/components/PtsCanvas.jsx | 160 +++ src/components/ScrollManager.jsx | 50 + src/hooks/useMouseEffect.js | 69 ++ src/hooks/useReducedMotion.js | 17 + src/main.jsx | 29 +- src/pages/Home.jsx | 288 ++--- src/pages/Manifesto.jsx | 145 ++- src/style.css | 1439 ++++++++++++++++++++++++ src/three/FlowField.js | 266 +++-- src/three/Scene.js | 355 +++--- src/three/Spheres.js | 166 ++- src/ui/CardEffects.js | 27 +- src/ui/ScrollEffects.js | 67 +- style.css | 1541 ++++++++++++++++++++++++-- vite.config.js | 99 +- 27 files changed, 5098 insertions(+), 1008 deletions(-) delete mode 100644 src/components/Canvas.jsx create mode 100644 src/components/CardInteraction.jsx create mode 100644 src/components/LayeredBackground.jsx create mode 100644 src/components/MouseInteraction.jsx create mode 100644 src/components/Navigation.jsx create mode 100644 src/components/ParallaxContainer.jsx create mode 100644 src/components/PlatformMarquee.jsx create mode 100644 src/components/PtsCanvas.jsx create mode 100644 src/components/ScrollManager.jsx create mode 100644 src/hooks/useMouseEffect.js create mode 100644 src/hooks/useReducedMotion.js create mode 100644 src/style.css diff --git a/index.html b/index.html index f63a437..71ccd89 100644 --- a/index.html +++ b/index.html @@ -7,6 +7,20 @@ + Intuition Labs LLC - AI Research & Consulting @@ -24,12 +38,12 @@ - - + + diff --git a/src/App.jsx b/src/App.jsx index f7ee7e8..30b9afb 100644 --- a/src/App.jsx +++ b/src/App.jsx @@ -1,43 +1,278 @@ -import React, { Suspense, lazy, useEffect } from 'react'; -import { - createBrowserRouter, - RouterProvider, - UNSAFE_DataRouterContext, - UNSAFE_DataRouterStateContext -} from 'react-router-dom'; +import React, { Suspense, lazy, useState, useEffect } from 'react'; +import { createBrowserRouter, RouterProvider } from 'react-router-dom'; +import { m, AnimatePresence } from 'framer-motion'; import LoadingState from './components/LoadingState'; -import Canvas from './components/Canvas'; import ErrorBoundary from './components/ErrorBoundary'; -import reportWebVitals from './utils/reportWebVitals'; +import Navigation from './components/Navigation'; +import CardInteraction from './components/CardInteraction'; +import LayeredBackground from './components/LayeredBackground'; +import MouseInteraction from './components/MouseInteraction'; +import { ScrollEffects } from '../src/ui/ScrollEffects'; -// Lazy load components const Home = lazy(() => import('./pages/Home')); -const Manifesto = lazy(() => import('./pages/Manifesto')); -const router = createBrowserRouter([ - { - path: "/", - element: , +const sections = [ + { id: 'hero', label: 'Home' }, + { id: 'services', label: 'Services' }, + { id: 'manifesto', label: 'Manifesto' }, + { id: 'ecosystem', label: 'Ecosystem' } +]; + +const ecosystemItems = { + platforms: { + title: 'Platforms', + links: [ + { label: 'Terminals', href: '/platforms/terminals' }, + { label: 'Radical', href: '/platforms/radical' }, + { label: 'PathFinder', href: '/platforms/pathfinder' }, + { label: 'Wuji', href: '/platforms/wuji' } + ] }, - { - path: "/manifesto", - element: , - } -], { - future: { - v7_startTransition: true, - v7_relativeSplatPath: true + hardware: { + title: 'Hardware', + links: [ + { label: 'Neural Interfaces', href: '/hardware/neural' }, + { label: 'Haptics', href: '/hardware/haptics' }, + { label: 'Sensors', href: '/hardware/sensors' } + ] + }, + experiences: { + title: 'Experiences', + links: [ + { label: 'Virtual Labs', href: '/experiences/labs' }, + { label: 'Training', href: '/experiences/training' }, + { label: 'Research', href: '/experiences/research' } + ] + }, + stash: { + title: 'Stash', + links: [ + { label: 'Documentation', href: '/stash/docs' }, + { label: 'Resources', href: '/stash/resources' }, + { label: 'Tools', href: '/stash/tools' } + ] } -}); +}; + +function MainApp() { + const [currentSection, setCurrentSection] = useState('hero'); + const [openDrawer, setOpenDrawer] = useState(null); + const scrollEffectsRef = React.useRef(null); -function App() { useEffect(() => { - if ('performance' in window) { - window.performance.mark('app_start'); + // Initialize scroll effects + scrollEffectsRef.current = new ScrollEffects(); + + return () => { + if (scrollEffectsRef.current) { + scrollEffectsRef.current.dispose(); + } + }; + }, []); + + useEffect(() => { + const handleScroll = () => { + const sections = document.querySelectorAll('.section'); + sections.forEach(section => { + const rect = section.getBoundingClientRect(); + if (rect.top >= -100 && rect.top <= window.innerHeight / 3) { + setCurrentSection(section.id); + section.classList.add('active'); + } else { + section.classList.remove('active'); + } + }); + }; + + const mainContent = document.querySelector('.main-content'); + if (mainContent) { + mainContent.addEventListener('scroll', handleScroll, { passive: true }); + return () => mainContent.removeEventListener('scroll', handleScroll); } - reportWebVitals(console.log); }, []); + const handleNavigate = (sectionId) => { + const section = document.getElementById(sectionId); + if (section) { + section.scrollIntoView({ behavior: 'smooth' }); + } + }; + + const toggleDrawer = (drawer) => { + setOpenDrawer(openDrawer === drawer ? null : drawer); + }; + + return ( +
+ + + + +
+
+ +
+ +
+
+ + Services +
+ +
+ +
+

Enterprise AI Strategy

+
    +
  • Comprehensive AI Readiness Assessment
  • +
  • Platform Selection & Integration Planning
  • +
  • Risk Analysis & Compliance Framework
  • +
  • ROI Modeling & Performance Metrics
  • +
  • Change Management & Training Programs
  • +
  • Scalability & Growth Planning
  • +
+
+
+ + +
+

LLM Systems

+
    +
  • Custom Model Architecture & Development
  • +
  • Advanced Prompt Engineering & Optimization
  • +
  • Multi-Modal System Integration
  • +
  • Performance Tuning & Scaling
  • +
  • Security & Privacy Implementation
  • +
  • Continuous Learning & Adaptation
  • +
+
+
+
+
+
+ +
+
+ + Manifesto +
+ + + +

Vision

+

We believe in creating AI systems that are ethical, transparent, and focused on enhancing human capabilities. Our approach combines cutting-edge technology with deep industry knowledge to deliver solutions that drive meaningful transformation.

+
+ +
+ +
+

Purpose

+

Building AI that serves humanity's highest potential

+
+
+ + +
+

Innovation

+

Pushing boundaries while maintaining ethical standards

+
+
+ + +
+

Collaboration

+

Working together to create meaningful solutions

+
+
+
+
+
+ +
+
+ + Ecosystem +
+ + +
+ {Object.entries(ecosystemItems).map(([key, item]) => ( + toggleDrawer(key)} + initial={{ opacity: 0, y: 20 }} + whileInView={{ opacity: 1, y: 0 }} + viewport={{ once: true }} + > +
+

{item.title}

+ + ↓ + +
+
+ +
+
+ ))} +
+
+
+
+
+ ); +} + +const router = createBrowserRouter([ + { + path: "/", + element: ( + }> + + + ), + errorElement: + } +]); + +function App() { return ( diff --git a/src/Video.jsx b/src/Video.jsx index a756429..8695f86 100644 --- a/src/Video.jsx +++ b/src/Video.jsx @@ -4,138 +4,203 @@ import { AbsoluteFill } from 'remotion'; import { createNoise4D } from 'simplex-noise'; const BackgroundAnimation = () => { - const noise4D = createNoise4D(); + const canvasRef = React.useRef(null); + const animationFrameRef = React.useRef(null); + const noise4D = React.useMemo(() => createNoise4D(), []); + const particlesRef = React.useRef(null); + const lastTimeRef = React.useRef(0); + const [dimensions, setDimensions] = React.useState({ width: window.innerWidth, height: window.innerHeight }); - + React.useEffect(() => { const handleResize = () => { - setDimensions({ - width: window.innerWidth, - height: window.innerHeight - }); + const canvas = canvasRef.current; + if (canvas) { + canvas.width = window.innerWidth * window.devicePixelRatio; + canvas.height = window.innerHeight * window.devicePixelRatio; + canvas.style.width = `${window.innerWidth}px`; + canvas.style.height = `${window.innerHeight}px`; + setDimensions({ + width: window.innerWidth, + height: window.innerHeight + }); + } }; - window.addEventListener('resize', handleResize); - return () => window.removeEventListener('resize', handleResize); + const debouncedResize = debounce(handleResize, 100); + window.addEventListener('resize', debouncedResize); + handleResize(); + + return () => window.removeEventListener('resize', debouncedResize); }, []); - + React.useEffect(() => { - const canvas = document.getElementById('remotionCanvas'); + const canvas = canvasRef.current; if (!canvas) return; - const ctx = canvas.getContext('2d'); - let animationFrame; - - // Increase number of particles and add variation - const particles = Array.from({ length: 150 }, () => ({ - x: Math.random() * dimensions.width, - y: Math.random() * dimensions.height, - size: Math.random() * 3 + 0.5, - speedX: (Math.random() - 0.5) * 0.3, - speedY: (Math.random() - 0.5) * 0.3, - opacity: Math.random() * 0.5 + 0.1 - })); + const ctx = canvas.getContext('2d', { + alpha: true, + desynchronized: true, + willReadFrequently: false + }); + if (!ctx) return; + + canvas.width = dimensions.width * window.devicePixelRatio; + canvas.height = dimensions.height * window.devicePixelRatio; + ctx.scale(window.devicePixelRatio, window.devicePixelRatio); + + // Initialize particles only once + if (!particlesRef.current) { + particlesRef.current = Array.from({ length: 200 }, () => ({ + x: Math.random() * dimensions.width, + y: Math.random() * dimensions.height, + size: Math.random() * 4 + 1, + speedX: (Math.random() - 0.5) * 0.5, + speedY: (Math.random() - 0.5) * 0.5, + opacity: Math.random() * 0.7 + 0.3, + hue: Math.random() * 20 + 180 + })); + } + + let isDestroyed = false; const animate = (time) => { - canvas.width = dimensions.width; - canvas.height = dimensions.height; + if (isDestroyed) return; + + // Throttle to 60fps + if (time - lastTimeRef.current < 16.67) { + animationFrameRef.current = requestAnimationFrame(animate); + return; + } + lastTimeRef.current = time; - // Enhanced gradient background + ctx.clearRect(0, 0, dimensions.width, dimensions.height); + + // Draw background gradient (cached version) const gradient = ctx.createRadialGradient( dimensions.width / 2, dimensions.height / 2, 0, dimensions.width / 2, dimensions.height / 2, - dimensions.width * 0.7 + dimensions.width * 0.8 ); - gradient.addColorStop(0, 'rgba(122, 158, 159, 0.03)'); - gradient.addColorStop(0.5, 'rgba(80, 108, 127, 0.02)'); - gradient.addColorStop(1, 'rgba(0, 0, 0, 0)'); + gradient.addColorStop(0, 'rgba(158, 207, 208, 0.05)'); + gradient.addColorStop(0.5, 'rgba(122, 158, 159, 0.03)'); + gradient.addColorStop(1, 'rgba(80, 108, 127, 0.02)'); ctx.fillStyle = gradient; - ctx.fillRect(0, 0, canvas.width, canvas.height); + ctx.fillRect(0, 0, dimensions.width, dimensions.height); - // Update and draw particles with enhanced effects - particles.forEach(particle => { - const noiseX = noise4D(particle.x * 0.001, particle.y * 0.001, time * 0.0001, 0) * 2; - const noiseY = noise4D(particle.x * 0.001, particle.y * 0.001, 0, time * 0.0001) * 2; + // Batch particle updates + ctx.globalCompositeOperation = 'screen'; + const particles = particlesRef.current; + const currentTime = time * 0.0002; + + for (let i = 0; i < particles.length; i++) { + const particle = particles[i]; + + // Update position with noise + const noiseX = noise4D(particle.x * 0.002, particle.y * 0.002, currentTime, 0) * 3; + const noiseY = noise4D(particle.x * 0.002, particle.y * 0.002, 0, currentTime) * 3; particle.x += particle.speedX + noiseX; particle.y += particle.speedY + noiseY; - // Improved wrapping logic - if (particle.x < -50) particle.x = canvas.width + 50; - if (particle.x > canvas.width + 50) particle.x = -50; - if (particle.y < -50) particle.y = canvas.height + 50; - if (particle.y > canvas.height + 50) particle.y = -50; + // Wrap around screen with buffer + if (particle.x < -100) particle.x = dimensions.width + 100; + if (particle.x > dimensions.width + 100) particle.x = -100; + if (particle.y < -100) particle.y = dimensions.height + 100; + if (particle.y > dimensions.height + 100) particle.y = -100; - // Draw particle with glow effect + // Draw particle with optimized glow const glow = ctx.createRadialGradient( particle.x, particle.y, 0, particle.x, particle.y, - particle.size * 2 + particle.size * 3 ); - glow.addColorStop(0, `rgba(122, 158, 159, ${particle.opacity})`); - glow.addColorStop(1, 'rgba(122, 158, 159, 0)'); + const color = `hsla(${particle.hue}, 70%, 70%,`; + glow.addColorStop(0, `${color} ${particle.opacity})`); + glow.addColorStop(0.5, `${color} ${particle.opacity * 0.5})`); + glow.addColorStop(1, `${color} 0)`); ctx.beginPath(); - ctx.arc(particle.x, particle.y, particle.size * 2, 0, Math.PI * 2); + ctx.arc(particle.x, particle.y, particle.size * 3, 0, Math.PI * 2); ctx.fillStyle = glow; ctx.fill(); - }); + } - // Draw flowing lines with enhanced effect + // Draw optimized flowing lines + ctx.globalCompositeOperation = 'screen'; ctx.beginPath(); - ctx.strokeStyle = 'rgba(122, 158, 159, 0.05)'; - ctx.lineWidth = 0.5; + ctx.strokeStyle = 'rgba(158, 207, 208, 0.1)'; + ctx.lineWidth = 1; + + const lineCount = Math.floor(dimensions.width / 40); + const stepSize = dimensions.width / lineCount; - for (let i = 0; i < canvas.width; i += 30) { - const startY = canvas.height / 2; - ctx.moveTo(i, startY); + for (let i = 0; i < lineCount; i++) { + const startX = i * stepSize; + const startY = dimensions.height / 2; + ctx.moveTo(startX, startY); - for (let x = 0; x < canvas.width; x += 5) { - const noise = noise4D(x * 0.002, startY * 0.002, time * 0.0001, 0); + for (let x = startX; x < startX + stepSize; x += 4) { + const noise = noise4D(x * 0.003, startY * 0.003, currentTime, 0); const y = startY + - Math.sin(x * 0.01 + time * 0.001) * 30 + - noise * 50; + Math.sin(x * 0.02 + time * 0.001) * 40 + + noise * 60; ctx.lineTo(x, y); } } ctx.stroke(); - animationFrame = requestAnimationFrame(animate); + animationFrameRef.current = requestAnimationFrame(animate); }; animate(0); return () => { - if (animationFrame) { - cancelAnimationFrame(animationFrame); + isDestroyed = true; + if (animationFrameRef.current) { + cancelAnimationFrame(animationFrameRef.current); } }; - }, [dimensions]); + }, [dimensions, noise4D]); return ( - + ); }; +// Debounce utility function +const debounce = (fn, ms) => { + let timer; + return (...args) => { + clearTimeout(timer); + timer = setTimeout(() => fn.apply(this, args), ms); + }; +}; + export const RemotionVideo = () => { const [dimensions, setDimensions] = React.useState({ width: window.innerWidth, @@ -143,12 +208,12 @@ export const RemotionVideo = () => { }); React.useEffect(() => { - const handleResize = () => { + const handleResize = debounce(() => { setDimensions({ width: window.innerWidth, height: window.innerHeight }); - }; + }, 100); window.addEventListener('resize', handleResize); return () => window.removeEventListener('resize', handleResize); @@ -167,7 +232,9 @@ export const RemotionVideo = () => { position: 'fixed', top: 0, left: 0, - zIndex: -1, + zIndex: 3, + opacity: 0.9, + mixBlendMode: 'screen' }} loop /> diff --git a/src/components/Canvas.jsx b/src/components/Canvas.jsx deleted file mode 100644 index 679973a..0000000 --- a/src/components/Canvas.jsx +++ /dev/null @@ -1,64 +0,0 @@ -import React, { useEffect, useRef, useState, useCallback } from 'react'; -import { Scene } from '../three/Scene'; - -function Canvas() { - const containerRef = useRef(null); - const sceneRef = useRef(null); - const [dimensions, setDimensions] = useState({ width: 0, height: 0 }); - - const handleResize = useCallback(() => { - if (containerRef.current) { - const { clientWidth, clientHeight } = containerRef.current; - setDimensions({ width: clientWidth, height: clientHeight }); - } - }, []); - - useEffect(() => { - const resizeObserver = new ResizeObserver(handleResize); - if (containerRef.current) { - resizeObserver.observe(containerRef.current); - } - return () => resizeObserver.disconnect(); - }, [handleResize]); - - useEffect(() => { - if (!containerRef.current || dimensions.width === 0) return; - - try { - if (!sceneRef.current) { - sceneRef.current = new Scene(containerRef.current, dimensions); - sceneRef.current.animate(0); - } else { - sceneRef.current.resize(dimensions); - } - } catch (error) { - console.error('Scene initialization error:', error); - } - - return () => { - if (sceneRef.current) { - sceneRef.current.dispose(); - sceneRef.current = null; - } - }; - }, [dimensions]); - - return ( -