feat: implement news layout components and context for improved structure and functionality
This commit is contained in:
parent
d9c36c5ff0
commit
77c9843267
36
app/contexts/news.tsx
Normal file
36
app/contexts/news.tsx
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import React, {
|
||||||
|
createContext,
|
||||||
|
useState,
|
||||||
|
useContext,
|
||||||
|
type PropsWithChildren,
|
||||||
|
} from 'react'
|
||||||
|
|
||||||
|
type NewsContextProperties = {
|
||||||
|
isLoginOpen: boolean
|
||||||
|
setIsLoginOpen: React.Dispatch<React.SetStateAction<boolean>>
|
||||||
|
isRegisterOpen: boolean
|
||||||
|
setIsRegisterOpen: React.Dispatch<React.SetStateAction<boolean>>
|
||||||
|
}
|
||||||
|
|
||||||
|
const NewsContext = createContext<NewsContextProperties | undefined>(undefined)
|
||||||
|
|
||||||
|
export const NewsProvider = ({ children }: PropsWithChildren) => {
|
||||||
|
const [isLoginOpen, setIsLoginOpen] = useState(false)
|
||||||
|
const [isRegisterOpen, setIsRegisterOpen] = useState(false)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<NewsContext.Provider
|
||||||
|
value={{ isLoginOpen, setIsLoginOpen, isRegisterOpen, setIsRegisterOpen }}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</NewsContext.Provider>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useNewsContext = (): NewsContextProperties => {
|
||||||
|
const context = useContext(NewsContext)
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('useNewsContext must be used within a NewsProvider')
|
||||||
|
}
|
||||||
|
return context
|
||||||
|
}
|
||||||
54
app/layouts/news/default.tsx
Normal file
54
app/layouts/news/default.tsx
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
import { type PropsWithChildren } from 'react'
|
||||||
|
|
||||||
|
import { PopupModal } from '~/components/popup/modal'
|
||||||
|
import Banner from '~/components/ui/banner'
|
||||||
|
import { FormLogin } from '~/components/ui/form-login'
|
||||||
|
import { FormRegister } from '~/components/ui/form-register'
|
||||||
|
import { useNewsContext } from '~/contexts/news'
|
||||||
|
|
||||||
|
import { FooterLinks } from './footer-links'
|
||||||
|
import { FooterNewsletter } from './footer-newsletter'
|
||||||
|
import { HeaderMenu } from './header-menu'
|
||||||
|
import { HeaderTop } from './header-top'
|
||||||
|
|
||||||
|
export const NewsDefaultLayout = (properties: PropsWithChildren) => {
|
||||||
|
const { children } = properties
|
||||||
|
const { isLoginOpen, setIsLoginOpen, isRegisterOpen, setIsRegisterOpen } =
|
||||||
|
useNewsContext()
|
||||||
|
return (
|
||||||
|
<main className="relative min-h-dvh bg-[#ECECEC]">
|
||||||
|
<header>
|
||||||
|
<HeaderTop />
|
||||||
|
<HeaderMenu />
|
||||||
|
</header>
|
||||||
|
<div className="grid sm:mx-[50px] sm:my-[25px] sm:gap-y-[25px]">
|
||||||
|
<Banner />
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<footer className="grid w-full grid-cols-1 gap-6 bg-[#2E2F7C] px-5 py-20 text-white sm:grid-cols-5 sm:gap-16 sm:px-16">
|
||||||
|
<FooterNewsletter />
|
||||||
|
<FooterLinks />
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
<PopupModal
|
||||||
|
isOpen={isLoginOpen}
|
||||||
|
onClose={() => setIsLoginOpen(false)}
|
||||||
|
description="Selamat Datang, silakan daftarkan akun Anda untuk melanjutkan!"
|
||||||
|
>
|
||||||
|
<FormLogin
|
||||||
|
setIsRegisterOpen={setIsRegisterOpen}
|
||||||
|
setIsLoginOpen={setIsLoginOpen}
|
||||||
|
/>
|
||||||
|
</PopupModal>
|
||||||
|
|
||||||
|
<PopupModal
|
||||||
|
isOpen={isRegisterOpen}
|
||||||
|
onClose={() => setIsRegisterOpen(false)}
|
||||||
|
description="Selamat Datang, silakan daftarkan akun Anda untuk melanjutkan!"
|
||||||
|
>
|
||||||
|
<FormRegister />
|
||||||
|
</PopupModal>
|
||||||
|
</main>
|
||||||
|
)
|
||||||
|
}
|
||||||
@ -3,12 +3,14 @@ import { Link } from 'react-router'
|
|||||||
|
|
||||||
import { CloseIcon } from '~/components/icons/close'
|
import { CloseIcon } from '~/components/icons/close'
|
||||||
import { MenuIcon } from '~/components/icons/menu'
|
import { MenuIcon } from '~/components/icons/menu'
|
||||||
import { HeaderSearch } from '~/layouts/header-search'
|
import { useNewsContext } from '~/contexts/news'
|
||||||
|
import { HeaderSearch } from '~/layouts/news/header-search'
|
||||||
|
|
||||||
import { MENU } from './menu'
|
import { MENU } from './menu'
|
||||||
|
|
||||||
export default function HeaderMenuMobile() {
|
export default function HeaderMenuMobile() {
|
||||||
const [isMenuOpen, setIsMenuOpen] = useState(false)
|
const [isMenuOpen, setIsMenuOpen] = useState(false)
|
||||||
|
const { setIsLoginOpen } = useNewsContext()
|
||||||
|
|
||||||
const handleToggleMenu = (): void => {
|
const handleToggleMenu = (): void => {
|
||||||
setIsMenuOpen(!isMenuOpen)
|
setIsMenuOpen(!isMenuOpen)
|
||||||
@ -53,7 +55,13 @@ export default function HeaderMenuMobile() {
|
|||||||
</li>
|
</li>
|
||||||
))}
|
))}
|
||||||
|
|
||||||
<button className="w-full bg-white px-[35px] py-3 text-center text-[#2E2F7C] sm:hidden">
|
<button
|
||||||
|
className="w-full bg-white px-[35px] py-3 text-center text-[#2E2F7C] sm:hidden"
|
||||||
|
onClick={() => {
|
||||||
|
setIsLoginOpen(true)
|
||||||
|
setIsMenuOpen(false)
|
||||||
|
}}
|
||||||
|
>
|
||||||
Akun
|
Akun
|
||||||
</button>
|
</button>
|
||||||
</ul>
|
</ul>
|
||||||
@ -1,6 +1,6 @@
|
|||||||
import { Link } from 'react-router'
|
import { Link } from 'react-router'
|
||||||
|
|
||||||
import HeaderMenuMobile from '~/layouts/header-menu-mobile'
|
import HeaderMenuMobile from '~/layouts/news/header-menu-mobile'
|
||||||
|
|
||||||
import { HeaderSearch } from './header-search'
|
import { HeaderSearch } from './header-search'
|
||||||
import { MENU } from './menu'
|
import { MENU } from './menu'
|
||||||
@ -1,16 +1,11 @@
|
|||||||
import { useState } from 'react'
|
|
||||||
import { Link } from 'react-router'
|
import { Link } from 'react-router'
|
||||||
|
|
||||||
import { PopupModal } from '~/components/popup/modal'
|
|
||||||
import { Button } from '~/components/ui/button'
|
import { Button } from '~/components/ui/button'
|
||||||
import { FormLogin } from '~/components/ui/form-login'
|
import { useNewsContext } from '~/contexts/news'
|
||||||
import { FormRegister } from '~/components/ui/form-register'
|
|
||||||
import { APP } from '~/data/meta'
|
import { APP } from '~/data/meta'
|
||||||
|
|
||||||
export const HeaderTop = () => {
|
export const HeaderTop = () => {
|
||||||
const [isLoginOpen, setIsLoginOpen] = useState(false)
|
const { setIsLoginOpen } = useNewsContext()
|
||||||
const [isRegisterOpen, setIsRegisterOpen] = useState(false)
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="flex h-[60px] items-center justify-between bg-white px-5 align-middle sm:h-[100px] sm:gap-[15px] sm:px-[50px] sm:py-[20px]">
|
<div className="flex h-[60px] items-center justify-between bg-white px-5 align-middle sm:h-[100px] sm:gap-[15px] sm:px-[50px] sm:py-[20px]">
|
||||||
@ -47,25 +42,6 @@ export const HeaderTop = () => {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<PopupModal
|
|
||||||
isOpen={isLoginOpen}
|
|
||||||
onClose={() => setIsLoginOpen(false)}
|
|
||||||
description="Selamat Datang, silakan daftarkan akun Anda untuk melanjutkan!"
|
|
||||||
>
|
|
||||||
<FormLogin
|
|
||||||
setIsRegisterOpen={setIsRegisterOpen}
|
|
||||||
setIsLoginOpen={setIsLoginOpen}
|
|
||||||
/>
|
|
||||||
</PopupModal>
|
|
||||||
|
|
||||||
<PopupModal
|
|
||||||
isOpen={isRegisterOpen}
|
|
||||||
onClose={() => setIsRegisterOpen(false)}
|
|
||||||
description="Selamat Datang, silakan daftarkan akun Anda untuk melanjutkan!"
|
|
||||||
>
|
|
||||||
<FormRegister />
|
|
||||||
</PopupModal>
|
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -1,28 +1,15 @@
|
|||||||
import { Outlet } from 'react-router'
|
import { Outlet } from 'react-router'
|
||||||
|
|
||||||
import Banner from '~/components/ui/banner'
|
import { NewsProvider } from '~/contexts/news'
|
||||||
import { FooterLinks } from '~/layouts/footer-links'
|
import { NewsDefaultLayout } from '~/layouts/news/default'
|
||||||
import { FooterNewsletter } from '~/layouts/footer-newsletter'
|
|
||||||
import { HeaderMenu } from '~/layouts/header-menu'
|
|
||||||
import { HeaderTop } from '~/layouts/header-top'
|
|
||||||
|
|
||||||
const NewsLayout = () => {
|
const NewsLayout = () => {
|
||||||
return (
|
return (
|
||||||
<main className="relative min-h-dvh bg-[#ECECEC]">
|
<NewsProvider>
|
||||||
<header>
|
<NewsDefaultLayout>
|
||||||
<HeaderTop />
|
|
||||||
<HeaderMenu />
|
|
||||||
</header>
|
|
||||||
<div className="grid sm:mx-[50px] sm:my-[25px] sm:gap-y-[25px]">
|
|
||||||
<Banner />
|
|
||||||
<Outlet />
|
<Outlet />
|
||||||
</div>
|
</NewsDefaultLayout>
|
||||||
|
</NewsProvider>
|
||||||
<footer className="grid w-full grid-cols-1 gap-6 bg-[#2E2F7C] px-5 py-20 text-white sm:grid-cols-5 sm:gap-16 sm:px-16">
|
|
||||||
<FooterNewsletter />
|
|
||||||
<FooterLinks />
|
|
||||||
</footer>
|
|
||||||
</main>
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -82,7 +82,7 @@ export default tseslint.config(
|
|||||||
'~/factories/*/*',
|
'~/factories/*/*',
|
||||||
'~/pages/*/*',
|
'~/pages/*/*',
|
||||||
'~/hooks/*/*',
|
'~/hooks/*/*',
|
||||||
'~/layouts/*/*/*',
|
'~/layouts/*/*/*/*',
|
||||||
'~/schemas/*/*/*',
|
'~/schemas/*/*/*',
|
||||||
'~/types/*/*',
|
'~/types/*/*',
|
||||||
'~/utils/*/**',
|
'~/utils/*/**',
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user