代码之家  ›  专栏  ›  技术社区  ›  tic

如何重用D3中两个(或更多)链式转换序列

  •  4
  • tic  · 技术社区  · 6 年前

    我必须应用两个非常长的链式转换序列,它们主要在转换顺序上有所不同,我正在寻找一种紧凑的编码方式。

    a , b c , d , e , f , g , h e , , , , b c , . 我试过下面的代码,但不起作用。请注意,转换可能具有不同的属性( delay duration , ease x , y , width height cx , cy , r ,等等)和样式。例如,过渡 可以参考 ,过渡 ,过渡 c d y ,过渡 e transform f color

    附言:这个问题和我的目的是一样的 previous

    有没有什么方法可以用一种简洁的方式来编码呢?

    var t1 = d3
    .transition() // transition "a" specifications 
    ...
    .transition() // transition "b" specifications 
    ...
    .transition() // transition "c" specifications 
    ...
    .transition() // transition "d" specifications 
    ...
    ;
    var t2 = d3
    .transition() // transition "e" specifications 
    ...
    .transition() // transition "f" specifications 
    ...
    .transition() // transition "g" specifications 
    ...
    .transition() // transition "h" specifications 
    ...
    ;
    someelement1
    .transition(t1).transition(t2); 
    someelement2
    .transition(t2).transition(t1); 
    
    1 回复  |  直到 6 年前
        1
  •  5
  •   i alarmed alien    6 年前

    如评论中所述,回答这个问题的原则与你先前的问题相同。在本例中,您有一组不同的转换,这些转换可以按不同键引用的任何顺序应用。让我们将它们存储在一个对象中:

    var transitions = {
      a: function(sel){ return sel.transition().duration(1000).delay(1000).attr('cy', 200) },
      b: function(sel){ return sel.transition().duration(2000).delay(0).attr('r', 40) },
      c: function(sel){ return sel.transition().duration(500).delay(1500).attr('fill', 'red') },
      d: function(sel){ return sel.transition().duration(1500).delay(500).attr('opacity', 0.5) },
      e: function(sel){ return sel.transition().duration(1000).delay(3000).attr('cy', 300) },
      f: function(sel){ return sel.transition().duration(2000).delay(0).attr('r', 60) },
      g: function(sel){ return sel.transition().duration(500).delay(1500).attr('fill', 'magenta') },
      h: function(sel){ return sel.transition().duration(1500).delay(500).attr('opacity', 0.25) }
    };
    

    每个函数都有一个 d3.selection 对象,并对其应用特定的转换参数和转换集。这些函数可以任意长和复杂。我懒惰,没有想象力,所以他们在这个版本中只做了一个转换。

    这里已经有一些代码重复,所以让我们把选择转换成转换,并使用 this

    var transitions = {
      a: function(){ return this.duration(1000).delay(1000).attr('cy', 200) },
      b: function(){ return this.duration(2000).delay(0).attr('r', 40) },
      c: function(){ return this.duration(500).delay(1500).attr('fill', 'red') },
      d: function(){ return this.duration(1500).delay(500).attr('opacity', 0.5) },
      e: function(){ return this.duration(1000).delay(3000).attr('cy', 300) },
      f: function(){ return this.duration(2000).delay(0).attr('r', 60) },
      g: function(){ return this.duration(500).delay(1500).attr('fill', 'magenta') },
      h: function(){ return this.duration(1500).delay(500).attr('opacity', 0.25) }
    };
    

    现在我们可以通过调用

    transitions['a'].call( selection.transition() )
    transitions.f.call( d3.select('circle').transition() )
    

    要指定要应用于选定内容的变换数组,如下所示:

    apply_transitions( group.select(":nth-child(1)"), ['a','b','c','d'] );
    apply_transitions( group.select(":nth-child(2)"), ['e','f','g','h'] );
    

    /**
    * apply a series of transitions to a selection
    *
    * @param selection - d3 selection
    * @param tr_arr - array of transition identifiers, referring to functions in the `transitions` object
    */
    function apply_transitions( selection, tr_arr ) {
    
      // turn the current selection into a d3.transition
      // call the transition function referred to by the first ID in the array
      // with the d3.transition as the `this` context
      // note that the function returns a transition object, so it can be chained
      transitions[ tr_arr[0] ].call( selection.transition() )
      // add a handler to be applied at the end of the transition
        .on('end', function(){
          // if there are more transitions to be applied, call
          // apply_transitions again with tr_arr minus the first element
          // note that the `this` context in the `on` function is a DOM element,
          // so use `d3.select(this)` to turn it into a d3 selection
          if ( tr_arr.length > 1 ) {
            apply_transitions( d3.select(this), tr_arr.slice(1) );
          }
        })
    }
    

    实景示例:

        var svg = d3.select('svg').attr('width', 500).attr('height', 500);
    
        var dataSet = [20, 20];
    
        var group=svg.append("g");
        var circles = group.selectAll('circle')
        .data(dataSet)
        .enter()
        .append('circle')
        .attr("r",function(d){ return d })
        .attr("cx",function(d, i){ return i * 100 + 50 })
        .attr("cy",50)
        .attr("fill",'black');
    
        apply_transitions( group.select(":nth-child(1)"), ['a','b','c','d'] );
    
        apply_transitions( group.select(":nth-child(2)"), ['e','f','g','h'] );
    
      function apply_transitions( selection, tr_arr ) {
    
        var transitions = {
          a: function(){ return this.duration(1000).delay(1000).attr('cy', 200) },
          b: function(){ return this.duration(2000).delay(0).attr('r', 40) },
          c: function(){ return this.duration(500).delay(1500).attr('fill', 'red') },
          d: function(){ return this.duration(1500).delay(500).attr('opacity', 0.5) },
          e: function(){ return this.duration(1000).delay(3000).attr('cy', 300) },
          f: function(){ return this.duration(2000).delay(0).attr('r', 60) },
          g: function(){ return this.duration(500).delay(1500).attr('fill', 'magenta') },
          h: function(){ return this.duration(1500).delay(500).attr('opacity', 0.25) }
        };
    
          transitions[ tr_arr[0] ].call( selection.transition() )
            .on('end', function(){
              if ( tr_arr.length > 1 ) {
                apply_transitions( d3.select(this), tr_arr.slice(1) );
              }
            })
      }
    <script src="http://d3js.org/d3.v5.js"></script>
    <svg></svg>