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

来源:https://www.cnblogs.com/dashnowords/archive/2019/03/10/10506921.html
-Advertisement-
Play Games

示例代碼托管在: "http://www.github.com/dashnowords/blogs" 博客園地址: "《大史住在大前端》原創博文目錄" 華為雲社區地址: "【你要的前端打怪升級指南】" [TOC] 一. 任務說明 使用原生 繪製柱狀圖。(柱狀圖截圖來自於百度Echarts官方示例庫 ...


目錄

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

博客園地址:《大史住在大前端》原創博文目錄

華為雲社區地址:【你要的前端打怪升級指南】

一. 任務說明

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

二. 重點提示

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

三. 示例代碼

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

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

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

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

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

/**
 * 繪製坐標軸
 */
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軸坐標
 */
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) {
        //繪製坐標文字
        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軸坐標
 */
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) {
        //繪製坐標文字
        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;
    });
}

/**
 * 繪製數據
 */
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;
    //創建漸變色
    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);
    });
}

瀏覽器中可查看效果:

四. 思考題

如果希望在坐標軸末端加一個箭頭,需要怎麼做呢?

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

/**
 * x軸繪製箭頭
 */
function drawArrow(options) {
    let factor = options.axisArrow.size;//獲取箭頭大小因數
    context.save();//保存當前設置的繪圖上下文
    context.translate(options.chartZone[2], options.chartZone[3]);//移動坐標系原點至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; //設置填充色透明度
    context.fillStyle = options.axisArrow.color;//獲取箭頭顏色
    context.fill();//填充箭頭路徑
    context.restore();//恢復繪圖上下文樣式設置
}        

箭頭效果:

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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 我們很榮幸地向大家發佈Wonder 1.0正式版-----WebGL 3D引擎和編輯器! ...
  • 續前文: "手工翻譯Vue.js源碼第一步:14個文件重命名" 對core/instance/索引中的變數, 方法進行重命名如下(題圖): javascript import { 混入初始化 } from './初始化' import { 混入狀態 } from './狀態' import { 混入 ...
  • 前言 正所謂金三銀四五吃土,因此這些天幾個前端技術群討論最多的話題就是面試題了。某日群內討論一道面試題:“前端如何實現大量圖片預載入以及預載入進度動畫進度條顯示?” 恰巧本人以前jquery、zepto一把梭快樂切圖的時候,寫過不少形式的圖片預載入動畫。對於圖片預載入的原理稍微瞭解接觸過那麼一點點, ...
  • 首先, 在兩天時間內安裝數破百, 多謝支持. VS Code插件市場地址: "英漢詞典 Visual Studio Marketplace" 開源庫地址同前文: "Visual Studio Code插件 英漢詞典初版發佈" 查詢單詞功能基本不變, 在詳細信息的開頭添加了原詞: 如題圖, 支持駝峰命 ...
  • 1.超引用:(...) 用於取出參數對象中的所有可遍歷屬性,拷貝到當前對象之中 2.解構賦值: 賦值: 解構: 3. for of迴圈: es5的for迴圈: for in 迴圈: for of 迴圈:(不能用來遍歷json) 4.Map對象 map對象是因為for of 的出現才出現的,一個map ...
  • webpack 用於編譯 JavaScript 模塊。一旦完成安裝,就可以通過 webpack 的 CLI 或 API 與其配合交互。 首先創建一個目錄,進入目錄中,初始化npm 然後本地安裝webpack,webpack-cli ,lodash 目錄結構: index.html webpack.c ...
  • 今天主要來講下前端的數據存儲,說起數據存儲,大家肯定第一時間想起cookie,localstorage,sessionstorage,而其實還有userData和IndexedDB這兩種數據存儲,接下來將對它們進行一個比較詳細的總結 一、為什麼要進行數據存儲 隨著Web應用程式的出現,慢慢的也開始產 ...
  • 一、escape和它們不是同一類 簡單來說,escape是對字元串(string)進行編碼(而另外兩種是對URL),作用是讓它們在所有電腦上可讀。編碼之後的效果是%XX或者%uXXXX這種形式。其中 ASCII字母、數字、@*/+ ,這幾個字元不會被編碼,其餘的都會。最關鍵的是,當你需要對URL編碼 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...