Added app badge support for the guest PWA and wired it to the existing counts (unread notifications + upload queue + pending uploads). When the total hits zero, the badge is cleared; when it’s >0, it’s set.

This commit is contained in:
Codex Agent
2025-12-27 14:15:14 +01:00
parent 1a48c9458e
commit a8b54b75ea
3 changed files with 103 additions and 0 deletions

View File

@@ -8,6 +8,7 @@ import {
type GuestNotificationItem,
} from '../services/notificationApi';
import { fetchPendingUploadsSummary } from '../services/pendingUploadsApi';
import { updateAppBadge } from '../lib/badges';
export type NotificationCenterValue = {
notifications: GuestNotificationItem[];
@@ -265,6 +266,10 @@ export function NotificationCenterProvider({ eventToken, children }: { eventToke
const loading = loadingNotifications || queueLoading || pendingLoading;
const totalCount = unreadCount + queueCount + pendingCount;
React.useEffect(() => {
void updateAppBadge(totalCount);
}, [totalCount]);
const value: NotificationCenterValue = {
notifications,
unreadCount,

View File

@@ -0,0 +1,52 @@
import { afterEach, describe, expect, it, vi } from 'vitest';
import { supportsBadging, updateAppBadge } from '../badges';
const originalSet = (navigator as any).setAppBadge;
const originalClear = (navigator as any).clearAppBadge;
const hadSet = 'setAppBadge' in navigator;
const hadClear = 'clearAppBadge' in navigator;
function restoreNavigator() {
if (hadSet) {
Object.defineProperty(navigator, 'setAppBadge', { configurable: true, value: originalSet });
} else {
delete (navigator as any).setAppBadge;
}
if (hadClear) {
Object.defineProperty(navigator, 'clearAppBadge', { configurable: true, value: originalClear });
} else {
delete (navigator as any).clearAppBadge;
}
}
describe('badges', () => {
afterEach(() => {
restoreNavigator();
});
it('sets the badge count when supported', async () => {
const setAppBadge = vi.fn();
Object.defineProperty(navigator, 'setAppBadge', { configurable: true, value: setAppBadge });
Object.defineProperty(navigator, 'clearAppBadge', { configurable: true, value: vi.fn() });
expect(supportsBadging()).toBe(true);
await updateAppBadge(4);
expect(setAppBadge).toHaveBeenCalledWith(4);
});
it('clears the badge when count is zero', async () => {
const clearAppBadge = vi.fn();
Object.defineProperty(navigator, 'setAppBadge', { configurable: true, value: vi.fn() });
Object.defineProperty(navigator, 'clearAppBadge', { configurable: true, value: clearAppBadge });
await updateAppBadge(0);
expect(clearAppBadge).toHaveBeenCalled();
});
it('no-ops when unsupported', async () => {
delete (navigator as any).setAppBadge;
delete (navigator as any).clearAppBadge;
expect(supportsBadging()).toBe(false);
await updateAppBadge(3);
});
});

View File

@@ -0,0 +1,46 @@
type BadgingNavigator = Navigator & {
setAppBadge?: (contents?: number) => Promise<void> | void;
clearAppBadge?: () => Promise<void> | void;
};
function getNavigator(): BadgingNavigator | null {
if (typeof navigator === 'undefined') {
return null;
}
return navigator as BadgingNavigator;
}
export function supportsBadging(): boolean {
const nav = getNavigator();
return Boolean(nav && (typeof nav.setAppBadge === 'function' || typeof nav.clearAppBadge === 'function'));
}
export async function updateAppBadge(count: number): Promise<void> {
const nav = getNavigator();
if (!nav) {
return;
}
const safeCount = Number.isFinite(count) ? Math.max(0, Math.floor(count)) : 0;
if (!supportsBadging()) {
return;
}
try {
if (safeCount > 0 && nav.setAppBadge) {
await nav.setAppBadge(safeCount);
return;
}
if (nav.clearAppBadge) {
await nav.clearAppBadge();
return;
}
if (nav.setAppBadge) {
await nav.setAppBadge(0);
}
} catch (error) {
console.warn('Updating app badge failed', error);
}
}