Linux dpw.dpwebtech.com 3.10.0-1160.88.1.el7.x86_64 #1 SMP Tue Mar 7 15:41:52 UTC 2023 x86_64
Apache
: 192.232.243.69 | : 18.223.172.243
54 Domain
7.3.33
dpclient
www.github.com/MadExploits
Terminal
AUTO ROOT
Adminer
Backdoor Destroyer
Linux Exploit
Lock Shell
Lock File
Create User
CREATE RDP
PHP Mailer
BACKCONNECT
UNLOCK SHELL
HASH IDENTIFIER
CPANEL RESET
CREATE WP USER
README
+ Create Folder
+ Create File
/
home /
dpclient /
public_html /
HRD-Test /
manual /
ko /
[ HOME SHELL ]
Name
Size
Permission
Action
resources
[ DIR ]
drwxr-xr-x
align-html-elements-to-3d.html
34.71
KB
-rw-r--r--
backgrounds.html
14.09
KB
-rw-r--r--
billboards.html
16.08
KB
-rw-r--r--
cameras.html
30.68
KB
-rw-r--r--
canvas-textures.html
18.89
KB
-rw-r--r--
cleanup.html
18.02
KB
-rw-r--r--
custom-buffergeometry.html
23.81
KB
-rw-r--r--
debugging-glsl.html
7.45
KB
-rw-r--r--
debugging-javascript.html
31.93
KB
-rw-r--r--
fog.html
14.89
KB
-rw-r--r--
fundamentals.html
31.61
KB
-rw-r--r--
game.html
91.06
KB
-rw-r--r--
indexed-textures.html
31.35
KB
-rw-r--r--
lang.css
589
B
-rw-r--r--
lights.html
30.72
KB
-rw-r--r--
load-gltf.html
35.02
KB
-rw-r--r--
load-obj.html
37.24
KB
-rw-r--r--
material-table.html
1.86
KB
-rw-r--r--
materials.html
21.47
KB
-rw-r--r--
multiple-scenes.html
29.43
KB
-rw-r--r--
offscreencanvas.html
49.49
KB
-rw-r--r--
optimize-lots-of-objects-anima...
23.92
KB
-rw-r--r--
optimize-lots-of-objects.html
27.17
KB
-rw-r--r--
picking.html
22.34
KB
-rw-r--r--
post-processing-3dlut.html
27.84
KB
-rw-r--r--
post-processing.html
17.92
KB
-rw-r--r--
prerequisites.html
23.54
KB
-rw-r--r--
primitives.html
25.21
KB
-rw-r--r--
rendering-on-demand.html
12.77
KB
-rw-r--r--
rendertargets.html
8.71
KB
-rw-r--r--
responsive.html
16.99
KB
-rw-r--r--
scenegraph.html
31.44
KB
-rw-r--r--
setup.html
5.14
KB
-rw-r--r--
shadertoy.html
25.15
KB
-rw-r--r--
shadows.html
29.24
KB
-rw-r--r--
textures.html
35.63
KB
-rw-r--r--
tips.html
17.83
KB
-rw-r--r--
transparency.html
19.85
KB
-rw-r--r--
voxel-geometry.html
45.92
KB
-rw-r--r--
webxr-basics.html
21.66
KB
-rw-r--r--
webxr-look-to-select.html
22.26
KB
-rw-r--r--
webxr-point-to-select.html
17.58
KB
-rw-r--r--
Delete
Unzip
Zip
${this.title}
Close
Code Editor : fundamentals.html
<!DOCTYPE html><html lang="ko"><head> <meta charset="utf-8"> <title>Three.js란?</title> <meta name="viewport" content="width=device-width, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0"> <meta name="twitter:card" content="summary_large_image"> <meta name="twitter:site" content="@threejs"> <meta name="twitter:title" content="Three.js – Three.js란?"> <meta property="og:image" content="https://threejs.org/files/share.png"> <link rel="shortcut icon" href="../../files/favicon_white.ico" media="(prefers-color-scheme: dark)"> <link rel="shortcut icon" href="../../files/favicon.ico" media="(prefers-color-scheme: light)"> <link rel="stylesheet" href="../resources/lesson.css"> <link rel="stylesheet" href="../resources/lang.css"> <!-- Import maps polyfill --> <!-- Remove this when import maps will be widely supported --> <script async src="https://unpkg.com/es-module-shims@1.8.0/dist/es-module-shims.js"></script> <script type="importmap"> { "imports": { "three": "../../build/three.module.js" } } </script> <link rel="stylesheet" href="/manual/ko/lang.css"> </head> <body> <div class="container"> <div class="lesson-title"> <h1>Three.js란?</h1> </div> <div class="lesson"> <div class="lesson-main"> <p>Three.js 입문하신 것을 환영합니다! <a href="https://threejs.org">Three.js</a>는 웹페이지에 3D 객체를 쉽게 렌더링하도록 도와주는 자바스크립트 3D 라이브러리입니다.</p> <p>대부분의 경우 Three.js는 3D 객체를 렌더링하는 데 WebGL을 사용합니다. 때문에 Three.js = WebGL이라고 착각하기 쉽죠. 하지만 <a href="https://webglfundamentals.org">WebGL은 점, 선, 삼각형만을 그리는 아주 단순한 시스템입니다</a>. WebGL로 직접 무언가를 만들려면 상당히 많은 양의 코드를 짜야 하죠. 만약 씬(scenes), 광원, 그림자, 물체, 텍스처 등 3차원 세계를 구현한다면 머리도 꽤 복잡하겠거니와 코드 자체도 굉장히 복잡할 겁니다. Three.js는 이런 3D 요소들의 처리를 도와 직관적인 코드를 짤 수 있도록 해줍니다.</p> <p>본 튜토리얼은 ES2015 문법을 적극적으로 사용할 것이므로, 여러분이 이미 최신 자바스크립트에 능숙하다는 가정 아래에 진행합니다. 자세한 내용에 대해서는 <a href="prerequisites.html">먼저 알아야 할 것</a>을 참고하세요. 대부분의 모던 브라우저가 자동 업데이트 기능을 사용하므로, 특이한 환경이 아니라면 예시에 나온 코드를 실행하는 데 문제가 없을 것입니다. 만약 구형 브라우저에서도 코드가 실행되길 원한다면, <a href="https://babeljs.io">바벨(Babel)</a> 같은 트랜스파일러를 찾아보세요. 물론 너무 오래된 엔진을 사용하는 브라우저에서는 Three.js를 아예 구동하지 못할 수 있습니다.</p> <p>새로운 프로그래밍 언어를 배울 때, <code class="notranslate" translate="no">"Hello World!"</code>를 출력하며 시작했던 것을 기억하나요? Three.js는 3차원 세계를 다루므로, 우리는 정육면체(cube)를 3차원으로 구현하는 것으로 튜토리얼을 시작하고자 합니다. 이른바 "Hello Cube"라는 것이죠!</p> <p>하지만 먼저 Three.js 앱의 구조부터 간략히 살펴보죠. Three.js 앱을 만들려면 다양한 객체를 생성해 연결해야 합니다. 아래는 Three.js 앱의 구조를 도식화한 것입니다.</p> <div class="threejs_center"><img src="../resources/images/threejs-structure.svg" style="width: 768px;"></div> <p>위 도표에서 중요한 것들을 추려보죠.</p> <ul> <li><p>먼저 <a href="/docs/#api/ko/constants/Renderer"><code class="notranslate" translate="no">Renderer</code></a>가 있습니다. Three.js의 핵심 객체이죠. <a href="/docs/#api/ko/constants/Renderer"><code class="notranslate" translate="no">Renderer</code></a>는 <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>과 <a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a> 객체를 넘겨 받아 카메라의 <a href="https://ko.wikipedia.org/wiki/%EC%A0%88%EB%91%90%EC%B2%B4"><em>절두체(frustum)</em></a> 안 3D 씬의 일부를 평면(2차원) 이미지로 렌더링합니다.</p> </li> <li><p><a href="scenegraph.html">씬 그래프(Scene graph)</a>는 <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a> 또는 다수의 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>, <a href="/docs/#api/ko/lights/Light"><code class="notranslate" translate="no">Light</code></a>, <a href="/docs/#api/ko/objects/Group"><code class="notranslate" translate="no">Group</code></a>, <a href="/docs/#api/ko/core/Object3D"><code class="notranslate" translate="no">Object3D</code></a>, <a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a>로 이루어진 트리 구조와 유사합니다. <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>은 씬 그래프의 최상위 노드로서 배경색(background color), 안개(fog) 등의 요소를 포함합니다. <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>에 포함된 객체들 또한 부모/자식의 트리 구조로 이루어지며, 이는 각 객체의 유래와 방향성을 나타냅니다. 쉽게 말해 자식 객체의 위치(position)와 방향(orientation)이 부모 기준이라는 거죠. 예를 들어 자동차의 바퀴가 자동차 객체의 자식 객체라면, 자동차 객체의 방향을 움직일 때, 바퀴 객체의 방향 또한 같이 움직입니다(더 자세한 내용은 <a href="scenegraph.html">씬 그래프에 관한 글</a>에서 확인할 수 있습니다)</p> <p><a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a>가 도표에서 반쯤 나간 것이 보이나요? 이는 의도된 것으로, 다른 객체와 달리 <a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a>는 굳이 씬 그래프에 포함될 필요가 없음을 보여주기 위함입니다. 물론 다른 객체와 마찬가지로 <a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a> 또한 다른 객체의 자식 객체가 될 수 있습니다. 이러면 부모 객체에 따라 <a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a> 또한 움직이겠죠. <a href="scenegraph.html">씬 그래프에 관한 글</a> 마지막에 여러개의 <a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a>를 넣는 예제가 있으니 참고하시기 바랍니다.</p> </li> <li><p><a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>는 어떤 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>로 하나의 <code class="notranslate" translate="no">Geometry</code>를 그리는 객체입니다. <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>, <code class="notranslate" translate="no">Geometry</code>는 재사용이 가능하여 여러개의 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>가 하나의 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a> 또는 <code class="notranslate" translate="no">Geometry</code>를 동시에 참조할 수 있습니다. 파란색 정육면체 2개를 그린다고 해보죠. 일단 두 정육면체의 위치가 달라야 하니, 2개의 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>가 필요합니다. 그리고 정점(vertex, 꼭지점) 데이터를 가진 한 개의 <code class="notranslate" translate="no">Geometry</code>와 채색을 위한 하나의 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>이 필요하겠죠. 이때 각 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>는 객체를 복사할 필요 없이, 같은 <code class="notranslate" translate="no">Geometry</code> 그리고 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>을 참조할 수 있습니다.</p> </li> <li><p><code class="notranslate" translate="no">Geometry</code>는 기하학 객체의 정점 데이터입니다. 구(sphere), 정육면체(cube), 면(plane), 개, 고양이, 사람, 나무, 건물 등 아주 다양한 것이 될 수 있죠. Three.js는 기본적으로 몇 가지의 내장(built-in) <a href="primitives.html">기하학 객체</a>를 제공합니다. 물론 <a href="custom-buffergeometry.html">직접 기하학 객체를 만들</a> 수도 있고, <a href="load-obj.html">파일에서 기하학 객체를 불러올</a> 수도 있죠.</p> </li> <li><p><a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>은 기하학 객체를 그리는 데 사용하는 <a href="materials.html">표면 속성</a>입니다. 색이나 밝기 등을 지정할 수 있죠. 하나의 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>는 여러개의 <a href="/docs/#api/ko/textures/Texture"><code class="notranslate" translate="no">Texture</code></a>를 사용할 수 있습니다. 기하학 객체의 표면을 이미지로 덮어씌울 때 주로 사용하죠.</p> </li> <li><p><a href="/docs/#api/ko/textures/Texture"><code class="notranslate" translate="no">Texture</code></a>는 이미지나 <a href="textures.html">파일에서 로드한 이미지</a>, <a href="canvas-textures.html">canvas로 생성한 이미지</a> 또는 <a href="rendertargets.html">다른 <code class="notranslate" translate="no">Scene</code> 객체에서 렌더링한 결과물</a>에 해당합니다.</p> </li> <li><p><a href="/docs/#api/ko/lights/Light"><code class="notranslate" translate="no">Light</code></a>는 <a href="lights.html">여러 종류의 광원에 해당</a>합니다.</p> </li> </ul> <p>이제 기본적인 구조에 대해 배웠으니 아래와 같은 구조의 <em>"Hello Cube"</em>를 만들어 봅시다.</p> <div class="threejs_center"><img src="../resources/images/threejs-1cube-no-light-scene.svg" style="width: 500px;"></div> <p>먼저, Three.js를 로드합니다.</p> <pre class="prettyprint showlinemods notranslate lang-html" translate="no"><script type="module"> import * as THREE from 'three'; </script> </pre> <p><code class="notranslate" translate="no">type="module"</code> 속성을 지정하는 것을 잊지 마세요. 이러면 <code class="notranslate" translate="no">import</code> 키워드를 통해 Three.js 모듈을 불러올 수 있습니다. 다른 방법으로 로드할 수도 있지만 모듈 로드가 r106 이후 빌드에서 추천하는 방식입니다. 모듈로 로드하면 다른 의존성 스크립트를 알아서 로드하기 때문에 <code class="notranslate" translate="no"><script></code> 태그를 일일이 작성하는 수고를 덜 수 있습니다.</p> <p>그리고 <code class="notranslate" translate="no"><canvas></code> 태그를 작성합니다.</p> <pre class="prettyprint showlinemods notranslate lang-html" translate="no"><body> <canvas id="c"></canvas> </body> </pre> <p>이제 Three.js에게 렌더링을 맡겨보죠.</p> <pre class="prettyprint showlinemods notranslate lang-html" translate="no"><script type="module"> import * as THREE from 'three'; +function main() { + const canvas = document.querySelector('#c'); + const renderer = new THREE.WebGLRenderer({antialias: true, canvas}); + ... </script> </pre> <p>canvas 요소(element)를 참조한 이후 <a href="/docs/#api/ko/renderers/WebGLRenderer"><code class="notranslate" translate="no">WebGLRenderer</code></a>를 생성했습니다. 이 렌더러(renderer)는 여러분이 입력한 데이터를 실제로 canvas에 그려주는</p> <p>만약 Three.js에 canvas 요소를 넘겨주지 않으면, Three.js는 canvas 요소를 자동으로 생성합니다. 그리고 여러분이 직접 이 canvas 요소를 문서(document) 에 삽입해야 하죠. 어디에 canvas 요소를 넣을지는 경우에 따라 다를 테니, 예제에서는 자바스크립트에서 요소를 참조하도록 하는 편이 호환성 면에서 더 낫다고 판단했습니다. 만약 canvas 요소를 동적으로 삽입한다면 여러분이 직접 코드를 고쳐야 할 테니까요.</p> <p>다음으로 카메라가 필요하니 <code class="notranslate" translate="no">PerspectiveCamera(원근 카메라)</code> 객체를 생성해봅시다.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const fov = 75; const aspect = 2; // the canvas default const near = 0.1; const far = 5; const camera = new THREE.PerspectiveCamera(fov, aspect, near, far); </pre> <p><code class="notranslate" translate="no">fov</code>는 <code class="notranslate" translate="no">field of view(시야각)</code>의 줄임말입니다. 예제의 경우 수직면 75도로 설정했습니다. 알아둬야 할 건 Three.js의 대부분이 각도 단위로 호도(radians)를 사용하는데, 원근 카메라만 특이하게 도(degrees)를 인자로 받는다는 점입니다.</p> <p><code class="notranslate" translate="no">aspect</code>는 canvas의 가로 세로 비율입니다. 이는 <a href="responsive.html">다른 글</a> 에서 자세히 다루겠지만, 기본 설정으로 canvas의 크기는 300x150이니 비율도 300/150, 2로 설정했습니다.</p> <p><code class="notranslate" translate="no">near</code>와 <code class="notranslate" translate="no">far</code>는 카메라 앞에 렌더링되는 공간 범위를 지정하는 요소입니다. 이 공간 바깥에 있는 요소는 화면에서 잘려나가며, 렌더링되지 않을 것입니다.</p> <p>위에서 살펴본 4가지 속성은 하나의 <em>"절두체"</em>를 만듭니다. <em>"절두체"</em>는 끝부분이 잘려나간 피라미드처럼 생긴 3차원 모양인데, 구, 정육면체, 각기둥(prism)처럼 3차원 모양의 하나로 생각하면 됩니다.</p> <p><img src="../resources/frustum-3d.svg" width="500" class="threejs_center"></p> <p><code class="notranslate" translate="no">near</code>와 <code class="notranslate" translate="no">far</code> 평면의 높이는 시야각(fov), 너비는 시야각과 <code class="notranslate" translate="no">aspect</code>에 의해 결정됩니다.</p> <p>그리고 앞서 말했듯 절두체 안에 있는 요소만 렌더링되며, 바깥에 있는 요소는 렌더링되지 않습니다.</p> <p>기본 설정으로 카메라는 -Z 축 +Y 축, 즉 아래를 바라봅니다. 정육면체를 원점에 놓을 예정이니 카메라를 조금 뒤로 움직여 화면 안에 들어오도록 해보죠.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">camera.position.z = 2; </pre> <p>우리가 원하는 결과물을 다음처럼 그려볼 수 있습니다.</p> <p><img src="../resources/scene-down.svg" width="500" class="threejs_center"></p> <p>카메라는 <code class="notranslate" translate="no">z = 2</code> 위치에서 -Z 방향을 바라봅니다. 절두체는 카메라 앞 0.1 칸에서 5칸까지를 차지하죠. 또한 아래를 바라보는 형태이기 때문에, 시야각은 canvas 크기의 영향을 받습니다. 앞서 만든 canvas는 두 배 더 크기 때문에 실제로는 시야각이 위에서 정의한 75도 보다는 훨씬 넓을 것입니다.</p> <p>이제 <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>을 만듭니다. Three.js에서 <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>이란 씬 그래프에서 가장 상단에 위치한 요소입니다. 뭔가를 화면에 렌더링하고 싶다면 먼저 <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>에 추가해야 하죠. 여기서는 간단하게 다룰 것이므로 자세한 내용은 <a href="scenegraph.html">이 글</a>을 참고하세요.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const scene = new THREE.Scene(); </pre> <p>다음으로 간단한 정육면체를 만들어보죠. Three.js에서 렌더링되는 대부분의 3D 요소는 정점 데이터가 정의된 기하학 객체를 필요로 합니다. 지금은 정육면체를 만들어야 하니 <a href="/docs/#api/ko/geometries/BoxGeometry"><code class="notranslate" translate="no">BoxGeometry</code></a> 생성자를 호출하겠습니다.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const boxWidth = 1; const boxHeight = 1; const boxDepth = 1; const geometry = new THREE.BoxGeometry(boxWidth, boxHeight, boxDepth); </pre> <p>다음으로 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>을 만들고 색을 지정합니다. 색을 지정할 때는 CSS처럼 숫자 형태의 hex 코드를 이용합니다(<code class="notranslate" translate="no">#</code> 대신 <code class="notranslate" translate="no">0x</code>. 역주).</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const material = new THREE.MeshBasicMaterial({color: 0x44aa88}); </pre> <p>그리고 앞서 만든 <code class="notranslate" translate="no">Geometry</code>(물체의 형태)와 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>(물체의 색, 밝기, 질감 등)을 이용해 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>를 만듭니다. <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>는 <code class="notranslate" translate="no">Geometry</code>, <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a> 외에도 물체의 위치, 방향, 크기 등을 담은 객체입니다.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const cube = new THREE.Mesh(geometry, material); </pre> <p>마지막으로 완성된 정육면체 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>를 <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>에 넣어줍니다.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">scene.add(cube); </pre> <p><code class="notranslate" translate="no">renderer</code>의 <code class="notranslate" translate="no">render</code> 메서드에 <a href="/docs/#api/ko/scenes/Scene"><code class="notranslate" translate="no">Scene</code></a>과 <a href="/docs/#api/ko/cameras/Camera"><code class="notranslate" translate="no">Camera</code></a>를 넘겨주면 화면을 렌더링할 수 있습니다.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">renderer.render(scene, camera); </pre> <p>아래는 전체 코드입니다.</p> <p></p><div translate="no" class="threejs_example_container notranslate"> <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/fundamentals.html"></iframe></div> <a class="threejs_center" href="/manual/examples/fundamentals.html" target="_blank">새 탭에서 보기</a> </div> <p></p> <p>물론 위 예제는 3D 정육면체라고 하긴 어렵습니다. 코드 상으로 Three.js는 분명 3D 정육면체를 그렸지만, 카메라가 -Z 방향을 바라보고, 정육면체도 Z 축에 맞추어 정렬되어 있기 때문에 한 면만 보입니다.</p> <p>한 번 이 물체에 애니메이션을 주고 3D처럼 보이는지 확인해보죠. 애니메이션을 구현하기 위해 <a href="https://developer.mozilla.org/en-US/docs/Web/API/window/requestAnimationFrame"><code class="notranslate" translate="no">requestAnimationFrame</code></a> 루프 안에서 렌더링 함수를 호출합니다.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function render(time) { time *= 0.001; // convert time to seconds cube.rotation.x = time; cube.rotation.y = time; renderer.render(scene, camera); requestAnimationFrame(render); } requestAnimationFrame(render); </pre> <p><code class="notranslate" translate="no">requestAnimationFrame</code>은 브라우저에 애니메이션 프레임을 요청하는 함수입니다. 인자로 실행할 함수를 전달하면 되죠(이 경우는 <code class="notranslate" translate="no">render</code> 함수). 브라우저는 넘겨받은 함수를 실행하고, 페이지에 변화가 있다면 페이지를 다시 렌더링합니다. 위 예제에서는 Three.js의 <code class="notranslate" translate="no">renderer.render</code> 메서드를 호출해 씬을 렌더링하도록 했죠.</p> <p><code class="notranslate" translate="no">requestAnimationFrame</code>은 매개변수로 넘겨받은 함수에 페이지가 로드된 이후의 시간값을 밀리초 단위로 넘겨줍니다. 전 초 단위가 더 익숙하기 때문에, 밀리초 단위를 초 단위로 변경하였습니다.</p> <p>그리고 정육면체의 X, Y축 회전값을 현재 시간값으로 설정합니다(이 회전값은 <a href="https://ko.wikipedia.org/wiki/%EB%9D%BC%EB%94%94%EC%95%88">라디안(radians)</a> 단위를 사용합니다). 360˚도는 2π 라디안이니 큐브는 각 축마다 약 6.28초에 한 바퀴를 돌게 됩니다.</p> <p>그리고 씬을 렌더링 한 후, 브라우저에 재귀적으로 애니메이션 프레임을 요청해 이 애니메이션이 반복되도록 합니다.</p> <p>마지막으로 루프 바깥에서 <code class="notranslate" translate="no">requestAnimationFrame</code>을 한 번 호출해 루프를 시작합니다.</p> <p></p><div translate="no" class="threejs_example_container notranslate"> <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/fundamentals-with-animation.html"></iframe></div> <a class="threejs_center" href="/manual/examples/fundamentals-with-animation.html" target="_blank">새 탭에서 보기</a> </div> <p></p> <p>아까보단 낫지만 아직 3D 물체라고 부르기엔 뭔가 부족합니다. 광원을 추가해 그림자가 지도록 하면 어떨까요? 나중에 <a href="lights.html">이 글</a>에서 자세히 다루겠지만, Three.js에는 다양한 종류의 광원이 있습니다. 다 살펴보기 힘드니 지금은 예시로 <a href="/docs/#api/ko/lights/DirectionalLight"><code class="notranslate" translate="no">DirectionalLight</code></a>를 사용해보도록 하죠.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">{ const color = 0xFFFFFF; const intensity = 3; const light = new THREE.DirectionalLight(color, intensity); light.position.set(-1, 2, 4); scene.add(light); } </pre> <p><code class="notranslate" translate="no">DirectionalLight</code>에는 <code class="notranslate" translate="no">위치(position)</code>와 <code class="notranslate" translate="no">목표(target)</code> 속성이 있습니다. 기본값은 <code class="notranslate" translate="no">0, 0, 0</code> 이죠. 먼저 <code class="notranslate" translate="no">position</code>을 <code class="notranslate" translate="no">-1, 2, 4</code>로 설정해 카메라보다 약간 동서쪽, Z축으로는 약간 위로 보냅니다. <code class="notranslate" translate="no">target</code>은 기본값 <code class="notranslate" translate="no">0, 0, 0</code> 그대로 두어 공간의 중앙을 비추도록 합니다.</p> <p>그리고 <code class="notranslate" translate="no">material</code>도 바꿔야 합니다. <a href="/docs/#api/ko/materials/MeshBasicMaterial"><code class="notranslate" translate="no">MeshBasicMaterial</code></a>은 광원에 반응하지 않으니, 광원에 반응하는 <a href="/docs/#api/ko/materials/MeshPhongMaterial"><code class="notranslate" translate="no">MeshPhongMaterial</code></a>로 바꿉니다(<code class="notranslate" translate="no">phong</code>은 광원 반사 모델을 처음 개발한 사람 이름. 역주).</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">-const material = new THREE.MeshBasicMaterial({color: 0x44aa88}); // greenish blue +const material = new THREE.MeshPhongMaterial({color: 0x44aa88}); // greenish blue </pre> <p>다음은 현재까지의 프로그램 구조를 도식화 한 것이고,</p> <div class="threejs_center"><img src="../resources/images/threejs-1cube-with-directionallight.svg" style="width: 500px;"></div> <p>아래는 결과물입니다.</p> <p></p><div translate="no" class="threejs_example_container notranslate"> <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/fundamentals-with-light.html"></iframe></div> <a class="threejs_center" href="/manual/examples/fundamentals-with-light.html" target="_blank">새 탭에서 보기</a> </div> <p></p> <p>이제 3D라고 불러도 어색하지 않네요.</p> <p>재미를 위해서 한 번 정육면체를 2개만 더 만들어 봅시다.</p> <p>미리 만들어 놨던 <code class="notranslate" translate="no">Geometry</code>를 사용하고, <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>만 바꿔 다른 색을 가진 큐브 2개를 만들겠습니다.</p> <p>먼저 함수를 하나 만들어보겠습니다. 이 함수는 넘겨받은 색상값으로 새로운 <a href="/docs/#api/ko/materials/Material"><code class="notranslate" translate="no">Material</code></a>을 만들고, 넘겨받은 <code class="notranslate" translate="no">Geometry</code>와 조합해 새로운 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>를 만듭니다. 그리고 씬에 추가한 후 넘겨받은 X축 값을 통해 물체를 이동시키죠.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function makeInstance(geometry, color, x) { const material = new THREE.MeshPhongMaterial({color}); const cube = new THREE.Mesh(geometry, material); scene.add(cube); cube.position.x = x; return cube; } </pre> <p>다음으로 3가지 다른 색상과 X축 값으로 이 함수를 호출해 결과물을 배열로 저장합니다.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const cubes = [ makeInstance(geometry, 0x44aa88, 0), makeInstance(geometry, 0x8844aa, -2), makeInstance(geometry, 0xaa8844, 2), ]; </pre> <p>마지막으로 <code class="notranslate" translate="no">render</code> 함수에서 3개의 정육면체를 회전시킵니다. 동적인 효과를 위해 각 큐브마다 약간 다른 값을 주도록 해보죠.</p> <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function render(time) { time *= 0.001; // convert time to seconds cubes.forEach((cube, ndx) => { const speed = 1 + ndx * .1; const rot = time * speed; cube.rotation.x = rot; cube.rotation.y = rot; }); ... </pre> <p>결과물은 다음과 같습니다.</p> <p></p><div translate="no" class="threejs_example_container notranslate"> <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/fundamentals-3-cubes.html"></iframe></div> <a class="threejs_center" href="/manual/examples/fundamentals-3-cubes.html" target="_blank">새 탭에서 보기</a> </div> <p></p> <p>아까 위에서 좌표로 도식한 그림과 위 결과물을 비교해보면, 예상대로 X축으로 -2, +2만큼 이동한 정육면체들의 일부가 절두체에서 약간 벗어났습니다. 또한 가운데 정육면체에 비해 굉장히 굴절되어 보이는데, 이는 우리가 설정한 시야각이 너무 좁은 탓입니다.</p> <p>위 프로그램은 구조는 다음과 같습니다.</p> <div class="threejs_center"><img src="../resources/images/threejs-3cubes-scene.svg" style="width: 610px;"></div> <p>그림을 보면 각 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a> 객체는 같은 <a href="/docs/#api/ko/geometries/BoxGeometry"><code class="notranslate" translate="no">BoxGeometry</code></a>를 참조합니다. 그러나 각 <a href="/docs/#api/ko/objects/Mesh"><code class="notranslate" translate="no">Mesh</code></a>는 다른 <a href="/docs/#api/ko/materials/MeshPhongMaterial"><code class="notranslate" translate="no">MeshPhongMaterial</code></a>을 참조하므로 다른 색을 띄죠.</p> <p>이 인트로가 Three.js를 시작하는 데 도움이 되었으면 합니다. <a href="responsive.html">다음 장에서는 코드를 반응형으로 만들어 다양한 상황에 적용할 수 있도록 해 볼 것입니다</a>.</p> <div id="es6" class="threejs_bottombar"> <h3>es6 모듈, Three.js, 프로젝트 구조</h3> <p>Three.js r106 릴리즈 이후에서는 three.js를 <a href="https://developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Statements/import">es6 모듈</a>로 사용하길 권장합니다.</p> <p> es6 모듈은 js 파일이나 인라인 <code class="notranslate" translate="no"><script type="module"></code> 태그 안에서 <code class="notranslate" translate="no">import</code> 키워드를 사용해 로드할 수 있습니다. </p> <pre class="prettyprint"><script type="module"> import * as THREE from 'three'; ... </script> </pre> <p> 경로는 절대경로나 상대경로를 사용해야 하며, 상대경로는 <code class="notranslate" translate="no"><img></code>, <code class="notranslate" translate="no"><a></code> 태그와 달리 <code class="notranslate" translate="no">./</code> 또는 <code class="notranslate" translate="no">../</code>로 시작해야 합니다. </p> <p> 만약 절대경로가 같다면 같은 script는 한 번만 로드됩니다. Three.js 기반 프로젝트의 폴더 구조를 잘 구성해야 하는 가장 큰 이유이죠. </p> <pre class="dos">someFolder | ├-build | | | +-three.module.js | +-examples | +-jsm | +-controls | | | +-OrbitControls.js | +-TrackballControls.js | +-... | +-loaders | | | +-GLTFLoader.js | +-... | ... </pre> <p> 이유인 즉 <a href="https://github.com/mrdoob/three.js/blob/master/examples/jsm/controls/OrbitControls.js"><code class="notranslate" translate="no">OrbitControls.js</code></a> 같은 코드는 다음처럼 상대경로로 모듈을 로드하기 때문에 </p> <pre class="prettyprint">import * as THREE from '../../../build/three.module.js'; </pre> <p> 같은 폴더 구조를 사용함으로써 모든 예제 스크립트가 같은 three.module.js 파일을 참조하도록 할 수 있죠. </p> <pre class="prettyprint">import * as THREE from './someFolder/build/three.module.js'; import {OrbitControls} from './someFolder/addons/controls/OrbitControls.js'; </pre> <p>아래는 CDN을 사용하는 예시입니다. <code class="notranslate" translate="no">three.modules.js</code>의 경로가 <code class="notranslate" translate="no">/build/three.modules.js</code> 로 끝나야 한다는 것을 명심하세요.</p> <pre class="prettyprint">import * as THREE from 'https://unpkg.com/three@<version>/<b>build/three.module.js</b>'; import {OrbitControls} from 'https://unpkg.com/three@<version>/addons/controls/OrbitControls.js'; </pre> </div> </div> </div> </div> <script src="../resources/prettify.js"></script> <script src="../resources/lesson.js"></script> </body></html>
Close