Compare commits
138 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| f5bcc6e40d | |||
| 03df96f083 | |||
| 6518d28772 | |||
| 5013b53765 | |||
| 06dcaf500d | |||
| 7cfd7b9740 | |||
| 9ba60256e2 | |||
| cd4d18a801 | |||
| 39fbb62e9d | |||
| 10e3fad348 | |||
| 101737d93b | |||
| c38c3287c1 | |||
| 3c8e21634d | |||
| 39736cb05d | |||
| bcd3ed80c7 | |||
| b958875873 | |||
| 4a760304bf | |||
| f51f6aac7f | |||
| 1f88228dbd | |||
| 7dfb6f3f95 | |||
| 58697ead0e | |||
| 037ef82c3d | |||
| fe948f40d8 | |||
| 3df05324f8 | |||
| aa174c3fba | |||
| 25fef0aec6 | |||
| b55725bda8 | |||
| ca9ddc0a90 | |||
| aa349f3d83 | |||
| a53e0030bb | |||
| 5f850bac37 | |||
| 2c9735cce2 | |||
| 521faa156b | |||
| 4966363406 | |||
| 10f33044ae | |||
| 39c2ca6ef4 | |||
| a5fc9943c4 | |||
| c1e8d6c8d5 | |||
| 0c037dd603 | |||
| be20bbb4dc | |||
| aade90a21f | |||
| 1286dc1aa5 | |||
| ca5ff42d8f | |||
| 5c2f7f374e | |||
| 2ccea76e4f | |||
| 02dad55fe9 | |||
| bfa4a8db78 | |||
| 14398cadd7 | |||
| 6c1bc0b01d | |||
| ea2ee7f715 | |||
| e8b2a950ad | |||
| e4e3ac93f3 | |||
| 8e9ebbfd20 | |||
| 5309779be9 | |||
| d114bef4d0 | |||
| 9102f58624 | |||
| bab04bed97 | |||
| c9b303b81a | |||
| a6d3b35969 | |||
| af2a39ee2a | |||
| ed81a59c66 | |||
| 2d7f08f1f9 | |||
| 810ec6d9a3 | |||
| 094645cc72 | |||
| 957f27d3f4 | |||
| bb604433c5 | |||
| 98927d5c65 | |||
| 278ac5189f | |||
| 8c13071105 | |||
| d2c9774934 | |||
| a9c442f17a | |||
| 49b118c1eb | |||
| 7f361fa741 | |||
| 561a19bc2e | |||
| 9604c47d9f | |||
| 869ce33d0b | |||
| 8e74cf1994 | |||
| fc8ac045cd | |||
| 2c5d2a4959 | |||
| 1d06ae95a3 | |||
| 32b3c9ef3f | |||
| f808026c6d | |||
| e2d3bbb4d0 | |||
| d16d1f0b26 | |||
| c876b1fd20 | |||
| 7e3f6595c7 | |||
| 936394f078 | |||
| 9185617dea | |||
| 64cadcb367 | |||
| 7962fb0ad5 | |||
| ca488ea512 | |||
| 326ea116f9 | |||
| 39aabdb7b7 | |||
| d1d6ebe559 | |||
| 2102e120b8 | |||
| 93e306122c | |||
| af3b5b15f1 | |||
| 29b3444668 | |||
| c1d17c5dec | |||
| 94567ac331 | |||
| e7460a8877 | |||
| cc961688fb | |||
| 69d3b51d51 | |||
| abf61bbeb1 | |||
| a1b6e7a70b | |||
| f0ed8b9744 | |||
| b6e0ce95e4 | |||
| 3de2f401cc | |||
| 583fd7d255 | |||
| 3cbb15e331 | |||
| dc29c790ab | |||
| 4e7e90f543 | |||
| 7dc92b519f | |||
| f253534b66 | |||
| bd2a303a19 | |||
| 0d1bb2cd3f | |||
| 6610d14553 | |||
| 0c09a6ce21 | |||
| 8dc9577019 | |||
| 9cf8e53beb | |||
| ae8604d266 | |||
| 2cd96e4e08 | |||
| 5888e66d4f | |||
| 8460ad5c63 | |||
| 13d9c450ac | |||
| c07adff29a | |||
| 788afa1f30 | |||
| 6294dc789d | |||
| 888aeff34a | |||
| 8bbf752b4d | |||
| 0aca3c2d2b | |||
| 700bc27fa7 | |||
| 27007b9bc5 | |||
| c94eeace0c | |||
| 9524e66ac2 | |||
| b19d8ebc55 | |||
| e7c7f72b82 | |||
| f1d7310e68 |
268
src/app/gym-3d/page.tsx
Normal file
268
src/app/gym-3d/page.tsx
Normal file
@@ -0,0 +1,268 @@
|
||||
"use client";
|
||||
|
||||
import { useEffect, useRef, useState } from "react";
|
||||
import * as THREE from "three";
|
||||
import NavbarLayoutFloatingOverlay from '@/components/navbar/NavbarLayoutFloatingOverlay/NavbarLayoutFloatingOverlay';
|
||||
import { ContactSplitForm } from '@/components/sections/contact/ContactSplitForm';
|
||||
import { ThemeProvider } from "@/providers/themeProvider/ThemeProvider";
|
||||
import { Phone } from "lucide-react";
|
||||
|
||||
export default function Gym3DPage() {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const sceneRef = useRef<THREE.Scene | null>(null);
|
||||
const cameraRef = useRef<THREE.PerspectiveCamera | null>(null);
|
||||
const rendererRef = useRef<THREE.WebGLRenderer | null>(null);
|
||||
const [isLoading, setIsLoading] = useState(true);
|
||||
|
||||
useEffect(() => {
|
||||
if (!containerRef.current) return;
|
||||
|
||||
// Scene setup
|
||||
const scene = new THREE.Scene();
|
||||
scene.background = new THREE.Color(0x0a0a0a);
|
||||
sceneRef.current = scene;
|
||||
|
||||
// Camera setup
|
||||
const camera = new THREE.PerspectiveCamera(
|
||||
75,
|
||||
containerRef.current.clientWidth / containerRef.current.clientHeight,
|
||||
0.1,
|
||||
1000
|
||||
);
|
||||
camera.position.set(0, 10, 20);
|
||||
camera.lookAt(0, 5, 0);
|
||||
cameraRef.current = camera;
|
||||
|
||||
// Renderer setup
|
||||
const renderer = new THREE.WebGLRenderer({ antialias: true });
|
||||
renderer.setSize(containerRef.current.clientWidth, containerRef.current.clientHeight);
|
||||
renderer.shadowMap.enabled = true;
|
||||
containerRef.current.appendChild(renderer.domElement);
|
||||
rendererRef.current = renderer;
|
||||
|
||||
// Lighting
|
||||
const ambientLight = new THREE.AmbientLight(0xffffff, 0.6);
|
||||
scene.add(ambientLight);
|
||||
|
||||
const directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);
|
||||
directionalLight.position.set(10, 15, 10);
|
||||
directionalLight.castShadow = true;
|
||||
directionalLight.shadow.mapSize.width = 2048;
|
||||
directionalLight.shadow.mapSize.height = 2048;
|
||||
scene.add(directionalLight);
|
||||
|
||||
// Ground/Floor
|
||||
const groundGeometry = new THREE.PlaneGeometry(40, 40);
|
||||
const groundMaterial = new THREE.MeshStandardMaterial({ color: 0x1a1a1a });
|
||||
const ground = new THREE.Mesh(groundGeometry, groundMaterial);
|
||||
ground.rotation.x = -Math.PI / 2;
|
||||
ground.receiveShadow = true;
|
||||
scene.add(ground);
|
||||
|
||||
// Gym Equipment - Power Rack
|
||||
const rackGeometry = new THREE.BoxGeometry(2, 3, 1.5);
|
||||
const rackMaterial = new THREE.MeshStandardMaterial({ color: 0x333333, metalness: 0.8 });
|
||||
const rack = new THREE.Mesh(rackGeometry, rackMaterial);
|
||||
rack.position.set(-8, 1.5, 0);
|
||||
rack.castShadow = true;
|
||||
scene.add(rack);
|
||||
|
||||
// Barbell
|
||||
const barbellGeometry = new THREE.CylinderGeometry(0.02, 0.02, 2.2, 16);
|
||||
const barbellMaterial = new THREE.MeshStandardMaterial({ color: 0xffff00, metalness: 1 });
|
||||
const barbell = new THREE.Mesh(barbellGeometry, barbellMaterial);
|
||||
barbell.position.set(-8, 1.8, 0);
|
||||
barbell.rotation.z = Math.PI / 2;
|
||||
barbell.castShadow = true;
|
||||
scene.add(barbell);
|
||||
|
||||
// Weight plates on barbell
|
||||
for (let i = 0; i < 4; i++) {
|
||||
const plateGeometry = new THREE.CylinderGeometry(0.5, 0.5, 0.05, 32);
|
||||
const plateMaterial = new THREE.MeshStandardMaterial({ color: 0xff0000, metalness: 0.6 });
|
||||
const plate = new THREE.Mesh(plateGeometry, plateMaterial);
|
||||
plate.position.set(-8 + (i - 1.5) * 0.6, 1.8, 0);
|
||||
plate.castShadow = true;
|
||||
scene.add(plate);
|
||||
}
|
||||
|
||||
// Dumbbells rack
|
||||
for (let i = 0; i < 5; i++) {
|
||||
const dumbellGeometry = new THREE.CylinderGeometry(0.15, 0.15, 0.5, 16);
|
||||
const dumbellMaterial = new THREE.MeshStandardMaterial({ color: 0xffaa00, metalness: 0.7 });
|
||||
const dumbell = new THREE.Mesh(dumbellGeometry, dumbellMaterial);
|
||||
dumbell.position.set(0 + i * 1.5, 0.5, -10);
|
||||
dumbell.castShadow = true;
|
||||
scene.add(dumbell);
|
||||
}
|
||||
|
||||
// Gymnastics Rig
|
||||
const rigGeometry = new THREE.BoxGeometry(0.1, 0.1, 5);
|
||||
const rigMaterial = new THREE.MeshStandardMaterial({ color: 0x333333, metalness: 0.8 });
|
||||
const rig = new THREE.Mesh(rigGeometry, rigMaterial);
|
||||
rig.position.set(0, 8, 0);
|
||||
rig.castShadow = true;
|
||||
scene.add(rig);
|
||||
|
||||
// Olympic Rings
|
||||
const ringRadius = 0.3;
|
||||
const ringPositions = [-1.2, -0.4, 0.4, 1.2];
|
||||
ringPositions.forEach((x) => {
|
||||
const torusGeometry = new THREE.TorusGeometry(ringRadius, 0.08, 16, 100);
|
||||
const torusMaterial = new THREE.MeshStandardMaterial({ color: 0xffffff, metalness: 0.9 });
|
||||
const ring = new THREE.Mesh(torusGeometry, torusMaterial);
|
||||
ring.position.set(x, 7, 0);
|
||||
ring.rotation.x = Math.PI / 5;
|
||||
ring.castShadow = true;
|
||||
scene.add(ring);
|
||||
});
|
||||
|
||||
// Rowing Machine
|
||||
const rowingGeometry = new THREE.BoxGeometry(0.8, 1, 2.5);
|
||||
const rowingMaterial = new THREE.MeshStandardMaterial({ color: 0x222222, metalness: 0.6 });
|
||||
const rowingMachine = new THREE.Mesh(rowingGeometry, rowingMaterial);
|
||||
rowingMachine.position.set(8, 0.8, -5);
|
||||
rowingMachine.castShadow = true;
|
||||
scene.add(rowingMachine);
|
||||
|
||||
// Treadmill
|
||||
const treadmillGeometry = new THREE.BoxGeometry(1, 1.2, 2);
|
||||
const treadmillMaterial = new THREE.MeshStandardMaterial({ color: 0x1a1a1a, metalness: 0.5 });
|
||||
const treadmill = new THREE.Mesh(treadmillGeometry, treadmillMaterial);
|
||||
treadmill.position.set(8, 0.8, 0);
|
||||
treadmill.castShadow = true;
|
||||
scene.add(treadmill);
|
||||
|
||||
// Kettlebells
|
||||
for (let i = 0; i < 3; i++) {
|
||||
const kettlebellGeometry = new THREE.SphereGeometry(0.4, 16, 16);
|
||||
const kettlebellMaterial = new THREE.MeshStandardMaterial({ color: 0x333333, metalness: 0.7 });
|
||||
const kettlebell = new THREE.Mesh(kettlebellGeometry, kettlebellMaterial);
|
||||
kettlebell.position.set(-10, 0.6, -5 + i * 2);
|
||||
kettlebell.castShadow = true;
|
||||
scene.add(kettlebell);
|
||||
}
|
||||
|
||||
// Battle Ropes anchor
|
||||
const ropeAnchorGeometry = new THREE.BoxGeometry(0.2, 3, 0.2);
|
||||
const ropeAnchorMaterial = new THREE.MeshStandardMaterial({ color: 0x444444, metalness: 0.8 });
|
||||
const ropeAnchor = new THREE.Mesh(ropeAnchorGeometry, ropeAnchorMaterial);
|
||||
ropeAnchor.position.set(0, 2, 10);
|
||||
ropeAnchor.castShadow = true;
|
||||
scene.add(ropeAnchor);
|
||||
|
||||
// Walls
|
||||
const wallGeometry = new THREE.BoxGeometry(40, 5, 0.5);
|
||||
const wallMaterial = new THREE.MeshStandardMaterial({ color: 0x0f0f0f });
|
||||
|
||||
const wallBack = new THREE.Mesh(wallGeometry, wallMaterial);
|
||||
wallBack.position.set(0, 2.5, -20);
|
||||
wallBack.receiveShadow = true;
|
||||
scene.add(wallBack);
|
||||
|
||||
// Loading complete
|
||||
setIsLoading(false);
|
||||
|
||||
// Animation loop
|
||||
const animate = () => {
|
||||
requestAnimationFrame(animate);
|
||||
|
||||
// Rotate dumbbells slightly
|
||||
scene.children.forEach((child) => {
|
||||
if (child instanceof THREE.Mesh && child.geometry instanceof THREE.CylinderGeometry) {
|
||||
if (child.position.z === -10) {
|
||||
child.rotation.x += 0.01;
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// Rotate camera slightly
|
||||
const time = Date.now() * 0.0005;
|
||||
camera.position.x = Math.sin(time) * 25;
|
||||
camera.position.z = Math.cos(time) * 25;
|
||||
camera.lookAt(0, 5, 0);
|
||||
|
||||
renderer.render(scene, camera);
|
||||
};
|
||||
animate();
|
||||
|
||||
// Handle window resize
|
||||
const handleResize = () => {
|
||||
if (!containerRef.current) return;
|
||||
const width = containerRef.current.clientWidth;
|
||||
const height = containerRef.current.clientHeight;
|
||||
camera.aspect = width / height;
|
||||
camera.updateProjectionMatrix();
|
||||
renderer.setSize(width, height);
|
||||
};
|
||||
window.addEventListener("resize", handleResize);
|
||||
|
||||
return () => {
|
||||
window.removeEventListener("resize", handleResize);
|
||||
containerRef.current?.removeChild(renderer.domElement);
|
||||
};
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<ThemeProvider
|
||||
defaultButtonVariant="text-shift"
|
||||
defaultTextAnimation="reveal-blur"
|
||||
borderRadius="pill"
|
||||
contentWidth="medium"
|
||||
sizing="largeSmallSizeMediumTitles"
|
||||
background="circleGradient"
|
||||
cardStyle="gradient-bordered"
|
||||
primaryButtonStyle="primary-glow"
|
||||
secondaryButtonStyle="radial-glow"
|
||||
headingFontWeight="normal"
|
||||
>
|
||||
<div id="nav" data-section="nav">
|
||||
<NavbarLayoutFloatingOverlay
|
||||
brandName="Iron Pulse"
|
||||
navItems={[
|
||||
{ name: "WOD", id: "wod" },
|
||||
{ name: "Pricing", id: "pricing" },
|
||||
{ name: "Gallery", id: "gallery" },
|
||||
{ name: "Testimonials", id: "testimonials" },
|
||||
{ name: "Home", id: "/" }
|
||||
]}
|
||||
button={{ text: "Start Your Trial", href: "contact" }}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="w-full h-screen bg-background" ref={containerRef}>
|
||||
{isLoading && (
|
||||
<div className="absolute inset-0 flex items-center justify-center bg-background/80 backdrop-blur-sm">
|
||||
<div className="text-center">
|
||||
<div className="inline-block animate-spin rounded-full h-12 w-12 border-t-2 border-primary-cta"></div>
|
||||
<p className="text-foreground mt-4">Loading 3D Gym Tour...</p>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div id="contact" data-section="contact" className="py-20">
|
||||
<ContactSplitForm
|
||||
title="Interested in joining Iron Pulse?"
|
||||
description="Fill out the form below and our coaching team will reach out within 24 hours to schedule your trial session."
|
||||
inputs={[
|
||||
{ name: "name", type: "text", placeholder: "Full Name", required: true },
|
||||
{ name: "email", type: "email", placeholder: "Email Address", required: true }
|
||||
]}
|
||||
textarea={{
|
||||
name: "message", placeholder: "Tell us about your fitness goals and experience level...", rows: 5,
|
||||
required: false
|
||||
}}
|
||||
useInvertedBackground={false}
|
||||
mediaAnimation="slide-up"
|
||||
buttonText="Schedule Trial"
|
||||
onSubmit={(data) => {
|
||||
console.log("Form submitted:", data);
|
||||
alert("Thank you for reaching out! We'll contact you soon.");
|
||||
}}
|
||||
ariaLabel="Contact form section"
|
||||
/>
|
||||
</div>
|
||||
</ThemeProvider>
|
||||
);
|
||||
}
|
||||
@@ -1,54 +1,22 @@
|
||||
import type { Metadata } from "next";
|
||||
import { Halant } from "next/font/google";
|
||||
import { Inter } from "next/font/google";
|
||||
import { Manrope } from "next/font/google";
|
||||
import { DM_Sans } from "next/font/google";
|
||||
import "./globals.css";
|
||||
import { ServiceWrapper } from "@/components/ServiceWrapper";
|
||||
import Tag from "@/tag/Tag";
|
||||
|
||||
const halant = Halant({
|
||||
variable: "--font-halant", subsets: ["latin"],
|
||||
weight: ["300", "400", "500", "600", "700"],
|
||||
});
|
||||
|
||||
const inter = Inter({
|
||||
variable: "--font-inter", subsets: ["latin"],
|
||||
});
|
||||
|
||||
const manrope = Manrope({
|
||||
variable: "--font-manrope", subsets: ["latin"],
|
||||
});
|
||||
|
||||
const dmSans = DM_Sans({
|
||||
variable: "--font-dm-sans", subsets: ["latin"],
|
||||
});
|
||||
const inter = Inter({ subsets: ["latin"] });
|
||||
|
||||
export const metadata: Metadata = {
|
||||
title: "Iron Pulse CrossFit | Elite Coaching & High-Intensity Training", description: "Transform your strength at Iron Pulse CrossFit. Elite coaching, daily WODs, and a community of athletes pushing limits. Start your trial today.", keywords: "CrossFit gym, high-intensity training, coaching, fitness community, WOD, elite athletes", openGraph: {
|
||||
title: "Iron Pulse CrossFit | Elite Coaching & High-Intensity Training", description: "Transform your strength at Iron Pulse CrossFit. Elite coaching, daily WODs, and a community of athletes pushing limits.", type: "website", siteName: "Iron Pulse"},
|
||||
twitter: {
|
||||
card: "summary_large_image", title: "Iron Pulse CrossFit | Elite Coaching & High-Intensity Training", description: "Transform your strength at Iron Pulse CrossFit. Elite coaching, daily WODs, and a community of athletes pushing limits."},
|
||||
robots: {
|
||||
index: true,
|
||||
follow: true,
|
||||
},
|
||||
};
|
||||
title: "Iron Pulse - CrossFit Gym", description: "Elite CrossFit training and coaching"};
|
||||
|
||||
export default function RootLayout({
|
||||
children,
|
||||
}: Readonly<{
|
||||
}: {
|
||||
children: React.ReactNode;
|
||||
}>) {
|
||||
}) {
|
||||
return (
|
||||
<html lang="en" suppressHydrationWarning>
|
||||
<ServiceWrapper>
|
||||
<body
|
||||
className={`${halant.variable} ${inter.variable} ${manrope.variable} ${dmSans.variable} antialiased`}
|
||||
>
|
||||
<Tag />
|
||||
{children}
|
||||
|
||||
<html lang="en">
|
||||
<body className={inter.className}>
|
||||
{children}
|
||||
|
||||
<script
|
||||
dangerouslySetInnerHTML={{
|
||||
__html: `
|
||||
@@ -1416,7 +1384,6 @@ export default function RootLayout({
|
||||
}}
|
||||
/>
|
||||
</body>
|
||||
</ServiceWrapper>
|
||||
</html>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -33,7 +33,7 @@ export default function LandingPage() {
|
||||
{ name: "Pricing", id: "pricing" },
|
||||
{ name: "Gallery", id: "gallery" },
|
||||
{ name: "Testimonials", id: "testimonials" },
|
||||
{ name: "Contact", id: "contact" }
|
||||
{ name: "3D Gym", id: "/gym-3d" }
|
||||
]}
|
||||
button={{ text: "Start Your Trial", href: "contact" }}
|
||||
/>
|
||||
|
||||
Reference in New Issue
Block a user