在 D3 Hierarchy Typescript 中,每个父子对都具有相同的颜色

Each parent-child pair has same color in D3 Hierarchy Typescript

我正在尝试使用来自 here 的“D3 可折叠力布局”的一个很好的例子。我将输入数据修改如下

 const data = {
 "name": "directory",
 "children": [
  {"name": "1st file", 
    "children": [{"name": "1f 1st function", "group": 1},  
    {"name": "1f 2nd function", "group": 1},
    {"name": "1f 3rd function", "group": 1},
    {"name": "1f 4th function", "group": 1}], "group": 1,},
  {"name": "2nd file", "children": [
    {"name": "2f 1st function", "group": 2},
    {"name": "2f 2nd function", "group": 2}], "group": 2},
  {"name": "3rd file", "children": [
    {"name": "3f 1st function", "group": 3}
  ], "group": 3},
  {"name": "4th file", "children": [
    {"name": "4f 1st function", "group": 4},
    {"name": "4f 2nd function", "group": 4},
    {"name": "4f 3rd function", "group": 4}
  ], "group": 4}
 ],
 "group": 0
}

我想根据组值为所有父子节点设置相同的颜色。 例如:所有的“group:1”都是红色,“group:2”是绿色等等。任何人都可以帮我弄清楚吗?谢谢你的时间。

您只需要一个序数表。例如:

const colorScale = d3.scaleOrdinal()
  .domain(d3.range(5))
  .range(d3.schemeDark2);

这里我使用了一个名为d3.schemeDark2的颜色数组,根据你想要的颜色来改变它。对于域,因为您的 group 是数字,所以我使用 d3.range.

然后,绘制圆圈:

.style('fill', function(d){
    return colorScale(d.data.group)
})

这是包含您的数据和这些更改的代码:

const data = {
  "name": "directory",
  "children": [{
      "name": "1st file",
      "children": [{
          "name": "1f 1st function",
          "group": 1
        },
        {
          "name": "1f 2nd function",
          "group": 1
        },
        {
          "name": "1f 3rd function",
          "group": 1
        },
        {
          "name": "1f 4th function",
          "group": 1
        }
      ],
      "group": 1,
    },
    {
      "name": "2nd file",
      "children": [{
          "name": "2f 1st function",
          "group": 2
        },
        {
          "name": "2f 2nd function",
          "group": 2
        }
      ],
      "group": 2
    },
    {
      "name": "3rd file",
      "children": [{
        "name": "3f 1st function",
        "group": 3
      }],
      "group": 3
    },
    {
      "name": "4th file",
      "children": [{
          "name": "4f 1st function",
          "group": 4
        },
        {
          "name": "4f 2nd function",
          "group": 4
        },
        {
          "name": "4f 3rd function",
          "group": 4
        }
      ],
      "group": 4
    }
  ],
  "group": 0
};

const colorScale = d3.scaleOrdinal()
  .domain(d3.range(5))
  .range(d3.schemeDark2);

const width = 500,
  height = 400;

let i = 0;

const root = d3.hierarchy(data);
const transform = d3.zoomIdentity;
let node, link;

const svg = d3.select('body').append('svg')
  .call(d3.zoom().scaleExtent([1 / 2, 8]).on('zoom', zoomed))
  .append('g')
  .attr('transform', 'translate(40,0)');



const simulation = d3.forceSimulation()
  .force('link', d3.forceLink().id(function(d) {
    return d.id;
  }))
  .force('charge', d3.forceManyBody().strength(-15).distanceMax(300))
  .force('center', d3.forceCenter(width / 2, height / 4))
  .on('tick', ticked)


function update() {
  const nodes = flatten(root)
  const links = root.links()

  link = svg
    .selectAll('.link')
    .data(links, function(d) {
      return d.target.id
    })

  link.exit().remove()

  const linkEnter = link
    .enter()
    .append('line')
    .attr('class', 'link')
    .style('stroke', '#000')
    .style('opacity', '0.2')
    .style('stroke-width', 2)

  link = linkEnter.merge(link)

  node = svg
    .selectAll('.node')
    .data(nodes, function(d) {
      return d.id
    })

  node.exit().remove()

  const nodeEnter = node
    .enter()
    .append('g')
    .attr('class', 'node')
    .attr('stroke', '#666')
    .attr('stroke-width', 2)
    .style('fill', function(d) {
      return colorScale(d.data.group)
    })
    .style('opacity', 1)
    .on('click', clicked)
    .call(d3.drag()
      .on('start', dragstarted)
      .on('drag', dragged)
      .on('end', dragended))

  nodeEnter.append('circle')
    .attr("r", function(d) {
      return Math.sqrt(d.data.size) / 10 || 4.5;
    })
    .style('text-anchor', function(d) {
      return d.children ? 'end' : 'start';
    })
    .text(function(d) {
      return d.data.name
    })

  node = nodeEnter.merge(node)
  simulation.nodes(nodes)
  simulation.force('link').links(links)
}

function sizeContain(num) {
  num = num > 1000 ? num / 1000 : num / 100
  if (num < 4) num = 4
  return num
}

function color(d) {
  return d._children ? "#51A1DC" // collapsed package
    :
    d.children ? "#51A1DC" // expanded package
    :
    "#F94B4C"; // leaf node
}

function radius(d) {
  return d._children ? 8 :
    d.children ? 8 :
    4
}

function ticked() {
  link
    .attr('x1', function(d) {
      return d.source.x;
    })
    .attr('y1', function(d) {
      return d.source.y;
    })
    .attr('x2', function(d) {
      return d.target.x;
    })
    .attr('y2', function(d) {
      return d.target.y;
    })

  node
    .attr('transform', function(d) {
      return `translate(${d.x}, ${d.y})`
    })
}

function clicked(d) {
  if (!d3.event.defaultPrevented) {
    if (d.children) {
      d._children = d.children;
      d.children = null;
    } else {
      d.children = d._children;
      d._children = null;
    }
    update()
  }
}


function dragstarted(d) {
  if (!d3.event.active) simulation.alphaTarget(0.3).restart()
  d.fx = d.x
  d.fy = d.y
}

function dragged(d) {
  d.fx = d3.event.x
  d.fy = d3.event.y
}

function dragended(d) {
  if (!d3.event.active) simulation.alphaTarget(0)
  d.fx = null
  d.fy = null
}

function flatten(root) {
  const nodes = []

  function recurse(node) {
    if (node.children) node.children.forEach(recurse)
    if (!node.id) node.id = ++i;
    else ++i;
    nodes.push(node)
  }
  recurse(root)
  return nodes
}

function zoomed() {
  svg.attr('transform', d3.event.transform)
}

update()
body {
  background-color: #E6E6E6;
}

svg {
  width: 100vw;
  height: 100vh;
}

.node {
  pointer-events: all;
  cursor: pointer;
  z-index: 1000;
}

.node text {
  font: 8px sans-serif;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/5.7.0/d3.min.js"></script>