基於 HTML5 的 WebGL 樓宇自控 3D 可視化監控

前言

智慧樓宇和人們的生活息息相關,樓宇智能化程度的提高,會極大程度的改善人們的生活品質,在當前工業互聯網大背景下受到很大關注。目前智慧樓宇可視化監控的主要優點包括:

  • 智慧化 -- 智慧樓宇是一個生態系統,像人一樣擁有感知能力、自我判斷能力以及控制能力。
  • 綠色化 -- 綠色建築在耗能、產能以及能源管理方面實現綠色化,樓宇安防實現綠色化監控。
  • 運行成本可控制 -- 基於可持續發展的要求,現代建築、商業建築需運行50年以上,建築在運行過程中能源消耗巨大,如何降低運營成本降低,使建築在低碳、環保的狀態下健康運行。

傳統的 智慧樓宇/樓宇自動化/樓宇安防/智慧園區 常會採用 BIM(建築信息模型 Building information modeling)軟件,如 Autodesk 的 Revit 或 Bentley 這類建築和工程軟件,但這些 BIM 建模模型的數據往往過於龐大臃腫,絕大部分細節信息對樓宇自控意義不大,反而影響拖累了行業 Web SCADA 或 Web 組態監控的趨勢,所以我們採用以 Hightopo 的 HT for Web 產品輕量化 HTML5/WebGL 建模的方案,實現快速建模、運行時輕量化到甚至手機終端瀏覽器即可 3D 可視化運維的良好效果。

本篇文章通過對智能建築的建模,頁面動畫效果的實現,以及頁面主要功能點進行闡述,幫助我們瞭解如何使用 HT 實現一個簡單的智慧樓宇可視化監控,以及幫助我們瞭解智慧樓宇,樓宇自動化的優勢。

預覽地址:基於 HTML5 的 WebGL 樓宇自控 3D 可視化監控 http://www.hightopo.com/demo/...

界面簡介及效果預覽

界面通過 2d 圖紙疊加在 3d 場景上來實現 2d 界面 與 3d 場景的融合,2d 界面通過自動佈局的機制實現了手機端與電腦端的響應式呈現。

界面初始化效果

界面初始化過程中的動畫包括地面路徑的實時渲染,樓層的展開,樓層的輝光掃描,樓層報警點動態水波,樓層監測數據面板的實時變化等等。

監控界面效果

監控界面包括

  1. 人員進入大廈的實時監控,面板中動態刷新人員進入大廈的頭像以及當前大廈人數等實時信息。
  2. 大廈電梯運行情況實時監控,系統中展示電梯當前的運行位置以及是否在運行等信息。
  3. 大廈某個具體樓層監控數據的實時監控,通過柱狀圖的形式展示了當前樓層具體信息的大小。
  4. 大廈管道的實時監控,展示了當前智能建築所有管道的運行情況。

智能建築建模

該 3d 場景中所有的模型均爲線段和六面體搭建,相比較通過 3d Max 建模然後通過 obj 導入來說場景中的三角面會少很多,更加的輕量化,例如場景中建築的樓層,通過 ht.Shape 類繪製,該類中記錄着樓層 points 點的信息以及 segmentsht.List 類型的線段數組信息,segments 代表着點的連接方式,用於告訴 ht.Shape 利用點的信息來繪製二次貝塞爾曲線或者三次貝塞爾曲線或者直線等信息,相關具體說明請參考 HT for Web 的形狀手冊,以下爲繪製單層的截圖:

通過上圖可以知道構建完一層模型之後其它幾層模型均爲相同的,只是 y 軸的數值不同,通過疊加幾層之後便可形成一幢大樓的輪廓。如果用戶需要搭建智慧園區,智慧樓宇等場景,完全可以使用這種基於 HTML5/WebGL 建模的方案,減少考慮使用 BIM 建模模型。場景中的管道以及背景地圖路線都爲點連線之後構成,只是通過修改線的顏色粗細或者進行貼圖來修改線或者面的樣式,場景中的電梯爲一個顏色爲黃色的簡單六面體,電梯線也爲一條線段而已,所以場景中的模型都是輕量化的建模,使 3d 場景運行渲染的更加流暢,提升用戶體驗。

場景關鍵動畫代碼分析

1. 地圖路線動畫代碼分析

通過上述智能建築建模的分析我們可以知道線路都是爲點與點之間進行連線而生成,所以當我們繪製完地圖的路徑之後可以得到所有點的信息,假如直線 AB 爲地圖中的某一條線段,那麼我們可以知道點 A 以及點 B 的點的座標,之後我們可以計算 AB 線段上任意一點 C 的點的座標,然後通過連接 A 點與 C 點來形成一條與 AB 線段位置方向相同但是大小比 AB 線段短的線,直到 AC 線段的長度等於 AB 線段長度之後再進行下一條路徑動畫的繪製,以下爲關鍵僞代碼展示:

 // currentIndex 爲當前路徑繪製到的點的索引 
 // points 爲當前路徑所有點的信息  currentPoints 爲繪製過程中點的信息 
 // segments 爲當前路徑所有點的連接方式信息 currentSegments 爲繪製過程中點的連接方式信息
 
 // 即上述此時 A 點信息
let fromPoint = points[currentIndex]; 
 // 即上述此時 B 點信息
let toPoint = points[currentIndex + 1]; 
 // 通過 AB 兩點信息組成一條 AB 方向的向量
let pointVector = new ht.Math.Vector2(toPoint.x - fromPoint.x, toPoint.y - fromPoint.y); 
 // 記錄該向量的長度,用於判斷 AC 是否大於等於 AB
let pointVectorLength = pointVector.length(); 

let currentPoints = [], currentSegments = [];

for(let i = 0; i < currentIndex + 1; i++) {
    currentPoints.push({
      x: points[i].x,
      y: points[i].y
    });
    currentSegments.push(segments[i]);
}

通過上述代碼可以知道我們獲取到了 currentPoints 以及 currentSegments 的信息了,之後便要計算在 fromPoint(A點) 與 toPoint(B點) 連線上點的座標,即 C 點,以下爲計算 C 點的關鍵僞代碼:

 // addLength 爲每次增加的線段長度值,該程序中使用 500 即每次長度增加 500
let nextVectorLength = currentVectorLength + addLength, 
    tempPoint;
    
roadData.currentVectorLength = nextVectorLength;

 // 判斷 AC 線段的長度是否大於 AB 
if(nextVectorLength > pointVectorLength) {
    nextVectorLength = pointVectorLength;
    roadData.currentVectorLength = 0;
    roadData.currentIndex = currentIndex + 1;
}

pointVector.setLength(nextVectorLength);

 // 即爲 C 點座標
tempPoint = {x: pointVector.x + fromPoint.x, y: pointVector.y + fromPoint.y}; 
 // 往 currentPoints 添加 C 點座標
currentPoints.push(tempPoint); 
 // 往 currentSegments 添加 C 點連接方式,此程序中都爲直線連接,所以值都爲 2
currentSegments.push(2); 
 // roadNode 即爲 ht.Shape 類 重新設置 ht.Shape 類點的信息
roadNode.setPoints(currentPoints);  
 // 重新設置 ht.Shape 類點的連接信息
roadNode.setSegments(currentSegments); 

以下爲動畫代碼執行流程圖:

以下爲繪製一條路線動畫的截圖:

程序中通過向量的計算方式來不斷獲取 C 點的座標,當然也可以用其它方式來計算 C 點的座標。

2. 水波以及掃描動畫代碼分析

水波以及掃描動畫都是通過 HT 提供的修改圖標矩形框信息 api 來進行控制,通過調度的方式不斷修改圖標矩形框大小來產生水波以及掃描的動畫效果,調度的具體用法可以參考 HT for Web 的調度手冊,以下爲水波動畫的關鍵僞代碼:

 // waterWaveNodes 所有水波節點
let waterWaveTask = {
    interval: 100, // 指每隔 100 ms 調用 action 函數一次
    action: function(data){
        // 判斷 waterWaveNodes 是包含 data
        if(waterWaveNodes.indexOf(data) > -1) { 
            // 獲取此時圖標矩形框信息 circleRect 是個長度爲 4 的數組 分別表示 x, y, width, height
            let circleRect = data.a('circleRect'); 
   
            if(circleRect) {
                // 通過修改高度來變大水波大小
                let nextHeight = circleRect[3] + 10; 
                
                // 高度最大值爲 250 
                if(nextHeight < 250) { 
                     // 對應修改 y 的大小,y 的增加大小爲高度的一半
                    circleRect[1] = circleRect[1] - 5;
                    circleRect[3] = nextHeight;
                    data.a('circleRect', circleRect);
                    data.a('borderColor', 'rgba(255, 133, 133, ' + (1 - circleRect[3] / 250) + ')');
                }
                else {
                    data.a('circleRect', [-0.5,128,257,0]);
                    data.a('borderColor', 'rgba(255, 133, 133)');
                }
                
            }
            else {
                data.a('circleRect', [-0.5,128,257,0]);
            }
            
        }
    }    
};
dm3d.addScheduleTask(waterWaveTask); // 新增該調度任務

下圖爲水波在 2d 中的截圖:

3. 數字變化動畫代碼分析

從程序的截圖中可以看到在 2d 面板以及 3d 場景中都有數字在動態的變化,這部分主要通過數據綁定動態來修改數值的大小,關於數據綁定可以參考 HT for Web 的數據綁定手冊,也是通過調度來不斷修改數值的大小,程序中我封裝了產生隨機數的代碼,用於每次產生隨機數之後綁定到對應的節點上,以下爲修改 2d 面板上數字的變化僞代碼:

 // numNode(1-7) 爲 2d 面板中對應數字的節點
 // data.a('ht.value', number) 即爲動態修改 attr 上的 ht.value 信息,之後圖紙會自動更新新賦予的數值
 // getRandomValue 爲自己封裝的產生隨機數的方法
this.change2dNumTask = {
        interval: 1000,
        action: (data) => {
            if(data === numNode1 || data === numNode2) {
                data.a('ht.value', util.getRandomValue([500, 999], 5));
            }
            if(data === numNode3 || data === numNode4) {
                data.s('text', util.getRandomValue([0, 30], 2) + '%');
            }
            if(data === numNode5) {
                data.a('ht.value', util.getRandomValue([0, 99999], 5, 3));
            }
            if(data === numNode6) {
                data.a('ht.value', util.getRandomValue([0, 100], 2));
            }
            if(data === numNode7) {
                data.a('ht.value', util.getRandomValue([0, 100], 2));
            }
       }        
    };
dm2d.addScheduleTask(this.change2dNumTask); // 新增該調度任務

通過以上代碼可以知道修改數值是通過修改節點的 attr 以及 style 對象的某個屬性值來動態變化數值,當然在程序中 2d 面板可能還會隱藏,此時該調度任務就不需要執行,可以調用 removeScheduleTask 方法來移除此調度任務。

4. 柱狀圖高度動畫代碼分析

在 3d 場景中柱狀體也是一個六面體,只是四周用了漸變的貼圖,以及頂面用了一張純色的貼圖構造出來,每個六面體都有高度的信息,HT 中通過 node.getTall() 來獲取當前六面體的高度值,根據上一節講的數據綁定,我們可以在展示柱狀圖的時候循環獲取所有柱狀體節點的高度值大小假如命名爲 tall,之後通過 node.a('tall', tall) 將該值存儲到當前柱狀圖節點的 attr 對象上面,之後在柱狀體初始化的時候可以不斷修改高度值來動態改變高度,當高度值大於 node.a('tall') 則說明當前柱狀體初始化的高度已經完成。以下爲相關的僞代碼:

charts.forEach((chart) => {
    !chart.a('tall') && chart.a('tall', chart.getTall()); // 將高度存儲到 attr 上
    chart.setTall(0); // 設置初始高度爲 0
});
this.chartAnimteTask = {
        interval: 25,
        action: function(data){
            if(charts.indexOf(data) > -1) {
                if(finishNum !== chartLength) {
                    if(data.getTall() !== data.a('tall')) {
                        let nextTall = data.getTall() + deep; // deep 爲每次增加的高度
                        let tall = data.a('tall'); // 獲取初始化高度 
                        // 判斷下一個高度是否大於初始化高度
                        if(nextTall < tall) {
                            data.setTall(nextTall);
                        }
                        else {
                            data.setTall(tall);
                            finishNum++;
                        }
                    }    
                }
            }
        }    
};
dm3d.addScheduleTask(this.chartAnimteTask); // 新增該調度任務

通過上面代碼可以知道動畫每一步的程序執行也是通過調度來完成的,與前文幾個動畫的實現方式類似。

5. 3d 鏡頭推進代碼分析

3d 場景中視野的推進後退都是通過 HT api 提供的修改 eye 以及 center 的數值方法來實現,通過不斷調用 setEye 以及 setCenter 方法來達到修改視角的目的,eye 類比人眼睛所處的位置,center 類比人眼睛聚焦的位置,以下爲實現鏡頭推進關鍵的僞代碼:

let e = ht.Default.clone(g3d.getEye()), // 獲取當前眼睛的位置
    c = ht.Default.clone(g3d.getCenter()); // 獲取當前眼睛聚焦的位置
 // eye 爲需要修改的對應 eye 值
 // center 爲需要修改的對應 center 值
 // 以下爲分別獲取 eye 與 center 在 xyz 三個座標軸之間的差值
    let edx = eye[0] - e[0],
        edz = eye[1] - e[1],
        edy = eye[2] - e[2],
        cdx = center[0] - c[0],
        cdz = center[1] - c[1],
        cdy = center[2] - c[2];
    // 開啓不斷修改 eye 與 center 的動畫
    ht.Default.startAnim({
        duration: time ? time : 3000,
        easing: function(t){ return t; },
        finishFunc: function() {
            if(typeof cb === 'function') {
                cb();
            }
        },
        action: function (v) {
            // v 爲從 0-1 變換的值 
            g3d.setEye([
                e[0] + edx * v,
                e[1] + edz * v,
                e[2] + edy * v
            ]);
            g3d.setCenter([
                c[0] + cdx * v,
                c[1] + cdz * v,
                c[2] + cdy * v
            ]);
        }
});

通過以上代碼可以知道通過修改 eye 與 center 分別對應的 xyz 軸的值與當前 e 與 c 分別對應的 xyz 軸的值之間的距離來達到視角的變化。

以下爲該代碼的一個應用截圖:

總結

物聯網通過各種信息傳感設備,實時採集任何需要監控、連接、互動的物體或過程等各種需要的信息,與互聯網結合形成的一個巨大網絡。實現了物與物、物與人,所有的物品與網絡的連接,方便識別、管理和控制。所以物聯網帶給我們的智慧樓宇的可視化監控需要監控的方面可能還有很多,該系統中針對人員出入,設備信息,管道信息等的監控實現了一個簡單的智慧樓宇監控系統,物聯網也將用戶端延伸和擴展到了任何物品與物品之間,讓我們更加了解搭建智慧園區,智慧校園等場景監控之後設備可視化,資產可視化帶給我們的直觀性。場景中的反光與景深等效果都是 HT 核心包提供的效果,所有的模型搭建與動畫也都是通過 HT 核心包提供的 api 進行建模與動畫驅動,所以在網頁中展示會十分流暢,大大提高了用戶的體驗,並且在移動端表現也十分友好。

以下爲移動端的程序運行截圖:

程序運行截圖:

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章