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

首頁 > 編程 > HTML > 正文

基于 HTML5 Canvas實現 的交互式地鐵線路圖

2024-08-26 00:20:29
字體:
來源:轉載
供稿:網友

前言

前兩天在 echarts 上尋找靈感的時候,看到了很多有關地圖類似的例子,地圖定位等等,但是好像就是沒有地鐵線路圖,就自己花了一些時間搗鼓出來了這個交互式地鐵線路圖的 Demo,地鐵線路上的點是在網上隨便下載了一個,這篇文章記錄自己的一些收獲(畢竟我還是個菜鳥)以及代碼的實現,希望能夠幫到一些朋友。當然,如果有什么意見的可以直接跟我說,大家一起交流才會進步。

效果圖

html5,canvas,地鐵線路圖

http://www.hightopo.com/demo/subway/index.html

地圖稍微內容有點多,要全部展示,字顯得有點小了,但是沒關系,可以按照需求放大縮小,字體和繪制的內容并不會失真,畢竟都是用矢量繪制的~

界面生成

底層的 div 是通過 ht.graph.GraphView 組件生成的,然后就可以利用 HT for Web 提供好的方法,調用 canvas 畫筆隨便繪制就好,先來看看怎么生成底層 div:
 

var dm = new ht.DataModel();//數據容器var gv = new ht.graph.GraphView(dm);//拓撲組件gv.addToDOM();//將拓撲圖組件添加進body中

addToDOM 函數聲明如下:

addToDOM = function(){       var self = this,        view = self.getView(),           style = view.style;    document.body.appendChild(view); //將組件底層div添加到body中               style.left = '0';//默認組件是絕對定位,所以要設置位置    style.right = '0';    style.top = '0';    style.bottom = '0';          window.addEventListener('resize', function () { self.iv(); }, false); //窗口變化事件           }

現在我就可以在這個 div 上亂涂亂畫了~首先我獲取下載好的地鐵線路圖上的點,我將它們放在 subway.js 中,這個 js 文件全部都是下載的內容,我沒有做其他的改動,主要是將這些點根據線路來分分配添加到數組中,比如:

mark_Point13 = [];//線路 數組內包含線路的起點和終點坐標以及這條線路的名稱t_Point13 = [];//換成站點 數組內包含線路中的換乘站點坐標以及換成站點名稱n_Point13 = [];//小站點 數組內包含線路中的小站點坐標以及小站點名稱mark_Point13.push({ name: '十三號線', value: [113.4973,23.1095]}); mark_Point13.push({ name: '十三號線', value: [113.4155,23.1080]}); t_Point13.push({ name: '魚珠', value: [113.41548,23.10547]}); n_Point13.push({ name: '裕豐圍', value: [113.41548,23.10004]}); 

接下來來描繪地鐵線路,我聲明了一個數組 lineNum,用來裝 js 中所有的地鐵線路的編號,以及一個 color 數組,用來裝所有的地鐵線的顏色,這些顏色的 index 與 lineNum 中地鐵線編號的 index 是一一對應的:

var lineNum = ['1', '2', '3', '30', '4', '5', '6', '7', '8', '9', '13', '14', '32', '18', '21', '22', '60', '68'];var color = ['#f1cd44', '#0060a1', '#ed9b4f', '#ed9b4f', '#007e3a', '#cb0447', '#7a1a57', '#18472c', '#008193', '#83c39e', '#8a8c29', '#82352b', '#82352b', '#09a1e0', '#8a8c29', '#82352b', '#b6d300', '#09a1e0'];

接著遍歷 lineNum,將 lineNum 中的元素和顏色傳到 createLine 函數中,根據這兩個參數來繪制地鐵線路以及配色,畢竟 js 文件中的命名方式也是有規律的,哪一條線路,則命名后面一定會加上對應的數字,所以我們只需要將字符串與這個編號結合即可獲得 js 中對應的數組了:

let lineName = 'Line' + num;let line = window[lineName];createLine 的定義也非常簡單,我的代碼設置了不少的樣式,所以看起來有點多。創建一個 ht.Polyline 管線,我們可以通過 polyline.addPoint() 函數向這個變量中添加具體的點,通過 setSegments 可以設置點的連接方式。function createLine(num, color) {//繪制地圖線    var polyline = new ht.Polyline();//多邊形 管線    polyline.setTag(num);//設置節點tag標簽,作為唯一標示    if(num === '68') polyline.setToolTip('A P M');//設置提示信息     else if(num === '60') polyline.setToolTip('G F');     else polyline.setToolTip('Line' + num);    if(color) {        polyline.s({//s 為 setStyle 的簡寫,設置樣式            'shape.border.width': 0.4,//設置多邊形的邊框寬度            'shape.border.color': color,//設置多邊形的邊框顏色            'select.width': 0.2,//設置選中節點的邊框寬度            'select.color': color//設置選中節點的邊框顏色        });    }    let lineName = 'Line' + num;    let line = window[lineName];    for(let i = 0; i < line.length; i++) {        for(let j = 0; j < line[i].coords.length; j++) {            polyline.addPoint({x: line[i].coords[j][0]*300, y: -line[i].coords[j][1]*300});            if(num === '68'){//APM線(有兩條,但是點是在同一個數組中的)                if(i === 0 && j === 0) {                    polyline.setSegments([1]);                }                else if(i === 1 && j === 0) {                    polyline.getSegments().push(1);                }                else {                    polyline.getSegments().push(2);                }            }            }    }    polyline.setLayer('0');//將線設置在下層,點設置在上層“top”    dm.add(polyline);//將管線添加進數據容器中儲存,不然這個管線屬于“游離”狀態,是不會顯示在拓撲圖上的    return polyline;}

上面代碼中添加地鐵線上的點有分為幾種情況,是因為 js 中設置線的時候 Line68 有一個“跳躍”點的現象,所以我們必須“跳躍”過去,篇幅有限 Line68 數組具體的聲明自行看 subway.js。

這里說明一點,如果用的是 addPoint 函數,不設置 segments 時,默認將添加進的點用直線連接,segments 的定義如下:

1: moveTo,占用 1 個點信息,代表一個新路徑的起點
 

2: lineTo,占用 1 個點信息,代表從上次最后點連接到該點
 

3: quadraticCurveTo,占用 2 個點信息,第一個點作為曲線控制點,第二個點作為曲線結束點
 

4: bezierCurveTo,占用 3 個點信息,第一和第二個點作為曲線控制點,第三個點作為曲線結束點
 

5: closePath,不占用點信息,代表本次路徑繪制結束,并閉合到路徑的起始點

所以我們要做“跳躍”的行為設置 segments 為 1 即可。

最后繪制這些地鐵線上的點,這個部分 subway.js 中也分離出來了,命名以“mark_Point”、“t_Point”以及“n_Point”開頭,我在前面 js 的展示部分有對這些數組進行解釋,大家動動中指劃上去看看。

我們在這些點的位置添加 ht.Node 節點,當節點一添加進 dm 數據容器中時,就會在拓撲圖上顯示,當然,前提是這個拓撲圖組件 gv 設置的數據容器是這個 dm。篇幅有限,添加地鐵線上的點的代碼部分我只展示添加“換乘站點”的點:

var tName = 't_Point' + num;var tP = window[tName];//大站點if(tP) {//有些線路沒有“換乘站點”    for(let i = 0; i < tP.length; i++) {        let node = createNode(tP[i].name, tP[i].value, color[index]);//在獲取的線路上的點的坐標位置添加節點        node.s({//設置節點的樣式style            'label.scale': 0.05,//文本縮放,可以避免瀏覽器限制的最小字號問題            'label.font': 'bold 12px arial, sans-serif'//設置文本的font        });        node.setSize(0.6, 0.6);//設置節點大小。由于js中每個點之間的偏移量太小,所以我不得不把節點設置小一些        node.setImage('images/旋轉箭頭.json');//設置節點的圖片        node.a('alarmColor1', 'rgb(150, 150, 150)');//attr屬性,可以在這里面設置任何的東西,alarmColor1是在上面設置的image的json中綁定的屬性,具體參看 HT for Web 矢量手冊(http://www.hightopo.com/guide/guide/core/vector/ht-vector-guide.html#ref_binding)        node.a('alarmColor2', 'rgb(150, 150, 150)');//同上        node.a('tpNode', true);//這個屬性設置只是為了用來區分“換乘站點”和“小站點”的,后面會用上    }}

所有的地鐵線路以及站點都添加完畢。但是!你可能會看不見自己繪制的圖,因為他們太小了,這個時候可以設置 graphView 拓撲組件上的 fitContent 函數,我們順便將拓撲圖上的所有東西不可移動也設置一下:

gv.fitContent(false, 0.00001);//自適應大小,參數1為是否動畫,參數2為gv與邊框的padding值gv.setMovableFunc(function(){    return false;//設置gv上的節點不可移動});

這下你的地鐵線路圖就可以顯示啦~接下來看看交互。

交互

首先是鼠標移動事件,鼠標滑過具體線路時,線路會變粗,懸停一會兒還能看到這條線路的編號;當鼠標移動到“換乘站點”或“小站點”,站點對應的圖標都會變大并且變色,字體也會變大,鼠標移開圖標變回原來的顏色并且字體變小。不同點在于鼠標移動到“換乘站點”時,“換乘站點”會旋轉。

html5,canvas,地鐵線路圖

鼠標滑動事件,我直接基于 gv 的底層 div 進行的 mousemove 事件,通過 ht 封裝的 getDataAt 函數傳入事件 event 參數,獲取事件下對應的節點,然后就可以隨意操作節點了:

gv.getView().addEventListener('mousemove', function(e) {    var data = gv.getDataAt(e);//傳入邏輯坐標點或者交互event事件參數,返回當前點下的圖元    if(name) {        originNode(name);//不管什么時候都要讓節點保持原來的大小    }    if (data instanceof ht.Polyline) {//判斷事件節點的類型        dm.sm().ss(data);//選中“管道”        name = '';        clearInterval(interval);    }    else if (data instanceof ht.Node) {        if(data.getTag() !== name && data.a('tpNode')) {//若不是同一個節點,并且mousemove的事件對象為ht.Node類型,那么設置節點的旋轉            interval = setInterval(function() {                data.setRotation(data.getRotation() - Math.PI/16); //在自身旋轉的基礎上再旋轉            }, 100);        }        if(data.a('npNode')) {//如果鼠標移到“小站點”也要停止動畫            clearInterval(interval);        }        expandNode(data, name);////自定義的放大節點函數,比較容易,我不粘代碼了,可以去http://hightopo.com/   查看        dm.sm().ss(data);//設置選中節點        name = data.getTag();//作為“上一個節點”的存儲變量,可以通過這個值來獲取節點    }    else {//其他任何情況則不選中任何內容并且清除“換乘站點”上的動畫        dm.sm().ss(null);        name = '';        clearInterval(interval);    }});

鼠標懸停在地鐵線路上時顯示“具體線路信息”,我是通過設置 tooltip 來完成的(注意:要打開 gv 的 tooltip 開關):

gv.enableToolTip();//打開 tooltip 的開關if(num === '68') polyline.setToolTip('A P M');//設置提示信息 else if(num === '60') polyline.setToolTip('G F'); else polyline.setToolTip('Line' + num);

然后我利用右下角的 form 表單,單擊表單上的具體線路,或者雙擊拓撲圖上任意一個“站點”或者線路,則拓撲圖會自適應到對應的部分,將被雙擊的部分展現到拓撲圖的中央。

html5,canvas,地鐵線路圖

form 表單的聲明部分我好像還沒有解釋。。。就是通過 new 一個 ht.widget.FomePane 類創建一個 form 表單組件,通過 form.getView() 獲取表單組件的底層 div,將這個 div 擺放在 body 右下角,然后通過 addRow 函數向 form 表單中添加一行的表單項,可以在這行中添加任意多個項,通過 addRow 函數的第二個參數(一個數組),對添加進的表單項進行寬度的設置,通過第三個參數設置這行的高度:

function createForm() {//創建右下角的form表單    var form = new ht.widget.FormPane();    form.setWidth(200);//設置表單寬度    form.setHeight(416);//設置表單高度    let view = form.getView();    document.body.appendChild(view);//將表單添加進body中    view.style.zIndex = 1000;    view.style.bottom = '10px';//ht組件幾乎都設置絕對路徑    view.style.right = '10px';    view.style.background = 'rgba(211, 211, 211, 0.8)';    names.forEach(function(nameString) {        form.addRow([//向表單中添加行            {//這一行中的第一個表單項                button: {//向表單中添加button按鈕                    icon: 'images/Line'+nameString.value+'.json',//設置按鈕的圖標                    background: '',//設置按鈕的背景                    borderColor: '',//設置按鈕的邊框顏色                    clickable: false//設置按鈕不可點擊                }            },            {//第二個表單項                button: {                    label: nameString.name,                    labelFont: 'bold 14px arial, sans-serif',                    labelColor: '#fff',                    background: '',                    borderColor: '',                    onClicked: function() {//按鈕點擊回調事件                        gv.sm().ss(dm.getDataByTag(nameString.value));//設置選中按下的按鈕對應的線路                        gv.fitData(gv.sm().ld(), true, 5);//將選中的地鐵線路顯示在拓撲圖的中央                    }                }            }        ], [0.1, 0.2], 23);//第二個參數是設置第一參數中的數組的寬度,小于1是比例,大于1是實際寬度。第三個參數是該行的高度    });}

單擊“站點”顯示紅色標注,雙擊節點自適應放置到拓撲圖中央以及雙擊空白處將紅色標注隱藏的內容都是通過對拓撲組件 gv 的事件監聽來控制的,非常清晰易懂,代碼如下:

var node = createRedLight();//創建一個新的節點,顯示為“紅燈”的樣式gv.mi(function(e) {//ht 中拓撲組件中的事件監聽    if(e.kind === 'clickData' && (e.data.a('tpNode') || e.data.a('npNode'))) {//e.kind獲取當前事件類型,e.data獲取當前事件下的節點        node.s('2d.visible', true);//設置node節點可見        node.setPosition(e.data.getPosition().x, e.data.getPosition().y);//設置node的坐標為當前事件下節點的位置    }    else if(e.kind === 'doubleClickData') {//雙擊節點        gv.fitData(e.data, false, 10);//將事件下的節點自適應到拓撲圖的中央,參數1為自適應的節點,參數2為是否動畫,參數3為gv與邊框的padding    }    else if(e.kind === 'doubleClickBackground') {//雙擊空白處        node.s('2d.visible', false);//設置node節點不可見 查看 HT for Web 樣式手冊(http://www.hightopo.com/guide/guide/core/theme/ht-theme-guide.html#ref_style)    }});

注意 s(style) 和 a(attr) 定義是這樣的,s 是 ht 預定義的一些樣式屬性,而 a 是我們用戶來自定義的屬性,一般是通過調用字符串來調用結果的,這個字符串對應的可以是常量也可以是函數,還是很靈活的。

最后還做了一個小小的部分,選中“站點”,則該“站點”的上方會顯示一個紅色的會“呼吸”的用來注明當前選中的“站點”。

html5,canvas,地鐵線路圖

“呼吸”的部分是利用 ht 的 setAnimation 函數來完成的,在用這個函數之前要先打開數據容器的動畫開關,然后設置動畫:

dm.enableAnimation();//打開數據容器的動畫開關function createRedLight() {    var node = new ht.Node();    node.setImage('images/紅燈.json');//設置節點的圖片    node.setSize(1, 1);//設置節點的大小    node.setLayer('firstTop');//設置節點顯示在gv的最上層    node.s('2d.visible', false);//節點不可見    node.s('select.width', 0);//節點選中時的邊框為0,不可見    node.s('2d.selectable', false);//設置這個屬性,則節點不可選中    node.setAnimation({//設置動畫 具體參見 HT for Web 動畫手冊(http://www.hightopo.com/guide/guide/plugin/animation/ht-animation-guide.html)        expandWidth: {            property: "width",//設置這個屬性,并且未設置 accessType,則默認通過 setWidth/getWidth 來設置和獲取屬性。這里的 width 和下面的 height 都是通過前面設置的 size 得到的            from: 0.5, //動畫開始時的屬性值            to: 1,//動畫結束時的屬性值            next: "collapseWidth"//字符串類型,指定當前動畫完成之后,要執行的下個動畫,可將多個動畫融合        },        collapseWidth: {            property: "width",            from: 1,             to: 0.5,            next: "expandWidth"        },        expandHeight: {            property: "height",            from: 0.5,             to: 1,            next: "collapseHeight"        },        collapseHeight: {            property: "height",            from: 1,             to: 0.5,            next: "expandHeight"        },        start: ["expandWidth", "expandHeight"]//數組,用于指定要啟動的一個或多個動畫    });    dm.add(node);    return node;}

全部代碼結束!

總結

這個 Demo 花了我兩天時間完成,總覺得有點不甘心啊,但是有時候思維又轉不過彎來,花費了不少的時間,但是總的來說收獲還是很多的,我以前一直以為只要通過 getPoints().push 來向多邊形中添加點就可以了,求助了大神之后,發現原來這個方法不僅繞彎路而且還會出現各種各樣的問題,比如 getPoints 之前,一定要在多邊形中已經有 points 才可以,但是在很多情況下,初始化的 points 并不好設置,而且會造成代碼很繁瑣,直接通過 addPoint 方法,直接將點添加進多邊形變量中,并且還會默認將點通過直線的方式連接,也不用設置 segments,多可愛的一個函數。

還有就是因為 ht 默認縮放大小是 20,而我這個 Demo 的間距又很小,導致縮放到最大地鐵線路圖顯示也很小,所以我在 htconfig 中更改了 ht 的默認 zoomMax 屬性,記住,更改這個值一定要在所有的 ht 調用之前,因為在 htconfig 中設置的值在后面定義都是不可更改的。

以上所述是小編給大家介紹的基于 HTML5 Canvas實現 的交互式地鐵線路圖,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對VeVb武林網網站的支持!

 

注:相關教程知識閱讀請移步到HTML教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久777国产线看观看精品| 亚洲精品中文字幕有码专区| 久久精品视频免费播放| 一区二区欧美激情| 日韩的一区二区| 91亚洲精品久久久久久久久久久久| 欧美中文字幕在线观看| 国产亚洲精品va在线观看| 亚洲欧美日韩高清| 国语自产精品视频在线看| 久久99精品视频一区97| 欧美疯狂性受xxxxx另类| 日韩欧美有码在线| 热久久这里只有精品| 国产mv久久久| 欧美精品一区二区三区国产精品| 97香蕉超级碰碰久久免费的优势| 国产精品第10页| 久久久久久12| 国产中文字幕91| 亚洲国产精品yw在线观看| 91中文字幕在线| 深夜成人在线观看| 亚洲国产精品网站| 亚洲天堂网在线观看| 久久精品国产久精国产思思| 91成人在线视频| 亚洲大胆人体视频| 国产精品一区二区久久国产| 亚洲第一页中文字幕| 国产一区二区三区四区福利| 欧美一区二区大胆人体摄影专业网站| 最近2019中文字幕在线高清| 色婷婷av一区二区三区久久| 国产精品嫩草视频| 日韩精品视频观看| 91欧美精品成人综合在线观看| 日本视频久久久| 亚洲欧洲在线免费| 亚洲最大在线视频| 97视频在线免费观看| 国产精品wwwwww| 欧美在线视频观看免费网站| 日韩免费电影在线观看| 亚洲成人网久久久| 欧美肥婆姓交大片| 亚洲最大的av网站| 日韩精品在线视频| www.欧美精品一二三区| 九九热精品在线| 中文字幕欧美日韩| 国产综合久久久久| 日本免费在线精品| 国产视频999| 国产精品一二区| 国产精品久久久久久中文字| 深夜福利一区二区| 国产视频精品在线| 中文字幕亚洲天堂| 欧美在线亚洲在线| 成人激情视频在线观看| 欧美疯狂性受xxxxx另类| 奇米成人av国产一区二区三区| 日韩电影中文字幕在线观看| 欧美成人亚洲成人日韩成人| 国产精品久久久久一区二区| 欧美刺激性大交免费视频| 日韩av在线不卡| 久久九九全国免费精品观看| 日韩av不卡电影| 久久国产精品影视| 一本色道久久综合狠狠躁篇的优点| 亚洲最大激情中文字幕| 68精品久久久久久欧美| 欧美日韩国产综合视频在线观看中文| 久久免费福利视频| 欧美视频一区二区三区…| 久精品免费视频| 伊人久久五月天| 51ⅴ精品国产91久久久久久| 国产日韩欧美电影在线观看| 亚洲综合视频1区| 欧美日韩激情小视频| 成人美女av在线直播| 久久成人精品视频| 国产欧美一区二区三区视频| 亚洲精品456在线播放狼人| 亚洲激情视频在线观看| 国外成人在线播放| 久久99精品久久久久久青青91| 欧美日韩国产麻豆| 国产一区红桃视频| 久久91精品国产91久久跳| 亚洲深夜福利网站| 91亚洲永久免费精品| 中文.日本.精品| 亚洲qvod图片区电影| 欧美激情一区二区三区高清视频| 久久精品一本久久99精品| 日韩人在线观看| 久久久噜久噜久久综合| 久久精品久久精品亚洲人| 深夜成人在线观看| 欧美日韩国产麻豆| 亚洲淫片在线视频| 成人免费xxxxx在线观看| 精品国产乱码久久久久久天美| 久久亚洲精品国产亚洲老地址| 91精品国产高清久久久久久久久| 欧美裸身视频免费观看| 欧美激情精品久久久久久大尺度| 国产精品jvid在线观看蜜臀| 亚洲成avwww人| 日韩黄在线观看| 在线国产精品播放| 色琪琪综合男人的天堂aⅴ视频| 欧美激情喷水视频| 久久精品青青大伊人av| 最新国产精品拍自在线播放| 日本久久久久久久久久久| 亚洲精品电影久久久| 亚洲国产精品人久久电影| 亚洲精品久久久久国产| 久久精品久久精品亚洲人| 乱亲女秽乱长久久久| 欧美日韩国产专区| 国产精品第2页| 日韩高清人体午夜| 国产专区精品视频| 久久精品国产96久久久香蕉| 亚洲第一免费网站| 国产视频观看一区| 精品夜色国产国偷在线| 欧美黑人狂野猛交老妇| 久久亚洲精品一区二区| 欧洲亚洲免费视频| 两个人的视频www国产精品| 91久久嫩草影院一区二区| 中文字幕日本精品| 亚洲欧洲美洲在线综合| 国产精品女人久久久久久| 亚洲美女动态图120秒| 久久久久久有精品国产| 亚洲japanese制服美女| 91久久精品视频| 欧美xxxx综合视频| 国产欧美一区二区三区久久人妖| 亚洲自拍欧美色图| 久久久中精品2020中文| 国产精品免费久久久久久| 国产福利视频一区二区| 久久久女人电视剧免费播放下载| 国产精品电影在线观看| 姬川优奈aav一区二区| 亚洲美女激情视频| 亚洲理论电影网| 欧美亚洲成人xxx| 日韩精品久久久久久福利| 亚洲美女精品久久| 国产精品海角社区在线观看| 国产精品亚洲第一区| 激情成人在线视频| 欧美日韩亚洲系列| 在线成人激情视频|