[譯] D3.js 嵌套選擇集 (Nested Selection)

譯者注:

原文: Mike Bostock (D3.js 做者) -- Nested Selectionsjavascript

譯者: ssthousehtml

本文講解的是關於 D3.js 中 d3-selection 的使用. d3-selection 是 d3 的核心所在, 它提供了一種和以往 Dom 操做數據操做 徹底不一樣的思路, 讓咱們能很是優雅的進行數據可視化工做. 本文是 d3 做者對於 d3-selection 中 嵌套選擇集 的講解, 本人閱讀後以爲頗有啓發, 因此翻譯成中文, 但願對讀者也能有所幫助.java

譯文

D3 的 選擇集是分層級的, 就像 Dom 元素和數據集是能夠有層級的同樣. 好比說 Table:node

<table>
  <thead>
    <tr><td>  A</td><td>  B</td><td>  C</td><td>  D</td></tr>
  </thead>
  <tbody>
    <tr><td>  0</td><td>  1</td><td>  2</td><td>  3</td></tr>
    <tr><td>  4</td><td>  5</td><td>  6</td><td>  7</td></tr>
    <tr><td>  8</td><td>  9</td><td> 10</td><td> 11</td></tr>
    <tr><td> 12</td><td> 13</td><td> 14</td><td> 15</td></tr>
  </tbody>
</table>
複製代碼

若是讓你只選中 tbody 元素中的 td, 你會如何操做? 直接使用 d3.selectAll('td') 顯然會選中全部的 td 元素(包括 thead 和 tbody). 若是想只選中存在與 B 元素中的 A 元素, 你須要這樣操做:git

var td = d3.selectAll('tbody td')
複製代碼

除了上面那種方法, 你還能夠先選中 tbody, 再選中 td 元素, 像這樣:github

var td = d3.select('tbody').selectAll('td')
複製代碼

由於 selectAll 會對當前選擇集中的每一個元素(如: tbody)選中其符合條件的子元素(如: td). 這種方法會獲得和 d3.selectAll('tbody td') 相同的結果. 但若是你以後想要對同一父元素的選擇集引伸出更多的選擇集的話 (好比區分選中 table 中的奇數行選擇集和偶數行選擇集), 這種嵌套選擇集方式會方便的多.數組

嵌套中索引的使用

接着上面的例子, 若是你使用 d3.selectAll('td') , 你會獲得一個平展的選擇集, 像這樣:app

flat selection

var td = d3.selectAll('tbody td')
複製代碼

平展的選擇集缺乏了層級結構: 不管是 thead 中的 td 仍是 tbody 中的 td 全都被展開成了一個數組, 而不是以父元素進行分組. 這讓咱們想對每一行或是每一列的 td 進行操做變得很困難. 與此相反的, D3 的嵌套選擇集能保存層級關係. 好比咱們想以行的方式對選擇集分組, 咱們首先選中 tr 元素. 而後選中 td 元素.函數

nested selection

var td = d3.selectAll('tbody tr').selectAll('td')
複製代碼

如今, 若是你想讓第一列的全部元素變紅, 你能夠利用 index 參數 i:post

td.style('color', function(d, i) {
  return i ? null : 'red'
})
複製代碼

上面的參數 i 是 td 元素在 tr 中的索引, 你還能夠經過添加一個 j 參數來得到當前的行數的索引. 像這樣:

td.style('color', function(d, i, j) {
  console.log(`current row: ${j}`)
  return i ? null : 'red'
})
複製代碼

嵌套和數據間的關聯

層級結構的 Dom 元素經常是由層級結構的數據來驅動的. 而層級的選擇集能更方便的處理數據綁定. 繼續上面的例子, 你能夠把 table 的數據表示爲一個矩陣:

var matrix = [
  [0, 1, 2, 3], 
  [4, 5, 6, 7], 
  [8, 9, 10, 11], 
  [12, 13, 14, 15]
]
複製代碼

爲了讓這些數據綁定上對應的 td 元素, 咱們首先將矩陣的每一行和 tr 綁定對應起來, 而後再將矩陣中一行的每個元素和 tr 中的每個 td 綁定起來:

var td = d3
  .selectAll('tbody tr')
  .data(matrix)
  .selectAll('td')
  .data(function(d, i) {
    return d
  }) // d is matrix[i]
複製代碼

須要注意的是, data() 不只能夠傳入一個數據, 它還能夠傳入一個 返回一個數組的 function. 平展的選擇集一般對應的是單個數組, 是由於平展的選擇集只有一個 group.

上面的 row 選擇集是一個平展的選擇集, 由於它是直接由 d3.selectAll 建立的:

tr flat selection

var tr = d3.selectAll('tbody tr').data(matrix)
複製代碼

而 td 的選擇集是嵌套的:

tr nested selection

var td = tr.selectAll('td').data(function(d) {
  return d
}) // matrix[i]
複製代碼

data 傳入的 操做函數給每個 group 綁定了一個數組數據. d3 會對每一行 tr 調用操做函數. 由於父元素數據是矩陣, 因此操做函數在每次被調用時只是簡單的返回矩陣中當前行的數據, 來和 tr 進行綁定.

嵌套中父節點做用

嵌套選擇集有一個微妙但可能形成嚴重影響的反作用: 它會給每一個 group 設置父節點. 父節點是選擇集的一個隱藏屬性, 它會在被調用 append 方法時使用, 將子元素添加到父節點的 Dom 元素當中. 好比: 若是你想經過下面的方式進行數據綁定操做, 你會獲得一個 error:

wrong append operation

d3.selectAll('table tr')
  .data(matrix)
  .enter()
  .append('tr') // error!
複製代碼

上面的代碼之因此會報錯, 是由於默認的父節點是 html 元素, 你不能直接將 tr 元素添加到 html 元素中. 因此, 咱們應該在進行數據綁定前, 先選擇好父節點:

choose parent node before append operation

d3.select('table')
  .selectAll('tr')
  .data(matrix)
  .enter()
  .append('tr') // success
複製代碼

這種方式能夠用來選擇任意層級的嵌套選擇集. 好比你想從頭建立一個 table, 並填入上面矩陣中的數據, 你能夠首選選中 body 元素:

create table manually

var body = d3.select('body')
複製代碼

接下來在父節點 body 中添加一個 table:

append table to body

var table = body.append('table')
複製代碼

接下來綁定矩陣數據, 建立 tr 元素. 由於 selectAll 是在 table 元素上進行調用的, 因此父節點是 table:

append tr to table

var tr = table
  .selectAll('tr')
  .data(matrix)
  .enter()
  .append('tr')
複製代碼

最後咱們以 tr 做爲父節點, 建立 td 元素:

append td to tr

var td = tr
  .selectAll('td')
  .data(function(d) {
    return d
  })
  .enter()
  .append('td')
複製代碼

要不要使用嵌套選擇集 ?

在 D3 中, select 和 selectAll 有一個很重要的區別: select 會繼續使用當前存在的 group, 而 selectAll 老是會建立新的 group. 所以調用 select 能保存原有 selection 的數據, 索引位置, 甚至父節點.

好比, 下面的平展的選擇集, 它的父節點仍然是 html 節點:

parent node is html

var td = d3.selectAll('tbody tr').select('td')
複製代碼

想要獲得嵌套的選擇集, 惟一的方法就是在已有的選擇集的基礎上, 調用 selectAll 方法. 這就是爲何數據綁定老是出如今 selectAll 以後, 而不是 select 以後.

這篇文章使用 table 做爲例子僅僅是爲了方便講解層級結構. 其實 table 的使用並非特別具備典型性. 其實還有許多其它 嵌套選擇集的例子(點我查看, 點我查看)

就像 用 join 的方式思考 同樣, 嵌套選擇集一樣使用了一種思想上徹底不一樣的處理 Dom 元素數據 的思路. 這種思路剛開始可能很難理解, 但你一旦掌握了, 你就能得心應手的使用 D3.js 來完成你的數據可視化任務了.

若是以爲不錯的話, 不妨點下面的連接關注一下 : )

github 主頁

知乎專欄

掘金

歡迎關注個人公衆號:

相關文章
相關標籤/搜索