- node 14버전(12나 15도 괜찮음)과 MySQL을 미리 설치하기
- cd back
- npm i
- .env 작성하기(COOKIET_SECRET과 MYSQL_PASSWORD 비밀번호 설정)
- config/config.json 설정(MYSQL 접속 설정)
- npx sequelize db:create(스키마 생성)
- npm run dev했다가 ctrl + c로 끄기(테이블 생성)
- npx sequelize db:seed:all(기초 데이터 넣기)
- npm run dev
- localhost:3095에서 서버 돌아가는 중
- 백엔드 개발자가 API.md와 typings/db.ts를 남겨둔 상황
- package.json
- npm init으로 생성
- npm i react react-dom typescript
- npm i @types/react @types/react-dom
- 설치 후 package-lock.json과 node_modules 폴더가 생성됨
- .eslintrc
- eslint 설정 파일
- 코드 점검 도구, 직접 설정하면 팀원간 의견 충돌이 있으니 prettier에 위임
- npm i -D eslint
- .prettierrc
- prettier 설정 파일
- 저장하면 알아서 코드를 수정해줌(에디터 설정 필요)
- npm i -D prettier eslint-plugin-prettier eslint-config-prettier
- tsconfig.json
- 타입스크립트 설정
- 언어 문법과 자바스크립트 결과물이 어떻게 나와야하는지 설정하는 파일
- lib은 ES2020, DOM(브라우저), module은 esnext처럼 최신 설정이지만 target은 es5로 IE 브라우저에서도 돌아갈 수 있게 변환
- strict: true를 켜놓아야 타입 체킹을 해줘서 의미가 있음.
- webpack.config.ts
- 웹팩 설정
- ts, css, json, 최신 문법 js 파일들을 하나로 합쳐줌.
- npm i -D webpack @types/webpack @types/node
- entry에서 파일을 선택하면 module에 정해진 rules대로 js로 변환하여 하나의 파일로 합쳐줌(output). plugins는 합치는 중 부가적인 효과를 줌
- ts는 babel-loader로, css는 style-loader와 css-loader를 통해 js로 변환
- babel에서는 @babel/preset-env(최신문법 변환) @babel/preset-react(리액트 jsx 변환), @babel/preset-typescript(타입스크립트 변환)
- npm i -D css-loader style-loader @babel/core babel-loader @babel/preset-env @babel/preset-react @babel/preset-typescript
- publicPath가 /dist/고 [name].js에서 [name]이 entry에 적힌대로 app으로 바뀌어 /dist/app.js가 결과물이 됨.
- index.html 작성
- /dist/app.js로 웹팩이 만들어낸 js파일 불러옴
- 아이콘, 폰트, 파비콘같은 것은 슬랙에서 그대로 사용
- #app 태그에 리액트가 렌더링됨.
- client.tsx에 간단한 tsx 작성
- tsconfig-for-webpack-config.json
- 개발용 서버인 devServer 옵션 추가(port는 3090, publicPath는 /dist/로
- webpack serve할 때 webpack.config.ts를 인식 못하는 문제
- npm i -D ts-node webpack-dev-server @types/webpack-dev-server webpack-cli
- npm i cross-env
- package.json의 scripts의 dev를 cross-env TS_NODE_PROJECT="tsconfig-for-webpack-config.json" webpack serve --env development
- npm run dev하면 localhost:3090에서 서버 실행됨.
- hot reloading 설정
- npm i -D @pmmmwh/react-refresh-webpack-plugin react-refresh
- webpack의 babel-loader 안에 설정(env) 및 plugin으로 추가
- 폴더 구조 세팅
- 페이지들은 pages
- 페이지간 공통되는 틀은 layouts
- 개별 컴포넌트는 components
- 각 컴포넌트는 컴포넌트 폴더 아래 index.tsx(JSX)와 styles.tsx(스타일링)
- ts와 webpack에서 alias 지정
- npm i -D tsconfig-paths
- tsconfig에서 baseUrl와 paths 설정
- webpack에서는 resolve안에 alias 설정
- ../layouts/App같은 것을 @layouts/App으로 접근 가능
- emotion 세팅
- styled components와 비슷하지만 설정이 간단함.
- npm i @emotion/core @emotion/styled
- npm i -D babel-plugin-emotion (웹팩에 babel 설정 추가)
- 스타일드 컴포넌트로 만들 때 변수를 많이 만드는 셈이므로 & 같은 선택자 적극 활용해야 변수 이름짓기를 최소화할 수 있음.
- @layouts/App 작성
- 리액트 라우터 적용하기
- npm i react-router react-router-dom @types/react-router @types/react-router-dom
- client.tsx에서 App을 BrowserRouter로 감싸기
- @layouts/App에 Switch, Redirect, Route 넣기
- @pages/SignUp 작성
- @loadable/component
- 라우터를 코드스플리팅 해줌
- 회원가입 페이지에 접근한 사람은 회원가입 페이지에 필요한 JS만 받음
- 3초 룰 기억하자!
- npm i @loadable/component @types/loadable__component
- fork-ts-checker-webpack-plugin
- webpack은 ts체크 후 eslint체크 후 빌드 시작
- ts랑 eslint는 동시에 체크하면 더 효율적
- 이 플러그인이 동시에 진행하게 해줌.
- 회원가입 axios로 진행
- npm i axios
- CORS 문제를 피하기 위해서 devServer에 proxy 세팅
- CORS는 브라우저에서 다른 도메인의 서버로 요청을 보낼 때 발생
- 같은 도메인의 서버로 요청을 보내거나, 서버끼리 요청을 보낼 때는 발생하지 않음
- 따라서 같은 도메인인 proxy서버를 띄워 CORS를 피해갈 수 있음.
- useInput 커스텀 훅 만들기
- 커스텀 훅으로 훅들간에 중복된 것을 제거할 수 있음
- 훅 내부에 훅을 작성할 수 있는 유일한 케이스
- useCallback은 return 안에 들어있는 함수에 꼭 적용해주자
- useMemo는 return 안에 들어있는 값에 적용하자
- @pages/LogIn 작성 및 SWR
- 로그인 한 사람이 회원가입/로그인 페이지에 접근한다면?
- GET 요청은 SWR로 하는 것도 괜찮음
- npm i swr
- SWR에 fetcher(axios를 사용)를 달아줌.
- 로그인했음을 증명하기 위해 withCredentials: true 잊으면 안 됨.
- @layouts/Workspace 작성
- 눈에 띄는 구역 단위로 스타일드컴포넌트로 만들어둠.
- 구역 내부의 태그들은 스타일드컴포넌트로 만들면 변수명 지어야 하니 css선택자로 선택
- 그라바타
- npm i gravatar @types/gravatar
- Github같은 아이콘을 만들 수 있음
- typescript 정의
- 기본적으로 변수, 매개변수, 리턴값에 타입을 붙여주면 됨.
- 남이 타이핑해둔 것 분석하는 게 어려움
- Go to Type Definition
- 자바스크립트 라이브러리 작성자와는 다른 사람이 만든 ts 라이브러리가 @types로 시작하는 것들
- @components/DMList 작성
- 현재 채널 참여자 목록 가져오기
- @pages/DirectMessage 작성
- Header와 ChatList, ChatBox로 구성
- @components/ChatBox 먼저 작성
- react-mentions 활용
- DM에서는 멘션 기능이 없지만 Channel에서는 있을 것
- DM 보내보기
- optimistic UI
- 먼저 프론트에서 표시하고, 서버로는 그 다음에 요청보냄
- 요청 실패하는 순간 프론트에서 제거하고 에러 메시지 띄움
- 보낼 때 에러가난 것은 서버쪽에서 socket 연결 여부를 확인하기 때문
- DM 로딩은 useSWRInfinite 사용
- 결과물이 2차원 배열 꼴로 나옴.
- 첫 번째 인자가 주소 문자열이 아닌 주소를 리턴하는 함수
- 이 함수의 매개변수로 페이지가 들어있어서 현재 몇 페이지인지 알 수 있음.
- Workspace에 소켓 연결
- DMList에 onlineList, dm 이벤트 연결
- @components/ChatList 작성 및 @components/Chat 구현
- react-custom-scrollbars
- makeSection 구현(dayjs)
- 프로파일링 하면서 Chat에 memo 적용하기
- 인피니트 스크롤링 구현
- @components/ChannelList 작성
- @pages/ChannelMessage 작성
- Channel Chat 보내보기
- 시간 남으면 모달들 구현
- 빌드 설정