基於HTML5 WebGL實現的工業隧道監控

前言

監控隧道內的車道堵塞狀況、隧道內的車禍現場,在隧道中顯示當前車禍位置並在隧道口給與提示等等功能都是很是有必要的。這個隧道 Demo 的主要內容包括:照明、風機、車道指示燈、交通訊號燈、情報板、消防、火災報警、車行橫洞、風向儀、微波車檢、隧道緊急逃生出口的控制以及事故模擬等等。html

效果圖

http://www.hightopo.com/demo/tunnel2/index.htmlnode

上圖中的各類設備均可以雙擊,此時 camera 的位置會從當前位置移動到雙擊的設備的正前方;隧道入口的展現牌會自動輪播,出現事故時會展現牌中的內容會由「限速80,請開車燈」變爲「超車道兩車追尾,請減速慢行」;兩隧道中間的逃生通道上方的指示牌是能夠點擊的,點擊切換爲藍綠色激活狀態,兩旁的逃生通道門也會打開,再單擊指示牌變爲灰色,門關閉;還有一個事故現場模擬,雙擊兩旁變壓器中其中一個,在隧道內會出現一個「事故現場圖標」,單擊此圖標,出現彈出框顯示事故等等等等。json

代碼實現

場景搭建

整個隧道都是基於 3D 場景上繪製的,先來看看怎麼搭建 3D 場景:數組

dm = new ht.DataModel();//數據容器
g3d = new ht.graph3d.Graph3dView(dm);// 3d 場景
g3d.addToDOM();//將場景添加到 body 中

上面代碼中的 addToDOM 函數,是一個將組件添加到 body 體中的函數的封裝,定義以下:瀏覽器

addToDOM = function(){
    var self = this,
         view = self.getView(),//獲取組件的底層 div
         style = view.style;
    document.body.appendChild(view);//將組件底層div添加進body中
    style.left = '0';//ht 默認將全部的組件的position都設置爲absolute絕對定位
    style.right = '0';
    style.top = '0';
    style.bottom = '0';
    window.addEventListener('resize', function () { self.iv(); }, false);//窗口大小改變事件,調用刷新函數
}

JSON 反序列化

整個場景是由名爲 隧道1.json 的文件導出而成的,我只須要用代碼將 json 文件中的內容轉換爲我須要的部分便可:緩存

ht.Default.xhrLoad('./scenes/隧道1.json', function(text) {//xhrLoad 函數是一個異步加載文件的函數
    var json = ht.Default.parse(text);//將 json 文件中的文本轉爲咱們須要的 json 格式的內容
    dm.deserialize(json);//反序列化數據容器,解析用於生成對應的Data對象並添加到數據容器 這裏至關於把 json 文件中生成的 ht.Node 節點反序列化到數據容器中,這樣數據容器中就有這個節點了
});

因爲 xhrLoad 函數是一個異步加載函數,因此若是 dm 數據容器反序列化未完成就直接調用了其中的節點,那麼會形成數據獲取不到的結果,因此通常來講我是將一些邏輯代碼寫在這個函數內部,或者給邏輯代碼設置 timeout 錯開時間差。app

首先,因爲數據都是存儲在 dm 數據容器中的(經過 dm.add(node) 添加的),因此咱們要獲取數據除了能夠經過 id、tag 等獨立的方式,還能夠經過遍歷數據容器來獲取多個元素。因爲這個場景比較複雜,模型的面也比較多,鑑於設備配置,我將能 Batch 批量的元素都進行了批量:異步

dm.each(function(data) {
    if (data.s('front.image') === 'assets/sos電話.png'){//對「電話」進行批量
        data.s('batch', 'sosBatch');
    }
    else if (data.s('all.color') === 'rgba(222,222,222,0.18)') {//逃生通道批量(透明度也會影響性能)
        data.s('batch', 'emergencyBatch');
    }
    else if (data.s('shape3d') === 'models/隧道/攝像頭.json' || data.s('shape3d') === 'models/隧道/橫洞.json' || data.s('shape3d') === 'models/隧道/捲簾門.json') {
        if(!data.s('shape3d.blend'))//個別攝像頭染色了 不作批量
            data.s('batch', 'basicBatch');//基礎批量什麼也不作
    }
    else if (data.s('shape3d') === 'models/大型變壓器/變壓器.json') {    
        data.s('batch', 'tileBatch');
        data.setToolTip('單擊漫遊,雙擊車禍地點出現圖標');
    }
    else if (data.getDisplayName() === '地面') {
        data.s('3d.selectable', false);//設置隧道「地面」不可選中
    }
    else if (data.s('shape3d') === 'models/隧道/排風.json') {
        data.s('batch', 'fanBatch');//排風扇的模型比較複雜,因此作批量
    }
    else if (data.getDisplayName() === 'arrow') {//隧道兩旁的箭頭路標
        if (data.getTag() === 'arrowLeft') data.s('shape3d.image', 'displays/abc.png');
        else data.s('shape3d.image', 'displays/abc2.png');
        data.s({
            'shape3d': 'billboard',
            'shape3d.image.cache': true,//緩存,設置了 cache 的代價是須要設置 invalidateShape3dCachedImage
            'shape3d.transparent': true //設置這個值,圖片上的鋸齒就不會太明顯了(若圖片類型爲 json,則設置 shape3d.dynamic.transparent)
        });
        g3d.invalidateShape3dCachedImage(data);
    }
    else if (data.getTag() === 'board' || data.getTag() === 'board1') {//隧道入口處的情報板
        data.a('textRect', [0, 2, 244, 46]); //業務屬性,用來控制文本的位置[x,y,width,height]
        data.a('limitText', '限速80,請開車燈');//業務屬性,設置文本內容
        var min = -245;
        var name = 'board' + data.getId();
        window[name] = setInterval(function() {
            circleFunc(data, window[name], min)//設置情報板中的文字向左滾動,而且當文字所有顯示時重複閃爍三次
        }, 100);
    }

    //給逃生通道上方的指示板 動態設置顏色
    var infos = ['人行橫洞1', '人行橫洞2', '人行橫洞3', '人行橫洞4', '車行橫洞1', '車行橫洞2', '車行橫洞3'];
    infos.forEach(function(info) {
        if(data.getDisplayName() === info) {
            data.a('emergencyColor', 'rgb(138, 138, 138)');
        }
    });

    infos = ['車道指示器', '車道指示器1', '車道指示器2', '車道指示器3'];
    infos.forEach(function(info) {
        if (data.getDisplayName() === info) {
            createBillboard(data, 'assets/車道信號-過.png', 'assets/車道信號-過.png', info)//考慮到性能問題 將六面體變換爲 billboard 類型元素
        }
    });
});

上面有一處設置了 tooltip 文字提示信息,在 3d 中,要顯示這個文字提示信息,就須要設置 g3d.enableToolTip() 函數,默認 3d 組件是關閉這個功能的。ide

邏輯代碼

情報板滾動條

我就直接按照上面代碼中提到的方法進行解釋,首先是 circleFunc 情報板文字循環移動的函數,在這個函數中咱們用到了業務屬性 limitText 設置情報板中的文字屬性以及 textRect 設置情報板中文字的移動位置屬性:函數

function circleFunc(data, timer, min) {//設置情報板中的文字向左滾動,而且當文字所有顯示時重複閃爍三次
    var text = data.a('limitText');//獲取當前業務屬性 limitText 的內容
    data.a('textRect', [data.a('textRect')[0]-5, 2, 244, 46]); //設置業務屬性 textRect 文本框的座標和大小
    if (parseInt(data.a('textRect')) <= parseInt(min)) {
        data.a('textRect', [255, 2, 244, 46]);
    }
    else if (data.a('textRect')[0] === 0) {
        clearInterval(timer);
        var index = 0;
        var testName = 'testTimer' + data.getId();//設置多個 timer 是由於可以進入這個函數中的不止一個 data,若是在同一時間多個 data 設置同一個 timer,那確定只會對最後一個節點進行動畫。後面還有不少這種陷阱,要注意
        window[testName] = setInterval(function() {
            index++;
            if(data.a('limitText') === '') {//若是情報板中文本內容爲空
                setTimeout(function() {
                    data.a('limitText', text);//設置爲傳入的 text 值
                }, 100);
            }
            else {
                setTimeout(function() {
                    data.a('limitText', ''); //若情報板中的文本內容不爲空,則設置爲空
                }, 100);
            }

            if(index === 11) { //重複三次 
                clearInterval(window[testName]);
                data.a('limitText', text);
            }
        }, 100);

        setTimeout(function() {
            timer = setInterval(function() {
                circleFunc(data, timer, min) //回調函數
            }, 100);
        }, 1500);
    }
}

因爲 WebGL 對瀏覽器的要求不低,爲了能儘可能多的適應各大瀏覽器,咱們將全部的「道路指示器」 ht.Node 類型的六面體所有換成 billboard 類型的節點,性能能提高很多。

http://www.hightopo.com

設置 billboard 的方法很簡單,獲取當前的六面體節點,而後給這些節點設置:

node.s({
    'shape3d': 'billboard',
    'shape3d.image': imageUrl,
    'shape3d.image.cache': true
});
g3d.invalidateShape3dCachedImage(node); //還記得用 shape3d.image.cache 的代價麼?

固然,由於 billboard 不能雙面顯示不一樣的圖片,只是一個「面」,因此咱們還得在這個節點的位置建立另外一個節點,在這個節點的「背面」顯示圖片,而且跟這個節點的配置如出一轍,不過位置要稍稍偏移一點。

Camera 緩慢偏移

其餘動畫部分比較簡單,我就不在這裏多說了,這裏有一個雙擊節點能將視線從當前 camera 位置移動到雙擊節點正前方的位置的動畫我提一下。我封裝了兩個函數 setEye 和 setCenter,分別用來設置 camera 的位置和目標位置的:

function setCenter(center, finish) {//設置「目標」位置
    var c = g3d.getCenter().slice(0), //獲取當前「目標」位置,爲一個數組,而 getCenter 數組會在視線移動的過程當中不斷變化,因此咱們先拷貝一份
        dx = center[0] - c[0], //當前x軸位置和目標位置的差值
        dy = center[1] - c[1],
        dz = center[2] - c[2];
    // 啓動 500 毫秒的動畫過分
    ht.Default.startAnim({
        duration: 500,
        action: function(v, t) {
            g3d.setCenter([ //將「目標」位置緩慢從當前位置移動到設置的位置處
                c[0] + dx * v,
                c[1] + dy * v,
                c[2] + dz * v
            ]);
        }
    });
};

function setEye(eye, finish) {//設置「眼睛」位置
    var e = g3d.getEye().slice(0),//獲取當前「眼睛」位置,爲一個數組,而 getEye 數組會在視線移動的過程當中不斷變化,因此咱們先拷貝一份
        dx = eye[0] - e[0],
        dy = eye[1] - e[1],
        dz = eye[2] - e[2];

    // 啓動 500 毫秒的動畫過分
    ht.Default.startAnim({
        duration: 500,
        action: function(v, t) {//將 Camera 位置緩慢地從當前位置移動到設置的位置
            g3d.setEye([
                e[0] + dx * v,
                e[1] + dy * v,
                e[2] + dz * v
            ]);
        }
    });
};

後期咱們要設置的時候就直接調用這兩個函數,並設置參數爲咱們目標的位置便可。好比我這個場景中的各個模型,因爲不一樣視角對應的各個模型的旋轉角度也不一樣,我只能找幾個比較有表明性的 0°,90°,180°以及360° 這四種比較典型的角度了。因此繪製 3D 場景的時候,我也儘可能設置節點的旋轉角度爲這四個中的一種(並且對於咱們這個場景來講,基本上只在 y 軸上旋轉了):

var p3 = e.data.p3(), //獲取事件對象的三維座標
    s3 = e.data.s3(),//獲取事件對象的三維尺寸
    r3 = e.data.r3();//獲取事件對象的三維旋轉值

setCenter(p3);//設置「目標」位置爲當前事件對象的三維座標值
if (r3[1] !== 0) {//若是節點的 y 軸旋轉值 不爲 0
    if (parseFloat(r3[1].toFixed(5)) === parseFloat(-3.14159)) { //浮點負數得作轉換才能進行比值
        setEye([p3[0], p3[1]+s3[1], p3[2] * Math.abs(r3[1]*2.3/6)]);//設置camera 的目標位置
    }
    else if (parseFloat(r3[1].toFixed(4)) === parseFloat(-1.5708)) {
        setEye([p3[0] * Math.abs(r3[1]/1.8), p3[1]+s3[1], p3[2]]);
    }
    else {
        setEye([p3[0] *r3[1], p3[1]+s3[1], p3[2]]);
    }
}
else {
    setEye([p3[0], p3[1]+s3[1]*2, p3[2]+1000]);
}

事故模擬現場

最後來講說模擬的事故現場吧,這段仍是比較接近實際項目的。操做流程以下:雙擊「變壓器」-->隧道中間某個部分會出現一個「事故現場」圖標-->單擊圖標,彈出對話框,顯示當前事故信息-->點擊肯定,則事故現場以前的燈都顯示爲紅色×,而且隧道入口的情報板上的文字顯示爲「超車道兩車追尾,請減速慢行」-->再雙擊一次「變壓器」,場景恢復事故以前的狀態。

在 HT 中,可經過 Graph3dView#addInteractorListener(簡寫爲 mi)來監聽交互過程:

g3d.addInteractorListener(function(e) {
    if(e.kind === 'doubleClickData') {
        if (e.data.getTag() === 'jam') return;//有「事故」圖標節點存在
        if (e.data.s('shape3d') === 'models/大型變壓器/變壓器.json') {//若是雙擊對象是變壓器
            index++;
            var jam = dm.getDataByTag('jam');//經過惟一標識tag標籤獲取「事故」圖標節點對象
            if(index === 1){
                var jam = dm.getDataByTag('jam');
                jam.s({
                    '3d.visible': true,//設置節點在 3d 上可見
                    'shape3d': 'billboard',//設置節點爲 billboard 類型
                    'shape3d.image': 'assets/車禍.png', //設置 billboard 的顯示圖片
                    'shape3d.image.cache': true,//設置 billboard 圖片是否緩存
                    'shape3d.autorotate': true,//是否始終面向鏡頭
                    'shape3d.fixSizeOnScreen': [30, 30],//默認保持圖片本來大小,設置爲數組模式則能夠設置圖片顯示在界面上的大小
                });
                g3d.invalidateShape3dCachedImage(jam);//cache 的代價是節點須要設置這個函數
             }
             else {
                 jam.s({
                     '3d.visible': false//第二次雙擊變壓器就將全部一切恢復「事故」以前的狀態
                });
                dm.each(function(data) {
                    var p3 = data.p3();
                    if ((p3[2] < jam.p3()[2]) && data.getDisplayName() === '車道指示器1') {
                        data.s('shape3d.image', 'assets/車道信號-過.png');
                    }
                    if(data.getTag() === 'board1') {
                        data.a('limitText', '限速80,請開車燈');
                    }
                });
                index = 0;
            }
                        
        }
    }
});

既然「事故」節點圖標出現了,接着點擊圖標出現「事故信息彈出框」,監聽事件一樣是在 mi(addInteractorListener)中,可是此次監聽的是單擊事件,咱們知道,監聽雙擊事件時會觸發一次單擊事件,爲了不這種狀況,我在單擊事件裏面作了延時:

else if (e.kind === 'clickData'){//點擊圖元
    timer = setTimeout(function() {
        clearTimeout(timer);
        if (e.data.getTag() === 'jam') {//若是是「事故」圖標節點
            createDialog(e.data);//建立一個對話框
        }
    }, 200);
}

在上面的雙擊事件中我沒有 clearTimeout,怕順序問題給你們形成困擾,要記得加一下。

彈出框以下:

這個彈出框是由兩個 ht.widget.FormPane 表單構成的,左邊的表單只有一行,行高爲 140,右邊的表單是由 5 行構成的,點擊肯定,則「事故」圖標節點以前的道路指示燈都換成紅色×的圖標:

function createForm4(node, dialog) {//彈出框右邊的表單
    var form = new ht.widget.FormPane();//表單組件
    form.setWidth(200);//設置表單組件的寬
    form.setHeight(200);//設置表單組件的高
    var view = form.getView();//獲取表單組件的底層 div 
    document.body.appendChild(view);//將表單組件添加到 body 中

    var infos = [
        '編輯框內容爲:2輛',
        '編輯框內容爲:客車-客車',
        '編輯框內容爲:無起火',
        '編輯框內容爲:超車道'
    ];
    infos.forEach(function(info) {
        form.addRow([ //向表單中添加行
            info
        ], [0.1]);//第二個參數爲行寬度,小於1的值爲相對值
    });
    
    form.addRow([
        {
            button: {//添加一行的「確認」按鈕
                label: '確認',
                onClicked: function() {//按鈕點擊事件觸發
                    dialog.hide();//隱藏對話框
                    dm.each(function(data) {
                        var p3 = data.p3();
                        if ((p3[2] < node.p3()[2]) && data.getDisplayName() === '車道指示器1') {//改變「車道指示器」的顯示圖片爲紅色×,這裏我是根據「事故」圖標節點的座標來判斷「車道顯示器」是在前仍是在後的
                            data.s('shape3d.image', 'assets/車道信號-禁止.png');
                        }
                        if(data.getTag() === 'board1') {//將隧道口的情報板上的文字替換
                            data.a('limitText', '超車道兩車追尾,請減速慢行');
                        }
                    });
                }
            }
        }
    ], [0.1]);
    return form;
}

結束語

這個工業隧道的 Demo 是我經過幾天不斷地完善完善而成的,可能仍是有不足的地方,可是整體來講我是挺滿意的了,可能以後還會繼續完善,也得靠你們不斷地給我意見和建議,我只但願在本身努力的同時也能夠幫助到別人。整個 Demo 中,我主要遇到了兩個問題,一個是我在代碼中提到過的設置 timer 的問題,多個節點若是同時用一個 timer,那就只有最後一個節點可以顯示出 timer 的效果;另外一個是 getEye 和 getCenter 的問題,這兩個值都是在不斷變化的,因此得先拷貝一份數據,再進行數據的變換。

相關文章
相關標籤/搜索