質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.34%
Three.js

Three.jsはWebGLをサポートしているJavaScriptの3D描画用ライブラリです。

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

WebGL

WebGL(ウェブジーエル)は、ウェブブラウザで 3次元コンピュータグラフィックスを表示させるための標準仕様です。

Blender

Blenderとは、オープンソースの3DCGソフトウェアです。フリーでありながら、3Dモデル作成、レンダリング、アニメーション、コンポジットなどのハイエンドに匹敵する高い機能を持ち、さらにゲームエンジンも搭載しています。

Q&A

1回答

329閲覧

【Three.js】アニメーション付きGLTFモデルのマテリアルを置換するとアニメーションが動かなくなる現象を解決したい

cocy_eng

総合スコア2

Three.js

Three.jsはWebGLをサポートしているJavaScriptの3D描画用ライブラリです。

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

WebGL

WebGL(ウェブジーエル)は、ウェブブラウザで 3次元コンピュータグラフィックスを表示させるための標準仕様です。

Blender

Blenderとは、オープンソースの3DCGソフトウェアです。フリーでありながら、3Dモデル作成、レンダリング、アニメーション、コンポジットなどのハイエンドに匹敵する高い機能を持ち、さらにゲームエンジンも搭載しています。

0グッド

1クリップ

投稿2024/09/05 02:30

編集2024/09/10 11:47

実現したいこと

  • Three.jsで、アニメーション付きGLTFモデルをマテリアル名で取得し、シェーダーマテリアルに置き換えたい。

前提

GLTFモデルはBlenderのシェイプキーアニメーションで作成しています。下のように球が変形するアニメーションをWeb上で再生するところまでは上手くいきました。
イメージ説明

発生している問題

後述のソースコードの39行目あたりで、メッシュをマテリアル名(bg_mat)で取得してシェーダーマテリアルに置き換えたところ、色は意図通りに変わったのですが、アニメーションが動かなくなってしまいました。
イメージ説明

アニメーションに関わる何かが上書きされてしまったのでしょうか?
アニメーションが消えてしまう原因や消えない方法をご教示いただけると幸いです。

該当のソースコード

JavaScript

1window.addEventListener('load', init); 2 3function init() { 4 // レンダラーを作成 5 const renderer = new THREE.WebGLRenderer({ 6 canvas: document.querySelector('#canvas'), 7 antialias: true, 8 alpha: true, 9 }); 10 // シーンを作成 11 const scene = new THREE.Scene(); 12 13 // カメラを作成 14 // new THREE.PerspectiveCamera(視野角, アスペクト比, near, far) 15 const camera = new THREE.PerspectiveCamera(45, 1, 1, 2000); 16 camera.fov = 50; 17 camera.position.set(0, 0.5, 8); 18 // 原点方向を見つめる 19 camera.lookAt(0, 0, 0); 20 //camera.rotation.x = THREE.Math.DEG2RAD * (-45); 21 22 // Load GLTF or GLB 23 const loader = new THREE.GLTFLoader(); 24 const url = '3d/sample.glb'; 25 26 let model = null; 27 // clockを作成する 28 const clock = new THREE.Clock(); 29 loader.load( 30 url, 31 function (gltf) { 32 model = gltf.scene; 33 model.traverse((node) => { 34 if (!node.isMesh) return; 35 }); 36 model.scale.set(1, 1, 1); 37 model.position.set(0, 0, 0); 38 39 //マテリアルを置換 40 const bgMesh = getMatbyName(model, "bg_mat"); 41 //バーテックスシェーダ 42 const vertexShader =`void main() {gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);}`; 43 //フラグメントシェーダ 44 const fragmentShader =`precision mediump float; 45 void main() { 46 // オレンジ色のRGB値を定義 (1.0, 0.5, 0.0) 47 vec3 orangeColor = vec3(1.0, 0.5, 0.0); 48 // 最終的に出力される色に代入 49 gl_FragColor = vec4(orangeColor, 1.0); 50 }`; 51 52 const bgMat = new THREE.ShaderMaterial({ 53 vertexShader:vertexShader, 54 fragmentShader:fragmentShader, 55 side:THREE.DoubleSide, 56 }); 57 58 bgMesh.material = bgMat; 59 60 61 // アニメーションを読み込む 62 let mixer = new THREE.AnimationMixer(model); 63 gltf.animations.forEach((clip) => { 64 mixer.clipAction(clip).play(); 65 }); 66 67 scene.add(model); 68 69 // アニメーションを更新する関数を定義する 70 function update() { 71 if (mixer) { 72 mixer.update(clock.getDelta()); 73 } 74 } 75 76 // アニメーションを再生するイベントを設定する 77 renderer.setAnimationLoop(() => { 78 update(); 79 renderer.render(scene, camera); 80 }); 81 82 }, 83 function (error) { 84 console.log('An error happened'); 85 } 86 ); 87 88 //テクスチャをガンマ補正 89 renderer.gammaOutput = true; 90 renderer.gammaFactor = 2.2; 91 92 // ライト追加 93 hemisphereLight = new THREE.HemisphereLight(0xffffff, 0x4169e1, 4); 94 scene.add(hemisphereLight); 95 96 // カメラコントローラーを作成 97 const controls = new THREE.OrbitControls(camera, renderer.domElement); 98 controls.enableDamping = true; 99 controls.dampingFactor = 0.2; 100 101 // フォグを設定 102 scene.fog = new THREE.Fog(0x2a2a5f, 150, 200); 103 104 // 初回実行 105 var frame_num = 0; 106 tick(); 107 onResize(); 108 109 // リサイズイベント発生で実行 110 window.addEventListener('resize', onResize); 111 112 function onResize() { 113 // サイズを取得 114 const w = window.innerWidth; 115 const h = window.innerHeight; 116 // 描画サイズ 117 renderer.setSize(w, h); 118 // ピクセル比 119 renderer.setPixelRatio(window.devicePixelRatio); 120 // カメラのアスペクト比を正す 121 camera.aspect = w / h; 122 camera.updateProjectionMatrix(); 123 console.log('h' + h + 'h' + h); 124 // レンダリング 125 renderer.render(scene, camera); 126 } 127 128 function tick() { 129 requestAnimationFrame(tick); 130 renderer.render(scene, camera); 131 controls.update(); 132 133 } 134} 135 136function update() { 137 controls.update(); 138 requestAnimationFrame(update); 139 140} 141 142// 指定したマテリアル名がセットされている[ mesh ] を返します 143function getMatbyName(_o, _name){ 144 let refO = null; 145 146 if(_o.material && _o.material.name.indexOf(_name) > -1 ){ 147 return _o; 148 } 149 150 if(_o.materials){ 151 for(let i =0; i < _o.materials.length; i++){ 152 refO = getMatbyName(_o.materials[i], _name ); 153 if(refO) {return refO;} 154 } 155 } 156 157 for(let i =0; i < _o.children.length; i++){ 158 refO = getMatbyName(_o.children[i], _name ); 159 if(refO) {return refO; } 160 } 161 162 return refO; 163}

補足情報(FW/ツールのバージョンなど)

Blender3.6
three.js r124

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

cocy_eng

2024/09/07 01:32

補足です。 標準のマテリアル、例えば MeshBasicMaterial などであればマテリアルに下記オプションを付けることでアニメーションを再生することができました。 morphTargets : true (アニメーションの種類によっては skinning : true が良い場合もあるようです) しかし、上記コードのようなシェーダーマテリアルでは機能しませんでした。 惜しいところまで行っている気がするのですが…。引き続き情報ある方はお待ちしております。
guest

回答1

0

[香車]東上☆Aho(英帆)☆海美「
teratail での回答を求めるなら [3DCG][Blender] 等のタグをつけた方がいい。

回答をもとめるなら、#3DCG や #Blender #b3d とか、つけて X (旧 Twitter) に投げた方がいいかも。

投稿2024/09/07 01:50

umimi

総合スコア526

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

cocy_eng

2024/09/07 01:55

ご指摘ありがとうございます。タグが消えてしまっていることに気づきました。 一度誤って解決済みにしてしまい、それを取り消したからだと思います。 タグを復活させる方法がないか、確認してみます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだベストアンサーが選ばれていません

会員登録して回答してみよう

アカウントをお持ちの方は

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.34%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問