feat: global css variables and app shell

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
M1ngdaXie
2026-03-26 14:15:55 -07:00
parent 814504db8c
commit 3f07963cb5
5 changed files with 51 additions and 304 deletions

View File

@@ -1,184 +1,32 @@
.counter { *, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
font-size: 16px;
padding: 5px 10px;
border-radius: 5px;
color: var(--accent);
background: var(--accent-bg);
border: 2px solid transparent;
transition: border-color 0.3s;
margin-bottom: 24px;
&:hover { :root {
border-color: var(--accent-border); --font-ui: 'Outfit', system-ui, sans-serif;
} --font-mono: 'Fira Code', 'Courier New', monospace;
&:focus-visible {
outline: 2px solid var(--accent); --accent-red: #ff453a;
outline-offset: 2px; --accent-yellow: #ffd60a;
} --accent-green: #30d158;
--accent-blue: #0a84ff;
--accent-purple: #bf5af2;
--accent-teal: #64d2ff;
--glass-bg: rgba(30, 30, 40, 0.55);
--glass-border: rgba(255, 255, 255, 0.08);
--glass-blur: blur(24px) saturate(1.4);
--glass-shadow: 0 8px 40px rgba(0, 0, 0, 0.35);
--radius-window: 12px;
--radius-icon: 14px;
--menubar-height: 28px;
--dock-height: 72px;
--titlebar-height: 36px;
} }
.hero { html, body, #root {
position: relative;
.base,
.framework,
.vite {
inset-inline: 0;
margin: 0 auto;
}
.base {
width: 170px;
position: relative;
z-index: 0;
}
.framework,
.vite {
position: absolute;
}
.framework {
z-index: 1;
top: 34px;
height: 28px;
transform: perspective(2000px) rotateZ(300deg) rotateX(44deg) rotateY(39deg)
scale(1.4);
}
.vite {
z-index: 0;
top: 107px;
height: 26px;
width: auto;
transform: perspective(2000px) rotateZ(300deg) rotateX(40deg) rotateY(39deg)
scale(0.8);
}
}
#center {
display: flex;
flex-direction: column;
gap: 25px;
place-content: center;
place-items: center;
flex-grow: 1;
@media (max-width: 1024px) {
padding: 32px 20px 24px;
gap: 18px;
}
}
#next-steps {
display: flex;
border-top: 1px solid var(--border);
text-align: left;
& > div {
flex: 1 1 0;
padding: 32px;
@media (max-width: 1024px) {
padding: 24px 20px;
}
}
.icon {
margin-bottom: 16px;
width: 22px;
height: 22px;
}
@media (max-width: 1024px) {
flex-direction: column;
text-align: center;
}
}
#docs {
border-right: 1px solid var(--border);
@media (max-width: 1024px) {
border-right: none;
border-bottom: 1px solid var(--border);
}
}
#next-steps ul {
list-style: none;
padding: 0;
display: flex;
gap: 8px;
margin: 32px 0 0;
.logo {
height: 18px;
}
a {
color: var(--text-h);
font-size: 16px;
border-radius: 6px;
background: var(--social-bg);
display: flex;
padding: 6px 12px;
align-items: center;
gap: 8px;
text-decoration: none;
transition: box-shadow 0.3s;
&:hover {
box-shadow: var(--shadow);
}
.button-icon {
height: 18px;
width: 18px;
}
}
@media (max-width: 1024px) {
margin-top: 20px;
flex-wrap: wrap;
justify-content: center;
li {
flex: 1 1 calc(50% - 8px);
}
a {
width: 100%;
justify-content: center;
box-sizing: border-box;
}
}
}
#spacer {
height: 88px;
border-top: 1px solid var(--border);
@media (max-width: 1024px) {
height: 48px;
}
}
.ticks {
position: relative;
width: 100%; width: 100%;
height: 100%;
&::before, overflow: hidden;
&::after { font-family: var(--font-ui);
content: ''; -webkit-font-smoothing: antialiased;
position: absolute;
top: -4.5px;
border: 5px solid transparent;
}
&::before {
left: 0;
border-left-color: var(--border);
}
&::after {
right: 0;
border-right-color: var(--border);
}
} }

View File

@@ -1,121 +1,19 @@
import { useState } from 'react' import { useState } from 'react';
import reactLogo from './assets/react.svg' import { WindowProvider } from './context/WindowContext';
import viteLogo from './assets/vite.svg' import BootScreen from './components/BootScreen/BootScreen';
import heroImg from './assets/hero.png' import Desktop from './components/Desktop/Desktop';
import './App.css' import './App.css';
function App() { export default function App() {
const [count, setCount] = useState(0) const [booted, setBooted] = useState(false);
return ( return (
<> <WindowProvider>
<section id="center"> {!booted ? (
<div className="hero"> <BootScreen onComplete={() => setBooted(true)} />
<img src={heroImg} className="base" width="170" height="179" alt="" /> ) : (
<img src={reactLogo} className="framework" alt="React logo" /> <Desktop />
<img src={viteLogo} className="vite" alt="Vite logo" /> )}
</div> </WindowProvider>
<div> );
<h1>Get started</h1>
<p>
Edit <code>src/App.tsx</code> and save to test <code>HMR</code>
</p>
</div>
<button
className="counter"
onClick={() => setCount((count) => count + 1)}
>
Count is {count}
</button>
</section>
<div className="ticks"></div>
<section id="next-steps">
<div id="docs">
<svg className="icon" role="presentation" aria-hidden="true">
<use href="/icons.svg#documentation-icon"></use>
</svg>
<h2>Documentation</h2>
<p>Your questions, answered</p>
<ul>
<li>
<a href="https://vite.dev/" target="_blank">
<img className="logo" src={viteLogo} alt="" />
Explore Vite
</a>
</li>
<li>
<a href="https://react.dev/" target="_blank">
<img className="button-icon" src={reactLogo} alt="" />
Learn more
</a>
</li>
</ul>
</div>
<div id="social">
<svg className="icon" role="presentation" aria-hidden="true">
<use href="/icons.svg#social-icon"></use>
</svg>
<h2>Connect with us</h2>
<p>Join the Vite community</p>
<ul>
<li>
<a href="https://github.com/vitejs/vite" target="_blank">
<svg
className="button-icon"
role="presentation"
aria-hidden="true"
>
<use href="/icons.svg#github-icon"></use>
</svg>
GitHub
</a>
</li>
<li>
<a href="https://chat.vite.dev/" target="_blank">
<svg
className="button-icon"
role="presentation"
aria-hidden="true"
>
<use href="/icons.svg#discord-icon"></use>
</svg>
Discord
</a>
</li>
<li>
<a href="https://x.com/vite_js" target="_blank">
<svg
className="button-icon"
role="presentation"
aria-hidden="true"
>
<use href="/icons.svg#x-icon"></use>
</svg>
X.com
</a>
</li>
<li>
<a href="https://bsky.app/profile/vite.dev" target="_blank">
<svg
className="button-icon"
role="presentation"
aria-hidden="true"
>
<use href="/icons.svg#bluesky-icon"></use>
</svg>
Bluesky
</a>
</li>
</ul>
</div>
</section>
<div className="ticks"></div>
<section id="spacer"></section>
</>
)
} }
export default App

View File

@@ -0,0 +1 @@
export default function BootScreen({ onComplete }: { onComplete: () => void }) { return <div onClick={onComplete}>Loading...</div>; }

View File

@@ -0,0 +1 @@
export default function Desktop() { return <div>Desktop</div>; }

View File

@@ -1,10 +1,9 @@
import { StrictMode } from 'react' import React from 'react';
import { createRoot } from 'react-dom/client' import ReactDOM from 'react-dom/client';
import './index.css' import App from './App.tsx';
import App from './App.tsx'
createRoot(document.getElementById('root')!).render( ReactDOM.createRoot(document.getElementById('root')!).render(
<StrictMode> <React.StrictMode>
<App /> <App />
</StrictMode>, </React.StrictMode>
) );