Three.js使用对象组合实例方法
将多个模型放到一个组里面,就是一个对象组合。 创建组非常简单,每个你创建的网格都可以包含子元素,子元素可以使用add
函数来添加。在组中添加子元素的效果是:你可以移动、缩放、旋转和变形父对象,而所有的子对象都将会受到影响。
对象组合的实现
对象组合很好实现,首先创建一个THREE.Object3D
的类的对象。这是THREE.Mesh
和THREE.Scene
的基类,但是它本身不包含任何东西,也不会渲染任何东西。请注意,在THREE.js
的最新版本中引入了一个名为THREE.Group
的新对象以支持分组。该对象与THREE.Object3D
对象完全相同,它两个可以互换。
var group = new THREE.Object3D(); //实例化一个THREE.Object3D对象 group.add(sphere); //在对象里面添加第一个子元素 group.add(cube); //在对象里面添加第二个子元素 scene.add(group); //将对象组添加到场景当中
代码如上,我们就实现了一个场景组。
注意:你旋转一个组时,并不是分别旋转组中的每一个对象,而是绕其中心旋转整个组(在我们的例子里,是绕着group
对象的中心旋转整个组)。
使用组的时候,你依然可以引用、修改和定位每一个单独的几何体。唯一需要记住的是:所有的定位、旋转和变形都是相对父对象的。
案例代码
<!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8"> <title>Title</title> <style type="text/css"> html, body { margin: 0; height: 100%; } canvas { display: block; } </style></head><body onload="draw();"></body><script src="/lib/three.js"></script><script src="/lib/js/controls/OrbitControls.js"></script><script src="/lib/js/libs/stats.min.js"></script><script src="/lib/js/libs/dat.gui.min.js"></script><script> var renderer; function initRender() { renderer = new THREE.WebGLRenderer({antialias:true}); renderer.setSize(window.innerWidth, window.innerHeight); //告诉渲染器需要阴影效果 renderer.shadowMap.enabled = true; renderer.shadowMap.type = THREE.PCFSoftShadowMap; // 默认的是,没有设置的这个清晰 THREE.PCFShadowMap document.body.appendChild(renderer.domElement); } var camera; function initCamera() { camera = new THREE.PerspectiveCamera(45, window.innerWidth/window.innerHeight, 0.1, 1000); camera.position.set(0, 40, 50); camera.lookAt(new THREE.Vector3(0,0,0)); } var scene; function initScene() { scene = new THREE.Scene(); } //初始化dat.GUI简化试验流程 var gui; function initGui() { //声明一个保存需求修改的相关数据的对象 gui = { sphereX:-5, //球的x轴的位置 sphereY:5, //球的y轴的位置 sphereZ:0, //球的z轴的位置 sphereScale:1, //球的缩放 cubeX:15, //立方体的x轴位置 cubeY:5, //立方体的y轴位置 cubeZ:-5, //立方体的z轴的位置 cubeScale:1, //立方体的缩放 groupX:0, //模型组的x轴位置 groupY:0, //模型组的y轴位置 groupZ:0, //模型组的z轴的位置 groupScale:1, //模型组的缩放 grouping:false, //是否整个模型组旋转 rotate:false, //是否旋转 }; var datGui = new dat.GUI(); //将设置属性添加到gui当中,gui.add(对象,属性,最小值,最大值) //球型的操作 var sphereFolder = datGui.addFolder("sphere"); sphereFolder.add(gui,"sphereX",-30,30).onChange(function (e) { sphere.position.x = e; }); sphereFolder.add(gui,"sphereY",-30,30).onChange(function (e) { sphere.position.y = e; }); sphereFolder.add(gui,"sphereZ",-30,30).onChange(function (e) { sphere.position.z = e; }); sphereFolder.add(gui,"sphereScale",0,3).onChange(function (e) { sphere.scale.set(e, e, e); }); //立方体的操作 var cubeFolder = datGui.addFolder("cube"); cubeFolder.add(gui,"cubeX",-30,30).onChange(function (e) { cube.position.x = e; }); cubeFolder.add(gui,"cubeY",-30,30).onChange(function (e) { cube.position.y = e; }); cubeFolder.add(gui,"cubeZ",-30,30).onChange(function (e) { cube.position.z = e; }); cubeFolder.add(gui,"cubeScale",0,3).onChange(function (e) { cube.scale.set(e, e, e); }); //场景组的操作 var groupFolder = datGui.addFolder("group"); groupFolder.add(gui,"groupX",-30,30).onChange(function (e) { group.position.x = e; }); groupFolder.add(gui,"groupY",-30,30).onChange(function (e) { group.position.y = e; }); groupFolder.add(gui,"groupZ",-30,30).onChange(function (e) { group.position.z = e; }); groupFolder.add(gui,"groupScale",0,3).onChange(function (e) { group.scale.set(e, e, e); }); //添加旋转功能 datGui.add(gui, "grouping"); datGui.add(gui, "rotate"); } var light; function initLight() { scene.add(new THREE.AmbientLight(0x444444)); light = new THREE.PointLight(0xffffff); light.position.set(15,50,10); //告诉平行光需要开启阴影投射 light.castShadow = true; scene.add(light); } var sphere,cube,group; function initModel() { //模型组 group = new THREE.Object3D(); scene.add(group); //球 var sphereGeometry = new THREE.SphereGeometry(5,200,200); var sphereMaterial = new THREE.MeshLambertMaterial({color:0xaaaaaa}); sphere = new THREE.Mesh(sphereGeometry, sphereMaterial); sphere.position.x = -5; sphere.position.y = 5; //告诉球需要投射阴影 sphere.castShadow = true; group.add(sphere); //辅助工具 var helper = new THREE.AxisHelper(50); scene.add(helper); //立方体 var cubeGeometry = new THREE.CubeGeometry(10,10,8); var cubeMaterial = new THREE.MeshLambertMaterial({color:0x00ffff}); cube = new THREE.Mesh(cubeGeometry, cubeMaterial); cube.position.x = 15; cube.position.y = 5; cube.position.z = -5; //告诉立方体需要投射阴影 cube.castShadow = true; group.add(cube); //底部平面 var planeGeometry = new THREE.PlaneGeometry(100,100); var planeMaterial = new THREE.MeshStandardMaterial({color:0xaaaaaa}); var plane = new THREE.Mesh(planeGeometry, planeMaterial); plane.rotation.x = - 0.5 * Math.PI; plane.position.y = -0; //告诉底部平面需要接收阴影 plane.receiveShadow = true; scene.add(plane); } //初始化性能插件 var stats; function initStats() { stats = new Stats(); document.body.appendChild(stats.dom); } //用户交互插件 鼠标左键按住旋转,右键按住平移,滚轮缩放 var controls; function initControls() { controls = new THREE.OrbitControls( camera, renderer.domElement ); // 如果使用animate方法时,将此函数删除 //controls.addEventListener( 'change', render ); // 使动画循环使用时阻尼或自转 意思是否有惯性 controls.enableDamping = true; //动态阻尼系数 就是鼠标拖拽旋转灵敏度 //controls.dampingFactor = 0.25; //是否可以缩放 controls.enableZoom = true; //是否自动旋转 controls.autoRotate = false; //设置相机距离原点的最远距离 controls.minDistance = 100; //设置相机距离原点的最远距离 controls.maxDistance = 200; //是否开启右键拖拽 controls.enablePan = true; } var step = 0.02; //模型旋转的速度 function render() { //判断当前是否自动旋转 if(gui.rotate){ //判断是单个模型自转,还是模型组自转 if(gui.grouping){ group.rotation.y += step; } else{ sphere.rotation.y += step; cube.rotation.y += step; } } renderer.render( scene, camera ); } //窗口变动触发的函数 function onWindowResize() { camera.aspect = window.innerWidth / window.innerHeight; camera.updateProjectionMatrix(); render(); renderer.setSize( window.innerWidth, window.innerHeight ); } function animate() { //更新控制器 render(); //更新性能插件 stats.update(); controls.update(); requestAnimationFrame(animate); } function draw() { initGui(); initRender(); initScene(); initCamera(); initLight(); initModel(); initControls(); initStats(); animate(); window.onresize = onWindowResize; }</script></html>
以上是Three.js使用对象组合实例方法的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

如何使用WebSocket和JavaScript实现在线语音识别系统引言:随着科技的不断发展,语音识别技术已经成为了人工智能领域的重要组成部分。而基于WebSocket和JavaScript实现的在线语音识别系统,具备了低延迟、实时性和跨平台的特点,成为了一种被广泛应用的解决方案。本文将介绍如何使用WebSocket和JavaScript来实现在线语音识别系

WebSocket与JavaScript:实现实时监控系统的关键技术引言:随着互联网技术的快速发展,实时监控系统在各个领域中得到了广泛的应用。而实现实时监控的关键技术之一就是WebSocket与JavaScript的结合使用。本文将介绍WebSocket与JavaScript在实时监控系统中的应用,并给出代码示例,详细解释其实现原理。一、WebSocket技

如何利用JavaScript和WebSocket实现实时在线点餐系统介绍:随着互联网的普及和技术的进步,越来越多的餐厅开始提供在线点餐服务。为了实现实时在线点餐系统,我们可以利用JavaScript和WebSocket技术。WebSocket是一种基于TCP协议的全双工通信协议,可以实现客户端与服务器的实时双向通信。在实时在线点餐系统中,当用户选择菜品并下单

如何使用WebSocket和JavaScript实现在线预约系统在当今数字化的时代,越来越多的业务和服务都需要提供在线预约功能。而实现一个高效、实时的在线预约系统是至关重要的。本文将介绍如何使用WebSocket和JavaScript来实现一个在线预约系统,并提供具体的代码示例。一、什么是WebSocketWebSocket是一种在单个TCP连接上进行全双工

JavaScript和WebSocket:打造高效的实时天气预报系统引言:如今,天气预报的准确性对于日常生活以及决策制定具有重要意义。随着技术的发展,我们可以通过实时获取天气数据来提供更准确可靠的天气预报。在本文中,我们将学习如何使用JavaScript和WebSocket技术,来构建一个高效的实时天气预报系统。本文将通过具体的代码示例来展示实现的过程。We

JavaScript教程:如何获取HTTP状态码,需要具体代码示例前言:在Web开发中,经常会涉及到与服务器进行数据交互的场景。在与服务器进行通信时,我们经常需要获取返回的HTTP状态码来判断操作是否成功,根据不同的状态码来进行相应的处理。本篇文章将教你如何使用JavaScript获取HTTP状态码,并提供一些实用的代码示例。使用XMLHttpRequest

JavaScript是一种广泛应用于Web开发的编程语言,而WebSocket则是一种用于实时通信的网络协议。结合二者的强大功能,我们可以打造一个高效的实时图像处理系统。本文将介绍如何利用JavaScript和WebSocket来实现这个系统,并提供具体的代码示例。首先,我们需要明确实时图像处理系统的需求和目标。假设我们有一个摄像头设备,可以采集实时的图像数

Golang是一门功能强大且高效的编程语言,可以用于开发各种应用程序和服务。在Golang中,指针是一种非常重要的概念,它可以帮助我们更灵活和高效地操作数据。指针转换是指在不同类型之间进行指针操作的过程,本文将通过具体的实例来学习Golang中指针转换的最佳实践。1.基本概念在Golang中,每个变量都有一个地址,地址就是变量在内存中的位置。
