575.리액트 프로젝트 생성하기
1. Node.js 설치
리액트 프로젝트를 생성하고 실행하려면 Node.js가 필요합니다. Node.js는 JavaScript 런타임으로, 프로젝트 생성 도구와 빌드 도구들이 이를 이용합니다.
- Node.js 최신 버전 또는 최신 LTS(Long Term Support) 버전을 Node.js 공식 웹사이트에서 다운로드하여 설치하세요.
2. Create React App 또는 Vite 사용
리액트 프로젝트를 쉽게 생성할 수 있는 두 가지 주요 도구는 Create React App과 Vite입니다. 각 도구의 특징은 다음과 같습니다:
- Create React App: 리액트 앱을 빠르게 설정하고 실행할 수 있는 도구입니다. 자동으로 설정을 관리하고, 개발 서버를 제공하여 코드 변경 사항을 자동으로 반영합니다.
- Vite: 빠른 개발 환경을 제공하는 최신 빌드 도구입니다. Create React App보다 더 빠른 빌드 속도와 모듈 핫 리로딩을 제공합니다.
3. 프로젝트 생성
Create React App을 이용한 프로젝트 생성
터미널을 열고 다음 명령어를 실행하세요:
npx create-react-app my-app
my-app을 원하는 프로젝트 이름으로 바꾸세요.
프로젝트 디렉토리로 이동하고, 개발 서버를 시작하세요:
cd my-app npm start
Vite을 이용한 프로젝트 생성
터미널을 열고 다음 명령어를 실행하세요:
npm init vite@latest my-app --template react
npm create vite
- TypeScript: React 프로젝트에 타입스크립트(TypeScript)를 사용합니다. 타입스크립트는 자바스크립트에 타입을 추가하여 더 안정적인 코드를 작성할 수 있게 해줍니다.
- TypeScript + SWC: 타입스크립트와 함께 SWC(Speedy Web Compiler)를 사용합니다. SWC는 빠른 컴파일 속도를 제공하는 자바스크립트/타입스크립트 컴파일러입니다.
- JavaScript: 기본 자바스크립트를 사용합니다. 자바스크립트는 웹 개발에서 가장 널리 사용되는 스크립트 언어입니다.
- JavaScript + SWC: 자바스크립트에 SWC 컴파일러를 사용합니다.
- Remix ↗: Remix는 React 기반의 웹 프레임워크로, 더 나은 사용자 경험과 개발자 경험을 제공하기 위해 설계되었습니다.
my-app을 원하는 프로젝트 이름으로 바꾸세요.
프로젝트 디렉토리로 이동하고, 필요한 패키지를 설치하세요:
cd my-app npm install
개발 서버를 시작하세요:
npm run dev
. 프로젝트 파일 열기
선호하는 코드 편집기(예: Visual Studio Code)로 프로젝트 디렉토리를 엽니다. Visual Studio Code를 사용하는 경우, 터미널에서 다음 명령어로 바로 열 수 있습니다:
code .
. 기본 프로젝트 구조 이해
프로젝트가 생성되면 기본적인 폴더와 파일 구조가 설정됩니다. 주요 디렉토리와 파일은 다음과 같습니다:
- src/: 소스 코드 디렉토리
- index.js 또는 main.jsx: 리액트 앱의 진입점
- App.js 또는 App.jsx: 주요 컴포넌트 파일
- public/: 정적 파일 디렉토리
- index.html: HTML 템플릿 파일
6. 패키지 설치
프로젝트를 생성한 후에는 필요한 서드파티 패키지를 설치해야 합니다. 예를 들어, 리액트 라우터를 설치하려면 다음 명령어를 실행하세요:
npm install react-router-dom
7. 프로젝트 실행
프로젝트 디렉토리에서 개발 서버를 시작하여 앱을 실행하세요:
Create React App:
npm start
Vite:
npm run dev
8. 브라우저에서 앱 보기
터미널에 표시된 로컬 서버 주소를 브라우저에서 열어 리액트 앱을 확인하세요. 일반적으로 http://localhost:3000 또는 http://localhost:5173이 됩니다.
예제 프로젝트 스냅샷
강의에서 제공된 스냅샷을 사용하여 동일한 환경에서 작업할 수 있습니다. 해당 스냅샷을 다운로드하고 프로젝트 디렉토리에 복사한 후,
패키지를 설치하고 개발 서버를 시작하면 됩니다.
npm install npm run dev
이제 기본 리액트 프로젝트가 준비되었습니다. 이 프로젝트를 통해 리액트의 기본 개념과 기능을 배울 수 있습니다.
앞으로의 학습을 위해 이 프로젝트를 기반으로 작업을 진행하세요.
1.main.tsx
import React from "react"; import ReactDOM from "react-dom/client"; import "./index.css"; import { RouterProvider, createBrowserRouter } from "react-router-dom"; import NewPost, { action as newPostAction } from "./routes/NewPost"; import RootLayout from "./routes/RootLayout"; import Posts, { loader as postsLoader } from "./routes/Posts"; import PostDetails,{loader as postDetailsLoader , action as postDeleteAction} from "./routes/PostDetails"; const router = createBrowserRouter([ { path: "/", element: <RootLayout />, children: [ { path: "/", element: <Posts />, loader: postsLoader, children: [ { path: "create-post", element: <NewPost />, action: newPostAction, // action을 직접 전달 }, { path: "/:postId", element: <PostDetails />, loader: postDetailsLoader , action:postDeleteAction }, ], }, ], }, ]); ReactDOM.createRoot(document.getElementById("root")!).render( <React.StrictMode> <RouterProvider router={router} /> </React.StrictMode> );
이 React 코드에서는 react-router-dom을 사용하여 블로그와 같은 애플리케이션을 위한 클라이언트 사이드 라우팅 시스템을 설정합니다. 주요 부분을 하나씩 설명하겠습니다.
임포트
- React와 ReactDOM: React 애플리케이션을 빌드하고 DOM에 렌더링하기 위한 핵심 라이브러리입니다.
- CSS: 애플리케이션의 스타일을 정의한 파일입니다.
- RouterProvider와 createBrowserRouter: react-router-dom에서 가져온 함수로, 라우팅을 구성하고 제공하는 역할을 합니다.
- 컴포넌트와 액션/로더: 여러 컴포넌트와 그에 대응하는 액션/로더를 임포트합니다.
- NewPost와 newPostAction: 새로운 게시물을 생성하기 위한 컴포넌트와 액션입니다.
- RootLayout: 메인 레이아웃 컴포넌트입니다.
- Posts와 postsLoader: 게시물 목록을 표시하기 위한 컴포넌트와 로더입니다.
- PostDetails와 postDetailsLoader: 특정 게시물의 세부 정보를 표시하기 위한 컴포넌트와 로더입니다.
라우터 구성
라우터 구성은 createBrowserRouter를 사용하여 정의합니다.
루트 경로 (/):
- element: RootLayout 컴포넌트를 렌더링합니다.
- children: 루트 경로 아래에 중첩된 라우트입니다.
게시물 경로 (/):
- element: Posts 컴포넌트를 렌더링합니다.
- loader: 게시물 목록을 렌더링하는 데 필요한 데이터를 불러오는 postsLoader입니다.
- children: 게시물 경로 아래에 중첩된 라우트입니다.
- 새 게시물 생성 경로 (create-post):
- element: NewPost 컴포넌트를 렌더링합니다.
- action: 폼 제출을 처리하는 newPostAction입니다.
- 게시물 세부 정보 경로 (/:postId):
- element: PostDetails 컴포넌트를 렌더링합니다.
- loader: postId를 기반으로 특정 게시물의 데이터를 불러오는 postDetailsLoader입니다.
- 새 게시물 생성 경로 (create-post):
렌더링
ReactDOM.createRoot 메서드는 RouterProvider 컴포넌트를 렌더링하여 React 애플리케이션을 초기화하고, 구성된 router를 prop으로 전달합니다. 이는 애플리케이션에 라우터 컨텍스트를 설정합니다.
요약
- RootLayout: 메인 레이아웃 역할을 합니다.
- Posts: 게시물 목록을 표시하며, postsLoader를 통해 데이터를 불러옵니다.
- NewPost: 새로운 게시물을 생성합니다.
- PostDetails: 특정 게시물의 세부 정보를 표시하며, postDetailsLoader를 통해 데이터를 불러옵니다.
- 라우팅 구조: 루트 경로 (/)는 게시물 (/), 새 게시물 (/create-post), 게시물 세부 정보 (/:postId)를 포함합니다.
이 설정은 애플리케이션에서 게시물을 관리하기 위한 구조화되고 확장 가능한 라우팅 시스템을 보장합니다.
2.RootLayout.tsx
import { Outlet } from "react-router-dom"; import MainHeader from "../components/MainHeader"; const RootLayout = () => { return ( <> <MainHeader /> <main> <Outlet /> </main> </> ); }; export default RootLayout;
RootLayout 컴포넌트
- MainHeader 컴포넌트: 애플리케이션의 헤더 역할을 합니다. 헤더에는 주로 네비게이션 링크나 로고 등이 포함됩니다.
- <main> 요소: 애플리케이션의 주요 콘텐츠 영역입니다.
- Outlet 컴포넌트: react-router-dom에서 제공하는 컴포넌트로, 중첩된 라우트를 렌더링하는 역할을 합니다. 라우팅 설정에서 지정한 자식 컴포넌트가 이 위치에 렌더링됩니다.
설명
MainHeader 컴포넌트
- MainHeader 컴포넌트를 가져와서 <MainHeader />로 렌더링합니다. 이 컴포넌트는 애플리케이션의 모든 페이지에서 공통적으로 표시될 헤더를 정의합니다.
Outlet 컴포넌트
- Outlet은 react-router-dom에서 제공하는 컴포넌트로, 부모 라우트 컴포넌트 내에서 자식 라우트 컴포넌트가 렌더링될 위치를 지정합니다.
- RootLayout 컴포넌트가 루트 레이아웃으로 설정되면, 그 하위에 정의된 모든 자식 라우트는 <Outlet /> 위치에 렌더링됩니다.
JSX 구조
- <MainHeader />는 항상 화면 상단에 표시되는 헤더 컴포넌트입니다.
- <main> 태그 내에 <Outlet />이 위치하여 중첩된 라우트 컴포넌트가 렌더링될 공간을 만듭니다.
요약
RootLayout 컴포넌트는 애플리케이션의 기본 레이아웃을 정의합니다. 이 레이아웃은 다음과 같은 구조로 구성됩니다:
- 항상 표시되는 헤더(MainHeader).
- 중첩된 라우트를 렌더링하기 위한 <main> 태그 내의 <Outlet>.
이 설정을 통해 애플리케이션의 페이지 간에 일관된 레이아웃을 유지하면서, 각 페이지의 고유한 콘텐츠를 적절한 위치에 렌더링할 수 있습니다.
3.MainHeader.tsx
MainHeader 컴포넌트
- 아이콘: react-icons/md 라이브러리에서 MdPostAdd와 MdMessage 아이콘을 가져와 사용합니다.
- CSS 모듈: MainHeader.module.css 파일에서 정의된 스타일을 classes 객체로 가져옵니다.
- 링크: react-router-dom의 Link 컴포넌트를 사용하여 페이지 간 네비게이션을 처리합니다.
import { MdPostAdd, MdMessage } from 'react-icons/md'; import classes from './MainHeader.module.css'; import React from 'react'; import { Link } from 'react-router-dom'; const MainHeader:React.FC=()=> { return ( <header className={classes.header}> <h1 className={classes.logo}> <MdMessage /> <Link to="/" className='white'>리액트 포스트</Link> </h1> <p> <Link to="/create-post" className={classes.button}> <MdPostAdd size={18} /> 포스트하기 </Link> </p> </header> ); } export default MainHeader;
아이콘 사용
- MdMessage 아이콘은 로고 옆에 표시됩니다.
- MdPostAdd 아이콘은 "포스트하기" 버튼 옆에 표시됩니다.
헤더 구조
- <header> 태그는 전체 헤더를 감쌉니다.
- <h1> 태그는 로고를 포함하며, MdMessage 아이콘과 "리액트 포스트" 텍스트가 포함된 Link를 렌더링합니다.
- <p> 태그는 "포스트하기" 버튼을 포함하며, MdPostAdd 아이콘과 "포스트하기" 텍스트가 포함된 Link를 렌더링합니다.
CSS 모듈
- MainHeader.module.css 파일에서 정의된 스타일을 적용하여 헤더와 버튼의 스타일을 지정합니다.
- classes.header, classes.logo, classes.button은 각각 헤더, 로고, 버튼에 적용되는 클래스입니다.
요약
MainHeader 컴포넌트는 애플리케이션의 헤더 역할을 합니다. 주요 기능은 다음과 같습니다:
- 로고: MdMessage 아이콘과 "리액트 포스트" 텍스트로 구성된 링크. 메인 페이지로 이동합니다.
- 포스트하기 버튼: MdPostAdd 아이콘과 "포스트하기" 텍스트로 구성된 링크. 새로운 게시물을 작성하는 페이지로 이동합니다.
이 컴포넌트를 통해 애플리케이션의 헤더 부분을 일관되게 유지하면서, 사용자에게 메인 페이지와 포스트 작성 페이지로 쉽게 접근할 수 있는 네비게이션 기능을 제공합니다.
4.PostsList.tsx
import React from "react"; import Post from "./Post"; import classes from "./PostsList.module.css"; import { useLoaderData } from "react-router-dom"; export interface PostData { id:string; author: string; body: string; } const PostsList: React.FC = () => { // useLoaderData의 반환 타입을 PostData[]로 지정 const posts = useLoaderData() as PostData[]; return ( <> {posts.length > 0 && ( <ul className={classes.posts}> {posts.map((post, index) => ( <Post key={index} id={post.id} author={post.author} body={post.body} /> ))} </ul> )} {posts.length === 0 && ( <div style={{ textAlign: "center", color: "white" }}> <h2>게시글이 없습니다.</h2> <p>포스트를 작성해 주세요!</p> </div> )} </> ); }; export default PostsList;
PostsList 컴포넌트
- PostData 인터페이스: 게시물 데이터의 타입을 정의합니다.
- CSS 모듈: PostsList.module.css 파일에서 정의된 스타일을 classes 객체로 가져옵니다.
- useLoaderData 훅: react-router-dom에서 제공하는 훅으로, 로더가 반환한 데이터를 가져옵니다.
설명
PostData 인터페이스
- 게시물 데이터의 구조를 정의합니다. 각 게시물은 id, author, body 속성을 가집니다.
useLoaderData 훅
- useLoaderData를 사용하여 로더에서 반환된 데이터를 가져옵니다.
- 반환된 데이터를 PostData[] 타입으로 지정합니다.
렌더링
- 게시물이 있을 때 (posts.length > 0):
- 게시물 목록을 <ul> 요소로 감싸서 렌더링합니다.
- 각 게시물은 Post 컴포넌트를 사용하여 렌더링됩니다. key 속성으로는 게시물의 id를 사용해야 하지만, 여기서는 배열의 인덱스를 사용하고 있습니다. 이는 잠재적인 성능 문제를 일으킬 수 있으므로 실제로는 post.id를 사용하는 것이 좋습니다.
- 게시물이 없을 때 (posts.length === 0):
- 중앙 정렬된 메시지를 표시합니다. "게시글이 없습니다."와 "포스트를 작성해 주세요!"라는 메시지를 표시합니다.
- 게시물이 있을 때 (posts.length > 0):
CSS 모듈
- PostsList.module.css 파일에서 정의된 스타일을 적용하여 게시물 목록의 스타일을 지정합니다.
요약
PostsList 컴포넌트는 게시물 목록을 렌더링합니다. 주요 기능은 다음과 같습니다:
- useLoaderData 훅을 사용하여 게시물 데이터를 로드합니다.
- 게시물이 있을 경우, 각 게시물을 Post 컴포넌트로 렌더링하여 목록을 표시합니다.
- 게시물이 없을 경우, "게시글이 없습니다."라는 메시지를 표시합니다.
이 컴포넌트를 통해 애플리케이션에서 게시물 목록을 일관되게 표시하고, 게시물이 없을 때 사용자에게 알림을 제공할 수 있습니다.
5.NewPost.tsx
import React from "react"; import classes from "./NewPost.module.css"; import Modal from "./../components/Modal"; import { Form, Link, redirect, ActionFunction } from "react-router-dom"; import { API_BASE_URL } from "../api-config"; const NewPost: React.FC = () => { return ( <Modal> <Form method="post" className={classes.form}> <p> <label htmlFor="body">내용</label> <textarea id="body" required rows={3} name="body" /> </p> <p> <label htmlFor="name">작성자</label> <input type="text" id="name" required name="author" /> </p> <p className={classes.actions}> <Link to=".." className="button"> 취소 </Link> <button type="submit">전송</button> </p> </Form> </Modal> ); }; export default NewPost; export const action: ActionFunction = async ({ request }) => { const formData = await request.formData(); const postData = Object.fromEntries(formData) as { [key: string]: string }; await fetch(API_BASE_URL + "/posts", { method: "POST", body: JSON.stringify(postData), headers: { "Content-Type": "application/json; charset=utf-8", }, }); return redirect("/"); };
NewPost 컴포넌트
NewPost 컴포넌트는 사용자가 새로운 게시물을 작성할 수 있는 모달 폼을 렌더링합니다.
- CSS 모듈: NewPost.module.css 파일에서 정의된 스타일을 classes 객체로 가져옵니다.
- Modal 컴포넌트: 폼을 모달 형태로 렌더링합니다.
- Form 컴포넌트: react-router-dom의 Form 컴포넌트를 사용하여 폼을 생성합니다. 이 폼은 post 메서드로 제출됩니다.
- 내용 입력 필드: <textarea> 요소로 구성되며, id가 "body"이고 name 속성이 "body"입니다. 필수 입력 필드로 설정되어 있습니다.
- 작성자 입력 필드: <input type="text"> 요소로 구성되며, id가 "name"이고 name 속성이 "author"입니다. 필수 입력 필드로 설정되어 있습니다.
- 액션 버튼: 취소 버튼은 <Link> 요소로 구성되어 폼을 제출하지 않고 이전 페이지로 이동합니다. 제출 버튼은 폼을 제출합니다.
action 함수
action 함수는 폼 제출 시 실행되며, API에 새로운 게시물을 생성하는 요청을 보냅니다.
- 폼 데이터 수집: request.formData()를 사용하여 폼 데이터를 가져옵니다. 그런 다음 Object.fromEntries를 사용하여 폼 데이터를 객체로 변환합니다.
- 게시물 데이터 전송: fetch 함수를 사용하여 API 엔드포인트 (API_BASE_URL + "/posts")에 POST 요청을 보냅니다. 요청 본문은 폼 데이터로 구성되며, Content-Type 헤더를 application/json; charset=utf-8로 설정합니다.
- 리다이렉트: 폼 제출 후, 사용자를 루트 경로 ("/")로 리다이렉트합니다.
요약
이 코드는 새로운 게시물을 작성할 수 있는 모달 폼을 제공하고, 폼이 제출될 때 API로 게시물 데이터를 전송한 후 루트 페이지로 리다이렉트합니다. 주요 기능은 다음과 같습니다:
NewPost 컴포넌트:
- 모달 내에 폼을 렌더링합니다.
- 폼에는 내용과 작성자를 입력할 수 있는 필드가 있습니다.
- 취소 및 제출 버튼이 포함되어 있습니다.
action 함수:
- 폼 데이터를 수집하여 객체로 변환합니다.
- API에 POST 요청을 보내 게시물을 생성합니다.
- 요청이 완료되면 루트 페이지로 리다이렉트합니다.
6.Posts.tsx
import React from "react"; import "../App.css"; import PostsList from "../components/PostsList"; import { Outlet, LoaderFunction } from "react-router-dom"; import { API_BASE_URL } from "../api-config"; const Posts: React.FC = () => { return ( <> <Outlet /> <main> <PostsList /> </main> </> ); }; export default Posts; export const loader: LoaderFunction = async () => { const response = await fetch(`${API_BASE_URL}/posts`); if (!response.ok) { throw new Error('Failed to fetch posts'); } const resData = await response.json(); return resData.posts; };
Posts 컴포넌트
Posts 컴포넌트는 게시물 목록을 렌더링하고, 중첩된 라우트를 처리합니다.
- Outlet: react-router-dom의 Outlet 컴포넌트를 사용하여 중첩된 라우트를 렌더링할 위치를 지정합니다.
- PostsList: PostsList 컴포넌트를 사용하여 게시물 목록을 렌더링합니다.
loader 함수
loader 함수는 게시물 데이터를 비동기적으로 로드합니다.
- API 요청: fetch 함수를 사용하여 API에서 게시물 데이터를 가져옵니다.
- 에러 처리: 요청이 실패할 경우 에러를 발생시킵니다.
- 데이터 반환: JSON 응답에서 게시물 데이터를 추출하여 반환합니다.
설명
Posts 컴포넌트
- <Outlet />: react-router-dom의 Outlet 컴포넌트를 사용하여 중첩된 라우트를 렌더링합니다. 이 위치에 중첩된 라우트 컴포넌트가 렌더링됩니다.
- <PostsList />: 게시물 목록을 렌더링합니다.
loader 함수
- API 요청: fetch를 사용하여 ${API_BASE_URL}/posts 엔드포인트에서 게시물 데이터를 가져옵니다.
- 에러 처리: 응답이 성공적이지 않으면 에러를 발생시킵니다.
- 데이터 반환: JSON 응답에서 게시물 데이터를 추출하여 반환합니다.
요약
Posts 컴포넌트는 게시물 목록을 렌더링하고, Outlet을 사용하여 중첩된 라우트를 처리합니다. loader 함수는 API로부터 게시물 데이터를 비동기적으로 로드하여 반환합니다. 이 설정을 통해 사용자는 애플리케이션 내에서 게시물 목록을 쉽게 조회하고, 새로운 게시물을 작성하는 페이지로 네비게이션할 수 있습니다.
7.Post.tsx
import React from "react"; import classes from "./Post.module.css"; import { Link } from "react-router-dom"; import { PostData } from "./PostsList"; const Post: React.FC<PostData> = ({ id, author, body }) => { return ( <Link to={id} className={classes.post}> <li> <p className={classes.author}>{author}</p> <p className={classes.text}>{body}</p> </li> </Link> ); }; export default Post;
Post 컴포넌트
Post 컴포넌트는 게시물의 정보를 표시하고, 클릭하면 해당 게시물의 상세 페이지로 이동할 수 있는 링크를 제공합니다.
- CSS 모듈: Post.module.css 파일에서 정의된 스타일을 classes 객체로 가져옵니다.
- Link: react-router-dom의 Link 컴포넌트를 사용하여 게시물의 상세 페이지로 이동할 수 있는 링크를 생성합니다.
- PostData 인터페이스: PostsList에서 가져온 PostData 인터페이스를 사용하여 props의 타입을 정의합니다.
설명
CSS 모듈
- Post.module.css 파일에서 정의된 스타일을 classes 객체로 가져와서 적용합니다.
- classes.post는 게시물 전체를 감싸는 링크의 스타일입니다.
- classes.author와 classes.text는 각각 작성자와 본문 텍스트의 스타일입니다.
Link 컴포넌트
- Link 컴포넌트를 사용하여 게시물을 클릭하면 해당 게시물의 id를 URL 경로로 사용하여 상세 페이지로 이동합니다.
- to={id}는 링크의 목적지 URL을 게시물의 id로 설정합니다.
게시물 정보 표시
- <li> 요소 내에 작성자와 본문 텍스트를 각각 <p> 요소로 감싸서 표시합니다.
- classes.author와 classes.text 스타일을 적용하여 작성자와 본문 텍스트의 스타일을 지정합니다.
요약
Post 컴포넌트는 단일 게시물을 렌더링하며, 게시물의 작성자와 본문을 표시하고, 게시물을 클릭하면 해당 게시물의 상세 페이지로 이동할 수 있는 링크를 제공합니다. 이 컴포넌트는 PostData 인터페이스를 사용하여 props의 타입을 정의하며, react-router-dom의 Link 컴포넌트를 사용하여 링크를 생성합니다. 이를 통해 사용자는 게시물 목록에서 개별 게시물을 클릭하여 상세 정보를 확인할 수 있습니다.
8.Modal.tsx
Modal 컴포넌트
Modal 컴포넌트는 모달을 표시하고, 모달 외부 영역을 클릭하면 모달을 닫습니다.
- CSS 모듈: Modal.module.css 파일에서 정의된 스타일을 classes 객체로 가져옵니다.
- useNavigate: react-router-dom의 useNavigate 훅을 사용하여 페이지를 탐색합니다.
- closeHandler 함수: 모달을 닫고 이전 페이지로 돌아가는 함수입니다.
import React from "react"; import classes from "./Modal.module.css"; import { useNavigate } from "react-router-dom"; interface ModalComponent { children?: React.ReactNode; } const Modal: React.FC<ModalComponent> = (props) => { const navigate = useNavigate(); function closeHandler() { navigate(".."); } return ( <> <div className={classes.backdrop} onClick={closeHandler} /> <dialog className={classes.modal}>{props.children}</dialog> </> ); }; export default Modal;
설명
CSS 모듈
- Modal.module.css 파일에서 정의된 스타일을 classes 객체로 가져와서 적용합니다.
- classes.backdrop은 모달의 배경을 어둡게 만드는 스타일입니다.
- classes.modal은 모달 창의 스타일입니다.
useNavigate 훅
- useNavigate 훅을 사용하여 페이지를 탐색합니다. 이 훅은 함수형 컴포넌트 내에서 사용할 수 있습니다.
closeHandler 함수
- 모달 외부 영역을 클릭하면 호출됩니다.
- navigate("..")를 사용하여 현재 경로의 부모 경로로 이동합니다.
렌더링
- 모달 외부 영역: <div> 요소로, classes.backdrop 스타일을 적용하고 클릭 이벤트를 처리합니다.
- 모달 창: <dialog> 요소로, classes.modal 스타일을 적용하고 자식 요소(props.children)를 렌더링합니다.
요약
Modal 컴포넌트는 다음과 같은 기능을 제공합니다:
- 모달 렌더링: 모달 창과 배경을 렌더링합니다.
- 클릭 이벤트 처리: 모달 외부 영역을 클릭하면 모달이 닫히고 이전 페이지로 이동합니다.
- children: 모달 창 내부에 자식 요소를 렌더링합니다.
9.PostDetails.tsx
import { useLoaderData, Link, LoaderFunction, ActionFunction, redirect, Form } from 'react-router-dom'; import Modal from '../components/Modal'; import classes from './PostDetails.module.css'; import { PostData } from '../components/PostsList'; import { API_BASE_URL } from '../api-config'; function PostDetails() { const post = useLoaderData() as PostData; if (!post) { return ( <Modal> <main className={classes.details}> <h1>Could not find post</h1> <p>Unfortunately, the requested post could not be found.</p> <p> <Link to=".." className={classes.btn}> Okay </Link> </p> </main> </Modal> ); } return ( <Modal> <main className={classes.details}> <p className={classes.author}>{post.author}</p> <p className={classes.text}>{post.body}</p> <Form method="post"> <button type="submit" className='button' name="intent" value="delete">삭제</button> </Form> </main> </Modal> ); } export default PostDetails; export const loader: LoaderFunction = async ({ params }) => { const { postId } = params; if (!postId) { throw new Error("Post ID is required"); } const response = await fetch(`${API_BASE_URL}/posts/${postId}`); if (!response.ok) { throw new Error("Failed to fetch post"); } const postData = await response.json(); return postData.post; }; export const action: ActionFunction = async ({ params, request }) => { const formData = await request.formData(); const intent = formData.get("intent"); if (intent === "delete") { const { postId } = params; if (!postId) { throw new Error("Post ID is required"); } const response = await fetch(`${API_BASE_URL}/posts/${postId}`, { method: "DELETE", headers: { "Content-Type": "application/json; charset=utf-8", }, }); if (!response.ok) { throw new Error("Failed to delete post"); } return redirect("/"); } return null; };
소스 :
https://github.com/braverokmc79/react-crash-course-typescript/tree/react-router-dom
댓글 ( 0)
댓글 남기기