HTML5 + JavaScript绘制饼图+1

简介: HTML5 + JavaScript绘制饼图+1

利用难得的假期继续改进“圳品”信息系统,在另一个模块中使用之前的Pie()代码编制饼图。

代码详见:改进“圳品”信息系统网页上的饼图:加标题+文字说明换行显示

却发现饼图标题没显示出来。

研究源代码后发现问题出在计算饼图标题输出位置的代码上,即:

this.ctx.fillText(this.title, (tcCanvas.width - this.title.length*fontSize)/2, 25);

在这里我们使用了tcCanvas.width,由于tcCanvas是一个实例,在另一个实例中调用这段代码就会出现问题。

改进的代码是把Canvas的width作为一个属性传递进来,即分为两步。

第一步,增加width属性:

var typeCountPie = new Pie({
    ctx: ctx,
    x: tcCanvas.width / 2,
    y: tcCanvas.height / 2,
    r: 150,
  width:tcCanvas.width,//增加width属性
  data:aTypeCount,
    title:'“圳品”产品类别分析'
  });

第2步,修改计算饼图标题输出位置的代码,改为:

this.ctx.fillText(this.title, (this.width - this.title.length*fontSize)/2, 25);


也就是用this.width 替代 了 tcCanvas.width,增强了代码通用性。完整的代码如下:

<!DOCTYPE html>
<html>
 <head>
  <meta charset="UTF-8">
  <meta name="Author" content="PurpleEndurer">
  <meta name="Keywords" content="">
  <meta name="Description" content="">
  <title>“圳品”信息系统</title>
 </head>
 <body>
 <script>
      const aType = [
    "水果",//0
    "粮食",//1,谷类、豆类、薯类
    "食用油",//2
    "饮用水",//3
    "畜",//4
    "禽",//5
    "渔",//6
    "其它"//7
    ];
 
    //结构:
    //类别,数量,颜色
    var aTypeCount = [
    [0, 3, "purple"],
    [1, 5, "gray"],
    [2,3, 'yellow'],
    [3,4,"green"],
    [4,8,"red"]
    ]; 
  </script> 
 <canvas id="typeCountCanvas" style="border:1px solid red"></canvas>
 <div>
 <textarea border="1" id="taDebug" cols="80" rows="15">debug:</textarea>
 </div>
 
 <script>
    var tcCanvas = document.getElementById("typeCountCanvas");
    //var w = window.innerWidth;
    //var h = window.innerHeight;
    tcCanvas.height = 480;
    tcCanvas.width = 640;
    var ctx = tcCanvas.getContext('2d');
 
    var taDbg = document.getElementById("taDebug");
 
  function Pie(obj)
  {
 
    for(var key in obj)
    {
      this[key] = obj[key];
    }
    this.init();
    this.render(this.slice);
    this.drawText();
  }
 
  Pie.prototype = {
    init: function () {
     this.start = 0; 
      //1、累计数据求合
      var sum = 0;
      this.data.forEach(function (v) {
        sum += v[1];
      });
      //2、计算每一个数据所占的比重
      this.slice = this.data.map(function (v) {
        var obj = {};
        obj.number = v[1];
        obj.ratio = v[1] / sum;//每个数据占据的比重
        obj.radian = 2 * Math.PI * v[1] / sum;//该扇形所占据的弧度
        obj.start = this.start;
        obj.end = this.start + obj.radian;
        this.start = obj.end;
        obj.color = v[2];
        obj.text = aType[v[0]] + "类产品"+ v[1] + "个,占比" + Math.ceil(obj.ratio*100) + "%";
        return obj;
      },this);
    },
  
    //渲染页面(画饼图)
    render: function (obj) {
      //3、计算每一个扇形的起始弧度和结束弧度
      this.slice.forEach(function (v, i) {
        var obj = {};
        //第一个扇形的起始弧度:start 结束:start+第一个扇形占据的弧度差
        obj.start = this.start;
        obj.end = this.start + v.radian;
        this.start += v.radian;
        //绘制扇形
        this.ctx.beginPath();
        this.ctx.moveTo(this.x, this.y);
        this.ctx.arc(this.x, this.y, this.r, obj.start, obj.end);
        this.ctx.fillStyle = v.color;
        this.ctx.fill();
      }, this);
    },
 
    //输出文字说明
    drawText:function(){
 
  var fontSize = 20;//标题文字大小
 
  //如果标题属性已赋值且长度大于0,则输出标题文件
  if (typeof(this.title) != "undefined" && 0 < this.title.length)
  {
    this.ctx.font = "bolder "+ fontSize + "px 微软雅黑";
    this.ctx.fillStyle = 'black';
    this.ctx.fillText(this.title, (this.width - this.title.length*fontSize)/2, 25);
  }
 
      fontSize= 16;
      this.ctx.font = fontSize+"px 微软雅黑";
      var textWidth= 100;
      this.slice.forEach(function(obj){
      this.ctx.fillStyle = obj.color;// 'black';
 
        //计算文字所在的弧度
        r2 = obj.start + obj.radian/2;
 
        //计算相对于圆心文字偏移的位置
        b = this.r * Math.cos(r2) ;
        h = this.r * Math.sin(r2);
 
        //文字的x坐标位置
     var x2 = this.x + b;
        if (x2 <= this.x)
        {
            //在圆心的左侧
     x2 -= textWidth+10;
        }
 
        //文字的y坐标位置
        var y2 = this.y + h;
        if (y2 >= this.y)
        {
            y2 += fontSize; 
        }
        else
        {
            //在圆心的上方
            y2 -= fontSize;
        }
 
       //this.ctx.fillText(obj.text, x2, y2);//在一行输出
        taDbg.value += "\nthis.r=" + this.r + "  obj.text= " + obj.text + " x2=" + x2 + " y2=" + y2;
 
  //换行输出方法1
  var charNumPerLine = Math.ceil(textWidth / fontSize);
  var t = obj.text, i = 0;
  while (t.length > charNumPerLine)
  {
    this.ctx.fillText(t.substr(i,charNumPerLine), x2,y2);//绘制截取部分
    i += charNumPerLine;
    t = t.substring(i);
    y2 += fontSize;
  }
  if (i <= obj.text.length)
  {
    this.ctx.fillText(t, x2,y2);//绘制剩余部分
  }
 
/*
        //换行输出方法2
  var i, lastSubStrIndex = lineWidth=0;
  for (i=0;i< obj.text.length; i++)
  {
          taDbg.value += '\nobj.text['+i +']=' + obj.text[i];
    lineWidth += ctx.measureText(obj.text[i]).width; 
    if (lineWidth > textWidth)
    {  
      this.ctx.fillText(obj.text.substring(lastSubStrIndex,i), x2,y2);//绘制截取部分
      y2 +=  fontSize;
      lineWidth = 0;
      lastSubStrIndex = i;
    } 
    if(i==obj.text.length-1)
    { //绘制剩余部分
      this.ctx.fillText(obj.text.substring(lastSubStrIndex,i+1),x2,y2);
      break;
    }
  }//for*/        
      },this);
    }
  };
 
 var typeCountPie = new Pie({
    ctx: ctx,
    x: tcCanvas.width / 2,
    y: tcCanvas.height / 2,
    r: 150,
  width:tcCanvas.width,//增加width属性
  data:aTypeCount,
    title:'“圳品”产品类别分析'
  });
 
 </script>
 </body>
</html>

image.png

相关文章
|
3月前
|
存储 JavaScript 前端开发
用 HTML + JavaScript DIY 渐进式延迟法定退休年龄测算器
用 HTML + JavaScript DIY 渐进式延迟法定退休年龄测算器
|
3月前
|
JavaScript 前端开发
用html+javascript打造公文一键排版系统12:删除附件说明中“附件:”里的空格
用html+javascript打造公文一键排版系统12:删除附件说明中“附件:”里的空格
|
3月前
|
前端开发
用html+javascript打造公文一键排版系统3:获取参数设置、公文标题排版
用html+javascript打造公文一键排版系统3:获取参数设置、公文标题排版
用html+javascript打造公文一键排版系统1:设计界面
用html+javascript打造公文一键排版系统1:设计界面
|
14天前
|
Web App开发 移动开发 HTML5
html5 + Three.js 3D风雪封印在棱镜中的梅花鹿动效源码
html5 + Three.js 3D风雪封印在棱镜中的梅花鹿动效源码。画面中心是悬浮于空的梅花鹿,其四周由白色线段组成了一个6边形将中心的梅花鹿包裹其中。四周漂浮的白雪随着多边形的转动而同步旋转。建议使用支持HTML5与css3效果较好的火狐(Firefox)或谷歌(Chrome)等浏览器预览本源码。
47 2
|
1月前
|
JavaScript
JS鼠标框选并删除HTML源码
这是一个js鼠标框选效果,可实现鼠标右击出现框选效果的功能。右击鼠标可拖拽框选元素,向下拖拽可实现删除效果,简单实用,欢迎下载
42 4
|
1月前
|
移动开发 HTML5
html5+three.js公路开车小游戏源码
html5公路开车小游戏是一款html5基于three.js制作的汽车开车小游戏源代码,在公路上开车网页小游戏源代码。
58 0
html5+three.js公路开车小游戏源码
|
1月前
|
JSON 移动开发 数据格式
html5+css3+js移动端带歌词音乐播放器代码
音乐播放器特效是一款html5+css3+js制作的手机移动端音乐播放器代码,带歌词显示。包括支持单曲循环,歌词显示,歌曲搜索,音量控制,列表循环等功能。利用json获取音乐歌单和歌词,基于html5 audio属性手机音乐播放器代码。
110 6
|
3月前
|
前端开发 JavaScript
HTML+JavaScript+CSS DIY 分隔条splitter
HTML+JavaScript+CSS DIY 分隔条splitter
|
3月前
|
小程序 JavaScript 前端开发
你的生日是星期几?HTML+JavaScript帮你列出来
你的生日是星期几?HTML+JavaScript帮你列出来
下一篇
DataWorks