·场景搭建
使用npm或者其他获取安装three,就像npm i three,之后在需要演示模型的vue组件内import * as THREE from 'three',此时我们就可以创建场景scene、灯光light、镜头、几何体等等开始渲染循环了
镜头控制是必要的,我们不会加载一个3D模型只为了看一面的剪影,从官网可以找到许多种控制器,本项目使用的是npm i three-orbitcontrols
import OrbitControls from 'three-orbitcontrols'
create(){ //创建场景
this.renderer = new THREE.WebGLRenderer(); this.camera = new THREE.PerspectiveCamera( 75, window.innerWidth / window.innerHeight, 0.1, 5000 )
this.camera.position.z = 160
this.camera.position.x = 0
this.camera.position.y = 0 this.scene = new THREE.Scene();
this.scene.add(this.camera) //鼠标控制镜头
this.controls = new OrbitControls(this.camera) //创建一个环境灯光
var ambientLight = new THREE.AmbientLight( 0xffffff, 0.4 );
this.scene.add( ambientLight ); //创建一个点灯光
var pointLight = new THREE.PointLight( 0xffffff, 0.5 ); //给相机添加光源
this.camera.add( pointLight ); //渲染器样式
this.renderer.setClearColor(new THREE.Color('black'));
this.renderer.setSize(window.innerWidth, window.innerHeight);
this.renderer.domElement.setAttribute("style"," height:100%;width: 100%;") 30 },
现在有了一个有灯光和控制器但没有模型的场景,而且没有挂载到页面
·引入加载器
显然,我们无法使用代码建立所有的模型,尽管three提供了建立多种几何体的强大功能让我们可以以纯代码开发三维场景,但项目中仍需要three提供的多种3D模型文件的Loader.js来帮助我们将建模软件生产的模型加载成可以添加到scene的object
你可能需要展示stl|obj|3ds|3mf|....等等文件类型,我们可以在已安装的three路径 “\node_modules\three\examples\js\loaders” 下找到所有已支持的类型加载器,如果你找到了3MFLoader.js但没有看到3DSLoader.js,可以到threejs的官方示例下查找使用样例,页面右下角查看源代码就会发现3ds加载器叫TDSLoader.js ;P
官方样例统统使用head写法引入js,这不适合vue,除非你的网站就只有这一个组件演示这一类模型(要知道vue大都是单页面网站),按照不同文件类型按需加载对应的加载器js才是效率的使用方式,那么。。使用ES6提供的import()实现运行时加载
例如objloader.js,将文件从node_modules中移到用户可以获取到的路径下(比如static),就可以按如下方式使用了
somefunction:(path){//模型文件url
......
import('@/static/OBJLoader.js').then(module => { let Loader = new THREE.OBJLoader(); that.loadermashs(Loader,path) }).catch(err => { this.message = "加载器获取失败:"+err;
});
} onProgress( xhr ) { if ( xhr.lengthComputable ) { var percentComplete = xhr.loaded / xhr.total * 100; //message承接加载进度
this.message = Math.round(percentComplete, 2) + '% 已加载' ;
}
}, onError( xhr ) { this.message = xhr;
}, loadermashs(Loader,path){ let that = this; Loader.setPath(path); Loader.load('', function(object) { that.scene.add(object); }, that.onProgress, that.onError);
},
注意:需要手动在OBJLoader.js中首行加入:var THREE = require('three')
e.g 编译时warning:"export 'OBJLoader' (imported as 'THREE') was not found in 'three',是因为THREE.OBJLoader()关系是运行时建立的,怎么消除这个warning我也不知道。。
·挂载与卸载
start(){//挂载到页面开始循环
document.getElementById("hello").appendChild(this.renderer.domElement)
this.running = true; this.animate()
}, animate() { this.renderer.render(this.scene, this.camera) if(this.running)
requestAnimationFrame(this.animate)//再次调用animate
}
running作为关闭标识
webGL中,手动卸载是有必要的,按照开发者原意,系统判断哪些资源可以废弃是不可能的:用户可能在某一帧放弃使用某个模型,也可能在下一帧重新启用,因此自动丢弃不被使用的资源是不合理的,而不做手动处理的后果是当你多次打开演示后:
Error: WebGL: Exceeded 16 live WebGL contexts for this principal, losing the least recently used one on...
所以请在关闭演示之前卸载他们:相关api
if(this.running){ let canvas = document.getElementsByTagName('canvas')[0]; if(canvas){
let gl = canvas.getContext('webgl'); canvas.addEventListener('webglcontextlost', function(e) {
//console.log(e);
}, false); gl.getExtension('WEBGL_lose_context').loseContext();
}
this.running = false this.renderer.dispose(); this.scene.dispose(); this.controls.dispose();
}
·其他
网页应用的dialog中展示某obj模型,最终演示效果:
此外还有材质加载、贴图、stl格式jzip问题等等。。有的坑我还没踩只是远远的望了一下,如果同为初学者,那么希望能帮到你
如有见解还请大佬指教:D