Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion apps/profile/src/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ const router = createBrowserRouter([
{
path: '/:userCode',
element: <ProfilePage />,
errorElement: <NotFound />,
},
{
path: '/:userCode/shows',
Expand All @@ -36,6 +35,10 @@ const router = createBrowserRouter([
path: '/',
element: <NotFound />,
},
{
path: '*',
element: <NotFound />,
},
]);

const App = () => {
Expand Down
169 changes: 128 additions & 41 deletions apps/profile/src/components/NotFound.tsx
Original file line number Diff line number Diff line change
@@ -1,78 +1,165 @@
import styled from '@emotion/styled';
import { Helmet } from 'react-helmet-async';
import { useState, useEffect } from 'react';
import { useParams } from 'react-router-dom';
import Layout from '~/components/Layout';
import { EXTERNAL_DOMAIN } from '~/constants/external';

const NotFound = () => {
const { userCode } = useParams<{ userCode: string }>();
const [isDesktop, setIsDesktop] = useState(false);
const [showModal, setShowModal] = useState(true);

useEffect(() => {
const mediaQuery = window.matchMedia('(min-width: 1024px)');
setIsDesktop(mediaQuery.matches);

const handleMediaChange = (e: MediaQueryListEvent) => {
setIsDesktop(e.matches);
};

mediaQuery.addEventListener('change', handleMediaChange);
return () => mediaQuery.removeEventListener('change', handleMediaChange);
}, []);

return (
<>
<Helmet>
<title>페이지를 찾을 수 없습니다 - 불티</title>
</Helmet>

<Container>
<Content>
<ErrorCode>404</ErrorCode>
<Title>페이지를 찾을 수 없습니다</Title>
<Description>
요청하신 페이지가 존재하지 않거나 이동되었습니다.
<br />
URL을 다시 확인해 주세요.
</Description>
<HomeLink href={EXTERNAL_DOMAIN.SHOW_MANAGER}>불티 홈으로 가기</HomeLink>
</Content>
</Container>
<Layout>
<CoverSection isCover={false} isDesktop={isDesktop}>
<CoverImage
src=""
alt="기본 프로필"
/>
<CoverOverlay>
<ProfileInfo>
<Nickname>닉네임</Nickname>
<UserName>@{userCode || 'username'}</UserName>
</ProfileInfo>
</CoverOverlay>
</CoverSection>

{showModal && (
<ModalOverlay onClick={() => setShowModal(false)}>
<ModalContent onClick={(e) => e.stopPropagation()}>
<ModalText>존재하지 않는 프로필입니다.</ModalText>
<ModalButton href={EXTERNAL_DOMAIN.SHOW_MANAGER}>불티 홈 바로 가기</ModalButton>
</ModalContent>
</ModalOverlay>
)}
</Layout>
</>
);
};

export default NotFound;

const Container = styled.div`
const CoverSection = styled.div<{ isCover: boolean; isDesktop: boolean }>`
position: relative;
width: 100%;
height: 0;
padding-top: 100%;
overflow: hidden;
border-radius: ${({ isCover, isDesktop }) => (isCover && isDesktop ? '20px 20px 0 0' : '0')};
:after {
content: '';
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: linear-gradient(to bottom, rgba(18, 19, 24, 0.2) 0%, rgba(18, 19, 24, 1) 100%);
}
`;

const CoverImage = styled.img`
width: 100%;
height: 100%;
object-fit: cover;
position: absolute;
top: 0;
left: 0;
`;

const CoverOverlay = styled.div`
position: absolute;
bottom: 0;
left: 0;
right: 0;
z-index: 1;
padding: 0 20px;
display: flex;
align-items: center;
justify-content: center;
min-height: 100vh;
background-color: #f8f9fa;
padding: 20px;
flex-direction: column;
`;

const Content = styled.div`
text-align: center;
max-width: 480px;
const ProfileInfo = styled.div`
display: flex;
flex-direction: column;
`;

const ErrorCode = styled.div`
font-size: 72px;
const Nickname = styled.h1`
${({ theme }) => theme.typo.point.p3};
font-weight: bold;
color: #667eea;
margin-bottom: 16px;
color: ${({ theme }) => theme.palette.grey.g10};
`;

const Title = styled.h1`
font-size: 28px;
font-weight: bold;
color: #212529;
margin-bottom: 12px;
const UserName = styled.h1`
${({ theme }) => theme.typo.b1};
color: ${({ theme }) => theme.palette.grey.g50};
margin: 0;
`;

const ModalOverlay = styled.div`
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.7);
display: flex;
align-items: center;
justify-content: center;
z-index: 1000;
padding: 20px;
`;

const Description = styled.p`
font-size: 16px;
color: #495057;
line-height: 1.6;
margin-bottom: 32px;
const ModalContent = styled.div`
background-color: white;
border-radius: 20px;
padding: 40px 24px 24px;
max-width: 382px;
width: 100%;
display: flex;
flex-direction: column;
gap: 16px;
`;

const ModalText = styled.p`
${({ theme }) => theme.typo.sh1};
color: #121318;
text-align: center;
margin: 0;
`;

const HomeLink = styled.a`
display: inline-block;
padding: 12px 24px;
background-color: #667eea;
color: #ffffff;
const ModalButton = styled.a`
width: 100%;
height: 48px;
display: flex;
align-items: center;
justify-content: center;
background-color: #ff5623;
color: white;
text-decoration: none;
border-radius: 8px;
${({ theme }) => theme.typo.sh1};
font-weight: 600;
cursor: pointer;
transition: background-color 0.2s;

&:hover {
background-color: #5568d3;
background-color: #e64d1f;
}
`;
Loading