Prevents users who upgrade across multiple versions from seeing all interim notices at once. Version bounds are evaluated server-side using semver.coerce so prerelease builds compare as their base release. Range is lower-inclusive, upper-exclusive: maxVersion: '4.0.0' hides the notice once 4.0.0 ships.
27 KiB
System Notices — Technical Documentation & Dev Guide
System notices are server-evaluated, user-targeted messages shown in the TREK UI as modals, banners, or toasts. They are used for onboarding, upgrade announcements, breaking change warnings, and time-boxed campaigns. Every aspect — targeting, display, copy, and dismissal — is controlled from one place: the server-side registry.
Table of Contents
- Architecture overview
- Data flow
- Database schema
- The notice registry
- Notice fields reference
- Condition system
- Display types
- CTAs (call to action)
- i18n — translation keys
- Client store & dismissal
- Sorting & priority
- How-to recipes
- Testing
- Rules & constraints
1. Architecture overview
server/src/systemNotices/
├── types.ts — TypeScript types (SystemNotice, NoticeCondition, …)
├── registry.ts — Authoritative list of all notices (edit here to add/change/remove)
├── conditions.ts — Condition evaluators + custom predicate registry
└── service.ts — Queries DB, evaluates conditions, sorts, strips server-only fields
server/src/routes/systemNotices.ts — REST endpoints
client/src/store/systemNoticeStore.ts — Zustand store (fetch + optimistic dismiss)
client/src/components/SystemNotices/
├── SystemNoticeHost.tsx — Renders all three channels (modal / banner / toast)
├── SystemNoticeModal.tsx — Modal renderer (pager, animations, keyboard nav)
├── SystemNoticeBanner.tsx — Banner + toast renderers
└── noticeActions.ts — Client-side action registry for action-kind CTAs
client/src/pages/Trips/noticeActions.ts — Example domain action registration
There are no database rows for notice definitions. The registry is code-only. The database only stores which notices a user has dismissed.
2. Data flow
1. User authenticates
│
▼
2. authStore.loadUser() completes
│
▼
3. SystemNoticeHost mounts → calls useSystemNoticeStore.fetch()
│ (also triggered on cold page reload if store not yet loaded)
▼
4. GET /api/system-notices/active
│
▼
5. service.getActiveNoticesFor(userId)
├── reads user row (login_count, first_seen_version, role)
├── counts user trips
├── reads user_notice_dismissals
├── filters SYSTEM_NOTICES:
│ – not dismissed
│ – within [minVersion, maxVersion] range for the running app version
│ – all conditions pass (AND logic)
├── sorts by priority → severity → publishedAt (desc)
└── strips server-only fields (conditions, publishedAt, minVersion, maxVersion, priority)
│
▼
6. Client receives SystemNoticeDTO[]
│
▼
7. SystemNoticeHost partitions by display type
├── modal → ModalRenderer (multi-page pager, slide transitions)
├── banner → BannerRenderer (sticky top bar, max 2)
└── toast → ToastRenderer (fires window.__addToast, auto-dismisses)
│
▼
8. User dismisses → POST /api/system-notices/:id/dismiss
├── Server: INSERT OR IGNORE into user_notice_dismissals
└── Client: optimistic remove from store (retry once on failure)
3. Database schema
Added in migration 101 (server/src/db/migrations.ts).
users columns (added by migration 101)
| Column | Type | Default | Purpose |
|---|---|---|---|
first_seen_version |
TEXT |
'0.0.0' |
App version at account creation. Used by existingUserBeforeVersion condition. Backfilled users get '0.0.0'. |
login_count |
INTEGER |
0 |
Incremented on each successful login. Used by firstLogin condition. |
user_notice_dismissals
| Column | Type | Notes |
|---|---|---|
user_id |
INTEGER |
FK → users.id CASCADE DELETE |
notice_id |
TEXT |
Matches SystemNotice.id from registry |
dismissed_at |
INTEGER |
Unix ms timestamp |
Primary key: (user_id, notice_id) — dismissals are idempotent.
4. The notice registry
server/src/systemNotices/registry.ts is the single source of truth. Add, change, or retire notices here.
export const SYSTEM_NOTICES: SystemNotice[] = [
{
id: 'my-notice', // ← globally unique, never reuse
display: 'modal',
severity: 'info',
titleKey: 'system_notice.my_notice.title',
bodyKey: 'system_notice.my_notice.body',
dismissible: true,
conditions: [{ kind: 'firstLogin' }],
publishedAt: '2026-05-01T00:00:00Z',
priority: 50,
},
];
The golden rule for IDs
Never remove or renumber an entry. Never reuse an ID.
Dismissals are stored in the database keyed by id. Removing an entry means dismissed users would see it again if you ever add a notice with the same ID. If a notice is no longer needed, set maxVersion to the last app version on which it should appear — do not delete the entry.
5. Notice fields reference
Required fields
| Field | Type | Description |
|---|---|---|
id |
string |
Globally unique, stable identifier. Use kebab-case, descriptive, version-scoped when appropriate (v3-photos, welcome-v1). Max recommended length: 40 chars. |
display |
'modal' | 'banner' | 'toast' |
How the notice is rendered. See §7 Display types. |
severity |
'info' | 'warn' | 'critical' |
Affects colour scheme and accessibility role. critical notices cannot be toasts. |
titleKey |
string |
i18n key for the title. |
bodyKey |
string |
i18n key for the body. Markdown supported in modals; plain text only in banners/toasts. |
dismissible |
boolean |
If false, the X button and ESC key are hidden/blocked. Use only for critical notices that require action before proceeding. |
conditions |
NoticeCondition[] |
Empty array ([]) means always shown (same as [{ kind: 'always' }]). All conditions must pass (AND logic). |
publishedAt |
string |
ISO 8601 date. Used as a tiebreaker in sorting. Set to the deployment date. |
Optional fields
| Field | Type | Description |
|---|---|---|
priority |
number |
Higher number = shown first. Primary sort key. Default: 0. |
minVersion |
string |
Lowest app version (inclusive, semver) that should show this notice. Omit for no lower bound. |
maxVersion |
string |
Upper bound (exclusive, semver) — notice is hidden once this version ships. maxVersion: '4.0.0' means shown on < 4.0.0. Omit for no upper bound. |
icon |
string |
Lucide icon name (e.g. 'Sparkles', 'ImageOff'). Shown in the modal's severity icon circle. Falls back to the severity default icon if absent or unrecognised. |
bodyParams |
Record<string, string> |
Interpolation parameters for bodyKey. Values replace {key} placeholders in the translated string. Never hardcode version numbers or dates directly in translation strings — use this instead. |
media |
NoticeMedia |
Image to display in the modal. See below. |
highlights |
Array<{ labelKey: string; iconName?: string }> |
Bullet-point feature list rendered below the body in modals. Each entry is a translation key + optional Lucide icon name. |
cta |
NoticeCta |
Primary action button. See §8 CTAs. |
Version bounds: The range is
[minVersion, maxVersion)— lower bound inclusive, upper bound exclusive. SomaxVersion: '4.0.0'hides the notice once the app reaches 4.0.0. Both bounds are compared after stripping prerelease/build metadata viasemver.coerce, so a server running3.0.0-pre.42is treated as3.0.0— consistent withexistingUserBeforeVersionand staging environments behave like production.
NoticeMedia
interface NoticeMedia {
src: string; // URL or path
srcDark?: string; // Optional dark-mode variant
altKey: string; // i18n key for alt text
placement?: 'hero' | 'inline'; // default: 'hero' (full-width above body)
aspectRatio?: string; // CSS aspect-ratio value, default '16/9'
}
Character limits
| Field | Modal | Banner | Toast |
|---|---|---|---|
| Title | ≤ 40 chars | ≤ 40 chars | ≤ 40 chars |
| Body | ≤ 400 chars (markdown) | ≤ 140 chars (plain) | ≤ 80 chars (plain) |
| CTA label | ≤ 20 chars, a verb | ≤ 20 chars | ≤ 20 chars |
6. Condition system
Conditions are evaluated server-side on every GET /api/system-notices/active call. The client never sees conditions — only the filtered result.
All conditions in conditions[] must pass (AND logic). To implement OR logic, create multiple notices with overlapping IDs is not possible — instead use a custom predicate with internal OR logic.
Built-in conditions
always
{ kind: 'always' }
Always passes. Equivalent to an empty conditions array.
firstLogin
{ kind: 'firstLogin' }
Passes when users.login_count <= 1. The counter is incremented during login, so this fires on the first fetch after the very first login. Useful for onboarding notices.
noTrips
{ kind: 'noTrips' }
Passes when the user has zero trips. Often combined with firstLogin.
existingUserBeforeVersion
{ kind: 'existingUserBeforeVersion', version: '3.0.0' }
Passes when:
users.first_seen_version < version(user existed before this version)- AND the running app version
>= version(the version has been deployed)
Backfilled/legacy users have first_seen_version = '0.0.0' and always pass the first condition. Use this for upgrade announcements targeting users who were around before a breaking change.
dateWindow
{ kind: 'dateWindow', startsAt: '2026-06-01T00:00:00Z', endsAt: '2026-07-01T00:00:00Z' }
Passes when the current server time is inside [startsAt, endsAt]. endsAt is optional (open-ended). Use for campaigns, maintenance banners, and time-limited promotions.
role
{ kind: 'role', roles: ['admin'] }
// or both roles:
{ kind: 'role', roles: ['admin', 'user'] }
Passes when the user's role is in the given list.
addonEnabled
{ kind: 'addonEnabled', addonId: 'journey' }
Passes when the named addon is enabled in admin settings. Addon IDs are the string values in server/src/addons.ts (ADDON_IDS). Use this to gate notices that promote features behind an addon.
custom
{ kind: 'custom', id: 'my-predicate-id' }
Delegates evaluation to a predicate registered server-side with registerPredicate. This is the escape hatch for logic not covered by the built-in conditions.
Registering a custom predicate:
// server/src/systemNotices/conditions.ts exports registerPredicate
import { registerPredicate } from '../systemNotices/conditions.js';
registerPredicate('has-immich-configured', (ctx) => {
// ctx.user = { login_count, first_seen_version, role, noTrips }
// ctx.currentAppVersion = string
// ctx.now = Date
return someDbCheck(ctx.user);
});
Register predicates at application startup before the first getActiveNoticesFor call.
Combining conditions (AND)
conditions: [
{ kind: 'existingUserBeforeVersion', version: '3.0.0' },
{ kind: 'addonEnabled', addonId: 'journey' },
]
// Only shows to pre-3.0 users AND only if the journey addon is enabled.
7. Display types
modal
Full-screen overlay with backdrop. On mobile: bottom sheet with drag-to-dismiss. On desktop: centered card.
Features:
- Markdown body (via
react-markdown+remark-gfm+rehype-sanitize) - Optional hero or inline image
- Optional highlights list (icon + label bullets)
- Optional CTA button + "Not now" link
- OK button when no CTA is defined
- Multi-page pager: when multiple modal notices are active simultaneously, they are rendered as a paginated single modal with prev/next arrows, dot indicators,
N / Mcounter, and keyboard arrow navigation - Slide transition between pages
- ESC to dismiss all (if current notice is dismissible)
- CTA and OK dismiss all active modal notices, not just the current page
- "Not now" dismisses only the current page
Non-dismissible modals (dismissible: false): X button, ESC key, and pager navigation are all disabled until the user acts on the CTA. Use only for critical severity.
banner
Sticky top bar below the navigation. Slides in with a translate-Y animation.
Constraints:
- Maximum 2 banners shown simultaneously (the 2 highest-priority active banners)
- Plain text only (no markdown)
- RTL-aware left-border accent
- Reports its height via a CSS variable
--banner-stack-hfor layout reflow
toast
Fires the global window.__addToast toast system. Auto-dismisses after 6 s (info) or 9 s (warn). The notice is dismissed from the store after the toast expires.
Constraints:
criticalseverity is not allowed as a toast — the renderer logs a warning and auto-dismisses it instead- Plain text only
- No interaction (no CTA rendered via toast)
8. CTAs (call to action)
A CTA renders as the primary blue button in modals and as an underline link in banners. There are two kinds.
nav — navigate to a route
cta: {
kind: 'nav',
labelKey: 'system_notice.my_notice.cta_label',
href: '/journey',
}
On click: navigates to href using React Router, then dismisses all active modal notices (or the current banner notice). The label is resolved through the i18n system.
action — run a registered client-side handler
cta: {
kind: 'action',
labelKey: 'system_notice.my_notice.cta_label',
actionId: 'open:trip-create',
dismissOnAction: true, // default true — set false to keep notice open after action
}
On click: looks up actionId in the client-side action registry and calls the handler, then dismisses all active modal notices.
To add a new action:
- Create (or extend) a
noticeActions.tsfile in the relevant feature directory:
// client/src/pages/MyFeature/noticeActions.ts
import { registerNoticeAction } from '../../components/SystemNotices/noticeActions.js';
registerNoticeAction('open:my-feature', ({ navigate }) => {
navigate('/my-feature?from=notice');
});
- Import it as a side-effect in
client/src/App.tsx:
import './pages/MyFeature/noticeActions.js'
- The registry integrity test (
server/tests/unit/systemNotices/registry.test.ts) automatically scans allnoticeActions.tsfiles and verifies that everyactionIdin the registry is registered. The test will fail if you add anactionIdto the registry without registering it on the client.
Action handler signature:
(ctx: NoticeActionContext) => void | Promise<void>
interface NoticeActionContext {
navigate: NavigateFunction; // React Router navigate function
}
Dismiss behaviour summary
| Trigger | What is dismissed |
|---|---|
| X button (modal) | All active modal notices |
| ESC key | All active modal notices (if current is dismissible) |
| CTA button | All active modal notices |
| OK button (no CTA) | All active modal notices |
| "Not now" link | Current page only |
| Banner dismiss (X) | That banner only |
| Backdrop click (modal) | Current page only |
| Swipe down (mobile) | Current page only |
| Toast expires | That toast only |
9. i18n — translation keys
Every notice field that is user-visible (titleKey, bodyKey, CTA labelKey, highlight labelKey, media altKey) is an i18n key resolved through useTranslation().t(). The key string is what gets stored in the registry; the display value lives in the translation files.
Translation files location: client/src/i18n/translations/ (15 files: en, de, fr, es, it, nl, pl, cs, hu, ru, zh, zhTw, ar, br, id)
Key naming convention
system_notice.<notice_id_snake>.<field>
Examples:
system_notice.welcome_v1.title
system_notice.welcome_v1.body
system_notice.welcome_v1.cta_label
system_notice.welcome_v1.highlight_plan
system_notice.welcome_v1.hero_alt
Adding keys
Add the English key to client/src/i18n/translations/en.ts first, then replicate to the other 14 files. Group related notice keys together with a comment:
// System notices — my feature
'system_notice.my_notice.title': 'My feature is here',
'system_notice.my_notice.body': 'Here is what changed.',
'system_notice.my_notice.cta_label': 'Explore',
bodyParams interpolation
For values that vary at runtime (version numbers, dates, counts), use {placeholder} syntax in the translation string and pass bodyParams in the registry entry:
// In registry:
bodyKey: 'system_notice.my_notice.body',
bodyParams: { version: '3.1.0', date: '1 May 2026' },
// In en.ts:
'system_notice.my_notice.body': 'TREK {version} was released on {date}.',
Never hardcode dynamic values directly in translation strings. The interpolation runs client-side in ModalRenderer before rendering.
Multiline bodies (modals only)
Use \n\n (escaped, not literal newlines) for paragraph breaks in modal body strings:
'system_notice.my_notice.body': 'First paragraph.\n\nSecond paragraph.',
Literal newlines in single-quoted TypeScript strings cause a parse error.
Pager i18n keys
The pager UI uses its own keys (already present in all 15 files):
system_notice.pager.prev → "Previous notice"
system_notice.pager.next → "Next notice"
system_notice.pager.counter → "{current} / {total}"
system_notice.pager.goto → "Go to notice {n}"
system_notice.pager.position → "Notice {current} of {total}" (aria-live)
10. Client store & dismissal
client/src/store/systemNoticeStore.ts (Zustand, no persistence).
| Action | Behaviour |
|---|---|
fetch() |
GET /api/system-notices/active. Fails silently (non-critical). Sets loaded = true regardless. |
dismiss(id) |
Optimistic: removes notice from store immediately. POSTs to /api/system-notices/{id}/dismiss in background with one retry on failure. |
SystemNoticeHost triggers fetch() on mount if loaded === false. Auth store also triggers it after login, so on a fresh login the fetch happens exactly once.
11. Sorting & priority
Notices are sorted before being sent to the client. The sort order is:
priority(descending) — primary key. Higher number appears first.severity(descending) — tiebreaker:critical(2) >warn(1) >info(0).publishedAt(descending) — final tiebreaker: more recent notices first.
This means priority always wins over severity. Assign priorities deliberately so the intended reading order is preserved when multiple notices are active simultaneously.
Current priority allocations in the registry:
| Range | Use |
|---|---|
| 100 | Onboarding / first-login |
| 80–90 | Major version upgrade notices |
| 50–70 | Feature announcements |
| 10–40 | Campaigns, banners |
| 0 (default) | Miscellaneous |
12. How-to recipes
Add a new modal notice
- Registry — add an entry to
SYSTEM_NOTICESinserver/src/systemNotices/registry.ts:
{
id: 'my-feature-v2',
display: 'modal',
severity: 'info',
icon: 'Zap',
titleKey: 'system_notice.my_feature_v2.title',
bodyKey: 'system_notice.my_feature_v2.body',
highlights: [
{ labelKey: 'system_notice.my_feature_v2.highlight_one', iconName: 'Check' },
],
cta: {
kind: 'nav',
labelKey: 'system_notice.my_feature_v2.cta_label',
href: '/my-feature',
},
dismissible: true,
conditions: [{ kind: 'existingUserBeforeVersion', version: '2.0.0' }],
publishedAt: '2026-06-01T00:00:00Z',
priority: 60,
},
-
i18n — add keys to
client/src/i18n/translations/en.tsand the 14 other language files. -
Test — run
cd server && npx vitest run tests/unit/systemNotices/to verify registry integrity.
Add a notice with an action CTA
- Create the action handler in the relevant feature directory:
// client/src/pages/MyFeature/noticeActions.ts
import { registerNoticeAction } from '../../components/SystemNotices/noticeActions.js';
registerNoticeAction('open:my-feature-dialog', ({ navigate }) => {
navigate('/my-feature?dialog=welcome');
});
- Import it in
client/src/App.tsx:
import './pages/MyFeature/noticeActions.js'
- Reference the
actionIdin the registry:
cta: {
kind: 'action',
labelKey: 'system_notice.my_notice.cta_label',
actionId: 'open:my-feature-dialog',
},
The registry integrity test will catch any actionId that appears in the registry but lacks a registerNoticeAction call.
Retire a notice (stop showing it)
Do not delete the entry. Set maxVersion to the last app version on which the notice should appear. Once the app is upgraded past that version, the service filters it out automatically. The database row for dismissed users remains harmless.
{
id: 'old-campaign',
// ... all existing fields unchanged ...
maxVersion: '3.1.0', // hidden once 3.1.0 ships (exclusive upper bound)
}
To scope a notice to a specific version window (e.g. a v3-only announcement), combine both bounds:
{
id: 'v3-only',
minVersion: '3.0.0',
maxVersion: '4.0.0', // shown on >= 3.0.0 and < 4.0.0
}
Show a notice only during a campaign window
Combine dateWindow with any other targeting conditions:
conditions: [
{ kind: 'dateWindow', startsAt: '2026-06-15T00:00:00Z', endsAt: '2026-06-30T23:59:59Z' },
{ kind: 'role', roles: ['admin'] },
],
Show a notice only if an addon is enabled
conditions: [
{ kind: 'addonEnabled', addonId: 'journey' },
],
Addon IDs are the string values in server/src/addons.ts → ADDON_IDS.
Add a custom condition
// server/src/startup.ts (or wherever your bootstrap code runs)
import { registerPredicate } from './systemNotices/conditions.js';
registerPredicate('has-no-profile-photo', (ctx) => {
const row = db.prepare('SELECT avatar FROM users WHERE id = ?').get(ctx.user.id);
return !row?.avatar;
});
Then reference it in the registry:
conditions: [{ kind: 'custom', id: 'has-no-profile-photo' }],
Create a multipage upgrade announcement
Give multiple notices the same conditions and adjacent priority values. The pager groups all active modal notices together automatically — no extra wiring required.
// Page 1 — breaking change (higher priority, warn severity)
{ id: 'v4-breaking', priority: 90, severity: 'warn', conditions: [{ kind: 'existingUserBeforeVersion', version: '4.0.0' }], ... },
// Page 2 — new feature (lower priority, info severity)
{ id: 'v4-feature', priority: 80, severity: 'info', conditions: [{ kind: 'existingUserBeforeVersion', version: '4.0.0' }], ... },
Users who have already dismissed page 1 will only see page 2 on their next session.
13. Testing
Server unit tests
server/tests/unit/systemNotices/conditions.test.ts
Tests each condition kind in isolation using evaluate() directly. No DB required.
server/tests/unit/systemNotices/registry.test.ts
Validates registry integrity:
- No duplicate
idvalues - All
actionCTAactionIds have a correspondingregisterNoticeAction()call in the client source (scanned via regex — no JSON file needed) - All
publishedAtvalues parse as valid ISO dates
Run: cd server && npx vitest run tests/unit/systemNotices/
server/tests/integration/systemNotices.test.ts
Integration tests against a real in-memory SQLite database:
GET /api/system-notices/activereturns 401 without auth, returns correct notices per user statePOST /api/system-notices/:id/dismissstores the dismissal and filters on subsequent requests- Dismissing an unknown ID returns 404
Run: cd server && npx vitest run tests/integration/systemNotices.test.ts
Client unit tests
client/src/components/SystemNotices/SystemNoticeModal.test.tsx
Tests ModalRenderer with fake timers (vi.useFakeTimers()) and MSW for the dismiss endpoint. Key helpers:
// Flush the 500 ms grace delay that gates the modal's visible state
async function flushGraceDelay() {
await act(async () => { vi.runAllTimers(); });
}
// Minimal notice factory
function makeNotice(overrides?: Partial<SystemNoticeDTO>): SystemNoticeDTO
Covered cases (FE-SN-MODAL-001 to 018):
- Grace delay before visibility
- Dismiss button, X button, ESC key
- Non-dismissible notices (all affordances blocked)
- CTA nav button — dismisses all notices
- Body param interpolation
- Pager: counter, dots, prev/next buttons, keyboard arrows, dot click, non-dismissible lock
- Dismiss-does-not-skip regression
- X and ESC dismiss all in multipage scenario
- Last notice close
Run: cd client && npm run test -- SystemNoticeModal
Running all notice tests
cd server && npx vitest run tests/unit/systemNotices/ tests/integration/systemNotices.test.ts
cd client && npm run test -- SystemNoticeModal
14. Rules & constraints
| Rule | Reason |
|---|---|
Never delete or reuse a notice id |
Dismissal records are keyed by id. Deletion causes dismissed users to see the notice again. |
| Never use literal newlines in translation strings | Single-quoted TS strings with literal newlines cause esbuild parse errors. Use \n\n (escaped). |
| Never hardcode version numbers or dates in translation strings | Use bodyParams so strings stay translatable without retranslation per release. |
critical severity must have dismissible: false |
critical toasts are auto-dismissed with a warning; a dismissible critical modal is inconsistent UX. |
critical must not use display: 'toast' |
The toast renderer logs a warning and auto-dismisses critical toasts rather than showing them. |
| CTA labels ≤ 20 chars, sentence case, a verb | Consistent button copy across the app. |
| Priorities must be set explicitly for upgrade notices | Adjacent notices form a multipage group; ordering matters for the reading flow. |
action CTA actionId must be registered client-side |
The registry integrity test enforces this. Add both the registry entry and the registerNoticeAction call in the same PR. |
maxVersion over deletion for retiring notices |
See §12. Deletion would cause dismissed users to re-see the notice if the ID were ever reused. |