使用three.js搭建室内场景

公司做商城、消防、用电等项目,需要实现楼层和设备的可视化,以前都是使用其他建模工具创建的整体模型,再使用three.js的加载器加载到场景中,但是这样的加载存在缺陷,比如不能给模型的元素赋属性、不能单个点击元素、渲染单调等。所以本次参考了一些资料,不使用模型倒入,完全使用three.js搭建场景,代码有些粗燥勿怪。

 在线预览地址

CSDN下载地址https://download.csdn.net/download/u014529917/85432876

1.创建地板

       地板是一个类似盒子,有顶部有底部有侧面,但是不一定是规则的盒子,因此我放弃了常用的BoxGeometry的方式,改用顶点+面的形式创建任意多边形地板,已知多边形底部坐标,底部坐标加上高度得到顶部坐标,通过Earcut可以计算出底部和顶部的三角面,侧面的三角面可以直接通过坐标序号得到,由此可以创建一个通用的Geometry。

Floor.prototype.getGeometry = function(points,height){
    var topPoints = [];
    for(var i=0;i<points.length;i++){
        var vertice = points[i];
        topPoints.push([vertice[0],vertice[1]+height,vertice[2]]);
    }
    var totalPoints = points.concat(topPoints);
    var vertices =[];           //所有的顶点
    for(var i=0;i<totalPoints.length;i++){
        vertices.push(new THREE.Vector3(totalPoints[i][0],totalPoints[i][1],totalPoints[i][2]))
    }
    var length = points.length;
    var faces = [];
    for(var j=0;j<length;j++){                      //侧面生成三角形
        if(j!=length-1){
            faces.push(new THREE.Face3(j,j+1,length+j+1));
            faces.push(new THREE.Face3(length+j+1,length+j,j));
        }else{
            faces.push(new THREE.Face3(j,0,length));
            faces.push(new THREE.Face3(length,length+j,j));
        }
    }
    var data=[];
    for(var i=0;i<length;i++){
        data.push(points[i][0],points[i][2]);
    }
    var triangles = Earcut.triangulate(data);
    if(triangles && triangles.length != 0){
        for(var i=0;i<triangles.length;i++){
            var tlength = triangles.length;
            if(i%3==0 && i < tlength-2){
                faces.push(new THREE.Face3(triangles[i],triangles[i+1],triangles[i+2]));                            //底部的三角面
                faces.push(new THREE.Face3(triangles[i]+length,triangles[i+1]+length,triangles[i+2]+length));        //顶部的三角面
            }
        }
    }
    var geometry = new THREE.Geometry();
    geometry.vertices = vertices;
    geometry.faces = faces;
    geometry.computeFaceNormals();      //自动计算法向量
    return geometry;
}

效果:

2.创建墙体

墙体我使用了BoxGeometry,墙体上的窗户的洞、门洞,我们可以使用ThreeBSP库中差集函数来进行模型相减来实现。

Floor.prototype.addWall = function(size,position,rotation,holes){
    var geometry = new THREE.BoxGeometry(size[0], size[1], size[2]);
    var materials = new THREE.MeshLambertMaterial({color: 0xb0cee0,side:THREE.DoubleSide})
    var result = new THREE.Mesh(geometry,materials);
    if(holes){
        result = cube;
        for(var i=0;i<holes.length;i++){
            var totalBSP = new ThreeBSP(result);
            var hole = holes[i];
            var holeGeometry = new THREE.BoxGeometry(hole.size[0], hole.size[1], hole.size[2]);
            var holeCube = new THREE.Mesh( holeGeometry); 
            holeCube.position.x = hole.position[0];
            holeCube.position.y = hole.position[1] + hole.size[1]/2;
            holeCube.position.z = hole.position[2];
            var clipBSP = new ThreeBSP(holeCube);
            var  resultBSP = totalBSP.subtract(clipBSP);
            result = resultBSP.toMesh();
        }
        result.material = materials;
    }
    this.container.add(result);             //添加填充
}

效果:

3.门框

在添加门之前,为了更加形象一点,我添加了门框。先使用墙体减去门框的洞,再添加减去门洞的门框,跟前面类似,具体代码不放了。

效果:

4.门、窗、主机、显示屏、桌子

门、窗、主机、显示屏、桌子 我都是使用BoxGeometry的形式,再给相应的面贴纹理,跟前面类似,效果如下:

5.盆栽

盆栽的盆体可以使用CylinderBufferGeometry来创建顶部大于底部的圆台,盆栽的叶子是使用多个PlaneGeometry贴上植物纹理以不同的角度展示,代码如下:

//盆栽
Floor.prototype.addPlant = function(position,scale){
    var plant = new THREE.Object3D();
    var geometry = new THREE.CylinderBufferGeometry( 0.15, 0.1, 0.4, 22 );
    var material = new THREE.MeshLambertMaterial( {color: 0xffffff} );

    var cylinder = new THREE.Mesh( geometry, material );
    cylinder.position.x = 0;
    cylinder.position.y = 0.2;
    cylinder.position.z = 0;
    plant.add( cylinder );

    var leafTexture = new THREE.TextureLoader().load('meeting/plant.png');
    var leafMaterial = new THREE.MeshBasicMaterial({map:leafTexture,side:THREE.DoubleSide,transparent:true});
    var geom = new THREE.PlaneGeometry(0.4, 0.8);
    for(var i=0;i<4;i++){
        var leaf = new THREE.Mesh( geom, leafMaterial );
        leaf.position.y = 0.8;
        leaf.rotation.y = -Math.PI/(i+1);
        plant.add(leaf);
    }
    plant.position.x = position[0];
    plant.position.y = position[1];
    plant.position.z = position[2];
    this.container.add(plant);
}

效果(很粗燥):

6.椅子

椅子的模型有点复杂,因为这个差点放弃用three创建椅子,但看到一个同行完全用three创建的minicity,又有了信心和勇气。于是:4条椅子腿定位+旋转、椅子面、2条靠背腿定位+旋转、靠背定位+旋转,最终创建完成,代码太丑陋就不上了。效果:

8.开门动画

开门动画我使用了TWEEN库,Tween.js是一个包含各种经典动画算法的JS资源,动态改变门在z轴方向上的值。

if(status == "close"){
    status = "open";
    var desRotation = Math.PI/2;
    new TWEEN.Tween(door.rotation).to({
        y: desRotation
    }, 10000).easing(TWEEN.Easing.Elastic.Out).onComplete(function(){
    }).start();
}else{
    status = "close";
    new TWEEN.Tween(door.rotation).to({
        y: 0
    }, 10000).easing(TWEEN.Easing.Elastic.Out).onComplete(function(){
    }).start();
}

效果:

9.行走动画

行走动画我使用了three的animation模块,导入带动画的fbx模型,关于模型动画的制作很复杂,我们可以在网络上下载。导入动画之后播放动画。

var Mixers = [];
var animation;
var walkingMan;

var loader = new THREE.FBXLoader();
loader.load('file/walkingman.fbx', function ( object ) {  //Samba Dancing.fbx
    object.mixer = new THREE.AnimationMixer( object );
    Mixers.push( object.mixer );                                             //AnimationMixer
    animation = object.mixer.clipAction( object.animations[ 0 ] );         //AnimationAction AnimationClip
    walkingMan = object;
    walkingMan.scale.x = walkingMan.scale.y = walkingMan.scale.z = 0.8;
    walkingMan.position.x = firstPoint[0];
    walkingMan.position.y = firstPoint[1];
    walkingMan.position.z = firstPoint[2];
    walkingMan.rotation.y = rotation;       //角度 根据当前点和下一个点计算 
    scene.add( walkingMan );
    animation.play();
});

function updateWalkingMan(){
    if ( Mixers.length > 0 ) {
        for ( var i = 0; i < Mixers.length; i ++ ) {
            Mixers[ i ].update(clock.getDelta());//clock.getDelta()
        }
    }
}

function render() {
    updateWalkingMan();
    requestAnimationFrame(render);
    renderer.render(scene, camera);
}

效果:

在播放动画的同时,我们可以更改人物模型的位置、角度,达到在场景中走动的效果:

会议室建模告一段落,这也是一次探索吧。后续的目标是封装常用的模型、在web中建立用户交互的建模方式,更加标准、快速的搭建室内场景。

  • 26
    点赞
  • 145
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 50
    评论
### 回答1: three.js 是一款非常强大的 JavaScript 3D 渲染引擎,可以用来制作复杂的3D场景。通过使用它,我们可以很容易地搭建一个简单的3D仓库。 首先,我们需要准备一个模型。可以从一些公开的模型库中下载一个简单的仓库模型。 在 three.js 中,所有的模型都需要被载入、渲染和组织起来。 我们可以使用Three.js的OBJLoader将模型加载到场景中。 然后我们可以建立一个场景场景包含了所有需要渲染的对象,这些对象被渲染到画布中。我们也需要添加一个相机,指定场景中哪些部分需要被渲染。 在场景中加入一些灯光,可以提高渲染效果。我们可以使用 Three.js 的光源来添加灯光。同时也可以使用一些效果增强技术,如阴影,自然地模拟光影效应。 最后,我们设置一些输入和动画控制,这可以使我们对场景进行控制和交互。 比如,使用 Three.js 的 OrbitControls 实现鼠标拖拽和缩放效果。 通过这些步骤,我们可以搭建一个简单的 3D 仓库,您可以在浏览器中进行漫游和交互。Three.js 使得3D开发变得简单,相信在不断学习和实践后,您可以开发出更加贴近客户需要的应用场景。 ### 回答2: three.js是一款基于JavaScript的WebGL库,用于创建三维图形场景使用three.js可以简便地创建3D场景、动态交互式动画效果和各种效果的实现。 下面将介绍如何使用three.js搭建简单的3D仓库: 步骤一:创建场景布局 使用three.js,我们可以在浏览器中创建一个三维环境场景。首先需要定义一个场景变量,以便添加所有的3D对象。同时,还需要定义一个相机、渲染器和灯光。 代码示例: var scene = new THREE.Scene(); //定义场景变量 var camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);//定义相机变量 var renderer = new THREE.WebGLRenderer(); //定义渲染器变量 renderer.setSize(window.innerWidth, window.innerHeight); //设置渲染器的大小 //添加灯光 var light = new THREE.PointLight(0xffffff, 1, 100); light.position.set(0, 0, 3); scene.add(light); //添加相机 camera.position.z = 5; //将渲染器添加到页面中 document.body.appendChild(renderer.domElement); 步骤二:添加物体 接下来需要向场景中添加3D对象。这里我们可以添加几个较为简单的3D模型,例如长方体、球、圆锥等等。 代码示例: var geometry = new THREE.BoxGeometry(1, 1, 1); //创建长方体对象(宽,高,深) var material = new THREE.MeshBasicMaterial({color: 0x00ff00}); //材质(颜色) var cube = new THREE.Mesh(geometry, material); //网格对象 scene.add(cube); //将长方体添加到场景中 var sphereGeometry = new THREE.SphereGeometry(0.5, 32, 32); //创建球体对象(半径,水平分段数,垂直分段数) var sphereMaterial = new THREE.MeshBasicMaterial({color: 0xff0000}); //材质(颜色) var sphere = new THREE.Mesh(sphereGeometry, sphereMaterial); //网格对象 sphere.position.x = 2; //球体沿x轴平移2个单位 scene.add(sphere); //将球体添加到场景中 var coneGeometry = new THREE.ConeGeometry(0.5, 1, 32); //创建圆锥对象(半径,高度,分段数) var coneMaterial = new THREE.MeshBasicMaterial({color: 0x0000ff}); //材质(颜色) var cone = new THREE.Mesh(coneGeometry, coneMaterial); //网格对象 cone.position.x = -2; //圆锥沿x轴平移-2个单位 scene.add(cone); //将圆锥添加到场景中 至此,我们已经成功地搭建了简单的3D仓库场景。你可以通过使用鼠标和键盘进行场景的旋转、缩放和平移。 ### 回答3: 随着3D技术的发展,越来越多的项目开始应用3D技术,而three.js作为一种流行的3D库,能够轻松地搭建3D场景,为项目提供更丰富的视觉体验。在本文中,我们将演示如何使用three.js搭建一个简单的3D仓库。 首先,我们需要准备一些材料,包括three.js库、一个3D模型文件、一个HTML文件和一些CSS样式。在这里,我们将使用three.js提供的mmdloader和OrbitControls插件来加载模型和控制场景的相机。我们将模型文件命名为“model.pmx”。 接下来,创建一个HTML文件,并将three.js、mmdloader.js和OrbitControls.js库的链接添加到文件中。同时,我们还需要在CSS文件中定义场景的大小和颜色。 在HTML文件的主体中,我们需要定义一个容器来承载场景使用THREE.WebGLRenderer创建一个渲染器,然后将其添加到容器中。使用THREE.Clock来创建一个时间对象来跟踪时间,以便在动画使用。 接下来,我们需要通过使用mmdloader插件加载3D模型文件。使用THREE.MMDLoader()创建一个加载器对象,并使用该对象的load函数加载模型文件。加载完成后,调用场景中的add()函数将模型添加到场景中。同时,我们还要设置模型动画循环、速度等信息。 完成模型加载后,我们需要定义场景中的相机并添加OrbitControls插件来控制相机的移动和缩放。同时,我们还需要定义灯光来照亮场景。 最后,在场景中添加地面和一些墙体,以及一些边角线来突出显示仓库的结构。通过调整场景中的元素和相机视角,可以创建一个简单而真实的3D仓库。 总之,通过three.js,我们可以轻松地创建出3D场景,并且可以通过设置不同的属性来调整场景元素的大小、位置和动态效果。虽然本文只是演示了如何创建一个简单的3D仓库,但是Three.js拥有丰富的API和插件,可以满足更多复杂场景的需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 50
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

evomap

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值