Vue中使用Three.js加載glTF模型

Three.js是一個跨瀏覽器的腳本暇韧,使用JavaScript函數(shù)庫或API來在網(wǎng)頁瀏覽器中創(chuàng)建和展示動畫的三維計算機圖形舷蒲,基于WebGL實現(xiàn)煤裙,對WebGL進行了進一步的封裝楼镐,簡化了多數(shù)復雜的接口扼劈。


Three.js支持包括 .obj、.gltf等類型的模型結(jié)構(gòu)蝌矛。

glTF(GL傳輸格式)是Khronos的一個開放項目道批,它為3D資產(chǎn)提供了一種通用的、可擴展的格式入撒,這種格式既高效又與現(xiàn)代web技術(shù)高度互操作隆豹。

obj格式的模型只支持頂點、法線茅逮、紋理坐標和基本材質(zhì)璃赡,而glTF模型除上述所有內(nèi)容外判哥,glTF還提供了如下功能:

? ? 層級對象

????場景信息(光源,相機)

????骨骼結(jié)構(gòu)與動畫

????更可靠的材質(zhì)和著色器


一碉考、安裝引入Three.js

npm install three

在需要使用3D模型的頁面導入包:

import * as Three from "three"

在Vue中導入glTF模型需要使用 Three.js 中的 GLTFLoader:

import { GLTFLoader }?from?"three/examples/jsm/loaders/GLTFLoader"

// 導入軌道模型控制器

import?{?OrbitControls?}?from?"three/examples/jsm/controls/OrbitControls


二塌计、頁面Dom元素渲染

在Vue中,我們需要使用一個 div 元素來作為3D模型的容器:

<div?id="container"></div>

頁面打開之后侯谁,Three.js會給 div 元素添加一個 canvas 子元素用來作為3D模型的畫布锌仅。

三、初始化

Three.js中最重要的三大組件:

????場景——Scene

? ? 相機——Camera

? ? 渲染器——Renderer

初始化:

mounted(){

? ? this.initScene()

? ? this.initContainer()

? ? this.initCamera()

? ? this.initRenderer()

? ? this.initControls()

},

methods:{

? ???initModelContainer()?{

??????this.model_container?=?document.getElementById("container");

??????this.model_container.style.height?=?window.innerHeight?+?"px";

??????this.model_container.style.width?=?window.innerWidth?+?"px";

??????this.height?=?this.model_container.clientHeight;

??????this.width?=?this.model_container.clientWidth;

????},

????initScene()?{

??????this.scene?=?new?Three.Scene();

????},

? ??

????initCamera()?{

????????//?照相機

??????this.camera?=?new?Three.PerspectiveCamera(70, this.width?/?this.height, 0.01, 1000);

??????this.camera.position.set(-100,?60,?0);

????},

????initRenderer()?{

??????this.renderer?=?new?Three.WebGLRenderer({?antialias:?true,?alpha:?true?});

??????this.renderer.setSize(this.width,?this.height);

??????//?兼容高清屏幕

??????this.renderer.setPixelRatio(window.devicePixelRatio);

? ? ? ?// 消除canvas的外邊框

? ? ? this.renderer.domElement.style.outline?=?"none";

??????this.model_container.appendChild(this.renderer.domElement);

????},

? ??initControls()?{

??????this.orbitControls?=?new?OrbitControls(

????????this.camera,

????????this.renderer.domElement

??????);

??????//?慣性

??????this.orbitControls.enableDamping?=?true;

??????//?動態(tài)阻尼系數(shù)

??????this.orbitControls.dampingFactor?=?0.25;

??????//?縮放

??????this.orbitControls.enableZoom?=?true;

??????//?右鍵拖拽

??????this.orbitControls.enablePan?=?true;

??????//?水平旋轉(zhuǎn)范圍

? ? ? this.orbitControls.maxAzimuthAngle?=?Math.PI?/?6;

? ? ? this.orbitControls.minAzimuthAngle?=?-Math.PI?/?6;

??????// 垂直旋轉(zhuǎn)范圍

? ? ? this.orbitControls.maxPolarAngle?=?Math.PI?/?6;

? ? ? this.orbitControls.minPolarAngle?=?-Math.PI?/?6;

????},

}

四墙贱、導入glTF模型

將你的 gltf 模型放在 Vue 項目中的 public 文件夾下热芹,注意,只有將 gltf 模型放在靜態(tài)資源文件夾下才能被訪問到惨撇。

在鉤子函數(shù) mounted 中進行模型加載:

mounted(){

? ? this.loadModel()

},

methods:{

? ? loadModel(){

? ? ? ? let that = this

? ? ? ? // gltf模型加載器

? ? ? ? let loader = new GLTFLoader()

? ? ? ? return new Promise(function(resolve, reject){

? ? ? ? ? ? loader.load(

? ? ? ? ? ? ? ? // 模型在 /public/static/building/文件夾下

? ? ? ? ? ? ? ? "static/building/scene.gltf",

? ? ? ? ? ? ? ? gltf => {

? ? ? ? ? ? ? ? ? ? console.log(gltf)

? ? ? ? ? ? ? ? ? ? gltf.scene.traverse(object => {

? ? ? ? ? ? ? ? ? ? ? ? // 修改模型材質(zhì)

? ? ? ? ? ? ? ? ? ? ? ? let material = ...

? ? ? ? ? ? ? ? ? ? ? ? object.material = material

????????????????????})

? ? ? ? ? ? ? ? ? ? let group = new Three.Group()

? ? ? ? ? ? ? ? ? ? group.add(gltf.scene)

? ? ? ? ? ? ? ? ? ? let box = new Three.Box3()

? ? ? ? ? ? ? ? ? ? box.setFromObject(group)

? ??????????????????let wrapper = new Three.Object3D()

? ? ? ? ? ? ? ? ? ? wrapper.add(group)

? ? ? ? ? ? ? ? ? ? // 根據(jù)自己模型的大小設(shè)置位置

? ? ? ? ? ? ? ? ? ? wrapper.position.set(100, -300, 120)

? ? ? ? ? ? ? ? ? ? // 將模型加入到場景中? ! important

? ? ? ? ? ? ? ? ? ? that.scene.add(wrapper)

????????????????},

? ? ? ? ? ? ? ? xhr => {

? ? ? ? ? ? ? ? ? ? // 模型加載期間的回調(diào)函數(shù)

? ??????????????????console.log(`${(xhr.loaded?/?xhr.total)?*?100%?building?model?loaded`

????????????);

????????????????},

? ? ? ? ? ? ? ? error => {

? ??????????????????// 模型加載出錯的回調(diào)函數(shù)

? ??????????????????console.log("error?while?loading",?error);

????????????????????reject("load?model?error",?error);

????????????????}

? ? ? ? ? ? )

????????})

? ? }

}

啟動項目伊脓,模型導入成功,可以根據(jù)自己的需求為模型渲染材質(zhì)魁衙。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末丽旅,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子纺棺,更是在濱河造成了極大的恐慌榄笙,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,451評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件祷蝌,死亡現(xiàn)場離奇詭異茅撞,居然都是意外死亡,警方通過查閱死者的電腦和手機巨朦,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評論 3 394
  • 文/潘曉璐 我一進店門米丘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人糊啡,你說我怎么就攤上這事拄查。” “怎么了棚蓄?”我有些...
    開封第一講書人閱讀 164,782評論 0 354
  • 文/不壞的土叔 我叫張陵堕扶,是天一觀的道長。 經(jīng)常有香客問我梭依,道長稍算,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,709評論 1 294
  • 正文 為了忘掉前任役拴,我火速辦了婚禮糊探,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己科平,他們只是感情好褥紫,可當我...
    茶點故事閱讀 67,733評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著瞪慧,像睡著了一般故源。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上汞贸,一...
    開封第一講書人閱讀 51,578評論 1 305
  • 那天,我揣著相機與錄音印机,去河邊找鬼矢腻。 笑死,一個胖子當著我的面吹牛射赛,可吹牛的內(nèi)容都是我干的多柑。 我是一名探鬼主播,決...
    沈念sama閱讀 40,320評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼楣责,長吁一口氣:“原來是場噩夢啊……” “哼竣灌!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起秆麸,我...
    開封第一講書人閱讀 39,241評論 0 276
  • 序言:老撾萬榮一對情侶失蹤初嘹,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后沮趣,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體屯烦,經(jīng)...
    沈念sama閱讀 45,686評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,878評論 3 336
  • 正文 我和宋清朗相戀三年房铭,在試婚紗的時候發(fā)現(xiàn)自己被綠了驻龟。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,992評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡缸匪,死狀恐怖翁狐,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情凌蔬,我是刑警寧澤露懒,帶...
    沈念sama閱讀 35,715評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站砂心,受9級特大地震影響隐锭,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜计贰,卻給世界環(huán)境...
    茶點故事閱讀 41,336評論 3 330
  • 文/蒙蒙 一钦睡、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧躁倒,春花似錦荞怒、人聲如沸洒琢。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,912評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽衰抑。三九已至,卻和暖如春荧嵌,著一層夾襖步出監(jiān)牢的瞬間呛踊,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,040評論 1 270
  • 我被黑心中介騙來泰國打工啦撮, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留谭网,地道東北人。 一個月前我還...
    沈念sama閱讀 48,173評論 3 370
  • 正文 我出身青樓赃春,卻偏偏與公主長得像愉择,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子织中,可洞房花燭夜當晚...
    茶點故事閱讀 44,947評論 2 355