원티드 프리온보딩 10월 챌린지 과제를 수행하면서 공부한 내용을 작성하였습니다.
이 글은 구현의 과정을 의식의 흐름대로 담고 있기 때문에 조금 길고 성가실 수 있습니다.
history api에 대한 자세한 내용은 포함되어있지 않습니다. 해당 정보가 필요하다면 MDN window.history를 검색하는 것을 추천합니다.
과제는 React와 History API 사용하여 SPA Router 기능 구현하기입니다.

react-router-dom을 설치해서 그냥 사용해봤을 뿐 url을 바꾼다거나 라우팅을 할때 코드의 내부 구조가 어떻다던지에 대해 그다지 깊게 생각해 본 적이 없다. 프로그래머스 챌린지 할 때 SPA를 구현해야하기 때문에 새로고침 없이 페이지를 전환하는데 histroy.pushState, location 등을 사용해서 구현했었다. 비슷하게 하면 될 것 같아서 '생각보다 간단하겠는데?'라고 생각하면서 과제를 시작했다.

처음 아이디어

처음 아이디어는 path를 받아서 component를 랜더링하는 Route컴포넌트를 작성하려고 했다. 그래서 코드를 다음과 같이 만들었다.

1import React, { useEffect } from "react";
2
3interface IRouteProps {
4 path: string;
5 component: React.ReactElement;
6 state?: {};
7}
8
9const Route = ({ path, component, state }: IRouteProps) => {
10 useEffect(() => window.history.pushState(state, "", path), [path]);
11 return component;
12};
13
14export default Route;
1import React from "react";
2
3interface IRoutesProps {
4 children: React.ReactElement;
5}
6
7const Router = ({ children }: IRoutesProps) => {
8 return children;
9};
10
11export default Router;

그리고 어플리케이션의 Router를 만들었다.

1interface IAppRoutersProps {}
2
3const AppRouters = () => {
4 return (
5 <Router>
6 <>
7 <Route
8 path="/"
9 component={<Main />}
10 />
11 <Route
12 path="/about"
13 component={<About />}
14 />
15 </>
16 </Router>
17 );
18};
19
20export default AppRouters;
당연히 안된다.(ㅋㅋㅋㅋㅋㅋㅋㅋㅋ)

왜 안될까?

  1. AppRouter는 함수니까 위에서 아래로 코드를 읽으면서 Main 컴포넌트를 실행하고 /about으로 pushState를 넣은 다음에 About 컴포넌트를 실행하기 때문에 한 화면에 두 컴포넌트가 동시에 출력된다.
  2. useEffect가 실행될 때, Route에 전달된 path를 보고 그냥 history.pushState를 실행한다.

해결하려면?

일단 기본 url은 기본 동작으로 가지고 있어야한다. 최초로 실행시킬 컴포넌트는 '/' 경로에서 그냥 동작하도록 하게 하고 나머지 url에 대해서는 브라우저 주소창에 어떤 url을 입력할 때마다 입력한 경로를 보고 경로에 해당되는 컴포넌트를 실행해야한다.

그래도 아얘 동작이 안되는 것은 아니기 때문에 일단 여기서부터 수정을 해야겠다. 하지만 여전히 감이 오지 않는다. 일단 다른 사람들이 작성한 코드와 문서를 레퍼런스로 삼아야겠다.

바닐라 자바스크립트에서 SPA는 어떻게 구현할까?

프로그래머스 프론트앤드 챌린지에서 SPA를 어떻게 구현했는지 다시 찾아보기로했다.

프로그래머스의 블로그에 적혀있는데로 코드를 구현해보았다. 구현을 하면서 핵심이라고 생각한 부분은 다음과 같다.

  1. url이 바뀌었는지 그대로인지 알아야한다.
  2. url이 바뀌었다는 것을 감지했다면 해당 컴포넌트를 랜더링 해야한다. 이때 새로고침이 일어나지 않아야한다.

url이 바뀌었는지는 어떻게 알 수 있을까?

위의 예제 코드를 살펴보면 url의 변경 여부는 window.location.pathname의 변화 여부로 알 수 있다. 참고로 window.location.pathname으로도 url을 변경할 수 있다. 하지만 이렇게 변경을 하면 새로고침이 일어난다.

새로고침이 일어나지 않고 컴포넌트를 변경하기 위해서는 history.pushState를 사용해야한다. pushState는 url을 변경하지만 새로고침은 일어나지 않는다. 그래서 위의 코드는 조건문을 사용하여 pathname을 살펴보고 일치하는 url에 대해서 컴포넌트를 랜더링하는 방식으로 SPA를 구현하고 있다. 지금까지 살펴본 내용을 바탕으로 React 프로젝트에 적용해보기로 했다.

1export function changeRouter(pathname: string) {
2 return window.history.pushState(null, "", pathname);
3}
1import { changeRouter } from "@/lib/lib";
2import React, { useEffect, useState } from "react";
3
4interface IRouteProps {
5 path: string;
6 component: React.ReactElement;
7 state?: {};
8}
9
10const Route = ({ path, component, state }: IRouteProps) => {
11 const [isPath, setIsPath] = useState(false);
12
13 useEffect(() => {
14 const { pathname } = window.location;
15 if (path === pathname) {
16 setIsPath(true);
17 changeRouter(path);
18 }
19 }, [path]);
20
21 return isPath ? component : null;
22};
23
24export default Route;
url에 /나 /about을 입력해보세요.

로직은 정말 간단하게 Route만 수정했다. pathname이 path와 일치할 때만 component를 랜더링하도록 로직을 수정하였다. 그러자 라우터에 맞는 컴포넌트만 랜더링한다. Yeah!?

react-router-dom 깃허브 레포지토리 살펴보기

하지만 아직 갈길이 멀다. 위의 코드는 동작은 하더라도 요구사항과 많이 동떨어져있다.

  1. histroy에서 pushState의 맥락을 공유하는 무언가가 없기 때문에 컴포넌트 내부에서 버튼을 눌렀을 때, pushState를 실행시키면 주소창에 url만 변경되고 컴포넌트는 랜더링 되지 않는다.
  2. 뒤로 가기, 앞으로 가기가 동작하지 않는다.
  3. Router.tsx 컴포넌트가 불필요하다. 그냥 Route만 있어도 라우팅을 동작시킬 수 있다. Router안에 Route를 반드시 쓰도록 강제할 방법도 없다. 하지만 요구 조건은 Router안에 children으로 Route가 들어있어야한다. react-router-dom(v6) 라이브러리를 실제로 사용할 때, 그 사용 조건은 조금 까다롭다.
1function Router() {
2 return (
3 <Routes>
4 <Route
5 pathname="/"
6 element={<Main />}
7 />
8 </Routes>
9 );
10}
11
12function App() {
13 return <Router />;
14}
15
16function Root() {
17 return (
18 <React.StrictMode>
19 <BrowserRouter>
20 <App />
21 </BrowserRouter>
22 </React.StrictMode>
23 );
24}
25
26ReactDOM.render(<Root />, document.getElementById("root"));

위의 규칙을 지키지 않고 Route를 단독으로 사용하려고 시도하거나 BrowserRouter가 App을 감싸고 있지 않는 등의 코드를 작성하면 경고 메시지를 출력하면서 어플리케이션이 동작하지 않는다. 내가 생각했을 때는 그냥 Route만 있으면 될 것 같은데, 왜 이렇게 동작을 하도록 강제했는지 잘 모르겠다. 과제의 요구 조건을 만족 시키기 위해서는 몇가지 목적을 가지고 코드를 살펴봐야겠다.

  1. BrowserRouter는 어떤 역할인가?
  2. 왜 Routes로 Route를 감싸야 하는가?
  3. 반드시 Routes안에 Route를 쓰게 하도록 강제하는 방법은 무엇일까?

BrowserRouter는 어떤 역할을 하는가?

react-router의 깃허브 레포지토리를 살펴보면 BrowserRouter 함수를 찾을 수 있다. 결론적으로 말하면 BrowserRouter는 Router 컴포넌트를 반환하고 Router 컴포넌트는 NavigationContext.Provider에 감쌓여있는 LocationContext.Provider를 반환한다.

LocationContext와 NavigationContext는 React Context API다. BrowserRouter로 App 컴포넌트를 감싸으면 어플리케이션 전체 Router 상태를 공유하는 Context API가 생성된다.

Context API는 전역으로 상태를 공유할 수 있게 해준다. 그래서 부모에서 자녀로 Props를 전달하지 않더라도 어느 곳에서든지 바로 Context API에 저장된 상태 값을 사용할 수 있다.

하지만 이렇게 좋다고 생각되는 Context API는 단점을 가지고 있다. Context API의 값이 변경되면 Context Provider 하위에 있는 모든 컴포넌트는 랜더링을 다시 하게 된다. 그래서 Context API는 전역 상태 도구로 잘 사용하지 않는다. 만약 Context 값을 사용하지 않는 하위 컴포넌트의 랜더링을 막기 위해서는 useMemo를 사용해서 해결한다.

위의 코드를 힌트로 삼아서 Router를 수정하면 다음과 같다. 아직 useMemo를 사용하면 이번 프로젝트에서 어떤 효과를 거둘 수 있는지 명확하게 이해한 상태가 아니기 때문에 이해한 부분까지만 코드를 작성했다.

1import React, { createContext, useState } from "react";
2
3interface LocationContextProps {
4 location: {
5 pathName: string;
6 };
7}
8
9const LocationContext = createContext<LocationContextProps>(null!);
10
11interface IRoutesProps {
12 children: React.ReactNode;
13}
14
15const Router = ({ children }: IRoutesProps) => {
16 const [location, setLocation] = useState({ pathName: "/" });
17 return (
18 <LocationContext.Provider
19 children={children}
20 value={{ location }}
21 />
22 );
23};
24
25export default Router;

react-router-dom v6와 사용 방법이 다르지만 어쨌든 Router를 사용하면 LocationContext가 생성되면서 location을 전역으로 관리할 수 있게 되었다. 아직은 임시로 Provider의 value를 넣었다. 여기까지 코드를 작성하고 Pages/Main을 작성하려고 하는데 과제 요구 조건 중에 useRouter가 떠올랐다.

react-router-dom에서 Link 컴포넌트는 a 태그의 역할을 하면서도 새로고침이 되지 않는다. 왠지 이 컴포넌트와 useRouter 훅을 함께 사용하면 Link 컴포넌트를 비슷하게 흉내낼 수 있겠다 생각하게 되었다.

1export const useRouter = () => {
2 function push(to: string) {
3 return history.pushState(null, "", to);
4 }
5
6 return { push };
7};
1import { useRouter } from "@/lib/hook";
2import React from "react";
3
4interface ILinkProps {
5 children: React.ReactNode;
6 to: string;
7}
8
9const Link = ({ children, to }: ILinkProps) => {
10 const { push } = useRouter();
11 return (
12 <a
13 onClick={(e) => {
14 e.preventDefault();
15 push(to);
16 }}
17 >
18 {children}
19 </a>
20 );
21};
22
23export default Link;

위의 이미지처럼 링크를 클릭하면 주소는 변경되지만 새로고침은 되지 않는다. 그럼 바닐라 자바스크립트에서 했던 것처럼 링크가 변경될 때, 해당되는 컴포넌트를 랜더링 하기만 하면 된다. useRouter 훅과 LocationContext를 연결하면 문제가 해결 될 것 같다. 최종적으로 작성한 코드는 아래와 같다.

1import { LocationContext } from "@/Routes/Router";
2import { useContext } from "react";
3
4export const useRouter = () => {
5 const { setLocation } = useContext(LocationContext);
6 function push(to: string) {
7 setLocation({ pathName: to });
8 return history.pushState(null, "", to);
9 }
10
11 return { push };
12};
1import React, { useContext, useEffect, useState } from "react";
2import { LocationContext } from "./Router";
3
4interface IRouteProps {
5 path: string;
6 component: React.ReactNode;
7 state?: {};
8}
9
10const Route = ({ path, component, state }: IRouteProps) => {
11 const { pathname } = window.location;
12 const [isPath, setIsPath] = useState(false);
13 const { setLocation } = useContext(LocationContext);
14
15 const route = (pathname: string) => {
16 if (path === pathname) {
17 setIsPath(true);
18 } else {
19 setIsPath(false);
20 }
21
22 window.onpopstate = () => {
23 setLocation({ pathName: pathname });
24 };
25 };
26
27 useEffect(() => {
28 route(pathname);
29 }, [pathname]);
30
31 return isPath ? component : null;
32};
33
34export default Route;

왜 Routes로 Route를 감싸야하는가?

Routes로 Route를 감싸야하는 이유는 그렇게 설계 되었기 때문이다. react-router의 깃허브 레포지토리를 살펴보면 반드시 Routes로 Route를 감싸도록 설계가 되어있다. Route를 단독으로 사용하려고 하면 에러가 발생하게 하였다. Route 컴포넌트를 보고 조금 띠용 했는데 왜냐하면 Route 컴포넌트는 아무것도 반환하지 않기 때문이다. 내부에 어떤 함수가 설계되어있는 것도 아니다. 그냥 _props만 받도록 설계되어있다.

내부적으로 path tree(그냥 그렇게 표현하겠다.)를 만드는 역할은 Routes에서 한다. 만약에 Routes에 children이 있으면 createRoutesFromChildren 함수가 동작하면서 path tree를 만든다. 그렇기 때문에 react-router-dom v6를 사용하면서 root를 설정하고 root 안에 children으로 url 경로를 입력할 수 있다.

1function Router() {
2 return (
3 <Routes>
4 <Route
5 path="/main"
6 element={<MainLayout />}
7 >
8 <Route
9 path="next"
10 element={<Next />}
11 />
12 </Route>
13 </Routes>
14 );
15}

Route에 대한 자세한 설명은 링크를 참조하기를 바란다.

반드시 Routes안에 Route를 쓰게 하도록 강제하는 방법은 무엇일까?

이 방법은 invariant 함수로 boolean 값과 message를 넘겨서 value 값이 false일 때면 에러를 던지도록 설계되어있다. Route 컴포넌트는 넘겨지는 값이 false이기 때문에 단독으로 사용하면 무조건 에러를 던지게 된다. Routes와 함께 사용하게 되면 Routes가 Route를 반환하지 않고 Route의 _props값을 사용하여 라우팅을 하기 때문에 invariant 함수는 동작하지 않는다.

마무리

원티드 3차 챌린지 첫번째 과제를 해보았다. 처음에는 막막했는데 그냥 한 단계씩 아이디어를 구체화하고 다른 사람 코드를 참조해보고 확장해가면서 문제를 해결할 수 있었다. 하지만 한계가 많은 코드다. react-router 처럼 충첩 라우팅을 할 수 없고, Dynamic Segments를 사용할 수 없다. 또 params를 구분할 수 없다는 한계가 있다.

어쩌다보니 history API보다 react-router 라이브러리 내부 코드를 더 많이 살펴보게 된 것 같다. 과제의 의도는 그게 아니었던 것 같은데... react-router-dom을 그냥 사용하기만 했는데 이번 기회에 라이브러리의 내부 코드를 살펴 보았다. 코드를 보기 전에 그냥 Route가 path에 일치하는 컴포넌트를 반환 한다고만 생각했는데 나의 생각과는 완전히 다르게 설계 되어있었다.

BrowserRouter는 결론적으로 Context Provider를 반환한다. NavigationContext와 LocationContext를 반환하고 location, path 등은 전역으로 공유가 된다. 그렇기 때문에 url을 변경하거나 뒤로가기 앞으로 가기를 하게되면 url 주소에 맞게 컴포넌트를 반환할 수 있다.

path tree를 구성하는 방법은 재귀로 되어있었다. 맨날 알고리즘 공부한다고 하면서 재귀 함수 문제를 생각없이 풀곤 했는데 지루한 작업을 다시 용기내서 해봐야겠다는 생각이 들었다. 생각해보면 아이디어를 떠올리는 것 조차도 실력이겠구나 싶다.

어떤 컴포넌트 내부에서 다른 컴포넌트가 사용되도록 강제하는 방법에 힌트를 조금 얻어간다. invariant(불변)라는 함수로 다른 개발자가 조건과 다른 행동을 했을 때, 에러를 던지도록 설계하면 된다. 이렇게 하면 다른 개발자와 협업을 할때, 같은 기능 다른 이름을 가진 컴포넌트가 우후죽순으로 생겨나는 것을 막을 수 있을 것 같다.

바닐라 자바스크립트로 SPA를 구현하는 것은 상반기에 지원했던 기업 과제에서도 진행한적이 있었고, 프로그래머스 프론트앤드 챌린지 과제로도 구현한 적이 있었다. 그런데 왜 계속 할 때마다 처음부터 다시 시작해야하는지 모르겠다. 계속 쌓아 올려도 무너지는 모래성 같은 느낌이다. 이번 기회에 다시 구현을 해보면서 history.pushState와 window.onpopstate를 다시 살펴보았다. 지금은 익숙하게 이야기할 수 있지만 만약에 다시 구현을 해야하는 일이 생긴다면 이 아티클을 보고 처음부터 자료를 찾아 헤매지 않을 수 있을 것 같다.

참고