Spaces:
Running
Running
| import { useState, useEffect } from 'react'; | |
| import { | |
| Box, | |
| Container, | |
| Typography, | |
| Button, | |
| Grid, | |
| Card, | |
| CardContent, | |
| Chip, | |
| Stack, | |
| CircularProgress, | |
| } from '@mui/material'; | |
| import DownloadIcon from '@mui/icons-material/Download'; | |
| import AppleIcon from '@mui/icons-material/Apple'; | |
| import CheckCircleIcon from '@mui/icons-material/CheckCircle'; | |
| import OpenInNewIcon from '@mui/icons-material/OpenInNew'; | |
| import ExpandMoreIcon from '@mui/icons-material/ExpandMore'; | |
| import ExpandLessIcon from '@mui/icons-material/ExpandLess'; | |
| import Layout from '../components/Layout'; | |
| // Platform configuration | |
| const PLATFORMS = { | |
| 'darwin-aarch64': { | |
| name: 'macOS', | |
| subtitle: 'Apple Silicon', | |
| arch: 'M1, M2, M3, M4', | |
| format: '.dmg', | |
| icon: AppleIcon, | |
| color: '#a3a3a3', | |
| }, | |
| 'darwin-x86_64': { | |
| name: 'macOS', | |
| subtitle: 'Intel', | |
| arch: 'x86_64', | |
| format: '.dmg', | |
| icon: AppleIcon, | |
| color: '#a3a3a3', | |
| }, | |
| 'windows-x86_64': { | |
| name: 'Windows', | |
| subtitle: '64-bit', | |
| arch: 'x86_64', | |
| format: '.msi', | |
| icon: null, | |
| color: '#0078d4', | |
| }, | |
| 'linux-x86_64': { | |
| name: 'Linux', | |
| subtitle: 'Debian/Ubuntu', | |
| arch: 'x86_64', | |
| format: '.deb', | |
| icon: null, | |
| color: '#e95420', | |
| }, | |
| }; | |
| // URL to fetch latest release info (using GitHub API for CORS support) | |
| const GITHUB_RELEASES_API = 'https://api.github.com/repos/pollen-robotics/reachy-mini-desktop-app/releases/latest'; | |
| const GITHUB_RELEASES_LIST_API = 'https://api.github.com/repos/pollen-robotics/reachy-mini-desktop-app/releases?per_page=10'; | |
| // Detect user's platform | |
| function detectPlatform() { | |
| const ua = navigator.userAgent; | |
| const platform = navigator.platform || ''; | |
| if (/Mac/.test(platform) || /Mac/.test(ua)) { | |
| return 'darwin-aarch64'; | |
| } | |
| if (/Win/.test(platform) || /Windows/.test(ua)) { | |
| return 'windows-x86_64'; | |
| } | |
| if (/Linux/.test(platform) || /Linux/.test(ua)) { | |
| return 'linux-x86_64'; | |
| } | |
| return 'darwin-aarch64'; | |
| } | |
| // Format date | |
| function formatDate(dateString) { | |
| const date = new Date(dateString); | |
| return date.toLocaleDateString('en-US', { | |
| year: 'numeric', | |
| month: 'short', | |
| day: 'numeric', | |
| }); | |
| } | |
| // Parse release body and extract clean changes | |
| function parseReleaseChanges(body) { | |
| if (!body) return []; | |
| const changes = []; | |
| const lines = body.split('\n'); | |
| for (const line of lines) { | |
| const trimmed = line.trim(); | |
| // Skip empty lines, headers, and meta content | |
| if (!trimmed) continue; | |
| if (trimmed.startsWith('##')) continue; // Skip all headers (## What's Changed, etc.) | |
| if (trimmed.startsWith('**Full Changelog**')) continue; | |
| if (trimmed.startsWith('**New Contributors**')) continue; | |
| if (trimmed.includes('made their first contribution')) continue; | |
| if (trimmed.startsWith('<!--') || trimmed.endsWith('-->')) continue; | |
| if (trimmed === 'See the assets to download this version and install.') continue; | |
| // Parse change lines (starting with * or -) | |
| if (trimmed.startsWith('*') || trimmed.startsWith('-')) { | |
| let change = trimmed.replace(/^[\*\-]\s*/, ''); | |
| // Extract the description from markdown links: "fix: description by @user in https://..." | |
| // We want to keep: "fix: description" | |
| const byMatch = change.match(/^(.+?)\s+by\s+@\w+/i); | |
| if (byMatch) { | |
| change = byMatch[1].trim(); | |
| } | |
| // Remove trailing "in https://..." links | |
| change = change.replace(/\s+in\s+https:\/\/[^\s]+$/i, ''); | |
| // Clean up any remaining markdown link syntax [text](url) -> text | |
| change = change.replace(/\[([^\]]+)\]\([^)]+\)/g, '$1'); | |
| // Skip if it's just a contributor line or empty after cleaning | |
| if (change && change.length > 3 && !change.includes('first contribution')) { | |
| changes.push(change); | |
| } | |
| } | |
| } | |
| return changes; | |
| } | |
| // Windows Icon | |
| function WindowsIcon({ sx = {} }) { | |
| return ( | |
| <Box component="svg" viewBox="0 0 24 24" sx={{ width: 32, height: 32, ...sx }}> | |
| <path fill="currentColor" d="M0 3.449L9.75 2.1v9.451H0m10.949-9.602L24 0v11.4H10.949M0 12.6h9.75v9.451L0 20.699M10.949 12.6H24V24l-12.9-1.801" /> | |
| </Box> | |
| ); | |
| } | |
| // Linux Icon | |
| function LinuxIcon({ sx = {} }) { | |
| return ( | |
| <Box component="svg" viewBox="0 0 24 24" sx={{ width: 32, height: 32, ...sx }}> | |
| <path fill="currentColor" d="M12.504 0c-.155 0-.315.008-.48.021-4.226.333-3.105 4.807-3.17 6.298-.076 1.092-.3 1.953-1.05 3.02-.885 1.051-2.127 2.75-2.716 4.521-.278.832-.41 1.684-.287 2.489.117.779.456 1.511 1.044 2.127.473.497.986.902 1.504 1.238.518.336 1.03.62 1.523.836.483.23.93.42 1.297.606.368.187.656.37.83.544.36.364.575.747.658 1.162.084.416.036.864-.132 1.307-.168.443-.45.879-.808 1.257-.358.378-.79.698-1.232.919-.443.22-.893.337-1.284.337-.392 0-.722-.116-.963-.337-.241-.22-.393-.544-.429-.954-.037-.41.04-.894.198-1.402.159-.509.397-1.041.678-1.54.282-.5.606-.968.937-1.352.33-.383.666-.68.966-.844.3-.163.562-.193.742-.05.18.142.278.429.25.802-.028.373-.182.83-.457 1.284-.275.455-.67.908-1.137 1.27-.467.362-1.006.632-1.557.746-.55.113-1.112.069-1.612-.155-.5-.223-.938-.627-1.24-1.155-.3-.53-.463-1.183-.423-1.89.04-.706.284-1.467.716-2.183.43-.715 1.047-1.386 1.786-1.923.74-.538 1.603-.944 2.503-1.173.9-.23 1.837-.284 2.718-.173.88.11 1.705.385 2.4.786.694.4 1.26.927 1.63 1.52.37.593.545 1.252.482 1.9-.062.648-.362 1.286-.855 1.824-.492.537-1.178.974-1.96 1.24-.783.267-1.664.36-2.527.276-.863-.085-1.707-.347-2.41-.785-.703-.438-1.266-1.052-1.58-1.764-.315-.713-.38-1.524-.174-2.303.206-.78.682-1.528 1.356-2.105.674-.578 1.546-.985 2.486-1.173.94-.188 1.948-.156 2.866.127.917.284 1.744.769 2.343 1.445.6.675.972 1.54 1.022 2.478.05.938-.222 1.949-.793 2.85-.572.902-1.443 1.694-2.516 2.218-1.073.523-2.35.78-3.633.69-1.284-.09-2.575-.528-3.653-1.305-1.078-.778-1.944-1.895-2.436-3.22-.493-1.326-.612-2.86-.27-4.386.34-1.525 1.14-3.04 2.34-4.287 1.2-1.246 2.798-2.224 4.584-2.683 1.786-.46 3.76-.402 5.554.248 1.794.65 3.408 1.89 4.516 3.548 1.11 1.658 1.713 3.733 1.62 5.83-.093 2.098-.882 4.218-2.25 5.99-1.37 1.77-3.318 3.19-5.54 4.002-2.223.81-4.72 1.013-7.08.528-2.36-.485-4.583-1.656-6.263-3.35-1.68-1.693-2.816-3.91-3.153-6.292-.337-2.382.127-4.929 1.344-7.14 1.218-2.212 3.19-4.09 5.528-5.3 2.338-1.21 5.043-1.754 7.707-1.483 2.663.27 5.286 1.353 7.382 3.063 2.097 1.71 3.667 4.048 4.377 6.632.71 2.583.56 5.413-.577 7.948-1.137 2.534-3.14 4.774-5.648 6.265-2.508 1.492-5.52 2.235-8.503 2.05-2.983-.186-5.938-1.3-8.283-3.166C2.997 19.41 1.26 16.792.433 13.906c-.828-2.886-.745-6.04.336-8.896C1.85 2.152 4.023-.06 6.698.8" /> | |
| </Box> | |
| ); | |
| } | |
| // Platform Card component | |
| function PlatformCard({ platformKey, url, isActive, onClick }) { | |
| const platform = PLATFORMS[platformKey]; | |
| const Icon = platform?.icon; | |
| const isComingSoon = platformKey.includes('windows') || platformKey.includes('linux'); | |
| return ( | |
| <Card | |
| onClick={onClick} | |
| sx={{ | |
| cursor: 'pointer', | |
| position: 'relative', | |
| background: isActive | |
| ? 'linear-gradient(135deg, rgba(59, 130, 246, 0.15) 0%, rgba(139, 92, 246, 0.1) 100%)' | |
| : 'rgba(255, 255, 255, 0.03)', | |
| border: '1px solid', | |
| borderColor: isActive ? 'rgba(59, 130, 246, 0.4)' : 'rgba(255, 255, 255, 0.08)', | |
| backdropFilter: 'blur(10px)', | |
| transition: 'all 0.25s ease', | |
| opacity: isComingSoon ? 0.7 : 1, | |
| '&:hover': { | |
| borderColor: isActive ? 'rgba(59, 130, 246, 0.6)' : 'rgba(255, 255, 255, 0.2)', | |
| transform: 'translateY(-4px)', | |
| boxShadow: isActive | |
| ? '0 12px 40px rgba(59, 130, 246, 0.2)' | |
| : '0 12px 40px rgba(0, 0, 0, 0.3)', | |
| opacity: 1, | |
| }, | |
| }} | |
| > | |
| {/* Coming soon tag */} | |
| {isComingSoon && ( | |
| <Chip | |
| label="Coming soon" | |
| size="small" | |
| sx={{ | |
| position: 'absolute', | |
| top: 8, | |
| right: 8, | |
| backgroundColor: 'rgba(255, 149, 0, 0.2)', | |
| color: '#FF9500', | |
| fontSize: 10, | |
| fontWeight: 700, | |
| height: 20, | |
| '& .MuiChip-label': { px: 1 }, | |
| }} | |
| /> | |
| )} | |
| <CardContent | |
| component="a" | |
| href={isComingSoon ? undefined : url} | |
| sx={{ | |
| display: 'flex', | |
| flexDirection: 'column', | |
| alignItems: 'center', | |
| gap: 1.5, | |
| textDecoration: 'none', | |
| color: 'inherit', | |
| p: 3, | |
| '&:last-child': { pb: 3 }, | |
| pointerEvents: isComingSoon ? 'none' : 'auto', | |
| }} | |
| > | |
| <Box sx={{ | |
| width: 56, | |
| height: 56, | |
| display: 'flex', | |
| alignItems: 'center', | |
| justifyContent: 'center', | |
| borderRadius: 3, | |
| backgroundColor: 'rgba(255, 255, 255, 0.05)', | |
| }}> | |
| {Icon ? ( | |
| <Icon sx={{ fontSize: 32, color: platform?.color || 'white' }} /> | |
| ) : platformKey.includes('windows') ? ( | |
| <WindowsIcon sx={{ color: platform?.color }} /> | |
| ) : ( | |
| <LinuxIcon sx={{ color: platform?.color }} /> | |
| )} | |
| </Box> | |
| <Box sx={{ textAlign: 'center' }}> | |
| <Typography | |
| variant="subtitle1" | |
| sx={{ fontWeight: 600, color: 'white', lineHeight: 1.2 }} | |
| > | |
| {platform?.name} | |
| </Typography> | |
| <Typography | |
| variant="caption" | |
| sx={{ color: 'rgba(255,255,255,0.5)' }} | |
| > | |
| {platform?.subtitle} | |
| </Typography> | |
| </Box> | |
| <Chip | |
| label={platform?.format} | |
| size="small" | |
| sx={{ | |
| backgroundColor: 'rgba(255, 255, 255, 0.08)', | |
| color: 'rgba(255,255,255,0.7)', | |
| fontSize: 11, | |
| fontWeight: 600, | |
| height: 24, | |
| }} | |
| /> | |
| </CardContent> | |
| </Card> | |
| ); | |
| } | |
| export default function Download() { | |
| const [releaseData, setReleaseData] = useState(null); | |
| const [allReleases, setAllReleases] = useState([]); | |
| const [detectedPlatform, setDetectedPlatform] = useState(null); | |
| const [loading, setLoading] = useState(true); | |
| const [showAllReleases, setShowAllReleases] = useState(false); | |
| const [error, setError] = useState(null); | |
| useEffect(() => { | |
| setDetectedPlatform(detectPlatform()); | |
| // Fetch latest release info from GitHub API | |
| async function fetchReleases() { | |
| try { | |
| // Fetch latest release for download buttons | |
| const latestResponse = await fetch(GITHUB_RELEASES_API); | |
| // Fetch all releases for changelog | |
| const allResponse = await fetch(GITHUB_RELEASES_LIST_API); | |
| if (latestResponse.ok) { | |
| const data = await latestResponse.json(); | |
| // Transform GitHub API response to our format | |
| const version = data.tag_name?.replace('v', '') || ''; | |
| const platforms = {}; | |
| // Map assets to platforms (prioritize user-friendly formats) | |
| data.assets?.forEach(asset => { | |
| const name = asset.name.toLowerCase(); | |
| const url = asset.browser_download_url; | |
| // macOS Apple Silicon - prefer .dmg | |
| if (name.includes('arm64.dmg')) { | |
| platforms['darwin-aarch64'] = { url }; | |
| } else if (name.includes('darwin-aarch64') && !platforms['darwin-aarch64']) { | |
| platforms['darwin-aarch64'] = { url }; | |
| } | |
| // macOS Intel - prefer .dmg | |
| if (name.includes('x64.dmg') && !name.includes('arm64')) { | |
| platforms['darwin-x86_64'] = { url }; | |
| } else if (name.includes('darwin-x86_64') && !platforms['darwin-x86_64']) { | |
| platforms['darwin-x86_64'] = { url }; | |
| } | |
| // Windows - .msi | |
| if (name.includes('.msi')) { | |
| platforms['windows-x86_64'] = { url }; | |
| } | |
| // Linux - .deb | |
| if (name.includes('amd64.deb')) { | |
| platforms['linux-x86_64'] = { url }; | |
| } | |
| }); | |
| setReleaseData({ | |
| version, | |
| pub_date: data.published_at, | |
| platforms, | |
| }); | |
| } else { | |
| setError('Failed to fetch release info'); | |
| } | |
| // Set all releases for changelog | |
| if (allResponse.ok) { | |
| const releases = await allResponse.json(); | |
| setAllReleases(releases.filter(r => !r.draft)); | |
| } | |
| } catch (err) { | |
| console.error('Error fetching release:', err); | |
| setError('Failed to fetch release info'); | |
| } finally { | |
| setLoading(false); | |
| } | |
| } | |
| fetchReleases(); | |
| }, []); | |
| if (loading) { | |
| return ( | |
| <Layout transparentHeader> | |
| <Box sx={{ display: 'flex', justifyContent: 'center', alignItems: 'center', minHeight: '100vh', bgcolor: '#000' }}> | |
| <CircularProgress sx={{ color: 'white' }} /> | |
| </Box> | |
| </Layout> | |
| ); | |
| } | |
| if (error || !releaseData) { | |
| return ( | |
| <Layout transparentHeader> | |
| <Box sx={{ display: 'flex', flexDirection: 'column', justifyContent: 'center', alignItems: 'center', minHeight: '100vh', bgcolor: '#000', color: 'white', gap: 3 }}> | |
| <Typography variant="h5">Unable to load release info</Typography> | |
| <Button | |
| variant="outlined" | |
| href="https://github.com/pollen-robotics/reachy-mini-desktop-app/releases" | |
| target="_blank" | |
| sx={{ color: 'white', borderColor: 'rgba(255,255,255,0.3)' }} | |
| > | |
| View releases on GitHub | |
| </Button> | |
| </Box> | |
| </Layout> | |
| ); | |
| } | |
| const currentPlatform = PLATFORMS[detectedPlatform]; | |
| const currentUrl = releaseData?.platforms[detectedPlatform]?.url; | |
| return ( | |
| <Layout transparentHeader> | |
| <Box | |
| sx={{ | |
| minHeight: '100vh', | |
| background: 'linear-gradient(180deg, #000 0%, #0a0a12 50%, #0f0f1a 100%)', | |
| color: 'white', | |
| pt: 14, | |
| pb: 12, | |
| position: 'relative', | |
| overflow: 'hidden', | |
| }} | |
| > | |
| {/* Subtle gradient orbs - spread across the page */} | |
| <Box | |
| sx={{ | |
| position: 'absolute', | |
| top: 100, | |
| left: '-10%', | |
| width: 600, | |
| height: 600, | |
| background: 'radial-gradient(circle, rgba(59, 130, 246, 0.1) 0%, transparent 70%)', | |
| pointerEvents: 'none', | |
| }} | |
| /> | |
| <Box | |
| sx={{ | |
| position: 'absolute', | |
| top: '40%', | |
| right: '-15%', | |
| width: 700, | |
| height: 700, | |
| background: 'radial-gradient(circle, rgba(139, 92, 246, 0.08) 0%, transparent 70%)', | |
| pointerEvents: 'none', | |
| }} | |
| /> | |
| <Box | |
| sx={{ | |
| position: 'absolute', | |
| bottom: 100, | |
| left: '20%', | |
| width: 500, | |
| height: 500, | |
| background: 'radial-gradient(circle, rgba(255, 149, 0, 0.05) 0%, transparent 70%)', | |
| pointerEvents: 'none', | |
| }} | |
| /> | |
| <Container maxWidth="md" sx={{ position: 'relative', zIndex: 1 }}> | |
| {/* Hero Section */} | |
| <Box sx={{ textAlign: 'center', mb: 8 }}> | |
| {/* App icon */} | |
| <Box | |
| sx={{ | |
| display: 'flex', | |
| alignItems: 'center', | |
| justifyContent: 'center', | |
| mb: 4, | |
| }} | |
| > | |
| <Box | |
| sx={{ | |
| width: 100, | |
| height: 100, | |
| borderRadius: '24px', | |
| background: 'linear-gradient(135deg, rgba(255,255,255,0.1) 0%, rgba(255,255,255,0.05) 100%)', | |
| border: '1px solid rgba(255,255,255,0.1)', | |
| display: 'flex', | |
| alignItems: 'center', | |
| justifyContent: 'center', | |
| boxShadow: '0 8px 32px rgba(0,0,0,0.3)', | |
| }} | |
| > | |
| <Box | |
| component="img" | |
| src="/assets/reachy-icon.svg" | |
| alt="Reachy Mini Control" | |
| sx={{ width: 64, height: 64 }} | |
| /> | |
| </Box> | |
| </Box> | |
| <Typography | |
| variant="h2" | |
| sx={{ | |
| mb: 2, | |
| background: 'linear-gradient(135deg, #fff 0%, rgba(255,255,255,0.8) 100%)', | |
| backgroundClip: 'text', | |
| WebkitBackgroundClip: 'text', | |
| WebkitTextFillColor: 'transparent', | |
| }} | |
| > | |
| Reachy Mini Control | |
| </Typography> | |
| <Typography | |
| variant="h6" | |
| sx={{ | |
| color: 'rgba(255,255,255,0.6)', | |
| fontWeight: 400, | |
| mb: 3, | |
| maxWidth: 450, | |
| mx: 'auto', | |
| }} | |
| > | |
| The official desktop app to control, program, and play with your Reachy Mini. | |
| </Typography> | |
| {/* Version info */} | |
| <Stack | |
| direction="row" | |
| spacing={2} | |
| justifyContent="center" | |
| alignItems="center" | |
| sx={{ mb: 5 }} | |
| > | |
| <Chip | |
| icon={<Box sx={{ width: 8, height: 8, bgcolor: '#10b981', borderRadius: '50%', ml: 1 }} />} | |
| label={`v${releaseData?.version}`} | |
| sx={{ | |
| backgroundColor: 'rgba(16, 185, 129, 0.1)', | |
| color: '#10b981', | |
| fontWeight: 600, | |
| border: '1px solid rgba(16, 185, 129, 0.2)', | |
| }} | |
| /> | |
| <Typography variant="body2" sx={{ color: 'rgba(255,255,255,0.4)' }}> | |
| Released {formatDate(releaseData?.pub_date)} | |
| </Typography> | |
| </Stack> | |
| {/* Primary download button - different for macOS vs Windows/Linux */} | |
| {detectedPlatform?.startsWith('darwin') ? ( | |
| <> | |
| <Button | |
| variant="contained" | |
| size="large" | |
| href={currentUrl} | |
| startIcon={<DownloadIcon />} | |
| sx={{ | |
| px: 6, | |
| py: 2, | |
| fontSize: 17, | |
| fontWeight: 600, | |
| borderRadius: 3, | |
| background: 'linear-gradient(135deg, #FF9500 0%, #764ba2 100%)', | |
| boxShadow: '0 8px 32px rgba(255, 149, 0, 0.35)', | |
| transition: 'all 0.3s ease', | |
| '&:hover': { | |
| boxShadow: '0 12px 48px rgba(59, 130, 246, 0.5)', | |
| transform: 'translateY(-2px)', | |
| }, | |
| }} | |
| > | |
| Download for {currentPlatform?.name} | |
| </Button> | |
| <Typography | |
| variant="body2" | |
| sx={{ | |
| color: 'rgba(255,255,255,0.4)', | |
| mt: 2, | |
| fontSize: 13, | |
| }} | |
| > | |
| {currentPlatform?.subtitle} • {currentPlatform?.format?.replace('.', '').toUpperCase()} package | |
| </Typography> | |
| </> | |
| ) : ( | |
| <> | |
| {/* Windows/Linux - Coming soon message */} | |
| <Box | |
| sx={{ | |
| px: 5, | |
| py: 2.5, | |
| borderRadius: 3, | |
| background: 'linear-gradient(135deg, rgba(255, 149, 0, 0.15) 0%, rgba(139, 92, 246, 0.1) 100%)', | |
| border: '1px solid rgba(255, 149, 0, 0.3)', | |
| display: 'inline-block', | |
| }} | |
| > | |
| <Typography | |
| variant="h6" | |
| sx={{ | |
| color: '#FF9500', | |
| fontWeight: 600, | |
| fontSize: 17, | |
| }} | |
| > | |
| 🚧 {currentPlatform?.name} support coming soon! | |
| </Typography> | |
| </Box> | |
| <Typography | |
| variant="body2" | |
| sx={{ | |
| color: 'rgba(255,255,255,0.5)', | |
| mt: 2, | |
| fontSize: 14, | |
| maxWidth: 400, | |
| mx: 'auto', | |
| }} | |
| > | |
| We're working hard to bring Reachy Mini Control to {currentPlatform?.name}. | |
| In the meantime, macOS is fully supported! | |
| </Typography> | |
| </> | |
| )} | |
| {/* App screenshot */} | |
| <Box | |
| component="img" | |
| src="/assets/desktop-app-screenshot--white.png" | |
| alt="Reachy Mini Control Dashboard" | |
| sx={{ | |
| mt: 6, | |
| width: '100%', | |
| maxWidth: 700, | |
| mx: 'auto', | |
| display: 'block', | |
| borderRadius: '12px', | |
| }} | |
| /> | |
| </Box> | |
| {/* All platforms */} | |
| <Box sx={{ mb: 8 }}> | |
| <Typography | |
| variant="overline" | |
| sx={{ | |
| color: 'rgba(255,255,255,0.4)', | |
| display: 'block', | |
| textAlign: 'center', | |
| mb: 3, | |
| letterSpacing: 2, | |
| }} | |
| > | |
| Available for all platforms | |
| </Typography> | |
| <Grid container spacing={2}> | |
| {['darwin-aarch64', 'darwin-x86_64', 'windows-x86_64', 'linux-x86_64'].map((key) => ( | |
| <Grid size={{ xs: 6, sm: 3 }} key={key}> | |
| <PlatformCard | |
| platformKey={key} | |
| url={releaseData?.platforms[key]?.url} | |
| isActive={key === detectedPlatform} | |
| onClick={() => setDetectedPlatform(key)} | |
| /> | |
| </Grid> | |
| ))} | |
| </Grid> | |
| {/* Platform support notice - only show on macOS */} | |
| {detectedPlatform?.startsWith('darwin') && ( | |
| <Box | |
| sx={{ | |
| mt: 3, | |
| p: 2, | |
| background: 'rgba(255, 255, 255, 0.03)', | |
| border: '1px solid rgba(255, 255, 255, 0.08)', | |
| borderRadius: 2, | |
| textAlign: 'center', | |
| }} | |
| > | |
| <Typography | |
| variant="body2" | |
| sx={{ color: 'rgba(255,255,255,0.5)' }} | |
| > | |
| 🚧 Windows & Linux support coming soon | |
| </Typography> | |
| </Box> | |
| )} | |
| </Box> | |
| {/* Features / What's included */} | |
| <Box | |
| sx={{ | |
| background: 'rgba(255, 255, 255, 0.02)', | |
| border: '1px solid rgba(255, 255, 255, 0.06)', | |
| borderRadius: 4, | |
| p: 4, | |
| mb: 6, | |
| }} | |
| > | |
| <Typography | |
| variant="h6" | |
| sx={{ mb: 3, color: 'white', fontWeight: 600 }} | |
| > | |
| What's included | |
| </Typography> | |
| <Grid container spacing={2}> | |
| {[ | |
| '3D visualization of your robot', | |
| 'Real-time motor control', | |
| 'App Store with 30+ apps', | |
| 'Camera & microphone access', | |
| 'Record & playback movements', | |
| 'Full SDK integration', | |
| ].map((feature, i) => ( | |
| <Grid size={{ xs: 12, sm: 6 }} key={i}> | |
| <Stack direction="row" spacing={1.5} alignItems="center"> | |
| <CheckCircleIcon sx={{ color: '#10b981', fontSize: 20 }} /> | |
| <Typography variant="body2" sx={{ color: 'rgba(255,255,255,0.7)' }}> | |
| {feature} | |
| </Typography> | |
| </Stack> | |
| </Grid> | |
| ))} | |
| </Grid> | |
| </Box> | |
| {/* Requirements */} | |
| <Box sx={{ textAlign: 'center', mb: 8 }}> | |
| <Typography | |
| variant="body2" | |
| sx={{ color: 'rgba(255,255,255,0.4)', mb: 2 }} | |
| > | |
| Requires macOS 11+, Windows 10+, or Debian/Ubuntu Linux | |
| </Typography> | |
| <Button | |
| variant="text" | |
| size="small" | |
| href="https://github.com/pollen-robotics/reachy-mini-desktop-app/releases" | |
| target="_blank" | |
| endIcon={<OpenInNewIcon sx={{ fontSize: 16 }} />} | |
| sx={{ | |
| color: 'rgba(255,255,255,0.5)', | |
| '&:hover': { color: 'white' }, | |
| }} | |
| > | |
| View all releases on GitHub | |
| </Button> | |
| </Box> | |
| {/* Release Notes */} | |
| {allReleases.length > 0 && ( | |
| <Box | |
| id="release-notes" | |
| sx={{ | |
| background: 'rgba(255, 255, 255, 0.02)', | |
| border: '1px solid rgba(255, 255, 255, 0.06)', | |
| borderRadius: 4, | |
| p: 4, | |
| scrollMarginTop: '100px', // Offset for fixed header | |
| }} | |
| > | |
| <Typography | |
| variant="h6" | |
| sx={{ mb: 3, color: 'white', fontWeight: 600 }} | |
| > | |
| Release Notes | |
| </Typography> | |
| <Stack spacing={2.5}> | |
| {(showAllReleases ? allReleases : allReleases.slice(0, 5)) | |
| .map((release) => { | |
| const changes = parseReleaseChanges(release.body); | |
| return ( | |
| <Box | |
| key={release.id} | |
| sx={{ | |
| borderLeft: '2px solid rgba(255, 149, 0, 0.4)', | |
| pl: 3, | |
| py: 0.5, | |
| }} | |
| > | |
| <Stack direction="row" spacing={2} alignItems="center" flexWrap="wrap" sx={{ mb: changes.length > 0 ? 1 : 0 }}> | |
| <Typography | |
| variant="subtitle2" | |
| sx={{ color: 'white', fontWeight: 600 }} | |
| > | |
| {release.tag_name} | |
| </Typography> | |
| <Chip | |
| label={formatDate(release.published_at)} | |
| size="small" | |
| sx={{ | |
| backgroundColor: 'rgba(255, 255, 255, 0.05)', | |
| color: 'rgba(255,255,255,0.5)', | |
| fontSize: 10, | |
| height: 20, | |
| }} | |
| /> | |
| {release.prerelease && ( | |
| <Chip | |
| label="Pre-release" | |
| size="small" | |
| sx={{ | |
| backgroundColor: 'rgba(255, 149, 0, 0.15)', | |
| color: '#FF9500', | |
| fontSize: 10, | |
| height: 20, | |
| }} | |
| /> | |
| )} | |
| </Stack> | |
| {changes.length > 0 && ( | |
| <Box component="ul" sx={{ m: 0, pl: 2.5, listStyle: 'none' }}> | |
| {changes.map((change, i) => ( | |
| <Box | |
| component="li" | |
| key={i} | |
| sx={{ | |
| color: 'rgba(255,255,255,0.6)', | |
| fontSize: 13, | |
| lineHeight: 1.6, | |
| position: 'relative', | |
| '&::before': { | |
| content: '"•"', | |
| position: 'absolute', | |
| left: -14, | |
| color: 'rgba(255, 149, 0, 0.6)', | |
| } | |
| }} | |
| > | |
| {change} | |
| </Box> | |
| ))} | |
| </Box> | |
| )} | |
| </Box> | |
| ); | |
| })} | |
| </Stack> | |
| {allReleases.length > 5 && ( | |
| <Button | |
| variant="text" | |
| onClick={() => setShowAllReleases(!showAllReleases)} | |
| endIcon={showAllReleases ? <ExpandLessIcon /> : <ExpandMoreIcon />} | |
| sx={{ | |
| mt: 2, | |
| color: 'rgba(255,255,255,0.5)', | |
| '&:hover': { color: 'white' }, | |
| }} | |
| > | |
| {showAllReleases ? 'Show less' : 'Show older releases'} | |
| </Button> | |
| )} | |
| </Box> | |
| )} | |
| </Container> | |
| </Box> | |
| </Layout> | |
| ); | |
| } | |