game.html 91 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658
  1. <!DOCTYPE html><html lang="ko"><head>
  2. <meta charset="utf-8">
  3. <title>๋กœ ๊ฒŒ์ž„ ๋งŒ๋“ค๊ธฐ</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 โ€“ ๋กœ ๊ฒŒ์ž„ ๋งŒ๋“ค๊ธฐ">
  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. <!-- Import maps polyfill -->
  14. <!-- Remove this when import maps will be widely supported -->
  15. <script async src="https://unpkg.com/es-module-shims@1.6.3/dist/es-module-shims.js"></script>
  16. <script type="importmap">
  17. {
  18. "imports": {
  19. "three": "../../build/three.module.js"
  20. }
  21. }
  22. </script>
  23. <link rel="stylesheet" href="/manual/ko/lang.css">
  24. </head>
  25. <body>
  26. <div class="container">
  27. <div class="lesson-title">
  28. <h1>๋กœ ๊ฒŒ์ž„ ๋งŒ๋“ค๊ธฐ</h1>
  29. </div>
  30. <div class="lesson">
  31. <div class="lesson-main">
  32. <p>์ œ๊ฐ€ ๊ฝค ๋งŽ์ด ๋ฐ›์•˜๋˜ ์งˆ๋ฌธ ์ค‘ ํ•˜๋‚˜๊ฐ€ Three.js๋กœ ๊ฒŒ์ž„์„ ๋งŒ๋“œ๋Š” ๋ฐฉ๋ฒ•์— ๊ด€ํ•œ ๊ฒƒ์ด์—ˆ์Šต๋‹ˆ๋‹ค. ๊ธฐ์ดˆ์ ์ธ ๊ฒƒ์ด๊ธด ํ•ด๋„ ๋ถ€๋”” ์ด ๊ธ€์— ์—ฌ๋Ÿฌ๋ถ„์ด ์›ํ–ˆ๋˜ ๋‚ด์šฉ์ด ์žˆ๋‹ค๋ฉด ์ข‹๊ฒ ๋„ค์š”.</p>
  33. <p>๊ธ€์„ ์“ฐ๋Š” ํ˜„์žฌ๋ฅผ ๊ธฐ์ค€์œผ๋กœ, ์•„๋งˆ ์ด ๊ธ€์ด ์ด ์‹œ๋ฆฌ์ฆˆ์—์„œ ๊ฐ€์žฅ ๊ธด ๊ธ€์ด ๋  ๊ฒƒ ๊ฐ™์Šต๋‹ˆ๋‹ค. ์˜ˆ์ œ๋กœ ์“ด ์ฝ”๋“œ๊ฐ€ ์ง€๋‚˜์น˜๊ฒŒ ์ „๋ฌธ์ ์œผ๋กœ ๋ณด์ผ ์ˆ˜๋„ ์žˆ์ง€๋งŒ ๊ทธ๊ฑด ์˜ˆ์ œ๋ฅผ ๋งŒ๋“ค๋ฉฐ ๋ฌธ์ œ๊ฐ€ ์ƒ๊ธธ ๋•Œ๋งˆ๋‹ค ์ด์ „์— ์ œ๊ฐ€ ์‹ค์ œ๋กœ ๋งŒ๋“ค์—ˆ๋˜ ๊ฒŒ์ž„์˜ ์ฝ”๋“œ๋ฅผ ๊ฐ€์ ธ์™€์„œ ๊ทธ๋ ‡์Šต๋‹ˆ๋‹ค. ๋˜ํ•œ ์™œ ์ด๋Ÿฐ ํ•ด๊ฒฐ์ฑ…์„ ์ผ๋Š”์ง€ ์ตœ๋Œ€ํ•œ ์ ์œผ๋ ค๊ณ  ํ–ˆ์œผ๋‹ˆ ๊ธธ ์ˆ˜๋ฐ–์— ์—†์ฃ . ๋ฌผ๋ก  ๋งŒ๋“ค๋ ค๋Š” ๊ฒŒ์ž„์˜ ๊ทœ๋ชจ๊ฐ€ ์ž‘๋‹ค๋ฉด ์ด๋Ÿฐ ํ•ด๊ฒฐ์ฑ…์ด ์ „๋ถ€ ํ•„์š” ์—†์„ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์˜ˆ์ œ๋กœ ๊ตฌํ˜„ํ•œ ๊ฒƒ๋„ ๊ต‰์žฅํžˆ ๊ฐ„๋‹จํ•œ ๊ฒŒ์ž„์— ์†ํ•ฉ๋‹ˆ๋‹ค. ํ†ต์ƒ์ ์œผ๋กœ 3D ์บ๋ฆญํ„ฐ๊ฐ€ 2D ์บ๋ฆญํ„ฐ๋ณด๋‹ค ๋” ๋ณต์žกํ•˜๋‹ˆ ์ฒ˜๋ฆฌํ•ด์ค˜์•ผ ํ•  ๊ฒƒ์ด ๋งŽ์„ ์ˆ˜๋ฐ–์— ์—†์ฃ .</p>
  34. <p>ํŒฉ๋งจ(PacMan)์„ 2D๋กœ ๊ตฌํ˜„ํ•œ๋‹ค๋ฉด ํŒฉ๋งจ์ด ์ฝ”๋„ˆ๋ฅผ ๋Œ ๋•Œ ๋ฐ”๋กœ 90๋„ ๊บพ๊ธฐ๋งŒ ํ•˜๋ฉด ๋ฉ๋‹ˆ๋‹ค. ํ”„๋ ˆ์ž„ ์‚ฌ์ด์— ๋”ฐ๋กœ ์ฒ˜๋ฆฌํ•ด์ค˜์•ผ ํ•  ๊ฒƒ์ด ์—†์ฃ . ํ•˜์ง€๋งŒ 3D์˜ ์„ธ๊ณ„์—์„œ๋Š” ๋ฐ”๋กœ ๋ฐฉํ–ฅ์„ ํ‹€๊ธฐ๋ณด๋‹ค ๋ช‡ ํ”„๋ ˆ์ž„์— ๊ฑธ์ณ ์„œ์„œํžˆ ๋ฐฉํ–ฅ์„ ํŠธ๋Š” ๊ฒŒ ์ผ๋ฐ˜์ ์ž…๋‹ˆ๋‹ค. ์•„์ฃผ ๊ฐ„๋‹จํ•œ ์ฐจ์ด์ ์ด์ง€๋งŒ, ์ด๊ฒƒ ๋•Œ๋ฌธ์— ์ž‘์—…์ด ํ›จ์”ฌ ๋ณต์žกํ•ด์ง‘๋‹ˆ๋‹ค.</p>
  35. <p>์ด ๊ธ€์—์„œ ๋‹ค๋ฃฐ ๋‚ด์šฉ์€ Three.js์— ๊ด€ํ•œ ๊ฒƒ์ด๋ผ๊ณ  ๋ณด๊ธฐ ์–ด๋ ต์Šต๋‹ˆ๋‹ค. ์™œ๋ƒํ•˜๋ฉด <strong>Three.js๋Š” ๊ฒŒ์ž„ ์—”์ง„์ด ์•„๋‹ˆ๊ธฐ ๋•Œ๋ฌธ</strong>์ด์ฃ . Three.js๋Š” 3D ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ์ž…๋‹ˆ๋‹ค. 3D ์š”์†Œ๋ฅผ ๊ณ„์—ดํ™”ํ•˜๋Š” <a href="scenegraph.html">์”ฌ ๊ทธ๋ž˜ํ”„</a>์™€ 3D ์š”์†Œ๋ฅผ ๋ Œ๋”๋งํ•˜๋„๋ก ๋„์™€์ฃผ๋Š” ๊ธฐ๋Šฅ ๋“ฑ์„ ์ œ๊ณตํ•˜์ฃ . ํ•˜์ง€๋งŒ ๊ฒŒ์ž„๊ณผ ๊ด€๋ จํ•œ ๊ธฐ๋Šฅ์€ ์ง€์›ํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค. ์ถฉ๋Œ(collision), ๋ฌผ๋ฆฌ(physics), ์ž…๋ ฅ ์‹œ์Šคํ…œ, ํŒจ์Šค ํŒŒ์ธ๋”ฉ(path finding) ๋“ฑ๋“ฑ.. ์ด๋Ÿฐ ๊ธฐ๋Šฅ์€ ์ง์ ‘ ๋งŒ๋“ค์–ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.</p>
  36. <p>๊ฒฐ๊ตญ ์ด ๊ธ€์˜ <em>๋ฏธ์™„์„ฑ</em> ๊ฒŒ์ž„์„ ๋งŒ๋“œ๋Š” ๋ฐ ๊ฝค ๋งŽ์€ ์ฝ”๋“œ๋ฅผ ์ผ์Šต๋‹ˆ๋‹ค. ์•„๊นŒ ๋งํ–ˆ๋“ฏ ์ œ๊ฐ€ ์ฝ”๋“œ๋ฅผ ๋„ˆ๋ฌด ์ง€๋‚˜์น˜๊ฒŒ ์งฐ์„ ์ˆ˜๋„ ์žˆ๊ณ , ๋” ๊ฐ„๋‹จํ•œ ํ•ด๊ฒฐ์ฑ…์ด ์žˆ์„ ์ˆ˜๋„ ์žˆ์œผ๋‚˜, ์ €๋Š” ๊ธ€์„ ๋งˆ๋ฌด๋ฆฌํ•œ ์ง€๊ธˆ๋„ ์ถฉ๋ถ„ํžˆ ๋งŽ์€ ์ฝ”๋“œ๋ฅผ ์ผ๋Š”์ง€, ์„ค๋ช…์„ ๋น ๋œจ๋ฆฐ ๊ฒƒ์ด ์—†๋Š”์ง€ ๊ฑฑ์ •๋ฉ๋‹ˆ๋‹ค.</p>
  37. <p>์ด ๊ธ€์—์„œ ์“ด ๋ฐฉ๋ฒ•์€ ๋Œ€๋ถ€๋ถ„ <a href="https://unity.com">์œ ๋‹ˆํ‹ฐ(Unity) ์—”์ง„</a>์˜ ์˜ํ–ฅ์„ ํฌ๊ฒŒ ๋ฐ›์•˜์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์œ ๋‹ˆํ‹ฐ๋ฅผ ์ž˜ ๋ชจ๋ฅธ๋‹ค๊ณ  ํ•ด์„œ ์ด ๊ธ€์„ ์ฝ๋Š” ๊ฒŒ ์–ด๋ ต์ง„ ์•Š์„ ๊ฒ๋‹ˆ๋‹ค. 1000๊ฐœ์˜ ๊ธฐ๋Šฅ์ด ์žˆ๋‹ค๋ฉด ๊ทธ ์ค‘ 10๊ฐœ ์ •๋„ ๋ฐ–์— ์“ฐ์ง€ ์•Š์•˜๊ฑฐ๋“ ์š”.</p>
  38. <p>๋จผ์ € Three.js ๋ถ€๋ถ„๋ถ€ํ„ฐ ์‹œ์ž‘ํ•ด๋ด…์‹œ๋‹ค. ๊ฒŒ์ž„์— ์“ธ ๋ชจ๋ธ๋“ค๋ถ€ํ„ฐ ์ฐพ์•„๋ณด์ฃ .</p>
  39. <p><a href="https://opengameart.org">opengameart.org</a> ์‚ฌ์ดํŠธ์—์„œ <a href="https://opengameart.org/users/quaternius">quaternius</a> ์ž‘๊ฐ€์˜ <a href="https://opengameart.org/content/lowpoly-animated-knight">์›€์ง์ด๋Š” ๊ธฐ์‚ฌ ๋ชจ๋ธ</a>์„ ์ฐพ์•˜์Šต๋‹ˆ๋‹ค.</p>
  40. <div class="threejs_center"><img src="../resources/images/knight.jpg" style="width: 375px;"></div>
  41. <p><a href="https://opengameart.org/users/quaternius">๊ฐ™์€ ์ž‘๊ฐ€</a>๊ฐ€ ๋งŒ๋“  ์ž‘ํ’ˆ ์ค‘์— <a href="https://opengameart.org/content/lowpoly-animated-farm-animal-pack">์›€์ง์ด๋Š” ๋™๋ฌผ๋“ค</a>๋„ ์žˆ๋”๊ตฐ์š”.</p>
  42. <div class="threejs_center"><img src="../resources/images/animals.jpg" style="width: 606px;"></div>
  43. <p>์ด ๋ชจ๋ธ๋“ค๋กœ ๊ฝค ๊ดœ์ฐฎ์€ ๊ฒŒ์ž„์„ ๋งŒ๋“ค ์ˆ˜ ์žˆ์„ ๊ฒƒ ๊ฐ™์Šต๋‹ˆ๋‹ค. ๋ชจ๋ธ๋“ค์„ ๋ถˆ๋Ÿฌ์™€๋ณด์ฃ .</p>
  44. <p><a href="load-gltf.html">glTF ํŒŒ์ผ ๋ถˆ๋Ÿฌ์˜ค๊ธฐ</a>์— ๋Œ€ํ•ด์„œ๋Š” ์ด์ „์— ๋‹ค๋ค˜์—ˆ์Šต๋‹ˆ๋‹ค. ๋™์ผํ•œ ๋ฐฉ๋ฒ•์„ ์‚ฌ์šฉํ•˜์ง€๋งŒ ์ด๋ฒˆ์—๋Š” ๋ชจ๋ธ์ด ์—ฌ๋Ÿฌ ๊ฐœ์ด๊ธฐ๋„ ํ•˜๊ณ , ๋ชจ๋ธ์„ ์ „๋ถ€ ๋ถˆ๋Ÿฌ์˜ค๊ธฐ ์ „์— ๊ฒŒ์ž„์„ ์‹œ์ž‘ํ•ด์„  ์•ˆ ๋ฉ๋‹ˆ๋‹ค.</p>
  45. <p>์ด๋Ÿฐ ๊ฒฝ์šฐ๋ฅผ ๋Œ€๋น„ํ•ด Three.js๋Š” <a href="/docs/#api/ko/loaders/managers/LoadingManager"><code class="notranslate" translate="no">LoadingManager</code></a>๋ฅผ ์ œ๊ณตํ•ฉ๋‹ˆ๋‹ค. <a href="/docs/#api/ko/loaders/managers/LoadingManager"><code class="notranslate" translate="no">LoadingManager</code></a>์˜ ์ธ์Šคํ„ด์Šค๋ฅผ ์ƒ์„ฑํ•ด ๋‹ค๋ฅธ ๋กœ๋”(loader)์— ๋„˜๊ฒจ์ฃผ๊ธฐ๋ฉด ๋˜์ฃ . <a href="/docs/#api/ko/loaders/managers/LoadingManager"><code class="notranslate" translate="no">LoadingManager</code></a>์˜ <a href="/docs/#api/ko/loaders/managers/LoadingManager#onProgress"><code class="notranslate" translate="no">onProgress</code></a>์™€ <a href="/docs/#api/ko/loaders/managers/LoadingManager#onLoad"><code class="notranslate" translate="no">onLoad</code></a> ์†์„ฑ์— ์ฝœ๋ฐฑ ํ•จ์ˆ˜๋ฅผ ์ง€์ •ํ•˜๋ฉด ๋˜๋Š”๋ฐ, <a href="/docs/#api/ko/loaders/managers/LoadingManager#onLoad"><code class="notranslate" translate="no">onLoad</code></a>๋Š” ๋ชจ๋“  ํŒŒ์ผ์„ ๋ถˆ๋Ÿฌ์˜จ ๋’ค ํ˜ธ์ถœํ•˜๊ณ , <a href="/docs/#api/ko/loaders/managers/LoadingManager#onProgress"><code class="notranslate" translate="no">onProgress</code></a>๋Š” ๊ฐ ํŒŒ์ผ์„ ๋ถˆ๋Ÿฌ์™”์„ ๋•Œ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค. <a href="/docs/#api/ko/loaders/managers/LoadingManager#onProgress"><code class="notranslate" translate="no">onProgress</code></a>๋ฅผ ์ด์šฉํ•˜๋ฉด ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ๋ณด์—ฌ์ค„ ์ˆ˜ ์žˆ์ฃ .</p>
  46. <p><a href="load-gltf.html">glTF ํŒŒ์ผ ๋ถˆ๋Ÿฌ์˜ค๊ธฐ</a> ์˜ˆ์ œ๋ฅผ ๊ฐ€์ ธ์™€ ์นด๋ฉ”๋ผ ์ ˆ๋‘์ฒด(frustum)๋ฅผ ์กฐ์ •ํ•˜๋Š” ์ฝ”๋“œ๋ฅผ ์ง€์šฐ๊ณ  ์•„๋ž˜ ์ฝ”๋“œ๋ฅผ ์ถ”๊ฐ€ํ•ฉ๋‹ˆ๋‹ค.</p>
  47. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const manager = new THREE.LoadingManager();
  48. manager.onLoad = init;
  49. const models = {
  50. pig: { url: 'resources/models/animals/Pig.gltf' },
  51. cow: { url: 'resources/models/animals/Cow.gltf' },
  52. llama: { url: 'resources/models/animals/Llama.gltf' },
  53. pug: { url: 'resources/models/animals/Pug.gltf' },
  54. sheep: { url: 'resources/models/animals/Sheep.gltf' },
  55. zebra: { url: 'resources/models/animals/Zebra.gltf' },
  56. horse: { url: 'resources/models/animals/Horse.gltf' },
  57. knight: { url: 'resources/models/knight/KnightCharacter.gltf' },
  58. };
  59. {
  60. const gltfLoader = new GLTFLoader(manager);
  61. for (const model of Object.values(models)) {
  62. gltfLoader.load(model.url, (gltf) =&gt; {
  63. model.gltf = gltf;
  64. });
  65. }
  66. }
  67. function init() {
  68. // ๋‚˜์ค‘์— ์ž‘์„ฑํ•  ์˜ˆ์ •
  69. }
  70. </pre>
  71. <p>์œ„ ์ฝ”๋“œ๋Š” <code class="notranslate" translate="no">models</code> ๊ฐ์ฒด์— ์žˆ๋Š” ํŒŒ์ผ์„ ๋ถˆ๋Ÿฌ์˜ค๊ณ , <a href="/docs/#api/ko/loaders/managers/LoadingManager"><code class="notranslate" translate="no">LoadingManager</code></a>๊ฐ€ ํŒŒ์ผ์„ ์ „๋ถ€ ๋ถˆ๋Ÿฌ์™”์„ ๋•Œ <code class="notranslate" translate="no">init</code> ํ•จ์ˆ˜๋ฅผ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค. <code class="notranslate" translate="no">models</code> ๊ฐ์ฒด๋ฅผ ์ „์—ญ์œผ๋กœ ์„ ์–ธํ•œ ๊ฑด ๋‚˜์ค‘์— <a href="/docs/#examples/loaders/GLTFLoader"><code class="notranslate" translate="no">GLTFLoader</code></a>์˜ ์ฝœ๋ฐฑ์„ ์ด์šฉํ•ด ๊ฐ ๋ชจ๋ธ์˜ ์ •๋ณด๋ฅผ ์‚ฌ์šฉํ•  ๋•Œ ๋ถˆ๋Ÿฌ์˜จ ๊ฐ ๋ชจ๋ธ์— ์ ‘๊ทผํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•˜๊ธฐ ์œ„ํ•จ์ž…๋‹ˆ๋‹ค.</p>
  72. <p>๋ชจ๋“  ๋ชจ๋ธ๊ณผ ๋ชจ๋ธ์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜ ๋ฐ์ดํ„ฐ๋Š” ํ˜„์žฌ ์•ฝ 6.6MB์ž…๋‹ˆ๋‹ค. ๊ฝค ์šฉ๋Ÿ‰์ด ํฌ๋„ค์š”. ์—ฌ๋Ÿฌ๋ถ„์˜ ์„œ๋ฒ„๊ฐ€ ์••์ถ•์„ ์ง€์›(์ด ์‚ฌ์ดํŠธ์˜ ์›น ์„œ๋ฒ„๊ฐ€ ์ด ๊ธฐ๋Šฅ์„ ์ง€์›ํ•ฉ๋‹ˆ๋‹ค)ํ•œ๋‹ค๋ฉด ์šฉ๋Ÿ‰์€ ์•ฝ 1.4MB๊นŒ์ง€ ์ค„์–ด๋“ค ๊ฒ๋‹ˆ๋‹ค. 6.6MB์™€ ๋น„๊ตํ•˜๋ฉด ํ™•์‹คํžˆ ์ ์€ ๋ฐ์ดํ„ฐ์ง€๋งŒ ์ ˆ๋Œ€ ์ž‘์€ ๋ฐ์ดํ„ฐ๋Š” ์•„๋‹™๋‹ˆ๋‹ค. ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ๋งŒ๋“ค์–ด ์‚ฌ์šฉ์ž์—๊ฒŒ ์–ผ๋งˆ๋‚˜ ๊ธฐ๋‹ค๋ ค์•ผ ํ•˜๋Š”์ง€๋ฅผ ํ‘œ์‹œํ•ด์ค€๋‹ค๋ฉด ์ข‹๊ฒ ๋„ค์š”.</p>
  73. <p><a href="/docs/#api/ko/loaders/managers/LoadingManager#onProgress"><code class="notranslate" translate="no">onProgress</code></a>์— ์ฝœ๋ฐฑ ํ•จ์ˆ˜๋ฅผ ์ง€์ •ํ•ด์ค์‹œ๋‹ค. ์ด ์ฝœ๋ฐฑ ํ•จ์ˆ˜๋Š” ํ˜ธ์ถœํ•  ๋•Œ 3๊ฐœ์˜ ๋งค๊ฐœ๋ณ€์ˆ˜๋ฅผ ๋ฐ›์Šต๋‹ˆ๋‹ค. ๊ฐ๊ฐ ๋งˆ์ง€๋ง‰์— ๋ถˆ๋Ÿฌ์˜จ ํŒŒ์ผ์˜ <code class="notranslate" translate="no">url</code>, ๊ทธ๋ฆฌ๊ณ  ๋ถˆ๋Ÿฌ์˜จ ํŒŒ์ผ์˜ ๊ฐœ์ˆ˜, ์ „์ฒด ํŒŒ์ผ์˜ ๊ฐœ์ˆ˜์ž…๋‹ˆ๋‹ค.</p>
  74. <p>ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋Š” HTML๋กœ ๊ฐ„๋‹จํžˆ ๊ตฌํ˜„ํ•˜๋„๋ก ํ•˜์ฃ .</p>
  75. <pre class="prettyprint showlinemods notranslate lang-html" translate="no">&lt;body&gt;
  76. &lt;canvas id="c"&gt;&lt;/canvas&gt;
  77. + &lt;div id="loading"&gt;
  78. + &lt;div&gt;
  79. + &lt;div&gt;...loading...&lt;/div&gt;
  80. + &lt;div class="progress"&gt;&lt;div id="progressbar"&gt;&lt;/div&gt;&lt;/div&gt;
  81. + &lt;/div&gt;
  82. + &lt;/div&gt;
  83. &lt;/body&gt;
  84. </pre>
  85. <p><code class="notranslate" translate="no">#progressbar</code>๋ฅผ ์ฐธ์กฐํ•œ ๋’ค <code class="notranslate" translate="no">width</code>๋ฅผ ํผ์„ผํŠธ(%) ๋‹จ์œ„๋กœ ํ‘œ์‹œํ•ด ํ˜„์žฌ ์ง„ํ–‰์œจ์„ ๋ณด์—ฌ์ค„ ๊ฒ๋‹ˆ๋‹ค. ์ฝœ๋ฐฑ์—์„œ ์ด ์Šคํƒ€์ผ๋งŒ ์ฒ˜๋ฆฌํ•ด์ฃผ๋ฉด ๋˜๊ฒ ๋„ค์š”.</p>
  86. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const manager = new THREE.LoadingManager();
  87. manager.onLoad = init;
  88. +const progressbarElem = document.querySelector('#progressbar');
  89. +manager.onProgress = (url, itemsLoaded, itemsTotal) =&gt; {
  90. + progressbarElem.style.width = `${ itemsLoaded / itemsTotal * 100 | 0 }%`;
  91. +};
  92. </pre>
  93. <p>์ด๋ฏธ ๋ชจ๋“  ๋ชจ๋ธ์„ ๋ถˆ๋Ÿฌ์™”์„ ๋•Œ <code class="notranslate" translate="no">init</code> ํ•จ์ˆ˜๋ฅผ ํ˜ธ์ถœํ•˜๊ฒŒ ํ•ด๋†“์•˜์œผ๋‹ˆ, ์—ฌ๊ธฐ์„œ <code class="notranslate" translate="no">#loading</code> ์š”์†Œ๋ฅผ ์ˆจ๊ฒจ ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์—†์•ฑ๋‹ˆ๋‹ค.</p>
  94. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function init() {
  95. + // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  96. + const loadingElem = document.querySelector('#loading');
  97. + loadingElem.style.display = 'none';
  98. }
  99. </pre>
  100. <p>์•„๋ž˜๋Š” ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ๊พธ๋ฏธ๊ธฐ ์œ„ํ•œ CSS์ž…๋‹ˆ๋‹ค. <code class="notranslate" translate="no">#loading</code>์€ ํŽ˜์ด์ง€ ์ „์ฒด๋ฅผ ๊ฝ‰ ์ฑ„์šฐ๊ณ  ์ž์‹ ์š”์†Œ๋ฅผ ๊ฐ€์šด๋ฐ ์ •๋ ฌ์‹œํ‚ต๋‹ˆ๋‹ค. ๊ทธ๋ฆฌ๊ณ  <code class="notranslate" translate="no">.progress</code>๋Š” ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๊ฐ€ ๋“ค์–ด๊ฐˆ ์˜์—ญ์„ ์ •์˜ํ•˜์ฃ . ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”์— ๊ฐ„๋‹จํ•œ ์• ๋‹ˆ๋ฉ”์ด์…˜๋„ ๋„ฃ์–ด์คฌ์Šต๋‹ˆ๋‹ค.</p>
  101. <pre class="prettyprint showlinemods notranslate lang-css" translate="no">#loading {
  102. position: absolute;
  103. left: 0;
  104. top: 0;
  105. width: 100%;
  106. height: 100%;
  107. display: flex;
  108. align-items: center;
  109. justify-content: center;
  110. text-align: center;
  111. font-size: xx-large;
  112. font-family: sans-serif;
  113. }
  114. #loading&gt;div&gt;div {
  115. padding: 2px;
  116. }
  117. .progress {
  118. width: 50vw;
  119. border: 1px solid black;
  120. }
  121. #progressbar {
  122. width: 0;
  123. transition: width ease-out .5s;
  124. height: 1em;
  125. background-color: #888;
  126. background-image: linear-gradient(
  127. -45deg,
  128. rgba(255, 255, 255, .5) 25%,
  129. transparent 25%,
  130. transparent 50%,
  131. rgba(255, 255, 255, .5) 50%,
  132. rgba(255, 255, 255, .5) 75%,
  133. transparent 75%,
  134. transparent
  135. );
  136. background-size: 50px 50px;
  137. animation: progressanim 2s linear infinite;
  138. }
  139. @keyframes progressanim {
  140. 0% {
  141. background-position: 50px 50px;
  142. }
  143. 100% {
  144. background-position: 0 0;
  145. }
  146. }
  147. </pre>
  148. <p>์ด์ œ ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๊ฐ€ ์ƒ๊ฒผ์œผ๋‹ˆ ๋ชจ๋ธ์„ ์ฒ˜๋ฆฌํ•  ์ฐจ๋ก€์ž…๋‹ˆ๋‹ค. ์ด ๋ชจ๋ธ๋“ค์—๋Š” ์• ๋‹ˆ๋ฉ”์ด์…˜์ด ์žˆ๋Š”๋ฐ, ์ด ์• ๋‹ˆ๋ฉ”์ด์…˜์„ ์ œ์–ดํ•  ์ˆ˜ ์—†๋‹ค๋ฉด ์• ๋‹ˆ๋ฉ”์ด์…˜์ด ์žˆ๋Š” ์˜๋ฏธ๊ฐ€ ์—†๊ฒ ์ฃ . Three.js๋Š” ๊ธฐ๋ณธ์ ์œผ๋กœ ์• ๋‹ˆ๋ฉ”์ด์…˜๋“ค์„ ๋ฐฐ์—ด ํ˜•ํƒœ๋กœ ์ €์žฅํ•ฉ๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ๋ฐฐ์—ด์ด ์•„๋‹Œ ์ด๋ฆ„ ํ˜•ํƒœ๋กœ ์ €์žฅํ•˜๋Š” ๊ฒŒ ๋‚˜์ค‘์— ์“ฐ๊ธฐ์— ํŽธํ•˜๋‹ˆ ๊ฐ ๋ชจ๋ธ์— <code class="notranslate" translate="no">animation</code> ์†์„ฑ์„ ๋งŒ๋“ค๊ฒ ์Šต๋‹ˆ๋‹ค. ๋ฌผ๋ก  ๊ฐ ์• ๋‹ˆ๋ฉ”์ด์…˜์˜ ์ด๋ฆ„์€ ๊ณ ์œ ํ•œ ๊ฐ’์ด์–ด์•ผ ํ•˜๊ฒ ์ฃ .</p>
  149. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">+function prepModelsAndAnimations() {
  150. + Object.values(models).forEach(model =&gt; {
  151. + const animsByName = {};
  152. + model.gltf.animations.forEach((clip) =&gt; {
  153. + animsByName[clip.name] = clip;
  154. + });
  155. + model.animations = animsByName;
  156. + });
  157. +}
  158. function init() {
  159. // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  160. const loadingElem = document.querySelector('#loading');
  161. loadingElem.style.display = 'none';
  162. + prepModelsAndAnimations();
  163. }
  164. </pre>
  165. <p>์ด์ œ ์• ๋‹ˆ๋ฉ”์ด์…˜์ด ๋“ค์–ด๊ฐ„ ๋ชจ๋ธ์„ ํ™”๋ฉด์— ๋„์›Œ๋ด…์‹œ๋‹ค.</p>
  166. <p><a href="load-gltf.html">์ด์ „ glTF ํŒŒ์ผ ์˜ˆ์ œ</a>์™€ ๋‹ฌ๋ฆฌ ์ด๋ฒˆ์—๋Š” ๊ฐ ๋ชจ๋ธ์„ ํ•˜๋‚˜ ์ด์ƒ ๋ฐฐ์น˜ํ•  ๊ณ„ํš์ž…๋‹ˆ๋‹ค. ๊ทธ๋Ÿฌ๋‹ˆ ํŒŒ์ผ์„ ๋ถˆ๋Ÿฌ์˜จ ๋’ค ๋ฐ”๋กœ ์žฅ๋ฉด์— ๋„ฃ๋Š” ๋Œ€์‹  ๊ฐ glTF์˜ ์”ฌ ๊ทธ๋ž˜ํ”„(scene), ์ด ๊ฒฝ์šฐ์—๋Š” ์›€์ง์ด๋Š” ์บ๋ฆญํ„ฐ๋ฅผ ๋ณต์‚ฌํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ๋‹คํ–‰ํžˆ Three.js์—๋Š” <code class="notranslate" translate="no">SkeletonUtil.clone</code>์ด๋ผ๋Š” ํ•จ์ˆ˜๊ฐ€ ์žˆ์–ด ์ด๋ฅผ ์‰ฝ๊ฒŒ ๊ตฌํ˜„ํ•  ์ˆ˜ ์žˆ์ฃ . ๋จผ์ € ํ•ด๋‹น ๋ชจ๋“ˆ์„ ๋ถˆ๋Ÿฌ์˜ค๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  167. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">import * as THREE from 'three';
  168. import { OrbitControls } from 'three/addons/controls/OrbitControls.js';
  169. import { GLTFLoader } from 'three/addons/loaders/GLTFLoader.js';
  170. +import { SkeletonUtils } from 'three/addons/utils/SkeletonUtils.js';
  171. </pre>
  172. <p>๊ทธ๋ฆฌ๊ณ  ์•„๊นŒ ๋ถˆ๋Ÿฌ์™”๋˜ ๋ชจ๋ธ์„ ๋ณต์‚ฌํ•ฉ๋‹ˆ๋‹ค.</p>
  173. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function init() {
  174. // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  175. const loadingElem = document.querySelector('#loading');
  176. loadingElem.style.display = 'none';
  177. prepModelsAndAnimations();
  178. + Object.values(models).forEach((model, ndx) =&gt; {
  179. + const clonedScene = SkeletonUtils.clone(model.gltf.scene);
  180. + const root = new THREE.Object3D();
  181. + root.add(clonedScene);
  182. + scene.add(root);
  183. + root.position.x = (ndx - 3) * 3;
  184. + });
  185. }
  186. </pre>
  187. <p>์œ„ ์ฝ”๋“œ์—์„œ๋Š” ๋ถˆ๋Ÿฌ์˜จ ๊ฐ ๋ชจ๋ธ์˜ <code class="notranslate" translate="no">gltf.scene</code>์„ ๋ณต์‚ฌํ•ด ์ƒˆ๋กœ์šด <a href="/docs/#api/ko/core/Object3D"><code class="notranslate" translate="no">Object3D</code></a>์˜ ์ž์‹์œผ๋กœ ์ถ”๊ฐ€ํ–ˆ์Šต๋‹ˆ๋‹ค. ๋ถ€๋ชจ๋ฅผ ๋”ฐ๋กœ ๋งŒ๋“  ๊ฑด ๋ชจ๋ธ์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜์ด ๋ชจ๋ธ์˜ ๊ฐ ์š”์†Œ์˜ ์œ„์น˜๊ฐ’์— ์˜ํ–ฅ์„ ๋ฏธ์น˜๊ธฐ์— ์ฝ”๋“œ๋กœ ์ง์ ‘ ์œ„์น˜๊ฐ’์„ ์ˆ˜์ •ํ•˜๊ธฐ๋„ ์–ด๋ ต๊ณ , ์ œ๋Œ€๋กœ ๋ฐ˜์˜๋„ ์•ˆ ๋  ๊ฒƒ์ด๊ธฐ ๋•Œ๋ฌธ์ž…๋‹ˆ๋‹ค.</p>
  188. <p>๊ฐ ๋ชจ๋ธ์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜์„ ์žฌ์ƒํ•˜๋ ค๋ฉด <a href="/docs/#api/ko/animation/AnimationMixer"><code class="notranslate" translate="no">AnimationMixer</code></a>๋ฅผ ์จ์•ผ ํ•ฉ๋‹ˆ๋‹ค. <a href="/docs/#api/ko/animation/AnimationMixer"><code class="notranslate" translate="no">AnimationMixer</code></a>๋Š” ํ•˜๋‚˜ ์ด์ƒ์˜ <a href="/docs/#api/ko/animation/AnimationAction"><code class="notranslate" translate="no">AnimationAction</code></a>์œผ๋กœ ์ด๋ฃจ์–ด์ง€๊ณ , ๊ฐ <a href="/docs/#api/ko/animation/AnimationAction"><code class="notranslate" translate="no">AnimationAction</code></a>์—๋Š” ํ•˜๋‚˜์˜ <a href="/docs/#api/ko/animation/AnimationClip"><code class="notranslate" translate="no">AnimationClip</code></a>์ด ์žˆ์Šต๋‹ˆ๋‹ค. <a href="/docs/#api/ko/animation/AnimationAction"><code class="notranslate" translate="no">AnimationAction</code></a>์—๋Š” ์—ฌ๋Ÿฌ ์•ก์…˜(action)์„ ์ด์–ด์„œ ์žฌ์ƒํ•˜๊ฑฐ๋‚˜, ๋‹ค๋ฅธ ์• ๋‹ˆ๋ฉ”์ด์…˜์œผ๋กœ ๋ถ€๋“œ๋Ÿฝ๊ฒŒ ์ „ํ™˜ํ•˜๊ธฐ ๋“ฑ ๋‹ค์–‘ํ•œ ์„ค์ •์ด ์žˆ์ฃ . ๋‹น์žฅ์€ ์ฒซ ๋ฒˆ์งธ <a href="/docs/#api/ko/animation/AnimationClip"><code class="notranslate" translate="no">AnimationClip</code></a>์œผ๋กœ ์•ก์…˜์„ ๋งŒ๋“ค์–ด๋ด…์‹œ๋‹ค. ์„ค์ •์„ ๋ฐ”๊พธ์ง€ ์•Š๋Š”๋‹ค๋ฉด ํ•ด๋‹น ์• ๋‹ˆ๋ฉ”์ด์…˜ ํด๋ฆฝ(clip)์„ ๋ฐ˜๋ณตํ•ด ์žฌ์ƒํ•  ๊ฒ๋‹ˆ๋‹ค.</p>
  189. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">+const mixers = [];
  190. function init() {
  191. // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  192. const loadingElem = document.querySelector('#loading');
  193. loadingElem.style.display = 'none';
  194. prepModelsAndAnimations();
  195. Object.values(models).forEach((model, ndx) =&gt; {
  196. const clonedScene = SkeletonUtils.clone(model.gltf.scene);
  197. const root = new THREE.Object3D();
  198. root.add(clonedScene);
  199. scene.add(root);
  200. root.position.x = (ndx - 3) * 3;
  201. + const mixer = new THREE.AnimationMixer(clonedScene);
  202. + const firstClip = Object.values(model.animations)[0];
  203. + const action = mixer.clipAction(firstClip);
  204. + action.play();
  205. + mixers.push(mixer);
  206. });
  207. }
  208. </pre>
  209. <p>์• ๋‹ˆ๋ฉ”์ด์…˜์„ ์‹œ์ž‘ํ•˜๊ธฐ ์œ„ํ•ด <a href="/docs/#api/ko/animation/AnimationAction#play"><code class="notranslate" translate="no">play</code></a> ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ–ˆ์Šต๋‹ˆ๋‹ค. ๊ทธ๋ฆฌ๊ณ  ์ƒ์„ฑํ•œ <a href="/docs/#api/ko/animation/AnimationMixer"><code class="notranslate" translate="no">AnimationMixer</code></a>๋“ค์„ ์ „๋ถ€ <code class="notranslate" translate="no">mixers</code> ๋ฐฐ์—ด์— ๋„ฃ์—ˆ์ฃ . ๋งˆ์ง€๋ง‰์œผ๋กœ ๋ Œ๋”๋ง ๋ฃจํ”„์—์„œ ๊ฐ <a href="/docs/#api/ko/animation/AnimationMixer"><code class="notranslate" translate="no">AnimationMixer</code></a>์˜ <a href="/docs/#api/ko/animation/AnimationMixer.update"><code class="notranslate" translate="no">AnimationMixer.update</code></a> ๋ฉ”์„œ๋“œ์— ๋ฐ”๋กœ ์ง์ „ ํ”„๋ ˆ์ž„๊ณผ ํ˜„์žฌ ํ”„๋ ˆ์ž„์˜ ์‹œ๊ฐ„๊ฐ’์„ ๋„˜๊ฒจ์ฃผ์–ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.</p>
  210. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">+let then = 0;
  211. function render(now) {
  212. + now *= 0.001; // ์ดˆ ๋‹จ์œ„๋กœ ๋ณ€ํ™˜
  213. + const deltaTime = now - then;
  214. + then = now;
  215. if (resizeRendererToDisplaySize(renderer)) {
  216. const canvas = renderer.domElement;
  217. camera.aspect = canvas.clientWidth / canvas.clientHeight;
  218. camera.updateProjectionMatrix();
  219. }
  220. + for (const mixer of mixers) {
  221. + mixer.update(deltaTime);
  222. + }
  223. renderer.render(scene, camera);
  224. requestAnimationFrame(render);
  225. }
  226. </pre>
  227. <p>์ด์ œ ๊ฐ ๋ชจ๋ธ๊ณผ ๋ชจ๋ธ์˜ ์ฒซ ๋ฒˆ์งธ ์• ๋‹ˆ๋ฉ”์ด์…˜์ด ๋ณด์ผ ๊ฒ๋‹ˆ๋‹ค.</p>
  228. <p></p><div translate="no" class="threejs_example_container notranslate">
  229. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/game-load-models.html"></iframe></div>
  230. <a class="threejs_center" href="/manual/examples/game-load-models.html" target="_blank">์ƒˆ ํƒญ์—์„œ ๋ณด๊ธฐ</a>
  231. </div>
  232. <p></p>
  233. <p>๋ชจ๋“  ์• ๋‹ˆ๋ฉ”์ด์…˜์„ ํ™•์ธํ•  ์ˆ˜ ์žˆ๋„๋ก ์˜ˆ์ œ๋ฅผ ์ˆ˜์ •ํ•ด๋ด…์‹œ๋‹ค. ์• ๋‹ˆ๋ฉ”์ด์…˜ ํด๋ฆฝ์„ ์ „๋ถ€ ์•ก์…˜์œผ๋กœ ๋งŒ๋“ค์–ด ์žฌ์ƒํ•  ์ˆ˜ ์žˆ๋„๋ก ๋งŒ๋“ค๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  234. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">-const mixers = [];
  235. +const mixerInfos = [];
  236. function init() {
  237. // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  238. const loadingElem = document.querySelector('#loading');
  239. loadingElem.style.display = 'none';
  240. prepModelsAndAnimations();
  241. Object.values(models).forEach((model, ndx) =&gt; {
  242. const clonedScene = SkeletonUtils.clone(model.gltf.scene);
  243. const root = new THREE.Object3D();
  244. root.add(clonedScene);
  245. scene.add(root);
  246. root.position.x = (ndx - 3) * 3;
  247. const mixer = new THREE.AnimationMixer(clonedScene);
  248. - const firstClip = Object.values(model.animations)[0];
  249. - const action = mixer.clipAction(firstClip);
  250. - action.play();
  251. - mixers.push(mixer);
  252. + const actions = Object.values(model.animations).map((clip) =&gt; {
  253. + return mixer.clipAction(clip);
  254. + });
  255. + const mixerInfo = {
  256. + mixer,
  257. + actions,
  258. + actionNdx: -1,
  259. + };
  260. + mixerInfos.push(mixerInfo);
  261. + playNextAction(mixerInfo);
  262. });
  263. }
  264. +function playNextAction(mixerInfo) {
  265. + const { actions, actionNdx } = mixerInfo;
  266. + const nextActionNdx = (actionNdx + 1) % actions.length;
  267. + mixerInfo.actionNdx = nextActionNdx;
  268. + actions.forEach((action, ndx) =&gt; {
  269. + const enabled = ndx === nextActionNdx;
  270. + action.enabled = enabled;
  271. + if (enabled) {
  272. + action.play();
  273. + }
  274. + });
  275. +}
  276. </pre>
  277. <p>์œ„ ์ฝ”๋“œ์—์„œ๋Š” ๊ฐ ๋ชจ๋ธ๋งˆ๋‹ค ์•ก์…˜ ๋ฐฐ์—ด๊ณผ <a href="/docs/#api/ko/animation/AnimationMixer"><code class="notranslate" translate="no">AnimationMixer</code></a>๋ฅผ ๊ฐ์ฒด๋กœ ๋งŒ๋“ค์–ด ์ €์žฅํ–ˆ์Šต๋‹ˆ๋‹ค. ์•ก์…˜ ๋ฐฐ์—ด์€ ๋ชจ๋ธ์˜ ๊ฐ <a href="/docs/#api/ko/animation/AnimationClip"><code class="notranslate" translate="no">AnimationClip</code></a>๋งˆ๋‹ค <a href="/docs/#api/ko/animation/AnimationAction"><code class="notranslate" translate="no">AnimationAction</code></a>์„ ํ•˜๋‚˜์”ฉ ์ƒ์„ฑํ•ด ๋ฐฐ์—ด๋กœ ๋งŒ๋“  ๊ฒƒ์ด์ฃ . ๊ทธ๋ฆฌ๊ณ  ํ•˜๋‚˜์˜ ์•ก์…˜์„ ์ œ์™ธํ•œ ๋‚˜๋จธ์ง€ ์•ก์…˜์˜ <code class="notranslate" translate="no">enabled</code> ์†์„ฑ์„ ๋„๋Š” <code class="notranslate" translate="no">playNextAction</code>์„ ํ˜ธ์ถœํ–ˆ์Šต๋‹ˆ๋‹ค.</p>
  278. <p>๋ฐ์ดํ„ฐ ํ˜•์‹์ด ๋ฐ”๋€Œ์—ˆ์œผ๋‹ˆ ๋ Œ๋”๋ง ๋ฃจํ”„์˜ ์—…๋ฐ์ดํŠธ ์ชฝ ์ฝ”๋“œ๋„ ์ˆ˜์ •ํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.</p>
  279. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">-for (const mixer of mixers) {
  280. +for (const { mixer } of mixerInfos) {
  281. mixer.update(deltaTime);
  282. }
  283. </pre>
  284. <p>์ˆซ์žํ‚ค 1-8๋ฒˆ์„ ๋ˆŒ๋Ÿฌ ๊ฐ ๋ชจ๋ธ์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜์„ ์„ ํƒํ•  ์ˆ˜ ์žˆ๋„๋ก ๋ฆฌ์Šค๋„ˆ๋ฅผ ์ง€์ •ํ•ฉ๋‹ˆ๋‹ค.</p>
  285. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">window.addEventListener('keydown', (e) =&gt; {
  286. const mixerInfo = mixerInfos[e.keyCode - 49];
  287. if (!mixerInfo) {
  288. return;
  289. }
  290. playNextAction(mixerInfo);
  291. });
  292. </pre>
  293. <p>์ด์ œ ์˜ˆ์ œ๋ฅผ ํด๋ฆญํ•œ ๋’ค ์ˆซ์žํ‚ค 1-8๋ฒˆ์„ ๋ˆ„๋ฅด๋ฉด ๊ฐ ๋ฒˆํ˜ธ์— ํ•ด๋‹นํ•˜๋Š” ๋ชจ๋ธ์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜์ด ๋ฐ”๋€” ๊ฒ๋‹ˆ๋‹ค.</p>
  294. <p></p><div translate="no" class="threejs_example_container notranslate">
  295. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/game-check-animations.html"></iframe></div>
  296. <a class="threejs_center" href="/manual/examples/game-check-animations.html" target="_blank">์ƒˆ ํƒญ์—์„œ ๋ณด๊ธฐ</a>
  297. </div>
  298. <p></p>
  299. <p>Three.js ๊ด€๋ จ ๋‚ด์šฉ์€ ์—ฌ๊ธฐ๊นŒ์ง€์ž…๋‹ˆ๋‹ค. ์—ฌํƒœ๊นŒ์ง€ ๋‹ค์ˆ˜์˜ ํŒŒ์ผ์„ ๋ถˆ๋Ÿฌ์˜ค๋Š” ๋ฒ•, ํ…์Šค์ฒ˜๊ฐ€ ์”Œ์›Œ์ง„ ๋ชจ๋ธ์„ ๋ณต์‚ฌํ•˜๋Š” ๋ฒ•, ํ•ด๋‹น ๋ชจ๋ธ์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜์„ ์žฌ์ƒํ•˜๋Š” ๋ฒ•์„ ์•Œ์•„๋ดค์ฃ . ์‹ค์ œ ๊ฒŒ์ž„์—์„œ๋Š” <a href="/docs/#api/ko/animation/AnimationAction"><code class="notranslate" translate="no">AnimationAction</code></a> ๊ฐ์ฒด๋กœ ๋‹ค์–‘ํ•œ ๋™์ž‘์„ ์ง์ ‘ ์ฒ˜๋ฆฌํ•ด์ค˜์•ผ ํ•ฉ๋‹ˆ๋‹ค.</p>
  300. <p>์ž, ์ด์ œ ๊ฒŒ์ž„์˜ ๊ธฐ๋ณธ ํ‹€์„ ๋งŒ๋“ค์–ด๋ด…์‹œ๋‹ค.</p>
  301. <p>์ตœ์‹  ๊ฒŒ์ž„์„ ๋งŒ๋“ค ๋•Œ๋Š” ๋ณดํ†ต <a href="https://www.google.com/search?q=entity+component+system">Entity Component System(ECS)</a>์„ ๋งŽ์ด ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค. Entity Component System์—์„œ๋Š” ๊ฒŒ์ž„์˜ ์š”์†Œ๋ฅผ ์—ฌ๋Ÿฌ ๊ฐœ์˜ <em>์ปดํฌ๋„ŒํŠธ(component)</em>๋กœ ์ด๋ฃจ์–ด์ง„ <em>์—”ํ‹ฐํ‹ฐ(entity)</em>๋ผ ๋ถ€๋ฅด์ฃ . ์ƒˆ๋กœ์šด ์—”ํ‹ฐํ‹ฐ๋ฅผ ์ƒ์„ฑํ•  ๋•Œ๋Š” ๋ชจ๋“  ์ฝ”๋“œ๋ฅผ ์ƒˆ๋กœ ์“ฐ๋Š” ๊ฒƒ์ด ์•„๋‹Œ, ๋ฏธ๋ฆฌ ๋งŒ๋“ค์–ด ๋†“์€ ์ปดํฌ๋„ŒํŠธ๋“ค์„ ์—ฎ์–ด ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค.</p>
  302. <p>์˜ˆ์ œ์—์„œ๋Š” ์—”ํ‹ฐํ‹ฐ๋ฅผ <code class="notranslate" translate="no">GameObject</code>๋ผ ๋ถ€๋ฅด๊ฒ ์Šต๋‹ˆ๋‹ค. ์ด๋Š” ๋‹จ์ˆœํžˆ ์ปดํฌ๋„ŒํŠธ ๋ฐฐ์—ด๊ณผ <a href="/docs/#api/ko/core/Object3D"><code class="notranslate" translate="no">THREE.Object3D</code></a>๋ฅผ ํ•ฉ์นœ ๊ฒƒ์ž…๋‹ˆ๋‹ค.</p>
  303. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function removeArrayElement(array, element) {
  304. const ndx = array.indexOf(element);
  305. if (ndx &gt;= 0) {
  306. array.splice(ndx, 1);
  307. }
  308. }
  309. class GameObject {
  310. constructor(parent, name) {
  311. this.name = name;
  312. this.components = [];
  313. this.transform = new THREE.Object3D();
  314. parent.add(this.transform);
  315. }
  316. addComponent(ComponentType, ...args) {
  317. const component = new ComponentType(this, ...args);
  318. this.components.push(component);
  319. return component;
  320. }
  321. removeComponent(component) {
  322. removeArrayElement(this.components, component);
  323. }
  324. getComponent(ComponentType) {
  325. return this.components.find(c =&gt; c instanceof ComponentType);
  326. }
  327. update() {
  328. for (const component of this.components) {
  329. component.update();
  330. }
  331. }
  332. }
  333. </pre>
  334. <p><code class="notranslate" translate="no">GameObject.update</code> ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด ๊ฐ ์ปดํฌ๋„ŒํŠธ์˜ <code class="notranslate" translate="no">update</code> ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค.</p>
  335. <p>name ์†์„ฑ์€ ๋‹จ์ˆœํžˆ ๋””๋ฒ„๊น…์„ ์œ„ํ•œ ๊ฒƒ์ž…๋‹ˆ๋‹ค. ์ฝ˜์†”์—์„œ <code class="notranslate" translate="no">GameObject</code>๋ฅผ ๋ดค์„ ๋•Œ ์–ด๋–ค ์š”์†Œ์ธ์ง€ ์‰ฝ๊ฒŒ ํ™•์ธํ•  ์ˆ˜ ์žˆ๊ฒ ์ฃ .</p>
  336. <p>์ƒ์†Œํ•ด ๋ณด์ผ ์ˆ˜ ์žˆ๋Š” ๊ฒƒ๋“ค ๋ช‡ ๊ฐ€์ง€๋งŒ ์ง‘๊ณ  ๋„˜์–ด๊ฐ€๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  337. <p><code class="notranslate" translate="no">GameObject.addComponent</code>๋Š” ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ƒ์„ฑํ•  ๋•Œ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค. GameObject ์•ˆ์—์„œ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ƒ์„ฑํ•˜๋Š” ๊ฒŒ ์ตœ์„ ์ธ์ง€๋Š” ๋ชจ๋ฅด๊ฒ ์œผ๋‚˜, ๊ฐœ์ธ์ ์œผ๋กœ ์ปดํฌ๋„ŒํŠธ๊ฐ€ GameObject ๋ฐ–์— ์กด์žฌํ•˜๋Š” ๊ฑด ์˜๋ฏธ๊ฐ€ ์—†์–ด ๋ณด์˜€์Šต๋‹ˆ๋‹ค. ๊ทธ๋ž˜์„œ ์ƒ์„ฑํ•œ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ž๋™์œผ๋กœ GameObject์˜ ๋ฐฐ์—ด์— ์ถ”๊ฐ€ํ•˜๊ณ , GameObject ์ž์ฒด๋„ ์ปดํฌ๋„ŒํŠธ์˜ constructor์— ๋„˜๊ฒจ์ค„ ์ˆ˜ ์žˆ์œผ๋ฉด ํŽธํ•˜๊ฒ ๋‹ค๊ณ  ์ƒ๊ฐํ–ˆ์ฃ . ์‰ฝ๊ฒŒ ๋งํ•ด ์ง€๊ธˆ์€ ๋‹ค์Œ์ฒ˜๋Ÿผ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ถ”๊ฐ€ํ•˜์ง€๋งŒ,</p>
  338. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const gameObject = new GameObject(scene, 'foo');
  339. gameObject.addComponent(TypeOfComponent);
  340. </pre>
  341. <p>์œ„์™€ ๊ฐ™์€ ๋ฐฉ์‹์„ ์„ ํ˜ธํ•˜์ง€ ์•Š๋Š”๋‹ค๋ฉด ๋‹ค์Œ์ฒ˜๋Ÿผ ์ถ”๊ฐ€ํ•  ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  342. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const gameObject = new GameObject(scene, 'foo');
  343. const component = new TypeOfComponent(gameObject);
  344. gameObject.addComponent(component);
  345. </pre>
  346. <p>์ฒซ ๋ฒˆ์งธ ์ฝ”๋“œ๊ฐ€ ์งง๊ณ  ์ž๋™ํ™”๋๋‹ค๋Š” ๋ฉด์—์„œ ๋” ์ข‹์„๊นŒ์š”, ์•„๋‹ˆ๋ฉด ๊ธฐ์กด ํ˜•์‹์„ ํ•ด์ณ์„œ ๋” ๋ณ„๋กœ์ผ๊นŒ์š”? ์ €๋Š” ์–ด๋–ป๋‹ค๊ณ  ํŒ๋‹จํ•˜๊ธฐ๊ฐ€ ์–ด๋ ต๋„ค์š”.</p>
  347. <p><code class="notranslate" translate="no">GameObject.getComponent</code>๋Š” ์ปดํฌ๋„ŒํŠธ์˜ ํƒ€์ž…์„ ์ด์šฉํ•ด ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ฐพ์Šต๋‹ˆ๋‹ค. ์ด๋Š” ํ•˜๋‚˜์˜ GameObject๊ฐ€ ๊ฐ™์€ ํƒ€์ž…์˜ ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋‘ ๊ฐœ ์ด์ƒ ์‚ฌ์šฉํ•  ์ˆ˜ ์—†๋‹ค๋Š” ์ด์•ผ๊ธฐ์ฃ . ๋ฌผ๋ก  ๋‘ ๊ฐœ ์ด์ƒ ์‚ฌ์šฉํ•œ๋‹ค๊ณ  ์—๋Ÿฌ๊ฐ€ ๋‚˜๊ฑฐ๋‚˜ ํ•˜์ง„ ์•Š๊ฒ ์ง€๋งŒ, ๋ณ„๋„์˜ API๋ฅผ ์ถ”๊ฐ€ํ•˜์ง€ ์•Š๋Š” ํ•œ ์ € ๋ฉ”์„œ๋“œ๋Š” ํ•ญ์ƒ ๊ฐ™์€ ํƒ€์ž… ์ค‘ ์ฒซ ๋ฒˆ์งธ ์ปดํฌ๋„ŒํŠธ๋งŒ์„ ๋ฐ˜ํ™˜ํ•  ๊ฒ๋‹ˆ๋‹ค.</p>
  348. <p>์ปดํฌ๋„ŒํŠธ๊ฐ€ ๋‹ค๋ฅธ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ฐพ๋Š” ๊ฑด ํ”ํ•œ ์ผ์ž…๋‹ˆ๋‹ค. ๊ทธ๋ฆฌ๊ณ  ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ฐพ์„ ๋•Œ๋Š” ์ž˜๋ชป ์ฐธ์กฐํ•˜๋Š” ์ผ์ด ์—†๋„๋ก ํƒ€์ž…์„ ์ฒดํฌํ•ด์•ผ ํ•˜์ฃ . ๊ทธ๋ƒฅ ๊ฐ ์ปดํฌ๋„ŒํŠธ์— ๊ณ ์œ ํ•œ ์ด๋ฆ„ ์†์„ฑ์„ ์ฃผ๊ณ  ๊ทธ ์ด๋ฆ„์œผ๋กœ ํ•ด๋‹น ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ฐพ์„ ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค. ์ด๋ ‡๊ฒŒ ํ•˜๋ฉด ๊ฐ™์€ ํƒ€์ž…์˜ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์—ฌ๋Ÿฌ ๊ฐœ ์“ธ ์ˆ˜ ์žˆ์œผ๋‹ˆ ํ™•์žฅ์„ฑ ๋ฉด์—์„œ ์œ ๋ฆฌํ•  ๊ฒ๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์ด ๋ฐฉ๋ฒ•์€ ๊ทธ๋‹ค์ง€ ์ผ๊ด€์„ฑ์ด ์—†์Šต๋‹ˆ๋‹ค. ์ด๋ฒˆ์—๋„ ์–ด๋–ค ์ชฝ์ด ๋” ์ข‹๋‹ค๊ณ  ํŒ๋‹จํ•˜๊ธฐ๊ฐ€ ์–ด๋ ต๋„ค์š”.</p>
  349. <p>์•„๋ž˜๋Š” ์ปดํฌ๋„ŒํŠธ์˜ ๊ธฐ์ดˆ ํด๋ž˜์Šค์ž…๋‹ˆ๋‹ค.</p>
  350. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">// ๋ชจ๋“  ์ปดํฌ๋„ŒํŠธ์˜ ๊ธฐ์ดˆ
  351. class Component {
  352. constructor(gameObject) {
  353. this.gameObject = gameObject;
  354. }
  355. update() {
  356. }
  357. }
  358. </pre>
  359. <p>์ปดํฌ๋„ŒํŠธ์— ๊ธฐ์ดˆ ํด๋ž˜์Šค๊ฐ€ ํ•„์š”ํ• ๊นŒ์š”? ์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ๋Š” ํƒ€์ž…์ด ๋А์Šจํ•œ ์–ธ์–ด์ด๊ธฐ์— ๊ตณ์ด ๊ธฐ์ดˆ ํด๋ž˜์Šค๋ฅผ ์“ธ ํ•„์š”๋Š” ์—†์Šต๋‹ˆ๋‹ค. ๊ฐ ์ปดํฌ๋„ŒํŠธ์˜ constructor์—์„œ ์ฒซ ๋ฒˆ์งธ ์ธ์ž๊ฐ€ GameObject์ด๊ธฐ๋งŒ ํ•˜๋ฉด ๋˜์ฃ . ๋งŒ์•ฝ GameObject๋ฅผ ๋‚˜์ค‘์— ์ฐธ์กฐํ•  ํ•„์š”๊ฐ€ ์—†๋‹ค๋ฉด ๊ตณ์ด ์ €์žฅํ•˜์ง€ ์•Š์•„๋„ ๋  ๊ฒ๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์ €๋Š” ์™ ์ง€ ์ด ํ˜•์‹์ด ๋” ์ข‹์•„ ๋ณด์ด๋„ค์š”. ๊ธฐ์ดˆ ํด๋ž˜์Šค๋ฅผ ๋‘๋ฉด ๋ถ€๋ชจ์˜ GameObject์— ์‰ฝ๊ฒŒ ์ ‘๊ทผํ•  ์ˆ˜ ์žˆ์„ ๋ฟ๋งŒ ์•„๋‹ˆ๋ผ ๋‹ค๋ฅธ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์‰ฝ๊ฒŒ ์ฐพ์„ ์ˆ˜ ์žˆ๊ณ , ์–ด๋–ค ์ฐจ์ด์ ์ด ์žˆ๋Š”์ง€๋„ ์‰ฝ๊ฒŒ ์•Œ ์ˆ˜ ์žˆ์„ ํ…Œ๋‹ˆ๊นŒ์š”.</p>
  360. <p>GameObject๋ฅผ ๋‹ค๋ฃจ๋ ค๋ฉด GameObject๋ฅผ ๊ด€๋ฆฌํ•˜๋Š” ํด๋ž˜์Šค๋ฅผ ๋งŒ๋“œ๋Š” ๊ฒŒ ์ข‹์„ ๋“ฏํ•ฉ๋‹ˆ๋‹ค. ์–ผํ• GameObject๋ฅผ ๋ฐฐ์—ด ํ˜•์‹์œผ๋กœ ๊ฐ–๊ณ  ์žˆ์–ด๋„ ๊ดœ์ฐฎ์ง€ ์•Š๋‚˜ ์‹ถ์„ ์ˆ˜ ์žˆ์œผ๋‚˜, ์‹ค์ œ๋กœ ๊ฒŒ์ž„์„ ํ”Œ๋ ˆ์ดํ•  ๋•Œ๋Š” ์š”์†Œ๊ฐ€ ์ถ”๊ฐ€๋˜๊ธฐ๋„ ํ•˜๊ณ , ์—†์–ด์ง€๊ธฐ๋„ ํ•ฉ๋‹ˆ๋‹ค. ์˜ˆ๋ฅผ ๋“ค์–ด ์ด GameObject๋Š” ์ด์„ ๋ฐœ์‚ฌํ•  ๋•Œ๋งˆ๋‹ค ์ด์•Œ GameObject๋ฅผ ์ถ”๊ฐ€ํ•  ๊ฒ๋‹ˆ๋‹ค. ๋ชฌ์Šคํ„ฐ GameObject๊ฐ€ ๋ˆ„๊ตฐ๊ฐ€์— ์˜ํ•ด ์ฃฝ๋Š”๋‹ค๋ฉด ํ•ด๋‹น GameObject๋Š” ์‚ฌ๋ผ์ง€๊ฒ ์ฃ . GameObject๋ฅผ ๋ฐฐ์—ด๋กœ ์ €์žฅํ•œ๋‹ค๋ฉด ์‹ญ์ค‘ํŒ”๊ตฌ ๋‹ค์Œ๊ณผ ๊ฐ™์€ ์‹์˜ ์ฝ”๋“œ๋ฅผ ์“ธ ๊ฒ๋‹ˆ๋‹ค.</p>
  361. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">for (const gameObject of globalArrayOfGameObjects) {
  362. gameObject.update();
  363. }
  364. </pre>
  365. <p>์œ„ ๋ฐ˜๋ณต๋ฌธ์€ <code class="notranslate" translate="no">globalArrayOfGameObjects</code>์— GameObject๊ฐ€ ์ถ”๊ฐ€๋˜๊ฑฐ๋‚˜ ์ œ๊ฑฐ๋์„ ๊ฒฝ์šฐ, ํŠน์ • ์ปดํฌ๋„ŒํŠธ์˜ <code class="notranslate" translate="no">update</code> ๋ฉ”์„œ๋“œ์—์„œ ์—๋Ÿฌ๋ฅผ ๋˜์ง€๊ฑฐ๋‚˜ ์˜ˆ์ƒ ๋ฐ–์˜ ๋™์ž‘์„ ํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  366. <p>์ด๋Ÿฐ ์ผ์„ ๋ฐฉ์ง€ํ•˜๊ธฐ ์œ„ํ•ด ์•ˆ์ „ ์žฅ์น˜๋ฅผ ์ถ”๊ฐ€ํ•ด๋ณด๋„๋ก ํ•˜์ฃ .</p>
  367. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class SafeArray {
  368. constructor() {
  369. this.array = [];
  370. this.addQueue = [];
  371. this.removeQueue = new Set();
  372. }
  373. get isEmpty() {
  374. return this.addQueue.length + this.array.length &gt; 0;
  375. }
  376. add(element) {
  377. this.addQueue.push(element);
  378. }
  379. remove(element) {
  380. this.removeQueue.add(element);
  381. }
  382. forEach(fn) {
  383. this._addQueued();
  384. this._removeQueued();
  385. for (const element of this.array) {
  386. if (this.removeQueue.has(element)) {
  387. continue;
  388. }
  389. fn(element);
  390. }
  391. this._removeQueued();
  392. }
  393. _addQueued() {
  394. if (this.addQueue.length) {
  395. this.array.splice(this.array.length, 0, ...this.addQueue);
  396. this.addQueue = [];
  397. }
  398. }
  399. _removeQueued() {
  400. if (this.removeQueue.size) {
  401. this.array = this.array.filter(element =&gt; !this.removeQueue.has(element));
  402. this.removeQueue.clear();
  403. }
  404. }
  405. }
  406. </pre>
  407. <p>์œ„ ํด๋ž˜์Šค๋Š” <code class="notranslate" translate="no">SafeArray</code>์˜ ์š”์†Œ๋ฅผ ๋”ํ•˜๊ฑฐ๋‚˜ ์ œ๊ฑฐํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•ด์ค๋‹ˆ๋‹ค. ์›๋ณธ ๋ฐฐ์—ด์˜ ๋ฐ˜๋ณต๋˜๋Š” ๋™์•ˆ ์›๋ณธ ๋ฐฐ์—ด์„ ๋ณ€๊ฒฝํ•˜์ง€ ์•Š๋Š”๋‹ค๋Š” ๊ฒŒ ์ฐจ์ด์ ์ด์ฃ . ๋Œ€์‹  ๋ฐ˜๋ณต ์ค‘๊ฐ„์— ์ถ”๊ฐ€๋œ ์š”์†Œ๋Š” <code class="notranslate" translate="no">addQueue</code>์—, ์ œ๊ฑฐ๋œ ์š”์†Œ๋Š” <code class="notranslate" translate="no">removeQueue</code>์— ๋“ค์–ด๊ฐ„ ๋’ค, ๋ฐ˜๋ณต๋ฌธ์ด ๋Œ์•„๊ฐ€์ง€ ์•Š์„ ๋•Œ ์›๋ณธ ๋ฐฐ์—ด์— ์ œ๊ฑฐ/์ถ”๊ฐ€๋ฉ๋‹ˆ๋‹ค.</p>
  408. <p>์•„๋ž˜๋Š” ์œ„ ํด๋ž˜์Šค๋ฅผ ์ด์šฉํ•œ GameObject์˜ ๊ด€๋ฆฌ ํด๋ž˜์Šค์ž…๋‹ˆ๋‹ค.</p>
  409. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class GameObjectManager {
  410. constructor() {
  411. this.gameObjects = new SafeArray();
  412. }
  413. createGameObject(parent, name) {
  414. const gameObject = new GameObject(parent, name);
  415. this.gameObjects.add(gameObject);
  416. return gameObject;
  417. }
  418. removeGameObject(gameObject) {
  419. this.gameObjects.remove(gameObject);
  420. }
  421. update() {
  422. this.gameObjects.forEach(gameObject =&gt; gameObject.update());
  423. }
  424. }
  425. </pre>
  426. <p>์—ฌํƒœ๊นŒ์ง€ ๋งŒ๋“  ์š”์†Œ๋กœ ์ฒซ ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋งŒ๋“ค์–ด๋ด…์‹œ๋‹ค. ์ด ์ปดํฌ๋„ŒํŠธ๋Š” ์•„๊นŒ ๋งŒ๋“ค์—ˆ๋˜ ๊ฒƒ๊ณผ ๊ฐ™์€ Three.js glTF ๊ฐ์ฒด๋ฅผ ๊ด€๋ฆฌํ•  ๊ฒ๋‹ˆ๋‹ค. ๊ฐ„๋‹จํžˆ ์• ๋‹ˆ๋ฉ”์ด์…˜์˜ ์ด๋ฆ„์„ ๋ฐ›์•„ ํ•ด๋‹น ์• ๋‹ˆ๋ฉ”์ด์…˜์„ ์žฌ์ƒํ•˜๋Š” <code class="notranslate" translate="no">setAnimation</code> ๋ฉ”์„œ๋“œ๋งŒ ์ƒˆ๋กœ ๋งŒ๋“ค๋„๋ก ํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  427. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class SkinInstance extends Component {
  428. constructor(gameObject, model) {
  429. super(gameObject);
  430. this.model = model;
  431. this.animRoot = SkeletonUtils.clone(this.model.gltf.scene);
  432. this.mixer = new THREE.AnimationMixer(this.animRoot);
  433. gameObject.transform.add(this.animRoot);
  434. this.actions = {};
  435. }
  436. setAnimation(animName) {
  437. const clip = this.model.animations[animName];
  438. // ๋ชจ๋“  ์•ก์…˜์„ ๋•๋‹ˆ๋‹ค.
  439. for (const action of Object.values(this.actions)) {
  440. action.enabled = false;
  441. }
  442. // ํ•ด๋‹น ํด๋ฆฝ์— ํ•ด๋‹นํ•˜๋Š” ์•ก์…˜์„ ์ƒ์„ฑ ๋˜๋Š” ๊ฐ€์ ธ์˜ต๋‹ˆ๋‹ค.
  443. const action = this.mixer.clipAction(clip);
  444. action.enabled = true;
  445. action.reset();
  446. action.play();
  447. this.actions[animName] = action;
  448. }
  449. update() {
  450. this.mixer.update(globals.deltaTime);
  451. }
  452. }
  453. </pre>
  454. <p>์ด ํด๋ž˜์Šค๋Š” ์•„๊นŒ ํ–ˆ๋˜ ๊ฒƒ์ฒ˜๋Ÿผ ๋ถˆ๋Ÿฌ์˜จ ์”ฌ ๊ทธ๋ž˜ํ”„๋ฅผ ๋ณต์‚ฌํ•˜๊ณ , <a href="/docs/#api/ko/animation/AnimationMixer"><code class="notranslate" translate="no">AnimationMixer</code></a>๋ฅผ ๋งŒ๋“ญ๋‹ˆ๋‹ค. ํด๋ž˜์Šค์˜ <code class="notranslate" translate="no">setAnimation</code> ๋ฉ”์„œ๋“œ๋Š” ํ•ด๋‹น ํด๋ฆฝ์— ๋Œ€ํ•œ ์•ก์…˜์ด ์กด์žฌํ•˜์ง€ ์•Š๋Š”๋‹ค๋ฉด ์ƒˆ๋กœ ์ƒ์„ฑํ•˜๊ณ , ๋‹ค๋ฅธ ์•ก์…˜์„ ์ „๋ถ€ ๋„๋Š” ์—ญํ• ์„ ํ•ฉ๋‹ˆ๋‹ค.</p>
  455. <p>์ด ์ฝ”๋“œ๋Š” <code class="notranslate" translate="no">globals.deltaTime</code>์„ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค. ์ด ์ „์—ญ ๊ฐ์ฒด๋„ ๋งŒ๋“ค์–ด์•ผ๊ฒ ์ฃ .</p>
  456. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const globals = {
  457. time: 0,
  458. deltaTime: 0,
  459. };
  460. </pre>
  461. <p>๊ทธ๋ฆฌ๊ณ  ๋ Œ๋”๋ง ๋ฃจํ”„์—์„œ ์ „์—ญ ๊ฐ์ฒด๋ฅผ ์—…๋ฐ์ดํŠธํ•˜๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.</p>
  462. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">let then = 0;
  463. function render(now) {
  464. // ์ดˆ ๋‹จ์œ„๋กœ ๋ณ€ํ™˜
  465. globals.time = now * 0.001;
  466. // ์‹œ๊ฐ„๊ฐ’์ด ๋„ˆ๋ฌด ํฌ์ง€ ์•Š๋„๋ก ์ œํ•œํ•ฉ๋‹ˆ๋‹ค.
  467. globals.deltaTime = Math.min(globals.time - then, 1 / 20);
  468. then = globals.time;
  469. </pre>
  470. <p>์œ„ ์ฝ”๋“œ์—์„œ๋Š” ์‹œ๊ฐ„๊ฐ’์˜ ๋ฒ”์œ„๊ฐ€ 1/20์ดˆ๋ฅผ ๋„˜์ง€ ์•Š๋„๋ก ํ–ˆ์Šต๋‹ˆ๋‹ค. ์ด๋Š” ์‚ฌ์šฉ์ž๊ฐ€ ํƒญ์„ ์ˆจ๊ธฐ๊ฑฐ๋‚˜ ํ–ˆ์„ ๊ฒฝ์šฐ ์‹œ๊ฐ„๊ฐ’์ด ๋„ˆ๋ฌด ์ปค์ง€์ง€ ์•Š๊ฒŒ ํ•˜๊ธฐ ์œ„ํ•œ ๊ฒƒ์ด์ฃ . ๋งŒ์•ฝ ์ด๋ ‡๊ฒŒ ์ œํ•œ์„ ๋‘์ง€ ์•Š๋Š”๋‹ค๋ฉด ์‚ฌ์šฉ์ž๊ฐ€ ํƒญ์„ ๋ช‡ ์ดˆ, ๋˜๋Š” ๋ช‡ ๋ถ„ ์ˆจ๊ฒผ๋‹ค๊ฐ€ ๋‹ค์‹œ ํƒญ์„ ์—ด์—ˆ์„ ๋•Œ ํ”„๋ ˆ์ž„ ๊ฐ„ ์‹œ๊ฐ„๊ฐ’์ด ๋„ˆ๋ฌด ์ปค์งˆ ํ…Œ๊ณ , ์•„๋ž˜์™€ ๊ฐ™์ด ์‹œ๊ฐ„์œผ๋กœ ์†๋ ฅ์„ ๊ณ„์‚ฐํ•˜๋Š” ๊ฒฝ์šฐ ์บ๋ฆญํ„ฐ๊ฐ€ ์ˆœ๊ฐ„์ด๋™ํ•˜๋Š” ๊ฒƒ์ฒ˜๋Ÿผ ๋ณด์ผ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  471. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">position += velocity * deltaTime;
  472. </pre>
  473. <p><code class="notranslate" translate="no">deltaTime</code>์˜ ์ตœ๋Œ“๊ฐ’์„ ์„ค์ •ํ•˜๋ฉด ์ด๋Ÿฐ ๋ฌธ์ œ๋ฅผ ๋ง‰์„ ์ˆ˜ ์žˆ์ฃ .</p>
  474. <p>์ด์ œ ํ”Œ๋ ˆ์ด์–ด ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋งŒ๋“ค์–ด๋ด…์‹œ๋‹ค.</p>
  475. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Player extends Component {
  476. constructor(gameObject) {
  477. super(gameObject);
  478. const model = models.knight;
  479. this.skinInstance = gameObject.addComponent(SkinInstance, model);
  480. this.skinInstance.setAnimation('Run');
  481. }
  482. }
  483. </pre>
  484. <p>ํ”Œ๋ ˆ์ด์–ด ์ปดํฌ๋„ŒํŠธ๋Š” ์ดˆ๊ธฐํ™” ์‹œ์— <code class="notranslate" translate="no">'Run'</code>์„ ์ธ์ž๋กœ <code class="notranslate" translate="no">setAnimation</code>์„ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค. ๊ฐœ์ธ์ ์œผ๋กœ ๋ฏธ๋ฆฌ ์–ด๋–ค ์• ๋‹ˆ๋ฉ”์ด์…˜์ด ์žˆ๋Š”์ง€ ๋ณด๋ ค๊ณ  ์ด์ „ ์˜ˆ์ œ๋ฅผ ์ˆ˜์ •ํ•ด ์• ๋‹ˆ๋ฉ”์ด์…˜์˜ ์ด๋ฆ„์„ ์ถœ๋ ฅํ•˜๋„๋ก ํ–ˆ์ฃ .</p>
  485. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function prepModelsAndAnimations() {
  486. Object.values(models).forEach(model =&gt; {
  487. + console.log('-------&gt;:', model.url);
  488. const animsByName = {};
  489. model.gltf.animations.forEach((clip) =&gt; {
  490. animsByName[clip.name] = clip;
  491. + console.log(' ', clip.name);
  492. });
  493. model.animations = animsByName;
  494. });
  495. }
  496. </pre>
  497. <p>์•„๋ž˜๋Š” ์‹ค์ œ๋กœ <a href="https://developers.google.com/web/tools/chrome-devtools/console/javascript">์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ ๊ฐœ๋ฐœ์ž ์ฝ˜์†”</a>์— ์ถœ๋ ฅ๋œ ๊ฒฐ๊ณผ์ž…๋‹ˆ๋‹ค.</p>
  498. <pre class="prettyprint showlinemods notranslate notranslate" translate="no"> -------&gt;: resources/models/animals/Pig.gltf
  499. Idle
  500. Death
  501. WalkSlow
  502. Jump
  503. Walk
  504. -------&gt;: resources/models/animals/Cow.gltf
  505. Walk
  506. Jump
  507. WalkSlow
  508. Death
  509. Idle
  510. -------&gt;: resources/models/animals/Llama.gltf
  511. Jump
  512. Idle
  513. Walk
  514. Death
  515. WalkSlow
  516. -------&gt;: resources/models/animals/Pug.gltf
  517. Jump
  518. Walk
  519. Idle
  520. WalkSlow
  521. Death
  522. -------&gt;: resources/models/animals/Sheep.gltf
  523. WalkSlow
  524. Death
  525. Jump
  526. Walk
  527. Idle
  528. -------&gt;: resources/models/animals/Zebra.gltf
  529. Jump
  530. Walk
  531. Death
  532. WalkSlow
  533. Idle
  534. -------&gt;: resources/models/animals/Horse.gltf
  535. Jump
  536. WalkSlow
  537. Death
  538. Walk
  539. Idle
  540. -------&gt;: resources/models/knight/KnightCharacter.gltf
  541. Run_swordRight
  542. Run
  543. Idle_swordLeft
  544. Roll_sword
  545. Idle
  546. Run_swordAttack
  547. </pre><p>์šด ์ข‹๊ฒŒ๋„ ๋™๋ฌผ๋“ค์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜ ์ด๋ฆ„์ด ์ „๋ถ€ ๋˜‘๊ฐ™๋„ค์š”. ๋‚˜์ค‘์— ํŽธํ•  ๋“ฏํ•ฉ๋‹ˆ๋‹ค. ๋ญ, ๊ทธ๊ฑด ๋‚˜์ค‘ ์–˜๊ธฐ๊ณ , ์ง€๊ธˆ์€ ํ”Œ๋ ˆ์ด์–ด์˜ ์• ๋‹ˆ๋ฉ”์ด์…˜ ์ค‘ <code class="notranslate" translate="no">Run</code>๋งŒ ์‹ ๊ฒฝ์”์‹œ๋‹ค.</p>
  548. <p>์ด์ œ ๋งŒ๋“  ์ปดํฌ๋„ŒํŠธ๋ฅผ ์จ ๋ณด๊ฒ ์Šต๋‹ˆ๋‹ค. ๋จผ์ € <code class="notranslate" translate="no">init</code> ํ•จ์ˆ˜๋ฅผ ์•ฝ๊ฐ„ ์ˆ˜์ •ํ•ฉ๋‹ˆ๋‹ค. <code class="notranslate" translate="no">init</code> ํ•จ์ˆ˜๋Š” <code class="notranslate" translate="no">GameObject</code>๋ฅผ ๋งŒ๋“ค๊ณ  ๊ฑฐ๊ธฐ์— <code class="notranslate" translate="no">Player</code> ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ถ”๊ฐ€ํ•˜๋Š” ์—ญํ• ์„ ํ•  ๊ฒ๋‹ˆ๋‹ค.</p>
  549. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const globals = {
  550. time: 0,
  551. deltaTime: 0,
  552. };
  553. +const gameObjectManager = new GameObjectManager();
  554. function init() {
  555. // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  556. const loadingElem = document.querySelector('#loading');
  557. loadingElem.style.display = 'none';
  558. prepModelsAndAnimations();
  559. + {
  560. + const gameObject = gameObjectManager.createGameObject(scene, 'player');
  561. + gameObject.addComponent(Player);
  562. + }
  563. }
  564. </pre>
  565. <p>๋ Œ๋”๋ง ๋ฃจํ”„์—์„œ <code class="notranslate" translate="no">gameObjectManager.update</code>๋ฅผ ํ˜ธ์ถœํ•˜๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.</p>
  566. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">let then = 0;
  567. function render(now) {
  568. // ์ดˆ ๋‹จ์œ„๋กœ ๋ณ€ํ™˜
  569. globals.time = now * 0.001;
  570. // ์‹œ๊ฐ„๊ฐ’์ด ๋„ˆ๋ฌด ํฌ์ง€ ์•Š๋„๋ก ์ œํ•œํ•ฉ๋‹ˆ๋‹ค.
  571. globals.deltaTime = Math.min(globals.time - then, 1 / 20);
  572. then = globals.time;
  573. if (resizeRendererToDisplaySize(renderer)) {
  574. const canvas = renderer.domElement;
  575. camera.aspect = canvas.clientWidth / canvas.clientHeight;
  576. camera.updateProjectionMatrix();
  577. }
  578. - for (const { mixer } of mixerInfos) {
  579. - mixer.update(deltaTime);
  580. - }
  581. + gameObjectManager.update();
  582. renderer.render(scene, camera);
  583. requestAnimationFrame(render);
  584. }
  585. </pre>
  586. <p>์˜ˆ์ œ๋ฅผ ์‹คํ–‰ํ•˜๋ฉด ํ”Œ๋ ˆ์ด์–ด ํ•˜๋‚˜๋งŒ ๋ณด์ผ ๊ฒ๋‹ˆ๋‹ค.</p>
  587. <p></p><div translate="no" class="threejs_example_container notranslate">
  588. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/game-just-player.html"></iframe></div>
  589. <a class="threejs_center" href="/manual/examples/game-just-player.html" target="_blank">์ƒˆ ํƒญ์—์„œ ๋ณด๊ธฐ</a>
  590. </div>
  591. <p></p>
  592. <p>๋‹จ์ˆœํžˆ Entity Component System์„ ๊ตฌํ˜„ํ•˜๋Š” ๋ฐ๋งŒ ๋„ˆ๋ฌด ๋งŽ์€ ์ฝ”๋“œ๋ฅผ ์“ด ๊ฒŒ ์•„๋‹Œ๊ฐ€ ์‹ถ์ง€๋งŒ, ์ด ์ •๋„๊ฐ€ ๋Œ€๋ถ€๋ถ„์˜ ๊ฒŒ์ž„์ด ๊ฐ–์ถฐ์•ผํ•  ๊ธฐ๋ณธ์ž…๋‹ˆ๋‹ค.</p>
  593. <p>์ด์ œ ์‚ฌ์šฉ์ž ์ž…๋ ฅ ์‹œ์Šคํ…œ์„ ์ถ”๊ฐ€ํ•ด๋ด…์‹œ๋‹ค. ๋‹จ์ˆœํžˆ ํ‚ค๋ณด๋“œ ์ด๋ฒคํŠธ์— ์ง์ ‘ ์ฝ”๋“œ๋ฅผ ์ž‘์„ฑํ•˜๊ธฐ๋ณด๋‹ค ํด๋ž˜์Šค๋ฅผ ๋งŒ๋“ค์–ด ์ฝ”๋“œ์˜ ๋‹ค๋ฅธ ๋ถ€๋ถ„์—์„œ๋„ <code class="notranslate" translate="no">์™ผ์ชฝ</code>, <code class="notranslate" translate="no">์˜ค๋ฅธ์ชฝ</code>์„ ํ™•์ธํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค. ์ด๋Ÿฌ๋ฉด <code class="notranslate" translate="no">์™ผ์ชฝ</code>, <code class="notranslate" translate="no">์˜ค๋ฅธ์ชฝ</code> ๋“ฑ ๋‹ค์–‘ํ•œ ํ‚ค๋ฅผ ๋‹ค์–‘ํ•œ ๋ฐฉ๋ฒ•์œผ๋กœ ์ง€์ •ํ•  ์ˆ˜ ์žˆ๊ฒ ์ฃ . ๋จผ์ € ํ‚ค๋ณด๋“œ ์ด๋ฒคํŠธ๋ถ€ํ„ฐ ์ง€์ •ํ•ฉ์‹œ๋‹ค.</p>
  594. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">/**
  595. * ํ‚ค ๋˜๋Š” ๋ฒ„ํŠผ์˜ ์ƒํƒœ๋ฅผ ์ถ”์ ํ•ฉ๋‹ˆ๋‹ค.
  596. *
  597. * ์™ผ์ชฝ ๋ฐฉํ–ฅํ‚ค๊ฐ€ ๋ˆŒ๋ ธ๋Š”์ง€ ํ™•์ธํ•˜๋ ค๋ฉด
  598. *
  599. * inputManager.keys.left.down
  600. *
  601. * ์„ ํ™•์ธํ•˜๊ณ , ํ˜„์žฌ ํ”„๋ ˆ์ž„์—์„œ ์™ผ์ชฝ ํ‚ค๋ฅผ ๋ˆŒ๋ €๋Š”์ง€ ํ™•์ธํ•˜๋ ค๋ฉด
  602. *
  603. * inputManager.keys.left.justPressed
  604. *
  605. * ๋ฅผ ํ™•์ธํ•˜๋ฉด ๋ฉ๋‹ˆ๋‹ค.
  606. *
  607. * ํ˜„์žฌ ๋“ฑ๋ก๋œ ํ‚ค๋Š” 'left', 'right', 'a', 'b', 'up', 'down' ์ž…๋‹ˆ๋‹ค.
  608. **/
  609. class InputManager {
  610. constructor() {
  611. this.keys = {};
  612. const keyMap = new Map();
  613. const setKey = (keyName, pressed) =&gt; {
  614. const keyState = this.keys[keyName];
  615. keyState.justPressed = pressed &amp;&amp; !keyState.down;
  616. keyState.down = pressed;
  617. };
  618. const addKey = (keyCode, name) =&gt; {
  619. this.keys[name] = { down: false, justPressed: false };
  620. keyMap.set(keyCode, name);
  621. };
  622. const setKeyFromKeyCode = (keyCode, pressed) =&gt; {
  623. const keyName = keyMap.get(keyCode);
  624. if (!keyName) {
  625. return;
  626. }
  627. setKey(keyName, pressed);
  628. };
  629. addKey(37, 'left');
  630. addKey(39, 'right');
  631. addKey(38, 'up');
  632. addKey(40, 'down');
  633. addKey(90, 'a');
  634. addKey(88, 'b');
  635. window.addEventListener('keydown', (e) =&gt; {
  636. setKeyFromKeyCode(e.keyCode, true);
  637. });
  638. window.addEventListener('keyup', (e) =&gt; {
  639. setKeyFromKeyCode(e.keyCode, false);
  640. });
  641. }
  642. update() {
  643. for (const keyState of Object.values(this.keys)) {
  644. if (keyState.justPressed) {
  645. keyState.justPressed = false;
  646. }
  647. }
  648. }
  649. }
  650. </pre>
  651. <p>์œ„ ์ฝ”๋“œ๋Š” ํ‚ค๊ฐ€ ๋ˆŒ๋ ธ๋Š”์ง€, ๋—๋Š”์ง€๋ฅผ ์ถ”์ ํ•ฉ๋‹ˆ๋‹ค. ํŠน์ • ํ‚ค๋ฅผ ๋ˆŒ๋ €๋Š”์ง€ ํ™•์ธํ•˜๋ ค๋ฉด ์˜ˆ๋ฅผ ๋“ค์–ด <code class="notranslate" translate="no">inputManager.keys.left.down</code>์„ ์ฒดํฌํ•˜๋ฉด ๋˜๊ณ , ํ•ด๋‹น ๊ฐ์ฒด์— <code class="notranslate" translate="no">justPressed</code>๋ฅผ ์ฒดํฌํ•˜๋ฉด ์‚ฌ์šฉ์ž๊ฐ€ ํ•ด๋‹น ํ”„๋ ˆ์ž„์—์„œ ํ‚ค๋ฅผ ๋ˆŒ๋ €๋Š”์ง€ ํ™•์ธํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค. ์˜ˆ๋ฅผ ๋“ค์–ด ์ ํ”„๋ฅผ ๊ตฌํ˜„ํ•  ๊ฒฝ์šฐ ์œ ์ €๊ฐ€ ํ‚ค๋ฅผ ๋ˆ„๋ฅด๊ณ  ์žˆ๋Š”์ง€๋ฅผ ์ถ”์ ํ•  ์ด์œ ๋Š” ์—†๊ฒ ์ฃ . ๋‹จ์ˆœํžˆ ํ•ด๋‹น ํ”„๋ ˆ์ž„์—์„œ ํ‚ค๋ฅผ ๋ˆŒ๋ €๋Š”์ง€๋งŒ ํ™•์ธํ•˜๋ฉด ๋  ๊ฒ๋‹ˆ๋‹ค.</p>
  652. <p>์ด์ œ <code class="notranslate" translate="no">InputManager</code>์˜ ์ธ์Šคํ„ด์Šค๋ฅผ ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค.</p>
  653. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const globals = {
  654. time: 0,
  655. deltaTime: 0,
  656. };
  657. const gameObjectManager = new GameObjectManager();
  658. +const inputManager = new InputManager();
  659. </pre>
  660. <p>๊ทธ๋ฆฌ๊ณ  ๋ Œ๋”๋ง ๋ฃจํ”„์—์„œ <code class="notranslate" translate="no">update</code> ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•˜๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.</p>
  661. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function render(now) {
  662. ...
  663. gameObjectManager.update();
  664. + inputManager.update();
  665. ...
  666. }
  667. </pre>
  668. <p><code class="notranslate" translate="no">gameObjectManager.update</code> ์ „์— ์ด ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด <code class="notranslate" translate="no">justPressed</code>๊ฐ€ ํ•ญ์ƒ <code class="notranslate" translate="no">false</code>์ผ ํ…Œ๋‹ˆ <code class="notranslate" translate="no">gameObjectManager.update</code> ๋’ค์— ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•˜๋„๋ก ํ–ˆ์Šต๋‹ˆ๋‹ค.</p>
  669. <p>์ด์ œ <code class="notranslate" translate="no">Player</code> ์ปดํฌ๋„ŒํŠธ์— ์‚ฌ์šฉ์ž ์ž…๋ ฅ์„ ์ถ”๊ฐ€ํ•ด๋ด…์‹œ๋‹ค.</p>
  670. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">+const kForward = new THREE.Vector3(0, 0, 1);
  671. const globals = {
  672. time: 0,
  673. deltaTime: 0,
  674. + moveSpeed: 16,
  675. };
  676. class Player extends Component {
  677. constructor(gameObject) {
  678. super(gameObject);
  679. const model = models.knight;
  680. this.skinInstance = gameObject.addComponent(SkinInstance, model);
  681. this.skinInstance.setAnimation('Run');
  682. + this.turnSpeed = globals.moveSpeed / 4;
  683. }
  684. + update() {
  685. + const { deltaTime, moveSpeed } = globals;
  686. + const { transform } = this.gameObject;
  687. + const delta = (inputManager.keys.left.down ? 1 : 0) +
  688. + (inputManager.keys.right.down ? -1 : 0);
  689. + transform.rotation.y += this.turnSpeed * delta * deltaTime;
  690. + transform.translateOnAxis(kForward, moveSpeed * deltaTime);
  691. + }
  692. }
  693. </pre>
  694. <p>์œ„ ์ฝ”๋“œ์—์„œ๋Š” ํ”Œ๋ ˆ์ด์–ด๋ฅผ ์•ž์œผ๋กœ ์›€์ง์ด๊ธฐ ์œ„ํ•ด <a href="/docs/#api/ko/core/Object3D.transformOnAxis"><code class="notranslate" translate="no">Object3D.transformOnAxis</code></a>๋ฅผ ์‚ฌ์šฉํ–ˆ์Šต๋‹ˆ๋‹ค. <a href="/docs/#api/ko/core/Object3D.transformOnAxis"><code class="notranslate" translate="no">Object3D.transformOnAxis</code></a>๋Š” ์ง€์—ญ ๊ณต๊ฐ„์„ ๊ธฐ์ค€์œผ๋กœ ํ•˜๊ธฐ์— ํ•ด๋‹น ๊ฐ์ฒด๊ฐ€ ์žฅ๋ฉด์˜ ๋ฃจํŠธ(root) ์š”์†Œ์— ์†ํ•  ๋•Œ๋งŒ ์ •์ƒ์ ์œผ๋กœ ์ž‘๋™ํ•ฉ๋‹ˆ๋‹ค. <a class="footnote" href="#parented" id="parented-backref">1</a></p>
  695. <p>๋˜ํ•œ ์ „์—ญ ๊ฐ์ฒด์— <code class="notranslate" translate="no">moveSpeed</code>๋ฅผ ์ถ”๊ฐ€ํ–ˆ๊ณ  ์ด๋ฅผ <code class="notranslate" translate="no">turnSpeed</code>์˜ ๊ธฐ์ค€์œผ๋กœ ์‚ผ์•˜์Šต๋‹ˆ๋‹ค. ์ด๋Š” ์บ๋ฆญํ„ฐ๊ฐ€ ๋ชฉํ‘œ๋ฅผ ํ–ฅํ•ด ์ƒ๋Œ€์ ์œผ๋กœ ๋น ๋ฅด๊ฒŒ ๋Œ๋„๋ก ๋งŒ๋“  ๊ฒƒ์œผ๋กœ, ์ด <code class="notranslate" translate="no">turnSpeed</code>์˜ ๊ฐ’์ด ๋„ˆ๋ฌด ์ž‘๋‹ค๋ฉด ์บ๋ฆญํ„ฐ๋Š” ๋ชฉํ‘œ ์ฃผ์œ„๋ฅผ ๋น™๋น™ ๋Œ๊ธฐ๋งŒ ํ•˜๊ณ  ์ ˆ๋Œ€ ๋ชฉํ‘œ์— ๋‹ฟ์ง€๋Š” ๋ชปํ•  ๊ฒ๋‹ˆ๋‹ค. ๋ฌผ๋ก  ์œ„ ๊ฐ’์€ ์–ด๋–ค ์ˆ˜ํ•™์  ๊ณต์‹์„ ์‚ฌ์šฉํ•œ ๊ฒƒ์ด ์•„๋‹™๋‹ˆ๋‹ค. ๊ทธ๋ƒฅ ๋Œ€์ถฉ ๋•Œ๋ ค ๋„ฃ์€ ๊ฒƒ์ด์ฃ .</p>
  696. <p>์ด๋Œ€๋กœ๋„ ์˜ˆ์ œ๋Š” ์ž˜ ์ž‘๋™ํ•  ํ…Œ์ง€๋งŒ ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ํ™”๋ฉด์„ ๋ฒ—์–ด๋‚˜๋ฉด ์บ๋ฆญํ„ฐ๊ฐ€ ์–ด๋”” ์žˆ๋Š”์ง€ ์ฐพ๊ธฐ๊ฐ€ ์–ด๋ ค์šธ ๊ฒ๋‹ˆ๋‹ค. ์ผ๋‹จ ํ™”๋ฉด์—์„œ ๋ฒ—์–ด๋‚œ ๋’ค ์ผ์ • ์‹œ๊ฐ„์ด ์ง€๋‚˜๋ฉด ํ”Œ๋ ˆ์ด์–ด๋ฅผ ๋‹ค์‹œ ์ค‘์ ์œผ๋กœ ์ˆœ๊ฐ„์ด๋™์‹œํ‚ค๊ธฐ๋กœ ํ•ฉ์‹œ๋‹ค. Three.js์˜ <a href="/docs/#api/ko/math/Frustum"><code class="notranslate" translate="no">Frustum</code></a> ํด๋ž˜์Šค๋ฅผ ์ด์šฉํ•˜๋ฉด ํŠน์ • ์ ์ด ์นด๋ฉ”๋ผ์˜ ์ ˆ๋‘์ฒด(frustum) ์•ˆ์— ์žˆ๋Š”์ง€ ์•Œ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  697. <p>๋จผ์ € ์นด๋ฉ”๋ผ๋กœ ์ ˆ๋‘์ฒด๋ฅผ ๋งŒ๋“ค์–ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ํ”Œ๋ ˆ์ด์–ด ์ปดํฌ๋„ŒํŠธ์—์„œ ์ด๊ฑธ ์ฒ˜๋ฆฌํ•  ์ˆ˜๋„ ์žˆ์ง€๋งŒ, ๋‹ค๋ฅธ ์š”์†Œ๋„ ์ด ๋ฐฉ๋ฒ•์„ ์จ์•ผ ํ•  ์ˆ˜ ์žˆ์œผ๋‹ˆ ์นด๋ฉ”๋ผ์˜ ์ ˆ๋‘์ฒด๋ฅผ ๊ด€๋ฆฌํ•˜๋Š” ์ƒˆ๋กœ์šด ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋งŒ๋“ค๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  698. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class CameraInfo extends Component {
  699. constructor(gameObject) {
  700. super(gameObject);
  701. this.projScreenMatrix = new THREE.Matrix4();
  702. this.frustum = new THREE.Frustum();
  703. }
  704. update() {
  705. const { camera } = globals;
  706. this.projScreenMatrix.multiplyMatrices(
  707. camera.projectionMatrix,
  708. camera.matrixWorldInverse);
  709. this.frustum.setFromProjectionMatrix(this.projScreenMatrix);
  710. }
  711. }
  712. </pre>
  713. <p>๋‹ค์Œ์œผ๋กœ <code class="notranslate" translate="no">init</code> ํ•จ์ˆ˜์—์„œ ๋ฐฉ๊ธˆ ๋งŒ๋“  ์ปดํฌ๋„ŒํŠธ๋กœ ์ƒˆ๋กœ์šด GameObject๋ฅผ ์ถ”๊ฐ€ํ•ฉ๋‹ˆ๋‹ค.</p>
  714. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function init() {
  715. // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  716. const loadingElem = document.querySelector('#loading');
  717. loadingElem.style.display = 'none';
  718. prepModelsAndAnimations();
  719. + {
  720. + const gameObject = gameObjectManager.createGameObject(camera, 'camera');
  721. + globals.cameraInfo = gameObject.addComponent(CameraInfo);
  722. + }
  723. {
  724. const gameObject = gameObjectManager.createGameObject(scene, 'player');
  725. gameObject.addComponent(Player);
  726. }
  727. }
  728. </pre>
  729. <p><code class="notranslate" translate="no">Player</code> ์ปดํฌ๋„ŒํŠธ์— ๋ฐฉ๊ธˆ ๋งŒ๋“  GameObject๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ์ฝ”๋“œ๋ฅผ ์ถ”๊ฐ€ํ•ฉ๋‹ˆ๋‹ค.</p>
  730. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Player extends Component {
  731. constructor(gameObject) {
  732. super(gameObject);
  733. const model = models.knight;
  734. this.skinInstance = gameObject.addComponent(SkinInstance, model);
  735. this.skinInstance.setAnimation('Run');
  736. this.turnSpeed = globals.moveSpeed / 4;
  737. + this.offscreenTimer = 0;
  738. + this.maxTimeOffScreen = 3;
  739. }
  740. update() {
  741. - const { deltaTime, moveSpeed } = globals;
  742. + const { deltaTime, moveSpeed, cameraInfo } = globals;
  743. const { transform } = this.gameObject;
  744. const delta = (inputManager.keys.left.down ? 1 : 0) +
  745. (inputManager.keys.right.down ? -1 : 0);
  746. transform.rotation.y += this.turnSpeed * delta * deltaTime;
  747. transform.translateOnAxis(kForward, moveSpeed * deltaTime);
  748. + const { frustum } = cameraInfo;
  749. + if (frustum.containsPoint(transform.position)) {
  750. + this.offscreenTimer = 0;
  751. + } else {
  752. + this.offscreenTimer += deltaTime;
  753. + if (this.offscreenTimer &gt;= this.maxTimeOffScreen) {
  754. + transform.position.set(0, 0, 0);
  755. + }
  756. + }
  757. }
  758. }
  759. </pre>
  760. <p>์˜ˆ์ œ๋ฅผ ์‹คํ–‰ํ•˜๊ธฐ ์ „์— ๋ชจ๋ฐ”์ผ ํ™˜๊ฒฝ์„ ์œ„ํ•œ ํ„ฐ์น˜ ์ธํ„ฐํŽ˜์ด์Šค๋ฅผ ์ถ”๊ฐ€ํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค. ๋จผ์ € ํ„ฐ์น˜ ์ด๋ฒคํŠธ๋ฅผ ๋ฐ›์„ HTML ์š”์†Œ๋ฅผ ๋งŒ๋“ญ๋‹ˆ๋‹ค.</p>
  761. <pre class="prettyprint showlinemods notranslate lang-html" translate="no">&lt;body&gt;
  762. &lt;canvas id="c"&gt;&lt;/canvas&gt;
  763. + &lt;div id="ui"&gt;
  764. + &lt;div id="left"&gt;&lt;img src="../resources/images/left.svg"&gt;&lt;/div&gt;
  765. + &lt;div style="flex: 0 0 40px;"&gt;&lt;/div&gt;
  766. + &lt;div id="right"&gt;&lt;img src="../resources/images/right.svg"&gt;&lt;/div&gt;
  767. + &lt;/div&gt;
  768. &lt;div id="loading"&gt;
  769. &lt;div&gt;
  770. &lt;div&gt;...loading...&lt;/div&gt;
  771. &lt;div class="progress"&gt;&lt;div id="progressbar"&gt;&lt;/div&gt;&lt;/div&gt;
  772. &lt;/div&gt;
  773. &lt;/div&gt;
  774. &lt;/body&gt;
  775. </pre>
  776. <p>๋ฒ„ํŠผ์˜ ์Šคํƒ€์ผ๋„ ์ž‘์„ฑํ•ฉ๋‹ˆ๋‹ค.</p>
  777. <pre class="prettyprint showlinemods notranslate lang-css" translate="no">#ui {
  778. position: absolute;
  779. left: 0;
  780. top: 0;
  781. width: 100%;
  782. height: 100%;
  783. display: flex;
  784. justify-items: center;
  785. align-content: stretch;
  786. }
  787. #ui&gt;div {
  788. display: flex;
  789. align-items: flex-end;
  790. flex: 1 1 auto;
  791. }
  792. .bright {
  793. filter: brightness(2);
  794. }
  795. #left {
  796. justify-content: flex-end;
  797. }
  798. #right {
  799. justify-content: flex-start;
  800. }
  801. #ui img {
  802. padding: 10px;
  803. width: 80px;
  804. height: 80px;
  805. display: block;
  806. }
  807. </pre>
  808. <p>์ œ๊ฐ€ ์‚ฌ์šฉํ•œ ๋ฐฉ๋ฒ•์€ ํ•˜๋‚˜์˜ div ์š”์†Œ, <code class="notranslate" translate="no">#ui</code>๋กœ ํ™”๋ฉด ์ „์ฒด๋ฅผ ์ฑ„์šฐ๊ณ , ํ•ด๋‹น ์š”์†Œ์˜ ์ž์‹์œผ๋กœ ํ™”๋ฉด์˜ ๋Œ€๋žต ๋ฐ˜์„ ์ฐจ์ง€ํ•˜๋Š” <code class="notranslate" translate="no">#left</code>์™€ <code class="notranslate" translate="no">#right</code>๋ฅผ ๊ฐ๊ฐ ์–‘์ชฝ์—, ๊ฐ€์šด๋ฐ์—๋Š” 40px์งœ๋ฆฌ ๊ตฌ๋ถ„์„ ์„ ๋„ฃ์–ด ํ™”๋ฉด ์ „์ฒด๊ฐ€ ์ด๋ฒคํŠธ๋ฅผ ๊ฐ์ง€ํ•˜๋„๋ก ํ•œ ๊ฒƒ์ž…๋‹ˆ๋‹ค. ์ด๋Ÿฌ๋ฉด ์‚ฌ์šฉ์ž๊ฐ€ ์™ผ์ชฝ ํ™”์‚ดํ‘œ๋ฅผ ๋ˆ„๋ฅธ ๋’ค ์™ผ์ชฝ์—์„œ ์˜ค๋ฅธ์ชฝ์œผ๋กœ ์†๊ฐ€๋ฝ์„ ์›€์ง์˜€์„ ๋•Œ <code class="notranslate" translate="no">InputManager</code>์˜ <code class="notranslate" translate="no">keys.left</code>๊ณผ <code class="notranslate" translate="no">keys.right</code>๋ฅผ ์—…๋ฐ์ดํŠธํ•  ์ˆ˜ ์žˆ๊ฒ ์ฃ . ๊ตณ์ด ์ž‘์€ ํ™”์‚ดํ‘œ๋ฅผ ๋ˆ„๋ฅด๋А๋ผ ๊ณ ์ƒํ•˜์ง€ ์•Š์•„๋„ ๋˜๋‹ˆ ์ด ํŽธ์ด ํ›จ์”ฌ ๋‚˜์„ ๊ฒ๋‹ˆ๋‹ค.</p>
  809. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class InputManager {
  810. constructor() {
  811. this.keys = {};
  812. const keyMap = new Map();
  813. const setKey = (keyName, pressed) =&gt; {
  814. const keyState = this.keys[keyName];
  815. keyState.justPressed = pressed &amp;&amp; !keyState.down;
  816. keyState.down = pressed;
  817. };
  818. const addKey = (keyCode, name) =&gt; {
  819. this.keys[name] = { down: false, justPressed: false };
  820. keyMap.set(keyCode, name);
  821. };
  822. const setKeyFromKeyCode = (keyCode, pressed) =&gt; {
  823. const keyName = keyMap.get(keyCode);
  824. if (!keyName) {
  825. return;
  826. }
  827. setKey(keyName, pressed);
  828. };
  829. addKey(37, 'left');
  830. addKey(39, 'right');
  831. addKey(38, 'up');
  832. addKey(40, 'down');
  833. addKey(90, 'a');
  834. addKey(88, 'b');
  835. window.addEventListener('keydown', (e) =&gt; {
  836. setKeyFromKeyCode(e.keyCode, true);
  837. });
  838. window.addEventListener('keyup', (e) =&gt; {
  839. setKeyFromKeyCode(e.keyCode, false);
  840. });
  841. + const sides = [
  842. + { elem: document.querySelector('#left'), key: 'left' },
  843. + { elem: document.querySelector('#right'), key: 'right' },
  844. + ];
  845. +
  846. + const clearKeys = () =&gt; {
  847. + for (const {key} of sides) {
  848. + setKey(key, false);
  849. + }
  850. + };
  851. +
  852. + const handleMouseMove = (e) =&gt; {
  853. + e.preventDefault();
  854. + // this is needed because we call preventDefault();
  855. + // we also gave the canvas a tabindex so it can
  856. + // become the focus
  857. + canvas.focus();
  858. + window.addEventListener('pointermove', handleMouseMove);
  859. + window.addEventListener('pointerup', handleMouseUp);
  860. +
  861. + for (const {elem, key} of sides) {
  862. + let pressed = false;
  863. + const rect = elem.getBoundingClientRect();
  864. + const x = e.clientX;
  865. + const y = e.clientY;
  866. + const inRect = x &gt;= rect.left &amp;&amp; x &lt; rect.right &amp;&amp;
  867. + y &gt;= rect.top &amp;&amp; y &lt; rect.bottom;
  868. + if (inRect) {
  869. + pressed = true;
  870. + }
  871. + setKey(key, pressed);
  872. + }
  873. + };
  874. +
  875. + function handleMouseUp() {
  876. + clearKeys();
  877. + window.removeEventListener('pointermove', handleMouseMove, {passive: false});
  878. + window.removeEventListener('pointerup', handleMouseUp);
  879. + }
  880. +
  881. + const uiElem = document.querySelector('#ui');
  882. + uiElem.addEventListener('pointerdown', handleMouseMove, {passive: false});
  883. +
  884. + uiElem.addEventListener('touchstart', (e) =&gt; {
  885. + // prevent scrolling
  886. + e.preventDefault();
  887. + }, {passive: false});
  888. }
  889. update() {
  890. for (const keyState of Object.values(this.keys)) {
  891. if (keyState.justPressed) {
  892. keyState.justPressed = false;
  893. }
  894. }
  895. }
  896. }
  897. </pre>
  898. <p>์ด์ œ ํ™”์‚ดํ‘œ ํ‚ค๋‚˜ ํ™”๋ฉด์„ ํ„ฐ์น˜ํ•ด ์บ๋ฆญํ„ฐ๋ฅผ ์›€์ง์ผ ์ˆ˜ ์žˆ์„ ๊ฒ๋‹ˆ๋‹ค.</p>
  899. <p></p><div translate="no" class="threejs_example_container notranslate">
  900. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/game-player-input.html"></iframe></div>
  901. <a class="threejs_center" href="/manual/examples/game-player-input.html" target="_blank">์ƒˆ ํƒญ์—์„œ ๋ณด๊ธฐ</a>
  902. </div>
  903. <p></p>
  904. <p>๋ฌผ๋ก  ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ํ™”๋ฉด ๋ฐ–์œผ๋กœ ๋‚˜๊ฐ”์„ ๋•Œ ์นด๋ฉ”๋ผ๋ฅผ ์›€์ง์ด๊ฑฐ๋‚˜, "ํ™”๋ฉด ๋ฐ– = ์ฃฝ์Œ"์ด๋ผ๋Š” ์„ค์ •์„ ๋„ฃ์„ ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์ด๊ฒƒ๊นŒ์ง€ ๋‹ค๋ฃฌ๋‹ค๋ฉด ์•ˆ ๊ทธ๋ž˜๋„ ๊ธด ๊ธ€์ด ๋” ๊ธธ์–ด์งˆ ํ…Œ๋‹ˆ ์ด ๋ฐฉ๋ฒ•์œผ๋กœ ๋งŒ์กฑํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  905. <p>์ด์ œ ๋™๋ฌผ์„ ์ถ”๊ฐ€ํ•ด๋ด…์‹œ๋‹ค. <code class="notranslate" translate="no">Player</code>์™€ ๋น„์Šทํ•œ ๋ฐฉ๋ฒ•์œผ๋กœ <code class="notranslate" translate="no">Animal</code> ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋งŒ๋“ญ๋‹ˆ๋‹ค.</p>
  906. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Animal extends Component {
  907. constructor(gameObject, model) {
  908. super(gameObject);
  909. const skinInstance = gameObject.addComponent(SkinInstance, model);
  910. skinInstance.mixer.timeScale = globals.moveSpeed / 4;
  911. skinInstance.setAnimation('Idle');
  912. }
  913. }
  914. </pre>
  915. <p>์œ„ ์ฝ”๋“œ์—์„œ๋Š” <a href="/docs/#api/ko/animation/AnimationMixer.timeScale"><code class="notranslate" translate="no">AnimationMixer.timeScale</code></a>์„ ์„ค์ •ํ•ด ์• ๋‹ˆ๋ฉ”์ด์…˜ ์†๋„๊ฐ€ ์ด๋™ ์†๋„์— ๋น„๋ก€ํ•˜๋„๋ก ๋งŒ๋“ค์—ˆ์Šต๋‹ˆ๋‹ค. ์ด๋Ÿฌ๋ฉด ์ด๋™ ์†๋„์™€ ๊ฐ™์ด ์• ๋‹ˆ๋ฉ”์ด์…˜ ์†๋„๊ฐ€ ๋นจ๋ผ์ง€๊ณ  ๋А๋ ค์ง€๊ฒ ์ฃ .</p>
  916. <p>๋‹ค์Œ์œผ๋กœ <code class="notranslate" translate="no">init</code> ํ•จ์ˆ˜์—์„œ ๊ฐ ๋™๋ฌผ์„ ๋ฐฐ์น˜ํ•ฉ๋‹ˆ๋‹ค.</p>
  917. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function init() {
  918. // ํ”„๋กœ๊ทธ๋ž˜์Šค ๋ฐ”๋ฅผ ์ˆจ๊น๋‹ˆ๋‹ค.
  919. const loadingElem = document.querySelector('#loading');
  920. loadingElem.style.display = 'none';
  921. prepModelsAndAnimations();
  922. {
  923. const gameObject = gameObjectManager.createGameObject(camera, 'camera');
  924. globals.cameraInfo = gameObject.addComponent(CameraInfo);
  925. }
  926. {
  927. const gameObject = gameObjectManager.createGameObject(scene, 'player');
  928. globals.player = gameObject.addComponent(Player);
  929. globals.congaLine = [gameObject];
  930. }
  931. + const animalModelNames = [
  932. + 'pig',
  933. + 'cow',
  934. + 'llama',
  935. + 'pug',
  936. + 'sheep',
  937. + 'zebra',
  938. + 'horse',
  939. + ];
  940. + animalModelNames.forEach((name, ndx) =&gt; {
  941. + const gameObject = gameObjectManager.createGameObject(scene, name);
  942. + gameObject.addComponent(Animal, models[name]);
  943. + gameObject.transform.position.x = (ndx + 1) * 5;
  944. + });
  945. }
  946. </pre>
  947. <p>๋™๋ฌผ๋“ค์„ ๋ฐฐ์น˜ํ•˜๊ณ  ๋๋‚ด๋ฉด ์‹ฌ์‹ฌํ•˜๋‹ˆ ๋ญ”๊ฐ€๋ฅผ ์ถ”๊ฐ€ํ•ด์•ผ๊ฒ ๋„ค์š”.</p>
  948. <p>๋™๋ฌผ์ด ํ”Œ๋ ˆ์ด์–ด๋ฅผ ๋”ฐ๋ผ ๊ธฐ์ฐจ๋†€์ด๏ผŠ๋ฅผ ํ•˜๊ฒŒ ํ•ด๋ด…์‹œ๋‹ค. ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ๋™๋ฌผ์— ๊ฐ€๊นŒ์ด ๊ฐ”์„ ๋•Œ๋งŒ ๊ธฐ์ฐจ์— ํ•ฉ๋ฅ˜ํ•˜๋„๋ก ํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค. ์ด๋ฅผ ๊ตฌํ˜„ํ•˜๋ ค๋ฉด ์•„๋ž˜์™€ ๊ฐ™์€ ๋ชจ์…˜(์ƒํƒœ, state)์ด ํ•„์š”ํ•  ๊ฒ๋‹ˆ๋‹ค.</p>
  949. <p>โ€ป ์›๋ฌธ์€ "conga line"์ž…๋‹ˆ๋‹ค. ๊ธฐ์ฐจ๋†€์ด์™€ ์œ ์‚ฌํ•œ ๊ผฌ๋ฆฌ์ž‡๊ธฐ ๋†€์ด๋กœ, ์šฐ๋ฆฌ์—๊ฒŒ ๋” ์ต์ˆ™ํ•œ "๊ธฐ์ฐจ๋†€์ด"๋กœ ์˜์—ญํ–ˆ์Šต๋‹ˆ๋‹ค. ์—ญ์ฃผ.</p>
  950. <ul>
  951. <li><p>๊ฐ€๋งŒํžˆ ์„œ ์žˆ๋Š” ๋ชจ์…˜(Idle):</p>
  952. <p>ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ๊ฐ€๊นŒ์›Œ์ง€๊ธฐ ์ „๊นŒ์ง€์˜ ๋ชจ์…˜์ž…๋‹ˆ๋‹ค.</p>
  953. </li>
  954. <li><p>๊ธฐ์ฐจ์˜ ๋์— ๊ฐˆ ๋•Œ๊นŒ์ง€ ๊ธฐ๋‹ค๋ฆฌ๋Š” ๋ชจ์…˜(Wait for End of Line):</p>
  955. <p>ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ๋™๋ฌผ๊ณผ ๋‹ฟ๋”๋ผ๋„ ๊ธฐ์ฐจ์˜ ๋์— ํ•ฉ๋ฅ˜ํ•ด์•ผ ํ•˜๋ฏ€๋กœ ๊ทธ ์ „๊นŒ์ง€ ๊ธฐ๋‹ค๋ฆฌ๋Š” ๋ชจ์…˜์ž…๋‹ˆ๋‹ค.</p>
  956. </li>
  957. <li><p>๋”ฐ๋ผ๋ถ™๊ธฐ(Go to Last):</p>
  958. <p>์ž์‹ ์ด ๋”ฐ๋ผ๊ฐˆ ๋Œ€์ƒ์ด ์žˆ๋˜ ์œ„์น˜๋กœ ์ด๋™ํ•จ๊ณผ ๋™์‹œ์— ๋”ฐ๋ผ๊ฐˆ ๋Œ€์ƒ์ด ์–ด๋”” ์žˆ๋Š”์ง€ ๊ธฐ๋กํ•ฉ๋‹ˆ๋‹ค.</p>
  959. </li>
  960. <li><p>๋”ฐ๋ผ๊ฐ€๊ธฐ(Follow):</p>
  961. <p>์ž์‹ ์ด ๋”ฐ๋ผ๊ฐ€๋Š” ๋Œ€์ƒ์˜ ํ˜„์žฌ ์œ„์น˜๋ฅผ ๊ธฐ๋กํ•จ๊ณผ ๋™์‹œ์— ๋Œ€์ƒ์ด ์žˆ์—ˆ๋˜ ์œ„์น˜๋กœ ์ด๋™ํ•ฉ๋‹ˆ๋‹ค.</p>
  962. </li>
  963. </ul>
  964. <p>์ด๋Ÿฐ ์ƒํƒœ๋ฅผ ๋‹ค๋ฃฐ ๋ฐฉ๋ฒ•์€ ์•„์ฃผ ๋‹ค์–‘ํ•ฉ๋‹ˆ๋‹ค. ๋ณดํ†ต์€ <a href="https://www.google.com/search?q=finite+state+machine">์œ ํ•œ ์ƒํƒœ ๊ธฐ๊ณ„(Finite State Machine)</a>์™€ ์ด๋Ÿฐ ์ƒํƒœ๋ฅผ ๋‹ค๋ฃฐ ํ—ฌํผ ํด๋ž˜์Šค๋ฅผ ์‚ฌ์šฉํ•˜์ฃ .</p>
  965. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class FiniteStateMachine {
  966. constructor(states, initialState) {
  967. this.states = states;
  968. this.transition(initialState);
  969. }
  970. get state() {
  971. return this.currentState;
  972. }
  973. transition(state) {
  974. const oldState = this.states[this.currentState];
  975. if (oldState &amp;&amp; oldState.exit) {
  976. oldState.exit.call(this);
  977. }
  978. this.currentState = state;
  979. const newState = this.states[state];
  980. if (newState.enter) {
  981. newState.enter.call(this);
  982. }
  983. }
  984. update() {
  985. const state = this.states[this.currentState];
  986. if (state.update) {
  987. state.update.call(this);
  988. }
  989. }
  990. }
  991. </pre>
  992. <p>์œ„ ํด๋ž˜์Šค๋Š” ์•ž์„œ ๋งํ•œ ํ—ฌํผ ํด๋ž˜์Šค๋ฅผ ๊ฐ„๋‹จํžˆ ๊ตฌํ˜„ํ•œ ๊ฒƒ์ž…๋‹ˆ๋‹ค. ํด๋ž˜์Šค๋Š” ์ƒ์„ฑ ์‹œ ์ƒํƒœ๋“ค์˜ ๊ฐ์ฒด๋ฅผ ๋ฐ›๊ณ , ๊ฐ ์ƒํƒœ์—๋Š” <code class="notranslate" translate="no">enter</code>, <code class="notranslate" translate="no">update</code>, <code class="notranslate" translate="no">exit</code>์ด๋ผ๋Š” ๋ฉ”์„œ๋“œ๊ฐ€ ์žˆ์Šต๋‹ˆ๋‹ค. ์ƒํƒœ๋ฅผ ๋ฐ”๊พธ๋ ค๋ฉด <code class="notranslate" translate="no">FiniteStateMachine.transition</code>์„ ํ˜ธ์ถœํ•  ๋•Œ ์ƒˆ๋กœ์šด ์ด๋ฆ„์„ ๋„˜๊ฒจ์ฃผ๋ฉด ๋˜์ฃ . ๋งŒ์•ฝ ํ˜„์žฌ ์ƒํƒœ์— <code class="notranslate" translate="no">exit</code> ๋ฉ”์„œ๋“œ๊ฐ€ ์žˆ๋‹ค๋ฉด ํ•ด๋‹น ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค. ๊ทธ๋ฆฌ๊ณ  ์ƒˆ๋กœ์šด ์ƒํƒœ์— <code class="notranslate" translate="no">enter</code> ๋ฉ”์„œ๋“œ๊ฐ€ ์žˆ์„ ๊ฒฝ์šฐ <code class="notranslate" translate="no">enter</code> ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค. ๋งˆ์ง€๋ง‰์œผ๋กœ ๋งค ํ”„๋ ˆ์ž„๋งˆ๋‹ค <code class="notranslate" translate="no">FiniteStateMachine.update</code>๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด ๊ฐ ์ƒํƒœ์˜ <code class="notranslate" translate="no">update</code> ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค.</p>
  993. <p>์ด์ œ ์ด ํด๋ž˜์Šค๋ฅผ ํ™œ์šฉํ•ด ๋™๋ฌผ๋“ค์˜ ์ƒํƒœ๋ฅผ ๋ฐ”๊ฟ”๋ด…์‹œ๋‹ค.</p>
  994. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">// ๋งค๊ฐœ๋ณ€์ˆ˜ obj1๊ณผ obj2์ด ๊ฐ€๊น๋‹ค๋ฉด true๋ฅผ ๋ฐ˜ํ™˜ํ•ฉ๋‹ˆ๋‹ค.
  995. function isClose(obj1, obj1Radius, obj2, obj2Radius) {
  996. const minDist = obj1Radius + obj2Radius;
  997. const dist = obj1.position.distanceTo(obj2.position);
  998. return dist &lt; minDist;
  999. }
  1000. // v ์˜ ๊ฐ’์ด -min๊ณผ +min ์‚ฌ์ด๊ฐ€ ๋˜๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.
  1001. function minMagnitude(v, min) {
  1002. return Math.abs(v) &gt; min
  1003. ? min * Math.sign(v)
  1004. : v;
  1005. }
  1006. const aimTowardAndGetDistance = function() {
  1007. const delta = new THREE.Vector3();
  1008. return function aimTowardAndGetDistance(source, targetPos, maxTurn) {
  1009. delta.subVectors(targetPos, source.position);
  1010. // ๋ฐ”๋ผ๋ณผ ๋ฐฉํ–ฅ์„ ๊ณ„์‚ฐํ•ฉ๋‹ˆ๋‹ค.
  1011. const targetRot = Math.atan2(delta.x, delta.z) + Math.PI * 1.5;
  1012. // ๋” ๊ฐ€๊นŒ์šด ๋ฐฉํ–ฅ์œผ๋กœ ํšŒ์ „ํ•ฉ๋‹ˆ๋‹ค.
  1013. const deltaRot = (targetRot - source.rotation.y + Math.PI * 1.5) % (Math.PI * 2) - Math.PI;
  1014. // maxTurn๋ณด๋‹ค ๋น ๋ฅธ ์†๋„๋กœ ๋Œ์ง€ ์•Š๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.
  1015. const deltaRotation = minMagnitude(deltaRot, maxTurn);
  1016. // rotation ๊ฐ’์„ 0์—์„œ Math.PI * 2 ์‚ฌ์ด๋กœ ์œ ์ง€ํ•ฉ๋‹ˆ๋‹ค.
  1017. source.rotation.y = THREE.MathUtils.euclideanModulo(
  1018. source.rotation.y + deltaRotation, Math.PI * 2);
  1019. // ๋ชฉํ‘œ๊นŒ์ง€์˜ ๊ฑฐ๋ฆฌ๋ฅผ ๋ฐ˜ํ™˜ํ•ฉ๋‹ˆ๋‹ค.
  1020. return delta.length();
  1021. };
  1022. }();
  1023. class Animal extends Component {
  1024. constructor(gameObject, model) {
  1025. super(gameObject);
  1026. + const hitRadius = model.size / 2;
  1027. const skinInstance = gameObject.addComponent(SkinInstance, model);
  1028. skinInstance.mixer.timeScale = globals.moveSpeed / 4;
  1029. + const transform = gameObject.transform;
  1030. + const playerTransform = globals.player.gameObject.transform;
  1031. + const maxTurnSpeed = Math.PI * (globals.moveSpeed / 4);
  1032. + const targetHistory = [];
  1033. + let targetNdx = 0;
  1034. +
  1035. + function addHistory() {
  1036. + const targetGO = globals.congaLine[targetNdx];
  1037. + const newTargetPos = new THREE.Vector3();
  1038. + newTargetPos.copy(targetGO.transform.position);
  1039. + targetHistory.push(newTargetPos);
  1040. + }
  1041. +
  1042. + this.fsm = new FiniteStateMachine({
  1043. + idle: {
  1044. + enter: () =&gt; {
  1045. + skinInstance.setAnimation('Idle');
  1046. + },
  1047. + update: () =&gt; {
  1048. + // ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ๊ทผ์ฒ˜์— ์žˆ๋Š”์ง€ ํ™•์ธํ•ฉ๋‹ˆ๋‹ค.
  1049. + if (isClose(transform, hitRadius, playerTransform, globals.playerRadius)) {
  1050. + this.fsm.transition('waitForEnd');
  1051. + }
  1052. + },
  1053. + },
  1054. + waitForEnd: {
  1055. + enter: () =&gt; {
  1056. + skinInstance.setAnimation('Jump');
  1057. + },
  1058. + update: () =&gt; {
  1059. + // ๊ธฐ์ฐจ์˜ ๊ฐ€์žฅ ๋งˆ์ง€๋ง‰์— ์žˆ๋Š” gameObject๋ฅผ ๊ฐ€์ ธ์˜ต๋‹ˆ๋‹ค.
  1060. + const lastGO = globals.congaLine[globals.congaLine.length - 1];
  1061. + const deltaTurnSpeed = maxTurnSpeed * globals.deltaTime;
  1062. + const targetPos = lastGO.transform.position;
  1063. + aimTowardAndGetDistance(transform, targetPos, deltaTurnSpeed);
  1064. + // ๊ธฐ์ฐจ์˜ ๋งˆ์ง€๋ง‰์— ์žˆ๋Š” ์š”์†Œ๊ฐ€ ๊ทผ์ฒ˜์— ์žˆ๋Š”์ง€ ํ™•์ธํ•ฉ๋‹ˆ๋‹ค.
  1065. + if (isClose(transform, hitRadius, lastGO.transform, globals.playerRadius)) {
  1066. + this.fsm.transition('goToLast');
  1067. + }
  1068. + },
  1069. + },
  1070. + goToLast: {
  1071. + enter: () =&gt; {
  1072. + // ๋”ฐ๋ผ๊ฐˆ ๋Œ€์ƒ์„ ๊ธฐ๋กํ•ฉ๋‹ˆ๋‹ค. remember who we're following
  1073. + targetNdx = globals.congaLine.length - 1;
  1074. + // ๊ธฐ์ฐจ์˜ ๋งˆ์ง€๋ง‰์— ์Šค์Šค๋กœ๋ฅผ ์ถ”๊ฐ€ํ•ฉ๋‹ˆ๋‹ค.
  1075. + globals.congaLine.push(gameObject);
  1076. + skinInstance.setAnimation('Walk');
  1077. + },
  1078. + update: () =&gt; {
  1079. + addHistory();
  1080. + // ๊ธฐ๋ก๋œ ์œ„์น˜ ์ค‘ ๊ฐ€์žฅ ๋‚˜์ค‘ ์œ„์น˜๋กœ ์ด๋™ํ•ฉ๋‹ˆ๋‹ค.
  1081. + const targetPos = targetHistory[0];
  1082. + const maxVelocity = globals.moveSpeed * globals.deltaTime;
  1083. + const deltaTurnSpeed = maxTurnSpeed * globals.deltaTime;
  1084. + const distance = aimTowardAndGetDistance(transform, targetPos, deltaTurnSpeed);
  1085. + const velocity = distance;
  1086. + transform.translateOnAxis(kForward, Math.min(velocity, maxVelocity));
  1087. + if (distance &lt;= maxVelocity) {
  1088. + this.fsm.transition('follow');
  1089. + }
  1090. + },
  1091. + },
  1092. + follow: {
  1093. + update: () =&gt; {
  1094. + addHistory();
  1095. + // ๊ฐ€์žฅ ์˜ค๋ž˜๋œ ์œ„์น˜๊ฐ’์„ ์ง€์šฐ๊ณ  ์ž๊ธฐ ์ž์‹ ์˜ ์œ„์น˜๊ฐ’์„ ์ถ”๊ฐ€ํ•ฉ๋‹ˆ๋‹ค.
  1096. + const targetPos = targetHistory.shift();
  1097. + transform.position.copy(targetPos);
  1098. + const deltaTurnSpeed = maxTurnSpeed * globals.deltaTime;
  1099. + aimTowardAndGetDistance(transform, targetHistory[0], deltaTurnSpeed);
  1100. + },
  1101. + },
  1102. + }, 'idle');
  1103. + }
  1104. + update() {
  1105. + this.fsm.update();
  1106. + }
  1107. }
  1108. </pre>
  1109. <p>ํ•œ ๋ฒˆ์— ๋„ˆ๋ฌด ๋งŽ์€ ์ฝ”๋“œ๋ฅผ ๋ณด์—ฌ์ค€ ๋“ฏํ•˜์ง€๋งŒ ์œ„ ์ฝ”๋“œ๋Š” ๋ฐฉ๊ธˆ ์–ธ๊ธ‰ํ•œ ์—ญํ• ์„ ํ•ฉ๋‹ˆ๋‹ค. ๊ฐ ์ƒํƒœ์— ๋Œ€ํ•œ ์ฝ”๋“œ๋ฅผ ๋ณด๊ณ  ์–ด๋–ค ์‹์œผ๋กœ ์ž‘๋™ํ•˜๋Š”์ง€ ๋ถ„์„ํ•ด๋ณด๊ธฐ ๋ฐ”๋ž๋‹ˆ๋‹ค.</p>
  1110. <p>์—ฌ๊ธฐ์— ๋ช‡ ๊ฐ€์ง€ ์š”์†Œ๋ฅผ ์ถ”๊ฐ€ํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ์ž๊ธฐ ์ž์‹ ์„ ์ „์—ญ ๊ฐ์ฒด(globals)์— ์ถ”๊ฐ€ํ•ด ๋‹ค๋ฅธ ๋™๋ฌผ์ด ์ž์‹ ์˜ ์œ„์น˜๋ฅผ ์ถ”์ ํ•˜๋„๋ก ํ•ด์•ผ ํ•˜๊ณ , ๋˜ ๊ธฐ์ฐจ์˜ ๋จธ๋ฆฌ๋ฅผ ํ”Œ๋ ˆ์ด์–ด์˜ <code class="notranslate" translate="no">GameObject</code>๋กœ ์ง€์ •ํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.</p>
  1111. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function init() {
  1112. ...
  1113. {
  1114. const gameObject = gameObjectManager.createGameObject(scene, 'player');
  1115. + globals.player = gameObject.addComponent(Player);
  1116. + globals.congaLine = [gameObject];
  1117. }
  1118. }
  1119. </pre>
  1120. <p>๊ฐ ๋ชจ๋ธ์˜ ํฌ๊ธฐ๋„ ๊ณ„์‚ฐํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.</p>
  1121. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function prepModelsAndAnimations() {
  1122. + const box = new THREE.Box3();
  1123. + const size = new THREE.Vector3();
  1124. Object.values(models).forEach(model =&gt; {
  1125. + box.setFromObject(model.gltf.scene);
  1126. + box.getSize(size);
  1127. + model.size = size.length();
  1128. const animsByName = {};
  1129. model.gltf.animations.forEach((clip) =&gt; {
  1130. animsByName[clip.name] = clip;
  1131. // ์ด๋Ÿฐ ๋ถ€๋ถ„์€ .blend ํŒŒ์ผ์—์„œ ์ˆ˜์ •ํ•˜๋Š” ๊ฒŒ ์ข‹์Šต๋‹ˆ๋‹ค.
  1132. if (clip.name === 'Walk') {
  1133. clip.duration /= 2;
  1134. }
  1135. });
  1136. model.animations = animsByName;
  1137. });
  1138. }
  1139. </pre>
  1140. <p>๊ทธ๋ฆฌ๊ณ  ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ์ž๊ธฐ ์ž์‹ ์˜ ํฌ๊ธฐ๋ฅผ ๊ธฐ๋กํ•˜๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.</p>
  1141. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Player extends Component {
  1142. constructor(gameObject) {
  1143. super(gameObject);
  1144. const model = models.knight;
  1145. + globals.playerRadius = model.size / 2;
  1146. </pre>
  1147. <p>์ด์ œ ์™€ ์ƒ๊ฐํ•ด๋ณด๋‹ˆ ํ”Œ๋ ˆ์ด์–ด๊ฐ€ ์•„๋‹ˆ๋ผ ๊ธฐ์ฐจ์˜ ๋จธ๋ฆฌ๋ฅผ ๋ฐ”๋ผ๋ณด๊ฒŒ ํ•˜๋Š” ํŽธ์ด ๋” ๋‚˜์•˜๊ฒ ๋„ค์š”. ์ด๊ฑด ๋‚˜์ค‘์— ๋Œ์•„์™€ ๊ณ ์น˜๋„๋ก ํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  1148. <p>์˜ˆ์ œ๋ฅผ ์ฒ˜์Œ ๋งŒ๋“ค์—ˆ์„ ๋•Œ๋Š” ๋™๋ฌผ๋“ค์ด ๋ชจ๋‘ ๊ฐ™์€ ํฌ๊ธฐ์˜ ๊ฒฝ๊ณ„ ์›(radius)์„ ์ผ์ง€๋งŒ, ์ด๋ ‡๊ฒŒ ํ•˜๊ณ  ๋ณด๋‹ˆ ๋ง๊ณผ ํผ๊ทธ(๊ฐ•์•„์ง€)์˜ ํฌ๊ธฐ๊ฐ€ ๊ฐ™์€ ๊ฒŒ ๋ง์ด ์•ˆ ๋œ๋‹ค๋Š” ์ƒ๊ฐ์ด ๋“ค์—ˆ์Šต๋‹ˆ๋‹ค. ๊ทธ๋ž˜์„œ ๊ฐ ๋ชจ๋ธ์˜ ํฌ๊ธฐ์— ๋”ฐ๋ผ ๊ฒฝ๊ณ„ ์›์„ ๋”ฐ๋กœ ์ง€์ •ํ–ˆ์ฃ . ๊ทธ๋ฆฌ๊ณ  ์ƒํƒœ๋ฅผ ๋ณด์—ฌ์ฃผ๋ฉด ์ข‹๊ฒ ๋‹ค๋Š” ์ƒ๊ฐ์ด ๋“ค์–ด ์ƒํƒœ๋ฅผ ๋ณด์—ฌ ์ค„ <code class="notranslate" translate="no">StatusDisplayHelper</code> ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ถ”๊ฐ€ํ–ˆ์Šต๋‹ˆ๋‹ค.</p>
  1149. <p>๋˜ํ•œ <a href="/docs/#api/ko/helpers/PolarGridHelper"><code class="notranslate" translate="no">PolarGridHelper</code></a>๋ฅผ ์จ ๊ฐ ์บ๋ฆญํ„ฐ์˜ ๊ฒฝ๊ณ„ ์›์ด ๋ณด์ด๋„๋ก ํ–ˆ๊ณ , <a href="align-html-elements-to-3d.html">HTML ์š”์†Œ๋ฅผ 3D๋กœ ์ •๋ ฌํ•˜๊ธฐ</a>์—์„œ ์ผ๋˜ ๋ฐฉ๋ฒ•์œผ๋กœ ๊ฐ ์บ๋ฆญํ„ฐ์˜ ์ƒํƒœ๋ฅผ HTML๋กœ ๋ณด์—ฌ์ฃผ๋„๋ก ํ–ˆ์Šต๋‹ˆ๋‹ค.</p>
  1150. <p>๋จผ์ € ๊ฐ ์š”์†Œ๋ฅผ ๋‹ด์„ HTML์„ ์ถ”๊ฐ€ํ•ฉ๋‹ˆ๋‹ค.</p>
  1151. <pre class="prettyprint showlinemods notranslate lang-html" translate="no">&lt;body&gt;
  1152. &lt;canvas id="c"&gt;&lt;/canvas&gt;
  1153. &lt;div id="ui"&gt;
  1154. &lt;div id="left"&gt;&lt;img src="../resources/images/left.svg"&gt;&lt;/div&gt;
  1155. &lt;div style="flex: 0 0 40px;"&gt;&lt;/div&gt;
  1156. &lt;div id="right"&gt;&lt;img src="../resources/images/right.svg"&gt;&lt;/div&gt;
  1157. &lt;/div&gt;
  1158. &lt;div id="loading"&gt;
  1159. &lt;div&gt;
  1160. &lt;div&gt;...loading...&lt;/div&gt;
  1161. &lt;div class="progress"&gt;&lt;div id="progressbar"&gt;&lt;/div&gt;&lt;/div&gt;
  1162. &lt;/div&gt;
  1163. &lt;/div&gt;
  1164. + &lt;div id="labels"&gt;&lt;/div&gt;
  1165. &lt;/body&gt;
  1166. </pre>
  1167. <p>CSS๋„ ์ž‘์„ฑํ•ฉ๋‹ˆ๋‹ค.</p>
  1168. <pre class="prettyprint showlinemods notranslate lang-css" translate="no">#labels {
  1169. position: absolute; /* ๊ธฐ์ค€ ์š”์†Œ ์œ„๋กœ ์˜ฌ๋ผ๊ฐ€๋„๋ก ํ•ฉ๋‹ˆ๋‹ค. */
  1170. left: 0; /* ๊ธฐ์ค€ ์š”์†Œ ์™ผ์ชฝ ์œ„๋กœ ์ •๋ ฌํ•ฉ๋‹ˆ๋‹ค. */
  1171. top: 0;
  1172. color: white;
  1173. width: 100%;
  1174. height: 100%;
  1175. overflow: hidden;
  1176. pointer-events: none;
  1177. }
  1178. #labels&gt;div {
  1179. position: absolute; /* ๊ธฐ์ค€ ์š”์†Œ๋ฅผ ๊ธฐ์ค€์œผ๋กœ ํ•ฉ๋‹ˆ๋‹ค. */
  1180. left: 0; /* ๊ธฐ์ค€ ์š”์†Œ์˜ ์™ผ์ชฝ ์œ„๋กœ ์ •๋ ฌํ•ฉ๋‹ˆ๋‹ค. */
  1181. top: 0;
  1182. font-size: large;
  1183. font-family: monospace;
  1184. user-select: none; /* ํ…์ŠคํŠธ๋ฅผ ์„ ํƒํ•  ์ˆ˜ ์—†๋„๋ก ํ•ฉ๋‹ˆ๋‹ค. */
  1185. text-shadow: /* ๊ธ€์ž์— ๊ฒ€์€ ์œค๊ณฝ์„ ์„ ๋„ฃ์Šต๋‹ˆ๋‹ค. */
  1186. -1px -1px 0 #000,
  1187. 0 -1px 0 #000,
  1188. 1px -1px 0 #000,
  1189. 1px 0 0 #000,
  1190. 1px 1px 0 #000,
  1191. 0 1px 0 #000,
  1192. -1px 1px 0 #000,
  1193. -1px 0 0 #000;
  1194. }
  1195. </pre>
  1196. <p>์•„๋ž˜๋Š” <code class="notranslate" translate="no">StateDisplayHelper</code> ์ปดํฌ๋„ŒํŠธ์ž…๋‹ˆ๋‹ค.</p>
  1197. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const labelContainerElem = document.querySelector('#labels');
  1198. class StateDisplayHelper extends Component {
  1199. constructor(gameObject, size) {
  1200. super(gameObject);
  1201. this.elem = document.createElement('div');
  1202. labelContainerElem.appendChild(this.elem);
  1203. this.pos = new THREE.Vector3();
  1204. this.helper = new THREE.PolarGridHelper(size / 2, 1, 1, 16);
  1205. gameObject.transform.add(this.helper);
  1206. }
  1207. setState(s) {
  1208. this.elem.textContent = s;
  1209. }
  1210. setColor(cssColor) {
  1211. this.elem.style.color = cssColor;
  1212. this.helper.material.color.set(cssColor);
  1213. }
  1214. update() {
  1215. const { pos } = this;
  1216. const { transform } = this.gameObject;
  1217. const { canvas } = globals;
  1218. pos.copy(transform.position);
  1219. /**
  1220. * ํ•ด๋‹น ์œ„์น˜๊ฐ’์„ ์ •๊ทœํ™”ํ•˜๋ฉด x์™€ y ๊ฐ’์€ -1์—์„œ +1 ์‚ฌ์ด์˜ ๊ฐ’์ด ๋ฉ๋‹ˆ๋‹ค.
  1221. * x = -1 ์ด๋ฉด ์™ผ์ชฝ, y = -1 ์ด๋ฉด ์˜ค๋ฅธ์ชฝ์ด์ฃ .
  1222. **/
  1223. pos.project(globals.camera);
  1224. // ์ •๊ทœํ™”ํ•œ ์œ„์น˜๊ฐ’์„ CSS ์œ„์น˜๊ฐ’์œผ๋กœ ๋ณ€ํ™˜ํ•ฉ๋‹ˆ๋‹ค.
  1225. const x = (pos.x * .5 + .5) * canvas.clientWidth;
  1226. const y = (pos.y * -.5 + .5) * canvas.clientHeight;
  1227. // HTML ์š”์†Œ๋ฅผ ํ•ด๋‹น ์œ„์น˜๋กœ ์˜ฎ๊น๋‹ˆ๋‹ค.
  1228. this.elem.style.transform = `translate(-50%, -50%) translate(${ x }px, ${ y }px)`;
  1229. }
  1230. }
  1231. </pre>
  1232. <p>๋™๋ฌผ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ƒ์„ฑํ•  ๋•Œ ์œ„ ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ถ”๊ฐ€ํ•˜๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.</p>
  1233. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Animal extends Component {
  1234. constructor(gameObject, model) {
  1235. super(gameObject);
  1236. + this.helper = gameObject.addComponent(StateDisplayHelper, model.size);
  1237. ...
  1238. }
  1239. update() {
  1240. this.fsm.update();
  1241. + const dir = THREE.MathUtils.radToDeg(this.gameObject.transform.rotation.y);
  1242. + this.helper.setState(`${ this.fsm.state }:${ dir.toFixed(0) }`);
  1243. }
  1244. }
  1245. </pre>
  1246. <p>์ถ”๊ฐ€๋กœ lil-gui๋ฅผ ์ด์šฉํ•ด ์œ„ ๋””๋ฒ„๊น… ์š”์†Œ๋“ค๋ฅผ ์ผœ๊ณ  ๋Œ ์ˆ˜ ์žˆ๋„๋ก ํ•ฉ๋‹ˆ๋‹ค.</p>
  1247. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">import * as THREE from 'three';
  1248. import { OrbitControls } from 'three/addons/controls/OrbitControls.js';
  1249. import { GLTFLoader } from 'three/addons/loaders/GLTFLoader.js';
  1250. import { SkeletonUtils } from 'three/addons/utils/SkeletonUtils.js';
  1251. +import { GUI } from 'three/addons/libs/lil-gui.module.min.js';
  1252. </pre>
  1253. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">+const gui = new GUI();
  1254. +gui.add(globals, 'debug').onChange(showHideDebugInfo);
  1255. +showHideDebugInfo();
  1256. const labelContainerElem = document.querySelector('#labels');
  1257. +function showHideDebugInfo() {
  1258. + labelContainerElem.style.display = globals.debug ? '' : 'none';
  1259. +}
  1260. +showHideDebugInfo();
  1261. class StateDisplayHelper extends Component {
  1262. ...
  1263. update() {
  1264. + this.helper.visible = globals.debug;
  1265. + if (!globals.debug) {
  1266. + return;
  1267. + }
  1268. ...
  1269. }
  1270. }
  1271. </pre>
  1272. <p>๊ฒŒ์ž„์˜ ๊ฐ€์žฅ ๊ธฐ๋ณธ์ ์ธ ํ‹€์„ ์™„์„ฑํ–ˆ๋„ค์š”.</p>
  1273. <p></p><div translate="no" class="threejs_example_container notranslate">
  1274. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/game-conga-line.html"></iframe></div>
  1275. <a class="threejs_center" href="/manual/examples/game-conga-line.html" target="_blank">์ƒˆ ํƒญ์—์„œ ๋ณด๊ธฐ</a>
  1276. </div>
  1277. <p></p>
  1278. <p>์›๋ž˜ ์ฒ˜์Œ์—๋Š” <a href="https://www.google.com/search?q=snake+game">์ง€๋ ์ด ๊ฒŒ์ž„</a>์„ ๋งŒ๋“ค๋ ค๊ณ  ํ–ˆ์Šต๋‹ˆ๋‹ค. ๋™๋ฌผ์ด ๊ธฐ์ฐจ์— ๋ถ™์–ด ๊ธฐ์ฐจ๊ฐ€ ๊ธธ์–ด์งˆ์ˆ˜๋ก ์žฅ์• ๋ฌผ์„ ํ”ผํ•˜๊ธฐ ์–ด๋ ค์›Œ์ง€๋Š” ๊ฒŒ์ž„์ด์ฃ . ์˜ˆ์ œ์— ๋ช‡ ๊ฐ€์ง€ ์žฅ์• ๋ฌผ ๋†“๊ฑฐ๋‚˜ ํ™”๋ฉด ๋‘˜๋ ˆ์— ๋ฒฝ์„ ์„ธ์šฐ๊ธฐ๋„ ํ–ˆ์Šต๋‹ˆ๋‹ค.</p>
  1279. <p>ํ•˜์ง€๋งŒ ์˜ˆ์ œ์—์„œ ์‚ฌ์šฉํ•œ ๋™๋ฌผ์€ ์ด์— ์ ํ•ฉํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค. ์˜ˆ์ œ์˜ ๋™๋ฌผ๋“ค์€ ๋Œ€๋ถ€๋ถ„ ์œ„์—์„œ ๋ดค์„ ๋•Œ ๊ธธ๊ณ  ํญ์ด ์–‡๊ฑฐ๋“ ์š”. ์•„๋ž˜๋Š” ์–ผ๋ฃฉ๋ง์„ ์œ„์—์„œ ๋ณธ ๊ฒƒ์ž…๋‹ˆ๋‹ค.</p>
  1280. <div class="threejs_center"><img src="../resources/images/zebra.png" style="width: 113px;"></div>
  1281. <p>์˜ˆ์ œ๋Š” ์› ๋ชจ์–‘์˜ ๊ฒฝ๊ณ„๋กœ ์š”์†Œ๋ผ๋ฆฌ์˜ ์ถฉ๋Œ์„ ๊ฐ์ง€ํ•˜๊ธฐ์— ์•„๋ž˜์™€ ๊ฐ™์ด ์šธํƒ€๋ฆฌ์— ๋‹ฟ๋Š” ๊ฒฝ์šฐ๋„ ์ถฉ๋Œ๋กœ ๊ฐ์ง€ํ•  ๊ฒ๋‹ˆ๋‹ค.</p>
  1282. <div class="threejs_center"><img src="../resources/images/zebra-collisions.svg" style="width: 400px;"></div>
  1283. <p>๋™๋ฌผ๊ณผ ๋™๋ฌผ์ด ๋ถ€๋”ชํžˆ๋Š” ๊ฒฝ์šฐ์—๋„ ๋งˆ์ฐฌ๊ฐ€์ง€์ž…๋‹ˆ๋‹ค. ํŠนํžˆ ๊ฒŒ์ž„์—์„œ๋Š” ์ด๋ž˜์„œ ์ข‹์„ ๊ฒŒ ์—†์ฃ .</p>
  1284. <p>2D ์‚ฌ๊ฐํ˜•์„ ๋งŒ๋“ค์–ด ์ถฉ๋Œ์„ ๊ฐ์ง€ํ•˜๋Š” ๊ฒƒ๋„ ์ƒ๊ฐํ–ˆ์œผ๋‚˜, ๋ฐ”๋กœ ๋„ˆ๋ฌด ๋งŽ์€ ์ฝ”๋“œ๋ฅผ ์จ์•ผ ํ•œ๋‹ค๋Š” ๊ฑธ ๊นจ๋‹ฌ์•˜์Šต๋‹ˆ๋‹ค. ์˜ˆ์ œ์˜ ๊ฐ ๋ชจ๋ธ์— ๋‹ค๋ฅธ ํฌ๊ธฐ์˜ ์‚ฌ๊ฐํ˜•์„ ์ถ”๊ฐ€ํ•˜๋Š” ๋ฐ๋Š” ๊ทธ๋‹ค์ง€ ๋งŽ์€ ์ฝ”๋“œ๊ฐ€ ๋“ค์–ด๊ฐ€์ง€ ์•Š์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์ด๋ ‡๊ฒŒ ๋ช‡ ๊ฐ€์ง€ ๋ชจ๋ธ์— ์‚ฌ๊ฐํ˜•์„ ์ถ”๊ฐ€ํ•ด๋ณด๋ฉด ๊ณง ์ถฉ๋Œ์„ ๊ฐ์ง€ํ•˜๋Š” ์ฝ”๋“œ๋ฅผ ์†๋ด์•ผ ํ•  ํ•„์š”๊ฐ€ ์ƒ๊ธธ ๊ฒ๋‹ˆ๋‹ค. ๋จผ์ € ๊ฐ ๋ชจ๋ธ์ด ์„œ๋กœ ์ถฉ๋Œํ•˜๋Š”์ง€ ํ™•์ธํ•ด์•ผ ํ•˜๋‹ˆ ๊ฐ ๋ชจ๋ธ์˜ ๊ฒฝ๊ณ„ ์ •์œก๋ฉด์ฒด๋‚˜ ๊ฒฝ๊ณ„ ๊ตฌ์ฒด, ๋˜๋Š” ๋ชจ๋ธ๊ณผ ๊ฐ™์€ ๋ฐฉํ–ฅ์œผ๋กœ ์ •๋ ฌ๋œ ๊ฒฝ๊ณ„ ์œก๋ฉด์ฒด๋ฅผ ๊ฒ€์‚ฌํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ๊ฐ ๋ชจ๋ธ์˜ ๊ฒฝ๊ณ„๊ฐ€ ์ถฉ๋Œํ–ˆ๋‹ค๋Š” ๊ฑด ๋‘ ๋ชจ๋ธ์ด <em>์–ด์ฉŒ๋ฉด</em> ์„œ๋กœ ์ถฉ๋Œํ–ˆ์„ ์ˆ˜๋„ ์žˆ๋‹ค๋Š” ์ด์•ผ๊ธฐ์ด๊ธฐ์—, ๊ฐ ๋ชจ๋ธ์ด <em>์‹ค์ œ๋กœ</em> ์ถฉ๋Œํ–ˆ๋Š”์ง€ ๊ฒ€์‚ฌํ•˜๊ธฐ ์œ„ํ•ด ํ•ด๋‹น ๋ชจ๋ธ๋“ค์„ ๋‹ค์‹œ ๊ฒ€์‚ฌํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ๋Œ€์ฒด๋กœ ๊ฒฝ๊ณ„ ๊ตฌ์ฒด๋ฅผ ๊ฒ€์‚ฌํ•˜๋Š” ๊ฒƒ๋งŒ ํ•ด๋„ ๊ฝค ๋งŽ์€ ์ž‘์—…์ด ํ•„์š”ํ•ฉ๋‹ˆ๋‹ค. ๊ฐ€๋Šฅํ•˜๋‹ค๋ฉด ๊ฐ ์š”์†Œ๊ฐ€ ๊ทผ์ ‘ํ–ˆ๋Š”์ง€์˜ ์—ฌ๋ถ€๋งŒ ๊ฒ€์‚ฌํ•˜๋Š” ๋“ฑ ๋” ํŠน์ˆ˜ํ•œ ๋ฐฉ๋ฒ•์„ ์‚ฌ์šฉํ•˜๋Š” ๊ฒŒ ๋” ๊ฒฝ์ œ์ ์ด์ฃ .</p>
  1285. <p>๋˜ํ•œ ์ถฉ๋Œ ์—ฌ๋ถ€๋ฅผ ๊ฒ€์‚ฌํ•˜๊ธฐ๋งŒ ํ•˜๋Š” ๊ฒƒ์œผ๋กœ ๋๋‚˜๋Š” ๊ฒƒ์ด ์•„๋‹ˆ๋ผ ์ถฉ๋Œ ์‹œ์Šคํ…œ๋„ ๊ตฌ์ถ•ํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ๊ทธ๋•Œ ๊ทธ๋•Œ ๊ฐ ๋ชจ๋ธ์—๊ฒŒ "๋„ˆ ๋‹ค๋ฅธ ์• ๋ž‘ ์ถฉ๋Œํ–ˆ๋‹ˆ?" ์ด๋ ‡๊ฒŒ ๋ฌผ์–ด๋ณด๋Š” ๊ฒƒ๋ณด๋‹ค ์‹œ์Šคํ…œ์ด ์ง์ ‘ ์ถฉ๋Œ ์—ฌ๋ถ€๋ฅผ ์ด๋ฒคํŠธ ๋“ฑ์œผ๋กœ ์•Œ๋ ค์ฃผ๋Š” ๊ฒŒ ๋” ํŽธํ•  ํ…Œ๋‹ˆ๊นŒ์š”. ์ถฉ๋Œ ์‹œ์Šคํ…œ์€ ์ถฉ๋Œ๊ณผ ๊ด€๋ จํ•œ ์ด๋ฒคํŠธ๋‚˜ ์ฝœ๋ฐฑ์„ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค. ์ด ๋ฐฉ๋ฒ•์˜ ์žฅ์ ์€ ๋ชจ๋“  ์ถฉ๋Œ์„ ํ•œ ๋ฒˆ๋งŒ ๊ฒ€์‚ฌํ•˜๊ธฐ์— ๊ฐ ๋ชจ๋ธ์ด "๋‚ด๊ฐ€ ๋‹ค๋ฅธ ์• ๋ž‘ ์ถฉ๋Œํ–ˆ๋‚˜?" ์ด๋ ‡๊ฒŒ ๊ฒ€์‚ฌ๋ฅผ ๋”ฐ๋กœ ํ•  ํ•„์š”๊ฐ€ ์—†๋‹ค๋Š” ๊ฑฐ์ฃ . ์—ฐ์‚ฐ๋Ÿ‰์„ ํ›จ์”ฌ ์ค„์ผ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  1286. <p>์‚ฌ๊ฐํ˜•์„ ํ™•์ธํ•˜๋Š” ์ •๋„์˜ ๊ฐ„๋‹จํ•œ ์ถฉ๋Œ ์‹œ์Šคํ…œ์„ ๋งŒ๋“œ๋Š” ์ฝ”๋“œ๋Š” 100-300 ์ค„ ์ •๋„๋ฅผ ๋„˜์ง€ ์•Š์„ ๊ฒ๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์˜ˆ์ œ์™€ ๋น„๊ตํ•˜๋ฉด ์—ฌ์ „ํžˆ ๋งŽ์€ ์ฝ”๋“œ์ด๋‹ˆ ์ง€๊ธˆ์€ ์ด๋Œ€๋กœ ๋‚จ๊ฒจ ๋‘๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  1287. <p>์‹œ๋„ํ•ด๋ด„์งํ•œ ๋‹ค๋ฅธ ๋ฐฉ๋ฒ•์€ ๋‹ค๋ฅธ ์บ๋ฆญํ„ฐ ์ค‘ ์œ„์—์„œ ๋ฐ”๋ผ๋ดค์„ ๋•Œ ๊ฐ€์žฅ ์›ํ˜•์— ๊ฐ€๊นŒ์šด ์บ๋ฆญํ„ฐ๋ฅผ ์ฐพ๋Š” ๊ฒ๋‹ˆ๋‹ค๏ผŠ. ์ธ๊ฐ„ํ˜• ์บ๋ฆญํ„ฐ์˜ ๊ฒฝ์šฐ๋Š” ๋Œ€๋ถ€๋ถ„ ์ž˜ ์ž‘๋™ํ•  ํ…Œ๊ณ , ๋™๋ฌผ๊ณผ ๋™๋ฌผ์˜ ๊ฒฝ์šฐ๋„ ์ผ๋ถ€ ๊ฒฝ์šฐ๋Š” ์ž˜ ์ž‘๋™ํ•  ๊ฒ๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ๋™๋ฌผ๊ณผ ์šธํƒ€๋ฆฌ์˜ ๊ฒฝ์šฐ๋Š” ๊ฐ์ง€ํ•˜์ง€ ๋ชปํ•˜๊ฒ ์ฃ . ์›๋ž˜ ํ™”๋ฉด ์ฃผ์œ„์— ์šธํƒ€๋ฆฌ๋‚˜ ๋ค๋ถˆ, ๋‘ฅ๊ทผ ๋ง‰๋Œ€๋ฅผ ๋‘˜๋Ÿฌ๋ณด๋ ค๊ณ  ํ–ˆ์œผ๋‚˜ ์ด๋Ÿฌ๋ ค๋ฉด 120์—์„œ 200๊ฐœ ์ •๋„์˜ ์š”์†Œ๋ฅผ ๋” ๋งŒ๋“ค์–ด์•ผ ํ•˜๊ณ , ์œ„์—์„œ ์–ธ๊ธ‰ํ•œ ์ตœ์ ํ™” ๋ฌธ์ œ์— ๋ถ€๋”ช์ณค์„ ๊ฒ๋‹ˆ๋‹ค.</p>
  1288. <p>โ€ป ์‹œ์•ผ๊ฐ ๋•Œ๋ฌธ์— ์นด๋ฉ”๋ผ์˜ ์ค‘์‹ฌ์—์„œ ๋ฒ—์–ด๋‚ ์ˆ˜๋ก ๋จธ๋ฆฌ ์œ„๊ฐ€ ์•„๋‹Œ ์˜†์ด ๋ณด์ด๋Š” ๊ฑธ ์ด์šฉํ•œ ๋ฐฉ๋ฒ•. ์—ญ์ฃผ.</p>
  1289. <p>์ด๋Ÿฐ ์—ฌ๋Ÿฌ ๋ฌธ์ œ ๋•Œ๋ฌธ์— ๋Œ€๋ถ€๋ถ„์˜ ๊ฒŒ์ž„๋“ค์ด ๊ธฐ์กด์— ์“ฐ๋˜ ๋ฐฉ๋ฒ•์„ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค. ๊ทธ๋ฆฌ๊ณ  ์ด ๋ฐฉ๋ฒ•๋“ค ์ค‘์—๋Š” ๋ฌผ๋ฆฌ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ์—์„œ ์“ฐ๋Š” ๊ฒƒ๋“ค๋„ ์žˆ์ฃ . ๋ฌผ๋ฆฌ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋Š” ์š”์†Œ๊ฐ€ ์„œ๋กœ ์ถฉ๋Œํ•˜๋Š”์ง€ ํ™•์ธํ•˜๋Š” ๊ธฐ๋Šฅ์ด ํ•„์ˆ˜๊ธฐ์—, ์ œ๊ฐ€ ์œ„์—์„œ ์‚ฌ์šฉํ–ˆ๋˜ ๋ฐฉ๋ฒ•์„ ์‚ฌ์šฉํ•˜๊ธฐ๋„ ํ•ฉ๋‹ˆ๋‹ค.</p>
  1290. <p>Three.js์˜ ์˜ˆ์ œ ์ค‘ <a href="https://github.com/kripken/ammo.js/">ammo.js</a>๋ฅผ ์‚ฌ์šฉํ•œ ๊ฒƒ์„ ๋ณด๋ฉด ์ด๋Ÿฐ ํ•ด๊ฒฐ ๋ฐฉ๋ฒ•์„ ์ฐพ๋Š” ๋ฐ ๋„์›€์ด ๋ ์ง€๋„ ๋ชจ๋ฅด๊ฒ ๋„ค์š”.</p>
  1291. <p>๋˜ ๋‹ค๋ฅธ ๋ฐฉ๋ฒ•์€ ์žฅ์• ๋ฌผ์„ ์ผ์ •ํ•œ ๊ฒฉ์ž(grid)์— ๋†“๊ณ  ํ”Œ๋ ˆ์ด์–ด์™€ ๋™๋ฌผ์ด ํ•ด๋‹น ๊ฒฉ์ž๋งŒ ์ฐธ์กฐํ•˜๊ฒŒ ํ•˜๋Š” ๊ฒ๋‹ˆ๋‹ค. ์„ฑ๋Šฅ ๋ฉด์—์„œ ๊ต‰์žฅํžˆ ์ข‹์€ ๋ฐฉ๋ฒ•์ธ๋ฐ, ์ด ๋˜ํ•œ ์—ฌ๋Ÿฌ๋ถ„์ด ์ง์ ‘ ์—ฐ์Šตํ•  ์ˆ˜ ์žˆ๋Š” ๐Ÿ˜œ ์š”์†Œ๋กœ ๋‚จ๊ฒจ ๋‘๋ฉด ์ข‹๊ฒ ๋‹ค๋Š” ์ƒ๊ฐ์ด ๋“ค๋”๊ตฐ์š”.</p>
  1292. <p>๋ง๋ถ™์—ฌ ๋Œ€๋ถ€๋ถ„์˜ ๊ฒŒ์ž„ ์‹œ์Šคํ…œ์—๋Š” <a href="https://www.google.com/search?q=coroutines">์ฝ”๋ฃจํ‹ด(coroutine)</a>์ด๋ผ๋Š” ๊ฒƒ์ด ์žˆ์Šต๋‹ˆ๋‹ค. ์ฝ”๋ฃจํ‹ด์€ ํŠน์ • ์ž‘์—…์„ ํ•˜๋Š” ๋™์•ˆ ๋ฉˆ์ท„๋‹ค๊ฐ€ ๋‚˜์ค‘์— ๋‹ค์‹œ ์‹œ์ž‘ํ•˜๋Š” ๋ฃจํ‹ด(routine)์„ ๋งํ•˜์ฃ .</p>
  1293. <p>ํ”Œ๋ ˆ์ด์–ด ์œ„์— ์Œํ‘œ๋ฅผ ๋„์›Œ ๋…ธ๋ž˜๋กœ ๋™๋ฌผ๋“ค์„ ๊ผฌ์‹œ๋Š” ๊ฒƒ์ฒ˜๋Ÿผ ํ•ด๋ณด๊ฒ ์Šต๋‹ˆ๋‹ค. ๊ตฌํ˜„ํ•  ์ˆ˜ ์žˆ๋Š” ๋ฐฉ๋ฒ•์€ ์•„์ฃผ ๋งŽ์ง€๋งŒ, ์˜ˆ์ œ์—์„œ๋Š” ์ฝ”๋ฃจํ‹ด์„ ์‚ฌ์šฉํ•ด ์ด๋ฅผ ๊ตฌํ˜„ํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  1294. <p>๋จผ์ € ์ฝ”๋ฃจํ‹ด์„ ๊ด€๋ฆฌํ•˜๋Š” ํด๋ž˜์Šค๋ฅผ ๋งŒ๋“ญ๋‹ˆ๋‹ค.</p>
  1295. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function* waitSeconds(duration) {
  1296. while (duration &gt; 0) {
  1297. duration -= globals.deltaTime;
  1298. yield;
  1299. }
  1300. }
  1301. class CoroutineRunner {
  1302. constructor() {
  1303. this.generatorStacks = [];
  1304. this.addQueue = [];
  1305. this.removeQueue = new Set();
  1306. }
  1307. isBusy() {
  1308. return this.addQueue.length + this.generatorStacks.length &gt; 0;
  1309. }
  1310. add(generator, delay = 0) {
  1311. const genStack = [generator];
  1312. if (delay) {
  1313. genStack.push(waitSeconds(delay));
  1314. }
  1315. this.addQueue.push(genStack);
  1316. }
  1317. remove(generator) {
  1318. this.removeQueue.add(generator);
  1319. }
  1320. update() {
  1321. this._addQueued();
  1322. this._removeQueued();
  1323. for (const genStack of this.generatorStacks) {
  1324. const main = genStack[0];
  1325. // ๋‹ค๋ฅธ ์ฝ”๋ฃจํ‹ด์ด ํ•ด๋‹น ์š”์†Œ๋ฅผ ์ œ๊ฑฐํ–ˆ์„ ๊ฒฝ์šฐ
  1326. if (this.removeQueue.has(main)) {
  1327. continue;
  1328. }
  1329. while (genStack.length) {
  1330. const topGen = genStack[genStack.length - 1];
  1331. const { value, done } = topGen.next();
  1332. if (done) {
  1333. if (genStack.length === 1) {
  1334. this.removeQueue.add(topGen);
  1335. break;
  1336. }
  1337. genStack.pop();
  1338. } else if (value) {
  1339. genStack.push(value);
  1340. } else {
  1341. break;
  1342. }
  1343. }
  1344. }
  1345. this._removeQueued();
  1346. }
  1347. _addQueued() {
  1348. if (this.addQueue.length) {
  1349. this.generatorStacks.splice(this.generatorStacks.length, 0, ...this.addQueue);
  1350. this.addQueue = [];
  1351. }
  1352. }
  1353. _removeQueued() {
  1354. if (this.removeQueue.size) {
  1355. this.generatorStacks = this.generatorStacks.filter(genStack =&gt; !this.removeQueue.has(genStack[0]));
  1356. this.removeQueue.clear();
  1357. }
  1358. }
  1359. }
  1360. </pre>
  1361. <p>์œ„ ํด๋ž˜์Šค๋Š” ๋‹ค๋ฅธ ์ฝ”๋ฃจํ‹ด์ด ์‹คํ–‰๋˜๋Š” ๋™์•ˆ ์š”์†Œ๋ฅผ ์•ˆ์ „ํ•˜๊ฒŒ ์ œ๊ฑฐ/์ถ”๊ฐ€ํ•˜๋„๋ก <code class="notranslate" translate="no">SafeArray</code>์™€ ๋น„์Šทํ•œ ๊ตฌ์กฐ๋กœ ๋งŒ๋“ค์—ˆ์Šต๋‹ˆ๋‹ค. ๋˜ํ•œ ์ด ํด๋ž˜์Šค๋Š” ์ค‘์ฒฉ๋œ ์ฝ”๋ฃจํ‹ด๋„ ์ฒ˜๋ฆฌํ•ฉ๋‹ˆ๋‹ค.</p>
  1362. <p>์ฝ”๋ฃจํ‹ด์„ ๋งŒ๋“ค๋ ค๋ฉด ์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ์˜ <a href="https://developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Statements/function*">์ œ๋„ˆ๋ ˆ์ดํ„ฐ ํ•จ์ˆ˜</a>๋ฅผ ๋งŒ๋“ค์–ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ์ œ๋„ˆ๋ ˆ์ดํ„ฐ ํ•จ์ˆ˜๋Š” <code class="notranslate" translate="no">function*</code>์ด๋ผ๋Š” ํ‚ค์›Œ๋“œ๋กœ ์ƒ์„ฑํ•˜์ฃ (๋ณ„ํ‘œ๋ฅผ ๋ถ™์—ฌ์•ผ ํ•ฉ๋‹ˆ๋‹ค!).</p>
  1363. <p>์ œ๋„ˆ๋ ˆ์ดํ„ฐ ํ•จ์ˆ˜๋Š” <code class="notranslate" translate="no">yield</code> ํ‚ค์›Œ๋“œ๋กœ ์‹คํ–‰ ์ˆœ์„œ๋ฅผ <strong>์–‘๋ณด(yield)</strong>ํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  1364. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function* countOTo9() {
  1365. for (let i = 0; i &lt; 10; ++i) {
  1366. console.log(i);
  1367. yield;
  1368. }
  1369. }
  1370. </pre>
  1371. <p>์ด ํ•จ์ˆ˜๋ฅผ ์•„๊นŒ ๋งŒ๋“  <code class="notranslate" translate="no">CoroutineRunner</code>์— ์ถ”๊ฐ€ํ•˜๋ฉด ํ•œ ํ”„๋ ˆ์ž„, ๋˜๋Š” <code class="notranslate" translate="no">runner.update</code>๋ฅผ ํ˜ธ์ถœํ•  ๋•Œ๋งˆ๋‹ค 0๋ถ€ํ„ฐ 9๊นŒ์ง€์˜ ์ˆซ์ž๋ฅผ ์ฐจ๋ก€๋Œ€๋กœ ์ถœ๋ ฅํ•  ๊ฒ๋‹ˆ๋‹ค.</p>
  1372. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const runner = new CoroutineRunner();
  1373. runner.add(count0To9);
  1374. while(runner.isBusy()) {
  1375. runner.update();
  1376. }
  1377. </pre>
  1378. <p>์ฝ”๋ฃจํ‹ด์€ ๋™์ž‘์ด ๋๋‚ฌ์„ ๋•Œ ์ž๋™์œผ๋กœ ์ œ๊ฑฐ๋ฉ๋‹ˆ๋‹ค. ์ฝ”๋ฃจํ‹ด์ด ๋๋‚˜๊ธฐ ์ „์— ์ œ๊ฑฐํ•˜๋ ค๋ฉด ์ œ๋„ˆ๋ ˆ์ดํ„ฐ๋ฅผ ๋ฏธ๋ฆฌ ์ฐธ์กฐํ•œ ๋’ค <code class="notranslate" translate="no">remove</code> ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.</p>
  1379. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">const gen = count0To9();
  1380. runner.add(gen);
  1381. // ์–ผ๋งˆ ํ›„
  1382. runner.remove(gen);
  1383. </pre>
  1384. <p>์ด์ œ ํ”Œ๋ ˆ์ด์–ด๊ฐ€ 0.5์—์„œ 1์ดˆ ์‚ฌ์ด๋งˆ๋‹ค ํ•œ ๋ฒˆ์”ฉ ์Œํ‘œ๋ฅผ ๋ฑ‰๋„๋ก ํ•ด๋ด…์‹œ๋‹ค.</p>
  1385. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Player extends Component {
  1386. constructor(gameObject) {
  1387. ...
  1388. + this.runner = new CoroutineRunner();
  1389. +
  1390. + function* emitNotes() {
  1391. + for (;;) {
  1392. + yield waitSeconds(rand(0.5, 1));
  1393. + const noteGO = gameObjectManager.createGameObject(scene, 'note');
  1394. + noteGO.transform.position.copy(gameObject.transform.position);
  1395. + noteGO.transform.position.y += 5;
  1396. + noteGO.addComponent(Note);
  1397. + }
  1398. + }
  1399. +
  1400. + this.runner.add(emitNotes());
  1401. }
  1402. update() {
  1403. + this.runner.update();
  1404. ...
  1405. }
  1406. }
  1407. function rand(min, max) {
  1408. if (max === undefined) {
  1409. max = min;
  1410. min = 0;
  1411. }
  1412. return Math.random() * (max - min) + min;
  1413. }
  1414. </pre>
  1415. <p>์œ„ ์ฝ”๋“œ์—์„œ๋Š” <code class="notranslate" translate="no">CoroutineRunner</code>๋ฅผ ๋งŒ๋“ค๊ณ  <code class="notranslate" translate="no">emitNotes</code> ์ฝ”๋ฃจํ‹ด์„ ์ถ”๊ฐ€ํ–ˆ์Šต๋‹ˆ๋‹ค. ์ด ํ•จ์ˆ˜๋Š” 0.5์ดˆ์—์„œ 1์ดˆ ์‚ฌ์ด๋งˆ๋‹ค ๊ณ„์†ํ•ด์„œ <code class="notranslate" translate="no">Note</code> ์ปดํฌ๋„ŒํŠธ๋ฅผ ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค.</p>
  1416. <p><code class="notranslate" translate="no">Note</code> ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋งŒ๋“ค๋ ค๋ฉด ๋จผ์ € ํ…์Šค์ฒ˜๊ฐ€ ํ•„์š”ํ•ฉ๋‹ˆ๋‹ค. ์Œํ‘œ ์ด๋ฏธ์ง€๋ฅผ ๋ถˆ๋Ÿฌ์˜ฌ ์ˆ˜๋„ ์žˆ์ง€๋งŒ, <a href="canvas-textures.html">์บ”๋ฒ„์Šค๋กœ ํ…์Šค์ฒ˜ ๋งŒ๋“ค๊ธฐ</a>์—์„œ ๋‹ค๋ค˜๋˜ ๊ฒƒ์ฒ˜๋Ÿผ ์บ”๋ฒ„์Šค๋ฅผ ์ด์šฉํ•ด ์ง์ ‘ ์Œํ‘œ๋ฅผ ๋งŒ๋“ค๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  1417. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function makeTextTexture(str) {
  1418. const ctx = document.createElement('canvas').getContext('2d');
  1419. ctx.canvas.width = 64;
  1420. ctx.canvas.height = 64;
  1421. ctx.font = '60px sans-serif';
  1422. ctx.textAlign = 'center';
  1423. ctx.textBaseline = 'middle';
  1424. ctx.fillStyle = '#FFF';
  1425. ctx.fillText(str, ctx.canvas.width / 2, ctx.canvas.height / 2);
  1426. return new THREE.CanvasTexture(ctx.canvas);
  1427. }
  1428. const noteTexture = makeTextTexture('โ™ช');
  1429. </pre>
  1430. <p>์œ„์—์„œ ๋งŒ๋“  ํ…์Šค์ฒ˜๋Š” ํ•˜์–€์ƒ‰์œผ๋กœ, ๋‚˜์ค‘์— ํ…์Šค์ฒ˜๋ฅผ ์‚ฌ์šฉํ•  ๋•Œ ์ƒ‰์„ ๋”ฐ๋กœ ์ง€์ •ํ•ด ์›ํ•˜๋Š” ์ƒ‰์˜ ์Œํ‘œ๋ฅผ ๊ทธ๋ฆด ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  1431. <p>์ด์ œ ์Œํ‘œ ํ…์Šค์ฒ˜๋ฅผ ๋งŒ๋“ค์—ˆ์œผ๋‹ˆ <code class="notranslate" translate="no">Note</code> ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋งŒ๋“ค ์ฐจ๋ก€์ž…๋‹ˆ๋‹ค. ์Œํ‘œ ์ปดํฌ๋„ŒํŠธ๋Š” <a href="billboards.html">๋นŒ๋ณด๋“œ์— ๊ด€ํ•œ ๊ธ€</a>์—์„œ ๋‹ค๋ค˜๋˜ <a href="/docs/#api/ko/materials/SpriteMaterial"><code class="notranslate" translate="no">SpriteMaterial</code></a>๊ณผ <a href="/docs/#api/ko/objects/Sprite"><code class="notranslate" translate="no">Sprite</code></a>๋ฅผ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค.</p>
  1432. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Note extends Component {
  1433. constructor(gameObject) {
  1434. super(gameObject);
  1435. const { transform } = gameObject;
  1436. const noteMaterial = new THREE.SpriteMaterial({
  1437. color: new THREE.Color().setHSL(rand(1), 1, 0.5),
  1438. map: noteTexture,
  1439. side: THREE.DoubleSide,
  1440. transparent: true,
  1441. });
  1442. const note = new THREE.Sprite(noteMaterial);
  1443. note.scale.setScalar(3);
  1444. transform.add(note);
  1445. this.runner = new CoroutineRunner();
  1446. const direction = new THREE.Vector3(rand(-0.2, 0.2), 1, rand(-0.2, 0.2));
  1447. function* moveAndRemove() {
  1448. for (let i = 0; i &lt; 60; ++i) {
  1449. transform.translateOnAxis(direction, globals.deltaTime * 10);
  1450. noteMaterial.opacity = 1 - (i / 60);
  1451. yield;
  1452. }
  1453. transform.parent.remove(transform);
  1454. gameObjectManager.removeGameObject(gameObject);
  1455. }
  1456. this.runner.add(moveAndRemove());
  1457. }
  1458. update() {
  1459. this.runner.update();
  1460. }
  1461. }
  1462. </pre>
  1463. <p>์ด ์ปดํฌ๋„ŒํŠธ๋Š” <a href="/docs/#api/ko/objects/Sprite"><code class="notranslate" translate="no">Sprite</code></a>๋ฅผ ๋งŒ๋“ค๊ณ  ๋ฌด์ž‘์œ„๋กœ ์†๋„๋ฅผ ์ •ํ•ด 60ํ”„๋ ˆ์ž„ ๋™์•ˆ ๊ทธ ์†๋„๋กœ ์ด๋™ํ•˜๊ฒŒ ํ•ฉ๋‹ˆ๋‹ค. ๋™์‹œ์— ์žฌ์งˆ์˜ <a href="/docs/#api/ko/materials/Material#opacity"><code class="notranslate" translate="no">opacity</code></a> ์†์„ฑ์„ ๋ฐ”๊ฟ” ํŽ˜์ด๋“œ-์•„์›ƒ ํšจ๊ณผ๋„ ์ฃผ์ฃ . ๋ฐ˜๋ณต๋ฌธ์ด ๋๋‚˜๋ฉด ์ด ์ปดํฌ๋„ŒํŠธ๋Š” ์œ„์น˜๊ฐ’๊ณผ ์Œํ‘œ๋ฅผ ํ•ด๋‹น GameObject์—์„œ ์ œ๊ฑฐํ•ฉ๋‹ˆ๋‹ค.</p>
  1464. <p>์ •๋ง ๋งˆ์ง€๋ง‰์œผ๋กœ, ๋™๋ฌผ์˜ ์ˆ˜๋ฅผ ์ข€ ๋Š˜๋ ค๋ณด๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  1465. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">function init() {
  1466. ...
  1467. const animalModelNames = [
  1468. 'pig',
  1469. 'cow',
  1470. 'llama',
  1471. 'pug',
  1472. 'sheep',
  1473. 'zebra',
  1474. 'horse',
  1475. ];
  1476. + const base = new THREE.Object3D();
  1477. + const offset = new THREE.Object3D();
  1478. + base.add(offset);
  1479. +
  1480. + // ์†Œ์šฉ๋Œ์ด ํ˜•ํƒœ๋กœ ๋™๋ฌผ๋“ค์„ ๋ฐฐ์น˜ํ•ฉ๋‹ˆ๋‹ค.
  1481. + const numAnimals = 28;
  1482. + const arc = 10;
  1483. + const b = 10 / (2 * Math.PI);
  1484. + let r = 10;
  1485. + let phi = r / b;
  1486. + for (let i = 0; i &lt; numAnimals; ++i) {
  1487. + const name = animalModelNames[rand(animalModelNames.length) | 0];
  1488. const gameObject = gameObjectManager.createGameObject(scene, name);
  1489. gameObject.addComponent(Animal, models[name]);
  1490. + base.rotation.y = phi;
  1491. + offset.position.x = r;
  1492. + offset.updateWorldMatrix(true, false);
  1493. + offset.getWorldPosition(gameObject.transform.position);
  1494. + phi += arc / r;
  1495. + r = b * phi;
  1496. }
  1497. </pre>
  1498. <p></p><div translate="no" class="threejs_example_container notranslate">
  1499. <div><iframe class="threejs_example notranslate" translate="no" style=" " src="/manual/examples/resources/editor.html?url=/manual/examples/game-conga-line-w-notes.html"></iframe></div>
  1500. <a class="threejs_center" href="/manual/examples/game-conga-line-w-notes.html" target="_blank">์ƒˆ ํƒญ์—์„œ ๋ณด๊ธฐ</a>
  1501. </div>
  1502. <p></p>
  1503. <p>๋ˆ„๊ตฐ๊ฐ€ <code class="notranslate" translate="no">setTimeout</code>์„ ์“ฐ๋ฉด ์•ˆ ๋˜๋ƒ๊ณ  ๋ฌผ์„์ง€๋„ ๋ชจ๋ฅด๊ฒ ์Šต๋‹ˆ๋‹ค. <code class="notranslate" translate="no">setTimeout</code>์„ ์“ฐ์ง€ ์•Š์€ ๊ฑด <code class="notranslate" translate="no">setTimeout</code>์€ ๊ฒŒ์ž„์˜ ํ”„๋ ˆ์ž„ ์ฃผ๊ธฐ์™€ ๋ฌด๊ด€ํ•˜๊ธฐ ๋•Œ๋ฌธ์ž…๋‹ˆ๋‹ค. ์˜ˆ๋ฅผ ๋“ค์–ด ์˜ˆ์ œ์—์„œ๋Š” ํ”„๋ ˆ์ž„ ๊ฐ„ ์‹œ๊ฐ„๊ฐ’์„ ์ตœ๋Œ€ 1/20์ดˆ๋กœ ์ œํ•œํ–ˆ์ฃ . ๋ฐฉ๊ธˆ ๊ตฌ์ถ•ํ•œ ์ฝ”๋ฃจํ‹ด ์‹œ์Šคํ…œ๋„ ์ด ์ œํ•œ์„ ๋”ฐ๋ฅผ ํ…Œ์ง€๋งŒ, <code class="notranslate" translate="no">setTimeout</code>์„ ์“ฐ๋ฉด ๊ทธ๋ ‡์ง€ ์•Š์„ ๊ฒ๋‹ˆ๋‹ค.</p>
  1504. <p>๋ฌผ๋ก  ์ข€ ๋” ๊ฐ„๋‹จํ•œ ํƒ€์ด๋จธ๋ฅผ ๋งŒ๋“ค ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  1505. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">class Player ... {
  1506. update() {
  1507. this.noteTimer -= globals.deltaTime;
  1508. if (this.noteTimer &lt;= 0) {
  1509. // ํƒ€์ด๋จธ๋ฅผ ์ดˆ๊ธฐํ™”ํ•ฉ๋‹ˆ๋‹ค.
  1510. this.noteTimer = rand(0.5, 1);
  1511. // GameObject๋กœ ์Œํ‘œ ์ปดํฌ๋„ŒํŠธ๋ฅผ ๋งŒ๋“ญ๋‹ˆ๋‹ค.
  1512. }
  1513. }
  1514. </pre>
  1515. <p>ํŠน์ • ๊ฒฝ์šฐ์—์•ผ ์ด ๋ฐฉ๋ฒ•์ด ๋” ์ข‹์„ ์ˆ˜๋„ ์žˆ์ง€๋งŒ, ๋” ๋งŽ์€ ์š”์†Œ๋ฅผ ์ถ”๊ฐ€ํ•˜๋ฉด ๊ทธ๋งŒํผ ๋” ๋งŽ์€ ๋ณ€์ˆ˜์™€ ์ฝ”๋ฃจํ‹ด์„ ์ถ”๊ฐ€ํ•ด์•ผ ํ• ํ…Œ๊ณ , ๊ทธ๋Ÿด์ˆ˜๋ก <code class="notranslate" translate="no">setTimeout</code>์„ <em>์„ค์ •ํ•˜๊ณ  ๊นŒ๋จน์„</em> ํ™•๋ฅ ์ด ๋†’์•„์งˆ ๊ฒ๋‹ˆ๋‹ค.</p>
  1516. <p>๋™๋ฌผ๋“ค์˜ ์ƒํƒœ๋ฅผ ์„ค์ •ํ•  ๋•Œ๋„ ์•„๋ž˜์™€ ๊ฐ™์ด ์ฝ”๋ฃจํ‹ด์„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  1517. <pre class="prettyprint showlinemods notranslate lang-js" translate="no">// ์‹ค์ œ๋กœ ์‚ฌ์šฉํ•˜์ง€ ์•Š๋Š” ํ•จ์ˆ˜
  1518. function* animalCoroutine() {
  1519. setAnimation('Idle');
  1520. while(playerIsTooFar()) {
  1521. yield;
  1522. }
  1523. const target = endOfLine;
  1524. setAnimation('Jump');
  1525. while(targetIsTooFar()) {
  1526. aimAt(target);
  1527. yield;
  1528. }
  1529. setAnimation('Walk')
  1530. while(notAtOldestPositionOfTarget()) {
  1531. addHistory();
  1532. aimAt(target);
  1533. yield;
  1534. }
  1535. for(;;) {
  1536. addHistory();
  1537. const pos = history.unshift();
  1538. transform.position.copy(pos);
  1539. aimAt(history[0]);
  1540. yield;
  1541. }
  1542. }
  1543. </pre>
  1544. <p>์ด ๋ฐฉ๋ฒ•์„ ์จ๋„ ๋”ฑํžˆ ๋ฌธ์ œ๋Š” ์—†์—ˆ๊ฒ ์ง€๋งŒ, ์ƒํƒœ๊ฐ€ ์ผ์ •ํ•˜์ง€ ์•Š์•„ ๋‹ค์‹œ <code class="notranslate" translate="no">FiniteStateMachine</code>์„ ์ฐพ๊ฒŒ ๋  ๊ฒ๋‹ˆ๋‹ค.</p>
  1545. <p>๋˜ ์ €๋Š” ์ฝ”๋ฃจํ‹ด์„ ํ•ด๋‹น ์ปดํฌ๋„ŒํŠธ์™€ ๋…๋ฆฝ์ ์œผ๋กœ ์‹คํ–‰ํ•˜๋Š” ๊ฒŒ ์ข‹์€์ง€ ์ž˜ ๋ชจ๋ฅด๊ฒ ์Šต๋‹ˆ๋‹ค. ๋ฌผ๋ก  ๊ทธ๋ƒฅ ์ „์—ญ์— <code class="notranslate" translate="no">CoroutineRunner</code>๋ฅผ ๋งŒ๋“ค์–ด ๋ชจ๋“  ์ฝ”๋ฃจํ‹ด์„ ์—ฌ๊ธฐ์— ์ง‘์–ด ๋„ฃ์„ ์ˆ˜๋Š” ์žˆ์ฃ . ํ•˜์ง€๋งŒ ์ด๋Ÿฌ๋ฉด ์ฝ”๋ฃจํ‹ด์„ ์—†์• ๊ธฐ๊ฐ€ ํž˜๋“ค์–ด์งˆ ๊ฒ๋‹ˆ๋‹ค. ์ง€๊ธˆ ์˜ˆ์ œ๋Š” GameObject๋ฅผ ์ œ๊ฑฐํ•˜๋ฉด ํ•ด๋‹น ์ปดํฌ๋„ŒํŠธ๋„ ์ œ๊ฑฐ๋˜๊ณ , ๊ทธ๋Ÿฌ๋ฉด ์ƒ์„ฑํ•œ <code class="notranslate" translate="no">CoroutineRunner</code>์˜ ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•  ์ผ๋„ ์—†์œผ๋‹ˆ ์ฝ”๋ฃจํ‹ด๋„ ์ „๋ถ€ ๊ฐ€๋น„์ง€ ์ปฌ๋ ‰์…˜์— ๋“ค์–ด๊ฐˆ ๊ฒ๋‹ˆ๋‹ค. ์ „์—ญ์— <code class="notranslate" translate="no">CoroutineRunner</code>๋ฅผ ๋‘๋ฉด ์ปดํฌ๋„ŒํŠธ์—์„œ ์ง์ ‘ ์ด ์ „์—ญ ๊ฐ์ฒด์˜ ์ฝ”๋ฃจํ‹ด์„ ์ œ๊ฑฐํ•˜๊ฑฐ๋‚˜ ์ž๋™์œผ๋กœ ์ฝ”๋ฃจํ‹ด์„ ์ œ๊ฑฐํ•  ๋‹ค๋ฅธ ๋ฐฉ๋ฒ•์ด ํ•„์š”ํ•  ๊ฒ๋‹ˆ๋‹ค.</p>
  1546. <p>์‹ค์ œ ๊ฒŒ์ž„ ์—”์ง„์ด๋ผ๋ฉด ๋” ๊ณ ๋ คํ•ด์•ผ ํ•  ๋ฌธ์ œ๊ฐ€ ๋งŽ์Šต๋‹ˆ๋‹ค. ์ง€๊ธˆ์€ GameObject๋‚˜ ์ปดํฌ๋„ŒํŠธ์— ๋”ฐ๋กœ ์ˆœ์„œ๋ฅผ ์ง€์ •ํ•  ์ˆ˜ ์—†์ฃ . ๊ทธ๋ƒฅ ์ถ”๊ฐ€ํ•œ ์ˆœ์„œ๊ฐ€ ํ•ด๋‹น ์š”์†Œ์˜ ์ˆœ์„œ๊ฐ€ ๋ฉ๋‹ˆ๋‹ค. ๋Œ€๋ถ€๋ถ„์˜ ๊ฒŒ์ž„ ์—”์ง„์€ ์šฐ์„  ์ˆœ์œ„๋ฅผ ์ •ํ•ด ์ˆœ์„œ๋ฅผ ๋ฐ”๊ฟ€ ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.</p>
  1547. <p>๋‹ค๋ฅธ ๋ฌธ์ œ๋Š” <code class="notranslate" translate="no">Note</code> ์ปดํฌ๋„ŒํŠธ๊ฐ€ ์žฅ๋ฉด ์œ„ GameObject์˜ transform ์†์„ฑ์„ ๋ณ€๊ฒฝํ•œ๋‹ค๋Š” ๊ฒ๋‹ˆ๋‹ค. ์• ์ดˆ์— <code class="notranslate" translate="no">GameObject</code>๊ฐ€ transform ์†์„ฑ์„ ๋ณ€๊ฒฝํ–ˆ์œผ๋‹ˆ ์ข€ ๋” ์ œ๋Œ€๋กœ ๊ตฌํ˜„ํ•˜๋ ค๋ฉด <code class="notranslate" translate="no">GameObject</code>๊ฐ€ ๊ณ„์† transform ์†์„ฑ์„ ๊ด€๋ฆฌํ•˜๋Š” ๊ฒŒ ๋งž๊ฒ ์ฃ . <code class="notranslate" translate="no">GameObject</code>์— <code class="notranslate" translate="no">dispose</code> ๊ฐ™์€ ๋ฉ”์„œ๋“œ๋ฅผ ๋‘๊ณ  <code class="notranslate" translate="no">GameObjectManager.removeGameObject</code>์—์„œ ์ด ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ–ˆ๋‹ค๋ฉด ์–ด๋–จ๊นŒ์š”?</p>
  1548. <p>๋˜ <code class="notranslate" translate="no">gameObjectManager.update</code>๋‚˜ <code class="notranslate" translate="no">inputManager.update</code>๋ฅผ ์ง์ ‘ ํ˜ธ์ถœํ•˜๋Š” ๋Œ€์‹  <code class="notranslate" translate="no">SystemManager</code>๋ฅผ ๋งŒ๋“ค์–ด <code class="notranslate" translate="no">update</code>๋ฉ”์„œ๋“œ๋ฅผ ๊ฐ€์ง„ ์š”์†Œ๋ฅผ ์ „๋ถ€ ์ถ”๊ฐ€ํ•ด ์ด ํด๋ž˜์Šค๊ฐ€ ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•˜๋„๋ก ํ•˜๋Š” ๊ฒŒ ๋” ๋‚˜์„ ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค. ์ด๋ ‡๊ฒŒ ํ•˜๋ฉด <code class="notranslate" translate="no">CollisionManager</code> ๋“ฑ ์ƒˆ๋กœ์šด ์‹œ์Šคํ…œ์„ ๋งŒ๋“ค์—ˆ์„ ๋•Œ <code class="notranslate" translate="no">render</code> ํ•จ์ˆ˜๋ฅผ ์ˆ˜์ •ํ•˜๋Š” ๊ฒŒ ์•„๋‹ˆ๋ผ <code class="notranslate" translate="no">SystemManager</code>์— ์ด ์‹œ์Šคํ…œ์„ ์ถ”๊ฐ€ํ•˜๊ธฐ๋งŒ ํ•˜๋ฉด ๋  ๊ฒ๋‹ˆ๋‹ค.</p>
  1549. <p>์ €๋Š” ์ด๋Ÿฐ ๋ฌธ์ œ๋“ค์„ ์ „๋ถ€ ๋‹ค๋ฃจ๊ธฐ๋ณด๋‹ค ์—ฌ๋Ÿฌ๋ถ„์˜ ๋ชซ์œผ๋กœ ๋‚จ๊ฒจ ๋‘๊ณ ์ž ํ•ฉ๋‹ˆ๋‹ค. ๋ถ€๋”” ์ด ๊ธ€์ด ์—ฌ๋Ÿฌ๋ถ„๋งŒ์˜ ๊ฒŒ์ž„ ์—”์ง„์„ ๋งŒ๋“œ๋Š” ๋ฐ ๋„์›€์ด ๋˜์—ˆ๋‹ค๋ฉด ์ข‹๊ฒ ๋„ค์š”.</p>
  1550. <p>์–ด์ฉŒ๋ฉด ์ œ๊ฐ€ ๊ฒŒ์ž„ ์žผ(game jam)๏ผŠ์„ ์—ด ์ˆ˜๋„ ์žˆ๊ฒ ๋„ค์š”. ์œ„ ์˜ˆ์ œ์˜ <em>jsfiddle</em>์ด๋‚˜ <em>codepen</em>์„ ํด๋ฆญํ•ด๋ณด๋ฉด ์ฝ”๋“œ๋ฅผ ๋ฐ”๋กœ ํŽธ์ง‘ํ•ด ๋ณผ ์ˆ˜ ์žˆ๋Š” ์‚ฌ์ดํŠธ๊ฐ€ ์—ด๋ฆด ๊ฒ๋‹ˆ๋‹ค. ํŠน์ • ๊ธฐ๋Šฅ์„ ์ถ”๊ฐ€ํ•˜๊ฑฐ๋‚˜ ํ•ด์„œ ์˜ˆ์ œ๋ฅผ ํผ๊ทธ๊ฐ€ ๊ธฐ์‚ฌ(knight)์„ ๋Œ๊ณ  ๋‹ค๋‹ˆ๋Š” ๊ฒŒ์ž„์„ ๋งŒ๋“ค๊ฑฐ๋‚˜, ๊ธฐ์‚ฌ์˜ ๊ตฌ๋ฅด๊ธฐ ์• ๋‹ˆ๋ฉ”์ด์…˜์„ ๋ณผ๋ง๊ณต์œผ๋กœ์จ ๋™๋ฌผ ๋ณผ๋ง ๊ฒŒ์ž„์„ ๋งŒ๋“ค ์ˆ˜๋„ ์žˆ๊ฒ ์ฃ . ๋˜๋Š” ๋™๋ฌผ ์ด์–ด ๋‹ฌ๋ฆฌ๊ธฐ ๊ฒŒ์ž„์ด๋ผ๋“ ๊ฐ€์š”. ๊ดœ์ฐฎ์€ ๊ฒŒ์ž„์„ ๋งŒ๋“ค์—ˆ๋‹ค๋ฉด ์•„๋ž˜์— ๋Œ“๊ธ€๋กœ ๋งํฌ๋ฅผ ๋‚จ๊ฒจ์ฃผ์‹œ๋ฉด ๊ฐ์‚ฌํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค.</p>
  1551. <p>โ€ป ๊ฒŒ์ž„ ์žผ: ๋ณดํ†ต 24์‹œ๊ฐ„์—์„œ 72์‹œ๊ฐ„ ์ •๋„์˜ ์งง์€ ๊ธฐ๊ฐ„ ๋‚ด์— ํŒ€, ๋˜๋Š” ๊ฐœ์ธ์ด ๊ฒŒ์ž„์„ ๋งŒ๋“œ๋Š” ๋Œ€ํšŒ. ์—ญ์ฃผ.</p>
  1552. <div class="footnotes">
  1553. [<a id="parented">1</a>]: ๋ฌผ๋ก  ๋ถ€๋ชจ์˜ ์–ด๋–ค ์š”์†Œ๋„ translation, rotation, scale ์†์„ฑ์„ ๋ฐ”๊พธ์ง€ ์•Š์•˜๋‹ค๋ฉด ์ •์ƒ์ ์œผ๋กœ ์ž‘๋™ํ•ฉ๋‹ˆ๋‹ค.<a href="#parented-backref">[๋Œ์•„๊ฐ€๊ธฐ]</a>
  1554. </div>
  1555. </div>
  1556. </div>
  1557. </div>
  1558. <script src="../resources/prettify.js"></script>
  1559. <script src="../resources/lesson.js"></script>
  1560. </body></html>
็ฒคICPๅค‡19079148ๅท