15

使用 d3 图形库,我似乎无法使路径绘制缓慢,因此可以看到它们在增长。

该站点在“折线图(展开)”部分中有一个完美的示例,但没有为该部分提供代码。有人可以帮我解决可能实现这一目标的 D3 代码行吗?

当我尝试在下面的代码片段中附加 delay() 或 duration() 时,路径仍会立即绘制,并且此段之后的所有 SVG 代码都无法呈现。

    var mpath = svg.append ('path');
        mpath.attr ('d', 'M35 48 L22 48 L22 35 L22 22 L35 22 L35 35 L48 35 L48 48')
             .attr ('fill', 'none')
             .attr ('stroke', 'blue')
             .duration (1000);
4

3 回答 3

49

在 svg 中为线条设置动画时的一种常见模式是设置stroke-dasharray路径长度的 a ,然后设置 animate stroke-dashoffset

var totalLength = path.node().getTotalLength();

path
  .attr("stroke-dasharray", totalLength + " " + totalLength)
  .attr("stroke-dashoffset", totalLength)
  .transition()
    .duration(2000)
    .ease("linear")
    .attr("stroke-dashoffset", 0);

你可以在这里看到一个演示:http: //bl.ocks.org/4063326

于 2012-11-13T01:42:29.773 回答
18

我相信执行此操作的“D3 方式”是使用自定义补间函数。你可以在这里看到一个有效的实现:http: //jsfiddle.net/nrabinowitz/XytnD/

这假设您有一个名为 setup 的生成器lined3.svg.line计算路径:

// add element and transition in
var path = svg.append('path')
    .attr('class', 'line')
    .attr('d', line(data[0]))
  .transition()
    .duration(1000)
    .attrTween('d', pathTween);

function pathTween() {
    var interpolate = d3.scale.quantile()
            .domain([0,1])
            .range(d3.range(1, data.length + 1));
    return function(t) {
        return line(data.slice(0, interpolate(t)));
    };
}​

这里的pathTween函数返回一个插值器,它采用给定的线切片,由我们通过转换的距离定义,并相应地更新路径。

不过,值得注意的是,我怀疑通过采用简单的方法可以获得更好的性能和更流畅的动画:clipPath在线条上放置一个白色矩形(如果你的背景很简单)或一个(如果你的背景很复杂),然后将其过渡到右侧以显示下面的线。

于 2012-11-13T01:35:40.867 回答
1

根据您链接到的帖子,我想出了以下示例:

var i = 0,
    svg = d3.select("#main");

String.prototype.repeat = function(times) {
   return (new Array(times + 1)).join(this);
}

segments = [{x:35, y: 48}, {x: 22, y: 48}, {x: 22, y: 35}, {x: 34, y:35}, {x: 34, y:60}];
line = "M"+segments[0].x + " " + segments[0].y

new_line = line + (" L" + segments[0].x + " " + segments[0].y).repeat(segments.length);
 var mpath = svg.append ('path').attr ('d',new_line )
             .attr ('fill', 'none')
             .attr ('stroke', 'blue')

for (i=0; i<segments.length; i++)
    {
    new_segment = " " + "L"+segments[i].x + " " + segments[i].y
    new_line = line + new_segment.repeat(segments.length-i)
    mpath.transition().attr('d',new_line).duration(1000).delay(i*1000);
    line = line + new_segment

    }

这有点难看,但有效。你可以在jsFiddle上看到它

于 2012-11-13T01:11:16.617 回答