d3.js circle packing along a line

后端 未结 2 1231
遇见更好的自我
遇见更好的自我 2020-12-05 17:04

I got a data set that where each sample has a size (0-1000) and a value (grade 1-5). I want to visualise the data with circles of different sizes along a line (domain axis)

相关标签:
2条回答
  • 2020-12-05 17:27

    D3's packing layout is not the answer here. It places circles in a spiral fashion around the existing group. Here's me reverse-engineering the algorithm behind packing layout:

    enter image description here

    I would suggest a force layout-based approach. That way, you can give your nodes force towards a gravitational center, and then let gravity do its thing.

    Force layouts (e.g. Clustered Force Layout I) are usually animations, so you'll want to apply a static force layout.

    I've wrapped up this approach in an example block, which looks like this:

    0 讨论(0)
  • 2020-12-05 17:39

    Oooh, this one was a puzzle...

    If you look at the code for the NYTimes graphic, it uses pre-computed coordinates in the data file, so that's not much use.

    However, there's an unused variable declaration at the top of the script that hints that the original version used d3.geom.quadtree to lay out the circles. The quadtree isn't actually a layout method; it is used to create a search tree of adjacent nodes, so that when you need to find a node in a given area you don't have to search through the whole set. Example here.

    The quadtree can therefore be used to identify which of your datapoints might be overlapping each other on the x-axis. Then you have to figure out how much you need to offset them in order to avoid that overlap. The variable radii complicate both functions...

    I've got a test case implemented here: http://fiddle.jshell.net/6cW9u/5/

    The packing algorithm isn't perfect: I always add new circles to the outside of existing circles, without testing whether they could possibly fit closer in, so sometimes you get significant extra whitespace when it is just the far edges of circles bumping into each other. (Run it a few times to get an idea of the possibilities -- note that I've got x-variables distributed as random normal and r-variables distributed as random uniform.) I also got a stack overflow on the recursive methods during one iteration with N=100 -- the random distribution clearly wasn't distributed well enough for the quadtree optimization.

    But it's got the basic functionality. Leave a comment here if you can't follow the logic of my code comments.

    --ABR

    Update

    New fiddle here: http://fiddle.jshell.net/6cW9u/8/

    After a lot of re-arranging, I got the packing algorithm to search for gaps between existing bubbles. I've got the sort order switched (so that biggest circles get added first) to show off how little circles can get added in the gaps -- although as I mention in the code comments, this reduces the efficiency of the quadtree search.

    Also added various decoration and transition so you can clearly see how the circles are being positioned, and set the r-scale to be square root, so the area (not radius) is proportional to the value in the data (which is more realistic, and what the O.P. asked for).

    0 讨论(0)
提交回复
热议问题