Skip to content

Latest commit

 

History

History
222 lines (157 loc) · 7.09 KB

README.md

File metadata and controls

222 lines (157 loc) · 7.09 KB

Second Hand Max FE

Table of Contents

Tech Stack

Vite Styled Components TypeScript React React Router Axios React Query

Members

Bakha Kakamotobi
Bakha Kakamotobi

BE Repositories

Dev Log

범용 컴포넌트

SelectInput

SelectInput 컴포넌트 #2

Dropdown

Dropdown 컴포넌트 #6

Hooks

useViewportIntersection

V1
  • Viewport을 고려하여 위치를 동적으로 변경해야할 수 있는 경우(Ex: SelectInputDropdown)를 위해 viewport의 "left"/"right"과의 중첩 여부를 알려줌.
  • IntersectionObserver API를 사용.
V2
  • useViewportIntersection -> useIntersection
  • Default로 viewport를 기준으로 intersection을 확인하지만 별도로 boundaryElement을 설정할 수 있도록 리팩토링함.

refetch vs invalidateQueries

  • refetch는 일단 실행되면 바로 query를 refetch 함.
  • invalidateQueries는 해당 query를 stale로 마킹을 하고, 후에 해당 query를 observe하는 컴포넌트가 마운트 될 때 refetch 함.

    invalidation is a more "smart" refetching. | TkDodo

  • Reference: TanStack/query#2468

Route Protection

  • react-router-dom의 Navigate은 내부적으로 useEffect를 사용하고 있음.
  • Route protection을 구현할 때 Navigate을 사용하지 않고 따로 useEffect로 직접해도 렌더링 횟수는 동일.

Auth Token Flow

  • Access Token 만료는 서버와 403으로 지정.
  • Axios response interceptor를 활용하여 403을 받으면 refresh token으로 access token refresh 시도.
fetcher.interceptors.response.use(
  (res) => res,
  async (error) => {
    const originalRequest = error.config;
    // If the access token expired, attempt to refresh it.
    if (error.response.status === 403 && !originalRequest._retry) {
      originalRequest._retry = true;

      try {
        const res = await refreshAccessToken();
        localStorage.setItem("accessToken", res.data?.accessToken);
        return fetcher(originalRequest);
      } catch (refreshError) {
        // Refresh token expired.
        localStorage.removeItem("accessToken");
        localStorage.removeItem("refreshToken");

        // TODO: navigate the user to `/signin` while displaying the toast
        window.location.href = Routes.SIGNIN;
        return Promise.reject(refreshError);
      }
    }

    return Promise.reject(error);
  }
);

Query Error Handling

V1

  • QueryCache에서 일괄적으로 모든 query의 에러를 받아서 server에서 받은 에러 메시지를 toast에 담아서 보여줌.
  • 문제
    • Route protection을 사용자 정보 (nickname, profile image)의 유무에 따라서 적용하고 있기 때문에 기본적으로 사이트 접속시 사용자 정보를 받기 위한 요청을 보냄.
      • 즉, 로그인이 안된 사용자(i.e. localStorage에 유효한 access token이 없음)가 처음 사이트 접속시 error toast 보여짐.
    • 예상치 못한 다른 에러 또한 보여짐.

V2

  • 동일하게 QueryCache에서 일괄적으로 error를 핸들링하고, 특정 상황에서는 query.meta.errorMessage를 활용.
    • Ex: useUserInfoQuery의 "AccessToken이 유효하지 않습니다" -> "로그인을 먼저 해주세요".

비동기로 받아온 데이터로 초기값 복사 및 state 초기화

  • EditProductItemPage
    • 비동기 데이터를 이용한 input 초기값 설정을 useEffect를 사용하고 있음.

TypeScript Type System

  • TypeScript is a structural type system.
    • Structural Type System: when comparing types, only the members on the type is taken into account.
      • i.e. TS checks whether an object type has at least the required properties and their corresponding types, rather than an exact match of the properties.
    • cf. Nominal Type System
      • Two types that have been created cannot be assigned to each other.

Example 1

TypeScript (Structural Typing)
type CategoryType = {
  id: number;
  title: string;
  imageUrl: string;
};

type CategoryTag = {
  id: number;
  title: string;
};

const arr: CategoryTag[] = [];

const categoryA: CategoryType = {
  id: 1,
  title: "a",
  imageUrl: "blah",
};

arr.push(categoryA); // No Error
Swift (Nominal Typing)
struct CategoryType {
    var id: Int
    var title: String
    var imageUrl: String
}

struct CategoryTag {
    var id: Int
    var title: String
}

var arr: [CategoryTag] = []

let categoryA = CategoryType(
    id: 1,
    title: "a",
    imageUrl: "blah"
)

// PS: Swift throws an error even if `CategoryType` and `CategoryTag` have the same properties.
arr.append(categoryA) // Error

Example 2

TypeScript (Structural Typing)
type CategoryType = {
  id: number;
  title: string;
};

type CategoryTag = {
  id: number;
  title: string;
  imageUrl: string;
};

const arr: CategoryTag[] = [];

const categoryA: CategoryType = {
  id: 1,
  title: "a",
};

arr.push(categoryA); // Error

URL.createObjectURL()

https://developer.mozilla.org/en-US/docs/Web/API/URL/createObjectURL_static#memory_management

z-index 상수화

const base = 0;
const above = 1;
const below = -1;

export const zAppBar = base + above;
export const zDropdown = zAppBar + above;
export const zModal = zDropdown + above;
export const zAlert = zModal + above;

Reference: https://www.smashingmagazine.com/2021/02/css-z-index-large-projects/