2013-05-28 10 views
8

私はd3.jsで地図のビジュアライゼーションを構築しています。私は米国の州と郡の両方の塗りつぶしポリゴンを描画しています。郡のSVGレイヤーは、状態のレイヤーの下にあります。状態は塗りつぶされますが、塗りつぶし不透明度は0に設定されます。クリックイベントを捕捉するには塗りつぶしが必要です(私はそうだと思います)。オーバーラッピングSVGレイヤーでマウスイベントを処理する

州レベルでクリックイベントを捕捉したいのですが、私は郡レベルでマウスオーバーイベントをキャッチしたいと思います。

ただし、mouseoverイベントは状態によってキャプチャされ、郡に渡されません。

イベントをレイヤーに渡す方法や、マウスオーバーイベントを郡でトリガーする方法はありますか?

+0

使用している例へのリンクがありますか? –

答えて

2

州をクリックしたときに何かをしたい場合は、国境のスタイルや図形の不透明度を変更するなど、すべての郡が州に属し、州をグループ化してキャプチャ郡内の「クリック」イベントを選択し、対応する状態要素を選択し、その視覚属性を変更します。この戦略のモックアップは:http://jsfiddle.net/pnavarrc/PGTCM/5/

// Setup the visualization, assuming that you have a div with id 'chart' 
var width = 300, 
    height = 300, 
    div = d3.select('#chart'), 
    svg = div.append('svg') 
     .attr('width', width) 
     .attr('height', height); 

// Array of states, each one containing one or more counties. In this example, 
// the states are rectangles, but you get the idea 
var states = [ 
    { 
     width: 300, 
     height: 300, 
     name: 'Fake State', 
     counties: [ 
      {x: 5, y: 5, width: 95, height: 290, fill: '#b4a0a0'}, 
      {x: 100, y: 5, width: 150, height: 290, fill: '#b4c0a0'}, 
      {x: 250, y: 5, width: 45, height: 290, fill: '#a4a0c0'} 
     ] 
    }]; 

// Create a group for each state, with class state 
var gstates = svg.selectAll('g.state') 
    .data(states) 
    .enter() 
    .append('g') 
    .classed('state', true); 

// Add the state background, in this case, a transparent rectangle 
gstates 
    .append('rect') 
    .classed('state', true) 
    .attr('width', function(d) { return d.width; }) 
    .attr('height', function(d) { return d.height; }) 
    .attr('fill', '#444') 
    .attr('fill-opacity', 0.0); 

// For each group, add rectangles for each county, binding them to the 
// county array of each state. 
gstates.selectAll('rect.county') 
    .data(function(d) { return d.counties; }) 
    .enter() 
    .append('rect') 
    .classed('county', true) 
     .attr('x', function(d) { return d.x; }) 
     .attr('y', function(d) { return d.y; }) 
     .attr('width', function(d) { return d.width; }) 
     .attr('height', function(d) { return d.height; }) 
     .attr('fill', function(d) { return d.fill; }) 
     .on('mouseover', function() { 
      d3.select(this).attr('fill-opacity', 0.5); 
     }) 
     .on('mouseout', function() { 
      d3.select(this).attr('fill-opacity', 0.9); 
     }) 
     .on('click', function() { 
      // Retrive the parent group of each county, and then select 
      // the shape corresponding to the state and alter its properties 
      // you can also access the state data 
      var parent = d3.select(d3.select(this).node().parentNode), 
       state = parent.select('rect.state'); 
       // Output 'Fake State' 
       console.log(parent[0][0].__data__.name); 
       state.attr('fill-opacity', 1.0); 
     }); 

ここでの作業フィドルがあります。よろしく、

6

あなたの望むことを実行するd3関数です。

これは、最上位レイヤーのポインターイベントを一時的に無効にし、次のレイヤーでマウスイベントを手動でトリガーすることによって、イベントを下位レイヤーに渡します。

function passThruEvents(g) { 
    g .on('mousemove.passThru', passThru) 
     .on('mousedown.passThru', passThru) 
    ; 

    function passThru(d) { 
     var e = d3.event; 

     var prev = this.style.pointerEvents; 
     this.style.pointerEvents = 'none'; 

     var el = document.elementFromPoint(d3.event.x, d3.event.y); 

     var e2 = document.createEvent('MouseEvent'); 
     e2.initMouseEvent(e.type,e.bubbles,e.cancelable,e.view, e.detail,e.screenX,e.screenY,e.clientX,e.clientY,e.ctrlKey,e.altKey,e.shiftKey,e.metaKey,e.button,e.relatedTarget); 

     el.dispatchEvent(e2); 

     this.style.pointerEvents = prev; 
    } 
} 
+0

素晴らしいソリューション! – rockTheWorld

+1

この解決策は他の状況にも適応するのはすごくて簡単ですが、一番上の要素によって「隠された」要素は ':hover'を介してスタイル可能ではありません。 – luismreis