Typescript - useRef 읽기 전용 에러

배준형·2022년 3월 22일
6
post-thumbnail

1. useRef는 읽기 전용?

  • useRef는 DOM에 연결하여 해당 Element를 읽어오거나, .current 프로퍼티에 값을 저장해놓고 해당 값이 변동되어도 렌더링하고 싶지 않을 때 사용한다.
  • 만약 DOM에 연결하거나 동적인 데이터 값을 .current 값에 할당하여 사용할 때 초기엔 의미적으로 값이 없음을 나타내기 위해 초기 값을 null로 작성해야하지 않을까? 생각했었다.
  • 다만, useRef를 사용할 때 타입 지정을 해놓고 사용하려고 하니 .current 프로퍼티에 값을 할당하는데 읽기 전용이라고 에러가 뜨는 것이다.

2. MutableRefObject | RefObject

  • useRef에 마우스 커서를 올려 놓으면 MutableRefObject 이라는 타입을 반환하고 있다.
  • MutableRefObject 타입이 무엇인지 들어가보자 (ctrl + 우클릭)
interface MutableRefObject<T> {
    current: T;
}
  • useRef에서의 .current 프로퍼티는 제네릭으로 표현되어 있다. 그런데 왜 읽기 전용일까?

  • useRef에서 최초에는 null 값을 주고 컴포넌트가 마운트되면 ref를 연결하거나 ref.current에 값을 할당하는 방식으로 사용하므로 useRef에 Type을 지정해야 한다.

    • 그런데, 타입을 지정하니 useRef의 타입이 아래와 같이 바뀌었다.

  • 그럼 RefObject는 뭘까? 들어가보자
interface RefObject<T> {
    readonly current: T | null;
}
  • RefObject는 null 이거나 T 타입의 제네릭인데 readonly 처리 되어있다.
  • 왜 다르게 동작하는 것일까?

3. 파라미터의 Type이 설정한 Type과 같은지 확인?

  • useRef의 interface를 확인해보자
function useRef<T>(initialValue: T): MutableRefObject<T>;

function useRef<T>(initialValue: T|null): RefObject<T>;
  • useRef에 대해 위에서 확인한 두 가지 interface가 작성되어 있다.
  • 위 내용에 따르면 initialValue에 null 값을 할당하면 RefObject의 interface를 가진 useRef를 사용하게 되는 것이고, .current 프로퍼티는 읽기 전용이 되는 것이다.

4. 어떻게 사용해야 할까?

  • 그럼 어떻게 사용해야 할까?
  • useRef의 interface를 더 살펴보니 아래와 같은 interface도 존재했다.
function useRef<T = undefined>(): MutableRefObject<T | undefined>;
  • 해당 interface는 초기 값이 undefined일 경우 MutableRefObject interface가 지정되어 있다.
  • 즉, 초기 값을 null이 아닌 undefined라고 작성하면 읽기 전용이 아닌 값을 할당하여 사용할 수 있게 된다.

  • 수정 전
const ref = useRef<number>(null);
ref.current = 10; // Error. .current는 읽기 전용
  • 수정 후
const ref = useRef<number>();
ref.current = 10; // 정상 동작

5. DOM Element에 ref Prop으로 넘겨줄 때

  • 위 사항에 대해서는 이해가 됐지만, .current를 값으로 사용하는 것이 아닌 DOM Element의 ref에 할당할 때 undefined 값을 초기값으로 넣으니 또 다른 에러가 발생했다.
  • T | undefined 타입은 T | null 타입에 할당할 수 없다고 뜬다.
  • 마찬가지로 Element에서 prop으로 받을 수 있는 ref를 타고 들어가보자.
interface ClassAttributes<T> extends Attributes {
    ref?: LegacyRef<T> | undefined;
}
type LegacyRef<T> = string | Ref<T>;
type Ref<T> = RefCallback<T> | RefObject<T> | null;
type RefCallback<T> = (instance: T | null) => void;
  • ref는 LegacyRef<T> 타입의 prop을 넘겨줄 수 있다고 나와있는데 하나씩 순서대로 들어가보면
    • LegacyRef는 string이나 Ref 타입
    • Ref는 RefCallback이나 RefObject이나 null 타입 이어야 한다.

  • 결국 ref는 RefCallback | RefObject | string | null | undefined 중 하나여야 한다.
  • 여기서 RefCallback은 Ref 형태가 아닌 Callback 함수에 대한 정의인 것처럼 보이므로 제외하면 남는 것은 RefObject인데, 어디서 많이 본 형태다.

interface RefObject<T> {
    readonly current: T | null;
}
  • 아까 초기 값에 null 값을 넣어서 읽기 전용이 되어 .current 프로퍼티에 값을 할당할 수 없게 되어 MutableRefObject 형태로 사용하기 위해 undefined 값으로 초기값을 넣어줬었다.
  • 그런데 ref prop으로 넘겨주려는 데이터의 형태는 RefObject 형태여야 하고, 이는 초기값으로 null을 작성해줘야 사용할 수 있게 된다.

정리

  • .current 프로퍼티에 값을 할당해서 값이 변동되어도 렌더링이 되지 않도록 하는 변수처럼 사용하려면 MutableRefObject 형태의 useRef를 사용해야 하므로 초기값으로 undefined를 할당한다.
  • DOM Element에 ref prop을 넘겨주려먼 RefObject 형태의 useRef를 사용해야 하므로 초기값으로 null을 할당한다.
profile
프론트엔드 개발자 배준형입니다.

0개의 댓글