-
Notifications
You must be signed in to change notification settings - Fork 2
[재하] 1205(화) 개발기록
어드민 페이지 Full Stack, 전체 글 조회기능 구현
- emotion 설치
- Button, Table 컴포넌트 생성
- 전체 글 조회 API 구현
- Board 컴포넌트 생성, 전체 글 조회
- 트러블 슈팅: Vite dev/prod 환경변수 설정
- 테스트 (동작 화면)
yarn workspace admin add @emotion/styled @emotion/react
프론트 프로젝트와 의존성을 동일하게 가기 위해 @emotion/styled, @emotion/react를 설치해준다.
버튼, 테이블 등 공유해서 사용할 수 있는 베이스 컴포넌트들을 생성해준다. 학습메모 2를 참고하여 프론트와 동일한 코드 컨벤션으로 구현을 해보았다.
// Button.tsx
import styled from '@emotion/styled';
interface PropsType extends React.ButtonHTMLAttributes<HTMLButtonElement> {
onClick?: () => void;
children: React.ReactNode;
}
export default function Button({ children, ...args }: PropsType) {
return <CustomButton {...args}>{children}</CustomButton>;
}
const CustomButton = styled.button<PropsType>`
background-color: #fff;
border: 1px solid #ddd;
border-radius: 4px;
color: #212121;
font-size: 14px;
font-weight: 600;
padding: 6px 12px;
cursor: pointer;
outline: none;
&:hover {
background-color: #ddd;
}
`;
// Table.tsx
import styled from '@emotion/styled';
interface PropsType extends React.TableHTMLAttributes<HTMLTableElement> {
children: React.ReactNode;
}
export default function Table({ children, ...args }: PropsType) {
return <CustomTable {...args}>{children}</CustomTable>;
}
const CustomTable = styled.table<PropsType>`
border-collapse: collapse;
width: 100%;
border: 1px solid #ddd;
`;
스타일은 추후에 또 적당히 수정해보자.
백엔드에 전체 글 조회 API가 없기 때문에 만들어둔 admin 모듈에 새로운 API를 만들어준다.
// admin.controller.ts
import { Controller, Get } from '@nestjs/common';
import { AdminService } from './admin.service';
@Controller('admin')
export class AdminController {
constructor(private readonly adminService: AdminService) {}
@Get('post')
getAllPosts() {
return this.adminService.getAllPosts();
}
}
// admin.service.ts
import { Injectable } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm';
import { User } from 'src/auth/entities/user.entity';
import { Board } from 'src/board/entities/board.entity';
import { Repository } from 'typeorm';
@Injectable()
export class AdminService {
constructor(
@InjectRepository(User)
private readonly userRepository: Repository<User>,
@InjectRepository(Board)
private readonly boardRepository: Repository<Board>,
) {}
async getAllPosts() {
const posts = await this.boardRepository.find();
return posts;
}
}
board 메뉴에 보여줄 페이지를 컴포넌트로 만들어준다. 마찬가지로 학습메모 5, 프론트 분들의 코드를 참고하여 동일한 컨벤션으로 구현 시도.
import { useState } from 'react';
import Button from '../shared/Button';
import Table from '../shared/Table';
const baseUrl = import.meta.env.VITE_API_BASE_URL;
export default function Board() {
const [boardList, setBoardList] = useState([]);
const getBoardList = async () => {
const response = await fetch(baseUrl + '/admin/post');
const data = await response.json();
setBoardList(data);
};
return (
<div>
<Button onClick={getBoardList}>게시글 불러오기</Button>
<Table>
<thead>
<tr>
<th>번호</th>
<th>제목</th>
<th>작성자</th>
<th>좋아요</th>
<th>이미지 수</th>
<th>작성일시</th>
<th>수정일시</th>
</tr>
</thead>
<tbody>
{boardList.map((board: any) => (
<tr key={board.id}>
<td>{board.id}</td>
<td>{board.title}</td>
<td>{board.user.nickname}</td>
<td>{board.like_cnt}</td>
<td>{board.images.length}</td>
<td>{board.created_at}</td>
<td>{board.updated_at}</td>
</tr>
))}
</tbody>
</Table>
</div>
);
}
완성된 페이지는 라우트와 Nav바에 등록해줘야 한다.
Nav바는 아래와 같이 리팩토링해서 App.tsx에서 메뉴를 한번에 편집이 가능하도록 변경했다.
// Nav.tsx
import styled from '@emotion/styled';
interface PropsType extends React.HTMLAttributes<HTMLDivElement> {
children: React.ReactNode;
}
export default function Nav({ children, ...args }: PropsType) {
return <CustomNav {...args}>{children}</CustomNav>;
}
// 최상단에 위치하도록 설정
const CustomNav = styled.nav<PropsType>`
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 20px;
display: flex;
align-items: center;
justify-content: space-around;
background-color: #fff;
border-bottom: 1px solid #ddd;
padding: 12px 12px;
& > a {
margin-right: 16px;
text-decoration: none;
color: #212121;
font-size: 14px;
font-weight: 600;
}
& > a:hover {
text-decoration: underline;
}
`;
이제 최종적으로 board 페이지 등록.
// App.tsx
import { Route, Routes } from 'react-router-dom';
import './App.css';
import { TestComponent } from './components/TestComponent/TestComponent.tsx';
import Board from './components/Board/Board.tsx';
import Nav from './components/Nav.tsx';
function App() {
return (
<>
<Nav>
<a href="/">Home</a>
<a href="/about">About</a>
<a href="/abc">Test</a>
<a href="/board">Board</a>
</Nav>
<Routes>
<Route path="/" element={<div>Home</div>} />
<Route path="/about" element={<div>About</div>} />
<Route path="/abc" element={<TestComponent />} />
<Route path="/board" element={<Board />} />
</Routes>
</>
);
}
export default App;
board 페이지에서 게시글 전체 조회를 위한 fetch 시
baseUrl을 개발 환경에서는 http://localhost:3000
,
배포 환경에서는 https://www.별글.site/api
로 해줘야 하는데,
기존에 알고 있던 REACT 환경변수가 먹히지 않아 애를 좀 먹었다. 결국 본 프로젝트에서는 Vite를 이용하여 dev서버 실행과 build를 진행하므로, Vite에서 제공하는 환경변수 import 형태를 준수해야 했다.
학습메모 4를 참고하여 admin workspace 루트인 /packages/admin/
에
.env.development
, .env.production
두 파일을 추가해준다.
VITE_API_BASE_URL=http://localhost:3000
VITE_API_BASE_URL=https://www.별글.site/api
불러올 땐 아래와 같이 불러오면 된다.
const baseUrl = import.meta.env.VITE_API_BASE_URL;
yarn workspace admin dev
이제 dev서버에서 요청하면 http://localhost:3000
로 잘 가고
yarn workspace admin build
빌드해보면 https://www.별글.site/api
이 잘 등록되어 있음을 확인할 수 있다.
전체적인 실행 화면은 위와 같다.
© 2023 debussysanjang
- 🐙 [가은] Three.js와의 설레는 첫만남
- 🐙 [가은] JS로 자전과 공전을 구현할 수 있다고?
- ⚽️ [준섭] NestJS 강의 정리본
- 🐧 [동민] R3F Material 간단 정리
- 👾 [재하] 만들면서 배우는 NestJS 기초
- 👾 [재하] GitHub Actions을 이용한 자동 배포
- ⚽️ [준섭] 테스트 코드 작성 이유
- ⚽️ [준섭] TypeScript의 type? interface?
- 🐙 [가은] 우리 팀이 Zustand를 쓰는 이유
- 👾 [재하] NestJS, TDD로 개발하기
- 👾 [재하] AWS와 NCP의 주요 서비스
- 🐰 [백범] Emotion 선택시 고려사항
- 🐧 [동민] Yarn berry로 모노레포 구성하기
- 🐧 [동민] Vite, 왜 쓰는거지?
- ⚽️ [준섭] 동시성 제어
- 👾 [재하] NestJS에 Swagger 적용하기
- 🐙 [가은] 너와의 추억을 우주의 별로 띄울게
- 🐧 [동민] React로 멋진 3D 은하 만들기(feat. R3F)
- ⚽️ [준섭] NGINX 설정
- 👾 [재하] Transaction (트랜잭션)
- 👾 [재하] SSH 보안: Key Forwarding, Tunneling, 포트 변경
- ⚽️ [준섭] MySQL의 검색 - LIKE, FULLTEXT SEARCH(전문검색)
- 👾 [재하] Kubernetes 기초(minikube), docker image 최적화(멀티스테이징)
- 👾 [재하] NestJS, 유닛 테스트 각종 mocking, e2e 테스트 폼데이터 및 파일첨부
- 2주차(화) - git, monorepo, yarn berry, TDD
- 2주차(수) - TDD, e2e 테스트
- 2주차(목) - git merge, TDD
- 2주차(일) - NCP 배포환경 구성, MySQL, nginx, docker, docker-compose
- 3주차(화) - Redis, Multer 파일 업로드, Validation
- 3주차(수) - AES 암복호화, TypeORM Entity Relation
- 3주차(목) - NCP Object Storage, HTTPS, GitHub Actions
- 3주차(토) - Sharp(이미지 최적화)
- 3주차(일) - MongoDB
- 4주차(화) - 플랫폼 종속성 문제 해결(Sharp), 쿼리 최적화
- 4주차(수) - 코드 개선, 트랜잭션 제어
- 4주차(목) - 트랜잭션 제어
- 4주차(일) - docker 이미지 최적화
- 5주차(화) - 어드민 페이지(전체 글, 시스템 정보)
- 5주차(목) - 감정분석 API, e2e 테스트
- 5주차(토) - 유닛 테스트(+ mocking), e2e 테스트(+ 파일 첨부)
- 6주차(화) - ERD
- 2주차(화) - auth, board 모듈 생성 및 테스트 코드 환경 설정
- 2주차(목) - Board, Auth 테스트 코드 작성 및 API 완성
- 3주차(월) - Redis 연결 후 RedisRepository 작성
- 3주차(화) - SignUpUserDto에 ClassValidator 적용
- 3주차(화) - SignIn시 RefreshToken 발급 및 Redis에 저장
- 3주차(화) - 커스텀 AuthGuard 작성
- 3주차(수) - SignOut시 토큰 제거
- 3주차(수) - 깃헙 로그인 구현
- 3주차(토) - OAuth 코드 통합 및 재사용
- 4주차(수) - NestJS + TypeORM으로 MySQL 전문검색 구현
- 4주차(목) - NestJS Interceptor와 로거
- [전체] 10/12(목)
- [전체] 10/15(일)
- [전체] 10/30(월)
- [FE] 11/01(수)~11/03(금)
- [전체] 11/06(월)
- [전체] 11/07(화)
- [전체] 11/09(목)
- [전체] 11/11(토)
- [전체] 11/13(월)
- [BE] 11/14(화)
- [BE] 11/15(수)
- [FE] 11/16(목)
- [FE] 11/19(일)
- [BE] 11/19(일)
- [FE] 11/20(월)
- [BE] 11/20(월)
- [BE] 11/27(월)
- [FE] 12/04(월)
- [BE] 12/04(월)
- [FE] 12/09(금)
- [전체] 12/10(일)
- [FE] 12/11(월)
- [전체] 12/11(월)
- [전체] 12/12(화)