Skip to content

Commit

Permalink
keyword: 5주차 완료
Browse files Browse the repository at this point in the history
  • Loading branch information
gaaahee committed Nov 7, 2024
1 parent b177e69 commit 7dc9b8c
Show file tree
Hide file tree
Showing 21 changed files with 416 additions and 132 deletions.
257 changes: 257 additions & 0 deletions keyword/chapter05/keyword.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,257 @@
### 키워드 정리 🍠

- useRef 🍠

`useRef`는 컴포넌트 내에서 DOM 요소나 특정 값을 저장하고 관리할 때 사용됨

1. **DOM 요소 접근**: 예를 들어, 특정 `<input>` 필드에 포커스를 주고 싶을 때 `useRef`로 해당 요소에 접근할 수 있다.
- **폼 초기 포커스**: 페이지 로드 시 자동으로 첫 번째 필드에 포커스를 주고 싶을 때
- **유효성 검사 실패 시**: 입력값이 잘못된 경우, 해당 필드로 포커스를 이동해 사용자에게 쉽게 위치를 알려줄 때
- **사용자 경험 개선**: 사용자가 입력할 때마다 필요한 필드로 자연스럽게 포커스를 이동시키고 싶을 때
2. **값 저장 (리렌더링 없이 유지)**: useRef는 리렌더링 없이 값을 저장한다. `useState` 또한 값을 저장할 수 있지만, 값이 변화하면 리렌더링이 발생한다. `useRef` 는 UI에 반영하지 않고 값을 저장하거나 유지해야 할 때, `useState`는 UI에 실시간으로 반영해야 되는 값을 관리할 때 사용하기 적절하다.
3. **타이머나 애니메이션**: `setTimeout`, `setInterval` 같은 타이머 ID나 애니메이션 프레임을 참조할 때에도 `useRef`를 사용해 관리할 수 있습니다.
- input의 주요 프로퍼티 🍠
- 아래 내용 이외에, 자주 사용하는 프로퍼티가 있으면 추가로 더 정리해주세요! 🍠

### 1. `type`

- **설명**: 입력 필드의 종류를 설정합니다.
- **값 예시**: `"text"`, `"password"`, `"email"`, `"number"`, `"checkbox"`, `"radio"`
- **예시**: `<input type="text" />`

### 2. `value`

- **설명**: 입력 필드의 값을 설정하고 제어할 때 사용됩니다. **Controlled Component**에서 자주 사용됩니다.
- **값 예시**: 문자열 또는 숫자
- **예시**: `<input type="text" value={value} />`

### 3. `defaultValue`

- **설명**: 초기값을 설정하는 데 사용됩니다.
- **값 예시**: 문자열 또는 숫자
- **예시**: `<input type="text" defaultValue="초기값" />`

### 4. `onChange`

- **설명**: 사용자가 입력 필드에 값을 입력하거나 변경할 때 호출되는 이벤트 핸들러입니다.
- **값 예시**: 함수
- **예시**: `<input type="text" onChange={(e) => setValue(e.target.value)} />`

### 5. `placeholder`

- **설명**: 입력 필드가 비어 있을 때 표시되는 힌트 텍스트입니다.
- **값 예시**: 문자열
- **예시**: `<input type="text" placeholder="여기에 입력하세요" />`

### 6. `checked`

- **설명**: 체크박스나 라디오 버튼이 선택되었는지 여부를 제어합니다.
- **값 예시**: `true` 또는 `false`
- **예시**: `<input type="checkbox" checked={isChecked} />`

### 7. `defaultChecked`

- **설명**: 체크박스나 라디오 버튼의 초기 상태를 설정합니다.
- **값 예시**: `true` 또는 `false`
- **예시**: `<input type="checkbox" defaultChecked />`

### 8. `disabled`

- **설명**: 입력 필드를 비활성화하여 사용자 입력을 막습니다.
- **값 예시**: `true` 또는 `false`
- **예시**: `<input type="text" disabled />`

### 9. `readOnly`

- **설명**: 입력 필드의 값을 읽기 전용으로 설정합니다. 사용자는 값을 변경할 수 없습니다.
- **값 예시**: `true` 또는 `false`
- **예시**: `<input type="text" readOnly />`

### 10. `name`

- **설명**: 폼 데이터를 제출할 때 서버로 전송되는 데이터의 이름을 지정합니다.
- **값 예시**: 문자열
- **예시**: `<input type="text" name="username" />`

### 11. `maxLength`

- **설명**: 입력할 수 있는 최대 글자 수를 지정합니다.
- **값 예시**: 숫자
- **예시**: `<input type="text" maxLength={10} />`

### 12. `min` / `max`

- **설명**: 숫자 또는 날짜 입력에서 사용할 수 있는 최소/최대 값을 지정합니다.
- **값 예시**: 숫자 또는 날짜
- **예시**: `<input type="number" min={1} max={10} />`

### 13. `autoFocus`

- **설명**: 페이지가 로드될 때 자동으로 입력 필드에 포커스를 줍니다.
- **값 예시**: `true` 또는 `false`
- **예시**: `<input type="text" autoFocus />`

### 14. `required`

- **설명**: 입력 필드를 필수 입력으로 설정합니다. 폼을 제출할 때 이 필드가 비어 있으면 제출이 거부됩니다.
- **값 예시**: `true` 또는 `false`
- **예시**: `<input type="text" required />`

### 15. **`size`**

- **설명**: 텍스트 입력의 표시되는 크기를 지정합니다. 실제 입력 길이를 제한하지는 않지만, 사용자에게 보이는 텍스트 필드의 길이를 조절할 수 있습니다.
- **값 예시**: 숫자 (필드의 문자 길이)
- **예시**: `<input type="text" size={20} />`

### 16. `pattern`

- **설명**: 정규 표현식으로 입력값의 형식을 제한할 수 있습니다. 예를 들어, 전화번호 형식 또는 이메일 형식만 허용하고 싶을 때 유용합니다.
- **값 예시**: 정규 표현식 (`"^\d{10}$"` 등)
- **예시**: `<input type="text" pattern="^\d{10}$" title="10자리 숫자를 입력하세요" />`

### **`Controlled Input`** vs **`UnControlled Input`**

- Controlled Input

### Controlled Input

**`Controlled Input`**은 React의 상태(state)에 의해 폼의 값을 관리하는 방식입니다. Input의 value가 컴포넌트의 상태에 연결되어 있고, 상태값이 달라질 떄 마다, 입력 필드의 값도 갱신됩니다.

쉽게 말해서, Input의 값은, **`React Component가 제어`**하고, 모든 입력 변화 또한, **`컴포넌트의 상태로 반영`**되어집니다.

특징

- Input의 value는, React의 **`state`**로 관리됩니다.
- 상태 값이, **`Input의 value 속성에 직접적으로 연결`** 됨.
- **`Input`****`Value`****`변경`**되면, **`상태가 업데이트`**되고, **`상태가 다시 렌더링을 Trigger`** 함.
- **`React`****`Form 요소의 현재 값을 항상 알고 있기에 제어가 용이`**.

```tsx
import { useState } from 'react';

function ControlledInput() {
const [inputValue, setInputValue] = useState('');

const handleChange = (event) => {
setInputValue(event.target.value);
};

return (
<div>
<input type="text" value={inputValue} onChange={handleChange} />
<p>입력: {inputValue}</p>
</div>
);
}

export default ControlledInput;

```

### 장점:

- 상태를 통해 값이 직접적으로 제어되기 때문에 **`폼 데이터를 검증`**하거나 **`조작`**하기가 쉽습니다.
- **`사용자 입력을 실시간으로 검증`**하거나 **`포맷을 조정`**할 수 있습니다.

### 단점:

- **`컴포넌트에서 상태 관리가 복잡`**해질 수 있으며, 특히 폼 데이터에 너무 많은 value들을 관리하는 경우 **`성능에 부담`**이 될 수 있습니다.

- UnControlled Input

### UnControlled Input

**`Uncontrolled Input`**Reactstate가 아닌 DOM 자체에서 입력 값을 관리하는 방식입니다. 폼의 값은 **`React 컴포넌트가 직접 관리하지 않고`**, 필요할 때 DOM에서 직접 값을 참조하는 방법입니다. 이를 위해 `ref`를 사용하여 **`DOM 요소에 직접 접근`**할 수 있습니다.

### 특징

- 입력값이 `state`에 의존하지 않고, React의 제어 밖에서 관리됩니다.
- 폼 요소의 값은 사용자가 입력하는 대로 DOM에 저장되며, 필요할 때만 값을 읽어 옵니다.
- React의 상태를 사용하지 않기 때문에 폼을 간단하게 유지할 수 있습니다.

```tsx
import { useRef } from 'react';
function UncontrolledInput() {
const inputRef = useRef(null);
const handleSubmit = () => {
console.log(`입력: ${inputRef.current.value}`);
};
return (
<div>
<input type="text" ref={inputRef} />
<button onClick={handleSubmit}>제출</button>
</div>
);
}
export default UncontrolledInput;
```

### 장점

- 상태 관리가 필요 없으므로 간단한 폼에 적합하며, **`성능적으로도 유리`**할 수 있습니다.
- 작은 폼이나 **`상태가 필요 없는 경우 사용하기 쉽습니다.`**

### 단점

- 입력 값을 **`실시간으로 검증하거나 조작하기 어려우며`**, 폼 데이터와 관련된 논리를 관리하는 데 제약이 있을 수 있습니다.
- **`DOM에 직접 접근`**하기 때문에 **`React의 단방향 데이터 흐름을 벗어나는 경우`**가 생길 수 있습니다.


### react-hook-form & yup 🍠

**`react-hook-form`****`yup`**React에서 폼을 간편하게 관리하고, 유효성 검사를 수행하는 데 유용한 라이브러리 입니다. 두 개의 라이브러리를 함께 사용하면, 폼 입력 관리와 검증 과정을 매우 효율적으로 처리할 수 있습니다.

현재, 여기서는 **`yup`**을 통한, **`validation`**을 설명하지만, 혹시라도, 본인이 작업하시는 프로젝트가 **`TypeScript`**시라면, **`zod`**를 사용하는 것이 조금 더 유리할 수도 있으니, **`zod`**를 더욱 추천드립니다!

- **`react-hook-form`** 공식문서 설명

### react-hook-form

https://react-hook-form.com/get-started

주요 특징

1. DX (개발자 경험)

직관적이고, 기능이 완벽한 API로, 개발자가 폼을 구축할 떄 매끄러운 경험을 제공합니다.

2. HTML 표준

기존의 HTML 마크업을 활용하여 제약 기반 검증 API로 폼을 검증합니다.

3. 가벼움

패키지 크기는 매우 중요합니다. React Hook Form은 의존성이 없는 작은 라이브러리이다.

4. 성능

리렌더링 횟수를 최소화하고, 검증 계산을 줄이며, 더 빠른 마운팅을 제공.

5. 채택 가능성

폼 상태는 본질적으로 로컬에 있기 떄문에 다른 의존성 없이도 쉽게 채택 가능하다.

6. UX (사용자 경험)

최상의 사용자 경험을 제공하기 위해 일관된 검증 전략을 지향.

- **`yup`** 공식문서 설명

### yup


**`Yup`**은 런타임 값 파싱 및 검증을 위한 스키마 빌더입니다. 스키마를 정의하고, 값을 변환하여 일치시키거나, 기존 값의 형태를 검증하거나, 두 작업을 모두 수행할 수 있습니다. **`Yup 스키마`**는 매우 표현력이 뛰어나며 복잡하고 상호 의존적인 검증이나 값 변환을 모델링할 수 있습니다.

주요 기능:

1. 간결하면서도 표현력이 뛰어난 스키마 인터페이스로, 간단한 데이터 모델부터 복잡한 데이터 모델까지 설계 가능
2. 강력한 TypeScript 지원: 스키마에서 정적 타입을 추론하거나, 스키마가 올바르게 타입을 구현하는지 확인
3. 내장된 비동기 검증 지원: 서버 측 및 클라이언트 측 검증을 동일하게 모델링 가능
4. 확장성: 타입 안정성이 보장된 메서드와 스키마를 추가할 수 있음
5. 풍부한 오류 세부 정보 제공으로 디버깅이 쉬움
28 changes: 14 additions & 14 deletions mission/chapter04/mission_ch04/src/App.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,40 +2,40 @@ import React from 'react';
import { createBrowserRouter, RouterProvider } from "react-router-dom";
import RootLayout from "./layout/root-layout";
import HomePage from "./pages/Default/home";
import Login from "./pages/Default/login";
import Signup from "./pages/Default/signup";
import Search from './pages/Default/search';
import CategoryList from "./pages/CategoryList/CategoryList";
import MovieList from "./pages/MovieList/MovieList"; // 영화 목록 페이지
import LoginPage from "./pages/Default/login";
import SignupPage from "./pages/Default/signup";
import SearchPage from './pages/Default/search';
import CategoryListPage from "./pages/CategoryList/CategoryList";
import MovieRoute from "./pages/MoviePages/MovieRoute";

const router = createBrowserRouter([
{
path: '/',
element: <RootLayout />,
element: <RootLayout/>,
children: [
{
index: true,
element: <HomePage />
element: <HomePage/>
},
{
path: 'login',
element: <Login />
element: <LoginPage/>
},
{
path: 'signup',
element: <Signup />
element: <SignupPage/>
},
{
path: 'search',
element: <Search />
element: <SearchPage/>
},
{
path: 'movies',
element: <CategoryList />
element: <CategoryListPage/>
},
{
path: 'movies/:movieId',
element: <MovieList />
path: 'movies/:id',
element: <MovieRoute/>
}
]
}
Expand All @@ -45,4 +45,4 @@ function App() {
return <RouterProvider router={router} />;
}

export default App;
export default App;
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,8 @@ const CastCard = (props) =>{
<>
<div className="cast_card_box">
<div className="profile_box">
<img className='profile_img' src={`https://image.tmdb.org/t/p/w500${cast.profile_path}`}/></div>
<img className='profile_img'
src={`https://image.tmdb.org/t/p/w500${cast.profile_path}`}/></div>
<span>{cast.name}</span>
<p>{cast.character}</p>
</div>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import styled from "styled-components";
import React from 'react';
import {CardDiv, CardSpan} from './CategoryCard.style';

const CategoryCard = ({title})=>{
return(
Expand All @@ -9,26 +10,4 @@ const CategoryCard = ({title})=>{

}

export default CategoryCard;

const CardDiv = styled.div`
width : 200px;
height : 100px;
background-color : white;
border-radius: 10px;
display : flex;
align-items : flex-end;
&:hover{
transform: scale(1.05);
box-shadow: 0 10px 20px rgba(0, 0, 0, 0.5);
}
`
const CardSpan = styled.span`
background-color: grey;
color: white;
opacity: 0.5;
border-radius: 5px;
margin :10px 10px 10px 10px;
padding : 0px 3px 3px 3px;
vertical-align : bottom;
`
export default CategoryCard;
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import styled from "styled-components";

export const CardDiv = styled.div`
width : 200px;
height : 100px;
background-color : white;
border-radius: 10px;
display : flex;
align-items : flex-end;
&:hover{
transform: scale(1.05);
box-shadow: 0 10px 20px rgba(0, 0, 0, 0.5);
}
`
export const CardSpan = styled.span`
background-color: grey;
color: white;
opacity: 0.5;
border-radius: 5px;
margin :10px 10px 10px 10px;
padding : 0px 3px 3px 3px;
vertical-align : bottom;
`
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import React from "react";
import { NavbarContainer, Logo, NavLinks, NavButton } from "./navbar.styled";
import { NavbarContainer, Logo, NavLinks, NavButton } from "./navbar.style";

const Navbar = () => {
return (
Expand Down
Loading

0 comments on commit 7dc9b8c

Please sign in to comment.