feat: add animated text flip and pointer highlight components with home agent section
This commit is contained in:
58
src/components/ui/layout-text-flip.tsx
Normal file
58
src/components/ui/layout-text-flip.tsx
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
"use client";
|
||||||
|
import React, { useState, useEffect } from "react";
|
||||||
|
import { motion, AnimatePresence } from "motion/react";
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
|
||||||
|
export const LayoutTextFlip = ({
|
||||||
|
text = "Build Amazing",
|
||||||
|
words = ["Landing Pages", "Component Blocks", "Page Sections", "3D Shaders"],
|
||||||
|
duration = 3000,
|
||||||
|
}: {
|
||||||
|
text: string;
|
||||||
|
words: string[];
|
||||||
|
duration?: number;
|
||||||
|
}) => {
|
||||||
|
const [currentIndex, setCurrentIndex] = useState(0);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const interval = setInterval(() => {
|
||||||
|
setCurrentIndex((prevIndex) => (prevIndex + 1) % words.length);
|
||||||
|
}, duration);
|
||||||
|
|
||||||
|
return () => clearInterval(interval);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<motion.span
|
||||||
|
layoutId="subtext"
|
||||||
|
className="text-2xl font-bold tracking-tight drop-shadow-lg md:text-4xl"
|
||||||
|
>
|
||||||
|
{text}
|
||||||
|
</motion.span>
|
||||||
|
|
||||||
|
<motion.span
|
||||||
|
layout
|
||||||
|
className="relative w-fit overflow-hidden px-8 py-2 font-neuton font-medium italic tracking-tight"
|
||||||
|
>
|
||||||
|
<AnimatePresence mode="popLayout">
|
||||||
|
<motion.span
|
||||||
|
key={currentIndex}
|
||||||
|
initial={{ y: -40, filter: "blur(10px)" }}
|
||||||
|
animate={{
|
||||||
|
y: 0,
|
||||||
|
filter: "blur(0px)",
|
||||||
|
}}
|
||||||
|
exit={{ y: 50, filter: "blur(10px)", opacity: 0 }}
|
||||||
|
transition={{
|
||||||
|
duration: 0.5,
|
||||||
|
}}
|
||||||
|
className={cn("inline-block whitespace-nowrap")}
|
||||||
|
>
|
||||||
|
{words[currentIndex]}
|
||||||
|
</motion.span>
|
||||||
|
</AnimatePresence>
|
||||||
|
</motion.span>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
119
src/components/ui/pointer-highlight.tsx
Normal file
119
src/components/ui/pointer-highlight.tsx
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
"use client";
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
import { motion } from "motion/react";
|
||||||
|
import { useRef, useEffect, useState } from "react";
|
||||||
|
|
||||||
|
export function PointerHighlight({
|
||||||
|
children,
|
||||||
|
rectangleClassName,
|
||||||
|
pointerClassName,
|
||||||
|
containerClassName,
|
||||||
|
}: {
|
||||||
|
children: React.ReactNode;
|
||||||
|
rectangleClassName?: string;
|
||||||
|
pointerClassName?: string;
|
||||||
|
containerClassName?: string;
|
||||||
|
}) {
|
||||||
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
|
const [dimensions, setDimensions] = useState({ width: 0, height: 0 });
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (containerRef.current) {
|
||||||
|
const { width, height } = containerRef.current.getBoundingClientRect();
|
||||||
|
setDimensions({ width, height });
|
||||||
|
}
|
||||||
|
|
||||||
|
const resizeObserver = new ResizeObserver((entries) => {
|
||||||
|
for (const entry of entries) {
|
||||||
|
const { width, height } = entry.contentRect;
|
||||||
|
setDimensions({ width, height });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (containerRef.current) {
|
||||||
|
resizeObserver.observe(containerRef.current);
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
if (containerRef.current) {
|
||||||
|
resizeObserver.unobserve(containerRef.current);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={cn("relative w-fit", containerClassName)}
|
||||||
|
ref={containerRef}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
{dimensions.width > 0 && dimensions.height > 0 && (
|
||||||
|
<motion.div
|
||||||
|
className="pointer-events-none absolute inset-0 z-0"
|
||||||
|
initial={{ opacity: 0, scale: 0.95, originX: 0, originY: 0 }}
|
||||||
|
animate={{ opacity: 1, scale: 1 }}
|
||||||
|
transition={{ duration: 0.5, ease: "easeOut" }}
|
||||||
|
>
|
||||||
|
<motion.div
|
||||||
|
className={cn(
|
||||||
|
"absolute inset-0 border border-neutral-800 dark:border-neutral-200",
|
||||||
|
rectangleClassName,
|
||||||
|
)}
|
||||||
|
initial={{
|
||||||
|
width: 0,
|
||||||
|
height: 0,
|
||||||
|
}}
|
||||||
|
whileInView={{
|
||||||
|
width: dimensions.width,
|
||||||
|
height: dimensions.height,
|
||||||
|
}}
|
||||||
|
transition={{
|
||||||
|
duration: 1,
|
||||||
|
ease: "easeInOut",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<motion.div
|
||||||
|
className="pointer-events-none absolute"
|
||||||
|
initial={{ opacity: 0 }}
|
||||||
|
whileInView={{
|
||||||
|
opacity: 1,
|
||||||
|
x: dimensions.width + 4,
|
||||||
|
y: dimensions.height + 4,
|
||||||
|
}}
|
||||||
|
style={{
|
||||||
|
rotate: -90,
|
||||||
|
}}
|
||||||
|
transition={{
|
||||||
|
opacity: { duration: 0.1, ease: "easeInOut" },
|
||||||
|
duration: 1,
|
||||||
|
ease: "easeInOut",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Pointer
|
||||||
|
className={cn("h-5 w-5 text-blue-500", pointerClassName)}
|
||||||
|
/>
|
||||||
|
</motion.div>
|
||||||
|
</motion.div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Pointer = ({ ...props }: React.SVGProps<SVGSVGElement>) => {
|
||||||
|
return (
|
||||||
|
<svg
|
||||||
|
stroke="currentColor"
|
||||||
|
fill="currentColor"
|
||||||
|
strokeWidth="1"
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
viewBox="0 0 16 16"
|
||||||
|
height="1em"
|
||||||
|
width="1em"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<path d="M14.082 2.182a.5.5 0 0 1 .103.557L8.528 15.467a.5.5 0 0 1-.917-.007L5.57 10.694.803 8.652a.5.5 0 0 1-.006-.916l12.728-5.657a.5.5 0 0 1 .556.103z"></path>
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
};
|
||||||
65
src/pages/home/HomeAgent.tsx
Normal file
65
src/pages/home/HomeAgent.tsx
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
import { H2, P } from '@/components/Texts'
|
||||||
|
import { Button } from '@/components/Button'
|
||||||
|
import { LayoutTextFlip } from '@/components/ui/layout-text-flip' // make sure this import path is correct
|
||||||
|
|
||||||
|
export function HomeAgent() {
|
||||||
|
return (
|
||||||
|
<div className="relative isolate overflow-hidden bg-white">
|
||||||
|
<div className="px-6 py-24 sm:py-32 lg:px-8">
|
||||||
|
<div className="mx-auto max-w-4xl text-center">
|
||||||
|
<H2>
|
||||||
|
Deploy your own{" "}
|
||||||
|
<span className="font-neuton text-left text-black font-medium text-7xl italic bg-clip-text bg-gradient-to-r from-blue-400 via-cyan-400 to-violet-400">
|
||||||
|
<LayoutTextFlip
|
||||||
|
text=""
|
||||||
|
words={[
|
||||||
|
"GPT-5",
|
||||||
|
"Claude 3.5",
|
||||||
|
"Gemini 1.5",
|
||||||
|
"Mistral 7B",
|
||||||
|
"Llama 3.1",
|
||||||
|
|
||||||
|
"AI Agents",
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</H2>
|
||||||
|
|
||||||
|
<P className="mx-auto mt-6 max-w-xl text-lg/8 text-pretty text-gray-600">
|
||||||
|
Mycelium delivers enterprise-grade AI agents with unmatched customizability and the fastest time to production — all in the agent platform designed for real business use cases.
|
||||||
|
</P>
|
||||||
|
|
||||||
|
<div className="mt-10 flex items-center justify-center gap-x-6">
|
||||||
|
<Button variant="solid" color="cyan" href="/signup">
|
||||||
|
Get started
|
||||||
|
</Button>
|
||||||
|
<a href="/agents" className="text-sm/6 font-semibold text-gray-900 hover:text-gray-600">
|
||||||
|
Learn more <span aria-hidden="true">→</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<svg
|
||||||
|
viewBox="0 0 1024 1024"
|
||||||
|
aria-hidden="true"
|
||||||
|
className="absolute top-1/2 left-1/2 -z-10 size-256 -translate-x-1/2 mask-[radial-gradient(closest-side,white,transparent)]"
|
||||||
|
>
|
||||||
|
<circle
|
||||||
|
r={512}
|
||||||
|
cx={512}
|
||||||
|
cy={512}
|
||||||
|
fill="url(#8d958450-c69f-4251-94bc-4e091a323369)"
|
||||||
|
fillOpacity="0.7"
|
||||||
|
/>
|
||||||
|
<defs>
|
||||||
|
<radialGradient id="8d958450-c69f-4251-94bc-4e091a323369" cx="50%" cy="50%" r="50%">
|
||||||
|
<stop offset="0%" stopColor="#60A5FA" /> {/* blue-400 */}
|
||||||
|
<stop offset="50%" stopColor="#06B6D4" /> {/* cyan-500 */}
|
||||||
|
<stop offset="100%" stopColor="#A78BFA" /> {/* violet-400 */}
|
||||||
|
</radialGradient>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -40,7 +40,7 @@ export function HomeCloud() {
|
|||||||
))}
|
))}
|
||||||
</ul>
|
</ul>
|
||||||
<div className="mt-10 flex">
|
<div className="mt-10 flex">
|
||||||
<a href="#" className="text-sm/6 font-semibold text-cyan-600 hover:text-cyan-500">
|
<a href="/cloud" className="text-sm/6 font-semibold text-cyan-600 hover:text-cyan-500">
|
||||||
Learn more
|
Learn more
|
||||||
<span aria-hidden="true"> →</span>
|
<span aria-hidden="true"> →</span>
|
||||||
</a>
|
</a>
|
||||||
|
|||||||
@@ -7,8 +7,8 @@ import { HomeHeroDark } from './HomeHeroDark'
|
|||||||
import { HomeAurora } from './HomeAurora'
|
import { HomeAurora } from './HomeAurora'
|
||||||
import { HomeMapSection } from './HomeMap'
|
import { HomeMapSection } from './HomeMap'
|
||||||
import { HomeFeatures } from './HomeFeatures'
|
import { HomeFeatures } from './HomeFeatures'
|
||||||
import { HalfGlobe } from '@/components/ui/HalfGlobe'
|
|
||||||
import { HomeCloud } from './HomeCloud'
|
import { HomeCloud } from './HomeCloud'
|
||||||
|
import { HomeAgent } from './HomeAgent'
|
||||||
|
|
||||||
export default function HomePage() {
|
export default function HomePage() {
|
||||||
return (
|
return (
|
||||||
@@ -29,6 +29,10 @@ export default function HomePage() {
|
|||||||
<HomeCloud />
|
<HomeCloud />
|
||||||
</AnimatedSection>
|
</AnimatedSection>
|
||||||
|
|
||||||
|
<AnimatedSection>
|
||||||
|
<HomeAgent />
|
||||||
|
</AnimatedSection>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user