成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

【帶著canvas去流浪】(1)繪制柱狀圖

sarva / 1680人閱讀

摘要:示例代碼托管在博客園地址大史住在大前端原創(chuàng)博文目錄華為云社區(qū)地址你要的前端打怪升級指南一任務(wù)說明使用原生繪制柱狀圖。

示例代碼托管在:http://www.github.com/dashnowords/blogs

博客園地址:《大史住在大前端》原創(chuàng)博文目錄

華為云社區(qū)地址:【你要的前端打怪升級指南】

[TOC]

一. 任務(wù)說明

使用原生canvasAPI繪制柱狀圖。(柱狀圖截圖來自于百度Echarts官方示例庫【查看示例鏈接】)

二. 重點提示

柱狀圖或許是最容易實現(xiàn)的圖表類型了,矩形的部分直接使用fillRect()來繪制即可,為了將坐標(biāo)軸標(biāo)簽文字繪制在小分割線中間,需要用measureText()來測量文本的寬度,然后進(jìn)行相應(yīng)的偏移,否則直接繪制的話文字的左邊界會和直線相對齊。其他部分都是一些基本API的使用,希望各位小伙伴通過做練習(xí)來熟悉這些API的用法。

三. 示例代碼

提示:代碼中將個別圖表參數(shù)直接寫在了函數(shù)里(也就是所謂的“魔鬼數(shù)字”),這種做法是不提倡的,因為它違反了開發(fā)的基本原則之一“開放封閉原則”。如果你使用過Echarts圖表庫就會發(fā)現(xiàn),圖表中幾乎所有要素都可以通過參數(shù)來定制,此處只需要關(guān)注canvasAPI的實現(xiàn)方法即可。

/**
 * 獲取canvas繪圖上下文
 * @type {[type]}
 */
const canvas = document.getElementById("canvas");
const context = canvas.getContext("2d");

//繪圖配置
let options = {
        chartZone:[50,50,1000,700],//標(biāo)識繪圖區(qū)域
        yAxisLabel:["0","100","200","300","400"],//標(biāo)示Y軸坐標(biāo)
        yMax:400,//Y軸最大值
        xAxisLabel:["Mon","Tue","Wed","Thu","Fri","Sat","Sun"],//X軸坐標(biāo)
        data:[10,50,200,330,390,320,220],//柱狀圖數(shù)據(jù)
        barStyle:{
            width:70,//柱狀圖寬度
            color:"#1abc9c"http://柱狀圖顏色
        }
    }

/*Echarts使用時,會調(diào)用實例方法echartsInstance.setOptions(options)來啟動繪圖*/
drawBarChart(options);

/**
 * 繪制柱狀圖
 */
function drawBarChart(options) {
    drawAxis(options); //繪制坐標(biāo)軸
    drawYLabels(options); //繪制y軸坐標(biāo)
    drawXLabels(options); //繪制x軸坐標(biāo)
    //drawData(options);//繪制柱狀圖
    drawDataGradient(options);//繪制漸變色柱狀圖
}

/**
 * 繪制坐標(biāo)軸
 */
function drawAxis(options) {
    let chartZone = options.chartZone;
    context.strokeWidth = 4;
    context.strokeStyle = "#353535";
    context.moveTo(chartZone[0],chartZone[1]);
    context.lineTo(chartZone[0],chartZone[3]); //y軸總高從50到700
    context.lineTo(chartZone[2],chartZone[3]); //x軸總長從50到1000
    context.stroke();
}

/**
 * 繪制y軸坐標(biāo)
 */
function drawYLabels(options) {
    let labels = options.yAxisLabel;
    let yLength = (options.chartZone[3] - options.chartZone[1])*0.98;
    let gap = yLength / (labels.length - 1);
    
    labels.forEach(function (label, index) {
        //繪制坐標(biāo)文字
        let offset = context.measureText(label).width + 20;
        context.strokeStyle = "#eaeaea";
        context.font = "16px";
        context.fillText(label, options.chartZone[0] - offset ,options.chartZone[3] - index * gap);
        //繪制小間隔
        context.beginPath();
        context.strokeStyle = "#353535";
        context.moveTo(options.chartZone[0] - 10, options.chartZone[3] - index * gap);
        context.lineTo(options.chartZone[0], options.chartZone[3] - index * gap);
        context.stroke();
        //繪制輔助線
        context.beginPath();
        context.strokeStyle = "#eaeaea";
        context.strokeWidth = 2;
        context.moveTo(options.chartZone[0], options.chartZone[3] - index * gap);
        context.lineTo(options.chartZone[2], options.chartZone[3] - index * gap);
        context.stroke();
    });
}

/**
 * 繪制x軸坐標(biāo)
 */
function drawXLabels(options) {
    let labels = options.xAxisLabel;
    let xLength = (options.chartZone[2] - options.chartZone[0])*0.96;
    let gap = xLength / labels.length;
    
    labels.forEach(function (label, index) {
        //繪制坐標(biāo)文字
        let offset = context.measureText(label).width;
        context.strokeStyle = "#eaeaea";
        context.font = "18px";
        context.fillText(label, options.chartZone[0] + (index + 1) * gap - offset ,options.chartZone[3] + 20);
        //繪制小間隔
        context.beginPath();
        context.strokeStyle = "#353535";
        context.moveTo(options.chartZone[0] + (index + 1) * gap - offset / 2 ,options.chartZone[3]);
        context.lineTo(options.chartZone[0] + (index + 1) * gap - offset / 2,options.chartZone[3]+5);
        context.stroke();
        //存儲偏移量
        options.offsetXLabel = offset / 2;
    });
}

/**
 * 繪制數(shù)據(jù)
 */
function drawData(options) {
    let data = options.data;
    let xLength = (options.chartZone[2] - options.chartZone[0])*0.96;
    let yLength = (options.chartZone[3] - options.chartZone[1])*0.98;
    let gap = xLength / options.xAxisLabel.length;

    //繪制矩形
    data.forEach(function (item, index) {
        context.fillStyle = options.barStyle.color || "#1abc9c"; //02BAD4
        let x0 = options.chartZone[0] + (index + 1) * gap - options.barStyle.width / 2 - options.offsetXLabel;
        let height = item / options.yMax * (options.chartZone[3] - options.chartZone[1])*0.98;
        let y0 =  options.chartZone[3] - height;
        let width = options.barStyle.width;
        context.fillRect(x0,y0,width,height);
    });
}

/**
 * 繪制線性漸變色柱狀圖
 */
function drawDataGradient(options) {
    let data = options.data;
    let xLength = (options.chartZone[2] - options.chartZone[0])*0.96;
    let yLength = (options.chartZone[3] - options.chartZone[1])*0.98;
    let gap = xLength / options.xAxisLabel.length;
    //創(chuàng)建漸變色
    let fillStyleGradient = context.createLinearGradient(50,50,50,700);
        fillStyleGradient.addColorStop(0, options.barStyle.color);
        fillStyleGradient.addColorStop(1, "rgba(1,176,241,0.6)");

    //繪制矩形
    data.forEach(function (item, index) {
        context.fillStyle = fillStyleGradient;
        let x0 = options.chartZone[0] + (index + 1) * gap - options.barStyle.width / 2 - options.offsetXLabel;
        let height = item / options.yMax * (options.chartZone[3] - options.chartZone[1])*0.98;
        let y0 =  options.chartZone[3] - height;
        let width = options.barStyle.width;
        context.fillRect(x0,y0,width,height);
    });
}

瀏覽器中可查看效果:

四. 思考題

如果希望在坐標(biāo)軸末端加一個箭頭,需要怎么做呢?

/*x軸箭頭示例*/
//1.options中增加箭頭顏色和大小的設(shè)置
let options = {
    //...
    axisArrow:{
            size:2,
            color:"#DA5961"
        }
}
//箭頭繪制函數(shù)

/**
 * x軸繪制箭頭
 */
function drawArrow(options) {
    let factor = options.axisArrow.size;//獲取箭頭大小因子
    context.save();//保存當(dāng)前設(shè)置的繪圖上下文
    context.translate(options.chartZone[2], options.chartZone[3]);//移動坐標(biāo)系原點至x軸末端
    context.beginPath();//開始繪制箭頭
    context.moveTo(0,0);//移動至新原點
    context.lineTo(2 * factor,-3 * factor);
    context.lineTo(10 * factor,0);
    context.lineTo(2 * factor, 3 * factor);
    context.lineTo(0,0);
    context.globalAlpha = 0.7; //設(shè)置填充色透明度
    context.fillStyle = options.axisArrow.color;//獲取箭頭顏色
    context.fill();//填充箭頭路徑
    context.restore();//恢復(fù)繪圖上下文樣式設(shè)置
}        

箭頭效果:

y軸的箭頭請自行完成即可。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/104319.html

相關(guān)文章

發(fā)表評論

0條評論

sarva

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<