@payez/next-mvp 3.6.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/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/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,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 {
|