APICloud d3.js 画饼图带动画效果

学习文献:http://pkuwwt.github.io/d3-tutorial-cn/adding-elements.html (基础教学)
https://juejin.im/post/5a81a8946fb9a06334266f05 (讲解插值)
https://github.com/pshrmn/notes/blob/master/d3/interpolators.md (解释插值)
今天要学习总结下d3画饼图,先看效果图

饼图.gif

上面是饼图带点击动画的,然后中间的数字可以变化。这里面主要用到d3来实现效果。后面我会贴出我全部的代码

思路流程

创建SVG--->添加g标签元素--->数据转换得到我们需要的数据--->添加path开启动画和画圆--->点击圆让其变大--->数字慢慢增长

创建SVG
    var svg = d3.select('#d3Circle')
                .append("svg")
                .attr("width", sellCircleSVGWid)
                .attr("height", sellCircleSVGhei);

.select表示要选择的函数,append创建SVG,attr用来设置属性值,我这里设置了宽度和高度。我们创建的SVG是一个矢量图的格式可无限放大而不失真。

添加g标签
  var arcs = svg.selectAll("g")
                .data(piedata)//设置数据集,piedata元素集合
                .enter()
                .append("g")//添加g元素
                .attr("transform", "translate(" + sellCircleSVGWid / 2 + "," + sellCircleSVGhei / 2 + ")");

我通过SVG再创建一个g标签,并且让它移动到圆的中心位置
selectAll表示选中所有的g标签元素,但是我们在此之前并没有创建g标签元素,所以通过selectAll会返回一个空的集合给我。data用于设置数据源,piedata集合里面有多少数据,就会创建多少个g标签元素给我。enter()表示创建,append表示添加。
注意!!! piedata 必须是集合参数(我自己总结出来的,我试过几次了,不是集合就不会画图出来)
我们最好是把g标签创建出来,因为g是用来组合对象的容器。添加到g元素上的变换会应用到其所有的子元素上。添加到g元素的属性会被其所有的子元素继承。此外,g元素也可以用来定义复杂的对象。

数据转换得到我们需要的数据
       //创建圆弧生成器
        var arcPath = d3.svg.arc()
                .innerRadius(innerRaidus)
                .outerRadius(outerRadius);
       //创建圆弧生成器
        var arcPath2 = d3.svg.arc()
                .innerRadius(innerRaidus)
                .outerRadius(outerRadius + 5);

        var color = ['#c56b4f', '#e8b63e', '#435fd8'];
        var ptDetail = ["456", "123", "256"];//数据源
        var totalSales = "835";
      //这里数据过滤,拿到需要的数据
        var pie = d3.layout.pie().value(function (d) {
            return d - 0;//将字符串转为数字
        });
        var dataset = [];//数据集合
        for (var k in ptDetail) {
            dataset.push(ptDetail[k] / totalSales);//算出占用量
        }
        dataset.sort(function (a, b) {//从大到小排序
            return a < b ? 1 : -1
        });
        var piedata = pie(dataset);//拿到需要的数据
        //先定义位置
        var sum = 0;
        piedata.forEach(function (d, i) {//定义自己的颜色,执行实时间等
            d.color = color[i];
            d.position = i;
            d.duration = 2000 * (d.data / d3.sum(dataset));//动画时长2秒,计算每一个弧形所用动画时间
            d.delaytime = sum;
            sum += d.duration;
        });

var arcPath = d3.svg.arc().innerRadius(innerRaidus).outerRadius(outerRadius); 创建一个弧生成器,我们把数据传递给它,像这样arcPath(data),它就会返回一堆路径数据给你,这堆数据用于path上面画圆。
d3.layout.pie().value 主要用于数据过滤,然后内部计算得到开始角度,结束角度


piedata数据转换后的值.png

这里面的data 是原始的数据,startAngle是开始弧度,endAngle是结束弧度。dataset是未进行过滤的数据源

添加path开启动画和画圆
      arcs.append("path")
                .transition()
                .delay(function (d, i) {
                    return d.delaytime;
                })
                .duration(function (d, i) {
                    return d.duration;
                })
                .attrTween("d", function (d, j) {//过度器
                    var i = d3.interpolate(d.startAngle, d.endAngle);
                    return function (t) {
                        d.endAngle = i(t);
                        return arcPath(d);
                    }
                })
                .attr("stroke-width", "2px")
                .attr("fill", function (d, i) {
                    return d.color;
                });

transition()用于开启动画的标志,delay()多少时间后开始执行,duration()多少时间内完成,attrTween()过渡器,这个比较难立即了吧,举一个简单的例子说明

//下面我用书上的例子来展示
        var svg = d3.select('#d3Circle')
                .append("svg")
                .attr("width", 200)
                .attr("height", 180);
   //定义SVG的宽高
        var svg = d3.select('#d3Circle')
                .append("svg")
                .attr("width", sellCircleSVGWid)
                .attr("height", sellCircleSVGhei);

        var rect = svg.append("rect")
                .attr("fill", "steelblue")
                .attr("x", 10)
                .attr("y", 10)
                .attr("width", 100)
                .attr("height", 30);
        var rectTran = rect.transition()
                .duration(2000)
                .attrTween("width", function (d, i, a) {
                    return function (t) {
                        return Number(a) + t * 300;
                    }
                });

attrTween动画.gif

所以我们可以理解为,数字从0到1,慢慢变化的过度。继续上面的讲解
var i = d3.interpolate(d.startAngle, d.endAngle);在d3里面,interpolate解释为插值器的意思,我上面贴的链接也有对这个进行解释,然后我对它的理解如下


interpolate的理解.png

t的取值为[0,1],我们i就是作为一个函数了,如果我们duration时间很长那么通过i(t)得到的数据也会很多,所以我们过度动画就从0读到endAngle变化了。arcPath(d)拿到数据函数将其转换为路径值,然后赋值给d参数。

点击圆让其变大
 var isClick = false
        arcs.on("click", function (d) {//点击
            d3.select(this).select("path").transition().attr("d", function (d) {
                if (!isClick) {//放大效果
                    isClick = true;
                    return arcPath2(d);
                } else {//还原
                    isClick = false;
                    return arcPath(d);
                }
            });
        });

由于我上面写了2个arcPath,一个用于点击后放大的效果的。

数字慢慢增长
 function CusnumDd(el, param) {
        var sum = 0;
        var time = setInterval(function () {
            sum += 1;
            el.innerHTML = sum;
            if (param <= sum) {
                clearInterval(time);
                el.innerHTML = (param);
            }
        }, 100)
    }
全部代码
<script type="text/javascript">
    $(function () {
        var sellCircleSVGWid = 200;//宽
        var sellCircleSVGhei = 180;//高
        var innerRaidus = 60;
        var outerRadius = 80;

        //定义SVG的宽高
        var svg = d3.select('#d3Circle')
                .append("svg")
                .attr("width", sellCircleSVGWid)
                .attr("height", sellCircleSVGhei);

        /* var rect = svg.append("rect")
         .attr("fill", "steelblue")
         .attr("x", 10)
         .attr("y", 10)
         .attr("width", 100)
         .attr("height", 30);
         var rectTran = rect.transition()
         .duration(2000)
         .attrTween("width", function (d, i, a) {
         return function (t) {
         return Number(a) + t * 300;
         }
         });*/

        //画圆形图开始
        var arcPath = d3.svg.arc()
                .innerRadius(innerRaidus)
                .outerRadius(outerRadius);
        //画圆形图开始
        var arcPath2 = d3.svg.arc()
                .innerRadius(innerRaidus)
                .outerRadius(outerRadius + 5);

        var color = ['#c56b4f', '#e8b63e', '#435fd8'];
        var ptDetail = ["456", "123", "256"];
        var totalSales = "835";
        var pie = d3.layout.pie().value(function (d) {
            return d - 0;
        });
        var dataset = [];//数据集合
        for (var k in ptDetail) {
            dataset.push(ptDetail[k] / totalSales);
        }
        dataset.sort(function (a, b) {
            return a < b ? 1 : -1
        });//从大到小排序
        var piedata = pie(dataset);//元素集合
        //先定义位置
        var sum = 0;
        piedata.forEach(function (d, i) {
            d.color = color[i];
            d.position = i;
            d.duration = 2000 * (d.data / d3.sum(dataset));//动画时长2秒,计算每一个弧形所用动画时间
            d.delaytime = sum;
            sum += d.duration;
        });

        var arcs = svg.selectAll("g")
                .data(piedata)
                .enter()
                .append("g")
                .attr("transform", "translate(" + sellCircleSVGWid / 2 + "," + sellCircleSVGhei / 2 + ")");
        arcs.append("path")
                .transition()
                .delay(function (d, i) {
                    return d.delaytime;
                })
                .duration(function (d, i) {
                    return d.duration;
                })
                .attrTween("d", function (d, j) {
                    var i = d3.interpolate(d.startAngle, d.endAngle);
                    return function (t) {
                        d.endAngle = i(t);
                        return arcPath(d);
                    }
                })
                .attr("stroke-width", "2px")
                .attr("fill", function (d, i) {
                    return d.color;
                });

        var isClick = false
        arcs.on("click", function (d) {//点击
            d3.select(this).select("path").transition().attr("d", function (d) {
                if (!isClick) {
                    isClick = true;
                    return arcPath2(d);
                } else {
                    isClick = false;
                    return arcPath(d);
                }
            });
        });


        var s = document.getElementById("number");
        del_ff(s);    //清理空格
        var el = s.firstChild;   //获得s的第一个子节点
        CusnumDd(el, 8);


        var a = document.getElementById("dom");
        del_ff(a);
        console.log('获取a的全部子节点:')
        console.log(a.childNodes);//获取a的全部子节点;
        console.log('获取a的父节点:')
        console.log(a.parentNode); //获取a的父节点;
        console.log('获取a的下一个兄弟节点:')
        console.log(a.nextSibling);//获取a的下一个兄弟节点
        console.log('获取a的上一个兄弟节点:')
        console.log(a.previousSibling);//获取a的上一个兄弟节点
        console.log('获取a的第一个子节点:')
        console.log(a.firstChild);//获取a的第一个子节点
        console.log('获取a的最后一个子节点:')
        console.log(a.lastChild);//获取a的最后一个子节点*/
    });

    function del_ff(elem) {
        var elem_child = elem.childNodes;
        for (var i = 0; i < elem_child.length;) {
            if (elem_child[i].nodeName == "#text" && !/\s/.test(elem_child.nodeValue)) {
                elem.removeChild(elem_child[i])
            }
            else {
                i++;
            }
        }
    }

    /**
     * 功能: 数字滚动显示
     * @param el 选择对象
     * @param param 数字参数
     */
    function CusnumDd(el, param) {
        var sum = 0;
        var time = setInterval(function () {
            sum += 1;
            el.innerHTML = sum;
            if (param <= sum) {
                clearInterval(time);
                el.innerHTML = (param);
            }
        }, 100)
    }

</script>
html
<div id="container">
    <div class="circle" id="d3Circle">
        <div id="number">
            <span></span>
        </div>
    </div>

    <div id="dom">
        <div></div>
        <div></div>
        <div></div>

    </div>
</div>
css
<style type="text/css">
        body, html, #allmap {
            width: 100%;
            height: 100%;
            overflow: hidden;
            margin: 0;
            font-family: "微软雅黑";
        }

        #container {
            width: 100%;
            height: 500px;
            display: flex;
            display: -webkit-flex;
            flex-direction: column;
            align-items: center;
        }

        .circle {
            width: 100%;
            height: 200px;
            display: flex;
            display: -webkit-flex;
            flex-direction: column;
            align-items: center;
            justify-content: center;
            position: relative;
        }

        #number {
            color: black;
            width: 60px;
            height: 60px;
            line-height: 60px;
            text-align: center;
            position: absolute;
        }

    </style>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352