摘要:如果你嫌上面太瑣碎,可以直接取看我的試驗源碼,文件是歡迎本文后續(xù)從開始擼一個支持單軸輪播的雷達圖之末篇本文首發(fā)于
在線示例
如果你還不了解canvas,還不知道要講啥,建議從首篇看起:從0開始寫一個支持單軸輪播的雷達圖
在首篇我們已經(jīng)講了怎么實現(xiàn)輪播,在這里我們將要用這一篇文章來說一下雷達圖的單軸hover效果的實現(xiàn),也是我寫這篇文章的原因,因為echarts只支持單個series的hover。而我想要的是支持下面的效果:單軸hover。如果你只想知道怎么解決Echarts單軸輪播問題,并且你足夠機智,你可以直接看最后一節(jié)。
如果你用過Echarts的radar圖,如果你去看過生成雷達圖形成的dom結(jié)構(gòu),你hover后會發(fā)現(xiàn)如下圖所示
有沒有一種恍然大悟的感覺,其實就是在這畫布的容器(我們稱其為target)里里,添加了一個dom元素,來顯示提示詞。而這個dom元素的定位是絕對定位,定位點是根據(jù)hover捕獲點在畫布上的left和top計算值,然后顯示生成的tooltip值;所以我們可以思考,要達到前面這幾句話,我們需要提前做以下幾步:
1、綁定一個鼠標(biāo)移動事件,并試圖捕捉畫布上的拐點;
2、根據(jù)捕捉的點,獲取相應(yīng)的數(shù)據(jù)(標(biāo)簽,數(shù)值);
3:生成一個div元素(也可以是label這些元素),并設(shè)定位置,樣式,及顯示值,并添加到容器中,大功告成;
原諒我用了個英文單詞,我確實不知道該用個什么標(biāo)題。
如果你細(xì)心看過首篇的每一個實現(xiàn)步驟,并且理解整體思路,那接下來,事情會簡單很多。因為無論哪個js插件畫雷達圖,前面的幾步都是必須的,其核心除了canvas,其實就是計算和坐標(biāo)轉(zhuǎn)換,即將一塊高為width,寬為height的dom元素變成一張擁有直角坐標(biāo)系的畫布,就像下面這個簡圖:
我們?yōu)檫@個target元素綁定一個click事件,然后
const draw = document.getElementById("canvas"); const ctx = draw.getContext("2d"); /*獲取鼠標(biāo)在canvas畫布上的位置(**不是瀏覽器窗口的鼠標(biāo)位置) * clientX獲取的相對瀏覽器窗口左上角的位置,適用于所有瀏覽器 * 在chrome瀏覽器中,有一個zrX屬性,是相對于元素本身的相對位置 * getBoundingClientRect()函數(shù)是獲取元素邊框相對于瀏覽器窗口的位置 * */ function getMousePos(canvas, event) { var rect = canvas.getBoundingClientRect(); return { x: event.clientX - rect.left , y: event.clientY - rect.top } } draw.addEventListener("click",function(event){ console.log(getMousePos(draw,event)); //打印得到的,就是鼠標(biāo)點擊位置相對于canvas容器元素左上角的坐標(biāo)值,而不是相對于瀏覽器窗口。 });
如果你還不明白,你可以試著拷貝代碼,建立一個html頁面嘗試一下
根據(jù)獲取的坐標(biāo)點,判斷是不是需要捕捉的點你有可能會疑問,就算捕捉了這個點,那怎么才能判斷這個點是不是需要捕捉的拐點呢?答案是,前面繪制這個拐點的時候,我們已經(jīng)記錄下了這個拐點在坐標(biāo)位置,所以我們只需要將點擊獲取到的像素(px)值,轉(zhuǎn)化為建立的坐標(biāo)系坐標(biāo),我們就能輕易判斷是不是拐點了,當(dāng)然為了實現(xiàn)的靈敏度,我們需要給其配一個誤差值,保證在拐點某個面積內(nèi)都能觸發(fā)這個hover。代碼中有較多注釋,具體看代碼實現(xiàn):
function addListener(dom,canvas,pointData){ //獲取鼠標(biāo)在canvas畫布上的位置(**不是瀏覽器窗口的鼠標(biāo)位置) function getMousePos(canvas, evt) { //canvas就是dom中的canvas元素 var rect = canvas.getBoundingClientRect(); return { x: evt.clientX - rect.left * (canvas.width / rect.width), y: evt.clientY - rect.top * (canvas.height / rect.height) } } dom.addEventListener("click",function (evt) { //開始測試建議用點擊,后面弄清楚了改成mousemove事件就ok const mouse = getMousePos(canvas, evt); let point={}; let text = ""; let index =-1; const r =5; //這個r就是那個允許的誤差值,值越大,越容易觸發(fā)hover point.x=mouse.x-offset.x/2; //offset的值就是畫布的長和寬值,利用他們的中心點,可以計算鼠標(biāo)點擊位置在坐標(biāo)系中的坐標(biāo)值; point.y=mouse.y-offset.y/2;//同上 for(let i=0;i生成一個div元素,顯示相應(yīng)展示的輪播值(item[0] - r) && point.x < (item[0] + r) && point.y > (item[1] - r) && point.y < (item[1] + r)) { index = i; //滿足條件,就記錄下這個值,用于后面單軸顯示 break; } } if(index!==-1){ hoverLabel(dom,mouse,index); //在下一步,我們將實現(xiàn)這個方法 }else{ removeLabel(dom);//當(dāng)然,我們也將順便實現(xiàn)這個方法 } })
當(dāng)我們生成一個div,并試圖定位將其添加到畫布容器中顯示出來時,你需要明白下面兩點:
1:這個div需要絕對定位,裝他的容器需要一個非static定位;
2:裝這個div的元素是裝canvas容器的元素,而不是canvas,添加到canvas中 ,你將什么也看不到;
這個實現(xiàn)較簡單,就直接看源碼吧
/*dom 即要添加div的容器,point即觸發(fā)hover事件的位置,是位置,不是變換后的坐標(biāo),text即要顯示的值*/ function hoverLabel(dom,point,text){ removeLabel(dom);//生成前先判斷有無,有就移除這個顯示標(biāo)簽 let label =document.createElement("div"); label.style.position="absolute"; label.style.top=point.y+"px"; label.style.left=point.x+"px"; label.style.border="1px solid yellowgreen"; label.style.background = "gray"; label.style.zIndex = 999; label.innerHTML ="show:0999"+text; dom.appendChild(label); } function removeLabel(dom) { //移除這個顯示標(biāo)簽 (dom.querySelectorAll("label").length)&&(dom.removeChild(dom.querySelector("label"))); //有就刪除,沒有就不動作 }
提示,我們也可以提前生成一個標(biāo)簽元素,將其添加到畫布容器中,并不顯示它,后面我們只需動態(tài)改變它的位置和內(nèi)容。
至此,我們已經(jīng)完成了單軸的hover,也就是下面這種效果圖
到這里,似乎該告一段落了,但好戲才要真正開始,我們的標(biāo)題是解決Echarts單軸輪播,我不是一個標(biāo)題黨,所以接下來,將要討論這個有趣的話題。
通過上面我們已經(jīng)自己實現(xiàn)了一個Radar圖,并對其實現(xiàn)了單軸輪播和hover,我們已經(jīng)明白了其中的坐標(biāo)變換,所以我們可否用Echarts生成一個radar圖,自己為其寫一個tooltip呢,答案是必須的,但前提是,這是個Radar圖只有一個系列,而且最好半徑是設(shè)定好的。其實現(xiàn)的核心思想就是,通過我們option中的設(shè)置,獲取畫布大小,獲取標(biāo)簽和值,生成坐標(biāo)系,算出目標(biāo)點的坐標(biāo),然后添加上面說的hover事件,我們還是直接看他的具體實現(xiàn),直接上源碼:
/*target 添加畫布的容器,option即為setOption中的Option,autoOption用于設(shè)定是否自動輪播,是否hover觸發(fā),*/ function addHover(target,option,autoOption={autoShow:false,hoverEnable:true,time:1000}){ const m =Math; var center ={ //獲取畫布的中心偏離比,因為echarts支持百分?jǐn)?shù)控制radar圖在畫布中的位置,所以我們需要計算這個點,當(dāng)然我們也可以為了方便,不在option中設(shè)置,這樣我們就可以直接用畫布的中心點,即(0.5,0.5); pointx:(option.radar.center&&Number(option.radar.center[0].substr(0,option.radar.center[0].length-1))/100)||0.5, //計算X軸的偏離比例 pointy:(option.radar.center&&Number(option.radar.center[1].substr(0,option.radar.center[1].length-1))/100)||0.5, //計算y軸的偏離比例 }; var x=target.offsetWidth*center.pointx,y=target.offsetHeight*center.pointy; //計算radar中心點x的值,計算radar中心點x的值 var indicator = option.radar.indicator; //獲取option中radar的標(biāo)簽 var data = option.series[0].data[0].value;//獲取option中radar的值,我們在這里只去第一個series的值; var length = indicator.length; //獲取標(biāo)簽的長度,即雷達的拐點數(shù),這個很重股 var step =-1; //輪播要用到的參數(shù) var hovering =false ; //這個參數(shù)是控制輪播與hover同時觸發(fā),顯示hover值,暫停輪播顯示 var radius=option.radar.radius,pointData=[]; //獲取radar的半徑 var style ={ //hover顯示的樣式 color:"#fff", border:"1px solid rgb(51,51,51)", borderRadius:"4px", backgroundColor:"rgba(50,50,50,0.7)" }; const single = 2*m.PI /length*(-1); for(let i = 0;i(item[0] - r) && point.x < (item[0] + r) && point.y > (item[1] - r) && point.y < (item[1] + r)) { index = i; break; } } if(index!==-1){ var tag =indicator[index] var text = tag.name+":"+m.round(data[index]*100/tag.max)+"%"; hovering =true; hoverLabel(label,mouse,text,style); }else{ hovering =false; removeLabel(label); } })) }
至于調(diào)用,那就簡單了
第一步: var ele = document.getElementById("chart"); var draw = echarts.init(ele); 第二步: 配置你的option var option ={}; 第三步畫出你的雷達圖 draw.setOption(option); 第四步:綁定事件: addHover(ele,option,{autoShow:true,hoverEnable:true});
至此,我們就 完全解決了Echarts單軸輪播問題。
如果你嫌上面太瑣碎,可以直接取github看我的試驗源碼,文件是mouse.html,歡迎star
本文后續(xù):從0開始擼一個支持單軸輪播的雷達圖之末篇
本文首發(fā)于:http://closertb.site
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/85057.html
摘要:今天要用到的理論在前兩篇都講過,如果你錯過了前兩篇,你應(yīng)該先看看。但是我們可否用生成一個圖,自己為其寫一個呢答案是肯定的。但一切的一切前提是,要支持單軸輪播,這個圖只有一個系列。 今天要用到的理論在前兩篇都講過,如果你錯過了前兩篇,你應(yīng)該先看看。從0開始寫一個支持單軸輪播的雷達圖之首篇從0開始寫一個支持單軸輪播的雷達圖之中篇 前言 通過前面我們自己實現(xiàn)了一個Radar圖,并對其實現(xiàn)了單...
摘要:引子最近做公司的數(shù)據(jù)展示項目,用的核心插件是,但在雷達圖的展示案列上,需求上出現(xiàn)了需要單軸輪播標(biāo)簽和數(shù)據(jù),在看完上的后,這個不支持,看了一下源碼,似乎有點復(fù)雜,改了改,只實現(xiàn)了多個的輪播,和需求還是有差距,周末反正無聊,何不自己動手?jǐn)]一個。 引子 最近做公司的數(shù)據(jù)展示項目,用的核心插件是Echarts,但在雷達圖的展示案列上,需求上出現(xiàn)了需要單軸輪播標(biāo)簽和數(shù)據(jù),在看完github上的i...
摘要:以上三種動態(tài)樣式注入,不同的使用場景,各有利弊,至于你想用哪一種,需要你自己權(quán)衡,睡覺去啦。。。。 前言 作為一個前端,我們都聽過結(jié)構(gòu),樣式,行為分離;關(guān)于樣式,我們都聽過外聯(lián)樣式,內(nèi)聯(lián)樣式和行內(nèi)樣式;關(guān)于這三者,什么權(quán)重啊,啊,對了,這些都不會出現(xiàn)在這篇文章里,這篇文章就說一些那些我們不怎么使用的,動態(tài)引入css樣式的方法; 靜態(tài)樣式引入 前面說過外聯(lián)樣式,內(nèi)聯(lián)樣式和行內(nèi)樣式,所謂外...
摘要:其次父組件中負(fù)責(zé)通用的功能,以及輪播的整體架構(gòu),其結(jié)構(gòu)如下。下面的是一種移動端的適配方案。接下來實現(xiàn)函數(shù)運用動畫切換到指定下標(biāo)的子項到此為止,咱們就已經(jīng)完成了一個初步的滑動切換輪播圖的功能了。 前言 昨天寫了一篇側(cè)邊菜單組件的文章,閱讀人數(shù)挺多的,內(nèi)心很欣喜(偷著樂,第一篇文章有這么多人看)!乘著這股勁,今天在繼續(xù)寫一篇我們平時工作中更常用的滑動輪播組件的文章。 效果展示 老規(guī)矩,咱們...
閱讀 3261·2021-11-18 10:02
閱讀 1463·2021-10-12 10:08
閱讀 1264·2021-10-11 10:58
閱讀 1279·2021-10-11 10:57
閱讀 1178·2021-10-08 10:04
閱讀 2133·2021-09-29 09:35
閱讀 783·2021-09-22 15:44
閱讀 1283·2021-09-03 10:30