Plugin/bubblecompare/index.ts

  1. import {select as d3Select} from "d3-selection";
  2. import Plugin from "../Plugin";
  3. /**
  4. * Bubble compare diagram plugin.<br>
  5. * Compare data 3-dimensional ways: x-axis, y-axis & bubble-size.
  6. * - **NOTE:**
  7. * - Plugins aren't built-in. Need to be loaded or imported to be used.
  8. * - Non required modules from billboard.js core, need to be installed separately.
  9. * - **Required modules:**
  10. * - [d3-selection](https://github.com/d3/d3-selection)
  11. * @class plugin-bubblecompare
  12. * @requires d3-selection
  13. * @param {object} options bubble compare plugin options
  14. * @augments Plugin
  15. * @returns {BubbleCompare}
  16. * @example
  17. * // Plugin must be loaded before the use.
  18. * <script src="$YOUR_PATH/plugin/billboardjs-plugin-bubblecompare.js"></script>
  19. *
  20. * var chart = bb.generate({
  21. * data: {
  22. * columns: [ ... ],
  23. * type: "bubble"
  24. * }
  25. * ...
  26. * plugins: [
  27. * new bb.plugin.bubblecompare({
  28. * minR: 11,
  29. * maxR: 74,
  30. * expandScale: 1.1
  31. * }),
  32. * ]
  33. * });
  34. * @example
  35. * import {bb} from "billboard.js";
  36. * import BubbleCompare from "billboard.js/dist/billboardjs-plugin-bubblecompare";
  37. *
  38. * bb.generate({
  39. * plugins: [
  40. * new BubbleCompare({ ... })
  41. * ]
  42. * })
  43. */
  44. export default class BubbleCompare extends Plugin {
  45. static version = `0.0.1`;
  46. public $$;
  47. constructor(options) {
  48. super(options);
  49. return this;
  50. }
  51. $init(): void {
  52. const {$$} = this;
  53. $$.findClosest = this.findClosest.bind(this);
  54. $$.getBubbleR = this.getBubbleR.bind(this);
  55. $$.pointExpandedR = this.pointExpandedR.bind(this);
  56. }
  57. pointExpandedR(d): number {
  58. const baseR = this.getBubbleR(d);
  59. const {expandScale = 1} = this.options;
  60. BubbleCompare.raiseFocusedBubbleLayer(d);
  61. this.changeCursorPoint();
  62. return baseR * expandScale;
  63. }
  64. static raiseFocusedBubbleLayer(d): void {
  65. d.raise && d3Select(d.node().parentNode.parentNode).raise();
  66. }
  67. changeCursorPoint(): void {
  68. this.$$.$el.eventRect.style("cursor", "pointer");
  69. }
  70. findClosest(values, pos): number {
  71. const {$$} = this;
  72. return values
  73. .filter(v => v && !$$.isBarType(v.id))
  74. .reduce((acc, cur) => {
  75. const d = $$.dist(cur, pos);
  76. return d < this.getBubbleR(cur) ? cur : acc;
  77. }, 0);
  78. }
  79. getBubbleR(d): number {
  80. const {minR, maxR} = this.options;
  81. const curVal = this.getZData(d);
  82. if (!curVal) return minR;
  83. const [min, max] = this.$$.data.targets.reduce(
  84. ([accMin, accMax], cur) => {
  85. const val = this.getZData(cur.values[0]);
  86. return [Math.min(accMin, val), Math.max(accMax, val)];
  87. },
  88. [10000, 0]
  89. );
  90. const size = min > 0 && max === min ? 0 : curVal / max;
  91. return Math.abs(size) * (maxR - minR) + minR;
  92. }
  93. getZData(d): number {
  94. return this.$$.isBubbleZType(d) ? this.$$.getBubbleZData(d.value, "z") : d.value;
  95. }
  96. }