Files
claudemesh/apps/web/src/modules/marketing/home/toaster.tsx
Alejandro Gutiérrez 4c057be069
Some checks failed
CI / Lint (push) Has been cancelled
CI / Typecheck (push) Has been cancelled
CI / Broker tests (Postgres) (push) Has been cancelled
CI / Docker build (linux/amd64) (push) Has been cancelled
fix(web): re-apply all landing page content fixes (linter reverted)
A linter/formatter reverted our content edits. Re-applying:
- Hero: concrete claims, no WhatsApp/Slack promises, beta pricing
- Logo bar: tech stack instead of fake customer logos
- Pricing: single honest Public Beta tier (removed $12/$24/$99)
- FAQ: real install flow, honest pricing language
- Features: claudemesh.com/install URL
- Toaster: v0.1.4 announcement
- Copy: "volunteers" / "shares" instead of jargon
- Links: #docs → GitHub README, claudemesh.sh → claudemesh.com

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-06 10:02:44 +01:00

149 lines
6.0 KiB
TypeScript

"use client";
import { useState } from "react";
import Link from "next/link";
const NEWS = [
{
tag: "New",
title: "claudemesh launch (v0.1.4)",
body: "Real-time peer messages pushed into Claude Code mid-turn. One command. Source open at github.com/alezmad/claudemesh-cli.",
href: "https://github.com/alezmad/claudemesh-cli",
},
{
tag: "Beta",
title: "Mesh Dashboard",
body: "Watch every Claude Code session on your team. Routes, presence, priority — all live.",
href: "#",
},
{
tag: "New",
title: "MCP bridge",
body: "Expose mesh messages as MCP tools. Your agent can message peers without leaving its context.",
href: "#",
},
{
tag: "Launch",
title: "Self-hosted broker",
body: "One binary. SQLite-backed. Runs on a Pi. Your mesh, never the cloud's.",
href: "#",
},
];
export const LatestNewsToaster = () => {
const [index, setIndex] = useState(0);
const [hidden, setHidden] = useState(false);
if (hidden) return null;
const item = NEWS[index];
if (!item) return null;
return (
<div
className="fixed bottom-6 right-6 z-[100] hidden w-[384px] rounded-[12px] border border-[var(--cm-border)] bg-[var(--cm-bg)] p-5 shadow-[0_20px_60px_rgba(0,0,0,0.45)] md:block"
role="complementary"
aria-label="Latest news"
>
{/* head */}
<div className="mb-4 flex items-center justify-between">
<div
className="flex items-center gap-1.5 text-xs text-[var(--cm-fg-tertiary)]"
style={{ fontFamily: "var(--cm-font-sans)" }}
>
<svg width="14" height="14" viewBox="0 0 16 16" fill="none">
<path
d="M8 2C11.3137 2 14 4.68629 14 8C14 11.3137 11.3137 14 8 14C4.68629 14 2 11.3137 2 8C2 4.68629 4.68629 2 8 2ZM8 2.8C5.12812 2.8 2.8 5.12812 2.8 8C2.8 10.8719 5.12812 13.2 8 13.2C10.8719 13.2 13.2 10.8719 13.2 8C13.2 5.12812 10.8719 2.8 8 2.8ZM8.4 7.6V10H9.2C9.42091 10 9.6 10.1791 9.6 10.4C9.6 10.6209 9.42091 10.8 9.2 10.8H6.8C6.57909 10.8 6.4 10.6209 6.4 10.4C6.4 10.1791 6.57909 10 6.8 10H7.6V8H6.8C6.57909 8 6.4 7.82091 6.4 7.6C6.4 7.37909 6.57909 7.2 6.8 7.2H8C8.22091 7.2 8.4 7.37909 8.4 7.6ZM8 5.2C8.33137 5.2 8.6 5.46863 8.6 5.8C8.6 6.13137 8.33137 6.4 8 6.4C7.66863 6.4 7.4 6.13137 7.4 5.8C7.4 5.46863 7.66863 5.2 8 5.2Z"
fill="currentColor"
/>
</svg>
Latest news
</div>
<button
onClick={() => setHidden(true)}
className="rounded p-1 text-[var(--cm-fg-tertiary)] transition-colors hover:bg-[var(--cm-bg-elevated)] hover:text-[var(--cm-fg)]"
aria-label="Close"
>
<svg width="16" height="16" viewBox="0 0 20 20" fill="none">
<path
d="M15.15 4.15a.5.5 0 01.7.7L10.71 10l5.14 5.15a.5.5 0 01-.7.7L10 10.71l-5.15 5.14a.5.5 0 01-.7-.7L9.29 10 4.15 4.85a.5.5 0 01.7-.7L10 9.29l5.15-5.14z"
fill="currentColor"
/>
</svg>
</button>
</div>
{/* body */}
<div className="grid grid-cols-[1fr_108px] gap-4">
<div>
<h4
className="mb-2 text-[22px] font-medium leading-tight text-[var(--cm-fg)]"
style={{ fontFamily: "var(--cm-font-serif)" }}
>
{item.title}
</h4>
<p
className="mb-4 text-[12px] leading-[1.5] text-[var(--cm-fg-secondary)]"
style={{ fontFamily: "var(--cm-font-mono)" }}
>
{item.body}
</p>
<Link
href={item.href}
className="inline-flex items-center gap-1.5 rounded-[6px] bg-[var(--cm-fg)] px-3 py-1.5 text-[12px] font-medium text-[var(--cm-bg)] transition-colors hover:bg-[var(--cm-gray-150)]"
style={{ fontFamily: "var(--cm-font-sans)" }}
>
Learn more
</Link>
</div>
{/* illustration tile */}
<div className="flex h-[108px] w-[108px] items-center justify-center rounded-[8px] bg-[var(--cm-clay)]">
<svg width="68" height="68" viewBox="0 0 68 68" fill="none">
<circle cx="20" cy="20" r="4" fill="#141413" />
<circle cx="48" cy="16" r="4" fill="#141413" />
<circle cx="52" cy="40" r="4" fill="#141413" />
<circle cx="24" cy="44" r="4" fill="#141413" />
<path
d="M20 20L48 16L52 40L24 44L20 20z"
stroke="#141413"
strokeWidth="1.5"
fill="none"
/>
<path
d="M10 56c6-4 12-4 20 0s14 4 24-2"
stroke="#141413"
strokeWidth="1.5"
fill="none"
strokeLinecap="round"
/>
</svg>
</div>
</div>
{/* pager */}
<div className="mt-4 flex items-center justify-between border-t border-[var(--cm-border)] pt-3">
<div
className="text-[10px] uppercase tracking-wider text-[var(--cm-fg-tertiary)]"
style={{ fontFamily: "var(--cm-font-mono)" }}
>
{String(index + 1).padStart(2, "0")} / {String(NEWS.length).padStart(2, "0")}
</div>
<div className="flex gap-1">
<button
onClick={() =>
setIndex((i) => (i - 1 + NEWS.length) % NEWS.length)
}
className="rounded border border-[var(--cm-border)] px-2 py-1 text-xs text-[var(--cm-fg-secondary)] transition-colors hover:border-[var(--cm-fg)] hover:text-[var(--cm-fg)]"
style={{ fontFamily: "var(--cm-font-mono)" }}
aria-label="Previous"
>
</button>
<button
onClick={() => setIndex((i) => (i + 1) % NEWS.length)}
className="rounded border border-[var(--cm-border)] px-2 py-1 text-xs text-[var(--cm-fg-secondary)] transition-colors hover:border-[var(--cm-fg)] hover:text-[var(--cm-fg)]"
style={{ fontFamily: "var(--cm-font-mono)" }}
aria-label="Next"
>
</button>
</div>
</div>
</div>
);
};