From b945f8e9bd3654c6222a82a73113c7ccde819e61 Mon Sep 17 00:00:00 2001
From: Jiho Park <59248080+jihorobert@users.noreply.github.com>
Date: Thu, 2 Jan 2025 13:51:58 +0900
Subject: [PATCH] feat(fe): confirm modal when leaving settings page (#2261)
* feat(fe): confirm modal when leaving settings page
* feat(fe): add dark and bright mode
* chore(fe): add comments for basemodal
* fix(fe): decide darkmode at ConfirmModal
* chore(fe): add explanation for BaseModal component
* chore(fe): add explanation for Basemodal component_2
* chore(fe): change comment for BaseModal
* fix(fe): change description props type to string from ReactNode
* chore(fe): import types
* chore(fe): change comment for basemodal
* chore(fe): change comment for confirmmodal
* fix(fe): move confirmNavigation to utils file
* chore(fe): change darkmode props name for BaseModal
* chore(fe): change frament tag to p tag
* chore(fe): remove duplicate backdrop blur by setting it as a default class
* fix(fe): change dependency for ConfirmNavigation
* fix(fe): change dependency for ConfirmNavigation
---
.../settings/_components/ConfirmModal.tsx | 61 ++++++++++++++++++
.../ConfirmNavigation.tsx => _libs/utils.ts} | 18 ++++--
.../app/(client)/(main)/settings/page.tsx | 16 ++++-
apps/frontend/components/BaseModal.tsx | 64 +++++++++++++++++++
.../components/shadcn/alert-dialog.tsx | 9 ++-
5 files changed, 155 insertions(+), 13 deletions(-)
create mode 100644 apps/frontend/app/(client)/(main)/settings/_components/ConfirmModal.tsx
rename apps/frontend/app/(client)/(main)/settings/{_components/ConfirmNavigation.tsx => _libs/utils.ts} (75%)
create mode 100644 apps/frontend/components/BaseModal.tsx
diff --git a/apps/frontend/app/(client)/(main)/settings/_components/ConfirmModal.tsx b/apps/frontend/app/(client)/(main)/settings/_components/ConfirmModal.tsx
new file mode 100644
index 0000000000..9a33f03ccf
--- /dev/null
+++ b/apps/frontend/app/(client)/(main)/settings/_components/ConfirmModal.tsx
@@ -0,0 +1,61 @@
+import BaseModal from '@/components/BaseModal'
+import {
+ AlertDialogAction,
+ AlertDialogCancel,
+ AlertDialogFooter
+} from '@/components/shadcn/alert-dialog'
+
+interface ModalProps {
+ open: boolean
+ handleOpen: () => void
+ handleClose: () => void
+ confirmAction: () => void
+ title?: string
+ description?: string
+}
+
+/**
+ *
+ * ConfirmModal component renders a modal dialog with confirm and cancel actions.
+ *
+ * @param open - Determines if the modal is open.
+ * @param handleClose - Function to close the modal.
+ * @param confirmAction - Function to execute when the user confirms.
+ * @param title - Title of the modal.
+ * @param description - Description of the modal.
+ *
+ * @remarks
+ * * AlertDialogFooter section (Button section) is separated using ConfirmModal component for reusability.
+ */
+export default function ConfirmModal({
+ open,
+ handleClose,
+ confirmAction,
+ title = '',
+ description = ''
+}: ModalProps) {
+ return (
+
+
+
+ Leave
+
+
+ Stay
+
+
+
+ )
+}
diff --git a/apps/frontend/app/(client)/(main)/settings/_components/ConfirmNavigation.tsx b/apps/frontend/app/(client)/(main)/settings/_libs/utils.ts
similarity index 75%
rename from apps/frontend/app/(client)/(main)/settings/_components/ConfirmNavigation.tsx
rename to apps/frontend/app/(client)/(main)/settings/_libs/utils.ts
index 5e7afa67c6..60bc764a1d 100644
--- a/apps/frontend/app/(client)/(main)/settings/_components/ConfirmNavigation.tsx
+++ b/apps/frontend/app/(client)/(main)/settings/_libs/utils.ts
@@ -2,7 +2,7 @@ import type { Route } from 'next'
import type { NavigateOptions } from 'next/dist/shared/lib/app-router-context.shared-runtime'
import { useRouter } from 'next/navigation'
import type { MutableRefObject } from 'react'
-import { useEffect } from 'react'
+import { useEffect, useState } from 'react'
import { toast } from 'sonner'
// const beforeUnloadHandler = (event: BeforeUnloadEvent) => {
@@ -22,6 +22,9 @@ export const useConfirmNavigation = (
updateNow: boolean
) => {
const router = useRouter()
+ const [isConfirmModalOpen, setIsConfirmModalOpen] = useState(false)
+ const [confirmAction, setConfirmAction] = useState<() => void>(() => () => {})
+
useEffect(() => {
const originalPush = router.push
const newPush = (
@@ -37,12 +40,11 @@ export const useConfirmNavigation = (
return
}
if (!bypassConfirmation.current) {
- const isConfirmed = window.confirm(
- 'Are you sure you want to leave?\nYour changes have not been saved.\nIf you leave this page, all changes will be lost.\nDo you still want to proceed?'
- )
- if (isConfirmed) {
+ setIsConfirmModalOpen(true)
+ setConfirmAction(() => () => {
+ setIsConfirmModalOpen(false)
originalPush(href as Route, options)
- }
+ })
return
}
originalPush(href as Route, options)
@@ -51,5 +53,7 @@ export const useConfirmNavigation = (
return () => {
router.push = originalPush
}
- }, [router, bypassConfirmation.current])
+ }, [router, bypassConfirmation])
+
+ return { isConfirmModalOpen, setIsConfirmModalOpen, confirmAction }
}
diff --git a/apps/frontend/app/(client)/(main)/settings/page.tsx b/apps/frontend/app/(client)/(main)/settings/page.tsx
index b32b8352f8..7488911b59 100644
--- a/apps/frontend/app/(client)/(main)/settings/page.tsx
+++ b/apps/frontend/app/(client)/(main)/settings/page.tsx
@@ -9,7 +9,7 @@ import { useState } from 'react'
import { useForm } from 'react-hook-form'
import { toast } from 'sonner'
import { z } from 'zod'
-import { useConfirmNavigation } from './_components/ConfirmNavigation'
+import ConfirmModal from './_components/ConfirmModal'
import CurrentPwSection from './_components/CurrentPwSection'
import IdSection from './_components/IdSection'
import LogoSection from './_components/LogoSection'
@@ -20,6 +20,7 @@ import ReEnterNewPwSection from './_components/ReEnterNewPwSection'
import SaveButton from './_components/SaveButton'
import StudentIdSection from './_components/StudentIdSection'
import TopicSection from './_components/TopicSection'
+import { useConfirmNavigation } from './_libs/utils'
interface getProfile {
username: string // ID
@@ -91,8 +92,6 @@ export default function Page() {
fetchDefaultProfile()
}, [])
- useConfirmNavigation(bypassConfirmation, !!updateNow)
-
const {
register,
handleSubmit,
@@ -112,6 +111,8 @@ export default function Page() {
}
})
+ const { isConfirmModalOpen, setIsConfirmModalOpen, confirmAction } =
+ useConfirmNavigation(bypassConfirmation, !!updateNow)
const [isCheckButtonClicked, setIsCheckButtonClicked] =
useState(false)
const [isPasswordCorrect, setIsPasswordCorrect] = useState(false)
@@ -324,6 +325,15 @@ export default function Page() {
onSubmitClick={onSubmitClick}
/>
+
+ setIsConfirmModalOpen(true)}
+ handleClose={() => setIsConfirmModalOpen(false)}
+ confirmAction={confirmAction}
+ />
)
}
diff --git a/apps/frontend/components/BaseModal.tsx b/apps/frontend/components/BaseModal.tsx
new file mode 100644
index 0000000000..6a655ebf16
--- /dev/null
+++ b/apps/frontend/components/BaseModal.tsx
@@ -0,0 +1,64 @@
+import { Loader2 } from 'lucide-react'
+import React, { type ReactNode } from 'react'
+import {
+ AlertDialog,
+ AlertDialogContent,
+ AlertDialogDescription,
+ AlertDialogHeader,
+ AlertDialogOverlay,
+ AlertDialogTitle
+} from './shadcn/alert-dialog'
+
+interface BaseModalProps {
+ open: boolean
+ handleClose: () => void
+ children?: ReactNode
+ loading?: boolean
+ loadingMessage?: string
+ title?: string
+ description?: string
+ darkMode?: boolean
+}
+
+/**
+ *
+ * @remarks
+ * * Use BaseModal Component by creating a new component(which includes 'AlertDialogFooter') that extends BaseModal.
+ * * AlertDialogFooter section (Button section) is separated using ConfirmModal component for reusability.
+ */
+export default function BaseModal({
+ open,
+ handleClose,
+ children,
+ loading = false,
+ loadingMessage = '',
+ title = '',
+ description = '',
+ darkMode = false
+}: BaseModalProps) {
+ const formattedDescription =
+ description.split('\n').map((line, index) => {line}
) ??
+ ''
+
+ return (
+
+
+
+
+ {title}
+
+ {loading ? (
+
+
+ {loadingMessage}
+
+ ) : (
+ formattedDescription
+ )}
+
+
+ {children}
+
+
+ )
+}
diff --git a/apps/frontend/components/shadcn/alert-dialog.tsx b/apps/frontend/components/shadcn/alert-dialog.tsx
index baaf9e0e13..c3cecdc510 100644
--- a/apps/frontend/components/shadcn/alert-dialog.tsx
+++ b/apps/frontend/components/shadcn/alert-dialog.tsx
@@ -13,11 +13,14 @@ const AlertDialogPortal = AlertDialogPrimitive.Portal
const AlertDialogOverlay = React.forwardRef<
React.ElementRef,
- React.ComponentPropsWithoutRef
->(({ className, ...props }, ref) => (
+ React.ComponentPropsWithoutRef & {
+ darkMode?: boolean
+ }
+>(({ className, darkMode = false, ...props }, ref) => (