cameras.html 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536
  1. <!DOCTYPE html><html lang="ko"><head>
  2. <meta charset="utf-8">
  3. <title>카메라(Cameras)</title>
  4. <meta name="viewport" content="width=device-width, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0">
  5. <meta name="twitter:card" content="summary_large_image">
  6. <meta name="twitter:site" content="@threejs">
  7. <meta name="twitter:title" content="Three.js – 카메라(Cameras)">
  8. <meta property="og:image" content="https://threejs.org/files/share.png">
  9. <link rel="shortcut icon" href="../../files/favicon_white.ico" media="(prefers-color-scheme: dark)">
  10. <link rel="shortcut icon" href="../../files/favicon.ico" media="(prefers-color-scheme: light)">
  11. <link rel="stylesheet" href="../resources/lesson.css">
  12. <link rel="stylesheet" href="../resources/lang.css">
  13. <script type="importmap">
  14. {
  15. "imports": {
  16. "three": "../../build/three.module.js"
  17. }
  18. }
  19. </script>
  20. <link rel="stylesheet" href="/manual/ko/lang.css">
  21. </head>
  22. <body>
  23. <div class="container">
  24. <div class="lesson-title">
  25. <h1>카메라(Cameras)</h1>
  26. </div>
  27. <div class="lesson">
  28. <div class="lesson-main">
  29. <p>※ 이 글은 Three.js의 튜토리얼 시리즈로서,
  30. 먼저 <a href="fundamentals.html">Three.js의 기본 구조에 관한 글</a>을
  31. 읽고 오길 권장합니다.</p>
  32. <p>이번 장에서는 카메라(cameras)에 대해 알아보겠습니다. <a href="fundamentals.html">첫 번째 장</a>에서
  33. 일부 다루긴 했지만, 중요 요소인 만큼 더 자세히 살펴볼 필요가 있습니다.</p>
  34. <p>Three.js에서 가장 자주 사용하는 카메라는 여태까지 썼던 <a href="/docs/#api/ko/cameras/PerspectiveCamera"><code class="notranslate" translate="no">PerspectiveCamera</code></a>(원근 카메라)입니다.
  35. 이 카메라는 멀리 있는 물체를 가까이 있는 것보다 상대적으로 작게 보이도록 해주죠.</p>
  36. <p><a href="/docs/#api/ko/cameras/PerspectiveCamera"><code class="notranslate" translate="no">PerspectiveCamera</code></a>는 <em>절두체(frustum)</em>를 만듭니다. <a href="https://ko.wikipedia.org/wiki/%EC%A0%88%EB%91%90%EC%B2%B4">절두체(frustum)는 입체(보통 원뿔이나 각뿔)를
  37. 절단하는 하나나 두 평행면 사이의 부분</a>을
  38. 의미하죠. 여기서 입체란 정육면체, 원뿔, 구, 원통, 절두체 등의 3D 요소입니다.</p>
  39. <div class="spread">
  40. <div><div data-diagram="shapeCube"></div><div>정육면체(cube)</div></div>
  41. <div><div data-diagram="shapeCone"></div><div>원뿔(cone)</div></div>
  42. <div><div data-diagram="shapeSphere"></div><div>구(sphere)</div></div>
  43. <div><div data-diagram="shapeCylinder"></div><div>원통(cylinder)</div></div>
  44. <div><div data-diagram="shapeFrustum"></div><div>절두체(frustum)</div></div>
  45. </div>
  46. <p>이걸 굳이 언급하는 이유는 글을 쓰는 저도 몇 년 동안 이를 몰랐기 때문입니다. 책이든 인터넷 글이든,
  47. <em>절두체</em>라는 단어를 봤다면 눈이 뒤집어졌을 겁니다. 입체의 이름을 알면 이해하기도, 기억하기도 훨씬
  48. 쉽죠 😅.</p>
  49. <p><a href="/docs/#api/ko/cameras/PerspectiveCamera"><code class="notranslate" translate="no">PerspectiveCamera</code></a>는 4가지 속성을 바탕으로 절두체를 만듭니다. <code class="notranslate" translate="no">near</code>는 절두체가 어디서 시작할지
  50. 결정하는 속성이고, <code class="notranslate" translate="no">far</code>는 절두체의 끝입니다. <code class="notranslate" translate="no">fov</code>는 시아갹(field of view)으로, <code class="notranslate" translate="no">near</code>와 카메라의
  51. 거리에 따라 절두체의 높이를 계산해 적용합니다. <code class="notranslate" translate="no">aspect</code>는 절두체의 너비에 관여하는 비율으로, 절두체의
  52. 너비는 절두체의 높이에 이 비율을 곱한 값입니다.</p>
  53. <p><img src="../resources/frustum-3d.svg" width="500" class="threejs_center"></p>
  54. <p><a href="lights.html">이전 장</a>에서 썼던 바닥면, 구체, 정육면체로 이루어진 예제를 다시 사용해
  55. 카메라의 속성을 조정할 수 있도록 만들겠습니다.</p>
  56. <p><code class="notranslate" translate="no">near</code> 속성은 항상 <code class="notranslate" translate="no">far</code> 속성보다 커야 하니, 이를 제어할 <code class="notranslate" translate="no">MinMaxGUIHelper</code> 헬퍼 클래스를
  57. 만들겠습니다. 이 클래스는 lil-gui가 제어할 <code class="notranslate" translate="no">min</code>과 <code class="notranslate" translate="no">max</code> 속성이 있고, lil-gui가 이를 조정할
  58. 때 지정한 두 가지 속성을 동시에 변경합니다.</p>
  59. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class MinMaxGUIHelper {
  60. constructor(obj, minProp, maxProp, minDif) {
  61. this.obj = obj;
  62. this.minProp = minProp;
  63. this.maxProp = maxProp;
  64. this.minDif = minDif;
  65. }
  66. get min() {
  67. return this.obj[this.minProp];
  68. }
  69. set min(v) {
  70. this.obj[this.minProp] = v;
  71. this.obj[this.maxProp] = Math.max(this.obj[this.maxProp], v + this.minDif);
  72. }
  73. get max() {
  74. return this.obj[this.maxProp];
  75. }
  76. set max(v) {
  77. this.obj[this.maxProp] = v;
  78. this.min = this.min; // min setter로 작동
  79. }
  80. }
  81. </pre>
  82. <p>이제 GUI를 만들어보죠.</p>
  83. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function updateCamera() {
  84. camera.updateProjectionMatrix();
  85. }
  86. const gui = new GUI();
  87. gui.add(camera, 'fov', 1, 180).onChange(updateCamera);
  88. const minMaxGUIHelper = new MinMaxGUIHelper(camera, 'near', 'far', 0.1);
  89. gui.add(minMaxGUIHelper, 'min', 0.1, 50, 0.1).name('near').onChange(updateCamera);
  90. gui.add(minMaxGUIHelper, 'max', 0.1, 50, 0.1).name('far').onChange(updateCamera);
  91. </pre>
  92. <p>카메라의 속성을 변경할 때마다 카메라의 <a href="/docs/#api/ko/cameras/PerspectiveCamera#updateProjectionMatrix"><code class="notranslate" translate="no">updateProjectionMatrix</code></a>
  93. 메서드를 호출해야 하므로, <code class="notranslate" translate="no">updateCamera</code>라는 함수를 만들어 값이 변경될 때마다 함수를 호출하도록
  94. 합니다.</p>
  95. <p></p><div translate="no" class="threejs_example_container notranslate">
  96. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/cameras-perspective.html"></iframe></div>
  97. <a class="threejs_center" href="/manual/examples/cameras-perspective.html" target="_blank">새 탭에서 보기</a>
  98. </div>
  99. <p></p>
  100. <p>값을 조정하며 카메라가 어떤 식으로 작동하는지 확인해보세요. <code class="notranslate" translate="no">aspect</code>는 창의 비율을 그대로 사용하도록
  101. 설정되어 있으므로, 이를 바꾸고 싶다면 예제를 새 창에서 열어 코드를 직접 수정해야 합니다.</p>
  102. <p>아직도 카메라가 어떤 식으로 작동하는지 보기 어려운가요? 까짓것 그럼 카메라를 하나 더 만들어보죠.
  103. 하나는 위의 예제와 같은 방식의 카메라이고, 다른 하나는 이 카메라의 시야와 절두체를 렌더링해
  104. 카메라가 어떻게 움직이는지 관찰할 수 있도록 만들겠습니다.</p>
  105. <p>Three.js의 가위 함수(scissor function)을 이용하면 쉽습니다. 가위 함수를 사용해 양쪽에
  106. 장면 두 개, 카메라 두 개를 렌더링하겠습니다.</p>
  107. <p>먼저 HTML과 CSS로 양쪽에 div 요소를 배치합니다. 이러면 각각의 카메라에 <a href="/docs/#examples/controls/OrbitControls"><code class="notranslate" translate="no">OrbitControls</code></a>를
  108. 두어 이벤트 처리하기도 훨씬 간단합니다.</p>
  109. <pre class="prettyprint showlinemods notranslate lang-html" translate="no">&lt;body&gt;
  110. &lt;canvas id="c"&gt;&lt;/canvas&gt;
  111. + &lt;div class="split"&gt;
  112. + &lt;div id="view1" tabindex="1"&gt;&lt;/div&gt;
  113. + &lt;div id="view2" tabindex="2"&gt;&lt;/div&gt;
  114. + &lt;/div&gt;
  115. &lt;/body&gt;
  116. </pre>
  117. <p>CSS로 두 div 요소가 canvas 위 양쪽에 자리하게 합니다.</p>
  118. <pre class="prettyprint showlinemods notranslate lang-css" translate="no">.split {
  119. position: absolute;
  120. left: 0;
  121. top: 0;
  122. width: 100%;
  123. height: 100%;
  124. display: flex;
  125. }
  126. .split &gt; div {
  127. width: 100%;
  128. height: 100%;
  129. }
  130. </pre>
  131. <p>카메라의 절두체를 시각화할 <a href="/docs/#api/ko/helpers/CameraHelper"><code class="notranslate" translate="no">CameraHelper</code></a>를 추가합니다.</p>
  132. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const cameraHelper = new THREE.CameraHelper(camera);
  133. ...
  134. scene.add(cameraHelper);
  135. </pre>
  136. <p>다음으로 두 div 요소를 참조합니다.</p>
  137. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const view1Elem = document.querySelector('#view1');
  138. const view2Elem = document.querySelector('#view2');
  139. </pre>
  140. <p>그리고 기존 <a href="/docs/#examples/controls/OrbitControls"><code class="notranslate" translate="no">OrbitControls</code></a>가 왼쪽 div 요소의 이벤트에만 반응하도록 설정합니다.</p>
  141. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">-const controls = new OrbitControls(camera, canvas);
  142. +const controls = new OrbitControls(camera, view1Elem);
  143. </pre>
  144. <p>다음으로 <a href="/docs/#api/ko/cameras/PerspectiveCamera"><code class="notranslate" translate="no">PerspectiveCamera</code></a>와 두 번째 <a href="/docs/#examples/controls/OrbitControls"><code class="notranslate" translate="no">OrbitControls</code></a>를 추가합니다. 이 <a href="/docs/#examples/controls/OrbitControls"><code class="notranslate" translate="no">OrbitControls</code></a>를
  145. 두 번째 카메라에 종속시키고, 오른쪽 div 요소의 이벤트에만 반응하도록 합니다.</p>
  146. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const camera2 = new THREE.PerspectiveCamera(
  147. 60, // 시야각(fov)
  148. 2, // 비율(aspect)
  149. 0.1, // near
  150. 500, // far
  151. );
  152. camera2.position.set(40, 10, 30);
  153. camera2.lookAt(0, 5, 0);
  154. const controls2 = new OrbitControls(camera2, view2Elem);
  155. controls2.target.set(0, 5, 0);
  156. controls2.update();
  157. </pre>
  158. <p>끝으로 가위 함수를 사용해 화면을 분할하겠습니다. 카메라 각각의 시점에 따라
  159. 장면을 canvas의 양쪽에 나눠 렌더링하게끔 할 것입니다.</p>
  160. <p>아래의 함수는 canvas 위에 덮어 씌운 요소의 사각 좌표(rectangle)를 구합니다.
  161. 그리고 해당 사각 좌표로 <code class="notranslate" translate="no">renderer</code>의 화면(viewport)과 가위(scissor)의 값을
  162. 정의한 뒤, 사각 좌표의 가로세로 비율을 반환합니다.</p>
  163. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function setScissorForElement(elem) {
  164. const canvasRect = canvas.getBoundingClientRect();
  165. const elemRect = elem.getBoundingClientRect();
  166. // canvas에 대응하는 사각형을 구하기
  167. const right = Math.min(elemRect.right, canvasRect.right) - canvasRect.left;
  168. const left = Math.max(0, elemRect.left - canvasRect.left);
  169. const bottom = Math.min(elemRect.bottom, canvasRect.bottom) - canvasRect.top;
  170. const top = Math.max(0, elemRect.top - canvasRect.top);
  171. const width = Math.min(canvasRect.width, right - left);
  172. const height = Math.min(canvasRect.height, bottom - top);
  173. // canvas의 일부분만 렌더링하도록 scissor 적용
  174. const positiveYUpBottom = canvasRect.height - bottom;
  175. renderer.setScissor(left, positiveYUpBottom, width, height);
  176. renderer.setViewport(left, positiveYUpBottom, width, height);
  177. // 비율 반환
  178. return width / height;
  179. }
  180. </pre>
  181. <p>이제 이 함수를 사용해 <code class="notranslate" translate="no">render</code> 함수에서 장면을 두 번 렌더링할 수 있습니다.</p>
  182. <pre class="prettyprint showlinemods notranslate lang-js" translate="no"> function render() {
  183. - if (resizeRendererToDisplaySize(renderer)) {
  184. - const canvas = renderer.domElement;
  185. - camera.aspect = canvas.clientWidth / canvas.clientHeight;
  186. - camera.updateProjectionMatrix();
  187. - }
  188. + resizeRendererToDisplaySize(renderer);
  189. +
  190. + // 가위 활성화
  191. + renderer.setScissorTest(true);
  192. +
  193. + // 기존 화면 렌더링
  194. + {
  195. + const aspect = setScissorForElement(view1Elem);
  196. +
  197. + // 비율에 따라 카메라 조정
  198. + camera.aspect = aspect;
  199. + camera.updateProjectionMatrix();
  200. + cameraHelper.update();
  201. +
  202. + // 기존 화면에서 가이드라인(CameraHelper)이 노출되지 않도록 설정
  203. + cameraHelper.visible = false;
  204. +
  205. + scene.background.set(0x000000);
  206. +
  207. + // 렌더링
  208. + renderer.render(scene, camera);
  209. + }
  210. +
  211. + // 두 번째 카메라 렌더링
  212. + {
  213. + const aspect = setScissorForElement(view2Elem);
  214. +
  215. + // 비율에 따라 카메라 조정
  216. + camera2.aspect = aspect;
  217. + camera2.updateProjectionMatrix();
  218. +
  219. + // 가이드라인 활성화
  220. + cameraHelper.visible = true;
  221. +
  222. + scene.background.set(0x000040);
  223. +
  224. + renderer.render(scene, camera2);
  225. + }
  226. - renderer.render(scene, camera);
  227. requestAnimationFrame(render);
  228. }
  229. requestAnimationFrame(render);
  230. }
  231. </pre>
  232. <p>위 예제에서는 분할된 두 화면을 구별하기 쉽두록 두 번째 화면의 배경을 진한
  233. 파란색으로 칠했습니다.</p>
  234. <p>또한 <code class="notranslate" translate="no">render</code> 함수 안에서 모든 것을 처리하기에, <code class="notranslate" translate="no">updateCamera</code> 함수도
  235. 제거하였습니다.</p>
  236. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">-function updateCamera() {
  237. - camera.updateProjectionMatrix();
  238. -}
  239. const gui = new GUI();
  240. -gui.add(camera, 'fov', 1, 180).onChange(updateCamera);
  241. +gui.add(camera, 'fov', 1, 180);
  242. const minMaxGUIHelper = new MinMaxGUIHelper(camera, 'near', 'far', 0.1);
  243. -gui.add(minMaxGUIHelper, 'min', 0.1, 50, 0.1).name('near').onChange(updateCamera);
  244. -gui.add(minMaxGUIHelper, 'max', 0.1, 50, 0.1).name('far').onChange(updateCamera);
  245. +gui.add(minMaxGUIHelper, 'min', 0.1, 50, 0.1).name('near');
  246. +gui.add(minMaxGUIHelper, 'max', 0.1, 50, 0.1).name('far');
  247. </pre>
  248. <p>이제 두 번째 화면에서 첫 번째 카메라의 절두체를 확인할 수 있습니다.</p>
  249. <p></p><div translate="no" class="threejs_example_container notranslate">
  250. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/cameras-perspective-2-scenes.html"></iframe></div>
  251. <a class="threejs_center" href="/manual/examples/cameras-perspective-2-scenes.html" target="_blank">새 탭에서 보기</a>
  252. </div>
  253. <p></p>
  254. <p>왼쪽은 기존의 화면과 같고 오른쪽에 왼쪽 카메라의 절두체가 보입니다.
  255. 패널에서 <code class="notranslate" translate="no">near</code>, <code class="notranslate" translate="no">far</code>, <code class="notranslate" translate="no">fov</code> 값을 조정하거나 마우스로 화면을 움직여보면,
  256. 오른쪽 화면의 절두체 안에 있는 물체만 왼쪽 화면에 노출됨을 확인할 수
  257. 있을 겁니다.</p>
  258. <p>누군가 이렇게 물을지도 모르겠네요. 그냥 <code class="notranslate" translate="no">near</code>를 0.0000000001로 설정하고 <code class="notranslate" translate="no">far</code>를
  259. 10000000000000로 설정해버리면요? 이러면 모든 게 항상 다 보이지 않나요? 이유를
  260. 설명하자면, GPU는 어떤 물체가 앞에 있거나 다른 물체의 뒤에 있을 때만 정밀도가
  261. 높기 때문입니다. 정밀도는 일정량이 <code class="notranslate" translate="no">near</code>와 <code class="notranslate" translate="no">far</code> 사이에 퍼져 있는데, 기본적으로
  262. 카메라에 가까울 수록 정밀도가 높고 멀수록 정밀도가 낮아집니다.</p>
  263. <p>현상을 직접 확인해보죠. 위의 예제를 수정해 20개의 구체를 한 줄로 세우겠습니다.</p>
  264. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">{
  265. const sphereRadius = 3;
  266. const sphereWidthDivisions = 32;
  267. const sphereHeightDivisions = 16;
  268. const sphereGeo = new THREE.SphereGeometry(sphereRadius, sphereWidthDivisions, sphereHeightDivisions);
  269. const numSpheres = 20;
  270. for (let i = 0; i &lt; numSpheres; ++i) {
  271. const sphereMat = new THREE.MeshPhongMaterial();
  272. sphereMat.color.setHSL(i * .73, 1, 0.5);
  273. const mesh = new THREE.Mesh(sphereGeo, sphereMat);
  274. mesh.position.set(-sphereRadius - 1, sphereRadius + 2, i * sphereRadius * -2.2);
  275. scene.add(mesh);
  276. }
  277. }
  278. </pre>
  279. <p><code class="notranslate" translate="no">near</code> 속성을 0.00001로 설정합니다.</p>
  280. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const fov = 45;
  281. const aspect = 2; // canvas 기본값
  282. -const near = 0.1;
  283. +const near = 0.00001;
  284. const far = 100;
  285. const camera = new THREE.PerspectiveCamera(fov, aspect, near, far);
  286. </pre>
  287. <p>그리고 기존의 GUI 코드를 수정해 0.00001의 작은 단위도 설정할 수 있도록 합니다.</p>
  288. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">-gui.add(minMaxGUIHelper, 'min', 0.1, 50, 0.1).name('near').onChange(updateCamera);
  289. +gui.add(minMaxGUIHelper, 'min', 0.00001, 50, 0.00001).name('near').onChange(updateCamera);
  290. </pre>
  291. <p>어떤 결과가 나올 것 같나요?</p>
  292. <p></p><div translate="no" class="threejs_example_container notranslate">
  293. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/cameras-z-fighting.html"></iframe></div>
  294. <a class="threejs_center" href="/manual/examples/cameras-z-fighting.html" target="_blank">새 탭에서 보기</a>
  295. </div>
  296. <p></p>
  297. <p>이는 <em>z-파이팅(z-fighting, Stitching)</em>의 한 예입니다. 컴퓨터의 GPU가 어떤 픽셀이
  298. 앞이고 어떤 픽셀을 뒤로 보내야할지 결정할 정밀도가 모자를 때 발생하는 현상이죠.</p>
  299. <p>위 예제가 어떻게 해도 정상적으로 보인다면, 아래 이미지를 보기 바랍니다.</p>
  300. <div class="threejs_center"><img src="../resources/images/z-fighting.png" style="width: 570px;"></div>
  301. <p>한 가지 해결책은 Three.js에게 픽셀의 앞 뒤를 결정할 때 다른 방법을 쓰도록 설정하는
  302. 것입니다. <a href="/docs/#api/ko/renderers/WebGLRenderer"><code class="notranslate" translate="no">WebGLRenderer</code></a>를 생성할 때 <code class="notranslate" translate="no">logarithmicDepthBuffer</code> 속성을 활성화해주면
  303. 되죠.</p>
  304. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">-const renderer = new THREE.WebGLRenderer({antialias: true, canvas});
  305. +const renderer = new THREE.WebGLRenderer({
  306. + antialias: true,
  307. + canvas,
  308. + logarithmicDepthBuffer: true,
  309. +});
  310. </pre>
  311. <p>대게의 경우 정상적으로 보일 겁니다.</p>
  312. <p></p><div translate="no" class="threejs_example_container notranslate">
  313. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/cameras-logarithmic-depth-buffer.html"></iframe></div>
  314. <a class="threejs_center" href="/manual/examples/cameras-logarithmic-depth-buffer.html" target="_blank">새 탭에서 보기</a>
  315. </div>
  316. <p></p>
  317. <p>문제가 그대로라면 <em>이 해결책을 쓸 수 없는 건가?</em>라는 의문에 빠질 수 있습니다.
  318. 이 해결책이 먹히지 않은 이유는 일부 GPU만 이 기능을 지원하기 때문이죠. 2018년
  319. 9월 기준으로 거의 모든 데스크탑 GPU가 이 기능을 지원하나, 모바일 기기는 대부분
  320. 이 기능을 지원하지 않습니다.</p>
  321. <p>이 기능을 쓰지 말아야 하는 또 다른 이유는, 이 기능이 일반적인 해결책보다 훨씬
  322. 성능이 나쁘기 때문입니다.</p>
  323. <p>게다가 이 기능을 활성화해도, <code class="notranslate" translate="no">near</code>를 더 작게, <code class="notranslate" translate="no">far</code>를 더 멀게 설정하다보면
  324. 결국 같은 현상을 만나게 될 겁니다.</p>
  325. <p>이는 항상 <code class="notranslate" translate="no">near</code>와 <code class="notranslate" translate="no">far</code>를 설정하는데 많은 공을 들여야 한다는 의미입니다.
  326. <code class="notranslate" translate="no">near</code>는 대상이 보이는 한 가장 멀게, <code class="notranslate" translate="no">far</code>도 대상이 보이는 한 카메라와 가장
  327. 가깝게 설정하는 것이 좋죠. 만약 거대한 공간을 렌더링하는 경우, 예를 들어 사람의
  328. 속눈썹과 50km 떨어진 산을 동시에 보이게 하려면 다른 해결책-나중에 다룰지도 모르는-을
  329. 찾아야 합니다. 당장은 <code class="notranslate" translate="no">near</code>와 <code class="notranslate" translate="no">far</code>를 적절하게 설정하는 게 중요하다는 것만
  330. 알아둡시다.</p>
  331. <p>두 번째로 자주 사용하는 카메라는 <a href="/docs/#api/ko/cameras/OrthographicCamera"><code class="notranslate" translate="no">OrthographicCamera</code></a>(정사영 카메라)입니다.
  332. 절두체 대신 <code class="notranslate" translate="no">left</code>, <code class="notranslate" translate="no">right</code>, <code class="notranslate" translate="no">top</code>, <code class="notranslate" translate="no">bottom</code>, <code class="notranslate" translate="no">near</code>, <code class="notranslate" translate="no">far</code>로 육면체를
  333. 정의해 사용하죠. 육면체로 화면을 투사하기에 원근 효과가 없습니다.</p>
  334. <p>2분할 화면 예제를 수정해 첫 번째 화면을 <a href="/docs/#api/ko/cameras/OrthographicCamera"><code class="notranslate" translate="no">OrthographicCamera</code></a>로 바꾸겠습니다.</p>
  335. <p>먼저 <a href="/docs/#api/ko/cameras/OrthographicCamera"><code class="notranslate" translate="no">OrthographicCamera</code></a>를 만들어보죠.</p>
  336. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const left = -1;
  337. const right = 1;
  338. const top = 1;
  339. const bottom = -1;
  340. const near = 5;
  341. const far = 50;
  342. const camera = new THREE.OrthographicCamera(left, right, top, bottom, near, far);
  343. camera.zoom = 0.2;
  344. </pre>
  345. <p>먼저 <code class="notranslate" translate="no">left</code>와 <code class="notranslate" translate="no">bottom</code>을 -1로, <code class="notranslate" translate="no">right</code>와 <code class="notranslate" translate="no">top</code>을 1로 설정했습니다. 이러면 육면체는
  346. 기본적으로 너비 2칸, 높이 2칸이 되겠죠. 그리고 육면체의 비율을 조정해 <code class="notranslate" translate="no">left</code>와 <code class="notranslate" translate="no">top</code>의
  347. 값을 조정할 수 있도록, <code class="notranslate" translate="no">zoom</code> 속성을 이용해 카메라에 보이는 범위를 조정할 수 있도록
  348. 했습니다.</p>
  349. <p>다음으로 <code class="notranslate" translate="no">zoom</code> 속성을 조정할 GUI를 추가합니다.</p>
  350. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const gui = new GUI();
  351. +gui.add(camera, 'zoom', 0.01, 1, 0.01).listen();
  352. </pre>
  353. <p><code class="notranslate" translate="no">listen</code> 메서드를 호출하면 lil-gui가 변화를 감지합니다. 이렇게 한 이유는 <a href="/docs/#examples/controls/OrbitControls"><code class="notranslate" translate="no">OrbitControls</code></a>이
  354. 마우스 휠 스크롤을 감지해 <code class="notranslate" translate="no">zoom</code> 속성을 변경하기 때문이죠.</p>
  355. <p>끝으로 왼쪽 화면을 렌더링할 때 <a href="/docs/#api/ko/cameras/OrthographicCamera"><code class="notranslate" translate="no">OrthographicCamera</code></a>를 업데이트하도록 설정합니다.</p>
  356. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">{
  357. const aspect = setScissorForElement(view1Elem);
  358. // 요소의 비율에 맞춰 카메라 업데이트
  359. - camera.aspect = aspect;
  360. + camera.left = -aspect;
  361. + camera.right = aspect;
  362. camera.updateProjectionMatrix();
  363. cameraHelper.update();
  364. // 기존 화면에서 가이드라인(CameraHelper)이 노출되지 않도록 설정
  365. cameraHelper.visible = false;
  366. scene.background.set(0x000000);
  367. renderer.render(scene, camera);
  368. }
  369. </pre>
  370. <p>이제 <a href="/docs/#api/ko/cameras/OrthographicCamera"><code class="notranslate" translate="no">OrthographicCamera</code></a>가 어떻게 작동하는지 확인할 차례입니다.</p>
  371. <p></p><div translate="no" class="threejs_example_container notranslate">
  372. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/cameras-orthographic-2-scenes.html"></iframe></div>
  373. <a class="threejs_center" href="/manual/examples/cameras-orthographic-2-scenes.html" target="_blank">새 탭에서 보기</a>
  374. </div>
  375. <p></p>
  376. <p>Three.js에서 <a href="/docs/#api/ko/cameras/OrthographicCamera"><code class="notranslate" translate="no">OrthographicCamera</code></a>는 주로 2D 요소를 표현하기 위해 사용합니다.
  377. 카메라에 얼마나 많은 요소를 보여줄지만 결정하면 되죠. 만약 canvas의 1픽셀을
  378. 카메라의 한 칸과 같은 크기로 지정하고 싶다면...</p>
  379. <p>중점을 장면의 중심에 두고 1 픽셀을 Three.js의 한 칸으로 만들 수 있습니다.</p>
  380. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">camera.left = -canvas.width / 2;
  381. camera.right = canvas.width / 2;
  382. camera.top = canvas.height / 2;
  383. camera.bottom = -canvas.height / 2;
  384. camera.near = -1;
  385. camera.far = 1;
  386. camera.zoom = 1;
  387. </pre>
  388. <p>2D canvas처럼 중점을 상단 왼쪽에 두려면 다음과 같이 설정할 수 있죠.</p>
  389. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">camera.left = 0;
  390. camera.right = canvas.width;
  391. camera.top = 0;
  392. camera.bottom = canvas.height;
  393. camera.near = -1;
  394. camera.far = 1;
  395. camera.zoom = 1;
  396. </pre>
  397. <p>중점이 상단 왼쪽에 있을 경우의 좌표는 2D canvas처럼 0, 0입니다.</p>
  398. <p>한 번 만들어보죠! 먼저 카메라를 설정합니다.</p>
  399. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const left = 0;
  400. const right = 300; // canvas 기본 크기
  401. const top = 0;
  402. const bottom = 150; // canvas 기본 크기
  403. const near = -1;
  404. const far = 1;
  405. const camera = new THREE.OrthographicCamera(left, right, top, bottom, near, far);
  406. camera.zoom = 1;
  407. </pre>
  408. <p>다음으로 평면(plane) 6개를 만들어 각각 다른 텍스처를 적용하겠습니다.
  409. 각 평면마다 <a href="/docs/#api/ko/core/Object3D"><code class="notranslate" translate="no">THREE.Object3D</code></a> 인스턴스를 만들어 평면의 부모로 설정합니다.
  410. 이러면 중점을 0, 0, 상단 좌측으로 지정해 좌표를 지정하기가 쉽습니다.</p>
  411. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const loader = new THREE.TextureLoader();
  412. const textures = [
  413. loader.load('resources/images/flower-1.jpg'),
  414. loader.load('resources/images/flower-2.jpg'),
  415. loader.load('resources/images/flower-3.jpg'),
  416. loader.load('resources/images/flower-4.jpg'),
  417. loader.load('resources/images/flower-5.jpg'),
  418. loader.load('resources/images/flower-6.jpg'),
  419. ];
  420. const planeSize = 256;
  421. const planeGeo = new THREE.PlaneGeometry(planeSize, planeSize);
  422. const planes = textures.map((texture) =&gt; {
  423. const planePivot = new THREE.Object3D();
  424. scene.add(planePivot);
  425. texture.magFilter = THREE.NearestFilter;
  426. const planeMat = new THREE.MeshBasicMaterial({
  427. map: texture,
  428. side: THREE.DoubleSide,
  429. });
  430. const mesh = new THREE.Mesh(planeGeo, planeMat);
  431. planePivot.add(mesh);
  432. // 평면을 움직여 상단 좌측이 중점이 되도록 설정
  433. mesh.position.set(planeSize / 2, planeSize / 2, 0);
  434. return planePivot;
  435. });
  436. </pre>
  437. <p>그리고 <code class="notranslate" translate="no">render</code> 함수 안에 canvas의 사이즈가 변경되었을 때 카메라를 업데이트하는
  438. 코드를 추가합니다.</p>
  439. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function render() {
  440. if (resizeRendererToDisplaySize(renderer)) {
  441. camera.right = canvas.width;
  442. camera.bottom = canvas.height;
  443. camera.updateProjectionMatrix();
  444. }
  445. ...
  446. </pre>
  447. <p><code class="notranslate" translate="no">planes</code>는 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">THREE.Mesh</code></a>, 평면의 배열입니다. 시간값을 기반으로 이 평면이 따로
  448. 움직이도록 하겠습니다.</p>
  449. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function render(time) {
  450. time *= 0.001; // 초 단위로 변경
  451. ...
  452. const distAcross = Math.max(20, canvas.width - planeSize);
  453. const distDown = Math.max(20, canvas.height - planeSize);
  454. // total distance to move across and back
  455. const xRange = distAcross * 2;
  456. const yRange = distDown * 2;
  457. const speed = 180;
  458. planes.forEach((plane, ndx) =&gt; {
  459. // compute a unique time for each plane
  460. const t = time * speed + ndx * 300;
  461. // get a value between 0 and range
  462. const xt = t % xRange;
  463. const yt = t % yRange;
  464. // set our position going forward if 0 to half of range
  465. // and backward if half of range to range
  466. const x = xt &lt; distAcross ? xt : xRange - xt;
  467. const y = yt &lt; distDown ? yt : yRange - yt;
  468. plane.position.set(x, y, 0);
  469. });
  470. renderer.render(scene, camera);
  471. </pre>
  472. <p>이미지가 완벽하게 가장자리에서 튕기는 것을 확인할 수 있을 겁니다. 2D canvas에서
  473. 픽셀값을 이용해 구현할 때와 같은 방식이죠.</p>
  474. <p></p><div translate="no" class="threejs_example_container notranslate">
  475. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/cameras-orthographic-canvas-top-left-origin.html"></iframe></div>
  476. <a class="threejs_center" href="/manual/examples/cameras-orthographic-canvas-top-left-origin.html" target="_blank">새 탭에서 보기</a>
  477. </div>
  478. <p></p>
  479. <p><a href="/docs/#api/ko/cameras/OrthographicCamera"><code class="notranslate" translate="no">OrthographicCamera</code></a>는 게임 엔진 에디터 등에서처럼 3D 모델링 결과물의 상, 하, 좌, 우,
  480. 앞, 뒤를 렌더링할 때도 사용합니다.</p>
  481. <div class="threejs_center"><img src="../resources/images/quad-viewport.png" style="width: 574px;"></div>
  482. <p>위 스크린샷의 1개의 화면만 원근(perspective) 카메라이고 나머지 3개는 정사영(orthographic)
  483. 카메라입니다.</p>
  484. <p>여기까지 카메라의 기초에 대해 살펴보았습니다. 카메라를 움직이는 방법에 대해서는 다른
  485. 글에서 좀 더 상세히 설명할 거예요. 다음은 장에서는 <a href="shadows.html">그림자(shadows)</a>에
  486. 대해 먼저 살펴보겠습니다.</p>
  487. <p><canvas id="c"></canvas></p>
  488. <script type="module" src="../resources/threejs-cameras.js"></script>
  489. </div>
  490. </div>
  491. </div>
  492. <script src="../resources/prettify.js"></script>
  493. <script src="../resources/lesson.js"></script>
  494. </body></html>