January 22, 2022
현재 만들어 놓은 FlatButton은 이미 기본 스타일이 있다. 새로운 버튼을 만들기 위해서 이 컴포넌트를 그대로 사용하려 했더니 가지고 있던 padding, font-size 등을 모두 바꿔야 했다. 디자인 스펙 자체가 고정되어 있지 않다보니까 재사용할 수 있는 엘리먼트 자체를 만드는게 무의미 했다.
const FlatButtonContainer = styled.button`
position: relative;
border-radius: 5px;
padding: 6px;
font-size: 14px;
cursor: pointer;
transition: background 0.2s;
&:hover {
background: ${colors.darker4()};
}
`
버튼 마다 기본적으로 width랑 height, border-radius 값이 모두 다를 수 있다. 그럼에도 불구하고 아래와 같은 형태는 공통된 특징들을 가지고 있는데 모두 아이콘과 이름이 중앙 정렬 되어 있다는 점이다. 또한 버튼을 hover 했을 때, bgColor가 변한다거나 fontColor, iconColor가 변한 다는 점도 특징 중에 하나다.
재 사용 할 수 있는 컴포넌트를 만들어야 하는 또 하나의 이유는 동일한 디자인이라면, 중복 코드를 작성하지 않고 빠르게 컴포넌트를 만들 수 있기 때문이다. 그럼 매번 똑같은 코드를 작성하지 않으면서 새로운 버튼을 만들 때도 쉽게 가져다가 쓸 수 있는 형태라면 어때야 할까?
return(){
<TypedIconButton
icon={<RocketIcon isActive={isActive} />}
name='창업가'
/>
}
아이콘과 이름이 중앙 정렬 된 상태의 버튼이라면 이렇게 만드는게 이후에 가져다 쓰기 편할 것이다. 근데 아까 얘기했던 것 처럼 TypedIconButton에 width랑 height, padding 등의 값들이 다르다면 어떻게 해줘야 할까? 여러 방법들이 있을거 같은데 가장 쓰기 편한 형태를 찾아보자.
현재 프로젝트에서 emotion 을 사용하고 있기 때문에 css 와 cx 를 사용해서 스타일을 만들어줬다.
버튼에 공통적으로 필요하고, 변하지 않는 값은 직접 props로 전달 하고 변할 수 있는 스타일들 (hover, active 상태에 따른 스타일)은 각 버튼마다 다 다를 수 있으니 styles 객체를 넘겨주는게 좋을거 같다고 판단했다.
변하지 않는 값(웬만하면)
변할 수 있는 값
// interface 는 생략
import React from 'react'
import styled from '@emotion/styled'
import { cx } from '@emotion/css'
const IconButton = ({
width,
height,
icon,
name,
rootClassName = '',
iconClassName = '',
nameClassName = '',
onClick,
}: IconButtonProps) => {
return (
<IconButtonRoot
width={width}
height={height}
className={rootClassName}
onClick={onClick}
>
<div className={cx('typed-icon', `${iconClassName}`)}>{icon}</div>
<Name className={cx('button-name', `${nameClassName}`)}>{name}</Name>
</IconButtonRoot>
)
}
const IconButtonRoot = styled.div<IconButtonRootProps>`
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
width: ${props => props.width}px;
height: ${props => props.height}px;
cursor: pointer;
`
const Name = styled.div``
Element 컴포넌트는 위와 같이 만들고 이를 base로 사용하여 새로운 버튼을 만들었다.
import { css } from '@emotion/css'
const DocButton = () => {
return (
<IconButton
width={70}
height={90}
icon={<TypedIcon icon="docs_box" />}
name="DOCS"
rootClassName={rootClassName}
iconClassName={iconClassName}
nameClassName={nameClassName}
/>
)
}
const rootClassName = css`
background-color: white;
border: 1px solid #e5e5e5;
box-shadow: 0px 2px 5px rgba(0, 0, 0, 0.05);
border-radius: 5px;
&:hover {
background: linear-gradient(0deg, rgba(0, 0, 0, 0.04), rgba(0, 0, 0, 0.04)),
#ffffff;
}
`
const iconClassName = css`
font-size: 35px;
`
const nameClassName = css`
color: #555555;
font-weight: 500;
font-size: 12px;
line-height: 20px;
`
export default DocButton
DocButton은 IconButton이 가지고 있는 기본 구조는 가져가고
스타일이 추가로 필요하기 때문에 rootClassName 이라는 class를 만들어줬다.
const NavButton = () => {
const [isActive, setIsActive] = useState(false)
const handleButtonClick = () => {
setIsActive(true)
setTimeout(() => setIsActive(false), 1000)
}
const iconClassName = css`
font-size: 24px;
margin-bottom: 3px;
* {
fill: ${isActive ? '#098ED0' : '#333333'};
}
`
const rootClassName = css`
background: #f7f7f7;
line-height: 14px;
text-align: center;
letter-spacing: -0.02em;
color: ${isActive ? '#098ED0' : '#333333'};
&:hover {
background: rgba(0, 0, 0, 0.04);
}
`
return (
<IconButton
width={60}
height={54}
icon={<TypedIcon icon="inbox_small" />}
name="Inbox"
rootClassName={rootClassName}
iconClassName={iconClassName}
nameClassName={nameClassName}
onClick={handleButtonClick}
/>
)
}
NavButton
은 단순 hover 상태 뿐 만 아니라, isActive
state에 따라서 스타일이 변경될 수 도 있다.
만약 IconButton 이라는 base 컴포넌트가 없었더라면, Root, Icon, Name 엘리먼트를 모두 만들어줘야 했을 것이다.
return (
<Root>
<TypedIcon iconName='어쩌고' color>
<Name>{버튼 이름}</Name>
</Root>
)
@emotion/css
을 사용하고 있지 않다면 설치 후 css
, cx
API를 사용해야 한다는 점이다. 이미 현재 project에서는 @emotion/react
를 사용하고 있으니 이 모듈을 사용해서 비슷하게 만들 수 있는 방법을 찾아야겠다.IconButton은 icon과 이름으로 이루어진 수직 정렬 버튼이고 이를 재사용해서 여러 다양한 형태의 버튼들을 만들고 싶었다. 버튼들은 각각의 style이 있을 수 있으니 다른 className을 사용하게 해서 버튼 만의 style을 추가시켜줬다. Icon 같은 경우에는 미리 원하는 상태의 Icon을 전달해서 이를 보여줄 수 도 있지만, 변경 될 수 있는 값들은 Root 컴포넌트 이던, Icon, Name이건 일관성있게 className으로 변경하고 싶어서 이 방식을 사용했다.