forked from emre/www_projectmycelium_com
- Integrated next-themes for dark mode theming with default dark theme - Added new UI components (GridBlink, Spotlight) and enhanced world map with cyan glow effects - Updated homepage messaging to emphasize Mycelium as a living network with new audience imagery
126 lines
3.8 KiB
TypeScript
126 lines
3.8 KiB
TypeScript
"use client";
|
|
|
|
import { useRef } from "react";
|
|
import { motion } from "motion/react";
|
|
import DottedMap from "dotted-map";
|
|
|
|
interface MapProps {
|
|
dots?: Array<{
|
|
start: { lat: number; lng: number; label?: string };
|
|
end: { lat: number; lng: number; label?: string };
|
|
}>;
|
|
lineColor?: string;
|
|
}
|
|
|
|
export default function WorldMap({
|
|
dots = [],
|
|
lineColor = "#06b6d4", // cyan-500
|
|
}: MapProps) {
|
|
const svgRef = useRef<SVGSVGElement>(null);
|
|
|
|
// ✅ Force dark-dotted map theme
|
|
const map = new DottedMap({ height: 100, grid: "diagonal" });
|
|
const svgMap = map.getSVG({
|
|
radius: 0.22,
|
|
color: "#06b6d480", // cyan-500 at 50% opacity
|
|
shape: "circle",
|
|
backgroundColor: "#111111",
|
|
});
|
|
|
|
// ✅ Point projection stays the same
|
|
const projectPoint = (lat: number, lng: number) => {
|
|
const x = (lng + 180) * (800 / 360);
|
|
const y = (90 - lat) * (400 / 180);
|
|
return { x, y };
|
|
};
|
|
|
|
const createCurvedPath = (start: any, end: any) => {
|
|
const midX = (start.x + end.x) / 2;
|
|
const midY = Math.min(start.y, end.y) - 50;
|
|
return `M ${start.x} ${start.y} Q ${midX} ${midY} ${end.x} ${end.y}`;
|
|
};
|
|
|
|
return (
|
|
<div className="w-full aspect-[2/1] bg-[#111111] rounded-lg relative font-sans">
|
|
<img
|
|
src={`data:image/svg+xml;utf8,${encodeURIComponent(svgMap)}`}
|
|
className="h-full w-full pointer-events-none select-none opacity-[0.6]"
|
|
alt="world map"
|
|
draggable={false}
|
|
/>
|
|
|
|
{/* ✅ Lines + points */}
|
|
<svg
|
|
ref={svgRef}
|
|
viewBox="0 0 800 400"
|
|
className="w-full h-full absolute inset-0 pointer-events-none select-none"
|
|
>
|
|
{/* ✅ animated curved travel lines */}
|
|
{dots.map((dot, i) => {
|
|
const startPoint = projectPoint(dot.start.lat, dot.start.lng);
|
|
const endPoint = projectPoint(dot.end.lat, dot.end.lng);
|
|
|
|
return (
|
|
<motion.path
|
|
key={`path-${i}`}
|
|
d={createCurvedPath(startPoint, endPoint)}
|
|
fill="none"
|
|
stroke="url(#path-gradient)"
|
|
strokeWidth="1"
|
|
initial={{ pathLength: 0 }}
|
|
animate={{ pathLength: 1 }}
|
|
transition={{
|
|
duration: 1,
|
|
delay: 0.5 * i,
|
|
ease: "easeOut",
|
|
}}
|
|
/>
|
|
);
|
|
})}
|
|
|
|
{/* ✅ glowing path gradient */}
|
|
<defs>
|
|
<linearGradient id="path-gradient" x1="0%" y1="0%" x2="100%" y2="0%">
|
|
<stop offset="0%" stopColor="black" stopOpacity="0" />
|
|
<stop offset="5%" stopColor={lineColor} stopOpacity="1" />
|
|
<stop offset="95%" stopColor={lineColor} stopOpacity="1" />
|
|
<stop offset="100%" stopColor="black" stopOpacity="0" />
|
|
</linearGradient>
|
|
</defs>
|
|
|
|
{/* ✅ start & end points with pulsing cyan glow */}
|
|
{dots.map((dot, i) => {
|
|
const s = projectPoint(dot.start.lat, dot.start.lng);
|
|
const e = projectPoint(dot.end.lat, dot.end.lng);
|
|
|
|
return (
|
|
<g key={`points-${i}`}>
|
|
{[s, e].map((p, idx) => (
|
|
<g key={idx}>
|
|
<circle cx={p.x} cy={p.y} r="2" fill={lineColor} />
|
|
<circle cx={p.x} cy={p.y} r="2" fill={lineColor} opacity="0.5">
|
|
<animate
|
|
attributeName="r"
|
|
from="2"
|
|
to="7"
|
|
dur="1.4s"
|
|
repeatCount="indefinite"
|
|
/>
|
|
<animate
|
|
attributeName="opacity"
|
|
from="0.6"
|
|
to="0"
|
|
dur="1.4s"
|
|
repeatCount="indefinite"
|
|
/>
|
|
</circle>
|
|
</g>
|
|
))}
|
|
</g>
|
|
);
|
|
})}
|
|
</svg>
|
|
</div>
|
|
);
|
|
}
|