@devvistatech/devvista-kit 0.0.10 → 0.0.13
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/CHANGELOG.md +12 -12
- package/LICENSE +6 -6
- package/README.md +55 -15
- package/app/ClientLayout.tsx +66 -0
- package/app/about/page.tsx +61 -298
- package/app/adRequest/page.tsx +625 -549
- package/app/admin-profile/page.tsx +123 -0
- package/app/analytics/page.tsx +382 -346
- package/app/api/about/route.ts +290 -306
- package/app/api/adRequest/route.ts +547 -567
- package/app/api/analytics/[reportType]/route.ts +274 -337
- package/app/api/bio/route.ts +297 -313
- package/app/api/blog/route.ts +288 -306
- package/app/api/chat/route.ts +14 -14
- package/app/api/contact/route.ts +409 -409
- package/app/api/contacts/route.ts +179 -224
- package/app/api/files/route.ts +415 -429
- package/app/api/gallery-data/route.ts +727 -735
- package/app/api/schedule/route.ts +439 -455
- package/app/api/signup/route.ts +129 -0
- package/app/api/sync-user/route.ts +306 -132
- package/app/api/trial-request/route.ts +297 -297
- package/app/api/verify-admin/route.ts +46 -0
- package/app/blog/[id]/page.tsx +307 -288
- package/app/blog/page.tsx +249 -216
- package/app/contact/page.tsx +284 -284
- package/app/faq/page.tsx +191 -191
- package/app/favicon.ico +0 -0
- package/app/gallery/page.tsx +336 -315
- package/app/globals.css +58 -58
- package/app/layout.tsx +59 -110
- package/app/not-found.tsx +20 -20
- package/app/page.tsx +47 -338
- package/app/products/constants/product.ts +27 -0
- package/app/products/page.tsx +296 -0
- package/app/products/productOne/page.tsx +266 -0
- package/app/products/productTwo/page.tsx +272 -0
- package/app/schedule/page.tsx +698 -660
- package/bin/init.js +207 -219
- package/components/addOns/functional/CalendlyWidget.tsx +107 -107
- package/components/addOns/functional/ClassList.tsx +149 -145
- package/components/addOns/functional/ClassPopup.tsx +398 -398
- package/components/addOns/functional/ContactForm.tsx +284 -284
- package/components/addOns/functional/NewUserAnalytics.tsx +100 -100
- package/components/addOns/functional/ProductList.tsx +1027 -0
- package/components/addOns/functional/aboutSections/AboutSection.tsx +581 -544
- package/components/addOns/functional/aboutSections/constants/aboutSection.ts +70 -65
- package/components/addOns/functional/banner/Banner.tsx +150 -0
- package/components/addOns/functional/banner/BannerDashboard.tsx +283 -0
- package/components/addOns/functional/bioSections/BioEditor.tsx +471 -0
- package/components/addOns/functional/bioSections/constants/bioEditor.ts +36 -0
- package/components/addOns/functional/blogSections/BlogDashboard.tsx +184 -184
- package/components/addOns/functional/blogSections/BlogFormPopUp.tsx +555 -554
- package/components/addOns/functional/blogSections/BlogList.tsx +148 -148
- package/components/addOns/functional/blogSections/BlogSidebar.tsx +58 -58
- package/components/addOns/functional/blogSections/constants/blogDashboard.ts +28 -28
- package/components/addOns/functional/blogSections/constants/blogFormPopUp.ts +97 -97
- package/components/addOns/functional/blogSections/constants/blogList.ts +22 -22
- package/components/addOns/functional/blogSections/constants/blogSidebar.ts +15 -15
- package/components/addOns/functional/{ImageDescCarousel.tsx → carousels/ImageDescCarousel.tsx} +839 -730
- package/components/addOns/functional/carousels/ProductDescCarousel.tsx +1129 -0
- package/components/addOns/functional/{ScheduleCarousel.tsx → carousels/ScheduleCarousel.tsx} +231 -171
- package/components/addOns/functional/carousels/constants.ts/productDescCarousel.ts +197 -0
- package/components/addOns/functional/carousels/constants.ts/scheduleCarousel.ts +20 -0
- package/components/addOns/functional/contactsDashboard/ContactsDashboard.tsx +366 -366
- package/components/addOns/functional/contactsDashboard/constants/contactsDashboard.ts +70 -70
- package/components/addOns/functional/fileUploaders/FileUploader.tsx +437 -0
- package/components/addOns/functional/fileUploaders/constants/fileUploader.ts +45 -0
- package/components/addOns/functional/galleries/GalleryComplex.tsx +1037 -836
- package/components/addOns/functional/galleries/GallerySimple.tsx +537 -509
- package/components/addOns/functional/galleries/ThreeSetGallery.tsx +260 -0
- package/components/addOns/functional/galleries/constants/galleryComplex.ts +106 -106
- package/components/addOns/functional/galleries/constants/gallerySimple.ts +76 -76
- package/components/addOns/functional/schedules/ScheduleGridOne.tsx +276 -262
- package/components/addOns/functional/schedules/ScheduleGridTwo.tsx +299 -294
- package/components/addOns/functional/schedules/ScheduleGridTwoBasic.tsx +293 -288
- package/components/addOns/functional/schedules/SchedulerForm.tsx +428 -428
- package/components/addOns/functional/schedules/constants/ScheduleGridTwo.ts +40 -40
- package/components/addOns/functional/schedules/constants/ScheduleGridTwoBasic.ts +40 -40
- package/components/addOns/functional/schedules/constants/SchedulerForm.ts +65 -65
- package/components/addOns/functional/schedules/constants/scheduleGridOne.ts +54 -54
- package/components/addOns/non-functional/AnnouncementBanner.tsx +46 -46
- package/components/addOns/non-functional/IconBubble.tsx +49 -49
- package/components/addOns/non-functional/SampleCarousel.tsx +204 -204
- package/components/addOns/non-functional/Testimonials.tsx +334 -334
- package/components/addOns/non-functional/ThreeSetGallery.tsx +63 -63
- package/components/addOns/non-functional/aboutSections/AboutSection.tsx +62 -62
- package/components/addOns/non-functional/aboutSections/constants/aboutSection.ts +24 -24
- package/components/addOns/non-functional/featureSections/FeaturesSection.tsx +74 -0
- package/components/addOns/non-functional/featureSections/constants/featuresSection.ts +30 -0
- package/components/addOns/non-functional/{Heros/HeroSection.tsx → heros/HomeHero.tsx} +144 -142
- package/components/addOns/non-functional/heros/ProductHero.tsx +111 -0
- package/components/addOns/non-functional/heros/constants/hero.ts +62 -0
- package/components/addOns/non-functional/imageCarousels/ProductSlider.tsx +117 -117
- package/components/addOns/non-functional/imageCarousels/ProgramCarousel.tsx +232 -232
- package/components/addOns/non-functional/imageCarousels/constants/programCarousel.ts +39 -39
- package/components/addOns/non-functional/imageCarousels/constants/programSlider.ts +36 -36
- package/components/addOns/non-functional/spinner.tsx +21 -21
- package/components/footers/footer.tsx +416 -453
- package/components/navBars/navbar.tsx +310 -310
- package/components/other/accordion.tsx +58 -58
- package/components/other/admin-menu.tsx +68 -68
- package/components/other/alert-dialog.tsx +141 -141
- package/components/other/alert.tsx +59 -59
- package/components/other/aspect-ratio.tsx +7 -7
- package/components/other/avatar.tsx +50 -50
- package/components/other/badge.tsx +36 -36
- package/components/other/breadcrumb.tsx +115 -115
- package/components/other/button.tsx +738 -738
- package/components/other/calendar.tsx +66 -66
- package/components/other/card.tsx +86 -86
- package/components/other/carousel.tsx +274 -274
- package/components/other/chart.tsx +363 -363
- package/components/other/checkbox.tsx +30 -30
- package/components/other/collapsible.tsx +11 -11
- package/components/other/command.tsx +155 -155
- package/components/other/context-menu.tsx +200 -200
- package/components/other/dialog.tsx +122 -122
- package/components/other/drawer.tsx +118 -118
- package/components/other/dropdown-menu.tsx +200 -200
- package/components/other/form.tsx +179 -179
- package/components/other/hover-card.tsx +29 -29
- package/components/other/input-otp.tsx +71 -71
- package/components/other/input.tsx +25 -25
- package/components/other/label.tsx +26 -26
- package/components/other/menubar.tsx +236 -236
- package/components/other/mobile-icon.tsx +21 -21
- package/components/other/navigation-menu.tsx +128 -128
- package/components/other/pagination.tsx +117 -117
- package/components/other/popover.tsx +31 -31
- package/components/other/progress.tsx +28 -28
- package/components/other/radio-group.tsx +44 -44
- package/components/other/resizable.tsx +45 -45
- package/components/other/scroll-area.tsx +48 -48
- package/components/other/select.tsx +160 -160
- package/components/other/separator.tsx +31 -31
- package/components/other/sheet.tsx +140 -140
- package/components/other/skeleton.tsx +15 -15
- package/components/other/slider.tsx +28 -28
- package/components/other/social-icons.tsx +39 -39
- package/components/other/sonner.tsx +31 -31
- package/components/other/switch.tsx +29 -29
- package/components/other/table.tsx +117 -117
- package/components/other/tabs.tsx +55 -55
- package/components/other/textarea.tsx +24 -24
- package/components/other/toast.tsx +122 -122
- package/components/other/toaster.tsx +35 -35
- package/components/other/toggle-group.tsx +61 -61
- package/components/other/toggle.tsx +45 -45
- package/components/other/tooltip.tsx +30 -30
- package/components/theme-provider.tsx +8 -8
- package/hooks/use-toast.ts +188 -188
- package/lib/auth/auth-context.tsx +225 -0
- package/lib/auth/auth-utils.tsx +30 -0
- package/lib/constants/about.ts +34 -34
- package/lib/constants/adRequest.ts +256 -113
- package/lib/constants/admin-profile.ts +12 -0
- package/lib/constants/contact.ts +40 -40
- package/lib/constants/faq.ts +34 -34
- package/lib/constants/gallery.ts +42 -42
- package/lib/constants/page.ts +69 -69
- package/lib/constants/schedule.ts +71 -71
- package/lib/google/google-analytics-tracking.tsx +44 -0
- package/lib/{google-analytics.tsx → google/google-analytics.tsx} +97 -97
- package/lib/types.ts +235 -0
- package/lib/utils/compressImage.tsx +32 -0
- package/middleware.ts +46 -42
- package/netlify.toml +5 -5
- package/next.config.js +10 -10
- package/package.json +117 -116
- package/public/images/test.png +0 -0
- package/tailwind.config.ts +89 -89
- package/tsconfig.json +23 -23
- package/components/addOns/functional/BioEditor.tsx +0 -447
- package/components/addOns/functional/FileUploader.tsx +0 -295
- package/components/addOns/non-functional/FeaturesSection.tsx +0 -63
- package/components/types.ts +0 -50
- package/dist/.next/types/app/api/about/route.js +0 -52
- package/dist/.next/types/app/api/blog/route.js +0 -52
- package/dist/.next/types/app/api/files/route.js +0 -52
- package/dist/.next/types/app/api/schedule/route.js +0 -52
- package/dist/.next/types/app/api/sync-user/route.js +0 -52
- package/dist/.next/types/app/layout.js +0 -22
- package/dist/.next/types/app/page.js +0 -22
- package/dist/app/about/page.jsx +0 -258
- package/dist/app/adRequest/page.jsx +0 -531
- package/dist/app/analytics/page.jsx +0 -298
- package/dist/app/api/about/route.js +0 -285
- package/dist/app/api/adRequest/route.js +0 -440
- package/dist/app/api/analytics/[reportType]/route.js +0 -357
- package/dist/app/api/bio/route.js +0 -293
- package/dist/app/api/blog/route.js +0 -366
- package/dist/app/api/chat/route.js +0 -58
- package/dist/app/api/contact/route.js +0 -163
- package/dist/app/api/contacts/route.js +0 -234
- package/dist/app/api/files/route.js +0 -444
- package/dist/app/api/gallery-data/route.js +0 -719
- package/dist/app/api/schedule/route.js +0 -461
- package/dist/app/api/sync-user/route.js +0 -186
- package/dist/app/api/trial-request/route.js +0 -165
- package/dist/app/blog/[id]/page.jsx +0 -312
- package/dist/app/blog/page.jsx +0 -210
- package/dist/app/constants/about.js +0 -32
- package/dist/app/constants/adRequest.js +0 -113
- package/dist/app/constants/contact.js +0 -40
- package/dist/app/constants/faq.js +0 -36
- package/dist/app/constants/gallery.js +0 -42
- package/dist/app/constants/page.js +0 -69
- package/dist/app/constants/schedule.js +0 -71
- package/dist/app/contact/page.jsx +0 -119
- package/dist/app/faq/page.jsx +0 -97
- package/dist/app/gallery/page.jsx +0 -281
- package/dist/app/layout.jsx +0 -45
- package/dist/app/not-found.jsx +0 -14
- package/dist/app/page.jsx +0 -324
- package/dist/app/schedule/page.jsx +0 -500
- package/dist/components/addOns/functional/BioEditor.jsx +0 -187
- package/dist/components/addOns/functional/CalendlyWidget.jsx +0 -61
- package/dist/components/addOns/functional/ClassList.jsx +0 -158
- package/dist/components/addOns/functional/ClassPopup.jsx +0 -300
- package/dist/components/addOns/functional/ContactForm.jsx +0 -219
- package/dist/components/addOns/functional/FileUploader.jsx +0 -222
- package/dist/components/addOns/functional/ImageDescCarousel.jsx +0 -491
- package/dist/components/addOns/functional/NewUserAnalytics.jsx +0 -71
- package/dist/components/addOns/functional/ScheduleCarousel.jsx +0 -68
- package/dist/components/addOns/functional/aboutSections/AboutSection.jsx +0 -372
- package/dist/components/addOns/functional/aboutSections/constants/aboutSection.js +0 -65
- package/dist/components/addOns/functional/blogSections/BlogDashboard.jsx +0 -111
- package/dist/components/addOns/functional/blogSections/BlogFormPopUp.jsx +0 -465
- package/dist/components/addOns/functional/blogSections/BlogList.jsx +0 -170
- package/dist/components/addOns/functional/blogSections/BlogSidebar.jsx +0 -35
- package/dist/components/addOns/functional/blogSections/constants/blogDashboard.js +0 -28
- package/dist/components/addOns/functional/blogSections/constants/blogFormPopUp.js +0 -97
- package/dist/components/addOns/functional/blogSections/constants/blogList.js +0 -22
- package/dist/components/addOns/functional/blogSections/constants/blogSidebar.js +0 -15
- package/dist/components/addOns/functional/contactsDashboard/ContactsDashboard.jsx +0 -355
- package/dist/components/addOns/functional/contactsDashboard/constants/contactsDashboard.js +0 -70
- package/dist/components/addOns/functional/galleries/GalleryComplex.jsx +0 -605
- package/dist/components/addOns/functional/galleries/GallerySimple.jsx +0 -363
- package/dist/components/addOns/functional/galleries/constants/galleryComplex.js +0 -106
- package/dist/components/addOns/functional/galleries/constants/gallerySimple.js +0 -76
- package/dist/components/addOns/functional/schedules/ScheduleGridOne.jsx +0 -167
- package/dist/components/addOns/functional/schedules/ScheduleGridTwo.jsx +0 -100
- package/dist/components/addOns/functional/schedules/ScheduleGridTwoBasic.jsx +0 -97
- package/dist/components/addOns/functional/schedules/SchedulerForm.jsx +0 -188
- package/dist/components/addOns/functional/schedules/constants/ScheduleGridTwo.js +0 -40
- package/dist/components/addOns/functional/schedules/constants/ScheduleGridTwoBasic.js +0 -40
- package/dist/components/addOns/functional/schedules/constants/SchedulerForm.js +0 -65
- package/dist/components/addOns/functional/schedules/constants/scheduleGridOne.js +0 -54
- package/dist/components/addOns/non-functional/AnnouncementBanner.jsx +0 -24
- package/dist/components/addOns/non-functional/FeaturesSection.jsx +0 -38
- package/dist/components/addOns/non-functional/HeroSection.jsx +0 -71
- package/dist/components/addOns/non-functional/Heros/HeroSection.jsx +0 -71
- package/dist/components/addOns/non-functional/IconBubble.jsx +0 -36
- package/dist/components/addOns/non-functional/SampleCarousel.jsx +0 -114
- package/dist/components/addOns/non-functional/Testimonials.jsx +0 -177
- package/dist/components/addOns/non-functional/ThreeSetGallery.jsx +0 -40
- package/dist/components/addOns/non-functional/aboutSections/AboutSection.jsx +0 -35
- package/dist/components/addOns/non-functional/aboutSections/constants/aboutSection.js +0 -24
- package/dist/components/addOns/non-functional/imageCarousels/ProductSlider.jsx +0 -80
- package/dist/components/addOns/non-functional/imageCarousels/ProgramCarousel.jsx +0 -155
- package/dist/components/addOns/non-functional/imageCarousels/constants/programCarousel.js +0 -39
- package/dist/components/addOns/non-functional/imageCarousels/constants/programSlider.js +0 -36
- package/dist/components/addOns/non-functional/spinner.jsx +0 -13
- package/dist/components/footers/footer.jsx +0 -217
- package/dist/components/navBars/navbar.jsx +0 -159
- package/dist/components/other/accordion.jsx +0 -40
- package/dist/components/other/admin-menu.jsx +0 -34
- package/dist/components/other/alert-dialog.jsx +0 -64
- package/dist/components/other/alert.jsx +0 -41
- package/dist/components/other/aspect-ratio.jsx +0 -4
- package/dist/components/other/avatar.jsx +0 -31
- package/dist/components/other/badge.jsx +0 -32
- package/dist/components/other/breadcrumb.jsx +0 -57
- package/dist/components/other/button.jsx +0 -322
- package/dist/components/other/calendar.jsx +0 -43
- package/dist/components/other/card.jsx +0 -44
- package/dist/components/other/carousel.jsx +0 -140
- package/dist/components/other/chart.jsx +0 -182
- package/dist/components/other/checkbox.jsx +0 -26
- package/dist/components/other/collapsible.jsx +0 -6
- package/dist/components/other/command.jsx +0 -68
- package/dist/components/other/context-menu.jsx +0 -88
- package/dist/components/other/dialog.jsx +0 -60
- package/dist/components/other/drawer.jsx +0 -60
- package/dist/components/other/dropdown-menu.jsx +0 -90
- package/dist/components/other/form.jsx +0 -89
- package/dist/components/other/hover-card.jsx +0 -23
- package/dist/components/other/input-otp.jsx +0 -46
- package/dist/components/other/input.jsx +0 -19
- package/dist/components/other/label.jsx +0 -23
- package/dist/components/other/login-popup.jsx +0 -1
- package/dist/components/other/menubar.jsx +0 -96
- package/dist/components/other/mobile-icon.jsx +0 -11
- package/dist/components/other/navigation-menu.jsx +0 -62
- package/dist/components/other/pagination.jsx +0 -63
- package/dist/components/other/popover.jsx +0 -25
- package/dist/components/other/progress.jsx +0 -23
- package/dist/components/other/radio-group.jsx +0 -31
- package/dist/components/other/resizable.jsx +0 -29
- package/dist/components/other/scroll-area.jsx +0 -36
- package/dist/components/other/select.jsx +0 -83
- package/dist/components/other/separator.jsx +0 -21
- package/dist/components/other/sheet.jsx +0 -74
- package/dist/components/other/signup-popup.jsx +0 -1
- package/dist/components/other/skeleton.jsx +0 -17
- package/dist/components/other/slider.jsx +0 -26
- package/dist/components/other/social-icons.jsx +0 -15
- package/dist/components/other/sonner.jsx +0 -27
- package/dist/components/other/switch.jsx +0 -23
- package/dist/components/other/table.jsx +0 -56
- package/dist/components/other/tabs.jsx +0 -32
- package/dist/components/other/textarea.jsx +0 -19
- package/dist/components/other/toast.jsx +0 -58
- package/dist/components/other/toaster.jsx +0 -31
- package/dist/components/other/toggle-group.jsx +0 -41
- package/dist/components/other/toggle.jsx +0 -39
- package/dist/components/other/tooltip.jsx +0 -24
- package/dist/components/theme-provider.jsx +0 -18
- package/dist/components/types.js +0 -1
- package/dist/hooks/use-toast.js +0 -135
- package/dist/lib/auth-context.jsx +0 -144
- package/dist/lib/constants/about.js +0 -32
- package/dist/lib/constants/adRequest.js +0 -113
- package/dist/lib/constants/contact.js +0 -40
- package/dist/lib/constants/faq.js +0 -36
- package/dist/lib/constants/gallery.js +0 -42
- package/dist/lib/constants/page.js +0 -69
- package/dist/lib/constants/schedule.js +0 -71
- package/dist/lib/google-analytics.jsx +0 -148
- package/dist/lib/utils.js +0 -9
- package/dist/lib/verify-user.js +0 -142
- package/dist/middleware.js +0 -37
- package/dist/tailwind.config.js +0 -86
- package/dist/tsconfig.tsbuildinfo +0 -1
- package/lib/auth-context.tsx +0 -131
- package/lib/verify-user.ts +0 -118
package/hooks/use-toast.ts
CHANGED
|
@@ -1,188 +1,188 @@
|
|
|
1
|
-
'use client';
|
|
2
|
-
|
|
3
|
-
import * as React from 'react';
|
|
4
|
-
|
|
5
|
-
import type { ToastActionElement, ToastProps } from '@/components/other/toast';
|
|
6
|
-
|
|
7
|
-
const TOAST_LIMIT = 1;
|
|
8
|
-
const TOAST_REMOVE_DELAY = 1000000;
|
|
9
|
-
|
|
10
|
-
type ToasterToast = ToastProps & {
|
|
11
|
-
id: string;
|
|
12
|
-
title?: React.ReactNode;
|
|
13
|
-
description?: React.ReactNode;
|
|
14
|
-
action?: ToastActionElement;
|
|
15
|
-
};
|
|
16
|
-
|
|
17
|
-
const actionTypes = {
|
|
18
|
-
ADD_TOAST: 'ADD_TOAST',
|
|
19
|
-
UPDATE_TOAST: 'UPDATE_TOAST',
|
|
20
|
-
DISMISS_TOAST: 'DISMISS_TOAST',
|
|
21
|
-
REMOVE_TOAST: 'REMOVE_TOAST',
|
|
22
|
-
} as const;
|
|
23
|
-
|
|
24
|
-
let count = 0;
|
|
25
|
-
|
|
26
|
-
function genId() {
|
|
27
|
-
count = (count + 1) % Number.MAX_SAFE_INTEGER;
|
|
28
|
-
return count.toString();
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
type ActionType = typeof actionTypes;
|
|
32
|
-
|
|
33
|
-
type Action =
|
|
34
|
-
| {
|
|
35
|
-
type: ActionType['ADD_TOAST'];
|
|
36
|
-
toast: ToasterToast;
|
|
37
|
-
}
|
|
38
|
-
| {
|
|
39
|
-
type: ActionType['UPDATE_TOAST'];
|
|
40
|
-
toast: Partial<ToasterToast>;
|
|
41
|
-
}
|
|
42
|
-
| {
|
|
43
|
-
type: ActionType['DISMISS_TOAST'];
|
|
44
|
-
toastId?: ToasterToast['id'];
|
|
45
|
-
}
|
|
46
|
-
| {
|
|
47
|
-
type: ActionType['REMOVE_TOAST'];
|
|
48
|
-
toastId?: ToasterToast['id'];
|
|
49
|
-
};
|
|
50
|
-
|
|
51
|
-
interface State {
|
|
52
|
-
toasts: ToasterToast[];
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
const toastTimeouts = new Map<string, ReturnType<typeof setTimeout>>();
|
|
56
|
-
|
|
57
|
-
const addToRemoveQueue = (toastId: string) => {
|
|
58
|
-
if (toastTimeouts.has(toastId)) {
|
|
59
|
-
return;
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
const timeout = setTimeout(() => {
|
|
63
|
-
toastTimeouts.delete(toastId);
|
|
64
|
-
dispatch({
|
|
65
|
-
type: 'REMOVE_TOAST',
|
|
66
|
-
toastId: toastId,
|
|
67
|
-
});
|
|
68
|
-
}, TOAST_REMOVE_DELAY);
|
|
69
|
-
|
|
70
|
-
toastTimeouts.set(toastId, timeout);
|
|
71
|
-
};
|
|
72
|
-
|
|
73
|
-
export const reducer = (state: State, action: Action): State => {
|
|
74
|
-
switch (action.type) {
|
|
75
|
-
case 'ADD_TOAST':
|
|
76
|
-
return {
|
|
77
|
-
...state,
|
|
78
|
-
toasts: [action.toast, ...state.toasts].slice(0, TOAST_LIMIT),
|
|
79
|
-
};
|
|
80
|
-
|
|
81
|
-
case 'UPDATE_TOAST':
|
|
82
|
-
return {
|
|
83
|
-
...state,
|
|
84
|
-
toasts: state.toasts.map((t) =>
|
|
85
|
-
t.id === action.toast.id ? { ...t, ...action.toast } : t
|
|
86
|
-
),
|
|
87
|
-
};
|
|
88
|
-
|
|
89
|
-
case 'DISMISS_TOAST': {
|
|
90
|
-
const { toastId } = action;
|
|
91
|
-
|
|
92
|
-
if (toastId) {
|
|
93
|
-
addToRemoveQueue(toastId);
|
|
94
|
-
} else {
|
|
95
|
-
state.toasts.forEach((toast) => {
|
|
96
|
-
addToRemoveQueue(toast.id);
|
|
97
|
-
});
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
return {
|
|
101
|
-
...state,
|
|
102
|
-
toasts: state.toasts.map((t) =>
|
|
103
|
-
t.id === toastId || toastId === undefined
|
|
104
|
-
? {
|
|
105
|
-
...t,
|
|
106
|
-
open: false,
|
|
107
|
-
}
|
|
108
|
-
: t
|
|
109
|
-
),
|
|
110
|
-
};
|
|
111
|
-
}
|
|
112
|
-
case 'REMOVE_TOAST':
|
|
113
|
-
if (action.toastId === undefined) {
|
|
114
|
-
return {
|
|
115
|
-
...state,
|
|
116
|
-
toasts: [],
|
|
117
|
-
};
|
|
118
|
-
}
|
|
119
|
-
return {
|
|
120
|
-
...state,
|
|
121
|
-
toasts: state.toasts.filter((t) => t.id !== action.toastId),
|
|
122
|
-
};
|
|
123
|
-
}
|
|
124
|
-
};
|
|
125
|
-
|
|
126
|
-
const listeners: Array<(state: State) => void> = [];
|
|
127
|
-
|
|
128
|
-
let memoryState: State = { toasts: [] };
|
|
129
|
-
|
|
130
|
-
function dispatch(action: Action) {
|
|
131
|
-
memoryState = reducer(memoryState, action);
|
|
132
|
-
listeners.forEach((listener) => {
|
|
133
|
-
listener(memoryState);
|
|
134
|
-
});
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
type Toast = Omit<ToasterToast, 'id'>;
|
|
138
|
-
|
|
139
|
-
function toast({ ...props }: Toast) {
|
|
140
|
-
const id = genId();
|
|
141
|
-
|
|
142
|
-
const update = (props: ToasterToast) =>
|
|
143
|
-
dispatch({
|
|
144
|
-
type: 'UPDATE_TOAST',
|
|
145
|
-
toast: { ...props, id },
|
|
146
|
-
});
|
|
147
|
-
const dismiss = () => dispatch({ type: 'DISMISS_TOAST', toastId: id });
|
|
148
|
-
|
|
149
|
-
dispatch({
|
|
150
|
-
type: 'ADD_TOAST',
|
|
151
|
-
toast: {
|
|
152
|
-
...props,
|
|
153
|
-
id,
|
|
154
|
-
open: true,
|
|
155
|
-
onOpenChange: (open) => {
|
|
156
|
-
if (!open) dismiss();
|
|
157
|
-
},
|
|
158
|
-
},
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
return {
|
|
162
|
-
id: id,
|
|
163
|
-
dismiss,
|
|
164
|
-
update,
|
|
165
|
-
};
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
function useToast() {
|
|
169
|
-
const [state, setState] = React.useState<State>(memoryState);
|
|
170
|
-
|
|
171
|
-
React.useEffect(() => {
|
|
172
|
-
listeners.push(setState);
|
|
173
|
-
return () => {
|
|
174
|
-
const index = listeners.indexOf(setState);
|
|
175
|
-
if (index > -1) {
|
|
176
|
-
listeners.splice(index, 1);
|
|
177
|
-
}
|
|
178
|
-
};
|
|
179
|
-
}, [state]);
|
|
180
|
-
|
|
181
|
-
return {
|
|
182
|
-
...state,
|
|
183
|
-
toast,
|
|
184
|
-
dismiss: (toastId?: string) => dispatch({ type: 'DISMISS_TOAST', toastId }),
|
|
185
|
-
};
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
export { useToast, toast };
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import * as React from 'react';
|
|
4
|
+
|
|
5
|
+
import type { ToastActionElement, ToastProps } from '@/components/other/toast';
|
|
6
|
+
|
|
7
|
+
const TOAST_LIMIT = 1;
|
|
8
|
+
const TOAST_REMOVE_DELAY = 1000000;
|
|
9
|
+
|
|
10
|
+
type ToasterToast = ToastProps & {
|
|
11
|
+
id: string;
|
|
12
|
+
title?: React.ReactNode;
|
|
13
|
+
description?: React.ReactNode;
|
|
14
|
+
action?: ToastActionElement;
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
const actionTypes = {
|
|
18
|
+
ADD_TOAST: 'ADD_TOAST',
|
|
19
|
+
UPDATE_TOAST: 'UPDATE_TOAST',
|
|
20
|
+
DISMISS_TOAST: 'DISMISS_TOAST',
|
|
21
|
+
REMOVE_TOAST: 'REMOVE_TOAST',
|
|
22
|
+
} as const;
|
|
23
|
+
|
|
24
|
+
let count = 0;
|
|
25
|
+
|
|
26
|
+
function genId() {
|
|
27
|
+
count = (count + 1) % Number.MAX_SAFE_INTEGER;
|
|
28
|
+
return count.toString();
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
type ActionType = typeof actionTypes;
|
|
32
|
+
|
|
33
|
+
type Action =
|
|
34
|
+
| {
|
|
35
|
+
type: ActionType['ADD_TOAST'];
|
|
36
|
+
toast: ToasterToast;
|
|
37
|
+
}
|
|
38
|
+
| {
|
|
39
|
+
type: ActionType['UPDATE_TOAST'];
|
|
40
|
+
toast: Partial<ToasterToast>;
|
|
41
|
+
}
|
|
42
|
+
| {
|
|
43
|
+
type: ActionType['DISMISS_TOAST'];
|
|
44
|
+
toastId?: ToasterToast['id'];
|
|
45
|
+
}
|
|
46
|
+
| {
|
|
47
|
+
type: ActionType['REMOVE_TOAST'];
|
|
48
|
+
toastId?: ToasterToast['id'];
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
interface State {
|
|
52
|
+
toasts: ToasterToast[];
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const toastTimeouts = new Map<string, ReturnType<typeof setTimeout>>();
|
|
56
|
+
|
|
57
|
+
const addToRemoveQueue = (toastId: string) => {
|
|
58
|
+
if (toastTimeouts.has(toastId)) {
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const timeout = setTimeout(() => {
|
|
63
|
+
toastTimeouts.delete(toastId);
|
|
64
|
+
dispatch({
|
|
65
|
+
type: 'REMOVE_TOAST',
|
|
66
|
+
toastId: toastId,
|
|
67
|
+
});
|
|
68
|
+
}, TOAST_REMOVE_DELAY);
|
|
69
|
+
|
|
70
|
+
toastTimeouts.set(toastId, timeout);
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
export const reducer = (state: State, action: Action): State => {
|
|
74
|
+
switch (action.type) {
|
|
75
|
+
case 'ADD_TOAST':
|
|
76
|
+
return {
|
|
77
|
+
...state,
|
|
78
|
+
toasts: [action.toast, ...state.toasts].slice(0, TOAST_LIMIT),
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
case 'UPDATE_TOAST':
|
|
82
|
+
return {
|
|
83
|
+
...state,
|
|
84
|
+
toasts: state.toasts.map((t) =>
|
|
85
|
+
t.id === action.toast.id ? { ...t, ...action.toast } : t
|
|
86
|
+
),
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
case 'DISMISS_TOAST': {
|
|
90
|
+
const { toastId } = action;
|
|
91
|
+
|
|
92
|
+
if (toastId) {
|
|
93
|
+
addToRemoveQueue(toastId);
|
|
94
|
+
} else {
|
|
95
|
+
state.toasts.forEach((toast) => {
|
|
96
|
+
addToRemoveQueue(toast.id);
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
return {
|
|
101
|
+
...state,
|
|
102
|
+
toasts: state.toasts.map((t) =>
|
|
103
|
+
t.id === toastId || toastId === undefined
|
|
104
|
+
? {
|
|
105
|
+
...t,
|
|
106
|
+
open: false,
|
|
107
|
+
}
|
|
108
|
+
: t
|
|
109
|
+
),
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
case 'REMOVE_TOAST':
|
|
113
|
+
if (action.toastId === undefined) {
|
|
114
|
+
return {
|
|
115
|
+
...state,
|
|
116
|
+
toasts: [],
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
return {
|
|
120
|
+
...state,
|
|
121
|
+
toasts: state.toasts.filter((t) => t.id !== action.toastId),
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
};
|
|
125
|
+
|
|
126
|
+
const listeners: Array<(state: State) => void> = [];
|
|
127
|
+
|
|
128
|
+
let memoryState: State = { toasts: [] };
|
|
129
|
+
|
|
130
|
+
function dispatch(action: Action) {
|
|
131
|
+
memoryState = reducer(memoryState, action);
|
|
132
|
+
listeners.forEach((listener) => {
|
|
133
|
+
listener(memoryState);
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
type Toast = Omit<ToasterToast, 'id'>;
|
|
138
|
+
|
|
139
|
+
function toast({ ...props }: Toast) {
|
|
140
|
+
const id = genId();
|
|
141
|
+
|
|
142
|
+
const update = (props: ToasterToast) =>
|
|
143
|
+
dispatch({
|
|
144
|
+
type: 'UPDATE_TOAST',
|
|
145
|
+
toast: { ...props, id },
|
|
146
|
+
});
|
|
147
|
+
const dismiss = () => dispatch({ type: 'DISMISS_TOAST', toastId: id });
|
|
148
|
+
|
|
149
|
+
dispatch({
|
|
150
|
+
type: 'ADD_TOAST',
|
|
151
|
+
toast: {
|
|
152
|
+
...props,
|
|
153
|
+
id,
|
|
154
|
+
open: true,
|
|
155
|
+
onOpenChange: (open) => {
|
|
156
|
+
if (!open) dismiss();
|
|
157
|
+
},
|
|
158
|
+
},
|
|
159
|
+
});
|
|
160
|
+
|
|
161
|
+
return {
|
|
162
|
+
id: id,
|
|
163
|
+
dismiss,
|
|
164
|
+
update,
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function useToast() {
|
|
169
|
+
const [state, setState] = React.useState<State>(memoryState);
|
|
170
|
+
|
|
171
|
+
React.useEffect(() => {
|
|
172
|
+
listeners.push(setState);
|
|
173
|
+
return () => {
|
|
174
|
+
const index = listeners.indexOf(setState);
|
|
175
|
+
if (index > -1) {
|
|
176
|
+
listeners.splice(index, 1);
|
|
177
|
+
}
|
|
178
|
+
};
|
|
179
|
+
}, [state]);
|
|
180
|
+
|
|
181
|
+
return {
|
|
182
|
+
...state,
|
|
183
|
+
toast,
|
|
184
|
+
dismiss: (toastId?: string) => dispatch({ type: 'DISMISS_TOAST', toastId }),
|
|
185
|
+
};
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
export { useToast, toast };
|
|
@@ -0,0 +1,225 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { useState, useEffect, createContext, useContext } from "react";
|
|
4
|
+
import { useAuth as useClerkAuth, useUser } from "@clerk/nextjs";
|
|
5
|
+
import { StrapiUser } from "../types";
|
|
6
|
+
|
|
7
|
+
// Define the shape of the authentication context, including the Strapi user data,
|
|
8
|
+
// loading state, error state, and functions for session checking and post-signup handling.
|
|
9
|
+
interface AuthContextType {
|
|
10
|
+
user: StrapiUser | null;
|
|
11
|
+
authLoading: boolean;
|
|
12
|
+
error: string | null;
|
|
13
|
+
checkSession: () => Promise<void>;
|
|
14
|
+
handlePostSignUp: (clerkUserId: string, email: string, username: string) => Promise<void>;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
// Create the React context for sharing authentication state and methods across components.
|
|
18
|
+
const AuthContext = createContext<AuthContextType | undefined>(undefined);
|
|
19
|
+
|
|
20
|
+
// A simple in-memory cache to store Strapi user data by Clerk user ID, avoiding repeated API calls.
|
|
21
|
+
const userCache = new Map<string, StrapiUser>();
|
|
22
|
+
|
|
23
|
+
// The main AuthProvider component that wraps the app and manages authentication state.
|
|
24
|
+
export function AuthProvider({ children }: { children: React.ReactNode }) {
|
|
25
|
+
// Local state for the synced Strapi user, loading indicator, and any errors.
|
|
26
|
+
const [user, setUser] = useState<StrapiUser | null>(null);
|
|
27
|
+
const [authLoading, setAuthLoading] = useState(true);
|
|
28
|
+
const [error, setError] = useState<string | null>(null);
|
|
29
|
+
|
|
30
|
+
// Hooks from Clerk to get sign-in status, user ID, and user details.
|
|
31
|
+
const { isSignedIn, userId } = useClerkAuth();
|
|
32
|
+
const { user: clerkUser } = useUser();
|
|
33
|
+
|
|
34
|
+
// Function to synchronize a Clerk user with Strapi: check if the user exists, create if not, and cache the result.
|
|
35
|
+
const syncUserToStrapi = async (clerkUserData: { authId: string; email: string; username: string }) => {
|
|
36
|
+
// Early return if required data is missing.
|
|
37
|
+
if (!clerkUserData || !userId) {
|
|
38
|
+
console.warn("syncUserToStrapi: Missing clerkUserData or userId:", { clerkUserData, userId, timestamp: new Date().toISOString() });
|
|
39
|
+
return null;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Check cache first for quick retrieval.
|
|
43
|
+
if (userCache.has(userId)) {
|
|
44
|
+
const cachedUser = userCache.get(userId)!;
|
|
45
|
+
setUser(cachedUser);
|
|
46
|
+
setError(null);
|
|
47
|
+
return cachedUser;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
try {
|
|
51
|
+
// First, attempt to fetch the existing user from Strapi via the sync-user API.
|
|
52
|
+
const checkResponse = await fetch(`/api/sync-user?authId=${encodeURIComponent(clerkUserData.authId)}`, {
|
|
53
|
+
headers: {
|
|
54
|
+
"Content-Type": "application/json",
|
|
55
|
+
},
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
// If the user exists, cache and set it as the current user.
|
|
59
|
+
if (checkResponse.ok) {
|
|
60
|
+
const existingUser = await checkResponse.json();
|
|
61
|
+
userCache.set(userId, existingUser);
|
|
62
|
+
setUser(existingUser);
|
|
63
|
+
setError(null);
|
|
64
|
+
return existingUser;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// If no existing user, attempt to create a new one via the signup API.
|
|
68
|
+
const response = await fetch("/api/signup", {
|
|
69
|
+
method: "POST",
|
|
70
|
+
headers: {
|
|
71
|
+
"Content-Type": "application/json",
|
|
72
|
+
},
|
|
73
|
+
body: JSON.stringify(clerkUserData),
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
// Handle signup failure (e.g., duplicate user error) by retrying the check.
|
|
77
|
+
if (!response.ok) {
|
|
78
|
+
const errorData = await response.json();
|
|
79
|
+
console.warn("syncUserToStrapi: Failed to create user (likely duplicate):", { errorData, status: response.status, timestamp: new Date().toISOString() });
|
|
80
|
+
|
|
81
|
+
const retryResponse = await fetch(`/api/sync-user?authId=${encodeURIComponent(clerkUserData.authId)}`, {
|
|
82
|
+
headers: {
|
|
83
|
+
"Content-Type": "application/json",
|
|
84
|
+
},
|
|
85
|
+
});
|
|
86
|
+
if (retryResponse.ok) {
|
|
87
|
+
const existingUser = await retryResponse.json();
|
|
88
|
+
userCache.set(userId, existingUser);
|
|
89
|
+
setUser(existingUser);
|
|
90
|
+
setError(null);
|
|
91
|
+
return existingUser;
|
|
92
|
+
}
|
|
93
|
+
console.warn("syncUserToStrapi: Failed to sync user:", { errorData, status: response.status, timestamp: new Date().toISOString() });
|
|
94
|
+
setError(null);
|
|
95
|
+
setUser(null);
|
|
96
|
+
return null;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
// On successful creation, cache and set the new Strapi user.
|
|
100
|
+
const strapiUser = await response.json();
|
|
101
|
+
userCache.set(userId, strapiUser);
|
|
102
|
+
setUser(strapiUser);
|
|
103
|
+
setError(null);
|
|
104
|
+
return strapiUser;
|
|
105
|
+
} catch (error) {
|
|
106
|
+
// Log any unexpected errors but clear error state and set user to null.
|
|
107
|
+
console.warn("syncUserToStrapi: Error:", { error: error instanceof Error ? error.message : "Unknown error", timestamp: new Date().toISOString() });
|
|
108
|
+
setError(null);
|
|
109
|
+
setUser(null);
|
|
110
|
+
return null;
|
|
111
|
+
}
|
|
112
|
+
};
|
|
113
|
+
|
|
114
|
+
// Handler called after Clerk signup to sync the new user to Strapi.
|
|
115
|
+
const handlePostSignUp = async (clerkUserId: string, email: string, username: string) => {
|
|
116
|
+
// Early return if required data is missing.
|
|
117
|
+
if (!email || !username) {
|
|
118
|
+
console.warn("handlePostSignUp: Missing email or username", { clerkUserId, email, username, timestamp: new Date().toISOString() });
|
|
119
|
+
setError(null);
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
try {
|
|
124
|
+
// Delegate to the sync function.
|
|
125
|
+
const strapiUser = await syncUserToStrapi({
|
|
126
|
+
authId: clerkUserId,
|
|
127
|
+
email,
|
|
128
|
+
username,
|
|
129
|
+
});
|
|
130
|
+
if (!strapiUser) {
|
|
131
|
+
console.warn("handlePostSignUp: Failed to create Strapi user", { clerkUserId, timestamp: new Date().toISOString() });
|
|
132
|
+
setError(null);
|
|
133
|
+
}
|
|
134
|
+
} catch (error) {
|
|
135
|
+
// Log errors but clear error state.
|
|
136
|
+
console.warn("handlePostSignUp: Error:", {
|
|
137
|
+
error: error instanceof Error ? error.message : "Unknown error",
|
|
138
|
+
timestamp: new Date().toISOString(),
|
|
139
|
+
});
|
|
140
|
+
setError(null);
|
|
141
|
+
}
|
|
142
|
+
};
|
|
143
|
+
|
|
144
|
+
// Function to check and sync the current session on mount or changes.
|
|
145
|
+
const checkSession = async () => {
|
|
146
|
+
// Set loading state.
|
|
147
|
+
setAuthLoading(true);
|
|
148
|
+
// Early return if not signed in or missing user data.
|
|
149
|
+
if (!isSignedIn || !userId || !clerkUser) {
|
|
150
|
+
setUser(null);
|
|
151
|
+
setAuthLoading(false);
|
|
152
|
+
setError(null);
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
// Check cache first.
|
|
157
|
+
if (userCache.has(userId)) {
|
|
158
|
+
const cachedUser = userCache.get(userId)!;
|
|
159
|
+
setUser(cachedUser);
|
|
160
|
+
setAuthLoading(false);
|
|
161
|
+
setError(null);
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
try {
|
|
166
|
+
// Attempt to fetch existing user.
|
|
167
|
+
const checkResponse = await fetch(`/api/sync-user?authId=${encodeURIComponent(userId)}`, {
|
|
168
|
+
headers: {
|
|
169
|
+
"Content-Type": "application/json",
|
|
170
|
+
},
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
// If found, cache and set.
|
|
174
|
+
if (checkResponse.ok) {
|
|
175
|
+
const existingUser = await checkResponse.json();
|
|
176
|
+
userCache.set(userId, existingUser);
|
|
177
|
+
setUser(existingUser);
|
|
178
|
+
setAuthLoading(false);
|
|
179
|
+
setError(null);
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// Log and proceed to sync if not found.
|
|
184
|
+
console.warn("checkSession: No user found, proceeding to sync:", { userId, timestamp: new Date().toISOString() });
|
|
185
|
+
|
|
186
|
+
// Sync with fallback username generation if needed.
|
|
187
|
+
const strapiUser = await syncUserToStrapi({
|
|
188
|
+
authId: userId,
|
|
189
|
+
email: clerkUser.primaryEmailAddress?.emailAddress || "",
|
|
190
|
+
username: clerkUser.username || clerkUser.firstName || `user_${userId.slice(0, 8)}`,
|
|
191
|
+
});
|
|
192
|
+
if (!strapiUser) {
|
|
193
|
+
console.warn("checkSession: No Strapi user found, user set to null", { userId, timestamp: new Date().toISOString() });
|
|
194
|
+
setUser(null);
|
|
195
|
+
}
|
|
196
|
+
} catch (err) {
|
|
197
|
+
// Log errors and reset states.
|
|
198
|
+
console.warn("checkSession: Error:", { error: err instanceof Error ? err.message : "Unknown error", timestamp: new Date().toISOString() });
|
|
199
|
+
setUser(null);
|
|
200
|
+
setError(null);
|
|
201
|
+
} finally {
|
|
202
|
+
// Always clear loading state.
|
|
203
|
+
setAuthLoading(false);
|
|
204
|
+
}
|
|
205
|
+
};
|
|
206
|
+
|
|
207
|
+
// Effect to trigger session check whenever Clerk auth state changes.
|
|
208
|
+
useEffect(() => {
|
|
209
|
+
checkSession();
|
|
210
|
+
}, [isSignedIn, userId, clerkUser]);
|
|
211
|
+
|
|
212
|
+
// Provide the context value to children components.
|
|
213
|
+
return (
|
|
214
|
+
<AuthContext.Provider value={{ user, authLoading, error, checkSession, handlePostSignUp }}>
|
|
215
|
+
{children}
|
|
216
|
+
</AuthContext.Provider>
|
|
217
|
+
);
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
// Custom hook to access the auth context safely.
|
|
221
|
+
export function useStrapiAuth() {
|
|
222
|
+
const context = useContext(AuthContext);
|
|
223
|
+
if (!context) throw new Error("useStrapiAuth must be used within an AuthProvider");
|
|
224
|
+
return context;
|
|
225
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { StrapiUser } from "../types";
|
|
2
|
+
|
|
3
|
+
export async function isAdminUser(isSignedIn: boolean | undefined, user: StrapiUser | null): Promise<boolean> {
|
|
4
|
+
if (!isSignedIn || !user?.authId) {
|
|
5
|
+
|
|
6
|
+
return false;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
try {
|
|
10
|
+
|
|
11
|
+
const response = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL || ''}/api/verify-admin`, {
|
|
12
|
+
method: "POST",
|
|
13
|
+
headers: { "Content-Type": "application/json" },
|
|
14
|
+
body: JSON.stringify({ authId: user.authId }),
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
if (!response.ok) {
|
|
18
|
+
const errorData = await response.json();
|
|
19
|
+
console.error("isAdminUser: Verify admin failed:", { error: errorData, status: response.status, timestamp: new Date().toISOString() });
|
|
20
|
+
return false;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const { isAdmin } = await response.json();
|
|
24
|
+
|
|
25
|
+
return isAdmin;
|
|
26
|
+
} catch (error) {
|
|
27
|
+
console.error("isAdminUser error:", { error: error instanceof Error ? error.message : "Unknown error", timestamp: new Date().toISOString() });
|
|
28
|
+
return false;
|
|
29
|
+
}
|
|
30
|
+
}
|