前言html
智慧樓宇和人們的生活息息相關,樓宇智能化程度的提升,會極大程度的改善人們的生活品質,在當前工業互聯網大背景下受到很大關注。目前智慧樓宇可視化監控的主要優勢包括:node
傳統的 智慧樓宇/樓宇自動化/樓宇安防/智慧園區 常會採用 BIM(建築信息模型 Building information modeling)軟件,如 Autodesk 的 Revit 或 Bentley 這類建築和工程軟件,但這些 BIM 建模模型的數據每每過於龐大臃腫,絕大部分細節信息對樓宇自控意義不大,反而影響拖累了行業 Web SCADA 或 Web 組態監控的趨勢,因此咱們採用以 Hightopo 的 HT for Web 產品輕量化 HTML5/WebGL 建模的方案,實現快速建模、運行時輕量化到甚至手機終端瀏覽器便可 3D 可視化運維的良好效果。api
本篇文章經過對智能建築的建模,頁面動畫效果的實現,以及頁面主要功能點進行闡述,幫助咱們瞭解如何使用 HT 實現一個簡單的智慧樓宇可視化監控,以及幫助咱們瞭解智慧樓宇,樓宇自動化的優點。數組
預覽地址:基於 HTML5 的 WebGL 樓宇自控 3D 可視化監控 http://www.hightopo.com/demo/ht-smart-building/瀏覽器
界面簡介及效果預覽網絡
界面經過 2d 圖紙疊加在 3d 場景上來實現 2d 界面 與 3d 場景的融合,2d 界面經過自動佈局的機制實現了手機端與電腦端的響應式呈現。運維
界面初始化效果dom
界面初始化過程當中的動畫包括地面路徑的實時渲染,樓層的展開,樓層的輝光掃描,樓層報警點動態水波,樓層監測數據面板的實時變化等等。ide
監控界面效果函數
監控界面包括:
智能建築建模
該 3d 場景中全部的模型均爲線段和六面體搭建,相比較經過 3d Max 建模而後經過 obj 導入來講場景中的三角面會少不少,更加的輕量化,例如場景中建築的樓層,經過 ht.Shape 類繪製,該類中記錄着樓層 points 點的信息以及 segments 爲 ht.List 類型的線段數組信息,segments 表明着點的鏈接方式,用於告訴 ht.Shape 利用點的信息來繪製二次貝塞爾曲線或者三次貝塞爾曲線或者直線等信息,相關具體說明請參考 HT for Web 的[形狀手冊],如下爲繪製單層的截圖:
經過上圖能夠知道構建完一層模型以後其它幾層模型均爲相同的,只是 y 軸的數值不一樣,經過疊加幾層以後即可造成一幢大樓的輪廓。若是用戶須要搭建智慧園區,智慧樓宇等場景,徹底可使用這種基於 HTML5/WebGL 建模的方案,減小考慮使用 BIM 建模模型。場景中的管道以及背景地圖路線都爲點連線以後構成,只是經過修改線的顏色粗細或者進行貼圖來修改線或者面的樣式,場景中的電梯爲一個顏色爲黃色的簡單六面體,電梯線也爲一條線段而已,因此場景中的模型都是輕量化的建模,使 3d 場景運行渲染的更加流暢,提高用戶體驗。
場景關鍵動畫代碼分析
1. 地圖路線動畫代碼分析
經過上述智能建築建模的分析咱們能夠知道線路都是爲點與點之間進行連線而生成,因此當咱們繪製完地圖的路徑以後能夠獲得全部點的信息,假如直線 AB 爲地圖中的某一條線段,那麼咱們能夠知道點 A 以及點 B 的點的座標,以後咱們能夠計算 AB 線段上任意一點 C 的點的座標,而後經過鏈接 A 點與 C 點來造成一條與 AB 線段位置方向相同可是大小比 AB 線段短的線,直到 AC 線段的長度等於 AB 線段長度以後再進行下一條路徑動畫的繪製,如下爲關鍵僞代碼展現:
1 // currentIndex 爲當前路徑繪製到的點的索引
2 // points 爲當前路徑全部點的信息 currentPoints 爲繪製過程當中點的信息
3 // segments 爲當前路徑全部點的鏈接方式信息 currentSegments 爲繪製過程當中點的鏈接方式信息
4
5 // 即上述此時 A 點信息
6 let fromPoint = points[currentIndex];
7 // 即上述此時 B 點信息
8 let toPoint = points[currentIndex + 1]; 9 // 經過 AB 兩點信息組成一條 AB 方向的向量 10 let pointVector = new ht.Math.Vector2(toPoint.x - fromPoint.x, toPoint.y - fromPoint.y); 11 // 記錄該向量的長度,用於判斷 AC 是否大於等於 AB 12 let pointVectorLength = pointVector.length(); 13 14 let currentPoints = [], currentSegments = []; 15 16 for(let i = 0; i < currentIndex + 1; i++) { 17 currentPoints.push({ 18 x: points[i].x, 19 y: points[i].y 20 }); 21 currentSegments.push(segments[i]); 22 }
經過上述代碼能夠知道咱們獲取到了 currentPoints 以及 currentSegments 的信息了,以後便要計算在 fromPoint(A點) 與 toPoint(B點) 連線上點的座標,即 C 點,如下爲計算 C 點的關鍵僞代碼:
1 // addLength 爲每次增長的線段長度值,該程序中使用 500 即每次長度增長 500
2 let nextVectorLength = currentVectorLength + addLength,
3 tempPoint; 4 5 roadData.currentVectorLength = nextVectorLength; 6 7 // 判斷 AC 線段的長度是否大於 AB 8 if(nextVectorLength > pointVectorLength) { 9 nextVectorLength = pointVectorLength; 10 roadData.currentVectorLength = 0; 11 roadData.currentIndex = currentIndex + 1; 12 } 13 14 pointVector.setLength(nextVectorLength); 15 16 // 即爲 C 點座標 17 tempPoint = {x: pointVector.x + fromPoint.x, y: pointVector.y + fromPoint.y}; 18 // 往 currentPoints 添加 C 點座標 19 currentPoints.push(tempPoint); 20 // 往 currentSegments 添加 C 點鏈接方式,此程序中都爲直線鏈接,因此值都爲 2 21 currentSegments.push(2); 22 // roadNode 即爲 ht.Shape 類 從新設置 ht.Shape 類點的信息 23 roadNode.setPoints(currentPoints); 24 // 從新設置 ht.Shape 類點的鏈接信息 25 roadNode.setSegments(currentSegments);
如下爲動畫代碼執行流程圖
如下爲繪製一條路線動畫的截圖:
程序中經過向量的計算方式來不斷獲取 C 點的座標,固然也能夠用其它方式來計算 C 點的座標。
2. 水波以及掃描動畫代碼分析
水波以及掃描動畫都是經過 HT 提供的修改圖標矩形框信息 api 來進行控制,經過調度的方式不斷修改圖標矩形框大小來產生水波以及掃描的動畫效果,調度的具體用法能夠參考 HT for Web 的[調度手冊],如下爲水波動畫的關鍵僞代碼:
1 // waterWaveNodes 全部水波節點
2 let waterWaveTask = {
3 interval: 100, // 指每隔 100 ms 調用 action 函數一次
4 action: function(data){ 5 // 判斷 waterWaveNodes 是包含 data 6 if(waterWaveNodes.indexOf(data) > -1) { 7 // 獲取此時圖標矩形框信息 circleRect 是個長度爲 4 的數組 分別表示 x, y, width, height 8 let circleRect = data.a('circleRect'); 9 10 if(circleRect) { 11 // 經過修改高度來變大水波大小 12 let nextHeight = circleRect[3] + 10; 13 14 // 高度最大值爲 250 15 if(nextHeight < 250) { 16 // 對應修改 y 的大小,y 的增長大小爲高度的一半 17 circleRect[1] = circleRect[1] - 5; 18 circleRect[3] = nextHeight; 19 data.a('circleRect', circleRect); 20 data.a('borderColor', 'rgba(255, 133, 133, ' + (1 - circleRect[3] / 250) + ')'); 21 } 22 else { 23 data.a('circleRect', [-0.5,128,257,0]); 24 data.a('borderColor', 'rgba(255, 133, 133)'); 25 } 26 27 } 28 else { 29 data.a('circleRect', [-0.5,128,257,0]); 30 } 31 32 } 33 } 34 }; 35 dm3d.addScheduleTask(waterWaveTask); // 新增該調度任務
下圖爲水波在 2d 中的截圖:
3. 數字變化動畫代碼分析
從程序的截圖中能夠看到在 2d 面板以及 3d 場景中都有數字在動態的變化,這部分主要經過數據綁定動態來修改數值的大小,關於數據綁定能夠參考 HT for Web 的[數據綁定手冊],也是經過調度來不斷修改數值的大小,程序中我封裝了產生隨機數的代碼,用於每次產生隨機數以後綁定到對應的節點上,如下爲修改 2d 面板上數字的變化僞代碼:
1 // numNode(1-7) 爲 2d 面板中對應數字的節點
2 // data.a('ht.value', number) 即爲動態修改 attr 上的 ht.value 信息,以後圖紙會自動更新新賦予的數值
3 // getRandomValue 爲本身封裝的產生隨機數的方法
4 this.change2dNumTask = {
5 interval: 1000, 6 action: (data) => { 7 if(data === numNode1 || data === numNode2) { 8 data.a('ht.value', util.getRandomValue([500, 999], 5)); 9 } 10 if(data === numNode3 || data === numNode4) { 11 data.s('text', util.getRandomValue([0, 30], 2) + '%'); 12 } 13 if(data === numNode5) { 14 data.a('ht.value', util.getRandomValue([0, 99999], 5, 3)); 15 } 16 if(data === numNode6) { 17 data.a('ht.value', util.getRandomValue([0, 100], 2)); 18 } 19 if(data === numNode7) { 20 data.a('ht.value', util.getRandomValue([0, 100], 2)); 21 } 22 } 23 }; 24 dm2d.addScheduleTask(this.change2dNumTask); // 新增該調度任務
經過以上代碼能夠知道修改數值是經過修改節點的 attr 以及 style 對象的某個屬性值來動態變化數值,固然在程序中 2d 面板可能還會隱藏,此時該調度任務就不須要執行,能夠調用 removeScheduleTask 方法來移除此調度任務。
4. 柱狀圖高度動畫代碼分析
在 3d 場景中柱狀體也是一個六面體,只是四周用了漸變的貼圖,以及頂面用了一張純色的貼圖構造出來,每一個六面體都有高度的信息,HT 中經過 node.getTall() 來獲取當前六面體的高度值,根據上一節講的數據綁定,咱們能夠在展現柱狀圖的時候循環獲取全部柱狀體節點的高度值大小假如命名爲 tall,以後經過 node.a('tall', tall) 將該值存儲到當前柱狀圖節點的 attr 對象上面,以後在柱狀體初始化的時候能夠不斷修改高度值來動態改變高度,當高度值大於 node.a('tall') 則說明當前柱狀體初始化的高度已經完成。如下爲相關的僞代碼:
1 charts.forEach((chart) => {
2 !chart.a('tall') && chart.a('tall', chart.getTall()); // 將高度存儲到 attr 上
3 chart.setTall(0); // 設置初始高度爲 0
4 }); 5 this.chartAnimteTask = { 6 interval: 25, 7 action: function(data){ 8 if(charts.indexOf(data) > -1) { 9 if(finishNum !== chartLength) { 10 if(data.getTall() !== data.a('tall')) { 11 let nextTall = data.getTall() + deep; // deep 爲每次增長的高度 12 let tall = data.a('tall'); // 獲取初始化高度 13 // 判斷下一個高度是否大於初始化高度 14 if(nextTall < tall) { 15 data.setTall(nextTall); 16 } 17 else { 18 data.setTall(tall); 19 finishNum++; 20 } 21 } 22 } 23 } 24 } 25 }; 26 dm3d.addScheduleTask(this.chartAnimteTask); // 新增該調度任務
經過上面代碼能夠知道動畫每一步的程序執行也是經過調度來完成的,與前文幾個動畫的實現方式相似。
5. 3d 鏡頭推動代碼分析
3d 場景中視野的推動後退都是經過 HT api 提供的修改 eye 以及 center 的數值方法來實現,經過不斷調用 setEye 以及 setCenter 方法來達到修改視角的目的,eye 類比人眼睛所處的位置,center 類比人眼睛聚焦的位置,如下爲實現鏡頭推動關鍵的僞代碼:
1 let e = ht.Default.clone(g3d.getEye()), // 獲取當前眼睛的位置
2 c = ht.Default.clone(g3d.getCenter()); // 獲取當前眼睛聚焦的位置
3 // eye 爲須要修改的對應 eye 值
4 // center 爲須要修改的對應 center 值
5 // 如下爲分別獲取 eye 與 center 在 xyz 三個座標軸之間的差值
6 let edx = eye[0] - e[0],
7 edz = eye[1] - e[1], 8 edy = eye[2] - e[2], 9 cdx = center[0] - c[0], 10 cdz = center[1] - c[1], 11 cdy = center[2] - c[2]; 12 // 開啓不斷修改 eye 與 center 的動畫 13 ht.Default.startAnim({ 14 duration: time ? time : 3000, 15 easing: function(t){ return t; }, 16 finishFunc: function() { 17 if(typeof cb === 'function') { 18 cb(); 19 } 20 }, 21 action: function (v) { 22 // v 爲從 0-1 變換的值 23 g3d.setEye([ 24 e[0] + edx * v, 25 e[1] + edz * v, 26 e[2] + edy * v 27 ]); 28 g3d.setCenter([ 29 c[0] + cdx * v, 30 c[1] + cdz * v, 31 c[2] + cdy * v 32 ]); 33 } 34 });
經過以上代碼能夠知道經過修改 eye 與 center 分別對應的 xyz 軸的值與當前 e 與 c 分別對應的 xyz 軸的值之間的距離來達到視角的變化。
如下爲該代碼的一個應用截圖:
總結
物聯網經過各類信息傳感設備,實時採集任何須要監控、鏈接、互動的物體或過程等各類須要的信息,與互聯網結合造成的一個巨大網絡。實現了物與物、物與人,全部的物品與網絡的鏈接,方便識別、管理和控制。因此物聯網帶給咱們的智慧樓宇的可視化監控須要監控的方面可能還有不少,該系統中針對人員出入,設備信息,管道信息等的監控實現了一個簡單的智慧樓宇監控系統,物聯網也將用戶端延伸和擴展到了任何物品與物品之間,讓咱們更加了解搭建智慧園區,智慧校園等場景監控以後設備可視化,資產可視化帶給咱們的直觀性。場景中的反光與景深等效果都是 HT 核心包提供的效果,全部的模型搭建與動畫也都是經過 HT 核心包提供的 api 進行建模與動畫驅動,因此在網頁中展現會十分流暢,大大提升了用戶的體驗,而且在移動端表現也十分友好。
如下爲移動端的程序運行截圖:
程序運行截圖: