Three.js - 如何将对象预置位置以便合并?

3

我成功地合并了我的立方体,但它们似乎会重新回到它们原来的位置/旋转状态,忽略了预设的位置/旋转。

var geometries = [];

cube1.position.set( 0, 0, 0 );
geometries.push( cube1 );

cube2.position.set( 1, 0, 0 );
geometries.push( cube2 );

cube3.position.set( 0, 1, 0 );
geometries.push( cube3 );

const cubes = BufferGeometryUtils.mergeBufferGeometries( geometries );

scene.add( cubes );//All the cubes get merged on 0, 0, 0 - overlapping one another

我该如何将立方体放置在其声明的位置,以便它们可以合并?


你可以将合并的几何体添加到网格中,并获取其几何体。请参见此答案 - Anurag Srivastava
@AnuragSrivastava那个问题还没有答案 :/ - Vardan Betikyan
1个回答

4
这是一个使用BufferGeometryUtils.applyMatrix4()的工作示例:

body{
  overflow: hidden;
  margin: 0;
}
<script type="module">
import * as THREE from "https://cdn.skypack.dev/three@0.136.0";
import {
  OrbitControls
} from "https://cdn.skypack.dev/three@0.136.0/examples/jsm/controls/OrbitControls.js";
import * as BufferGeometryUtils from "https://cdn.skypack.dev/three@0.136.0/examples/jsm/utils/BufferGeometryUtils";

let scene = new THREE.Scene();
let camera = new THREE.PerspectiveCamera(60, innerWidth / innerHeight, 1, 1000);
camera.position.setScalar(1).setLength(6);
let renderer = new THREE.WebGLRenderer({
  antialias: true
});
renderer.setSize(innerWidth, innerHeight);
renderer.setClearColor(0x404040);
document.body.appendChild(renderer.domElement);

let controls = new OrbitControls(camera, renderer.domElement);

let light = new THREE.DirectionalLight(0xffffff, 1);
light.position.setScalar(1);
scene.add(light, new THREE.AmbientLight(0xffffff, 0.5));

let boxes = [
  makeBox(2, 0, 0, 0xff0000),
  makeBox(0, 2, 0, 0x00ff00),
  makeBox(0, 0, 2, 0x0000ff)
];
let materials = [];

let g = BufferGeometryUtils.mergeBufferGeometries(boxes.map(b => {
  materials.push(b.material);
  b.updateMatrixWorld(); // needs to be done to be sure that all transformations are applied
  return b.geometry.applyMatrix4(b.matrixWorld);
}), true);

let o = new THREE.Mesh(g, materials);
scene.add(o);

window.addEventListener("resize", onWindowResize);

animate();

function animate() {

  requestAnimationFrame(animate);
  renderer.render(scene, camera);

}

function onWindowResize() {

  camera.aspect = innerWidth / innerHeight;
  camera.updateProjectionMatrix();

  renderer.setSize(innerWidth, innerHeight);

}

function makeBox(x, y, z, color) {
  let g = new THREE.BoxGeometry();
  let m = new THREE.MeshLambertMaterial({
    color: color
  });
  let box = new THREE.Mesh(g, m);
  box.position.set(x, y, z);
  return box;
}

</script>


花了一些时间来实现,但似乎它在处理gltf模型时效果不太好。此外,调用计数保持不变,使合并变得毫无意义。尽管如此,我非常感谢您花费的时间来提供帮助 :) - Vardan Betikyan

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接