-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* feat: implement flippable card in main own organization section * chore: apply missed code convention * fix: apply code review - remove duplicated color, change flip listener by device type * feat: apply image * feat: change size by screen size * fix: edit breakpoint 375 -> 428
- Loading branch information
1 parent
cd633c4
commit 3792f7e
Showing
11 changed files
with
351 additions
and
0 deletions.
There are no files selected for viewing
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
import React from 'react'; | ||
import useBooleanState from '@src/hooks/useBooleanState'; | ||
import { useDeviceType } from '@src/hooks/useDevice'; | ||
import * as S from './style'; | ||
|
||
type FlippableCardProps = { | ||
frontContent: React.ReactNode; | ||
backContent: React.ReactNode; | ||
}; | ||
|
||
const useFlippableCard = () => { | ||
const [isFlipped, setIsFlipped, setIsUnflipped, toggleFlipped] = useBooleanState(false); | ||
const deviceType = useDeviceType(); | ||
|
||
if (deviceType === 'desktop') { | ||
return { isFlipped, onMouseEnter: setIsFlipped, onMouseLeave: setIsUnflipped }; | ||
} | ||
return { isFlipped, onClick: toggleFlipped }; | ||
}; | ||
|
||
export default function FlippableCard({ frontContent, backContent }: FlippableCardProps) { | ||
const { isFlipped, ...eventListeners } = useFlippableCard(); | ||
|
||
const variants = { | ||
front: { rotateY: 0 }, | ||
back: { rotateY: 180 }, | ||
}; | ||
|
||
return ( | ||
<div {...eventListeners}> | ||
<S.CardWrapper | ||
animate={isFlipped ? 'back' : 'front'} | ||
variants={variants} | ||
transition={{ duration: 0.2 }} | ||
> | ||
<S.FrontSideCardWrapper>{frontContent}</S.FrontSideCardWrapper> | ||
<S.BackSideCardWrapper>{backContent}</S.BackSideCardWrapper> | ||
</S.CardWrapper> | ||
</div> | ||
); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
import styled from '@emotion/styled'; | ||
import { motion } from 'framer-motion'; | ||
|
||
export const CardWrapper = styled(motion.div)` | ||
transition: 0.2s; | ||
display: inline-grid; | ||
transform: perspective(800px) rotateY(0deg); | ||
transform-style: preserve-3d; | ||
`; | ||
|
||
export const SideCardWrapper = styled.div` | ||
grid-area: 1 / 1 / 1 / 1; | ||
backface-visibility: hidden; | ||
`; | ||
|
||
export const FrontSideCardWrapper = styled(SideCardWrapper)``; | ||
|
||
export const BackSideCardWrapper = styled(SideCardWrapper)` | ||
transform: rotateY(180deg); | ||
`; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
50 changes: 50 additions & 0 deletions
50
src/views/MainPage/components/OwnOrganization/Card/index.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
import FlippableCard from '@src/components/common/FlippableCard'; | ||
import { TextWeightType } from '@src/lib/types/universal'; | ||
import * as S from './style'; | ||
|
||
type OwnOrganizationCardProps = { | ||
nameKor: string; | ||
nameEng: string; | ||
description: TextWeightType[]; | ||
frontSideBg: string; | ||
backSideBg: string; | ||
}; | ||
|
||
function Footer({ nameKor, nameEng }: Pick<OwnOrganizationCardProps, 'nameKor' | 'nameEng'>) { | ||
return ( | ||
<S.FooterWrapper> | ||
<S.FooterKorName>{nameKor}</S.FooterKorName> | ||
<S.FooterEngName>{nameEng}</S.FooterEngName> | ||
</S.FooterWrapper> | ||
); | ||
} | ||
|
||
export default function OwnOrganizationCard({ | ||
nameKor, | ||
nameEng, | ||
description, | ||
frontSideBg, | ||
backSideBg, | ||
}: OwnOrganizationCardProps) { | ||
return ( | ||
<FlippableCard | ||
frontContent={ | ||
<S.CardWrapper background={`url(${frontSideBg}) center`}> | ||
<Footer nameKor={nameKor} nameEng={nameEng} /> | ||
</S.CardWrapper> | ||
} | ||
backContent={ | ||
<S.CardWrapper background={backSideBg}> | ||
<S.ContentWrapper> | ||
{description.map((textNode, index) => ( | ||
<S.TextWrapper key={index} weight={textNode.weight}> | ||
{textNode.content} | ||
</S.TextWrapper> | ||
))} | ||
</S.ContentWrapper> | ||
<Footer nameKor={nameKor} nameEng={nameEng} /> | ||
</S.CardWrapper> | ||
} | ||
/> | ||
); | ||
} |
134 changes: 134 additions & 0 deletions
134
src/views/MainPage/components/OwnOrganization/Card/style.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,134 @@ | ||
import styled from '@emotion/styled'; | ||
import { colors } from '@sopt-makers/colors'; | ||
|
||
export const CardWrapper = styled.div<{ background: string }>` | ||
background: ${({ background }) => background}; | ||
background-position: center; | ||
background-size: cover; | ||
background-repeat: no-repeat; | ||
border-radius: 19px; | ||
padding: 39px 0; | ||
height: 380px; | ||
display: flex; | ||
flex-direction: column; | ||
justify-content: flex-end; | ||
@media (max-width: 1440px) { | ||
width: 512px; | ||
height: 432px; | ||
padding-top: 39px; | ||
padding-bottom: 23px; | ||
} | ||
@media (max-width: 768px) { | ||
width: max(416px, min(100vw - 200px, 512px)); | ||
height: calc(max(432px, min(100vw - 200px, 512px)) * 0.84); | ||
} | ||
@media (max-width: 428px) { | ||
width: 293px; | ||
height: 250px; | ||
padding-top: 20px; | ||
padding-bottom: 13px; | ||
} | ||
`; | ||
|
||
export const FooterKorName = styled.div` | ||
width: 144px; | ||
text-align: center; | ||
padding: 16px 0; | ||
color: ${colors.white}; | ||
border: 1px solid rgba(255, 255, 255, 0.5); | ||
background: rgba(117, 97, 79, 0.33); | ||
backdrop-filter: blur(2.949289321899414px); | ||
border-radius: 14px; | ||
font-size: 22px; | ||
font-weight: 600; | ||
line-height: 28px; | ||
letter-spacing: -0.904px; | ||
@media (max-width: 1440px) { | ||
width: 128px; | ||
padding: 12px 0; | ||
font-size: 21px; | ||
line-height: 24.425px; | ||
} | ||
@media (max-width: 428px) { | ||
border-radius: 6px; | ||
width: 74px; | ||
padding: 7px 0; | ||
font-size: 12px; | ||
line-height: 13px; | ||
} | ||
`; | ||
|
||
export const FooterEngName = styled.div` | ||
color: rgba(255, 255, 255, 0.7); | ||
font-family: SUIT; | ||
font-size: 20px; | ||
font-style: normal; | ||
font-weight: 400; | ||
line-height: 28.288px; /* 138.027% */ | ||
letter-spacing: -1.025px; | ||
padding-bottom: 4px; | ||
@media (max-width: 1440px) { | ||
font-size: 19px; | ||
line-height: 13.9px; | ||
} | ||
@media (max-width: 428px) { | ||
font-size: 11px; | ||
line-height: 14.9px; | ||
} | ||
`; | ||
|
||
export const FooterWrapper = styled.div` | ||
display: flex; | ||
justify-content: space-between; | ||
align-items: flex-end; | ||
padding-left: 30px; | ||
padding-right: 42px; | ||
@media (max-width: 1440px) { | ||
padding-left: 25px; | ||
padding-right: 27px; | ||
} | ||
@media (max-width: 428px) { | ||
padding-left: 14px; | ||
padding-right: 15px; | ||
} | ||
`; | ||
|
||
export const ContentWrapper = styled.div` | ||
padding: 0 41px; | ||
flex: 1; | ||
word-break: keep-all; | ||
@media (max-width: 428px) { | ||
padding: 0 20px; | ||
} | ||
`; | ||
|
||
export const TextWrapper = styled.span<{ weight: 'normal' | 'bold' }>` | ||
font-size: 20px; | ||
color: ${colors.white}; | ||
font-weight: ${({ weight }) => weight}; | ||
line-height: 35px; /* 175% */ | ||
letter-spacing: -0.8px; | ||
@media (max-width: 1440px) { | ||
font-size: 18px; | ||
line-height: 31px; /* 172.222% */ | ||
letter-spacing: -0.72px; | ||
} | ||
@media (max-width: 428px) { | ||
font-size: 11px; | ||
line-height: 18.103px; /* 164.575% */ | ||
letter-spacing: -0.44px; | ||
} | ||
`; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
import { OWN_ORGANIZATION_LIST } from '@src/lib/constants/main'; | ||
import Tab from '../Tab'; | ||
import OwnOrganizationCard from './Card'; | ||
import * as S from './style'; | ||
|
||
export default function OwnOrganization() { | ||
return ( | ||
<S.Background> | ||
<Tab | ||
tab={'(3) 자체 운영 기구'} | ||
title={'SOPT를 운영하는 자체 기구'} | ||
description={ | ||
'SOPT에는 솝트를 자체적으로 운영하는 두 가지의 기구가 존재합니다. 한 기수 이상 활동한\n사람들이 모여, 솝트가 보다 유연하고 열정적인 경험으로 채워질 수 있도록 노력하죠.' | ||
} | ||
/> | ||
<S.Wrapper> | ||
{OWN_ORGANIZATION_LIST.map((organization) => ( | ||
<OwnOrganizationCard key={organization.nameEng} {...organization} /> | ||
))} | ||
</S.Wrapper> | ||
</S.Background> | ||
); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
import styled from '@emotion/styled'; | ||
import { colors } from '@sopt-makers/colors'; | ||
|
||
export const Wrapper = styled.div` | ||
display: flex; | ||
gap: 28px; | ||
overflow-x: hidden; | ||
@media (max-width: 1440px) { | ||
gap: 24px; | ||
overflow-x: scroll; | ||
} | ||
@media (max-width: 428px) { | ||
gap: 14px; | ||
} | ||
`; | ||
|
||
export const Background = styled.section` | ||
background-color: ${colors.white}; | ||
padding-top: 146px; | ||
`; |