2023年11月24日WebGL & Three.js
Three.jsで数値をGUIで変更することができるようになるlil-gui(dat-gui)のカスタマイズ方法と、
パフォーマンスを測ることができるstat-jsの使い方を紹介します。
lil-guiを使えると、見た目で直感的に調整できるので大変便利です。
また、Webサイトが遅いな、などと思ったらstat-jsでチェックできるのでこちらも便利です。
デモを用意したので、確認してみてください。
lil-guiとstat-jsのサンプルデモ
ソースコード
import * as THREE from "three";
import { OrbitControls } from "three/examples/jsm/controls/OrbitControls";
import GUI from "lil-gui";
import Stats from "stats-js";
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(
75,
window.innerWidth / window.innerHeight,
0.1,
1000
);
const renderer = new THREE.WebGLRenderer({
antialias: true,
});
renderer.setSize(window.innerWidth, window.innerHeight);
renderer.setClearColor(0x090909);
document.getElementById("modal-container").appendChild(renderer.domElement);
const meshes = [];
const MESH_NUM = 250;
const POS_RANGE = 150;
const MAX_SCALE = 1.1;
const TARGET_MESH_NUM = 20;
const COLORS = {
COLOR1: "#27005D",
COLOR2: "#9400FF",
COLOR3: "#AED2FF",
COLOR4: "#E4F1FF",
};
function mapRand(min, max, isInt = false) {
let rand = Math.random() * (max - min) + min;
rand = isInt ? Math.round(rand) : rand;
return rand;
}
function randomMesh() {
const geometries = [
new THREE.BoxGeometry(8, 8, 8),
new THREE.PlaneGeometry(8, 8),
new THREE.TorusGeometry(8, 3, 200, 20),
new THREE.SphereGeometry(8),
new THREE.TorusKnotGeometry(8, 2),
];
const colors = [COLORS.COLOR1, COLORS.COLOR2, COLORS.COLOR3, COLORS.COLOR4];
const colorIndex = mapRand(0, colors.length - 1, true);
const color = new THREE.Color(colors[colorIndex]);
const pos = {
x: mapRand(-POS_RANGE, POS_RANGE),
y: mapRand(-POS_RANGE, POS_RANGE),
z: mapRand(-POS_RANGE, POS_RANGE),
};
const scale = mapRand(1, MAX_SCALE);
const material = new THREE.MeshStandardMaterial({ color });
const gIndex = mapRand(0, geometries.length - 1, true);
const mesh = new THREE.Mesh(geometries[gIndex], material);
mesh.colorIndex = colorIndex;
mesh.position.set(pos.x, pos.y, pos.z);
mesh.geometry.scale(scale, scale, scale);
return mesh;
}
for (let i = 0; i < MESH_NUM; i++) {
const mesh = randomMesh();
meshes.push(mesh);
}
scene.add(...meshes);
const axis = new THREE.AxesHelper(500);
scene.add(axis);
camera.position.z = 50;
const control = new OrbitControls(camera, renderer.domElement);
const amLight = new THREE.AmbientLight(0xa9a9a9);
scene.add(amLight);
const pointLight1 = new THREE.PointLight(0xdfdfdf, 1, 500);
pointLight1.position.set(10, 110, 120);
const pHelper1 = new THREE.PointLightHelper(pointLight1, 5, 0x00ff00);
pHelper1.visible = false;
const pointLight2 = new THREE.PointLight(0xffff00, 1, 700);
pointLight2.position.set(-100, -110, -220);
const pHelper2 = new THREE.PointLightHelper(pointLight2, 5, 0x00ff00);
pHelper2.visible = false;
const pointLight3 = new THREE.PointLight(0x00ffff, 1, 2900);
pointLight3.position.set(550, 100, 220);
const spherical = new THREE.Spherical(28, 0.5, 1);
pointLight3.position.setFromSpherical(spherical);
const pHelper3 = new THREE.PointLightHelper(pointLight3, 5, 0x00ff00);
pHelper3.visible = false;
scene.add(
pointLight1,
pHelper1,
pointLight2,
pHelper2,
pointLight3,
pHelper3
);
function getAction({ x, y, z }) {
const rand = mapRand(0.5, 2);
const ACTIONS = [
function () {
const direction = x < 0 ? rand : -rand;
this.position.x += direction;
},
function () {
const direction = y < 0 ? rand : -rand;
this.position.y += direction;
},
function () {
const direction = z < 0 ? rand : -rand; this.position.z += direction; }, ]; const action = ACTIONS[mapRand(0, ACTIONS.length - 1, true)]; return action; }
targetMeshes.forEach((mesh) => (mesh.__action = null));
targetMeshes = [];
for (let i = 0; i < TARGET_MESH_NUM; i++) { const mesh = meshes[mapRand(0, meshes.length - 1, true)]; mesh.__action = getAction(mesh.position); targetMeshes.push(mesh); } }, 1500); const gui = new GUI({ container: document.querySelector("#modal-container") });
const folder2 = gui.addFolder("色");
folder2.open();
folder2.addColor(COLORS, "COLOR1").onChange((newValue) => {
meshes.forEach((mesh) => {
if (mesh.colorIndex === 0) {
mesh.material.color.set(newValue);
}
});
});
folder2.addColor(COLORS, "COLOR2").onChange((newValue) => {
meshes.forEach((mesh) => {
if (mesh.colorIndex === 1) {
mesh.material.color.set(newValue);
}
});
});
folder2.addColor(COLORS, "COLOR3").onChange((newValue) => {
meshes.forEach((mesh) => {
if (mesh.colorIndex === 2) {
mesh.material.color.set(newValue);
}
});
});
folder2.addColor(COLORS, "COLOR4").onChange((newValue) => {
meshes.forEach((mesh) => {
if (mesh.colorIndex === 3) {
mesh.material.color.set(newValue);
}
});
});
const stats = new Stats();
stats.showPanel(0);
document.body.appendChild( stats.dom );
function animate() {
requestAnimationFrame(animate);
stats.begin();
targetMeshes.forEach((mesh) => mesh.__action());
if (POS_RANGE > camera.position.z) {
camera.position.z += 0.01;
}
spherical.theta += 0.01;
pointLight3.position.setFromSpherical(spherical);
control.update();
renderer.render(scene, camera);
stats.end();
}
animate();
こちらのコードのほとんどは、【Three.js】物体(メッシュ)を複数ランダムに生成して散りばめる方法(デモあり)と同じで詳しく解説しているので確認してみてください。
今回は、lil-guiとstat-js部分に絞って解説します。
lil-gui(dat-gui)のカスタマイズ方法
dat-guiは非推奨になったのですが、ほぼ同じ使い方のため、参考になると思います。
今から使う場合は、dat-guiではなく、lil-guiを使いましょう。
lil-gui公式サイト: https://lil-gui.georgealways.com/
package.jsonにいれる
まずは使えるようにパッケージをインストールします。
インポートする
インスタンス化する
インスタンス化するときに、デフォルトではbody直下にDOMができます。
body以外にlil-guiを入れたい場合は、上のようにcontainerに値を設定するとそこにlil-guiをいれることができます。
ライトを直感的に調整できるようにする
まず、いろいろ追加するとわかりにくくなるので、フォルダーを作ってグループで管理して折りたためるようにします。
そのためにaddFolderを使います。
あとはそのaddForderに項目をadd()で追加していくだけです。
たとえば、x軸なら、pointLight1のx軸を-500〜500まで、1刻みで変更できるように設定しています。
ここの値を自由に変えることができます。
色を直感的に調整できるようにする
色も同様で、まずはフォルダーを作ってあげます。
上の場合だと、ランダムに4種類の色から色を選択し、メッシュを生成しているため、
colorIndexという項目を目印にします。
このcolorIndexの配列のインデックスをメッシュのインデックスと一致する場合に、とすることで、
複数メッシュに適用しているような色でも一括して変更することができます。
基本的に、addColorで配列を指定したら、onChange()で新しい色をセットする必要があります。
メッシュに色が固定されている場合
上のような感じで色がメッシュに固定されている場合は、もっと簡単です。
上のようにして変更することができます。
uniformsの値を変更する
上のようなかたちでマテリアルでuniformsのプロパティを設定して、フラグメントシェーダーに渡したいときがあります。
その値をGUIでチェックできるようにするには、以下のようにします。
materialがインスタンス化されたオブジェクトなので、あとは単純にuniforms…以下を指定してあげます。
Vector2は値をみると分かるのですがxとyのプロパティが生成され、それに指定した値(上の例なら10と20)が入ります。
ラベル名を変更したいとき
上のようにname関数を使うと、そこに指定した名称のラベルに変更できます。
数値のデータをチェックボックスで管理したいとき
画像の切り替えなどで0から1に変化するものをチェックボックスで表示管理したいときがあります。
その場合は以下のようにします。
まずは、チェックボックスにするためブール値(true,false)に変更します。
そして、変更を検知するためにonChange()を使います。
あとはその中にコールバック関数を書いてあげます。
よくgsapなどで変化させたりしますが、その場合はブール値にした値を数値に変換してあげる必要があるので、Number()などを使います。
他の場所で変更された値と連動させたい
上のチェックボックスとたとえば進捗ラベル部分を連動させたいとします。
その場合、進捗ラベルの方にlisten関数をつけてあげると、チェックボックスのオンオフと連動して変化させることができます。