摘要:而可視化圖表,則是強大功能的表現之一。效果動畫效果圖片顯示不出來,可以到最下面找地址分析可以這個圖表由軸數據條形和標題組成。這里就需要監聽事件,當鼠標的位置位于柱狀的面積內,觸發事件。
前言
canvas 強大的功能讓它成為了 HTML5 中非常重要的部分,至于它是什么,這里就不需要我多作介紹了。而可視化圖表,則是 canvas 強大功能的表現之一。
現在已經有了很多成熟的圖表插件都是用 canvas 實現的,Chart.js、ECharts等可以制作出好看炫酷的圖表,而且幾乎覆蓋了所有圖表的實現。
有時候自己只想畫個柱狀圖,自己寫又覺得麻煩,用別人插件又感覺累贅,最后打開百度,拷段代碼,粘貼上來修修改改。還不如自己擼一個呢。
效果動畫效果圖片顯示不出來,可以到最下面找demo地址
分析可以這個圖表由 xy軸、數據條形和標題組成。
軸線:可以使用 moveTo() & lineTo() 實現
文字:可以使用 fillText() 實現
長方形:可以使用 fillRect() 實現
這樣看來,似乎并沒有多難。
實現 定義畫布canvas 標簽只是個容器,真正實現畫圖的還是 JavaScript。
畫坐標軸坐標軸就是兩條橫線,也就是canvas里最基礎的知識。
由 ctx.beginPath() 開始一條新的路徑
ctx.lineWidth=1 設置線條寬度
ctx.strokeStyle="#000000" 設置線條顏色
ctx.moveTo(x,y) 定義線條的起點
ctx.lineTo(x1,y1) 定義線條的終點
最后 ctx.stroke() 把起點和終點連成一條線
var canvas = document.getElementById("canvas"); var ctx = canvas.getContext("2d"); var width = canvas.width; var height = canvas.height; var padding = 50; // 坐標軸到canvas邊框的邊距,留邊距寫文字 ctx.beginPath(); ctx.lineWidth = 1; // y軸線 ctx.moveTo(padding + 0.5, height - padding + 0.5); ctx.lineTo(padding + 0.5, padding + 0.5); ctx.stroke(); // x軸線 ctx.moveTo(padding + 0.5, height - padding + 0.5); ctx.lineTo(width - padding + 0.5, height - padding + 0.5); ctx.stroke();畫坐標點
y軸上多少坐標點由自己來定義,需要獲取到數據的最大值來計算y軸上的坐標值。x軸的點則由傳入的數據長度決定,坐標值由傳入數據的 xAxis 屬性決定。
坐標值就是文字,由 ctx.fillText(value, x, y) 填充文字,value 為文字值,x y 為值的坐標
ctx.textAlign="center" 設置文字居中對齊
ctx.fillStyle="#000000" 設置文字填充顏色
var yNumber = 5; // y軸的段數 var yLength = Math.floor((height - padding * 2) / yNumber); // y軸每段的真實長度 var xLength = Math.floor((width - padding * 2) / data.length); // x軸每段的真實長度 ctx.beginPath(); ctx.textAlign = "center"; ctx.fillStyle = "#000000"; ctx.strokeStyle = "#000000"; // x軸刻度和值 for (var i = 0; i < data.length; i++) { var xAxis = data[i].xAxis; var xlen = xLength * (i + 1); ctx.moveTo(padding + xlen, height - padding); ctx.lineTo(padding + xlen, height - padding + 5); ctx.stroke(); // 畫軸線上的刻度 ctx.fillText(xAxis, padding + xlen - xLength / 2, height - padding + 15); // 填充文字 } // y軸刻度和值 for (var i = 0; i < yNumber; i++) { var y = yFictitious * (i + 1); var ylen = yLength * (i + 1); ctx.moveTo(padding, height - padding - ylen); ctx.lineTo(padding - 5, height - padding - ylen); ctx.stroke(); ctx.fillText(y, padding - 10, height - padding - ylen + 5); }柱狀動畫
接下來要把數據通過柱狀的高低顯示出來,這里有個動畫效果,柱狀會從0升到對應的值。在 canvas 上實現動畫我們可以使用 setInterval、setTimeout 和 requestAnimationFrame。
requestAnimationFrame 不需要自己設置定時時間,而是跟著瀏覽器的繪制走。這樣就不會掉幀,自然就流暢。
requestAnimationFrame 原本只支持IE10以上,不過可以通過兼容的寫法實現兼容到IE6都行。
function looping() { looped = requestAnimationFrame(looping); if(current < 100){ // current 用來計算當前柱狀的高度占最終高度的百分之幾,通過不斷循環實現柱狀上升的動畫 current = (current + 3) > 100 ? 100 : (current + 3); drawAnimation(); }else{ window.cancelAnimationFrame(looped); looped = null; } } function drawAnimation() { for(var i = 0; i < data.length; i++) { var x = Math.ceil(data[i].value * current / 100 * yRatio); var y = height - padding - x; ctx.fillRect(padding + xLength * (i + 0.25), y, xLength/2, x); // 保存每個柱狀的信息 data[i].left = padding + xLength / 4 + xLength * i; data[i].top = y; data[i].right = padding + 3 * xLength / 4 + xLength * i; data[i].bottom = height - padding; } } looping();
柱狀即是畫矩形,由 ctx.fillRect(x, y, width, height) 實現,x y 為矩形左上角的坐標,width height 為矩形的寬高,單位為像素
ctx.fillStyle="#1E9FFF" 設置填充顏色
到這里,一個最基本的柱狀圖就完成了。接下來,我們可以為他添加標題。
標題要放置標題,就會發現我們一大早定義的 padding 內邊距確實有用,總不能把標題給覆蓋到柱狀圖上吧。但是標題有的是在頂部,有的在底部,那么就不能寫死了。定一個變量 position 來判斷位置去畫出來。這個簡單。
// 標題 if(title){ // 也不一定有標題 ctx.textAlign = "center"; ctx.fillStyle = "#000000"; // 顏色,也可以不用寫死,個性化嘛 ctx.font = "16px Microsoft YaHei" if(titlePosition === "bottom" && padding >= 40){ ctx.fillText(title,width/2,height-5) }else{ ctx.fillText(title,width/2,padding/2) } }監聽鼠標移動事件
我們看到,有些圖表,把鼠標移上去,當前的柱狀就變色了,移開之后又變回原來的顏色。這里就需要監聽 mouseover 事件,當鼠標的位置位于柱狀的面積內,觸發事件。
那我怎么知道在柱狀里啊,發現在 drawAnimation() 里會有每個柱狀的坐標,那我干脆把坐標給保存到 data 里。那么鼠標在柱狀里的條件應該是:
ev.offsetX > data[i].left
ev.offsetX < data[i].right
ev.offsetY > data[i].top
ev.offsetY < data[i].bottom
canvas.addEventListener("mousemove",function(ev){ var ev = ev||window.event; for (var i=0;i總結data[i].left && ev.offsetX < data[i].right && ev.offsetY > data[i].top && ev.offsetY < data[i].bottom){ console.log("我在第"+i+"個柱狀里。"); } } })
為了更方便的使用,封裝成構造函數。通過
var chart = new sBarChart("canvas",data,{ title: "xxx公司年度盈利", // 標題 titleColor: "#000000", // 標題顏色 titlePosition: "top", // 標題位置 bgColor: "#ffffff", // 背景色 fillColor: "#1E9FFF", // 柱狀填充色 axisColor: "#666666", // 坐標軸顏色 contentColor: "#a5f0f6" // 內容橫線顏色 });
參數可配置,很簡單就生成一個個性化的柱狀圖。代碼地址:canvas-demo
最后加上折線圖、餅圖、環形圖,完整封裝成sChart.js插件,插件地址:sChart.js
更多文章:lin-xin/blog 微信贊賞文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/83152.html
摘要:前言月份開始出沒社區,現在差不多月了,按照工作的說法,就是差不多過了三個月的試用期,準備轉正了一般來說,差不多到了轉正的時候,會進行總結或者分享會議那么今天我就把看過的一些學習資源主要是博客,博文推薦分享給大家。 1.前言 6月份開始出沒社區,現在差不多9月了,按照工作的說法,就是差不多過了三個月的試用期,準備轉正了!一般來說,差不多到了轉正的時候,會進行總結或者分享會議!那么今天我就...
閱讀 3623·2021-11-24 10:22
閱讀 3692·2021-11-22 09:34
閱讀 2495·2021-11-15 11:39
閱讀 1534·2021-10-14 09:42
閱讀 3668·2021-10-08 10:04
閱讀 1562·2019-08-30 15:52
閱讀 851·2019-08-30 13:49
閱讀 3024·2019-08-30 11:21