Skip to content

Commit

Permalink
Provides repeatability, through .prng(...) API
Browse files Browse the repository at this point in the history
  • Loading branch information
Kcnarf committed Aug 16, 2018
1 parent 57551c6 commit 4f62a9a
Show file tree
Hide file tree
Showing 7 changed files with 285 additions and 94 deletions.
29 changes: 22 additions & 7 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ The drawback is that the computation of a Voronoï treemap is based on a iterati

## Examples

* [The Global Economy by GDP](https://bl.ocks.org/Kcnarf/fa95aa7b076f537c00aed614c29bb568), a remake of [HowMuch.net's article](https://howmuch.net/articles/the-global-economy-by-GDP)
- [The Global Economy by GDP](https://bl.ocks.org/Kcnarf/fa95aa7b076f537c00aed614c29bb568), a remake of [HowMuch.net's article](https://howmuch.net/articles/the-global-economy-by-GDP)

## Installing

Expand Down Expand Up @@ -57,8 +57,7 @@ Then, later in your javascript, in order to draw cells:

```javascript
var allNodes = rootNode.descendants;
d3
.selectAll('path')
d3.selectAll('path')
.data(allNodes)
.enter()
.append('path')
Expand All @@ -72,14 +71,14 @@ d3

## Reference

* based on [Computing Voronoï Treemaps - Faster, Simpler, and Resolution-independent ](https://www.uni-konstanz.de/mmsp/pubsys/publishedFiles/NoBr12a.pdf)
* [https://github.com/ArlindNocaj/power-voronoi-diagram](https://github.com/ArlindNocaj/power-voronoi-diagram) for a Java implementation
- based on [Computing Voronoï Treemaps - Faster, Simpler, and Resolution-independent ](https://www.uni-konstanz.de/mmsp/pubsys/publishedFiles/NoBr12a.pdf)
- [https://github.com/ArlindNocaj/power-voronoi-diagram](https://github.com/ArlindNocaj/power-voronoi-diagram) for a Java implementation

## API

<a name="voronoiTreemap" href="#voronoiTreemap">#</a> d3.<b>voronoiTreemap</b>()

Creates a new voronoiTreemap with the default [_clip_](#voronoiTreemap_clip), [_convergenceRatio_](#voronoiTreemap_convergenceRatio), [_maxIterationCount_](#voronoiTreemap_maxIterationCount) and [_minWeightRatio_](#voronoiTreemap_minWeightRatio) configuration values.
Creates a new voronoiTreemap with the default configuration values and functions ([_clip_](#voronoiTreemap_clip), [_convergenceRatio_](#voronoiTreemap_convergenceRatio), [_maxIterationCount_](#voronoiTreemap_maxIterationCount), [_minWeightRatio_](#voronoiTreemap_minWeightRatio) and [_prng_](#voronoiTreemap_prng)).

<a name="_voronoiTreemap" href="#_voronoiTreemap">#</a> <i>voronoiTreemap</i>(<i>root</i>)

Expand Down Expand Up @@ -125,9 +124,25 @@ var minWeightRatio = 0.01; // 1% of maxWeight

_minWeightRatio_ allows to mitigate flickerring behaviour (caused by too small weights), and enhances user interaction by not computing near-empty cells.

<a name="voronoiTreemap_prng" href="#voronoiTreemap_prng">#</a> <i>voronoiTreemap</i>.<b>prng</b>([<i>prng</i>])

If _prng_ is specified, sets the pseudorandom number generator which is used when randomness is required (i.e. when setting intial random position of data/seeds). The given pseudorandom number generator must implement the same interface as `Math.random` and must only return values in the range [0, 1). If _prng_ is not specified, returns the current _prng_ , which defaults to `Math.random`.

Considering the same set of data, severall Voronoï treemap computations lead to disctinct final arrangements, due to the non-seedable `Math.random` number generator. If _prng_ is set to a _seedable_ PRNG which produces repeatable results, then several computations will produce the exact same final arrangement. This is useful if you want the same arrangement for distinct page loads/reloads. For example, using [seedrandom](https://github.com/davidbau/seedrandom):

```js
<script src="//cdnjs.cloudflare.com/ajax/libs/seedrandom/2.4.3/seedrandom.min.js"></script>
<script>
var mySeededPrng = new Math.seedrandom('my seed'); // (from seedrandom's doc) Use "new" to create a local prng without altering Math.random
voronoiTreemap.prng(mySeededPrng);
</script>
```

You can also take a look at [d3-random](https://github.com/d3/d3-random) for random number generator from other-than-uniform distributions.

## Dependencies

* d3-voronoi-map.voronoiMap
- d3-voronoi-map.voronoiMap

## Testing

Expand Down
85 changes: 57 additions & 28 deletions build/d3-voronoi-treemap.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,82 +4,111 @@
(factory((global.d3 = global.d3 || {}),global.d3));
}(this, function (exports,d3VoronoiMap) { 'use strict';

function voronoiTreemap () {
function voronoiTreemap() {

//begin: constants
var DEFAULT_CONVERGENCE_RATIO = 0.01;
var DEFAULT_MAX_ITERATION_COUNT = 50;
var DEFAULT_MIN_WEIGHT_RATIO = 0.01;
var DEFAULT_PRNG = Math.random;
//end: constants

/////// Inputs ///////
var clip = [[0,0], [0,1], [1,1], [1,0]] // clipping polygon
var convergenceRatio = DEFAULT_CONVERGENCE_RATIO; // targeted allowed error ratio; default 0.01 stops computation when cell areas error <= 1% clipping polygon's area
var maxIterationCount = DEFAULT_MAX_ITERATION_COUNT; // maximum allowed iteration; stops computation even if convergence is not reached; use a large amount for a sole converge-based computation stop
var minWeightRatio = DEFAULT_MIN_WEIGHT_RATIO; // used to compute the minimum allowed weight; default 0.01 means 1% of max weight; handle near-zero weights, and leaves enought space for cell hovering

var clip = [
[0, 0],
[0, 1],
[1, 1],
[1, 0]
] // clipping polygon
var convergenceRatio = DEFAULT_CONVERGENCE_RATIO; // targeted allowed error ratio; default 0.01 stops computation when cell areas error <= 1% clipping polygon's area
var maxIterationCount = DEFAULT_MAX_ITERATION_COUNT; // maximum allowed iteration; stops computation even if convergence is not reached; use a large amount for a sole converge-based computation stop
var minWeightRatio = DEFAULT_MIN_WEIGHT_RATIO; // used to compute the minimum allowed weight; default 0.01 means 1% of max weight; handle near-zero weights, and leaves enought space for cell hovering
var prng = DEFAULT_PRNG; // pseudorandom number generator

//begin: internals
var _voronoiMap = d3VoronoiMap.voronoiMap();
//end: internals


///////////////////////
///////// API /////////
///////////////////////

function _voronoiTreemap (rootNode) {
_voronoiMap.weight(function(d){ return d.value; })
function _voronoiTreemap(rootNode) {
_voronoiMap.weight(function (d) {
return d.value;
})
.convergenceRatio(convergenceRatio)
.maxIterationCount(maxIterationCount)
.minWeightRatio(minWeightRatio);

.minWeightRatio(minWeightRatio)
.prng(prng);

recurse(clip, rootNode);
};

_voronoiTreemap.convergenceRatio = function (_) {
if (!arguments.length) { return convergenceRatio; }

if (!arguments.length) {
return convergenceRatio;
}

convergenceRatio = _;
return _voronoiTreemap;
};

_voronoiTreemap.maxIterationCount = function (_) {
if (!arguments.length) { return maxIterationCount; }

if (!arguments.length) {
return maxIterationCount;
}

maxIterationCount = _;
return _voronoiTreemap;
};

_voronoiTreemap.minWeightRatio = function (_) {
if (!arguments.length) { return minWeightRatio; }

if (!arguments.length) {
return minWeightRatio;
}

minWeightRatio = _;
return _voronoiTreemap;
};

_voronoiTreemap.clip = function (_) {
if (!arguments.length) { return _voronoiMap.clip(); }

if (!arguments.length) {
return clip;
}

//begin: use voronoiMap.clip() to handle borderline input (non-counterclockwise, non-convex, ...)
_voronoiMap.clip(_);
//end: use voronoiMap.clip() to handle
clip = _voronoiMap.clip();
return _voronoiTreemap;
};

_voronoiTreemap.prng = function (_) {
if (!arguments.length) {
return prng;
}

prng = _;
return _voronoiTreemap;
};

///////////////////////
/////// Private ///////
///////////////////////

function recurse(clippingPolygon, node) {
var voronoiMapRes;

//assign polygon to node
node.polygon = clippingPolygon;
if (node.height!=0) {

if (node.height != 0) {
//compute one-level Voronoi map of children
voronoiMapRes = _voronoiMap.clip(clippingPolygon)(node.children);
//begin: recurse on children
voronoiMapRes.polygons.forEach(function(cp){
voronoiMapRes.polygons.forEach(function (cp) {
recurse(cp, cp.site.originalObject.data.originalData);
})
//end: recurse on children
Expand Down
2 changes: 1 addition & 1 deletion build/d3-voronoi-treemap.min.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 2 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,6 @@
"dependencies": {
"d3-polygon": "1.*",
"d3-array": "1.*",
"d3-voronoi-map": "1.*"
"d3-voronoi-map": ">=1.2.0 1.*"
}
}
}
89 changes: 60 additions & 29 deletions src/d3-voronoi-treemap.js
Original file line number Diff line number Diff line change
@@ -1,81 +1,112 @@
import {voronoiMap} from 'd3-voronoi-map';
import {
voronoiMap
} from 'd3-voronoi-map';

export function voronoiTreemap() {

export function voronoiTreemap () {

//begin: constants
var DEFAULT_CONVERGENCE_RATIO = 0.01;
var DEFAULT_MAX_ITERATION_COUNT = 50;
var DEFAULT_MIN_WEIGHT_RATIO = 0.01;
var DEFAULT_PRNG = Math.random;
//end: constants

/////// Inputs ///////
var clip = [[0,0], [0,1], [1,1], [1,0]] // clipping polygon
var convergenceRatio = DEFAULT_CONVERGENCE_RATIO; // targeted allowed error ratio; default 0.01 stops computation when cell areas error <= 1% clipping polygon's area
var maxIterationCount = DEFAULT_MAX_ITERATION_COUNT; // maximum allowed iteration; stops computation even if convergence is not reached; use a large amount for a sole converge-based computation stop
var minWeightRatio = DEFAULT_MIN_WEIGHT_RATIO; // used to compute the minimum allowed weight; default 0.01 means 1% of max weight; handle near-zero weights, and leaves enought space for cell hovering

var clip = [
[0, 0],
[0, 1],
[1, 1],
[1, 0]
] // clipping polygon
var convergenceRatio = DEFAULT_CONVERGENCE_RATIO; // targeted allowed error ratio; default 0.01 stops computation when cell areas error <= 1% clipping polygon's area
var maxIterationCount = DEFAULT_MAX_ITERATION_COUNT; // maximum allowed iteration; stops computation even if convergence is not reached; use a large amount for a sole converge-based computation stop
var minWeightRatio = DEFAULT_MIN_WEIGHT_RATIO; // used to compute the minimum allowed weight; default 0.01 means 1% of max weight; handle near-zero weights, and leaves enought space for cell hovering
var prng = DEFAULT_PRNG; // pseudorandom number generator

//begin: internals
var _voronoiMap = voronoiMap();
//end: internals


///////////////////////
///////// API /////////
///////////////////////

function _voronoiTreemap (rootNode) {
_voronoiMap.weight(function(d){ return d.value; })
function _voronoiTreemap(rootNode) {
_voronoiMap.weight(function (d) {
return d.value;
})
.convergenceRatio(convergenceRatio)
.maxIterationCount(maxIterationCount)
.minWeightRatio(minWeightRatio);

.minWeightRatio(minWeightRatio)
.prng(prng);

recurse(clip, rootNode);
};

_voronoiTreemap.convergenceRatio = function (_) {
if (!arguments.length) { return convergenceRatio; }

if (!arguments.length) {
return convergenceRatio;
}

convergenceRatio = _;
return _voronoiTreemap;
};

_voronoiTreemap.maxIterationCount = function (_) {
if (!arguments.length) { return maxIterationCount; }

if (!arguments.length) {
return maxIterationCount;
}

maxIterationCount = _;
return _voronoiTreemap;
};

_voronoiTreemap.minWeightRatio = function (_) {
if (!arguments.length) { return minWeightRatio; }

if (!arguments.length) {
return minWeightRatio;
}

minWeightRatio = _;
return _voronoiTreemap;
};

_voronoiTreemap.clip = function (_) {
if (!arguments.length) { return _voronoiMap.clip(); }

if (!arguments.length) {
return clip;
}

//begin: use voronoiMap.clip() to handle borderline input (non-counterclockwise, non-convex, ...)
_voronoiMap.clip(_);
//end: use voronoiMap.clip() to handle
clip = _voronoiMap.clip();
return _voronoiTreemap;
};

_voronoiTreemap.prng = function (_) {
if (!arguments.length) {
return prng;
}

prng = _;
return _voronoiTreemap;
};

///////////////////////
/////// Private ///////
///////////////////////

function recurse(clippingPolygon, node) {
var voronoiMapRes;

//assign polygon to node
node.polygon = clippingPolygon;
if (node.height!=0) {

if (node.height != 0) {
//compute one-level Voronoi map of children
voronoiMapRes = _voronoiMap.clip(clippingPolygon)(node.children);
//begin: recurse on children
voronoiMapRes.polygons.forEach(function(cp){
voronoiMapRes.polygons.forEach(function (cp) {
recurse(cp, cp.site.originalObject.data.originalData);
})
//end: recurse on children
Expand Down
Loading

0 comments on commit 4f62a9a

Please sign in to comment.