Replace puffin-logo.svg with puffinOffset.webp in both the favicon link and JSON-LD structured data to use WebP image format instead of SVG.
317 lines
13 KiB
TypeScript
317 lines
13 KiB
TypeScript
import React, { useState, useEffect } from 'react';
|
|
import { Menu, X } from 'lucide-react';
|
|
import { motion, AnimatePresence } from 'framer-motion';
|
|
import { Home } from './components/Home';
|
|
import { YachtSearch } from './components/YachtSearch';
|
|
import { TripCalculator } from './components/TripCalculator';
|
|
import { HowItWorks } from './components/HowItWorks';
|
|
import { About } from './components/About';
|
|
import { Contact } from './components/Contact';
|
|
import { OffsetOrder } from './components/OffsetOrder';
|
|
import { getVesselData } from './api/aisClient';
|
|
import { calculateTripCarbon } from './utils/carbonCalculator';
|
|
import { analytics } from './utils/analytics';
|
|
import type { VesselData, CarbonCalculation, CalculatorType } from './types';
|
|
|
|
const sampleVessel: VesselData = {
|
|
imo: "1234567",
|
|
vesselName: "Sample Yacht",
|
|
type: "Yacht",
|
|
length: 50,
|
|
width: 9,
|
|
estimatedEnginePower: 2250
|
|
};
|
|
|
|
function App() {
|
|
const [loading, setLoading] = useState(false);
|
|
const [error, setError] = useState<string | null>(null);
|
|
const [vesselData, setVesselData] = useState<VesselData | null>(null);
|
|
const [currentPage, setCurrentPage] = useState<'home' | 'calculator' | 'how-it-works' | 'about' | 'contact'>('home');
|
|
const [showOffsetOrder, setShowOffsetOrder] = useState(false);
|
|
const [offsetTons, setOffsetTons] = useState(0);
|
|
const [monetaryAmount, setMonetaryAmount] = useState<number | undefined>();
|
|
const [calculatorType, setCalculatorType] = useState<CalculatorType>('trip');
|
|
const [mobileMenuOpen, setMobileMenuOpen] = useState(false);
|
|
|
|
useEffect(() => {
|
|
analytics.pageView(window.location.pathname);
|
|
}, [currentPage]);
|
|
|
|
const handleSearch = async (imo: string) => {
|
|
setLoading(true);
|
|
setError(null);
|
|
setVesselData(null);
|
|
|
|
try {
|
|
const vessel = await getVesselData(imo);
|
|
setVesselData(vessel);
|
|
} catch (err) {
|
|
setError('Unable to fetch vessel data. Please verify the IMO number and try again.');
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
};
|
|
|
|
const handleOffsetClick = (tons: number, monetaryAmount?: number) => {
|
|
setOffsetTons(tons);
|
|
setMonetaryAmount(monetaryAmount);
|
|
setShowOffsetOrder(true);
|
|
window.scrollTo({ top: 0, behavior: 'smooth' });
|
|
};
|
|
|
|
const handleNavigate = (page: 'home' | 'calculator' | 'how-it-works' | 'about' | 'contact') => {
|
|
setCurrentPage(page);
|
|
setMobileMenuOpen(false);
|
|
window.scrollTo({ top: 0, behavior: 'smooth' });
|
|
};
|
|
|
|
const renderPage = () => {
|
|
if (currentPage === 'calculator' && showOffsetOrder) {
|
|
return (
|
|
<div className="flex justify-center px-4 sm:px-0">
|
|
<OffsetOrder
|
|
tons={offsetTons}
|
|
monetaryAmount={monetaryAmount}
|
|
onBack={() => {
|
|
setShowOffsetOrder(false);
|
|
window.scrollTo({ top: 0, behavior: 'smooth' });
|
|
}}
|
|
calculatorType={calculatorType}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
switch (currentPage) {
|
|
case 'calculator':
|
|
return (
|
|
<div className="flex flex-col items-center">
|
|
<div className="text-center mb-12 max-w-4xl">
|
|
<h2 className="text-3xl sm:text-4xl font-bold text-gray-900 mb-4">
|
|
Calculate & Offset Your Yacht's Carbon Footprint
|
|
</h2>
|
|
<p className="text-base sm:text-lg text-gray-600">
|
|
Use the calculator below to estimate your carbon footprint and explore offsetting options through our verified projects.
|
|
</p>
|
|
</div>
|
|
|
|
<div className="flex flex-col items-center w-full max-w-4xl space-y-8">
|
|
<TripCalculator
|
|
vesselData={sampleVessel}
|
|
onOffsetClick={handleOffsetClick}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
case 'how-it-works':
|
|
return <HowItWorks onNavigate={handleNavigate} />;
|
|
case 'about':
|
|
return <About onNavigate={handleNavigate} />;
|
|
case 'contact':
|
|
return <Contact />;
|
|
default:
|
|
return <Home onNavigate={handleNavigate} />;
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className="min-h-screen bg-gradient-to-br from-slate-50 via-blue-50 to-cyan-50 wave-pattern">
|
|
<header className="glass-nav shadow-luxury relative z-50 sticky top-0">
|
|
<div className="max-w-7xl mx-auto px-4 py-3 sm:px-6 lg:px-8">
|
|
<div className="flex items-center justify-between">
|
|
<motion.div
|
|
className="flex items-center space-x-3 cursor-pointer group"
|
|
onClick={() => handleNavigate('home')}
|
|
whileHover={{ scale: 1.02 }}
|
|
transition={{ type: "spring", stiffness: 400, damping: 17 }}
|
|
>
|
|
<motion.img
|
|
src="/puffinOffset.webp"
|
|
alt="Puffin Offset Logo"
|
|
className="h-10 w-auto transition-transform duration-300 group-hover:scale-110"
|
|
initial={{ opacity: 0, rotate: -10 }}
|
|
animate={{ opacity: 1, rotate: 0 }}
|
|
transition={{ duration: 0.6, delay: 0.2 }}
|
|
/>
|
|
<motion.h1
|
|
className="text-xl font-bold heading-luxury"
|
|
initial={{ opacity: 0, x: -20 }}
|
|
animate={{ opacity: 1, x: 0 }}
|
|
transition={{ duration: 0.6, delay: 0.4 }}
|
|
>
|
|
Puffin Offset
|
|
</motion.h1>
|
|
</motion.div>
|
|
|
|
{/* Mobile menu button */}
|
|
<button
|
|
className="sm:hidden p-2 rounded-md text-gray-600 hover:text-gray-900"
|
|
onClick={() => setMobileMenuOpen(!mobileMenuOpen)}
|
|
>
|
|
{mobileMenuOpen ? <X size={24} /> : <Menu size={24} />}
|
|
</button>
|
|
|
|
{/* Desktop navigation */}
|
|
<nav className="hidden sm:flex space-x-2">
|
|
{['calculator', 'how-it-works', 'about', 'contact'].map((page, index) => (
|
|
<motion.button
|
|
key={page}
|
|
onClick={() => handleNavigate(page as any)}
|
|
className={`px-4 py-2 rounded-xl font-medium transition-all duration-300 ${
|
|
currentPage === page
|
|
? 'bg-gradient-to-r from-blue-600 to-blue-700 text-white shadow-lg'
|
|
: 'text-slate-600 hover:text-slate-900 hover:bg-white/60'
|
|
}`}
|
|
whileHover={{ scale: 1.05 }}
|
|
whileTap={{ scale: 0.95 }}
|
|
initial={{ opacity: 0, y: -10 }}
|
|
animate={{ opacity: 1, y: 0 }}
|
|
transition={{ duration: 0.4, delay: 0.6 + index * 0.1 }}
|
|
>
|
|
{page === 'how-it-works' ? 'How it Works' :
|
|
page.charAt(0).toUpperCase() + page.slice(1)}
|
|
</motion.button>
|
|
))}
|
|
</nav>
|
|
</div>
|
|
|
|
{/* Mobile navigation */}
|
|
{mobileMenuOpen && (
|
|
<nav className="sm:hidden mt-4 pb-2 space-y-2">
|
|
<button
|
|
onClick={() => handleNavigate('calculator')}
|
|
className={`block w-full text-left px-4 py-2 rounded-lg ${
|
|
currentPage === 'calculator'
|
|
? 'bg-blue-50 text-blue-600 font-semibold'
|
|
: 'text-gray-600 hover:bg-gray-50'
|
|
}`}
|
|
>
|
|
Calculator
|
|
</button>
|
|
<button
|
|
onClick={() => handleNavigate('how-it-works')}
|
|
className={`block w-full text-left px-4 py-2 rounded-lg ${
|
|
currentPage === 'how-it-works'
|
|
? 'bg-blue-50 text-blue-600 font-semibold'
|
|
: 'text-gray-600 hover:bg-gray-50'
|
|
}`}
|
|
>
|
|
How it Works
|
|
</button>
|
|
<button
|
|
onClick={() => handleNavigate('about')}
|
|
className={`block w-full text-left px-4 py-2 rounded-lg ${
|
|
currentPage === 'about'
|
|
? 'bg-blue-50 text-blue-600 font-semibold'
|
|
: 'text-gray-600 hover:bg-gray-50'
|
|
}`}
|
|
>
|
|
About
|
|
</button>
|
|
<button
|
|
onClick={() => handleNavigate('contact')}
|
|
className={`block w-full text-left px-4 py-2 rounded-lg ${
|
|
currentPage === 'contact'
|
|
? 'bg-blue-50 text-blue-600 font-semibold'
|
|
: 'text-gray-600 hover:bg-gray-50'
|
|
}`}
|
|
>
|
|
Contact
|
|
</button>
|
|
</nav>
|
|
)}
|
|
</div>
|
|
</header>
|
|
|
|
<main className="max-w-[1600px] mx-auto py-8 sm:py-12 px-4 sm:px-6 lg:px-8 overflow-hidden">
|
|
<AnimatePresence mode="wait">
|
|
<motion.div
|
|
key={currentPage + (showOffsetOrder ? '-offset' : '')}
|
|
initial={{ opacity: 0, y: 20 }}
|
|
animate={{ opacity: 1, y: 0 }}
|
|
exit={{ opacity: 0, y: -20 }}
|
|
transition={{
|
|
duration: 0.4,
|
|
ease: [0.25, 0.1, 0.25, 1.0]
|
|
}}
|
|
>
|
|
{renderPage()}
|
|
</motion.div>
|
|
</AnimatePresence>
|
|
</main>
|
|
|
|
<footer className="bg-gradient-to-r from-slate-900 via-blue-900 to-slate-900 mt-16 relative overflow-hidden">
|
|
<div className="absolute inset-0 bg-[url('data:image/svg+xml,%3csvg width='60' height='60' viewBox='0 0 60 60' xmlns='http://www.w3.org/2000/svg'%3e%3cg fill='none' fill-rule='evenodd'%3e%3cg fill='%23ffffff' fill-opacity='0.03'%3e%3cpath d='m36 34v-4h-2v4h-4v2h4v4h2v-4h4v-2h-4zm0-30V0h-2v4h-4v2h4v4h2V6h4V4h-4zM6 34v-4H4v4H0v2h4v4h2v-4h4v-2H6zM6 4V0H4v4H0v2h4v4h2V6h4V4H6z'/%3e%3c/g%3e%3c/g%3e%3c/svg%3e')] opacity-20"></div>
|
|
<div className="relative max-w-7xl mx-auto px-4 py-12 sm:px-6 lg:px-8">
|
|
<div className="grid grid-cols-1 md:grid-cols-3 gap-8 mb-8">
|
|
<motion.div
|
|
initial={{ opacity: 0, y: 20 }}
|
|
whileInView={{ opacity: 1, y: 0 }}
|
|
transition={{ duration: 0.6 }}
|
|
viewport={{ once: true }}
|
|
>
|
|
<div className="flex items-center space-x-3 mb-4">
|
|
<img
|
|
src="/puffinOffset.webp"
|
|
alt="Puffin Offset Logo"
|
|
className="h-8 w-auto"
|
|
/>
|
|
<h3 className="text-xl font-bold text-white">Puffin Offset</h3>
|
|
</div>
|
|
<p className="text-slate-300 leading-relaxed">
|
|
The world's most exclusive carbon offsetting platform for superyacht owners and operators.
|
|
</p>
|
|
</motion.div>
|
|
|
|
<motion.div
|
|
initial={{ opacity: 0, y: 20 }}
|
|
whileInView={{ opacity: 1, y: 0 }}
|
|
transition={{ duration: 0.6, delay: 0.2 }}
|
|
viewport={{ once: true }}
|
|
className="text-center md:text-left"
|
|
>
|
|
<h4 className="text-lg font-semibold text-white mb-4">Services</h4>
|
|
<ul className="space-y-2 text-slate-300">
|
|
<li className="hover:text-yellow-400 transition-colors cursor-pointer">Carbon Calculator</li>
|
|
<li className="hover:text-yellow-400 transition-colors cursor-pointer">Offset Portfolio</li>
|
|
<li className="hover:text-yellow-400 transition-colors cursor-pointer">Fleet Management</li>
|
|
<li className="hover:text-yellow-400 transition-colors cursor-pointer">Custom Solutions</li>
|
|
</ul>
|
|
</motion.div>
|
|
|
|
<motion.div
|
|
initial={{ opacity: 0, y: 20 }}
|
|
whileInView={{ opacity: 1, y: 0 }}
|
|
transition={{ duration: 0.6, delay: 0.4 }}
|
|
viewport={{ once: true }}
|
|
className="text-center md:text-left"
|
|
>
|
|
<h4 className="text-lg font-semibold text-white mb-4">Sustainability Partners</h4>
|
|
<p className="text-slate-300 mb-4">
|
|
Powered by verified carbon offset projects through Wren Climate
|
|
</p>
|
|
<div className="text-xs text-slate-400">
|
|
All projects are verified to international standards
|
|
</div>
|
|
</motion.div>
|
|
</div>
|
|
|
|
<motion.div
|
|
className="border-t border-slate-700 pt-8 text-center"
|
|
initial={{ opacity: 0 }}
|
|
whileInView={{ opacity: 1 }}
|
|
transition={{ duration: 0.6, delay: 0.6 }}
|
|
viewport={{ once: true }}
|
|
>
|
|
<p className="text-slate-400">
|
|
© 2024 Puffin Offset. Luxury meets sustainability.
|
|
</p>
|
|
</motion.div>
|
|
</div>
|
|
</footer>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default App;
|