Emotion with React CRA

CSS-in-JS 로는 SCSS, Styled-Components 를 써봤다. SCSS의 네스팅기능과 Styled-Components의 props를 내려받아서 활용하는 기능이 좋았다. 그러나 Styled-Components의 태그를 대신하는 스타일명은 컴포넌트와 헷갈렸다. 그리고 컴포넌트명이 대문자로 가득찬 코드가 이뻐보이지 않았다.

emotion이 CSS-in-JS에서 1위를 달리고 있고 많은 기업에서 사용중이다. 어떤 특징이 있고 왜 좋은지 알아보고 싶어서 사용해보려고 한다.

다운로드 npmtrends.com

개발자들도 연말연초는 쉬는 것 같다..!

emotion 공식문서가 잘 나와있다. 그런데 코드보기가 영 불편하다.
aside부분이 너무 커서 가로가 자꾸 잘려서 스크롤을 해야 한다. 쩝.

React에서 emotion 사용하는 두가지 방법

  • CSS prop
  • styled

첫번째 방법. CSS props

이 방법을 보고 옳다구나 싶었다. 왜냐하면 div, p 등 태그이름을 유지할 수 있기 때문이다. 하지만 결론적으로 이 방법은 props를 받아서 사용하려면 인라인으로 해야하기 때문에 쓰지 않기로 했다.

/** @jsx jsx */   // <-- JSX Pragma 밑에 설명
import { css, jsx } from '@emotion/react'

const color = 'white'
render(
//방법 1.
  <div             // div 유지 가능!!!
    css={css`
      padding: 32px;
      &:hover {
        color: ${color};
      }
    `}
  >

//방법 2.
  <div
    css={{
      backgroundColor: 'hotpink',
      '&:hover': {
        color: 'lightgreen'
      }
    }}
  >

    Hover to change color.
  </div>
)

JSX Pragma

CSS props 방법을 사용하려면 /** @jsx jsx */ 를 위에 붙여야 한다. 바벨을 직접 구성할 수 없을 때 이렇게 해주면 좋다. 예를 들면 CRA나 codesandbox를 사용할 경우이다. 위 코드는 jsx babel plugin이 React.createElement 대신 jsx을 사용할 수 있도록 설정한다. 음.

Similar to a comment containing linter configuration, this configures the jsx babel plugin to use the jsx function instead of React.createElement. CRA 4 이상이면 /** @jsxImportSource @emotion/react */를 사용하자. -공식문서-

*React.createElement 란?

공식문서의 JSX Pragma를 이해하려면 이 부분을 이해해야 할 것 같다. React에서 바벨적용 전 코드는 어떻게 되어있을까? 바벨은 내가 열심히 짠 리액트 코드를 아래와 같은 코드로 바꾸어준다.

<body>
 <div id="root"></div>
 <script>
	class LikeButton extends React.Component {
  		constructor(props) { super(props);
  }
  render() { return React.createElement("button", null, "Like");  // 참고로 null 부분은 onClick같은 함수자리다.
 </script>
 <script>
 ReactDOM.render(React.createElement(LikeButton),  document.querySelector("#root"));
 </script>
</body>

// 위 코드결과
// <div id='root'><button>Like</button></div>

어떻게 바꿀까? 바벨은 jsx문법을 지원하는 것이다. jsx란 html+xml이다. <div>는 소문자니까 html로 <LikeButton>은 대문자니까 xml로 변환한다.

이실직고 하자면 ‘React.createElement 대신 jsx을 사용할 수 있도록 설정한다’는 공식문서의 설명을 잘 모르겠다… 내 생각으로는

 <div
    css={css`
      padding: 32px;
      &:hover {
        color: ${color};
      }
    `}
  >

위의 코드에서 css 프로퍼티의 값이 소문자인데 이걸 React.createElement로 받아들이면 태그 안에 태그니까 당연히 에러가 난다. 그래서 에러가 뜨지 않도록 어떻게저떻게 하는 것 같다…

두번째 방법. styled Components

styled was heavily inspired by styled-components and glamorous

styled-components의 엄청난 영향을 받아서 그런지 정말 똑같다. 그런데 이렇게 막 갖다베껴도 되는건가. 뭐 저작권을 잘 지켜서 잘 하셨겠지만..

1. props 사용법

import styled from '@emotion/styled';

const Button = styled.button`
  color: ${props => (props.primary ? 'hotpink' : 'turquoise')};
`;

const Container = styled.div(props => ({
  display: 'flex',
  flexDirection: props.column && 'column',
}));

render(
  <Container column>
    <Button>This is a regular button.</Button>
    <Button primary>This is a primary button.</Button>
  </Container>
);

2. 다른 emotion component 사용하기

백틱사용법과 객체사용법 두가지가 있다. 나는 백틱으로 통일해서 사용할 것!

백틱 사용법

import styled from '@emotion/styled';

const Child = styled.div`
  color: red;
`;
const Parent = styled.div`
  ${Child} // 이렇게 !! {
    color: green;
  }
`;

render(
  <div>
    <Parent>
      <Child>Green because I am inside a Parent</Child>
    </Parent>
    <Child>Red because I am not inside a Parent</Child>
  </div>
);

객체사용법

import styled from '@emotion/styled';

const Child = styled.div(
  {
    color: 'red',
  },
  props => ({ fontSize: props.size })
);

const Parent = styled.div({
  [Child]: {
    color: 'green',
  },
});

render(
  <div>
    <Parent>
      <Child>green</Child>
    </Parent>
    <Child>red</Child>
  </div>
);

위 사용법을 쓰려면 바벨설정을 해야 한다. 공식문서를 보면 설정방법에 babelrc 건들기와 CLI 통해서 하기, node api 통해서 하기 세가지 방법이 나와있다. 일단 노드는 안쓰니까 탈락. 두번째 CLI는 어떻게 하는지 모르겠다. 찾아봐도 모르겠어서 패스.

그래서 하나 남은 방법. babelrc를 건드려야 한다. 그런데 나는 CRA로 앱을 만들어서 babel을 건들려면 eject를 해야 한다. 찾아보니까 eject를 하지 않고 바벨설정을 하는 방법이 있었다 ! 바로 craco 패키지.

npm install @craco/craco

설치를 해준다.

package.json에서 다음으로 수정한다.

"scripts": {
    "start": "craco start",
    "build": "craco build",
    "test": "craco test",
    "eject": "craco eject"
}

루트폴더에 craco.config.js를 추가하고 아래 내용을 넣는다.

module.exports = {
  babel: {
    plugins: ['@emotion'],
  },
};

성공!!!!! 공식문서 보고 응용(?)해서 뭔가 성공한게 처음이라 기분이 좋다.

그런데 찾는 도중에 아까 위에서 jsx Pragma를 항상 import해줘야 한다고 썼는데 이것도 바벨설정으로 생략할 수 있다고 한다. 방금 만든 craco.config.js 파일에

module.exports = {
  babel: {
    presets: ['@emotion/babel-preset-css-prop'],
  },
};

를 추가하면 된다.

3. 그외

다른 스타일 사용하기

import styled from '@emotion/styled';

const Button = styled.button`
  color: red;
  fontsize: 15px;
`;
const BlueBtn = styled(Button)`
  color: blue;
`;

render(<BlueBtn>btn</BlueBtn>);

Global styles랑 theme 지정은 styled-component랑 비슷해서 생략.

참고

Emotion 공식문서

제로초님 유튜브 - 바벨 부분


Jiyon Lee
Written by@Jiyon Lee
뜨거운 코드를 가르며

GitHub