first-commit
ci / Validate workspace (push) Has been cancelled
landing-page-ci / Validate landing page (push) Has been cancelled
landing-page-deploy / Deploy landing page (push) Has been cancelled
github-metrics / Generate repository metrics SVG (push) Has been cancelled
refresh-contributors-wall / Refresh contributors wall cache bust (push) Waiting to run
ci / Validate workspace (push) Has been cancelled
landing-page-ci / Validate landing page (push) Has been cancelled
landing-page-deploy / Deploy landing page (push) Has been cancelled
github-metrics / Generate repository metrics SVG (push) Has been cancelled
refresh-contributors-wall / Refresh contributors wall cache bust (push) Waiting to run
This commit is contained in:
@@ -0,0 +1,442 @@
|
||||
<!doctype html>
|
||||
<!--
|
||||
OD mobile-app seed.
|
||||
|
||||
A pixel-accurate iPhone 15 Pro frame (390 × 844) with Dynamic Island,
|
||||
status-bar SVG icons, and home indicator — drawn entirely in HTML/SVG, no
|
||||
external image. The screen content lives inside `<main class="screen">`;
|
||||
paste in one of the layouts from `references/layouts.md`.
|
||||
|
||||
Tokens at the top of `<style>` mirror the web-prototype seed so a single
|
||||
DESIGN.md flows into both. Mobile spacing is tighter (~25%) and type sizes
|
||||
drop one step from desktop — all pre-applied here.
|
||||
-->
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||
<title>[REPLACE] Screen name · brand</title>
|
||||
<style>
|
||||
:root {
|
||||
--bg: #fafaf7;
|
||||
--surface: #ffffff;
|
||||
--fg: #1a1916;
|
||||
--muted: #6b6964;
|
||||
--border: #e8e5df;
|
||||
--accent: #c96442;
|
||||
|
||||
--accent-soft: color-mix(in oklch, var(--accent) 14%, transparent);
|
||||
--fg-soft: color-mix(in oklch, var(--fg) 6%, transparent);
|
||||
|
||||
--font-display: 'Iowan Old Style', 'Charter', Georgia, serif;
|
||||
--font-body: -apple-system, BlinkMacSystemFont, 'SF Pro Text', system-ui, sans-serif;
|
||||
--font-mono: ui-monospace, 'SF Mono', Menlo, monospace;
|
||||
|
||||
/* mobile type — one step down from web-prototype defaults */
|
||||
--fs-h1: 26px;
|
||||
--fs-h2: 20px;
|
||||
--fs-h3: 16px;
|
||||
--fs-body: 15px;
|
||||
--fs-meta: 12px;
|
||||
|
||||
--radius-card: 18px;
|
||||
--radius-pill: 999px;
|
||||
}
|
||||
|
||||
*, *::before, *::after { box-sizing: border-box; }
|
||||
html, body { margin: 0; padding: 0; height: 100%; }
|
||||
body {
|
||||
background:
|
||||
radial-gradient(60% 80% at 50% 0%, color-mix(in oklch, var(--accent) 6%, var(--bg)) 0%, var(--bg) 60%);
|
||||
color: var(--fg);
|
||||
font-family: var(--font-body);
|
||||
font-size: var(--fs-body);
|
||||
line-height: 1.4;
|
||||
-webkit-font-smoothing: antialiased;
|
||||
display: grid;
|
||||
place-items: center;
|
||||
padding: 32px;
|
||||
}
|
||||
|
||||
/* ─── caption above the device ──────────────────────────────────── */
|
||||
.stage {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: 24px;
|
||||
}
|
||||
.caption {
|
||||
font-family: var(--font-mono);
|
||||
font-size: 12px;
|
||||
letter-spacing: 0.08em;
|
||||
text-transform: uppercase;
|
||||
color: var(--muted);
|
||||
}
|
||||
.caption strong { color: var(--fg); font-weight: 500; }
|
||||
|
||||
/* ─── device frame ──────────────────────────────────────────────── */
|
||||
.device {
|
||||
position: relative;
|
||||
width: 390px;
|
||||
height: 844px;
|
||||
border-radius: 56px;
|
||||
padding: 12px;
|
||||
background:
|
||||
linear-gradient(160deg, #2a2a2c 0%, #1a1a1c 50%, #0e0e10 100%);
|
||||
box-shadow:
|
||||
0 0 0 1px rgba(255,255,255,0.04) inset,
|
||||
0 0 0 2px #000 inset,
|
||||
0 28px 60px -12px rgba(0,0,0,0.45),
|
||||
0 8px 20px -8px rgba(0,0,0,0.35);
|
||||
isolation: isolate;
|
||||
}
|
||||
/* metallic side rails */
|
||||
.device::before, .device::after {
|
||||
content: '';
|
||||
position: absolute;
|
||||
width: 3px;
|
||||
background: linear-gradient(to bottom, transparent 0%, rgba(255,255,255,0.06) 8%, transparent 16%, transparent 84%, rgba(255,255,255,0.04) 92%, transparent 100%);
|
||||
top: 100px;
|
||||
bottom: 100px;
|
||||
pointer-events: none;
|
||||
}
|
||||
.device::before { left: -1px; }
|
||||
.device::after { right: -1px; }
|
||||
|
||||
/* Dynamic Island */
|
||||
.island {
|
||||
position: absolute;
|
||||
top: 22px;
|
||||
left: 50%;
|
||||
transform: translateX(-50%);
|
||||
width: 124px;
|
||||
height: 36px;
|
||||
background: #000;
|
||||
border-radius: 999px;
|
||||
z-index: 5;
|
||||
}
|
||||
|
||||
/* hardware buttons (subtle) */
|
||||
.btn-rail {
|
||||
position: absolute;
|
||||
width: 4px;
|
||||
background: #0a0a0c;
|
||||
border-radius: 2px;
|
||||
}
|
||||
.btn-rail.left-1 { left: -3px; top: 174px; height: 32px; } /* silent */
|
||||
.btn-rail.left-2 { left: -3px; top: 220px; height: 60px; } /* vol+ */
|
||||
.btn-rail.left-3 { left: -3px; top: 290px; height: 60px; } /* vol- */
|
||||
.btn-rail.right-1 { right: -3px; top: 250px; height: 100px; } /* power */
|
||||
|
||||
/* ─── screen surface ────────────────────────────────────────────── */
|
||||
.screen {
|
||||
position: relative;
|
||||
width: 100%; height: 100%;
|
||||
background: var(--bg);
|
||||
border-radius: 44px;
|
||||
overflow: hidden;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
/* Status bar — 47px to clear the island. SF-style time, signal/wifi/battery SVG. */
|
||||
.statusbar {
|
||||
flex: 0 0 47px;
|
||||
padding: 18px 26px 0;
|
||||
display: flex;
|
||||
align-items: flex-start;
|
||||
justify-content: space-between;
|
||||
font-family: var(--font-body);
|
||||
font-size: 15px;
|
||||
font-weight: 600;
|
||||
color: var(--fg);
|
||||
letter-spacing: -0.01em;
|
||||
}
|
||||
.statusbar .right { display: inline-flex; align-items: center; gap: 6px; }
|
||||
.statusbar svg { width: 17px; height: 11px; fill: var(--fg); }
|
||||
.statusbar .battery { width: 25px; }
|
||||
|
||||
/* Content region — owns its scroll, frame stays still */
|
||||
.content {
|
||||
flex: 1 1 auto;
|
||||
overflow-y: auto;
|
||||
overflow-x: hidden;
|
||||
-webkit-overflow-scrolling: touch;
|
||||
padding: 8px 0 28px;
|
||||
}
|
||||
.content::-webkit-scrollbar { display: none; }
|
||||
|
||||
/* Home indicator (must always be the last visible thing) */
|
||||
.home-indicator {
|
||||
flex: 0 0 28px;
|
||||
position: relative;
|
||||
}
|
||||
.home-indicator::after {
|
||||
content: '';
|
||||
position: absolute;
|
||||
left: 50%; bottom: 8px;
|
||||
transform: translateX(-50%);
|
||||
width: 134px; height: 5px;
|
||||
background: var(--fg);
|
||||
border-radius: 999px;
|
||||
opacity: 0.85;
|
||||
}
|
||||
|
||||
/* ─── screen primitives — used by layouts.md ────────────────────── */
|
||||
.pad { padding-inline: 20px; }
|
||||
.stack { display: flex; flex-direction: column; gap: 16px; }
|
||||
.row { display: flex; align-items: center; gap: 12px; }
|
||||
.row-between { display: flex; align-items: center; justify-content: space-between; gap: 12px; }
|
||||
.grid-2 { display: grid; grid-template-columns: 1fr 1fr; gap: 10px; }
|
||||
.grid-3 { display: grid; grid-template-columns: repeat(3, 1fr); gap: 10px; }
|
||||
|
||||
.header {
|
||||
padding: 8px 20px 12px;
|
||||
display: flex; align-items: center; justify-content: space-between; gap: 12px;
|
||||
}
|
||||
.header h1 {
|
||||
font-family: var(--font-display);
|
||||
font-size: var(--fs-h1);
|
||||
letter-spacing: -0.02em;
|
||||
line-height: 1.1;
|
||||
margin: 0;
|
||||
}
|
||||
.header .icon-btn {
|
||||
width: 36px; height: 36px;
|
||||
border-radius: 999px;
|
||||
background: var(--surface);
|
||||
border: 1px solid var(--border);
|
||||
display: grid; place-items: center;
|
||||
color: var(--fg);
|
||||
}
|
||||
.header .icon-btn svg { width: 18px; height: 18px; stroke: currentColor; fill: none; stroke-width: 1.7; }
|
||||
|
||||
.greeting {
|
||||
font-family: var(--font-mono);
|
||||
font-size: 11px;
|
||||
letter-spacing: 0.08em;
|
||||
text-transform: uppercase;
|
||||
color: var(--muted);
|
||||
margin: 0 0 4px;
|
||||
}
|
||||
|
||||
.h2 { font-family: var(--font-display); font-size: var(--fs-h2); letter-spacing: -0.015em; line-height: 1.2; margin: 0; }
|
||||
.h3 { font-size: var(--fs-h3); font-weight: 600; line-height: 1.3; margin: 0; }
|
||||
.meta { font-family: var(--font-mono); font-size: var(--fs-meta); color: var(--muted); }
|
||||
.num { font-family: var(--font-mono); font-variant-numeric: tabular-nums; }
|
||||
|
||||
/* card */
|
||||
.card {
|
||||
background: var(--surface);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: var(--radius-card);
|
||||
padding: 16px;
|
||||
}
|
||||
.card.accent {
|
||||
background: var(--accent);
|
||||
color: #fff;
|
||||
border-color: transparent;
|
||||
}
|
||||
.card.accent .meta { color: rgba(255,255,255,0.72); }
|
||||
.card.flat { background: transparent; border: 0; padding: 12px 0; border-top: 1px solid var(--border); border-radius: 0; }
|
||||
.card.flat:first-child { border-top: 0; padding-top: 0; }
|
||||
|
||||
/* list row */
|
||||
.list-row {
|
||||
display: grid;
|
||||
grid-template-columns: 40px 1fr auto;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
padding: 12px 0;
|
||||
border-top: 1px solid var(--border);
|
||||
}
|
||||
.list-row:first-child { border-top: 0; }
|
||||
.list-row .avatar {
|
||||
width: 40px; height: 40px;
|
||||
border-radius: 50%;
|
||||
background:
|
||||
linear-gradient(135deg, var(--accent-soft), var(--fg-soft)),
|
||||
var(--surface);
|
||||
border: 1px solid var(--border);
|
||||
}
|
||||
.list-row .body .title { font-size: 15px; font-weight: 500; line-height: 1.25; }
|
||||
.list-row .body .sub { color: var(--muted); font-size: 13px; line-height: 1.3; margin-top: 2px; }
|
||||
|
||||
/* tab bar */
|
||||
.tabbar {
|
||||
flex: 0 0 auto;
|
||||
display: grid;
|
||||
grid-template-columns: repeat(var(--tabs, 4), 1fr);
|
||||
padding: 8px 8px 0;
|
||||
border-top: 1px solid var(--border);
|
||||
background: color-mix(in oklch, var(--surface) 92%, transparent);
|
||||
backdrop-filter: blur(20px);
|
||||
}
|
||||
.tab {
|
||||
display: flex; flex-direction: column; align-items: center; gap: 2px;
|
||||
padding: 8px 0;
|
||||
color: var(--muted);
|
||||
font-size: 10px;
|
||||
letter-spacing: 0.02em;
|
||||
}
|
||||
.tab.active { color: var(--accent); }
|
||||
.tab svg { width: 22px; height: 22px; stroke: currentColor; fill: none; stroke-width: 1.7; }
|
||||
.tab.active svg { stroke-width: 2; }
|
||||
|
||||
/* primary button — full-width, 48px tap target */
|
||||
.btn-primary {
|
||||
display: flex; align-items: center; justify-content: center;
|
||||
width: 100%;
|
||||
min-height: 48px;
|
||||
padding: 14px 20px;
|
||||
background: var(--accent);
|
||||
color: #fff;
|
||||
border: 0;
|
||||
border-radius: 14px;
|
||||
font: inherit;
|
||||
font-size: 15px;
|
||||
font-weight: 600;
|
||||
letter-spacing: -0.005em;
|
||||
cursor: pointer;
|
||||
}
|
||||
.btn-secondary {
|
||||
display: flex; align-items: center; justify-content: center;
|
||||
width: 100%;
|
||||
min-height: 48px;
|
||||
padding: 14px 20px;
|
||||
background: transparent;
|
||||
color: var(--fg);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 14px;
|
||||
font: inherit;
|
||||
font-size: 15px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
/* image placeholder */
|
||||
.ph-img {
|
||||
background:
|
||||
linear-gradient(135deg, var(--accent-soft), var(--fg-soft)),
|
||||
var(--surface);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 14px;
|
||||
aspect-ratio: 4 / 3;
|
||||
display: grid; place-items: center;
|
||||
color: var(--muted);
|
||||
font-family: var(--font-mono);
|
||||
font-size: 11px;
|
||||
letter-spacing: 0.04em;
|
||||
}
|
||||
.ph-img.square { aspect-ratio: 1 / 1; }
|
||||
.ph-img.wide { aspect-ratio: 16 / 9; }
|
||||
|
||||
/* pill / tag */
|
||||
.pill {
|
||||
display: inline-flex; align-items: center; gap: 4px;
|
||||
padding: 4px 10px;
|
||||
background: var(--accent-soft);
|
||||
color: var(--accent);
|
||||
border-radius: 999px;
|
||||
font-family: var(--font-mono);
|
||||
font-size: 10px;
|
||||
letter-spacing: 0.06em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
.tag {
|
||||
display: inline-flex;
|
||||
padding: 3px 9px;
|
||||
background: transparent;
|
||||
color: var(--muted);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 999px;
|
||||
font-size: 11px;
|
||||
}
|
||||
|
||||
/* progress */
|
||||
.progress { height: 6px; background: rgba(255,255,255,0.25); border-radius: 999px; overflow: hidden; }
|
||||
.progress > span { display: block; height: 100%; background: #fff; }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="stage">
|
||||
<div class="caption"><strong>[REPLACE] App</strong> · [REPLACE] Screen name</div>
|
||||
|
||||
<div class="device" data-od-id="device">
|
||||
<span class="btn-rail left-1" aria-hidden></span>
|
||||
<span class="btn-rail left-2" aria-hidden></span>
|
||||
<span class="btn-rail left-3" aria-hidden></span>
|
||||
<span class="btn-rail right-1" aria-hidden></span>
|
||||
<span class="island" aria-hidden></span>
|
||||
|
||||
<div class="screen">
|
||||
<!-- ─── Status bar ─── -->
|
||||
<div class="statusbar">
|
||||
<span class="num">9:41</span>
|
||||
<span class="right">
|
||||
<!-- signal -->
|
||||
<svg viewBox="0 0 17 11" aria-hidden>
|
||||
<rect x="0" y="7" width="3" height="4" rx="0.6"/>
|
||||
<rect x="4" y="5" width="3" height="6" rx="0.6"/>
|
||||
<rect x="8" y="3" width="3" height="8" rx="0.6"/>
|
||||
<rect x="12" y="0" width="3" height="11" rx="0.6"/>
|
||||
</svg>
|
||||
<!-- wifi -->
|
||||
<svg viewBox="0 0 17 11" aria-hidden>
|
||||
<path d="M8.5 1.5C5.5 1.5 2.7 2.6 0.5 4.6L2 6.1C3.8 4.5 6.1 3.6 8.5 3.6c2.4 0 4.7 0.9 6.5 2.5l1.5-1.5c-2.2-2-5-3.1-8-3.1zM3.5 7.6L5 9.1c1-0.9 2.2-1.4 3.5-1.4 1.3 0 2.5 0.5 3.5 1.4l1.5-1.5c-1.4-1.3-3.1-2-5-2-1.9 0-3.6 0.7-5 2zM6.5 10.6l2 2 2-2c-0.5-0.5-1.2-0.8-2-0.8s-1.5 0.3-2 0.8z"/>
|
||||
</svg>
|
||||
<!-- battery -->
|
||||
<svg class="battery" viewBox="0 0 25 11" aria-hidden>
|
||||
<rect x="0.5" y="0.5" width="21" height="10" rx="2.5" fill="none" stroke="currentColor" stroke-opacity="0.45"/>
|
||||
<rect x="22" y="3.5" width="1.5" height="4" rx="0.4" fill="currentColor" fill-opacity="0.45"/>
|
||||
<rect x="2" y="2" width="18" height="7" rx="1.4"/>
|
||||
</svg>
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<!-- ─── Scrollable content (paste a layout from references/layouts.md HERE) ─── -->
|
||||
<main class="content" data-od-id="content">
|
||||
<div class="header" data-od-id="header">
|
||||
<div>
|
||||
<p class="greeting">Tuesday · April 22</p>
|
||||
<h1>[REPLACE] Hi there.</h1>
|
||||
</div>
|
||||
<button class="icon-btn" aria-label="Settings">
|
||||
<svg viewBox="0 0 24 24"><circle cx="12" cy="12" r="3"/><path d="M19.4 15a1.7 1.7 0 0 0 .3 1.8l.1.1a2 2 0 1 1-2.8 2.8l-.1-.1a1.7 1.7 0 0 0-1.8-.3 1.7 1.7 0 0 0-1 1.5V21a2 2 0 1 1-4 0v-.1a1.7 1.7 0 0 0-1.1-1.5 1.7 1.7 0 0 0-1.8.3l-.1.1a2 2 0 1 1-2.8-2.8l.1-.1a1.7 1.7 0 0 0 .3-1.8 1.7 1.7 0 0 0-1.5-1H3a2 2 0 1 1 0-4h.1a1.7 1.7 0 0 0 1.5-1.1 1.7 1.7 0 0 0-.3-1.8l-.1-.1a2 2 0 1 1 2.8-2.8l.1.1a1.7 1.7 0 0 0 1.8.3H9a1.7 1.7 0 0 0 1-1.5V3a2 2 0 1 1 4 0v.1a1.7 1.7 0 0 0 1 1.5 1.7 1.7 0 0 0 1.8-.3l.1-.1a2 2 0 1 1 2.8 2.8l-.1.1a1.7 1.7 0 0 0-.3 1.8V9a1.7 1.7 0 0 0 1.5 1H21a2 2 0 1 1 0 4h-.1a1.7 1.7 0 0 0-1.5 1z"/></svg>
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div class="pad stack" data-od-id="empty-slot">
|
||||
<div class="card" style="text-align: center; padding: 28px 20px;">
|
||||
<p class="meta" style="margin: 0 0 6px;">PASTE A LAYOUT FROM</p>
|
||||
<p class="h3" style="margin: 0 0 6px;">references/layouts.md</p>
|
||||
<p style="margin: 0; color: var(--muted); font-size: 13px;">into <code style="font-family: var(--font-mono);"><main class="content"></code></p>
|
||||
</div>
|
||||
</div>
|
||||
</main>
|
||||
|
||||
<!-- ─── Tab bar (drop if the screen kind doesn't have one) ─── -->
|
||||
<nav class="tabbar" style="--tabs: 4;" data-od-id="tabbar">
|
||||
<a class="tab active">
|
||||
<svg viewBox="0 0 24 24"><path d="M3 12 12 3l9 9"/><path d="M5 10v10h14V10"/></svg>
|
||||
Home
|
||||
</a>
|
||||
<a class="tab">
|
||||
<svg viewBox="0 0 24 24"><circle cx="11" cy="11" r="7"/><path d="m21 21-4.3-4.3"/></svg>
|
||||
Search
|
||||
</a>
|
||||
<a class="tab">
|
||||
<svg viewBox="0 0 24 24"><path d="M22 12c0 5.5-4.5 10-10 10S2 17.5 2 12 6.5 2 12 2s10 4.5 10 10z"/><path d="M12 6v6l4 2"/></svg>
|
||||
Activity
|
||||
</a>
|
||||
<a class="tab">
|
||||
<svg viewBox="0 0 24 24"><circle cx="12" cy="8" r="4"/><path d="M4 21c0-4 4-7 8-7s8 3 8 7"/></svg>
|
||||
Profile
|
||||
</a>
|
||||
</nav>
|
||||
|
||||
<div class="home-indicator" aria-hidden></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
Reference in New Issue
Block a user