@payez/next-mvp 3.5.0 → 3.6.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (32) hide show
  1. package/dist/api-handlers/admin/index.d.ts +1 -0
  2. package/dist/api-handlers/admin/index.js +3 -1
  3. package/dist/api-handlers/admin/stats.d.ts +21 -0
  4. package/dist/api-handlers/admin/stats.js +240 -0
  5. package/dist/auth/utils/idp-client.js +1 -0
  6. package/dist/components/account/MobileNavDrawer.d.ts +32 -0
  7. package/dist/components/account/MobileNavDrawer.js +81 -0
  8. package/dist/components/account/UserAvatarMenu.js +5 -1
  9. package/dist/components/account/index.d.ts +2 -0
  10. package/dist/components/account/index.js +5 -2
  11. package/dist/index.d.ts +2 -2
  12. package/dist/index.js +2 -1
  13. package/dist/pages/admin-page-permissions/PagePermissionsAdminPage.d.ts +18 -0
  14. package/dist/pages/admin-page-permissions/PagePermissionsAdminPage.js +276 -0
  15. package/dist/pages/admin-page-permissions/index.d.ts +6 -0
  16. package/dist/pages/admin-page-permissions/index.js +13 -0
  17. package/dist/pages/admin-roles/RolesAdminPage.d.ts +12 -11
  18. package/dist/pages/admin-roles/RolesAdminPage.js +249 -66
  19. package/dist/routes/auth/session.d.ts +1 -30
  20. package/dist/routes/auth/session.js +3 -4
  21. package/package.json +6 -1
  22. package/src/api-handlers/admin/index.ts +5 -0
  23. package/src/api-handlers/admin/stats.ts +240 -0
  24. package/src/auth/utils/idp-client.ts +1 -0
  25. package/src/components/account/MobileNavDrawer.tsx +305 -0
  26. package/src/components/account/UserAvatarMenu.tsx +47 -17
  27. package/src/components/account/index.ts +5 -0
  28. package/src/index.ts +2 -2
  29. package/src/pages/admin-page-permissions/PagePermissionsAdminPage.tsx +527 -0
  30. package/src/pages/admin-page-permissions/index.ts +7 -0
  31. package/src/pages/admin-roles/RolesAdminPage.tsx +494 -318
  32. package/src/routes/auth/session.ts +3 -4
@@ -0,0 +1,240 @@
1
+ /**
2
+ * Admin Stats API Handler
3
+ *
4
+ * Aggregates dashboard statistics from users, Redis sessions, and audit logs.
5
+ * Uses service account HMAC auth for Vibe API requests.
6
+ *
7
+ * @version 1.0
8
+ * @requires Admin role (vibe_app_admin or payez_admin)
9
+ */
10
+
11
+ import { NextRequest, NextResponse } from 'next/server';
12
+ import { getServerSession } from 'next-auth';
13
+ import { getStartupIDPConfig } from '../../lib/startup-init';
14
+ import { getRedis } from '../../lib/redis';
15
+ import { ADMIN_ROLES } from '../../lib/roles';
16
+
17
+ interface VibeRequestOptions {
18
+ method: 'GET' | 'POST' | 'PUT' | 'DELETE';
19
+ body?: unknown;
20
+ }
21
+
22
+ async function checkAdminRole(getAuthOptions: () => Promise<any>): Promise<{ isAdmin: boolean; error?: NextResponse }> {
23
+ const authOptions = await getAuthOptions();
24
+ const session = await getServerSession(authOptions) as any;
25
+
26
+ if (!session?.user) {
27
+ return {
28
+ isAdmin: false,
29
+ error: NextResponse.json({ success: false, error: 'Please sign in' }, { status: 401 }),
30
+ };
31
+ }
32
+
33
+ const userRoles = (session.user?.roles as string[]) || [];
34
+ const hasAdminRole = ADMIN_ROLES.some(role => userRoles.includes(role));
35
+
36
+ if (!hasAdminRole) {
37
+ return {
38
+ isAdmin: false,
39
+ error: NextResponse.json({ success: false, error: 'Admin access required' }, { status: 403 }),
40
+ };
41
+ }
42
+
43
+ return { isAdmin: true };
44
+ }
45
+
46
+ async function vibeServiceRequest<T = unknown>(
47
+ endpoint: string,
48
+ options: VibeRequestOptions
49
+ ): Promise<{ ok: boolean; status: number; data: T | null; error?: string }> {
50
+ const idpUrl = process.env.NEXT_PUBLIC_IDP_URL || process.env.IDP_URL;
51
+ const clientId = process.env.VIBE_CLIENT_ID;
52
+ const signingKey = process.env.VIBE_HMAC_KEY;
53
+
54
+ if (!idpUrl || !clientId || !signingKey) {
55
+ return { ok: false, status: 500, data: null, error: 'Vibe not configured' };
56
+ }
57
+
58
+ const timestamp = Math.floor(Date.now() / 1000);
59
+ const stringToSign = `${timestamp}|${options.method}|${endpoint}`;
60
+
61
+ const crypto = await import('crypto');
62
+ const signature = crypto
63
+ .createHmac('sha256', Buffer.from(signingKey, 'base64'))
64
+ .update(stringToSign)
65
+ .digest('base64');
66
+
67
+ const proxyUrl = `${idpUrl}/api/vibe/proxy`;
68
+
69
+ const idpConfig = getStartupIDPConfig();
70
+ const idpClientId = idpConfig?.clientSlug || idpConfig?.clientId;
71
+
72
+ try {
73
+ const res = await fetch(proxyUrl, {
74
+ method: 'POST',
75
+ headers: {
76
+ 'Content-Type': 'application/json',
77
+ 'X-Vibe-Client-Id': clientId,
78
+ 'X-Vibe-Timestamp': String(timestamp),
79
+ 'X-Vibe-Signature': signature,
80
+ ...(idpClientId && { 'X-Client-Id': idpClientId }),
81
+ },
82
+ body: JSON.stringify({
83
+ endpoint,
84
+ method: options.method,
85
+ data: options.body ?? null,
86
+ }),
87
+ cache: 'no-store',
88
+ });
89
+
90
+ if (res.status === 204) return { ok: true, status: 204, data: null };
91
+ if (!res.ok) {
92
+ const errorText = await res.text();
93
+ return { ok: false, status: res.status, data: null, error: errorText };
94
+ }
95
+
96
+ const body = await res.json();
97
+ return { ok: true, status: res.status, data: body };
98
+ } catch (error) {
99
+ return { ok: false, status: 0, data: null, error: String(error) };
100
+ }
101
+ }
102
+
103
+ export interface AdminStatsHandlerConfig {
104
+ getAuthOptions: () => Promise<any>;
105
+ appSlug?: string;
106
+ }
107
+
108
+ /**
109
+ * GET /api/admin/stats - Dashboard statistics
110
+ * Aggregates users + tier breakdown, active Redis sessions, and recent audit activity.
111
+ */
112
+ export function createStatsHandler(config: AdminStatsHandlerConfig) {
113
+ const getSessionPrefix = () => {
114
+ const appSlug = config.appSlug || process.env.APP_SLUG || process.env.CLIENT_ID || 'app';
115
+ return `${appSlug}:sess:`;
116
+ };
117
+
118
+ return {
119
+ async GET(_request: NextRequest) {
120
+ const adminCheck = await checkAdminRole(config.getAuthOptions);
121
+ if (adminCheck.error) return adminCheck.error;
122
+
123
+ try {
124
+ // Fetch from 3 sources in parallel
125
+ const [usersResult, sessionCount, auditResult] = await Promise.allSettled([
126
+ // 1. Users + tier breakdown via HMAC proxy (Vibe collection query)
127
+ vibeServiceRequest<any>('/v1/collections/vibe_app/tables/users/query', {
128
+ method: 'POST',
129
+ body: { page: 1, pageSize: 500, orderBy: 'created_at', orderDirection: 'desc' },
130
+ }),
131
+
132
+ // 2. Active sessions from Redis
133
+ (async () => {
134
+ const redis = getRedis();
135
+ const sessionPrefix = getSessionPrefix();
136
+ const sessionKeys: string[] = [];
137
+ let cursor = '0';
138
+ do {
139
+ const [newCursor, keys] = await redis.scan(cursor, 'MATCH', `${sessionPrefix}*`, 'COUNT', 100);
140
+ cursor = newCursor;
141
+ sessionKeys.push(...keys.filter((k: string) => !k.includes(':ver:')));
142
+ } while (cursor !== '0');
143
+ return sessionKeys.length;
144
+ })(),
145
+
146
+ // 3. Recent audit activity via HMAC proxy
147
+ vibeServiceRequest<any>('/v1/audit?pageSize=10&sortDir=desc', { method: 'GET' }),
148
+ ]);
149
+
150
+ // Parse users — deduplicate by user_id
151
+ let totalUsers = 0;
152
+ let tierBreakdown: Record<string, number> = {};
153
+ if (usersResult.status === 'fulfilled' && usersResult.value.ok && usersResult.value.data) {
154
+ const data = usersResult.value.data;
155
+ const rawUsers = data.data || data.documents || data.users || [];
156
+
157
+ // Deduplicate by user_id (keeps latest document_id)
158
+ const userMap = new Map();
159
+ for (const u of rawUsers) {
160
+ const uid = u.user_id || u.id || u.document_id;
161
+ const existing = userMap.get(uid);
162
+ if (!existing || (u.document_id || '') > (existing.document_id || '')) {
163
+ userMap.set(uid, u);
164
+ }
165
+ }
166
+ const uniqueUsers = Array.from(userMap.values());
167
+ totalUsers = uniqueUsers.length;
168
+
169
+ // Build tier breakdown from deduplicated users (unless API provides one)
170
+ tierBreakdown = data.tierBreakdown || data.tiers || {};
171
+ if (Object.keys(tierBreakdown).length === 0) {
172
+ for (const user of uniqueUsers) {
173
+ const tier = user.tier || 'free';
174
+ tierBreakdown[tier] = (tierBreakdown[tier] || 0) + 1;
175
+ }
176
+ }
177
+ }
178
+
179
+ // Parse active sessions count
180
+ let activeSessions = 0;
181
+ if (sessionCount.status === 'fulfilled') {
182
+ activeSessions = sessionCount.value;
183
+ }
184
+
185
+ // Parse audit events for recent activity
186
+ let recentActivity: any[] = [];
187
+ if (auditResult.status === 'fulfilled' && auditResult.value.ok && auditResult.value.data) {
188
+ const data = auditResult.value.data;
189
+
190
+ // Handle multiple possible response shapes
191
+ let events: any[] = [];
192
+ if (Array.isArray(data)) {
193
+ events = data;
194
+ } else if (Array.isArray(data.data)) {
195
+ events = data.data;
196
+ } else if (Array.isArray(data.entries)) {
197
+ events = data.entries;
198
+ } else if (Array.isArray(data.items)) {
199
+ events = data.items;
200
+ } else if (Array.isArray(data.documents)) {
201
+ events = data.documents;
202
+ } else if (data.success && Array.isArray(data.results)) {
203
+ events = data.results;
204
+ }
205
+
206
+ recentActivity = events.slice(0, 5).map((e: any) => ({
207
+ id: e.audit_log_id || e.id || e.document_id,
208
+ type: e.category || e.type || 'admin',
209
+ action: e.action || e.event || e.message || 'Unknown action',
210
+ actor: e.admin_email || e.actor || e.user || e.actor_email || 'System',
211
+ target: e.target_type ? `${e.target_type}:${e.target_id}` : (e.target || e.target_user),
212
+ details: e.description || e.details,
213
+ timestamp: e.created_at || e.timestamp || e.date,
214
+ success: e.is_success ?? e.success ?? true,
215
+ }));
216
+ }
217
+
218
+ // Calculate tier percentages
219
+ const tiers = Object.entries(tierBreakdown).map(([name, count]) => ({
220
+ name,
221
+ count: count as number,
222
+ pct: totalUsers > 0 ? Math.round(((count as number) / totalUsers) * 100) : 0,
223
+ }));
224
+
225
+ return NextResponse.json({
226
+ totalUsers,
227
+ activeSessions,
228
+ tiers,
229
+ recentActivity,
230
+ });
231
+ } catch (error: any) {
232
+ console.error('[admin/stats] Error:', error);
233
+ return NextResponse.json(
234
+ { error: error.message || 'Internal error' },
235
+ { status: 500 }
236
+ );
237
+ }
238
+ },
239
+ };
240
+ }
@@ -187,6 +187,7 @@ export async function idpOAuthCallback(oauthData: {
187
187
  access_token: oauthData.accessToken || '',
188
188
  refresh_token: oauthData.refreshToken || '',
189
189
  expires_at: oauthData.expiresAt || 0,
190
+ client_id: clientId,
190
191
  }),
191
192
  });
192
193
 
@@ -0,0 +1,305 @@
1
+ 'use client';
2
+
3
+ import { useEffect, useCallback } from 'react';
4
+ import { useSession, signIn } from 'next-auth/react';
5
+ import { usePathname } from 'next/navigation';
6
+ import Image from 'next/image';
7
+ import Link from 'next/link';
8
+ import { X } from 'lucide-react';
9
+
10
+ export interface NavItem {
11
+ href: string;
12
+ label: string;
13
+ icon?: React.ReactNode;
14
+ }
15
+
16
+ export interface NavSection {
17
+ title?: string;
18
+ items: Array<{
19
+ label: string;
20
+ icon?: React.ReactNode;
21
+ href?: string;
22
+ onClick?: () => void;
23
+ }>;
24
+ }
25
+
26
+ export interface MobileNavDrawerProps {
27
+ isOpen: boolean;
28
+ onClose: () => void;
29
+ navItems: NavItem[];
30
+ /** Extra sections like Admin, rendered after nav items with optional title */
31
+ customSections?: NavSection[];
32
+ /** Base path for account link (default: '/account') */
33
+ basePath?: string;
34
+ /** Custom sign-in handler (default: next-auth signIn) */
35
+ onSignIn?: () => void;
36
+ /** Callback URL after sign in (default: '/dashboard') */
37
+ signInCallbackUrl?: string;
38
+ /** Custom unauthenticated actions (replaces default Login + Start Free buttons) */
39
+ unauthActions?: React.ReactNode;
40
+ /** Custom authenticated footer (replaces default "Account Settings" link) */
41
+ authFooter?: React.ReactNode;
42
+ }
43
+
44
+ export function MobileNavDrawer({
45
+ isOpen,
46
+ onClose,
47
+ navItems,
48
+ customSections,
49
+ basePath = '/account',
50
+ onSignIn,
51
+ signInCallbackUrl = '/dashboard',
52
+ unauthActions,
53
+ authFooter,
54
+ }: MobileNavDrawerProps) {
55
+ const { data: session } = useSession();
56
+ const pathname = usePathname();
57
+ const isAuthenticated = !!session?.user;
58
+
59
+ const isActiveRoute = useCallback(
60
+ (href: string) => pathname?.startsWith(href) ?? false,
61
+ [pathname],
62
+ );
63
+
64
+ // Close on Escape key
65
+ useEffect(() => {
66
+ function handleEscape(event: KeyboardEvent) {
67
+ if (event.key === 'Escape') {
68
+ onClose();
69
+ }
70
+ }
71
+
72
+ if (isOpen) {
73
+ document.addEventListener('keydown', handleEscape);
74
+ return () => document.removeEventListener('keydown', handleEscape);
75
+ }
76
+ }, [isOpen, onClose]);
77
+
78
+ // Lock body scroll when open
79
+ useEffect(() => {
80
+ if (isOpen) {
81
+ document.body.style.overflow = 'hidden';
82
+ return () => {
83
+ document.body.style.overflow = '';
84
+ };
85
+ }
86
+ }, [isOpen]);
87
+
88
+ const handleSignIn = () => {
89
+ onClose();
90
+ if (onSignIn) {
91
+ onSignIn();
92
+ } else {
93
+ signIn(undefined, { callbackUrl: signInCallbackUrl });
94
+ }
95
+ };
96
+
97
+ const handleSectionItemClick = (item: NavSection['items'][number]) => {
98
+ onClose();
99
+ if (item.onClick) {
100
+ item.onClick();
101
+ }
102
+ };
103
+
104
+ // Derive display initial from name or email
105
+ const userName = (session?.user as any)?.name;
106
+ const userEmail = session?.user?.email;
107
+ const displaySource = userName || userEmail;
108
+ const userInitial = displaySource?.charAt(0).toUpperCase() || '?';
109
+
110
+ return (
111
+ <>
112
+ {/* Backdrop */}
113
+ <div
114
+ className={`
115
+ fixed inset-0 bg-black/50 backdrop-blur-sm z-40 lg:hidden
116
+ transition-opacity duration-300
117
+ ${isOpen ? 'opacity-100' : 'opacity-0 pointer-events-none'}
118
+ `}
119
+ onClick={onClose}
120
+ aria-hidden="true"
121
+ />
122
+
123
+ {/* Slide-out Drawer */}
124
+ <div
125
+ role="dialog"
126
+ aria-modal="true"
127
+ aria-label="Navigation menu"
128
+ aria-expanded={isOpen}
129
+ className={`
130
+ fixed top-0 right-0 bottom-0 w-80 max-w-[85vw]
131
+ bg-white dark:bg-slate-900
132
+ shadow-[-8px_0_32px_rgba(0,0,0,0.15)]
133
+ dark:shadow-[-8px_0_32px_rgba(0,0,0,0.4)]
134
+ z-50 lg:hidden
135
+ overflow-y-auto
136
+ transition-transform duration-300 ease-out
137
+ ${isOpen ? 'translate-x-0' : 'translate-x-full'}
138
+ `}
139
+ >
140
+ {/* Drawer Header */}
141
+ <div className="flex items-center justify-between p-4 border-b border-gray-200 dark:border-white/10">
142
+ <span className="text-lg font-semibold text-gray-900 dark:text-white">
143
+ Menu
144
+ </span>
145
+ <button
146
+ onClick={onClose}
147
+ className="
148
+ p-2 rounded-xl
149
+ text-gray-400 hover:text-gray-900
150
+ dark:hover:text-white
151
+ hover:bg-gray-100 dark:hover:bg-white/10
152
+ transition-colors
153
+ "
154
+ aria-label="Close menu"
155
+ >
156
+ <X className="h-5 w-5" />
157
+ </button>
158
+ </div>
159
+
160
+ {/* User Info (if authenticated) */}
161
+ {isAuthenticated && session?.user && (
162
+ <div className="p-4 border-b border-gray-200 dark:border-white/10">
163
+ <div className="flex items-center gap-3">
164
+ {session.user.image ? (
165
+ <Image
166
+ src={session.user.image}
167
+ alt=""
168
+ width={48}
169
+ height={48}
170
+ className="w-12 h-12 rounded-full"
171
+ unoptimized
172
+ />
173
+ ) : (
174
+ <div className="w-12 h-12 rounded-full bg-blue-500 flex items-center justify-center text-white font-semibold text-lg">
175
+ {userInitial}
176
+ </div>
177
+ )}
178
+ <div className="flex-1 min-w-0">
179
+ {userName && (
180
+ <p className="text-sm font-semibold text-gray-900 dark:text-white truncate">
181
+ {userName}
182
+ </p>
183
+ )}
184
+ {userEmail && (
185
+ <p className="text-xs text-gray-500 dark:text-slate-400 truncate">
186
+ {userEmail}
187
+ </p>
188
+ )}
189
+ </div>
190
+ </div>
191
+ </div>
192
+ )}
193
+
194
+ {/* Navigation Items */}
195
+ <div className="p-2">
196
+ {navItems.map((item) => (
197
+ <Link
198
+ key={item.href}
199
+ href={item.href}
200
+ onClick={onClose}
201
+ className={`
202
+ flex items-center gap-3 px-4 py-3.5 rounded-xl
203
+ transition-colors duration-200
204
+ ${isActiveRoute(item.href)
205
+ ? 'bg-blue-500/10 text-blue-500'
206
+ : 'text-gray-900 dark:text-white hover:bg-gray-100 dark:hover:bg-white/10'
207
+ }
208
+ `}
209
+ >
210
+ {item.icon && <span className="text-xl">{item.icon}</span>}
211
+ <span className="font-medium">{item.label}</span>
212
+ {isActiveRoute(item.href) && (
213
+ <span className="ml-auto w-2 h-2 rounded-full bg-blue-500" />
214
+ )}
215
+ </Link>
216
+ ))}
217
+ </div>
218
+
219
+ {/* Custom Sections */}
220
+ {customSections?.map((section, sectionIndex) => (
221
+ <div
222
+ key={sectionIndex}
223
+ className="p-2 border-t border-gray-200 dark:border-white/10"
224
+ >
225
+ {section.title && (
226
+ <p className="px-4 py-2 text-xs font-semibold text-gray-500 dark:text-slate-400 uppercase tracking-wider">
227
+ {section.title}
228
+ </p>
229
+ )}
230
+ {section.items.map((item, itemIndex) =>
231
+ item.href ? (
232
+ <Link
233
+ key={itemIndex}
234
+ href={item.href}
235
+ onClick={onClose}
236
+ className="
237
+ flex items-center gap-3 px-4 py-3 rounded-xl
238
+ text-gray-900 dark:text-white
239
+ hover:bg-gray-100 dark:hover:bg-white/10
240
+ transition-colors
241
+ "
242
+ >
243
+ {item.icon && <span className="text-xl">{item.icon}</span>}
244
+ <span className="font-medium">{item.label}</span>
245
+ </Link>
246
+ ) : (
247
+ <button
248
+ key={itemIndex}
249
+ onClick={() => handleSectionItemClick(item)}
250
+ className="
251
+ flex items-center gap-3 px-4 py-3 rounded-xl w-full text-left
252
+ text-gray-900 dark:text-white
253
+ hover:bg-gray-100 dark:hover:bg-white/10
254
+ transition-colors
255
+ "
256
+ >
257
+ {item.icon && <span className="text-xl">{item.icon}</span>}
258
+ <span className="font-medium">{item.label}</span>
259
+ </button>
260
+ ),
261
+ )}
262
+ </div>
263
+ ))}
264
+
265
+ {/* Auth Actions */}
266
+ <div className="p-4 mt-auto border-t border-gray-200 dark:border-white/10">
267
+ {!isAuthenticated ? (
268
+ unauthActions ?? (
269
+ <div className="space-y-3">
270
+ <button
271
+ onClick={handleSignIn}
272
+ className="
273
+ w-full px-4 py-3 rounded-xl
274
+ text-blue-500 font-semibold
275
+ border border-blue-500/30
276
+ hover:bg-blue-500/10
277
+ transition-colors
278
+ "
279
+ >
280
+ Login
281
+ </button>
282
+ </div>
283
+ )
284
+ ) : (
285
+ authFooter ?? (
286
+ <Link
287
+ href={basePath}
288
+ onClick={onClose}
289
+ className="
290
+ flex items-center justify-center gap-2
291
+ w-full px-4 py-3 rounded-xl
292
+ text-gray-500 dark:text-slate-400 font-medium
293
+ hover:bg-gray-100 dark:hover:bg-white/10
294
+ transition-colors
295
+ "
296
+ >
297
+ Account Settings
298
+ </Link>
299
+ )
300
+ )}
301
+ </div>
302
+ </div>
303
+ </>
304
+ );
305
+ }
@@ -3,6 +3,7 @@
3
3
  import { useState, useRef, useEffect } from 'react';
4
4
  import { useSession, signOut } from 'next-auth/react';
5
5
  import { useRouter } from 'next/navigation';
6
+ import Image from 'next/image';
6
7
  import { User, Settings, Shield, LogOut } from 'lucide-react';
7
8
 
8
9
  export interface UserAvatarMenuProps {
@@ -119,12 +120,23 @@ export function UserAvatarMenu({
119
120
  {/* Avatar trigger button */}
120
121
  <button
121
122
  onClick={() => setIsOpen(!isOpen)}
122
- className="flex items-center justify-center h-10 w-10 rounded-full bg-[#349AD5] text-white font-semibold text-lg hover:bg-[#2980b9] transition-colors focus:outline-none focus:ring-2 focus:ring-[#349AD5] focus:ring-offset-2 dark:focus:ring-offset-slate-900"
123
+ className="flex items-center justify-center h-10 w-10 rounded-full overflow-hidden bg-blue-500 text-white font-semibold text-lg hover:opacity-90 transition-opacity focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-offset-2 dark:focus:ring-offset-slate-900"
123
124
  aria-label="User menu"
124
125
  aria-expanded={isOpen}
125
126
  aria-haspopup="true"
126
127
  >
127
- {userInitial}
128
+ {session.user.image ? (
129
+ <Image
130
+ src={session.user.image}
131
+ alt=""
132
+ width={40}
133
+ height={40}
134
+ className="w-10 h-10 rounded-full object-cover"
135
+ unoptimized
136
+ />
137
+ ) : (
138
+ userInitial
139
+ )}
128
140
  </button>
129
141
 
130
142
  {/* Dropdown menu */}
@@ -138,21 +150,39 @@ export function UserAvatarMenu({
138
150
  >
139
151
  {/* User identity label */}
140
152
  <div className="px-4 py-3 border-b border-gray-200 dark:border-slate-700">
141
- {userName && (
142
- <p className="text-sm font-medium text-gray-700 dark:text-slate-200 truncate">
143
- {userName}
144
- </p>
145
- )}
146
- {userEmail && (
147
- <p className="text-sm text-gray-500 dark:text-slate-400 truncate">
148
- {userEmail}
149
- </p>
150
- )}
151
- {!userName && !userEmail && (
152
- <p className="text-sm text-gray-500 dark:text-slate-400">
153
- Signed in
154
- </p>
155
- )}
153
+ <div className="flex items-center gap-3">
154
+ {session.user.image ? (
155
+ <Image
156
+ src={session.user.image}
157
+ alt=""
158
+ width={32}
159
+ height={32}
160
+ className="w-8 h-8 rounded-full flex-shrink-0"
161
+ unoptimized
162
+ />
163
+ ) : (
164
+ <div className="w-8 h-8 rounded-full bg-blue-500 flex items-center justify-center text-white font-semibold text-sm flex-shrink-0">
165
+ {userInitial}
166
+ </div>
167
+ )}
168
+ <div className="min-w-0">
169
+ {userName && (
170
+ <p className="text-sm font-medium text-gray-700 dark:text-slate-200 truncate">
171
+ {userName}
172
+ </p>
173
+ )}
174
+ {userEmail && (
175
+ <p className="text-sm text-gray-500 dark:text-slate-400 truncate">
176
+ {userEmail}
177
+ </p>
178
+ )}
179
+ {!userName && !userEmail && (
180
+ <p className="text-sm text-gray-500 dark:text-slate-400">
181
+ Signed in
182
+ </p>
183
+ )}
184
+ </div>
185
+ </div>
156
186
  </div>
157
187
 
158
188
  {/* Menu items */}
@@ -1,3 +1,5 @@
1
+ 'use client';
2
+
1
3
  /**
2
4
  * Account Components for @payez/next-mvp
3
5
  *
@@ -6,3 +8,6 @@
6
8
 
7
9
  export { UserAvatarMenu } from './UserAvatarMenu';
8
10
  export type { UserAvatarMenuProps } from './UserAvatarMenu';
11
+
12
+ export { MobileNavDrawer } from './MobileNavDrawer';
13
+ export type { MobileNavDrawerProps, NavItem, NavSection } from './MobileNavDrawer';
package/src/index.ts CHANGED
@@ -29,8 +29,8 @@ export { isUnauthenticatedRoute, configurePublicRoutes, getRouteConfig } from '.
29
29
  export { createMvpMiddleware } from './middleware/create-middleware';
30
30
 
31
31
  // Account Components
32
- export { UserAvatarMenu } from './components/account';
33
- export type { UserAvatarMenuProps } from './components/account';
32
+ export { UserAvatarMenu, MobileNavDrawer } from './components/account';
33
+ export type { UserAvatarMenuProps, MobileNavDrawerProps, NavItem, NavSection } from './components/account';
34
34
 
35
35
  // Admin Logging & Analytics (client-side components and hooks)
36
36
  export {