Frontend

Dlaczego Styled Components pasują do Reacta? Na przykładzie dwóch aplikacji

Trafiając na ten artykuł prawdopodobnie wiesz czym jest React. Jeżeli obiło ci się o uszy CSS-in-JS, to pewnie słyszałeś o Styled Components. Jeżeli tak to po co pisać kolejny artykuł o tej bibliotece? Inni piszą o samym podejściu css-in-js, pokazują te same przykłady z podstawowego rozdziału dokumentacji lub takie, które nie łączyły się w całość.

Brakowało mi w nich głębszego spojrzenia czy analizy, która pomogłaby w podjęciu decyzji czy warto inwestować nasz cenny czas w aplikowaniu tej biblioteki do swojego projektu. Czy Styled Components (w dalszej części SC jako styled components jak oraz SC jako styled component) i praca z tą biblioteką ma jakieś pułapki, które mogłyby okazać się katastrofą? Dlatego też chciałbym się podzielić kilkoma spostrzeżeniami z codziennej pracy z użytkowania SC.

Krzysztof Kuziel. Software Engineer, założyciel firmy Kodern zajmującej się aplikacjami webowymi. Komercyjnie i prywatnie od wielu lat związany z IT. Wśród większych projektów, w których uczestniczył można wymienić m.in. platformę usług streamingu video oraz obieg dokumentów administracji publicznej. Jego motto: “Tyle szans ile odwagi”.


Na potrzeby tego artykułu stworzyłem mały projekt. Dwie małe aplikacje utworzone z pomocą create-react-app, by dać kompletny obraz dwóch rozwiązań, jak i typowych przyzwyczajeń:

Ideą było pokazanie kilku tematów, które zawsze występują, jak do nich można podejść oraz zestawieniem ich w spójną całość. Przyznacie pewnie, że to dosyć proste projekty, lecz zawierają kompletną konfigurację związaną SC, która dotyczy np. Babel i Jest. Nad konfiguracją nieraz trzeba poświęcić kilka godzin, szczególnie gdy są jakieś problemy z paczkami. Do boju!

1. Komponenty

Po co w ogóle zawracać sobie głowę SC? Skoro React ma jsx, czyli już łączy html z js no to może dałoby się jeszcze coś więcej z tego wykrzesać. Nie chciałbym się też rozwodzić nad samym faktem czy sam jsx jest the best. Dotychczas staraliśmy się rozdzielać warstwę wizualną od logiki komponentu. No a co jeśli potrzebujemy coś bardziej skomplikowanego i dynamicznego? Jak zapanować nad nazwami klas (globalny konflikt nazw), literówkami, zarządzaniem równolegle css i js? Tu z pomocą przychodzi SC.

// Card.js
<div className={sideFrontClassNames}>
   <div className={`${ns}__side__front__heading-wrapper`}>
      <Heading />
   </div>
   <div className={`${ns}__side__front__buttons-bottom`}>
      <LinkButton>Documentation</LinkButton>
   </div>
</div>
// Card.css
&__side {
   height: 100%;
   width: 100%;

  &__front {
    &__heading-wrapper {
      margin: -.1rem;
      width: calc(100% + .2rem);
      height: 13.2rem;
    }
  }
}

Da się zrobić to inaczej? Da:

// Card.js
<SideFront>
   <HeadingWrapper>
      <Heading />
   </HeadingWrapper>
   <ButtonsBottom>
      <LinkButton>Documentation</LinkButton>
   </ButtonsBottom>
</SideFront>

Będę pomijał fragmenty kodu, by skupić uwagę na najważniejszych elementach. W powyższym przykładzie deklarujemy komponenty, co przy sprawnym eslincie od razu wykrywa nam potencjalne pomyłki i nie dopuszcza do dalszej pracy. Współczesne IDE mają możliwość autosugestii nazwy klas na podstawie zewnętrznych stylów, jednakże to nie to samo co automatyczny feedback, ponieważ nie zadeklarowaliśmy zmiennej. Komponenty deklarujemy jako dowolne tagi html, możemy je też importować.

// Card.js
const Side = css`
   height: 100%;
   width: 100%;
`;

const SideFront = styled.div`
 ${Side};
`;

const HeadingWrapper = styled.div`
 margin: -.1rem;
 width: calc(100% + .2rem);
 height: 13.2rem;
`;

Za pomocą ‘css’ możemy deklarować większe bloki css i importować je w innych komponentach. Na takim prostym przykładzie widać, że rozwiązanie to sprawdza się idealnie. Ja w tym widzę jedno „ale” — bardzo łatwo można wpaść w pułapkę tworzenia wrapperów do komponentów, a do nich kolejnych wrapperów itd.

<div className="images-box__wrapper">
   <div className="images-box">
      <div className="image-wrapper__wrapper">
         <img className="image" />
      </div>
   </div>
</div>

Co to spowoduje? No to już zależy od naszej aplikacji, ponieważ za dużo śmieci sprawi, że będzie trudno dostępny np. dla readerów. Zamiast kilku klas dodanych dodatkowo do elementów będziemy mieć dużą ilość wrapperów. Natknąłem się na porównania schludnego kodu HTML z przyjemnymi nazwami klas do napchanego wraperami kodu z SC, gdzie autor porównania atakuje SC twierdząc, że jest tam sam syf. Trudno takie porównania ocenić. Z przykładu powyżej nie można wywnioskować jakie intencje miał autor kodu, czy chciał ostylować sam element czy go faktycznie w coś opakować?

Na pewno zbyt duża ilość wrapperów mogłaby świadczyć o nieumiejętnym posługiwaniu się SC. Po części w rozwiązywaniu problemu z wraperami przychodzi na pomoc kompozycja.

// Card.js
import { CenterElements } from '../styles/commonStyles';

const ButtonsCentered = styled(CenterElements)`
 padding: .1rem;
`;

Ważne jest, że dzięki temu nie tworzymy kolejnej warstwy/tagu html. W tym momencie warto przyjrzeć się co pokazuje inspektor na naszym projekcie. W obu przypadkach widzimy dokładnie to samo co mamy w kodzie:

(standard-css-react)

(styled-components-react)

Tutaj klasy to nazwy komponentów oraz hash. Spotkałem się z głosami, że SC jest fajne, bo pokazuje nazwy komponentów. Jest to bardzo przydatne tak samo jak nazwy klas w standardowym podejściu, ale to moim zdaniem powinno być ważne w środowisku developerskim. Na produkcji dla optymalizacji przydałby się hashing co w SC mamy domyślnie, natomiast żeby uzyskać nazwy komponentów, jak na przykładzie powyżej, trzeba zmienić konfiguracje.

2. Bajery: propsy, atrybuty, animacje, media query

Kolejnym elementem, który przekonał mnie do używania SC to prosty sposób sterowania stylami za pomocą propsów, możemy też korzystać z wyrażeń warunkowych.

// ThemeSwitcher.js
const Switch = styled.div`
 width: 100%;
 height: 100%;

 ${({ checked }) => (checked && css`
     &::after {
       left: calc(100% - .5rem);
       transform: translateX(-100%);
     }
 `)}
`;

const ThemeSwitcher = ({ theme, onThemeChange }) => (
 <ThemeSwitcherWrapper>
   <Switch onClick={onThemeChange} checked={theme === THEMES.dark} />
 </ThemeSwitcherWrapper>
);

Przekazujemy propsa, a SC już sobie sprawdza wartość. W przykładzie powyżej, gdy checked będzie true to dodatkowy styl zostanie dodany. Poniżej jedno z moich ulubionych zastosowań:

const gradient = `linear-gradient(to top right, ${COLORS.scPink}, ${COLORS.scOrange})`;

export const SPINNER_SIZES = {
 small: '2rem',
 medium: '3rem',
 large: '4rem',
};

export const Spinner = styled.div`
 background: ${gradient};
 width: ${({ size }) => SPINNER_SIZES[size]};
 height: ${({ size }) => SPINNER_SIZES[size]};
`;

Oczywiście można też korzystać ze zmiennych wcześniej zadeklarowanych, jak na przykładzie powyżej “gradient”, a to też daje duże pole do popisu. W świecie Reacta używamy propsów, staramy się tworzyć reużywalne komponenty lub korzystamy z gotowych. Dostajemy do dyspozycji poniższy komponent, komponent którego nie możemy zmienić, możemy podejrzeć jego kod:

// Heading.js
const Heading = ({ bottomColor, topColor }) => (
 <svg width="100%" height="100%" viewBox="0 0 100 100" preserveAspectRatio="none">
   defs>
     <linearGradient id="heading-linear-gradient" x1="0%" y1="100%" x2="100%" y2="0%">
       stop offset="0%" style={{ stopColor: bottomColor }} />
       <stop offset="100%" style={{ stopColor: topColor }} />
     </linearGradient>
   </defs>
   <g>
     path d="M0 0 L0 100 L100 80 L100 0 Z" fill="url(#heading-linear-gradient)" />
   </g>
 </svg>
);

// Card.js
const bottomColor = theme.mode === THEMES.light ? COLORS.scPink : COLORS.gradient4;
const topColor = theme.mode === THEMES.light ? COLORS.scOrange : COLORS.gradient2;

<Heading bottomColor={bottomColor} topColor={topColor} />

Tutaj sprawa użycia jest dosyć prosta — przekazujemy kolory. Chciałem zwrócić uwagę, że jeżeli używamy SC to kolory i tak mamy zadeklarowane w JS. Co ze standardowym podejściem? Kolory zapewne mamy zdefiniowane w plikach css, dlatego bez sensu je dublować do js. Opcji jest wiele, każdy ma na to swoje patenty. Jednym z brudnych sposobów dostania się do zmiennych z root w trakcie wykonywania kodu jest getPropertyValue. Jak to działa możecie zobaczyć w repo w części standard-css-react. Można by się pokusić, by style parsować na etapie webpacka.

Następne funkcjonalności SC to zagnieżdżanie oraz media query:

// Card.js
const SideBack = styled.div`
 background: ${gradient};
`;

const CardWrapper = styled.div`
 height: 47rem;

 @media ${BREAKPOINTS.desktop} {
   height: 44rem;

   &:hover ${SideBack} {
     transform: rotateY(0);
   }
 }
`;

Powyższy przykład powinien być dosyć jasny, jednakże chciałbym podkreślić, że selektorem też może być SC.

Animacje. Wystarczy użyć keyframes:

// variables.js
import styled, { keyframes } from 'styled-components';

const sway = keyframes`
 0% {
   transform: rotate(0deg);
 }

 25% {
   transform: rotate(5deg);
 }

 50% {
   transform: rotate(0deg);
 }

 75% {
   transform: rotate(-5deg);
 }
`;

export const ANIMATIONS = {
 sway,
};
// Logo.js
const StyledLogo = styled.img`
 position: absolute;
 top: 5rem;
 left: 30%;
 width: 40%;
 animation: ${ANIMATIONS.sway} 4s linear infinite;
`;

Czasem trzeba bardziej pokombinować, a kolejną możliwością, którą oferuje SC to dobranie się do atrybutów komponentu. Dobrym przykładem może być tu NavLink:

const ns = 'navigation';

<NavLink to="/pages" className={`${ns}__button`} activeClassName={`${ns}__button--active`}>

Potencjalne wyzwanie mogłoby tutaj stanowić activeClassName. Można by wykorzystać activeStyle jednakże akurat w tym przypadku można skorzystać z attrs:

const activeClassName = uuid();
const StyledNavLink = styled(NavLink).attrs({ activeClassName })`
 &.${activeClassName} {
   color: red;
 }
`

Minusem jest generowanie unikatowego identyfikatora klasy, za to według mnie plus jest taki, że nie definiujemy go sami.

3. Style globalne

Jeżeli chcemy zrezygnować całkiem z importowania css’ów to w SC istnieją style globalne. Deklarujemy style korzystając createGlobalStyle, a później go używamy jako komponentu:

// global.js
import { createGlobalStyle } from 'styled-components';

export default createGlobalStyle`
 html {
   box-sizing: border-box;
   font-size: 62.5%;
   font-family: sans-serif;
 }
`;
// App.js
import Global from './styles/global';

 render() {
   const { theme } = this.state;

   return (
     <Fragment>
       <Normalize />
       <Global />
       <ThemeProvider theme={{ mode: theme }}>
         <CustomFlex>
           <Card />
           <ThemeSwitcher onThemeChange={this.handleThemeChange} theme={theme} />
         </CustomFlex>
       </ThemeProvider>
     </Fragment>
   );
 }

Moim preferowanym edytorem tekstu jest VSCode, a wśród zainstalowanych dodatków znajduje się jak vscode-styled-components, jednakże przy createGlobalStyle jak i blokach “css” chciałem zauważyć, że mogą się pojawić problemy z kolorowaniem kodu czy intellisense.

Po lewej style w createGlobalStyle w pliku js, a po prawej zwykły css:

Na ten moment nie jestem w stanie stwierdzić, czy to niedopracowanie dodatku czy coś z moją konfiguracją. Dobrze mieć to na uwadze.

4. Theme Provider

SC oferuje także providera do styli. Jest to tak jakby kontekst reactowy. Ciekawą funkcjonalnością jest tzw. variants, gdzie oprócz tematu możemy zdefiniować variant komponentu:

// LinkButton.js
import styledTheming, { css } from 'styled-components';

const themesProps = styledTheming.variants('mode', 'variant', {
 [LINK_BUTTON_VARIANTS.default]: {
   [THEMES.light]: css`
     color: ${COLORS.palevioletred};
     border-color: ${COLORS.white};
     background: ${COLORS.white};
   `,
   [THEMES.dark]: css`
     color: ${COLORS.black};
     background: ${COLORS.white};
     border-color: ${COLORS.white};
   `,
 }
...
});

const StyledLink = styled(CustomLink)`
 ${themesProps}
 text-transform: uppercase;
`;
// Card.js
<LinkButton href="https://www.styled-components.com/docs/" target="_blank"     variant={LINK_BUTTON_VARIANTS.default}>Documentation</LinkButton>

Plusem jest to, że theme jest dostępne w SC, zatem nie trzeba go osobno przekazywać jako props. Minusem, że jeżeli chcemy mieć dostęp do wartości theme w samym komponencie reactowym to trzeba go opakować w withTeme. Jeżeli chcielibyśmy użyć własnego kontekstu reaktowego to wyjdzie na to samo, ponieważ jeżeli będziemy chcieli go używać to i tak go będziemy opakowywać. Przykład użycia poniżej, a jak zaimplementować własnego ThemeProvider i withLayout można zobaczyć w themeContext.js:

// LinkButton.css
.link-button {
 &--default--light {
   color: var(--color-palevioletred);
   border-color: var(--color-white);
   background: var(--color-white);
 }
}
// LinkButton.js
const LinkButton = (props) => {
 const { theme, variant, children } = props;
 const classNames = cx(
   [`${ns}`],
   {
     [`${ns}--${variant}--light`]: theme === THEMES.light,
     [`${ns}--${variant}--dark`]: theme === THEMES.dark,
   },
 );

 return <a className={classNames} {...props}>{children}</a>;
};

// Card.js
<LinkButton href="https://www.styled-components.com/docs/" target="_blank" variant={LINK_BUTTON_VARIANTS.default}>Documentation</LinkButton>

export default withTheme(LinkButton);

Podsumowując na ten moment wygodniejsze dla mnie jest korzystanie z reactowego kontekstu, z którego już korzysta się też w innych celach, więc nie trzeba wprowadzać dodatkowych elementów.

5. Testy jednostkowe

Implementując SC możemy testować style, a dokładniej react-test-renderer, który posiada toHaveStyleRule, dzięki czemu można sprawdzić czy element posiada dokładnie takie wartości jakie byśmy chcieli:

// Details.spec.js
const mockComponent = theme => (
 <ThemeProvider theme={{ mode: theme }}>
   <Details>
     <span>TEST</span>
     <span>TEST2</span>
     <span>TEST3</span>
   </Details>
 </ThemeProvider>
);

[
 THEMES.light,
 THEMES.dark,
].forEach((theme) => {
 const cmp = renderer.create(mockComponent(theme)).toJSON();

 describe(`${theme} theme`, () => {
   test('renders correctly child elements', () => {
     expect(cmp).toMatchSnapshot();
     expect(cmp.children.length).toEqual(3);
   });

   test('should list items have correct font size', () => {
     cmp.children.forEach((item, idx) => {
       expect(cmp.children[idx]).toHaveStyleRule('font-size', '1.4rem');
     });
   });
 });
});

W snapshocie mamy zapisane style, co jest bardzo fajne. Bez SC mielibyśmy w snapshocie tylko nazwy klas.

// Details.spec.js.snap
// Jest Snapshot v1, https://goo.gl/fbAQLP

exports[`dark theme renders correctly child elements  1`] = `
.c0 {
  list-style: none;
  padding: 0 4rem;
  margin: 4rem auto .6rem;
}

.c1 {
  text-align: center;
  font-size: 1.4rem;
  padding: 1rem;
  color: #222;
}

.c1:not(:last-child) {
  border-bottom: 1px solid rgba(0,0,0,0.2);
}

<ul
  className="c0"
>
  <li
    className="c1"
  >
    <span>
      TEST
    </span>
  </li>
  <li
    className="c1"
  >
    <span>
      TEST2
    </span>
  </li>
  <li
    className="c1"
  >
    <span>
      TEST3
    </span>
  </li>
</ul>
`;

Do testowania możemy również używać Enzyme. W snapshotach też zapisują się style jednakże nie ma takiej sprytnej metody jak z react-test-renderer. Enzyme spisuje się dość dobrze, jednakże z tego co zauważyłem to autorzy enzyme nie nadążają za nowymi wersjami SC, a i nawet za samym Reactem. Na przykład naturalnym byłoby użycie shallow, a nie mount i zrobienie shallowWithTheme. W momencie pisania tego artykułu SC ver. 3 wszytko będzie ok, ale już przy ver. 4 napotkamy problemy.

// Card.spec.js
const renderComponent = ({
 theme = THEMES.light,
}) => mount(
 <ThemeProvider theme={{ mode: theme }}>
   <Card
     theme={{ mode: theme }}
   />
 </ThemeProvider>,
);

test('renders correct HTML', () => {
 expect(renderComponent({})).toMatchSnapshot();
});

[
 { theme: THEMES.light, bottomColor: 'rgb(219, 112, 147)', topColor: 'rgb(218, 163, 87)' },
 { theme: THEMES.dark, bottomColor: '#667db6', topColor: '#0082c8' },
].forEach(({ theme, bottomColor, topColor }) => {
 test(`should Heading have correct props for ${theme} theme`, () => {
   const headingProps = renderComponent({ theme }).find('Heading').props();

   expect(headingProps.bottomColor).toEqual(bottomColor);
   expect(headingProps.topColor).toEqual(topColor);
 });

6. Kod wynikowy i czas transpilacji

Nasz przykładowy projekt jest dosyć mały, więc ciężko tutaj zrobić porównanie:

Z powyższej tabeli wynika, że jeżeli nie parsujemy css to projekt buduje się szybciej. Mając styczność z większymi projektami, z miksowanym podejściem raczej bym powiedział, że czas budowania projektu znacząco się nie zmienił, jednakże kod wynikowy waży więcej. Kiedy weźmiemy pod lupę np. aplikację, która służy do budowania stron statycznych na request użytkownika to głównym czynnikiem go interesującym jest czas budowania strony.

Spotkałem się z przypadkiem, że z dnia na dzień czas tworzenia całego bundla strony zwiększył się prawie czterokrotnie: z 30 s na 2 min. Zmiana nie do przyjęcia. Co się stało?

Problemem była nowa wersja jednej z paczek związanych z babelem i styled components. Minorowa zmiana o 0.0.1. Nie powinno to nic zmienić a jednak. Pomogło przywrócenie do wcześniejszej wersji. Takie sytuacje mogą się zdarzyć, szczególnie gdy korzystamy z bibliotek open source. Pomaga na pewno fixowanie wersji paczek oraz prywatne repo.

Podsumowanie

Każde rozwiązanie ma swoje wady i zalety i to też dotyczy SC. Trzeba wyważyć czy do naszego projektu ta biblioteka będzie pasować. Po jej użyciu świat CSS na pewno stanie się dla nas inny. Dla mnie Styled Components to przyjemniejszy React. Czy użyłbym SC w kolejnym projekcie? Tak.

Podobne artykuły

[wpdevart_facebook_comment curent_url="https://geek.justjoin.it/dlaczego-styled-components-pasuja-reacta-przykladzie-dwoch-aplikacji/" order_type="social" width="100%" count_of_comments="8" ]