aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/src/components
diff options
context:
space:
mode:
Diffstat (limited to 'src/components')
-rw-r--r--src/components/FutureCard.tsx28
-rw-r--r--src/components/StatsCard.tsx26
-rw-r--r--src/components/ThreeBackground.tsx72
3 files changed, 126 insertions, 0 deletions
diff --git a/src/components/FutureCard.tsx b/src/components/FutureCard.tsx
new file mode 100644
index 0000000..5f8a95d
--- /dev/null
+++ b/src/components/FutureCard.tsx
@@ -0,0 +1,28 @@
+import React from 'react';
+import { motion } from 'framer-motion';
+import { Card } from '@nextui-org/react';
+
+interface FeatureCardProps {
+ icon: React.ReactNode;
+ title: string;
+ description: string;
+}
+
+export const FeatureCard: React.FC<FeatureCardProps> = ({ icon, title, description }) => {
+ return (
+ <motion.div
+ whileHover={{ scale: 1.05 }}
+ transition={{ type: "spring", stiffness: 300 }}
+ >
+ <Card className="p-6 bg-gradient-to-br from-gray-800/50 to-gray-900/50 border border-gray-700">
+ <div className="flex flex-col items-center text-center">
+ <div className="mb-4">
+ {icon}
+ </div>
+ <h3 className="text-xl font-bold mb-2 text-white">{title}</h3>
+ <p className="text-gray-400">{description}</p>
+ </div>
+ </Card>
+ </motion.div>
+ );
+};
diff --git a/src/components/StatsCard.tsx b/src/components/StatsCard.tsx
new file mode 100644
index 0000000..237f5dc
--- /dev/null
+++ b/src/components/StatsCard.tsx
@@ -0,0 +1,26 @@
+import React from 'react';
+import { motion } from 'framer-motion';
+import { Card } from '@nextui-org/react';
+
+interface StatsCardProps {
+ title: string;
+ value: string;
+}
+
+export const StatsCard: React.FC<StatsCardProps> = ({ title, value }) => {
+ return (
+ <motion.div
+ initial={{ opacity: 0, y: 20 }}
+ whileInView={{ opacity: 1, y: 0 }}
+ transition={{ duration: 0.5 }}
+ viewport={{ once: true }}
+ >
+ <Card className="p-6 bg-gradient-to-br from-gray-800/50 to-gray-900/50 border border-gray-700">
+ <div className="text-center">
+ <h3 className="text-2xl font-bold text-white mb-2">{value}</h3>
+ <p className="text-gray-400">{title}</p>
+ </div>
+ </Card>
+ </motion.div>
+ );
+};
diff --git a/src/components/ThreeBackground.tsx b/src/components/ThreeBackground.tsx
new file mode 100644
index 0000000..1a6337c
--- /dev/null
+++ b/src/components/ThreeBackground.tsx
@@ -0,0 +1,72 @@
+import React, { useEffect, useRef } from 'react';
+import * as THREE from 'three';
+
+export const ThreeBackground: React.FC = () => {
+ const containerRef = useRef<HTMLDivElement>(null);
+
+ useEffect(() => {
+ if (!containerRef.current) return;
+
+ const scene = new THREE.Scene();
+ const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
+ const renderer = new THREE.WebGLRenderer({ alpha: true });
+
+ renderer.setSize(window.innerWidth, window.innerHeight);
+ containerRef.current.appendChild(renderer.domElement);
+
+ // Create particles
+ const geometry = new THREE.BufferGeometry();
+ const vertices = [];
+
+ for (let i = 0; i < 5000; i++) {
+ vertices.push(
+ THREE.MathUtils.randFloatSpread(2000), // x
+ THREE.MathUtils.randFloatSpread(2000), // y
+ THREE.MathUtils.randFloatSpread(2000) // z
+ );
+ }
+
+ geometry.setAttribute('position', new THREE.Float32BufferAttribute(vertices, 3));
+
+ const material = new THREE.PointsMaterial({
+ color: 0x88ccff,
+ size: 2,
+ transparent: true,
+ opacity: 0.5
+ });
+
+ const particles = new THREE.Points(geometry, material);
+ scene.add(particles);
+
+ camera.position.z = 1000;
+
+ const animate = () => {
+ requestAnimationFrame(animate);
+ particles.rotation.x += 0.0001;
+ particles.rotation.y += 0.0001;
+ renderer.render(scene, camera);
+ };
+
+ animate();
+
+ const handleResize = () => {
+ camera.aspect = window.innerWidth / window.innerHeight;
+ camera.updateProjectionMatrix();
+ renderer.setSize(window.innerWidth, window.innerHeight);
+ };
+
+ window.addEventListener('resize', handleResize);
+
+ return () => {
+ window.removeEventListener('resize', handleResize);
+ containerRef.current?.removeChild(renderer.domElement);
+ };
+ }, []);
+
+ return (
+ <div
+ ref={containerRef}
+ className="fixed top-0 left-0 w-full h-full z-0"
+ />
+ );
+};