@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.
- package/dist/api-handlers/admin/index.d.ts +1 -0
- package/dist/api-handlers/admin/index.js +3 -1
- package/dist/api-handlers/admin/stats.d.ts +21 -0
- package/dist/api-handlers/admin/stats.js +240 -0
- package/dist/auth/utils/idp-client.js +1 -0
- package/dist/components/account/MobileNavDrawer.d.ts +32 -0
- package/dist/components/account/MobileNavDrawer.js +81 -0
- package/dist/components/account/UserAvatarMenu.js +5 -1
- package/dist/components/account/index.d.ts +2 -0
- package/dist/components/account/index.js +5 -2
- package/dist/index.d.ts +2 -2
- package/dist/index.js +2 -1
- package/dist/pages/admin-page-permissions/PagePermissionsAdminPage.d.ts +18 -0
- package/dist/pages/admin-page-permissions/PagePermissionsAdminPage.js +276 -0
- package/dist/pages/admin-page-permissions/index.d.ts +6 -0
- package/dist/pages/admin-page-permissions/index.js +13 -0
- package/dist/pages/admin-roles/RolesAdminPage.d.ts +12 -11
- package/dist/pages/admin-roles/RolesAdminPage.js +249 -66
- package/dist/routes/auth/session.d.ts +1 -30
- package/dist/routes/auth/session.js +3 -4
- package/package.json +6 -1
- package/src/api-handlers/admin/index.ts +5 -0
- package/src/api-handlers/admin/stats.ts +240 -0
- package/src/auth/utils/idp-client.ts +1 -0
- package/src/components/account/MobileNavDrawer.tsx +305 -0
- package/src/components/account/UserAvatarMenu.tsx +47 -17
- package/src/components/account/index.ts +5 -0
- package/src/index.ts +2 -2
- package/src/pages/admin-page-permissions/PagePermissionsAdminPage.tsx +527 -0
- package/src/pages/admin-page-permissions/index.ts +7 -0
- package/src/pages/admin-roles/RolesAdminPage.tsx +494 -318
- 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
|
+
}
|
|
@@ -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-
|
|
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
|
-
{
|
|
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
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
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 {
|