자바스크립트

threejs) 프레임마다 업데이트되는 TextGeometry

1일1공부실천하자 2024. 8. 25. 20:06

안녕하세요. 무려 한달 반만에 작성하는 게시글입니다.

프로젝트를 진행하며 무척 애를 먹었던 부분이 생겨 글을 작성해봅니다.

 

우선 제목에서 알다시피 threejs의 TextGeometry부분입니다.

 

다른 지오메트리와 달리 textGeomtry는 매 프레임마다 업데이트 되면 성능문제를 피할 수 없습니다.

여기서 말하는 업데이트란, 텍스트의 변화입니다.

 

 

 

사진으로 보기엔 정상적으로 나오는 텍스트입니다. 하지만 이것을 매 프레임마다 업데이트한다면 어떻게 될까요?

제 경우에서는 "onmousemove"이벤트에서 실행될 것입니다.

좌표 (0,0,0)에서 현재 마우스 위치까지의 거리를 "distanceTo"메서드로 구한 후 이를 텍스트로 나타내는 기능입니다.

영상에서 보시다시피 마우스를 움직이면 즉각적으로 텍스트에 변화가 생기는 것이 아니라 몇 초의 짧은 텀을 두고 텍스트가 업데이트 됩니다.

마치 디바운스를 사용한 것 같습니다.

 

해당 성능문제는 저를 사흘을 넘게 괴롭혔습니다. 하나하나 체크하며 성능 문제를 발생시키는 부분을 확인하기도 했고, 모든 코드를 갈아엎기도 했습니다. 

예시로 보여드린 코드는 매우 간단합니다.

 

function onMouseMove(e){
	// ...do something
	if (scene.getObjectByName("text")) {
          scene.remove("text");
        }
      const t = new Text({ text: `${distance}` });
      t.name = "text";
      t.rotation.x = -Math.PI / 2;
      scene.add(t);
}

 

 

다른 지오메트리는 괜찮지만 유독 텍스트 지오메트리에서 성능이슈가 발생하는 이유는 TextGeometry의 생성 과정이 매우 비싼 연산이기 때문입니다. 

TextGeomtry는 텍스트를 폰트 매트릭스에 따라 3D 지오메트리로 변환하는 복잡한 과정을 거칩니다. 이 과정에는 세그먼트, 폰트 데이터 등의 복잡한 계산이 필요합니다.

 

그럼 어떻게해야 성능이슈 없이 프레임마다 폰트를 생성할 수 있을까요?

답은 threejs discourse에서 찾을 수 있었습니다.

 

 

매 프레임마다 TextGeometry를 업데이트하는 것을 피하고, "Troika-three-text"와 같은 threejs와 연계될 수 있는 외부 라이브러리를 사용하는 것을 권장한다 합니다.

 

해당 공식문서를 읽어보는 것을 추천합니다만, 사용법은 threejs보다 훨씬 쉽습니다.

 

  const t = new Text();
  scene.add(t);
  t.color = "red";
  t.fontSize = 5;
  t.name = "text";
  t.text = `${distance}`;
  t.rotation.x = -Math.PI / 2;
  t.sync();

 

 

영상이 저품질이라 끊기는 느낌이 들지만, 실제로는 매우 부드럽게 동작하고 있습니다.