@cyguin/announce
v0.1.0
Published
In-app announcement banner for Next.js
Readme
@cyguin/announce
DB-backed, per-user dismissible announcement banner for Next.js apps.
Install
npm install @cyguin/announceSetup
1. Create the API route
Create app/api/announce/[...cyguin]/route.ts in your Next.js app:
import { createAnnounceHandler } from '@cyguin/announce/next';
import Database from 'better-sqlite3';
import { createSQLiteAdapter } from '@cyguin/announce/adapters/sqlite';
const db = new Database('announce.db');
db.pragma('journal_mode = WAL');
const adapter = createSQLiteAdapter(db);
const handler = createAnnounceHandler({ adapter });
export { handler as GET, handler as POST, handler as DELETE };2. Run migrations
CREATE TABLE IF NOT EXISTS announcements (
id TEXT PRIMARY KEY,
title TEXT NOT NULL,
body TEXT NOT NULL,
active_from INTEGER,
active_until INTEGER,
created_at INTEGER NOT NULL
);
CREATE TABLE IF NOT EXISTS announcement_dismissals (
user_id TEXT NOT NULL,
announcement_id TEXT NOT NULL,
dismissed_at INTEGER NOT NULL,
PRIMARY KEY (user_id, announcement_id)
);3. Add the banner
import { AnnouncementBanner } from '@cyguin/announce/react';
export default function Layout({ children }: { children: React.ReactNode }) {
return (
<>
<AnnouncementBanner
apiBase="/api/announce"
userId={currentUser.id}
theme="light"
/>
{children}
</>
);
}Postgres Setup
import { createAnnounceHandler } from '@cyguin/announce/next';
import postgres from 'postgres';
import { createPostgresAdapter } from '@cyguin/announce/adapters/postgres';
const sql = postgres(process.env.DATABASE_URL!);
const adapter = createPostgresAdapter(sql);
const handler = createAnnounceHandler({ adapter });
export { handler as GET, handler as POST, handler as DELETE };API Routes
| Method | Route | Description |
|--------|-------|-------------|
| GET | /api/announce/cyguin?userId=xxx | List active announcements for user |
| POST | /api/announce/cyguin | Create announcement |
| DELETE | /api/announce/cyguin/:id | Delete announcement |
| POST | /api/announce/cyguin/:id/dismiss?userId=xxx | Dismiss for user |
Theming
Use --cyguin-* CSS custom properties on .cyguin-announce-banner:
.cyguin-announce-banner {
--cyguin-bg: #ffffff;
--cyguin-fg: #0a0a0a;
--cyguin-accent: #f5a800;
--cyguin-border: #e5e5e5;
--cyguin-radius: 6px;
--cyguin-shadow: 0 1px 4px rgba(0,0,0,0.08);
}Switch to dark theme with the theme prop:
<AnnouncementBanner theme="dark" ... />Dark theme applies these overrides automatically:
--cyguin-bg: #0a0a0a;
--cyguin-bg-subtle: #1a1a1a;
--cyguin-border: #2a2a2a;
--cyguin-fg: #f5f5f5;
--cyguin-shadow: 0 1px 4px rgba(0,0,0,0.4);Configuration
createAnnounceHandler({
adapter,
options: {
defaultActiveDays: 7, // auto-expire after N days
maxActive: 3, // max simultaneous active announcements
},
})License
MIT
