mirror of
https://github.com/different-ai/openwork
synced 2026-05-01 20:07:11 +02:00
549 lines
18 KiB
TypeScript
549 lines
18 KiB
TypeScript
import {
|
|
OPENWORK_DOWNLOAD_URL,
|
|
SHARE_EASE,
|
|
buildBundleNarrative,
|
|
buildBundleUrls,
|
|
buildOgImageUrl,
|
|
buildOpenInAppUrls,
|
|
collectBundleItems,
|
|
escapeHtml,
|
|
escapeJsonForScript,
|
|
getBundleCounts,
|
|
humanizeType,
|
|
parseBundle,
|
|
wantsDownload,
|
|
wantsJsonResponse,
|
|
} from "../_lib/share-utils.ts";
|
|
import type { RequestLike } from "../_lib/types.ts";
|
|
import type { PreviewItem } from "../../components/share-home-types.ts";
|
|
|
|
export { buildBundleUrls, wantsDownload, wantsJsonResponse } from "../_lib/share-utils.ts";
|
|
|
|
function toneInitial(kind: string): string {
|
|
if (kind === "Config") return "config";
|
|
if (kind === "MCP") return "mcp";
|
|
if (kind === "Command") return "command";
|
|
if (kind === "Agent") return "agent";
|
|
return "skill";
|
|
}
|
|
|
|
function renderItem(item: PreviewItem): string {
|
|
return `
|
|
<div class="included-item">
|
|
<div class="item-left">
|
|
<div class="item-dot dot-${escapeHtml(toneInitial(item.kind))}"></div>
|
|
<span class="item-title">${escapeHtml(item.name)}</span>
|
|
</div>
|
|
<span class="item-meta">${escapeHtml(item.kind)} · ${escapeHtml(item.meta)}</span>
|
|
</div>`;
|
|
}
|
|
|
|
export function renderBundlePage({ id, rawJson, req }: { id: string; rawJson: string; req: RequestLike }): string {
|
|
const bundle = parseBundle(rawJson);
|
|
const urls = buildBundleUrls(req, id);
|
|
const ogImageUrl = buildOgImageUrl(req, id);
|
|
const { openInAppDeepLink, openInWebAppUrl } = buildOpenInAppUrls(urls.shareUrl, {
|
|
label: bundle.name || "Shared worker package",
|
|
});
|
|
|
|
const counts = getBundleCounts(bundle);
|
|
const schemaVersion = bundle.schemaVersion == null ? "unknown" : String(bundle.schemaVersion);
|
|
const typeLabel = humanizeType(bundle.type);
|
|
const title = bundle.name || `OpenWork ${typeLabel}`;
|
|
const description = bundle.description || buildBundleNarrative(bundle);
|
|
const items = collectBundleItems(bundle, 8);
|
|
const installHint =
|
|
bundle.type === "skill"
|
|
? "Open in app to choose where to add this skill."
|
|
: bundle.type === "skills-set"
|
|
? "Open in app to create a new worker with this entire skills set already attached."
|
|
: "Open in app to create a new worker with these skills, agents, MCPs, and config already bundled.";
|
|
const metadataRows = [
|
|
["ID", id],
|
|
["Type", bundle.type || "unknown"],
|
|
["Schema", schemaVersion],
|
|
counts.skillCount ? ["Skills", String(counts.skillCount)] : null,
|
|
counts.agentCount ? ["Agents", String(counts.agentCount)] : null,
|
|
counts.mcpCount ? ["MCPs", String(counts.mcpCount)] : null,
|
|
counts.commandCount ? ["Commands", String(counts.commandCount)] : null,
|
|
counts.configCount ? ["Configs", String(counts.configCount)] : null,
|
|
]
|
|
.filter((row): row is [string, string] => row !== null)
|
|
.map(
|
|
([label, value]) =>
|
|
`<div class="metadata-row"><dt>${escapeHtml(label)}</dt><dd>${escapeHtml(value)}</dd></div>`,
|
|
)
|
|
.join("");
|
|
|
|
return `<!doctype html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="utf-8" />
|
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
<title>${escapeHtml(title)} - OpenWork Share</title>
|
|
<meta name="description" content="${escapeHtml(description)}" />
|
|
<meta name="openwork:bundle-id" content="${escapeHtml(id)}" />
|
|
<meta name="openwork:bundle-type" content="${escapeHtml(bundle.type || "unknown")}" />
|
|
<meta name="openwork:schema-version" content="${escapeHtml(schemaVersion)}" />
|
|
<meta name="openwork:open-in-app-url" content="${escapeHtml(openInAppDeepLink)}" />
|
|
<link rel="canonical" href="${escapeHtml(urls.shareUrl)}" />
|
|
<link rel="alternate" type="application/json" href="${escapeHtml(urls.jsonUrl)}" />
|
|
<meta property="og:type" content="website" />
|
|
<meta property="og:title" content="${escapeHtml(title)}" />
|
|
<meta property="og:description" content="${escapeHtml(description)}" />
|
|
<meta property="og:url" content="${escapeHtml(urls.shareUrl)}" />
|
|
<meta property="og:image" content="${escapeHtml(ogImageUrl)}" />
|
|
<meta name="twitter:card" content="summary_large_image" />
|
|
<meta name="twitter:title" content="${escapeHtml(title)}" />
|
|
<meta name="twitter:description" content="${escapeHtml(description)}" />
|
|
<meta name="twitter:image" content="${escapeHtml(ogImageUrl)}" />
|
|
<style>
|
|
@font-face {
|
|
font-family: "FK Raster Roman Compact Smooth";
|
|
src: url("https://openwork.software/fonts/FKRasterRomanCompact-Smooth.woff2") format("woff2");
|
|
font-weight: 400;
|
|
font-style: normal;
|
|
font-display: swap;
|
|
}
|
|
:root {
|
|
color-scheme: light;
|
|
--ow-bg: #f6f9fc;
|
|
--ow-ink: #011627;
|
|
--ow-muted: #5f6b7a;
|
|
--ow-card: #ffffff;
|
|
--ow-border: rgba(148, 163, 184, 0.16);
|
|
--ow-shadow: 0 20px 60px -24px rgba(15, 23, 42, 0.18);
|
|
--ow-primary: #011627;
|
|
--ow-ease: ${SHARE_EASE};
|
|
--ow-sans: Inter, "Segoe UI", "Helvetica Neue", sans-serif;
|
|
--ow-accent: "FK Raster Roman Compact Smooth", "Iowan Old Style", "Palatino Linotype", "Book Antiqua", Georgia, serif;
|
|
}
|
|
|
|
* { box-sizing: border-box; }
|
|
[hidden] { display: none !important; }
|
|
html, body { min-height: 100%; }
|
|
body {
|
|
margin: 0;
|
|
font-family: var(--ow-sans);
|
|
color: var(--ow-ink);
|
|
background-color: var(--ow-bg);
|
|
overflow-x: hidden;
|
|
-webkit-font-smoothing: antialiased;
|
|
-moz-osx-font-smoothing: grayscale;
|
|
}
|
|
|
|
body::after {
|
|
content: "";
|
|
position: absolute;
|
|
top: 0;
|
|
right: 0;
|
|
width: 60vw;
|
|
height: 80vh;
|
|
background: radial-gradient(circle at 70% 30%, rgba(100, 116, 139, 0.25) 0%, transparent 60%);
|
|
filter: blur(60px);
|
|
z-index: 0;
|
|
pointer-events: none;
|
|
}
|
|
|
|
a { color: inherit; }
|
|
|
|
.shell {
|
|
position: relative;
|
|
z-index: 10;
|
|
width: min(100%, 1024px);
|
|
margin: 0 auto;
|
|
padding: 8px 32px 64px;
|
|
}
|
|
|
|
.nav {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
height: 80px;
|
|
margin-bottom: 40px;
|
|
}
|
|
|
|
.brand {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
gap: 12px;
|
|
text-decoration: none;
|
|
font-weight: 600;
|
|
font-size: 20px;
|
|
letter-spacing: -0.02em;
|
|
color: var(--ow-ink);
|
|
}
|
|
|
|
.brand-mark {
|
|
width: 24px;
|
|
height: 24px;
|
|
background: url('data:image/svg+xml;utf8,<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5" stroke="%23011627" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></svg>') no-repeat center center;
|
|
}
|
|
|
|
.nav-links { display: none; }
|
|
@media (min-width: 768px) {
|
|
.nav-links {
|
|
display: flex;
|
|
gap: 32px;
|
|
font-size: 15px;
|
|
color: var(--ow-muted);
|
|
font-weight: 500;
|
|
}
|
|
.nav-links a { text-decoration: none; transition: color 0.2s; }
|
|
.nav-links a:hover { color: var(--ow-ink); }
|
|
}
|
|
|
|
.nav-actions { display: flex; align-items: center; gap: 12px; }
|
|
|
|
.button-primary {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
min-height: 48px;
|
|
padding: 0 24px;
|
|
border-radius: 999px;
|
|
border: none;
|
|
cursor: pointer;
|
|
text-decoration: none;
|
|
color: #fff;
|
|
background: var(--ow-primary);
|
|
box-shadow: 0 14px 32px -16px rgba(1, 22, 39, 0.55);
|
|
font-family: inherit;
|
|
font-weight: 500;
|
|
font-size: 16px;
|
|
transition: all 300ms var(--ow-ease);
|
|
will-change: transform, background-color, box-shadow;
|
|
}
|
|
|
|
.button-primary:hover {
|
|
background: rgb(110, 110, 110);
|
|
transform: translateY(-1px);
|
|
box-shadow:
|
|
rgba(0, 0, 0, 0.06) 0px 0px 0px 1px,
|
|
rgba(0, 0, 0, 0.04) 0px 1px 2px 0px,
|
|
rgba(0, 0, 0, 0.04) 0px 2px 4px 0px;
|
|
}
|
|
|
|
.button-secondary {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
min-height: 48px;
|
|
padding: 0 24px;
|
|
border-radius: 999px;
|
|
text-decoration: none;
|
|
background: rgb(255, 255, 255);
|
|
color: rgb(0, 0, 0);
|
|
border: none;
|
|
box-shadow:
|
|
rgba(0, 0, 0, 0.06) 0px 0px 0px 1px,
|
|
rgba(0, 0, 0, 0.04) 0px 1px 2px 0px;
|
|
font-family: inherit;
|
|
font-weight: 500;
|
|
font-size: 16px;
|
|
transition: all 300ms var(--ow-ease);
|
|
cursor: pointer;
|
|
will-change: transform, background-color, box-shadow;
|
|
}
|
|
|
|
.button-secondary:hover {
|
|
background: rgb(242, 242, 242);
|
|
box-shadow:
|
|
rgba(0, 0, 0, 0.06) 0px 0px 0px 1px,
|
|
rgba(0, 0, 0, 0.04) 0px 1px 2px 0px,
|
|
rgba(0, 0, 0, 0.04) 0px 2px 4px 0px;
|
|
}
|
|
|
|
.hero-layout {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 64px;
|
|
}
|
|
@media (min-width: 1024px) {
|
|
.hero-layout { flex-direction: row; align-items: flex-start; }
|
|
}
|
|
|
|
.hero-copy {
|
|
flex: 1.1;
|
|
max-width: 600px;
|
|
}
|
|
|
|
h1 {
|
|
margin: 0 0 24px 0;
|
|
font-size: clamp(3rem, 5.5vw, 4.5rem);
|
|
line-height: 1.1;
|
|
letter-spacing: -0.04em;
|
|
font-weight: 500;
|
|
color: var(--ow-ink);
|
|
}
|
|
|
|
h1 em {
|
|
font-style: normal;
|
|
font-family: var(--ow-accent);
|
|
font-weight: 400;
|
|
font-size: 1.05em;
|
|
display: inline-block;
|
|
vertical-align: baseline;
|
|
}
|
|
|
|
.hero-body {
|
|
margin: 0 0 32px 0;
|
|
font-size: 20px;
|
|
line-height: 1.6;
|
|
color: #374151;
|
|
max-width: 500px;
|
|
}
|
|
|
|
.hero-actions {
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
gap: 16px;
|
|
align-items: center;
|
|
}
|
|
|
|
.hero-artifact {
|
|
flex: 0.9;
|
|
width: 100%;
|
|
}
|
|
|
|
.app-window {
|
|
background: rgba(255, 255, 255, 0.95);
|
|
border: 1px solid rgba(255, 255, 255, 0.6);
|
|
border-radius: 1.5rem;
|
|
box-shadow: 0 20px 50px -24px rgba(15, 23, 42, 0.12);
|
|
overflow: hidden;
|
|
display: flex;
|
|
flex-direction: column;
|
|
width: 100%;
|
|
}
|
|
|
|
.app-window-header {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
padding: 12px 16px;
|
|
background: linear-gradient(to bottom, rgba(255,255,255,0.9), rgba(255,255,255,0.6));
|
|
border-bottom: 1px solid rgba(255,255,255,0.5);
|
|
position: relative;
|
|
}
|
|
|
|
.mac-dots {
|
|
position: absolute;
|
|
left: 16px;
|
|
display: flex;
|
|
gap: 6px;
|
|
}
|
|
.mac-dot { width: 12px; height: 12px; border-radius: 50%; }
|
|
.mac-dot.red { background: #ff5f56; border: 1px solid rgba(224, 68, 62, 0.2); }
|
|
.mac-dot.yellow { background: #ffbd2e; border: 1px solid rgba(222, 161, 35, 0.2); }
|
|
.mac-dot.green { background: #27c93f; border: 1px solid rgba(26, 171, 41, 0.2); }
|
|
|
|
.app-window-title {
|
|
font-size: 12px;
|
|
font-weight: 500;
|
|
color: var(--ow-muted);
|
|
letter-spacing: 0.02em;
|
|
}
|
|
|
|
.app-window-body {
|
|
padding: 24px;
|
|
background: #ffffff;
|
|
}
|
|
|
|
.included-section {
|
|
width: 100%;
|
|
}
|
|
.included-section h4 {
|
|
margin: 0 0 12px 0;
|
|
font-size: 12px;
|
|
font-weight: 600;
|
|
text-transform: uppercase;
|
|
letter-spacing: 0.05em;
|
|
color: var(--ow-muted);
|
|
}
|
|
.included-list {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 8px;
|
|
}
|
|
.included-item {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
padding: 12px 16px;
|
|
background: #f8fafc;
|
|
border: 1px solid rgba(148, 163, 184, 0.15);
|
|
border-radius: 12px;
|
|
}
|
|
.item-left { display: flex; align-items: center; gap: 12px; }
|
|
.item-dot { width: 24px; height: 24px; border-radius: 50%; }
|
|
.dot-agent { background: #f97316; }
|
|
.dot-skill { background: #2463eb; }
|
|
.dot-mcp { background: #0f9f7f; }
|
|
.dot-command { background: #8b5cf6; }
|
|
.dot-config { background: #475569; }
|
|
|
|
.item-title { font-size: 14px; font-weight: 500; color: var(--ow-ink); }
|
|
.item-meta { font-size: 12px; color: var(--ow-muted); }
|
|
|
|
.results-grid {
|
|
display: grid;
|
|
grid-template-columns: 1fr;
|
|
gap: 24px;
|
|
margin-top: 64px;
|
|
}
|
|
@media (min-width: 768px) {
|
|
.results-grid { grid-template-columns: 1fr 1fr; }
|
|
}
|
|
|
|
.result-card {
|
|
background: #ffffff;
|
|
border: 1px solid var(--ow-border);
|
|
border-radius: 1.5rem;
|
|
padding: 32px;
|
|
box-shadow: var(--ow-shadow);
|
|
}
|
|
.result-card h3 { margin: 0 0 8px 0; font-size: 20px; font-weight: 500; }
|
|
.result-card p { margin: 0 0 24px 0; font-size: 15px; color: var(--ow-muted); line-height: 1.6; }
|
|
|
|
.url-box {
|
|
background: #f8fafc;
|
|
border: 1px solid rgba(148, 163, 184, 0.2);
|
|
border-radius: 12px;
|
|
padding: 16px;
|
|
font-family: ui-monospace, monospace;
|
|
font-size: 13px;
|
|
color: var(--ow-ink);
|
|
word-break: break-all;
|
|
margin-bottom: 16px;
|
|
}
|
|
|
|
.metadata-list {
|
|
padding: 0;
|
|
margin: 0;
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 12px;
|
|
}
|
|
.metadata-row {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: baseline;
|
|
gap: 12px;
|
|
font-size: 13px;
|
|
}
|
|
.metadata-list dt { color: var(--ow-muted); }
|
|
.metadata-list dd { margin: 0; color: var(--ow-ink); font-weight: 500; }
|
|
</style>
|
|
</head>
|
|
<body
|
|
data-openwork-share="true"
|
|
data-openwork-bundle-id="${escapeHtml(id)}"
|
|
data-openwork-bundle-type="${escapeHtml(bundle.type || "unknown")}"
|
|
data-openwork-schema-version="${escapeHtml(schemaVersion)}"
|
|
>
|
|
<main class="shell">
|
|
<nav class="nav">
|
|
<a class="brand" href="/" aria-label="OpenWork Share home">
|
|
<span class="brand-mark" aria-hidden="true"></span>
|
|
<span>openwork</span>
|
|
</a>
|
|
<div class="nav-links">
|
|
<a href="https://openwork.software/docs" target="_blank" rel="noreferrer">Docs</a>
|
|
<a href="${escapeHtml(OPENWORK_DOWNLOAD_URL)}" target="_blank" rel="noreferrer">Download</a>
|
|
<a href="https://openwork.software/enterprise" target="_blank" rel="noreferrer">Enterprise</a>
|
|
</div>
|
|
<div class="nav-actions">
|
|
<a class="button-secondary" href="https://github.com/different-ai/openwork" target="_blank" rel="noreferrer">
|
|
<svg width="20" height="20" viewBox="0 0 24 24" fill="currentColor"><path d="M12 2C6.477 2 2 6.484 2 12.017c0 4.425 2.865 8.18 6.839 9.504.5.092.682-.217.682-.483 0-.237-.008-.868-.013-1.703-2.782.605-3.369-1.343-3.369-1.343-.454-1.158-1.11-1.466-1.11-1.466-.908-.62.069-.608.069-.608 1.003.07 1.531 1.032 1.531 1.032.892 1.53 2.341 1.088 2.91.832.092-.647.35-1.088.636-1.338-2.22-.253-4.555-1.113-4.555-4.951 0-1.093.39-1.988 1.029-2.688-.103-.253-.446-1.272.098-2.65 0 0 .84-.27 2.75 1.026A9.564 9.564 0 0112 6.844c.85.004 1.705.115 2.504.337 1.909-1.296 2.747-1.027 2.747-1.027.546 1.379.202 2.398.1 2.651.64.7 1.028 1.595 1.028 2.688 0 3.848-2.339 4.695-4.566 4.943.359.309.678.92.678 1.855 0 1.338-.012 2.419-.012 2.747 0 .268.18.58.688.482A10.019 10.019 0 0022 12.017C22 6.484 17.522 2 12 2z"/></svg>
|
|
GitHub
|
|
</a>
|
|
</div>
|
|
</nav>
|
|
|
|
<section class="hero-layout">
|
|
<div class="hero-copy">
|
|
<h1>${escapeHtml(title)} <em>ready</em></h1>
|
|
<p class="hero-body">${escapeHtml(description)}</p>
|
|
<div class="hero-actions">
|
|
<a class="button-primary" href="${escapeHtml(openInAppDeepLink)}">Open in app</a>
|
|
<a class="button-secondary" href="${escapeHtml(openInWebAppUrl)}" target="_blank" rel="noreferrer">Open in web app</a>
|
|
</div>
|
|
<p style="margin-top: 16px; font-size: 13px; color: var(--ow-muted);">
|
|
${escapeHtml(installHint)}
|
|
</p>
|
|
</div>
|
|
|
|
<div class="hero-artifact">
|
|
<div class="app-window">
|
|
<div class="app-window-header">
|
|
<div class="mac-dots">
|
|
<div class="mac-dot red"></div>
|
|
<div class="mac-dot yellow"></div>
|
|
<div class="mac-dot green"></div>
|
|
</div>
|
|
<div class="app-window-title">OpenWork</div>
|
|
</div>
|
|
<div class="app-window-body">
|
|
<div class="included-section">
|
|
<h4>Package Contents</h4>
|
|
<div class="included-list">
|
|
${items.length ? items.map(renderItem).join("") : `<div class="included-item"><div class="item-left"><div class="item-dot dot-skill"></div><span class="item-title">OpenWork bundle</span></div><span class="item-meta">Shared config</span></div>`}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
|
|
<section class="results-grid">
|
|
<div class="result-card">
|
|
<h3>Bundle details</h3>
|
|
<p>Stable metadata for parsing and direct OpenWork import.</p>
|
|
<dl class="metadata-list">
|
|
${metadataRows}
|
|
</dl>
|
|
</div>
|
|
<div class="result-card">
|
|
<h3>Raw endpoints</h3>
|
|
<p>Keep the human page and machine payload side by side.</p>
|
|
<div class="url-box"><a href="${escapeHtml(urls.jsonUrl)}">JSON payload</a></div>
|
|
<div style="display: flex; gap: 12px;">
|
|
<a class="button-secondary" href="${escapeHtml(urls.downloadUrl)}">Download JSON</a>
|
|
<button class="button-secondary" id="copy-link" type="button">Copy share link</button>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
|
|
</main>
|
|
|
|
<script id="openwork-bundle-json" type="application/json">${escapeJsonForScript(rawJson)}</script>
|
|
<script>
|
|
const shareUrl = ${JSON.stringify(urls.shareUrl)};
|
|
const copyButton = document.getElementById("copy-link");
|
|
copyButton?.addEventListener("click", async () => {
|
|
try {
|
|
if (navigator.clipboard?.writeText) {
|
|
await navigator.clipboard.writeText(shareUrl);
|
|
copyButton.textContent = "Copied!";
|
|
setTimeout(() => copyButton.textContent = "Copy share link", 2000);
|
|
return;
|
|
}
|
|
} catch {}
|
|
|
|
const input = document.createElement("textarea");
|
|
input.value = shareUrl;
|
|
input.style.position = "fixed";
|
|
input.style.left = "-99999px";
|
|
document.body.appendChild(input);
|
|
input.select();
|
|
document.execCommand("copy");
|
|
input.remove();
|
|
copyButton.textContent = "Copied!";
|
|
setTimeout(() => copyButton.textContent = "Copy share link", 2000);
|
|
});
|
|
</script>
|
|
</body>
|
|
</html>`;
|
|
}
|