亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > HTML5 > 正文

HTML5 WebGL 實現民航客機飛行監控系統

2024-09-05 07:19:32
字體:
來源:轉載
供稿:網友

前言

前些日子出差,在飛機上看到頭頂的監控面板,除了播放電視劇和廣告之外,還會時不時的切換到一個飛機航行的監控系統,不過整個監控系統讓人感到有一點點的簡陋,所以我就突發奇想制作了一個采用 HT for Web 的升級版監控系統,demo 的效果還行,發出來大家相互學習下。


 

demo

實現過程

云中穿行效果

為了達到飛機云中穿行的效果,最開始我遇到的問題是飛機飛行的層次感,也就通常所說的透視效果,這里我采用的是云通道和云背景以不同的速度流動,制造一種飛行的透視效果。

云我采用的是貼圖的方式呈現的,但是僅僅是貼圖會遮擋天空和飛機,非常影響飛機飛行的觀感,所以我開啟了相應圖元的 transparent 和 opacity ,云背景和云通道設置不同的透明度,不僅增加了層次感,還會讓人產生云朵從眼前飄過的錯覺。

云通道采用的是 ht.Polyline 類型,通道縮放拉大了 Y 軸的比例,使云通道有更大的縱向空間,設置 reverse.flip 背拷貝使云通道內部也顯示出貼圖,仿佛讓飛機置身于云海中穿梭;云背景采用 ht.Node 類型,只設置一個面顯示充當云背景。

整體的云流動效果采用 offset 偏移實現,改變相應圖元或相應圖元面的貼圖偏移量來達到飛機云中穿行的效果, 代碼如下:
 

var i = 1,     p = 0;setInterval(() => {    i -= 0.1; p += 0.005;    clouds.s('shape3d.uv.offset', [i, 0]);    cloudBackground.s('all.uv.offset', [p, 0]);}, 100);

升降顛簸效果

雖然達到了飛機云中穿行的效果,但是如果飛機只是直直的飛行,那也會降低飛行的實感,相信坐過飛機的朋友肯定都遇到過因氣流產生的顛簸,也經常感受到飛機飛行途中的爬升和下降,這其實是因為飛機的航線并不是一直固定在一個高度上,有時會爬升有時會下降,所以我就用 ht-animation.js HT 動畫擴展插件去實現飛機顛簸效果,代碼如下:

dm.enableAnimation(20);plane.setAnimation({    back1: {        from: 0,        to: 160,        easing: 'Cubic.easeInOut',        duration: 8000,        next: "up1",        onUpdate: function (value) {            value = parseInt(value);            var p3 = this.p3();            this.p3(value, p3[1], p3[2]);        }    },    //...省略相似    start: ["back1"]});

球扇形視角限制

飛行效果完善之后,這時我就遇到了一個比較棘手的問題,因為實際上雖然看著飛機是在云海中穿梭,但是僅僅是在通道中飛行,背景其實也只是平面貼圖,所以當視角到達某種程度的時候就會有強烈的違和感和不真實感,就需要一個視角限制,使視角的調整剛剛好在一個范圍內。

視角限制的話一般是限制 g3d 的 eye 和 center ,不太了解的朋友可以去看 hightopo 官網中的 3d 手冊,里面有詳細的說明,這里我就不再贅述了;因為視角范圍的關系,所以我決定固定 center 的位置,代碼如下:
 

g3d.addPropertyChangeListener(e => {    // 固定中心點    if (e.property === 'center') {        e.newValue[0] = center[0];        e.newValue[1] = center[1];        e.newValue[2] = center[2];    }}

然后再把 eye 限制在某一個范圍內就大功告成了,然而這里卻并不是那么簡單,最開始我把 eye 限制在一個立方體的空間內,但交互效果很不理想,考慮到 g3d 默認交互中,鼠標拖拽平移視角變換時,實際上 eye 是在一個以 center 為球心的球面上運動的,所以我決定從這個球中挖出來一塊作為 eye 的限制空間,也就是球扇形,不太理解的朋友可以參考這個圖:


 

球扇形視角限制,一共需要三個參數,分別是中心參考軸、中心軸和外邊所成角度、所在球限制半徑,其中中心參考軸可根據初始 eye 和 center 的連接延長線確定,所在球限制半徑又分最大限制和最小限制,代碼如下:

 

function limitEye(g3d, eye, center, options) {    var limitMaxL   = options.limitMaxL,        limitMinL   = options.limitMinL,        limitA      = options.limitA;    g3d.addPropertyChangeListener(e => {        // 固定中心點        if (e.property === 'center') {            e.newValue[0] = center[0];            e.newValue[1] = center[1];            e.newValue[2] = center[2];        }        // 限制視角        if (e.property === 'eye') {            var newEyeV = new ht.Math.Vector3(e.newValue),                centerV = new ht.Math.Vector3(center),                refEyeV = new ht.Math.Vector3(eye),                refVector = refEyeV.clone().sub(centerV),                newVector = newEyeV.clone().sub(centerV);            if (centerV.distanceTo(newEyeV) > limitMaxL) {                newVector.setLength(limitMaxL);                e.newValue[0] = newVector.x;                e.newValue[1] = newVector.y;                e.newValue[2] = newVector.z;            }            if (centerV.distanceTo(newEyeV) < limitMinL) {                newVector.setLength(limitMinL);                e.newValue[0] = newVector.x;                e.newValue[1] = newVector.y;                e.newValue[2] = newVector.z;            }            if (newVector.angleTo(refVector) > limitA) {                var oldLength = newVector.length(),                    oldAngle  = newVector.angleTo(refVector),                    refLength = oldLength * Math.cos(oldAngle),                    vertVector,                    realVector,                    realEye;                refVector.setLength(refLength);                newEyeV = newVector.clone().add(centerV);                refEyeV = refVector.clone().add(centerV);                vertVector = newEyeV.clone().sub(refEyeV);                vertLength = refLength * Math.tan(limitA);                vertVector.setLength(vertLength);                realVector = vertVector.clone().add(refEyeV).sub(centerV);                realVector.setLength(oldLength);                realEye = realVector.clone().add(centerV);                // 防止移動角度大于 180 度,視角反轉                if (oldAngle > Math.PI / 2) {                    realEye.negate();                }                e.newValue[0] = realEye.x;                e.newValue[1] = realEye.y;                e.newValue[2] = realEye.z;            }          }    })}

飛機監控系統

當然作為監控系統,自然要有監控了,增加右下角的小地圖,并提供三種模式,分別是聚焦飛機,聚焦飛行軌跡和聚焦地圖,并根據飛機的飛行方向控制飛行軌跡的流動效果,其中聚焦飛機會跟隨飛機移動進行 fitData ,使飛機一直處于小地圖的中心,代碼如下:

 

var fitFlowP = function (e) {    if (e.property === 'position' && e.data === plane) {        mapGV.fitData(plane, false);    }};buttonP.s({    'interactive': true,    'onClick': function (event, data, view, point, width, height) {        map.a('fitDataTag', 'plane2D');        mapGV.fitData(plane, false);        mapDM.md(fitFlowP);    }});buttonL.s({    'interactive': true,    'onClick': function (event, data, view, point, width, height) {        mapDM.umd(fitFlowP);        map.a('fitDataTag', 'flyLine');        mapGV.fitData(flyLine, false);    }});// ...省略

增加鼠標移到飛機相應位置進行名稱的提示、雙擊后顯示飛機相應位置的信息面板并將視角聚焦到面板上、點擊飛機任意地方切換回飛機飛行模式等效果。


 

左側增加監控面板替代上面提到的雙擊相應位置這步操作直接聚焦到相應位置的信息面板上,這里按鈕開啟了交互并添加了相應的交互邏輯,代碼如下:

 

button_JC.s({    'interactive': true,    'onClick': function (event, data, view, point, width, height) {        event.preventDefault();        let g3d = G.g3d,            g3dDM = G.g3d.dm();        g3d.fireInteractorEvent({            kind: 'doubleClickData',            data: g3dDM.getDataByTag(data.getTag())        })    }});//...省略

天空渲染效果

既然是監控系統肯定是 24 小時無差別的監控,這就涉及到一個問題,我總不可能半夜的時候飛機也從瓦藍瓦藍的天空上飛過,這就很欠缺真實性了,所以要有一個天空從亮到暗再從暗到亮的過程,這個過程我暫定到 06:00-06:30 和19:00-19:30 這兩個時間段。

天空采用的是 shape3d : 'sphere' 球形,包裹整個場景,然后使用 reverse.flip 背拷貝 和 blend 染色,之后天空就可以渲染成我想要的顏色,如果按照時間改變天空明暗只要改變染色值就可以了。

但是由于白天和晚上光照情況的不同,云反射光的強度也不同,就導致了白天和晚上云的差異,所以也要調整云道和云背景的貼圖的 opacity 透明度,晚間更為透明度,代碼如下:

if ((hour > 6 && hour < 19) || (hour == 6 && minutes >= 30)) {    timePane && timePane.a({        'morning.visible': false,        'day.visible': true,        'dusk.visible': false,        'night.visible': false,        'day.opacity': 1    })    skyBox.s({        "shape3d.blend": 'rgb(127, 200, 240)',    })    cloudBackground.s({        "back.opacity": 0.7,    })    clouds.s({        "shape3d.opacity": 0.7,    })} else if ((hour < 6 || hour > 19) || (hour == 19 && minutes >= 30)) {//...省略} else if (hour == 6 && minutes < 15 ) {//...省略} else if (hour == 6 && minutes >= 15 && minutes < 30) {//...省略} else if (hour == 19 && minutes < 15) {//...省略} else if (hour == 19 && minutes >= 15 && minutes < 30) {//...省略}

這里我還增加了對右上角時間面板時間狀態圖標的支持,并增加了圖標切換時的漸隱漸顯效果,同時給時間面板狀態圖標位置增加了點擊切換到下一時間狀態的功能。

為了演示效果我增加了時間倍速按鈕,下圖是 500 倍時間流速下的變化情況:


 

總結

通過這個 demo ,我發現生活中有很多沒有被人所注意到的細節都存在數據可視化的可能,在這個大數據的時代更多的可能性值得被人發掘出來,不要錯個身邊每一個值得數據可視化的細節,這樣不僅可以更好的挖掘 HT for Web 的潛力,也可以加強自身身為一個程序員的綜合素質。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
69精品小视频| 欧美中文字幕在线播放| 亚洲精品国产拍免费91在线| 秋霞av国产精品一区| 久久国产精品久久久久久久久久| 日韩综合视频在线观看| 91av视频在线免费观看| 亚洲欧洲日产国码av系列天堂| 亚洲午夜激情免费视频| 久久精品视频99| 亚洲精品网址在线观看| 日韩影视在线观看| 国产热re99久久6国产精品| 久久综合久久美利坚合众国| 国产精品白嫩初高中害羞小美女| 高清欧美性猛交xxxx黑人猛交| 欧美成人午夜激情| 91夜夜揉人人捏人人添红杏| 久久人人爽国产| 久久久久久av| 国产成+人+综合+亚洲欧美丁香花| 欧美性猛交xxxx久久久| 欧美精品在线免费观看| 91精品久久久久久久久| 色噜噜久久综合伊人一本| 色综合五月天导航| 日本国产精品视频| 国产精品一区二区3区| 欧美激情一区二区三区成人| 欧美日韩中文字幕日韩欧美| 亚洲男人的天堂网站| 亚洲精品电影网| 中文字幕久久亚洲| 日韩电影免费在线观看中文字幕| 午夜免费久久久久| 国产精品尤物福利片在线观看| 黑人狂躁日本妞一区二区三区| 欧美成人剧情片在线观看| 久久久久久com| 国产成人一区二区在线| 动漫精品一区二区| 97福利一区二区| 成人在线播放av| 亚洲图片在线综合| 97久久国产精品| 欧美俄罗斯性视频| 自拍偷拍亚洲欧美| 欧美xxxx18国产| 精品国产一区二区三区久久狼5月| 正在播放欧美视频| 疯狂做受xxxx高潮欧美日本| 欧美性猛交xxxx富婆弯腰| 久久九九有精品国产23| 蜜月aⅴ免费一区二区三区| 久久久精品一区| 亚洲国产精彩中文乱码av| 最新亚洲国产精品| 日本精品免费观看| 日韩精品免费在线视频观看| 国产精品久久久久久久久久久久| 亚洲欧美一区二区精品久久久| 国产精品黄色影片导航在线观看| 久久久亚洲国产天美传媒修理工| 国产视频丨精品|在线观看| 欧美日韩亚洲一区二区三区| 中国日韩欧美久久久久久久久| 国产精品久久久久久久久久久久久| 日韩有码在线电影| 97精品一区二区视频在线观看| 97精品久久久中文字幕免费| 欧美性色视频在线| 国产在线播放不卡| 国产精品嫩草影院一区二区| 成人有码在线播放| 久久久免费观看视频| 中文字幕日韩欧美在线视频| 久久天天躁狠狠躁夜夜av| 亚洲精品一区二区在线| 日韩精品在线影院| 久久福利视频导航| 成人av番号网| 91精品久久久久久久久久久久久久| 精品无人国产偷自产在线| 亚洲精品久久久一区二区三区| 最近中文字幕日韩精品| 亚洲精品短视频| 欧美亚洲免费电影| 欧美丰满少妇xxxxx做受| 亚洲黄色有码视频| 岛国av一区二区| 91高清免费在线观看| 欧美性感美女h网站在线观看免费| 96国产粉嫩美女| 亚洲一区二区三区四区视频| 亚洲free嫩bbb| 国产日韩欧美中文| zzjj国产精品一区二区| 海角国产乱辈乱精品视频| 欧美日韩精品在线视频| 欧美成人免费播放| 国内精品视频一区| 亚洲成在人线av| 国产视频久久久久| 中文字幕欧美专区| 国产mv免费观看入口亚洲| 中日韩美女免费视频网站在线观看| 亚洲第一免费播放区| 夜夜狂射影院欧美极品| 色与欲影视天天看综合网| 韩国国内大量揄拍精品视频| 亚洲国产女人aaa毛片在线| 国产有码一区二区| 久久免费成人精品视频| 欧美日韩亚洲天堂| 日韩成人小视频| 国产精品爽爽爽爽爽爽在线观看| 成人免费网站在线看| 国产精品国产三级国产专播精品人| 国产精品www网站| 日本免费久久高清视频| 国产欧洲精品视频| 亚洲精品一区在线观看香蕉| 2019亚洲男人天堂| 国产成人一区二区在线| 久久免费视频网站| 久久精品国产视频| 成人在线免费观看视视频| 国产精品视频区| 成人免费网视频| 久久精品国产96久久久香蕉| 欧美夫妻性生活xx| 国产精品678| 97国产精品视频| 久久香蕉国产线看观看av| 国产精品久久久久秋霞鲁丝| 欧美华人在线视频| 668精品在线视频| 蜜月aⅴ免费一区二区三区| 欧美精品久久久久| 国产一区视频在线| 久久久国产精品免费| 亚洲色图美腿丝袜| 国产精品专区h在线观看| 国产69久久精品成人看| 国产在线一区二区三区| 国内外成人免费激情在线视频网站| 日韩一二三在线视频播| 奇米四色中文综合久久| 日韩在线不卡视频| 成人av色在线观看| 国产日韩换脸av一区在线观看| 性色av一区二区三区红粉影视| 一级做a爰片久久毛片美女图片| 久久高清视频免费| 亚洲最大成人免费视频| 国产精品扒开腿做爽爽爽的视频| 国产精品免费看久久久香蕉| 国产精品99久久久久久白浆小说| 亚洲在线免费视频| 91在线高清免费观看| 久久久免费高清电视剧观看| 久久久中文字幕| 大胆人体色综合| 91久久精品一区|