@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.
@@ -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 {