d3代码如何改造成update结构(恰当处理enter和exit)

d3的enter和exit

网上有很多blog讲解。说的还凑合的见:https://blog.csdn.net/nicolecc/article/details/50786661

如何把自己的rude绘图代码,进行精致化(update)

 不多比比,上代码示例:

d3.selectAll('.circle_group').children().remove();
var circle_group = d3.selectAll('.circle_group')
    .data(circleData)
    .enter().append('g')
    .attr('class', 'circle_group brushNode')
    .attr('transform',
        function (d) {
            return 'translate(' + d.x + ',' + d.y + ')';
        })
    .on('click', function (d) {
        if (shiftKey) {
            //标记当前点
            d3.select(this).classed("selected", function (d) {

                d.selected = !d.selected;
                d.previouslySelected = !d.selected;
                return d.selected;
            })

            //阻止事件冒泡
            d3.event.stopPropagation();
        }
    })
    .on("contextmenu", function (node) {
        contextmenu("Rmenu");
    })

circle_group.append('circle')
    .attr('r', function (d) {
        return d.r;
    })
    .style("fill", function (d) {
        return color20(d.index);
    })

;
circle_group.append('text')
    .attr("dy", ".35em")
    .attr("text-anchor", "middle")//在圆圈中加上数据
    .style('fill', function (node) {
        return '#555';
    })
    .attr("y", -7)
    .text(d => d.text);

circle_group.call(d3.drag()
//定义了元素拖拽行为的原点,设置为圆的圆心位置可以避免明显的元素跳动, 与d3v3中的origin方法类似。
    .subject(function () {
        var thisData = d3.select(this);
        return {
            x: thisData.datum().x,
            y: thisData.datum().y
        };
    })
    .on("start", dragstarted)
    .on("drag", dragmove)
    .on("end", dragended));

 很明显,新手图省事都是这么绘图的。就绘图结果来看,如果你不加动画一点问题都没有,只要一加动画过渡动画,所有的图形都是从无到有的过程,而我们想看的是,如果点更新的话,能看到他从哪(位置)更新到哪(位置)。(这里就不加动画过渡了,就一行代码的事.transition().duration(300) )。

改造如下:

    //这部分代码是有则改之,无则添加的功能
const circle_data=d3.selectAll('.circle_group').data(circleData)  //更新部分,如果你数据的数目没变,那circle_data.size()=你数据内容改变的数目,你可以把circle_data考虑成update部分就行,这样编代码准没错
    .attr('transform',             //首次运行的时候,因为没有元素circle_data.size()=0, 所以这个transform不会运行到
        function (d) {
            return 'translate(' + d.x + ',' + d.y + ')';
        });
const circle_enter=circle_data.enter().append('g')  //add 部分,首次运行的时候,circle_enter.size()=全部元素,所以在circle_enter进行所有的初始化设置操作
    .attr('class', 'circle_group brushNode')
    .attr('transform',
        function (d) {
            return 'translate(' + d.x + ',' + d.y + ')';
        });
circle_data.exit().remove();  // 当你删一些数据的时候, .exit().size()>0



circle_enter.on('click', function (d) {
    if (shiftKey) {
        //标记当前点
        d3.select(this).classed("selected", function (d) {

            d.selected = !d.selected;
            d.previouslySelected = !d.selected;
            return d.selected;
        })

        //阻止事件冒泡
        d3.event.stopPropagation();
    }
})
    .on("contextmenu", function (node) {
        contextmenu("Rmenu");
    })

circle_enter.append('circle')     //add
    .attr('r', d=>d.r)
    .style("fill", function (d) {
        return color20(d.index);
    });
circle_data.select('circle')  //update,这里不要append ,因为元素已经在那了(enter().append()过了)。当然首次运行(视图首次显示)的时候,这几句代码是运行不到的。
    .attr('r', d=>d.r)
    .style("fill", function (d) {
        return color20(d.index);
    });

circle_enter.append('text')  //add
    .attr("dy", ".35em")
    .attr("text-anchor", "middle")//在圆圈中加上数据
    .style('fill', function (node) {
        return '#555';
    })
    .attr("y", -7)
    .text(d => d.text);
circle_data.select('text')  //update  不要append 
    .text(d => d.text);

const drag=d3.drag()
//定义了元素拖拽行为的原点,设置为圆的圆心位置可以避免明显的元素跳动, 与d3v3中的origin方法类似。
    .subject(function () {
        var thisData = d3.select(this);
        return {
            x: thisData.datum().x,
            y: thisData.datum().y
        };
    })
    .on("start", dragstarted)
    .on("drag", dragmove)
    .on("end", dragended);
circle_enter.call(drag);
circle_data.call(drag);   //update 注意,这里一定要重新绑定一下,这里涉及到drag的初始化(subject用于初始化drag拖动点的初始位置)

 

敲黑板

1、更新部分的所有与位置有关的事件(比如d3.drag()的初始位置)要重新绑定,否则会出现不可预料的结果。

2、update部分与数据有关的attr,style要重新设置,这时就不用append了。(因为这个元素既然有了,你已经在之前的.enter().append().append(其他元素)添加过了,这里只需要更新一下即可。)。一些固定的attr,style就不用重复设置了(之前enter已经绑定过了)

posted @ 2019-06-10 03:24  xunhanliu  阅读(420)  评论(0编辑  收藏  举报