工業互聯網,物聯網,可視化等名詞在咱們如今信息化的大背景下已是耳熟能詳,平常生活的交通,出行,吃穿等可能均可以用信息化的方式來爲咱們表達,在傳統的可視化監控領域,通常都是基於 Web SCADA 的前端技術來實現 2D 可視化監控,本系統採用 Hightopo 的 HT for Web 產品來構造輕量化的 3D 可視化場景,該 3D 場景從正面展現了一個地鐵站的現實場景,包括地鐵的實時運行狀況,地鐵上下行狀況,視頻監控,煙霧報警,電梯運行狀況等等,幫助咱們直觀的瞭解當前的地鐵站。javascript
系統中爲了幫助用戶更直觀友好的瀏覽當前地鐵站,提供了三種交互模式:html
本篇文章經過對地鐵站可視化場景的搭建,動畫代碼的實現,交互模式的原理解析,以及主要功能點的實現進行闡述,幫助咱們瞭解如何使用 HT 實現一個簡單的地鐵站可視化。前端
預覽地址:基於 HTML5 WebGL 的地鐵站 3D 可視化系統 http://www.hightopo.com/demo/...java
地鐵從站外開到站內的效果爲透明度逐漸增長,速度逐漸下降。node
上述爲自動巡檢的漫遊效果,場景自動進行前進旋轉。數組
當咱們點擊場景中的監控設備時能夠查看當前設備的運行狀況,運行數據等信息。ide
該系統中的大部分模型都是經過 3dMax 建模生成的,該建模工具能夠導出 obj 與 mtl 文件,在 HT 中能夠經過解析 obj 與 mtl 文件來生成 3d 場景中的全部複雜模型,固然若是是某些簡單的模型能夠直接使用 HT 來繪製,這樣會比 obj 模型更輕量化,因此大部分簡單的模型都是採用 HT for Web 產品輕量化 HTML5/WebGL 建模的方案,具體的解析代碼以下:函數
// 分別爲 obj 文件地址,mtl 文件地址 ht.Default.loadObj('obj/metro.obj', 'obj/metro.mtl', { center: true, // 模型是否居中,默認爲 false,設置爲 true 則會移動模型位置使其內容居中 r3: [0, -Math.PI/2, 0], // 旋轉變化參數,格式爲 [rx, ry, rz] s3: [0.15, 0.15, 0.15], // 大小變化參數,格式爲 [sx, sy, sz] finishFunc: function(modelMap, array, rawS3) { if(modelMap) { ht.Default.setShape3dModel('metro', array); // 註冊一個名字爲 metro 的模型 } } });
上面經過加載 obj 模型以後註冊了一個名字爲 metro 的模型,以後若是要使用該模型能夠經過如下代碼來實現:工具
var node = new ht.Node(); node.s({ 'shape3d': 'metro' });
上面代碼新建了一個 node 對象,經過設置 style 對象的 shape3d 屬性能夠把模型名稱爲 metro 用到該 node 對象上去,以後即是咱們場景中看到的地鐵列車模型。動畫
場景中地鐵的運行是經過 HT 提供的調度插件來實現,調度的具體用法能夠參考 HT for Web 的調度手冊,該調度主要用於在指定的時間間隔進行函數回調處理,回調函數的第一個參數爲 data 圖元,也就是 3D 場景中的模型節點,咱們能夠判斷當前 data 是否爲咱們剛纔建立的 metro 那個節點來進行後續的操做,場景中模擬了一個左開的地鐵和一個右開的地鐵,兩輛地鐵會交替出現。在 3D 場景中確定會有座標系,HT 中是用 x, y, z 來分別表示三個軸,因此地鐵的運動確定是改變地鐵在座標系中的位置來實現地鐵的運行,地鐵座標以下圖所示:
經過上圖能夠知道地鐵在 3D 場景中的座標系,若是要實現地鐵的移動則只須要將地鐵往圖中所示紅色箭頭的方向進行移動,即 x 軸的方向,經過 setX 這個方法不斷的修改地鐵的位置達到地鐵行進的目的,代碼中經過 getSpeedByX 以及 getOpacityByX 兩個方法來不斷獲取此時的列車速度以及列車透明度,如下爲關鍵代碼實現:
let metroTask = { interval: 50, // 每五十秒執行一次 action: (data) => { // 即上文所提回調函數 // 判斷當時傳進來的節點是否爲地鐵列車節點 if(data === currentMetro) { // 獲取地鐵此時的 X 軸位置以及行進的方向 let currentX = data.getX(), direction = data.a('direction'); // 根據當前的 X 軸位置獲取當前的列車速度 let speed = this.getSpeedByX(currentX); // 根據當前的 X 軸位置獲取當前的列車透明度 let opacity = this.getOpacityByX(currentX); // 判斷此時 X 軸位置是否超過某個值 即地鐵是在某個範圍內移動 if(Math.abs(currentX) <= 5000) { // 設置當前的透明度 opacity !== 1 ? currentMetro.s({ 'shape3d.transparent': true, 'shape3d.opacity': opacity }) : currentMetro.s({ 'shape3d.transparent': false}); // 設置當前的 X 軸位置 data.setX(currentX + direction * speed); // 判斷此時地鐵的速度爲 0,因此此時應該執行開門的動畫 if(speed === 0) this.doorAnimation(currentMetro, direction); } // 右方向地鐵開到頭,進行復位 if(currentX > 5000 && direction === 1) { currentMetro = leftMetro; currentMetro.setX(5000); } // 左方向地鐵開到頭,進行復位 if(currentX < -5000 && direction === -1) { currentMetro = rightMetro; currentMetro.setX(-5000); } } } }; dm3d.addScheduleTask(metroTask);
經過以上代碼能夠知道地鐵在運行的過程當中,主要經過修改地鐵的 x 軸位置來產生前進的動畫,而且須要讓地鐵在某個區間內進行運動,須要判斷邊界,並且爲了模擬出真實的效果須要根據地鐵當前的位置不斷獲取當前的列車速度以及列車透明度,如下爲流程圖:
上圖所示的爲地鐵進站時候的流程,當地鐵停靠完畢關門後須要進行出站,此時咱們只須要把地鐵位置從新設置一下不爲 0 便可,如下爲部分代碼實現:
currentMetro.setX(direction * 10); // 設置出站列車的位置
當執行上面那句代碼以後上方的 metroTask 調度任務執行到 getSpeedByX 這個方法以後獲取到的 speed 速度不爲 0,所以此時會繼續執行地鐵行進的動畫,此時的速度就是由慢至快,透明度由深至淺。如下爲開門動畫執行流程:
系統中自動巡檢的實現主要是經過修改 3D 場景中的 eye 以及 center 的值,HT 中提供了 rotate,walk 兩個方法來控制視角的旋轉以及視角的行進,rotate 方法在非第一人稱模式時,旋轉是以 center 爲中心進行旋轉,也就是圍繞中心物體旋轉,當爲第一人稱時旋轉以 eye 爲中心進行旋轉,也就是旋轉眼睛朝向方向。walk 函數同時改變 eye 和 center 的位置,也就是 eye 和 center 在兩點創建的矢量方向上同時移動相同的偏移量。該系統中我沒有采用 rotate 函數而是本身實現了視角的旋轉,由於本來的 rotate 函數旋轉某個角度會立刻旋轉過去而不會有一個旋轉的過程,因此我從新實現了旋轉的方法,該系統中視角旋轉是經過不斷修改 center 的數值來實現,具體實現過程原理以下圖所示:
部分實現代碼以下:
rotateStep() { // 即上圖輔助點 C let fromCenter = this.fromCenter; // 即上圖 B 點 let toCenter = this.toCenter; // 每幀轉一度 let rotateValue = this.rotateFrame || Math.PI / 180; // 輔助點 C 與 B 點之間創建一個方向向量 let centerVector = new ht.Math.Vector2(toCenter.x - fromCenter.x, toCenter.y - fromCenter.y); let centerVectorLength = centerVector.length(); // 此時旋轉百分比 let rotatePercent = rotateValue * this.stepNum / this.curRotateVal; if(rotatePercent >= 1) { rotatePercent = 1; this.stepNum = -2; } let newLength = rotatePercent * centerVectorLength; centerVector.setLength(newLength); let newCenterVector = centerVector.add(fromCenter); // 獲取旋轉過程當中 center 的點信息 let newCenterPosition = [newCenterVector.x, this.personHeight, newCenterVector.y]; // 設置當前 center 的大小 this.g3d.setCenter(newCenterPosition); }
經過上述代碼就實現了場景中的視角旋轉,而且能夠經過修改 rotateValue 的值控制旋轉的速度。
場景中電梯是一個 obj 模型,3D 模型是由最基礎的三角形面拼接合成,例如 1 個矩形能夠由 2 個三角形構成,1 個立方體由 6 個面即 12 個三角形構成,以此類推更復雜的模型能夠由許多的小三角形組合合成。所以 3D 模型定義即爲對構造模型的全部三角形的描述,而每一個三角形由三個頂點 vertex 構成,每一個頂點 vertex 由 x, y, z 三維空間座標決定,HT 中使用 vs 數組記錄構成三角面的全部頂點座標,因此若是想要讓電梯運行起來,只須要把全部的頂點座標往電梯運行的方向進行平移,如下爲部分關鍵僞代碼:
// vs 指的是構成電梯模型全部的三角面頂點座標數組 // 因爲場景中電梯的運行方向爲往對角線右上方運動,因此只須要修改 x 軸以及 y 軸座標值 // xStep yStep 爲每次電梯運動的距離 setInterval(() => { // i+3 是由於 vs 數組的順序爲 x, y, z 軸 因此每次 i 偏移三個單位大小 for(let i = 0, l = vs.length; i < l; i = i + 3) { // 該頂點座標下一個 x 軸座標的值 let nextX = vs[i] - xStep; // 該頂點座標下一個 y 軸座標的值 let nextY = vs[i + 1] + yStep; vs[i] = nextX < -0.5 ? 0.5 - (Math.abs(nextX) - 0.5) : nextX; vs[i + 1] = nextY > 0.5 ? -0.5 + (Math.abs(nextY) - 0.5) : nextY; } }, 200);
電梯運動動畫以下圖所示:
當點擊場景中的攝像頭以後右側頂部會顯示出當前攝像頭的監控畫面,如下爲實現效果圖:
煙霧報警會根據後臺實時傳遞過來的狀態值來變換當前煙霧報警模型的顏色,紅色爲報警狀態,如下爲實現效果圖:
平常地鐵站中會有專門的電視來展現下一班地鐵到站的時間表,該系統中也模擬該效果,不過該系統暫時作了電視的模型,時間暫無對接,如下爲效果圖:
3D 場景中交互是比較簡單的,主要是點擊攝像頭展現 2D 監控面板,在 2D 界面中主要是切換三種交互模式,三種交互模式爲互斥的關係,如下是 3D 交互註冊事件代碼:
g3d.mi((e) => { let {g2d, dm2d} = this; // 爲點擊類型 if(e.kind === 'clickData') { // data 爲當前點擊的圖元 let data = e.data; // 當前圖元的 shape3d 類型 let shape3d = data.s('shape3d'); // 判斷當前 shape3d 類型是否爲攝像頭 if(shape3d && shape3d.indexOf('攝像頭') > 0) { let cameraPanel = dm2d.getDataByTag('cameraPanel'); // toggle 切換攝像頭 2d 面板 g2d.isVisible(cameraPanel) ? cameraPanel.s('2d.visible', false) : cameraPanel.s('2d.visible', true); } } // 爲點擊 3d 場景背景類型 if(e.kind === 'clickBackground') { let cameraPanel = dm2d.getDataByTag('cameraPanel'); // 隱藏攝像頭 2d 面板 g2d.isVisible(cameraPanel) && cameraPanel.s('2d.visible', false); } });
工業互聯網將人,數據和機器鏈接起來,地鐵站 3D 可視化系統則是一個很好的展示,HT 的輕量化,數據的可視化,機器的可視化,資產的管理化幫助咱們更好的監控。而物聯網將經過各類信息傳感設備,實時採集任何須要監控、鏈接、互動的物體或過程等各類須要的信息,經過與 HT 的結合更好的展示出可視化的優點,固然地鐵站還能夠與 VR 進行結合,在各地科技展會中咱們能夠見到各類 VR 場景操做,HT 中也能夠結合 VR 設備進行操做,能夠戴上設備在地鐵站中漫遊,讓人有身臨其境的感受,因爲場景自己的輕量化,因此 VR 場景下的流暢性也是十分的高,讓用戶不會有頭暈的感受。固然系統自己也能夠在移動端運行,如下爲移動端運行截圖:
程序運行截圖: