@net7/components 4.0.0 → 4.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,21 +1,59 @@
1
1
  //---------------------------
2
2
  // BUBBLECHART.ts
3
3
  //---------------------------
4
+ /**
5
+ * The bubble chart is drawn using d3js, a library to create and update anything in svg.
6
+ *
7
+ * LEGEND:
8
+ * - svg -> the canvas/<svg> element where everything will be drawn on (appended).
9
+ * - g -> <g> is a group of svg elements.
10
+ * - leaf -> this is the svg circle, the bubble.
11
+ * - text -> this is all the text inside the circle.
12
+ * - tspan -> this is just one line of text.
13
+ *
14
+ * What are "join, enter, update, exit?"
15
+ * https://observablehq.com/@d3/learn-d3-joins?collection=@d3/learn-d3
16
+ * https://observablehq.com/@thetylerwolf/day-18-join-enter-update-exit
17
+ *
18
+ * Each line can have a different width, but all of the text inside a circle
19
+ * needs to be scaled by a constant factor.
20
+ * https://observablehq.com/@mbostock/fit-text-to-circle
21
+ */
4
22
  import { Component, Input } from '@angular/core';
5
23
  import * as i0 from "@angular/core";
6
24
  import * as i1 from "@angular/common";
7
25
  export class BubbleChartComponent {
8
26
  constructor() {
9
27
  this._loaded = false;
28
+ /**
29
+ * Reference for much of the new text scaling code comes from:
30
+ * https://observablehq.com/@mbostock/fit-text-to-circle
31
+ */
32
+ this.measureWidth = (text) => {
33
+ const context = document.createElement('canvas').getContext('2d');
34
+ // measure text with the correct font family and weight
35
+ if (this.data?.fontRendering?.label?.family && this.data?.fontRendering?.label?.weight) {
36
+ context.font = `${this.data.fontRendering.label.weight} ${this.data.fontRendering.label.family}`;
37
+ }
38
+ // calculated width + padding
39
+ return context.measureText(text).width * 1.15;
40
+ };
41
+ this.isValidNumber = (value) => !Number.isNaN(Number.parseFloat(value));
10
42
  this.draw = () => {
11
43
  const { d3 } = this;
12
44
  const { containerId, data, height, width, selected, transition, colorMatch, shuffle, fontRendering } = this.data;
45
+ // SVG shape path for the close icon
13
46
  const closeIconPath = 'M -50,40 L-40,50 0,10 40,50 50,40 10,0 50,-40 40,-50 0,-10 -40,-50 -50,-40 -10,0 -50,40';
47
+ const defaultLineHeight = 13;
48
+ // word count before truncating with ellipsis
49
+ const ellipsisThreshold = 4;
50
+ const textScalingFactor = 1;
14
51
  if (!Array.isArray(data)) {
15
52
  // Check if it is possible to draw with the current dataset
16
53
  console.warn('(n7-bubble-chart) The data object is not in the expected format!');
17
54
  return;
18
55
  }
56
+ // animation settings used to render changes in the chart
19
57
  let t = d3
20
58
  .transition()
21
59
  .duration(0);
@@ -25,17 +63,23 @@ export class BubbleChartComponent {
25
63
  .duration(transition)
26
64
  .ease(d3.easeCubicInOut);
27
65
  }
66
+ // calculate the bubble background color from it's type (domain)
28
67
  const colorMap = d3.scaleOrdinal()
29
68
  .domain(colorMatch ? colorMatch.domain : ['persona', 'luogo', 'organizzazione', 'cosa notevole'])
30
69
  .range(colorMatch ? colorMatch.range : d3.schemeTableau10);
70
+ // calculate how big the radius of the bubble should be
31
71
  const sizeScale = d3 // map entity count to bubble size
32
72
  .scaleLinear()
33
73
  .domain([0, d3.max(data, (d) => +d.count)])
34
74
  .range([3, d3.max(data, (d) => +d.count)]);
75
+ // pack is a d3js method which calculates the bubble's x & y position in the chart
76
+ // circle packing reference: https://observablehq.com/@d3/pack
35
77
  const pack = (children) => d3
36
78
  .pack()
37
79
  .size([width - 2, height - 2])
38
80
  .padding(1.5)(d3.hierarchy({ children }).sum((d) => sizeScale(d.count)));
81
+ // the bubbles are packed in a single level tree, this is the root
82
+ // see circle packing reference: https://observablehq.com/@d3/pack
39
83
  const root = () => {
40
84
  if (typeof shuffle === 'undefined' || shuffle) {
41
85
  const shuffData = data.slice(); // do not modify the source data!
@@ -43,25 +87,33 @@ export class BubbleChartComponent {
43
87
  } // if shuffle is set to false, skip the data shuffle
44
88
  return pack(data);
45
89
  };
90
+ // svg canvas where all the bubbles are drawn
46
91
  const svg = d3
47
92
  .select(`#${containerId}`)
48
93
  .attr('viewBox', [0, 0, width, height])
49
- .attr('font-family', 'Verdana, Geneva, sans-serif')
94
+ // .attr('font-family', 'Verdana, Geneva, sans-serif')
95
+ // .attr('font-size', '10px')
96
+ .style('font', '10px Verdana, Geneva, sans-serif')
97
+ .style('height', 'auto')
98
+ .style('max-width', '100%')
50
99
  .attr('text-anchor', 'middle');
100
+ this.removeUnneededNodes(svg);
101
+ // a leaf of the "pack tree" corresponds to a bubble
51
102
  const leaf = svg.selectAll('g').data(root().leaves(), (d) => d.data.entity.id);
52
103
  leaf
53
104
  .transition(t) // update transition on <g>
54
105
  .attr('fill-opacity', 1)
55
- .attr('transform', (d) => `translate(${d.x + 1},${d.y + 1})`)
56
- .attr('font-size', (d) => {
57
- let size = d.r / 5.5;
58
- size *= 1;
59
- size += 1;
60
- return `${Math.round(size)}px`;
61
- });
62
- leaf.selectAll('.close-icon').remove(); // clear all existing close icons
106
+ .attr('transform', (d) => `translate(${d.x + 1},${d.y + 1})`);
107
+ // .attr('font-size', (d) => {
108
+ // let size = d.r / 5.5;
109
+ // size *= 1;
110
+ // size += 1;
111
+ // return `${Math.round(size)}px`;
112
+ // });
113
+ // clear all existing close icons from the bubbles
114
+ leaf.selectAll('.close-icon').remove();
63
115
  if (selected) {
64
- leaf // render necessary close icons
116
+ leaf // render only the necessary close icons
65
117
  .filter((d) => selected.includes(d.data.entity.id))
66
118
  .append('path')
67
119
  .attr('class', 'close-icon')
@@ -79,86 +131,10 @@ export class BubbleChartComponent {
79
131
  .transition(t) // update transition on <circle>
80
132
  .attr('fill-opacity', 1)
81
133
  .attr('r', (d) => d.r);
82
- leaf
83
- .select('text')
84
- .attr('font-family', () => {
85
- if (fontRendering && fontRendering.label && fontRendering.label.family) {
86
- return fontRendering.label.family;
87
- }
88
- return 'inherit';
89
- })
90
- .attr('font-weight', () => {
91
- if (fontRendering && fontRendering.label && fontRendering.label.weight) {
92
- return fontRendering.label.weight;
93
- }
94
- return 'inherit';
95
- })
96
- .selectAll('tspan')
97
- .data((d) => {
98
- if (d.r / 4 > 4.5) {
99
- // show text and number threshhold
100
- let label = (d.data.entity.label.charAt(0).toUpperCase()
101
- + d.data.entity.label.slice(1)).split(/ +/g);
102
- if (label.length > 3) {
103
- label = label.slice(0, 3);
104
- label[2] += '…';
105
- }
106
- return label;
107
- }
108
- if (d.r / 4 > 2.5) {
109
- // show text threshhold
110
- let label = (d.data.entity.label.charAt(0).toUpperCase()
111
- + d.data.entity.label.slice(1)).split(/ +/g);
112
- if (label.length > 3) {
113
- label = label.slice(0, 3);
114
- label[2] += '…';
115
- }
116
- return label;
117
- }
118
- return '';
119
- })
120
- .join('tspan')
121
- .attr('x', 0)
122
- .attr('y', (d, i, nodes) => `${i - (nodes.length + 1) / 2 + 0.97}em`)
123
- .attr('fill', 'white')
124
- .text((d) => d);
125
- leaf
126
- .select('.label-count')
127
- .attr('font-family', () => {
128
- if (fontRendering && fontRendering.counter && fontRendering.counter.family) {
129
- return fontRendering.counter.family;
130
- }
131
- return 'inherit';
132
- })
133
- .attr('font-weight', () => {
134
- if (fontRendering && fontRendering.counter && fontRendering.counter.weight) {
135
- return fontRendering.counter.weight;
136
- }
137
- return 'inherit';
138
- })
139
- .attr('fill', 'white')
140
- .text((d) => {
141
- if (d.r / 4 > 2.5) {
142
- // show text and number threshhold
143
- return d.data.count;
144
- }
145
- return '';
146
- })
147
- .attr('y', (d) => {
148
- let labelLength = d.data.entity.label.split(/ +/g);
149
- if (labelLength.length > 3) {
150
- labelLength = labelLength.slice(0, 3);
151
- }
152
- return `${labelLength.length - (labelLength.length + 1) / 2 + 0.97}em`;
153
- });
134
+ // g represents a "group of svg items"
135
+ // items grouped together can be added / removed / scaled together
154
136
  const g = leaf.enter().append('g');
155
137
  g.attr('transform', (d) => `translate(${d.x + 1},${d.y + 1})`)
156
- .attr('font-size', (d) => {
157
- let size = d.r / 5.5;
158
- size *= 1;
159
- size += 1;
160
- return `${Math.round(size)}px`;
161
- })
162
138
  .attr('cursor', 'pointer')
163
139
  .on('click', (event, d) => {
164
140
  this.onClick(d.data.entity.id);
@@ -171,10 +147,12 @@ export class BubbleChartComponent {
171
147
  .attr('fill-opacity', 1)
172
148
  .attr('fill', (d) => colorMap(d.data.entity.typeOfEntity))
173
149
  .attr('r', (d) => d.r);
150
+ // prevents the text from overflowing the bubble
174
151
  g.append('clipPath')
175
152
  .attr('id', (d) => { d.clipUid = `Clip-${d.data.entity.id}`; })
176
153
  .append('use')
177
154
  .attr('xlink:href', (d) => d.leafUid.href);
155
+ /** NEW TEXT LOGIC */
178
156
  g.append('text')
179
157
  .attr('font-family', () => {
180
158
  if (fontRendering && fontRendering.label && fontRendering.label.family) {
@@ -188,40 +166,86 @@ export class BubbleChartComponent {
188
166
  }
189
167
  return 'inherit';
190
168
  })
191
- .selectAll('tspan')
192
- .data((d) => {
193
- if (d.r / 4 > 4.5) {
194
- // show text and number threshhold
195
- let label = (d.data.entity.label.charAt(0).toUpperCase()
196
- + d.data.entity.label.slice(1)).split(/ +/g);
197
- if (label.length > 3) {
198
- label = label.slice(0, 3);
199
- label[2] += '…';
200
- }
201
- return label;
169
+ .attr('fill', 'white')
170
+ .each((d) => {
171
+ // Capitalize the first letter of the label
172
+ d.data.entity.label = d.data.entity
173
+ .label.charAt(0).toUpperCase()
174
+ + d.data.entity.label.slice(1);
175
+ // 1. initialize meta object
176
+ if (!d._meta || typeof d._meta !== 'object')
177
+ d._meta = {};
178
+ // 2. tokenize label & count into words
179
+ const words = d.data.entity.label.split(/[\s]+/g); // To hyphenate: /\s+|(?<=-)/
180
+ // Truncate with ellipsis if the label is longer than the threshold
181
+ if (words.length > ellipsisThreshold) {
182
+ words.splice(ellipsisThreshold, words.length - ellipsisThreshold);
183
+ words[ellipsisThreshold - 1] += '…';
202
184
  }
185
+ // add counter
203
186
  if (d.r / 4 > 2.5) {
204
- // show text threshhold
205
- let label = (d.data.entity.label.charAt(0).toUpperCase()
206
- + d.data.entity.label.slice(1)).split(/ +/g);
207
- if (label.length > 3) {
208
- label = label.slice(0, 3);
209
- label[2] += '…';
187
+ // show text threshold
188
+ if (!words[words.length - 1])
189
+ words.pop();
190
+ if (!words[0])
191
+ words.shift();
192
+ }
193
+ if (d.r / 4 > 4.5) {
194
+ // show number threshold
195
+ words.push(`${d.data.count}`);
196
+ }
197
+ d._meta.words = words;
198
+ d._meta.lineHeight = defaultLineHeight;
199
+ const targetWidth = Math.sqrt(this.measureWidth(d._meta.words.join(' ').trim()) * defaultLineHeight);
200
+ // 3. build lines of text
201
+ d._meta.lines = [];
202
+ let line;
203
+ let lineWidth0 = Infinity;
204
+ for (let i = 0, n = d._meta.words.length; i < n; i += 1) {
205
+ const lineText1 = (line ? `${line.text} ` : '') + words[i];
206
+ const lineWidth1 = this.measureWidth(lineText1);
207
+ if ((lineWidth0 + lineWidth1) / 2 < targetWidth && i !== n - 1) {
208
+ line.width = lineWidth0;
209
+ lineWidth0 = lineWidth1;
210
+ line.text = lineText1;
211
+ }
212
+ else {
213
+ // if line is too long or this is the last line (counter), push to next line
214
+ lineWidth0 = this.measureWidth(words[i]);
215
+ line = { width: lineWidth0, text: words[i] };
216
+ d._meta.lines.push(line);
210
217
  }
211
- return label;
212
218
  }
213
- return '';
219
+ // 4. compute the bounding radius
220
+ let radius = 0;
221
+ for (let i = 0, n = d._meta.lines.length; i < n; i += 1) {
222
+ const dy = (Math.abs(i - n / 2) + 0.8) * d._meta.lineHeight;
223
+ const dx = d._meta.lines[i].width / 2;
224
+ radius = Math.max(radius, Math.sqrt(dx * dx + dy * dy));
225
+ }
226
+ d._meta.textRadius = radius;
227
+ return d;
214
228
  })
215
- .join('tspan')
229
+ .attr('transform', (d) => {
230
+ const scale = ((d.r * 0.8) / d._meta.textRadius) * textScalingFactor;
231
+ return `scale(${scale})`;
232
+ })
233
+ .filter((d) => (d.r / 4 > 2.5))
234
+ .selectAll('tspan')
235
+ .data((d) => d._meta.lines)
236
+ .enter()
237
+ .append('tspan')
216
238
  .attr('x', 0)
217
- .attr('y', (d, i, nodes) => `${i - (nodes.length + 1) / 2 + 0.97}em`)
218
- .attr('fill', 'white')
219
- .text((d) => d)
239
+ .attr('y', (d, i, n) => (i - n.length / 2 + 0.8) * defaultLineHeight)
240
+ .attr('class', (d, i, n) => (
241
+ // if it's the last label and a valid number, mark as counter
242
+ i === n.length - 1 && this.isValidNumber(d.text) ? 'label-counter' : 'label-text'))
243
+ .text((d) => d.text)
220
244
  .attr('fill-opacity', 0)
221
245
  .transition(t) // enter() transition on <tspan>
222
246
  .attr('fill-opacity', 1);
223
- g.append('text') // Count label
224
- .attr('class', 'label-count')
247
+ // custom style for the counter
248
+ g.selectAll('tspan.label-counter')
225
249
  .attr('font-family', () => {
226
250
  if (fontRendering && fontRendering.counter && fontRendering.counter.family) {
227
251
  return fontRendering.counter.family;
@@ -233,25 +257,7 @@ export class BubbleChartComponent {
233
257
  return fontRendering.counter.weight;
234
258
  }
235
259
  return 'inherit';
236
- })
237
- .attr('fill', 'white')
238
- .text((d) => {
239
- if (d.r / 4 > 2.5) {
240
- // show text and number threshhold
241
- return d.data.count;
242
- }
243
- return '';
244
- })
245
- .attr('y', (d) => {
246
- let labelLength = d.data.entity.label.split(/ +/g);
247
- if (labelLength.length > 3) {
248
- labelLength = labelLength.slice(0, 3);
249
- }
250
- return `${labelLength.length - (labelLength.length + 1) / 2 + 0.97}em`;
251
- })
252
- .attr('fill-opacity', 0)
253
- .transition(t) // enter() transition on <text>
254
- .attr('fill-opacity', 1);
260
+ });
255
261
  leaf
256
262
  .exit() // EXIT CYCLE
257
263
  .remove();
@@ -273,7 +279,8 @@ export class BubbleChartComponent {
273
279
  return 'scale(.08)';
274
280
  });
275
281
  }
276
- this.emit('d3end', data); // communicate end of draw
282
+ // communicate end of draw
283
+ this.emit('d3end', data);
277
284
  };
278
285
  }
279
286
  ngAfterContentChecked() {
@@ -301,6 +308,13 @@ export class BubbleChartComponent {
301
308
  return;
302
309
  this.emit('click', payload);
303
310
  }
311
+ removeUnneededNodes(svg) {
312
+ // select all nodes and rejoin data
313
+ const nodes = svg.selectAll('g')
314
+ .data(this.data);
315
+ // remove excess nodes
316
+ nodes.exit().remove();
317
+ }
304
318
  }
305
319
  BubbleChartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BubbleChartComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
306
320
  BubbleChartComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: BubbleChartComponent, selector: "n7-bubble-chart", inputs: { data: "data", emit: "emit" }, ngImport: i0, template: "<div *ngIf=\"data\" class=\"n7-bubble-chart {{ data.classes || '' }}\">\n <svg #bubbleChart id=\"{{data.containerId}}\"></svg>\n</div>", dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
@@ -312,4 +326,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
312
326
  }], emit: [{
313
327
  type: Input
314
328
  }] } });
315
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bubble-chart.js","sourceRoot":"","sources":["../../../../../../projects/dv-components-lib/src/lib/components/bubble-chart/bubble-chart.ts","../../../../../../projects/dv-components-lib/src/lib/components/bubble-chart/bubble-chart.html"],"names":[],"mappings":"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","sourcesContent":["//---------------------------\n// BUBBLECHART.ts\n//---------------------------\nimport { Component, Input, AfterContentChecked } from '@angular/core';\n\n/**\n * Interface for BubbleChartComponent's \"data\"\n *\n * @property containerId (required)\n * @property containerWidth (required)\n * @property containerHeight (required)\n * @property bubblesData (required)\n * @property isForceSimulationEnabled (required)\n * @property classes (optional)\n * @property reset (optional)\n*/\nexport interface BubbleChartData {\n  /**\n   * unique identifier for the bubble-chart\n   */\n  containerId: string;\n  /**\n   * total width for the bubble-chart\n   */\n  width: number;\n  /**\n   * total height for the bubble-chart\n   */\n  height: number;\n  /**\n   * data about the bubbles\n   */\n  data: BubbleChartDataItem[];\n  /**\n   * additional css classes\n   */\n  classes?: string;\n  /**\n   * list of ID's of selected items\n   * (selected items will render with a 'X' icon)\n   */\n  selected?: string[];\n  /**\n   * Specify a list of typeOfEntity types (domain),\n   * and a list of colors (range), which will be mapped together.\n   */\n  colorMatch?: {\n    domain: string[];\n    range: string[];\n  };\n  /**\n   * Define a max and min size for the bubbles\n   */\n  sizeRange: [number, number];\n  /**\n   * If defined, transition will be activated\n   * and the value will be used as duration\n   */\n  transition?: number;\n  /**\n   * If set to false, stops the shuffling of the data\n   * before rendering.\n   */\n  shuffle?: boolean;\n  /**\n   * Defines the font-family and the font-weight for each text group\n   */\n  fontRendering?: {\n    label?: {\n      family?: string;\n      weight?: string;\n    };\n    counter?: {\n      family?: string;\n      weight?: string;\n    };\n  };\n\n  /**\n   * expose the draw function outside of n7-frontend/components library\n   * this is needed to redraw bubble-chart-component on command\n   */\n  setDraw?: any;\n}\n\n/**\n * Interface for D3Chart's \"data\"\n *\n * @property entity (required)\n * - id (required)\n * - label (optional)\n * - typeOfEntity (optional)\n * @property count (required)\n*/\nexport interface BubbleChartDataItem {\n  entity: {\n    id: string;\n    label?: string;\n    typeOfEntity?: string;\n  };\n  count: number;\n}\n\n@Component({\n  selector: 'n7-bubble-chart',\n  templateUrl: './bubble-chart.html'\n})\nexport class BubbleChartComponent implements AfterContentChecked {\n  @Input() data: BubbleChartData;\n\n  @Input() emit: any;\n\n  private d3;\n\n  private _loaded = false;\n\n  ngAfterContentChecked() {\n    /*\n     Waits for the dom to be loaded, then fires the draw function\n     that renders the chart.\n    */\n    if (this.data) {\n      if (this._loaded) return;\n      this._loaded = true;\n      setTimeout(() => {\n        import('d3').then((module) => {\n          this.d3 = module;\n          this.draw();\n          if (this.data && this.data.setDraw) {\n            this.data.setDraw(this.draw);\n          }\n        });\n      });\n    }\n  }\n\n  onClick(payload) {\n    if (!this.emit) return;\n    this.emit('click', payload);\n  }\n\n  draw = () => {\n    const { d3 } = this;\n    const {\n      containerId, data, height,\n      width, selected, transition,\n      colorMatch, shuffle,\n      fontRendering\n    } = this.data;\n    const closeIconPath = 'M -50,40 L-40,50 0,10 40,50 50,40 10,0 50,-40 40,-50 0,-10 -40,-50 -50,-40 -10,0 -50,40';\n\n    if (!Array.isArray(data)) {\n      // Check if it is possible to draw with the current dataset\n      console.warn('(n7-bubble-chart) The data object is not in the expected format!');\n      return;\n    }\n\n    let t = d3\n      .transition()\n      .duration(0);\n    if (typeof transition === 'number') {\n      t = d3\n        .transition()\n        .duration(transition)\n        .ease(d3.easeCubicInOut);\n    }\n\n    const colorMap = d3.scaleOrdinal()\n      .domain(colorMatch ? colorMatch.domain : ['persona', 'luogo', 'organizzazione', 'cosa notevole'])\n      .range(colorMatch ? colorMatch.range : d3.schemeTableau10);\n\n    const sizeScale = d3 // map entity count to bubble size\n      .scaleLinear()\n      .domain([0, d3.max(data, (d) => +d.count)])\n      .range([3, d3.max(data, (d) => +d.count)]);\n\n    const pack = (children) => d3\n      .pack()\n      .size([width - 2, height - 2])\n      .padding(1.5)(\n        d3.hierarchy({ children }).sum((d) => sizeScale(d.count))\n      );\n\n    const root = () => { // if shuffle is undefined or true, shuffle the data\n      if (typeof shuffle === 'undefined' || shuffle) {\n        const shuffData = data.slice(); // do not modify the source data!\n        return pack(d3.shuffle(shuffData));\n      } // if shuffle is set to false, skip the data shuffle\n      return pack(data);\n    };\n\n    const svg = d3\n      .select(`#${containerId}`)\n      .attr('viewBox', [0, 0, width, height])\n      .attr('font-family', 'Verdana, Geneva, sans-serif')\n      .attr('text-anchor', 'middle');\n\n    const leaf = svg.selectAll('g').data(root().leaves(), (d) => d.data.entity.id);\n    leaf\n      .transition(t) // update transition on <g>\n      .attr('fill-opacity', 1)\n      .attr('transform', (d) => `translate(${d.x + 1},${d.y + 1})`)\n      .attr('font-size', (d) => {\n        let size = d.r / 5.5;\n        size *= 1;\n        size += 1;\n        return `${Math.round(size)}px`;\n      });\n\n    leaf.selectAll('.close-icon').remove(); // clear all existing close icons\n    if (selected) {\n      leaf // render necessary close icons\n        .filter((d) => selected.includes(d.data.entity.id))\n        .append('path')\n        .attr('class', 'close-icon')\n        .attr('d', closeIconPath)\n        .attr('fill', '#fff')\n        .attr('transform', (d) => {\n          if (d.r / 4 > 3) {\n            return `scale(.08) translate(0, ${d.r * 10 - 80})`;\n          }\n          return 'scale(.08)';\n        });\n    }\n\n    leaf\n      .select('circle')\n      .transition(t) // update transition on <circle>\n      .attr('fill-opacity', 1)\n      .attr('r', (d) => d.r);\n\n    leaf\n      .select('text')\n      .attr('font-family', () => {\n        if (fontRendering && fontRendering.label && fontRendering.label.family) {\n          return fontRendering.label.family;\n        }\n        return 'inherit';\n      })\n      .attr('font-weight', () => {\n        if (fontRendering && fontRendering.label && fontRendering.label.weight) {\n          return fontRendering.label.weight;\n        }\n        return 'inherit';\n      })\n      .selectAll('tspan')\n      .data((d) => {\n        if (d.r / 4 > 4.5) {\n          // show text and number threshhold\n          let label = (\n            d.data.entity.label.charAt(0).toUpperCase()\n            + d.data.entity.label.slice(1)\n          ).split(/ +/g);\n          if (label.length > 3) {\n            label = label.slice(0, 3);\n            label[2] += '…';\n          }\n          return label;\n        } if (d.r / 4 > 2.5) {\n          // show text threshhold\n          let label = (\n            d.data.entity.label.charAt(0).toUpperCase()\n            + d.data.entity.label.slice(1)\n          ).split(/ +/g);\n          if (label.length > 3) {\n            label = label.slice(0, 3);\n            label[2] += '…';\n          }\n          return label;\n        }\n        return '';\n      })\n      .join('tspan')\n      .attr('x', 0)\n      .attr('y', (d, i, nodes) => `${i - (nodes.length + 1) / 2 + 0.97}em`)\n      .attr('fill', 'white')\n      .text((d) => d);\n\n    leaf\n      .select('.label-count')\n      .attr('font-family', () => {\n        if (fontRendering && fontRendering.counter && fontRendering.counter.family) {\n          return fontRendering.counter.family;\n        }\n        return 'inherit';\n      })\n      .attr('font-weight', () => {\n        if (fontRendering && fontRendering.counter && fontRendering.counter.weight) {\n          return fontRendering.counter.weight;\n        }\n        return 'inherit';\n      })\n      .attr('fill', 'white')\n      .text((d) => {\n        if (d.r / 4 > 2.5) {\n          // show text and number threshhold\n          return d.data.count;\n        }\n        return '';\n      })\n      .attr('y', (d) => {\n        let labelLength = d.data.entity.label.split(/ +/g);\n        if (labelLength.length > 3) {\n          labelLength = labelLength.slice(0, 3);\n        }\n        return `${labelLength.length - (labelLength.length + 1) / 2 + 0.97}em`;\n      });\n\n\n    const g = leaf.enter().append('g');\n\n    g.attr('transform', (d) => `translate(${d.x + 1},${d.y + 1})`)\n      .attr('font-size', (d) => {\n        let size = d.r / 5.5;\n        size *= 1;\n        size += 1;\n        return `${Math.round(size)}px`;\n      })\n      .attr('cursor', 'pointer')\n      .on('click', (event, d) => {\n        this.onClick(d.data.entity.id);\n      })\n      .attr('id', (d) => `g_${d.data.entity.id}`)\n      .append('circle')\n      .attr('id', (d) => { d.leafUid = d.data.entity.id; })\n      .attr('r', 0)\n      .transition(t) // enter() transition on <circle>\n      .attr('fill-opacity', 1)\n      .attr('fill', (d) => colorMap(d.data.entity.typeOfEntity))\n      .attr('r', (d) => d.r);\n\n    g.append('clipPath')\n      .attr('id', (d) => { d.clipUid = `Clip-${d.data.entity.id}`; })\n      .append('use')\n      .attr('xlink:href', (d) => d.leafUid.href);\n\n    g.append('text')\n      .attr('font-family', () => {\n        if (fontRendering && fontRendering.label && fontRendering.label.family) {\n          return fontRendering.label.family;\n        }\n        return 'inherit';\n      })\n      .attr('font-weight', () => {\n        if (fontRendering && fontRendering && fontRendering.label && fontRendering.label.weight) {\n          return fontRendering.label.weight;\n        }\n        return 'inherit';\n      })\n      .selectAll('tspan')\n      .data((d) => {\n        if (d.r / 4 > 4.5) {\n          // show text and number threshhold\n          let label = (\n            d.data.entity.label.charAt(0).toUpperCase()\n            + d.data.entity.label.slice(1)\n          ).split(/ +/g);\n          if (label.length > 3) {\n            label = label.slice(0, 3);\n            label[2] += '…';\n          }\n          return label;\n        } if (d.r / 4 > 2.5) {\n          // show text threshhold\n          let label = (\n            d.data.entity.label.charAt(0).toUpperCase()\n            + d.data.entity.label.slice(1)\n          ).split(/ +/g);\n          if (label.length > 3) {\n            label = label.slice(0, 3);\n            label[2] += '…';\n          }\n          return label;\n        }\n        return '';\n      })\n      .join('tspan')\n      .attr('x', 0)\n      .attr('y', (d, i, nodes) => `${i - (nodes.length + 1) / 2 + 0.97}em`)\n      .attr('fill', 'white')\n      .text((d) => d)\n      .attr('fill-opacity', 0)\n      .transition(t) // enter() transition on <tspan>\n      .attr('fill-opacity', 1);\n\n    g.append('text') // Count label\n      .attr('class', 'label-count')\n      .attr('font-family', () => {\n        if (fontRendering && fontRendering.counter && fontRendering.counter.family) {\n          return fontRendering.counter.family;\n        }\n        return 'inherit';\n      })\n      .attr('font-weight', () => {\n        if (fontRendering && fontRendering.counter && fontRendering.counter.weight) {\n          return fontRendering.counter.weight;\n        }\n        return 'inherit';\n      })\n      .attr('fill', 'white')\n      .text((d) => {\n        if (d.r / 4 > 2.5) {\n          // show text and number threshhold\n          return d.data.count;\n        }\n        return '';\n      })\n      .attr('y', (d) => {\n        let labelLength = d.data.entity.label.split(/ +/g);\n        if (labelLength.length > 3) {\n          labelLength = labelLength.slice(0, 3);\n        }\n        return `${labelLength.length - (labelLength.length + 1) / 2 + 0.97}em`;\n      })\n      .attr('fill-opacity', 0)\n      .transition(t) // enter() transition on <text>\n      .attr('fill-opacity', 1);\n\n    leaf\n      .exit() // EXIT CYCLE\n      .remove();\n\n    if (selected) {\n      g.filter((d) => selected.includes(d.leafUid)) // append 'X' icon // only for selected bubbles\n        .append('path')\n        .attr('d', closeIconPath)\n        .attr('class', 'close-icon')\n        .attr('fill', (d) => {\n          if (d.r / 4 > 2.5) {\n            return '#fff';\n          }\n          return 'transparent';\n        })\n        .attr('transform', (d) => {\n          if (d.r / 4 > 3) {\n            return `scale(.08) translate(0, ${d.r * 10 - 80})`;\n          }\n          return 'scale(.08)';\n        });\n    }\n\n    this.emit('d3end', data); // communicate end of draw\n  }\n}\n","<div *ngIf=\"data\" class=\"n7-bubble-chart {{ data.classes || '' }}\">\n    <svg #bubbleChart id=\"{{data.containerId}}\"></svg>\n</div>"]}
329
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bubble-chart.js","sourceRoot":"","sources":["../../../../../../projects/dv-components-lib/src/lib/components/bubble-chart/bubble-chart.ts","../../../../../../projects/dv-components-lib/src/lib/components/bubble-chart/bubble-chart.html"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,iBAAiB;AACjB,6BAA6B;AAE7B;;;;;;;;;;;;;;;;;GAiBG;AAEH,OAAO,EAAE,SAAS,EAAE,KAAK,EAAuB,MAAM,eAAe,CAAC;;;AA0HtE,MAAM,OAAO,oBAAoB;IAJjC;QAWU,YAAO,GAAG,KAAK,CAAC;QA2BxB;;;WAGG;QACH,iBAAY,GAAG,CAAC,IAAI,EAAE,EAAE;YACtB,MAAM,OAAO,GAAG,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YAClE,uDAAuD;YACvD,IAAI,IAAI,CAAC,IAAI,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,IAAI,IAAI,CAAC,IAAI,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,EAAE;gBACtF,OAAO,CAAC,IAAI,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,MAAM,IAAI,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;aAClG;YACD,6BAA6B;YAC7B,OAAO,OAAO,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC;QAChD,CAAC,CAAC;QAEF,kBAAa,GAAG,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC;QAUnE,SAAI,GAAG,GAAG,EAAE;YACV,MAAM,EAAE,EAAE,EAAE,GAAG,IAAI,CAAC;YACpB,MAAM,EACJ,WAAW,EAAE,IAAI,EAAE,MAAM,EACzB,KAAK,EAAE,QAAQ,EAAE,UAAU,EAC3B,UAAU,EAAE,OAAO,EACnB,aAAa,EACd,GAAG,IAAI,CAAC,IAAI,CAAC;YACd,oCAAoC;YACpC,MAAM,aAAa,GAAG,yFAAyF,CAAC;YAChH,MAAM,iBAAiB,GAAG,EAAE,CAAC;YAC7B,6CAA6C;YAC7C,MAAM,iBAAiB,GAAG,CAAC,CAAC;YAC5B,MAAM,iBAAiB,GAAG,CAAC,CAAC;YAE5B,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;gBACxB,2DAA2D;gBAC3D,OAAO,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAC;gBACjF,OAAO;aACR;YAED,yDAAyD;YACzD,IAAI,CAAC,GAAG,EAAE;iBACP,UAAU,EAAE;iBACZ,QAAQ,CAAC,CAAC,CAAC,CAAC;YACf,IAAI,OAAO,UAAU,KAAK,QAAQ,EAAE;gBAClC,CAAC,GAAG,EAAE;qBACH,UAAU,EAAE;qBACZ,QAAQ,CAAC,UAAU,CAAC;qBACpB,IAAI,CAAC,EAAE,CAAC,cAAc,CAAC,CAAC;aAC5B;YAED,gEAAgE;YAChE,MAAM,QAAQ,GAAG,EAAE,CAAC,YAAY,EAAE;iBAC/B,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,OAAO,EAAE,gBAAgB,EAAE,eAAe,CAAC,CAAC;iBAChG,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,eAAe,CAAC,CAAC;YAE7D,uDAAuD;YACvD,MAAM,SAAS,GAAG,EAAE,CAAC,kCAAkC;iBACpD,WAAW,EAAE;iBACb,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;iBAC1C,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAE7C,kFAAkF;YAClF,8DAA8D;YAC9D,MAAM,IAAI,GAAG,CAAC,QAAQ,EAAE,EAAE,CAAC,EAAE;iBAC1B,IAAI,EAAE;iBACN,IAAI,CAAC,CAAC,KAAK,GAAG,CAAC,EAAE,MAAM,GAAG,CAAC,CAAC,CAAC;iBAC7B,OAAO,CAAC,GAAG,CAAC,CACX,EAAE,CAAC,SAAS,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAC1D,CAAC;YAEJ,kEAAkE;YAClE,kEAAkE;YAClE,MAAM,IAAI,GAAG,GAAG,EAAE;gBAChB,IAAI,OAAO,OAAO,KAAK,WAAW,IAAI,OAAO,EAAE;oBAC7C,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,iCAAiC;oBACjE,OAAO,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC;iBACpC,CAAC,oDAAoD;gBACtD,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC;YACpB,CAAC,CAAC;YAEF,6CAA6C;YAC7C,MAAM,GAAG,GAAG,EAAE;iBACX,MAAM,CAAC,IAAI,WAAW,EAAE,CAAC;iBACzB,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;gBACvC,sDAAsD;gBACtD,6BAA6B;iBAC5B,KAAK,CAAC,MAAM,EAAE,kCAAkC,CAAC;iBACjD,KAAK,CAAC,QAAQ,EAAE,MAAM,CAAC;iBACvB,KAAK,CAAC,WAAW,EAAE,MAAM,CAAC;iBAC1B,IAAI,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAC;YAEjC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,CAAC;YAE9B,oDAAoD;YACpD,MAAM,IAAI,GAAG,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YAC/E,IAAI;iBACD,UAAU,CAAC,CAAC,CAAC,CAAC,2BAA2B;iBACzC,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;iBACvB,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YAChE,8BAA8B;YAC9B,0BAA0B;YAC1B,eAAe;YACf,eAAe;YACf,oCAAoC;YACpC,MAAM;YAEN,kDAAkD;YAClD,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,MAAM,EAAE,CAAC;YACvC,IAAI,QAAQ,EAAE;gBACZ,IAAI,CAAC,wCAAwC;qBAC1C,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;qBAClD,MAAM,CAAC,MAAM,CAAC;qBACd,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC;qBAC3B,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC;qBACxB,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC;qBACpB,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE;oBACvB,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;wBACf,OAAO,2BAA2B,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;qBACpD;oBACD,OAAO,YAAY,CAAC;gBACtB,CAAC,CAAC,CAAC;aACN;YAED,IAAI;iBACD,MAAM,CAAC,QAAQ,CAAC;iBAChB,UAAU,CAAC,CAAC,CAAC,CAAC,gCAAgC;iBAC9C,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;iBACvB,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAEzB,sCAAsC;YACtC,kEAAkE;YAClE,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YAEnC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC;iBAC3D,IAAI,CAAC,QAAQ,EAAE,SAAS,CAAC;iBACzB,EAAE,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE;gBACxB,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACjC,CAAC,CAAC;iBACD,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;iBAC1C,MAAM,CAAC,QAAQ,CAAC;iBAChB,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,GAAG,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;iBACpD,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;iBACZ,UAAU,CAAC,CAAC,CAAC,CAAC,iCAAiC;iBAC/C,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;iBACvB,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;iBACzD,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAEzB,gDAAgD;YAChD,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC;iBACjB,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,GAAG,CAAC,CAAC,OAAO,GAAG,QAAQ,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;iBAC9D,MAAM,CAAC,KAAK,CAAC;iBACb,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YAE7C,qBAAqB;YACrB,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC;iBACb,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE;gBACxB,IAAI,aAAa,IAAI,aAAa,CAAC,KAAK,IAAI,aAAa,CAAC,KAAK,CAAC,MAAM,EAAE;oBACtE,OAAO,aAAa,CAAC,KAAK,CAAC,MAAM,CAAC;iBACnC;gBACD,OAAO,SAAS,CAAC;YACnB,CAAC,CAAC;iBACD,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE;gBACxB,IAAI,aAAa,IAAI,aAAa,IAAI,aAAa,CAAC,KAAK,IAAI,aAAa,CAAC,KAAK,CAAC,MAAM,EAAE;oBACvF,OAAO,aAAa,CAAC,KAAK,CAAC,MAAM,CAAC;iBACnC;gBACD,OAAO,SAAS,CAAC;YACnB,CAAC,CAAC;iBACD,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC;iBACrB,IAAI,CAAC,CAAC,CAAa,EAAE,EAAE;gBACtB,2CAA2C;gBAC3C,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,CAAC,CAAC,IAAI,CAAC,MAAM;qBAChC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE;sBAC5B,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBACjC,4BAA4B;gBAC5B,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,OAAO,CAAC,CAAC,KAAK,KAAK,QAAQ;oBAAE,CAAC,CAAC,KAAK,GAAG,EAAE,CAAC;gBAC1D,uCAAuC;gBACvC,MAAM,KAAK,GAAG,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,6BAA6B;gBAChF,mEAAmE;gBACnE,IAAI,KAAK,CAAC,MAAM,GAAG,iBAAiB,EAAE;oBACpC,KAAK,CAAC,MAAM,CAAC,iBAAiB,EAAE,KAAK,CAAC,MAAM,GAAG,iBAAiB,CAAC,CAAC;oBAClE,KAAK,CAAC,iBAAiB,GAAG,CAAC,CAAC,IAAI,GAAG,CAAC;iBACrC;gBACD,cAAc;gBACd,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,EAAE;oBACjB,sBAAsB;oBACtB,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC;wBAAE,KAAK,CAAC,GAAG,EAAE,CAAC;oBAC1C,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;wBAAE,KAAK,CAAC,KAAK,EAAE,CAAC;iBAC9B;gBAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,EAAE;oBACnB,wBAAwB;oBACxB,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;iBAC/B;gBACD,CAAC,CAAC,KAAK,CAAC,KAAK,GAAG,KAAK,CAAC;gBACtB,CAAC,CAAC,KAAK,CAAC,UAAU,GAAG,iBAAiB,CAAC;gBACvC,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAC3B,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC,GAAG,iBAAiB,CACtE,CAAC;gBACF,yBAAyB;gBACzB,CAAC,CAAC,KAAK,CAAC,KAAK,GAAG,EAAE,CAAC;gBACnB,IAAI,IAAI,CAAC;gBACT,IAAI,UAAU,GAAG,QAAQ,CAAC;gBAC1B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE;oBACvD,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;oBAC3D,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;oBAChD,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC,GAAG,CAAC,GAAG,WAAW,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE;wBAC9D,IAAI,CAAC,KAAK,GAAG,UAAU,CAAC;wBACxB,UAAU,GAAG,UAAU,CAAC;wBACxB,IAAI,CAAC,IAAI,GAAG,SAAS,CAAC;qBACvB;yBAAM;wBACL,4EAA4E;wBAC5E,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;wBACzC,IAAI,GAAG,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC;wBAC7C,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;qBAC1B;iBACF;gBACD,iCAAiC;gBACjC,IAAI,MAAM,GAAG,CAAC,CAAC;gBACf,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE;oBACvD,MAAM,EAAE,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC;oBAC5D,MAAM,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC;oBACtC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC;iBACzD;gBACD,CAAC,CAAC,KAAK,CAAC,UAAU,GAAG,MAAM,CAAC;gBAC5B,OAAO,CAAC,CAAC;YACX,CAAC,CAAC;iBACD,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE;gBACvB,MAAM,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,iBAAiB,CAAC;gBACrE,OAAO,SAAS,KAAK,GAAG,CAAC;YAC3B,CAAC,CAAC;iBACD,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC,CAAC;iBAC9B,SAAS,CAAC,OAAO,CAAC;iBAClB,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC;iBAC1B,KAAK,EAAE;iBACP,MAAM,CAAC,OAAO,CAAC;iBACf,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;iBACZ,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,MAAM,GAAG,CAAC,GAAG,GAAG,CAAC,GAAG,iBAAiB,CAAC;iBACpE,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;YAC1B,6DAA6D;YAC7D,CAAC,KAAK,CAAC,CAAC,MAAM,GAAG,CAAC,IAAI,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,YAAY,CAClF,CAAC;iBACD,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;iBACnB,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;iBACvB,UAAU,CAAC,CAAC,CAAC,CAAC,gCAAgC;iBAC9C,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;YAE3B,+BAA+B;YAC/B,CAAC,CAAC,SAAS,CAAC,qBAAqB,CAAC;iBAC/B,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE;gBACxB,IAAI,aAAa,IAAI,aAAa,CAAC,OAAO,IAAI,aAAa,CAAC,OAAO,CAAC,MAAM,EAAE;oBAC1E,OAAO,aAAa,CAAC,OAAO,CAAC,MAAM,CAAC;iBACrC;gBACD,OAAO,SAAS,CAAC;YACnB,CAAC,CAAC;iBACD,IAAI,CAAC,aAAa,EAAE,GAAG,EAAE;gBACxB,IAAI,aAAa,IAAI,aAAa,CAAC,OAAO,IAAI,aAAa,CAAC,OAAO,CAAC,MAAM,EAAE;oBAC1E,OAAO,aAAa,CAAC,OAAO,CAAC,MAAM,CAAC;iBACrC;gBACD,OAAO,SAAS,CAAC;YACnB,CAAC,CAAC,CAAC;YAEL,IAAI;iBACD,IAAI,EAAE,CAAC,aAAa;iBACpB,MAAM,EAAE,CAAC;YAEZ,IAAI,QAAQ,EAAE;gBACZ,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,+CAA+C;qBAC1F,MAAM,CAAC,MAAM,CAAC;qBACd,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC;qBACxB,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC;qBAC3B,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE;oBAClB,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,EAAE;wBACjB,OAAO,MAAM,CAAC;qBACf;oBACD,OAAO,aAAa,CAAC;gBACvB,CAAC,CAAC;qBACD,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE;oBACvB,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;wBACf,OAAO,2BAA2B,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;qBACpD;oBACD,OAAO,YAAY,CAAC;gBACtB,CAAC,CAAC,CAAC;aACN;YAED,0BAA0B;YAC1B,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;QAC3B,CAAC,CAAC;KACH;IA5TC,qBAAqB;QACnB;;;UAGE;QACF,IAAI,IAAI,CAAC,IAAI,EAAE;YACb,IAAI,IAAI,CAAC,OAAO;gBAAE,OAAO;YACzB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;YACpB,UAAU,CAAC,GAAG,EAAE;gBACd,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE;oBAC3B,IAAI,CAAC,EAAE,GAAG,MAAM,CAAC;oBACjB,IAAI,CAAC,IAAI,EAAE,CAAC;oBACZ,IAAI,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;wBAClC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;qBAC9B;gBACH,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAED,OAAO,CAAC,OAAO;QACb,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO;QACvB,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IAC9B,CAAC;IAkBD,mBAAmB,CAAC,GAAG;QACrB,mCAAmC;QACnC,MAAM,KAAK,GAAG,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC;aAC7B,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACnB,sBAAsB;QACtB,KAAK,CAAC,IAAI,EAAE,CAAC,MAAM,EAAE,CAAC;IACxB,CAAC;;kHAxDU,oBAAoB;sGAApB,oBAAoB,+FCjJjC,2IAEM;4FD+IO,oBAAoB;kBAJhC,SAAS;+BACE,iBAAiB;8BAIlB,IAAI;sBAAZ,KAAK;gBAEG,IAAI;sBAAZ,KAAK","sourcesContent":["//---------------------------\n// BUBBLECHART.ts\n//---------------------------\n\n/**\n * The bubble chart is drawn using d3js, a library to create and update anything in svg.\n *\n * LEGEND:\n * - svg -> the canvas/<svg> element where everything will be drawn on (appended).\n * - g -> <g> is a group of svg elements.\n * - leaf -> this is the svg circle, the bubble.\n * - text -> this is all the text inside the circle.\n * - tspan -> this is just one line of text.\n *\n * What are \"join, enter, update, exit?\"\n * https://observablehq.com/@d3/learn-d3-joins?collection=@d3/learn-d3\n * https://observablehq.com/@thetylerwolf/day-18-join-enter-update-exit\n *\n * Each line can have a different width, but all of the text inside a circle\n * needs to be scaled by a constant factor.\n * https://observablehq.com/@mbostock/fit-text-to-circle\n */\n\nimport { Component, Input, AfterContentChecked } from '@angular/core';\n\n/**\n * Interface for D3Chart's \"data\"\n *\n * @property entity (required)\n * - id (required)\n * - label (optional)\n * - typeOfEntity (optional)\n * @property count (required)\n*/\nexport interface BubbleChartDataItem {\n  entity: {\n    id: string;\n    label?: string;\n    typeOfEntity?: string;\n  };\n  count: number;\n}\n\n/**\n * Interface for a Circle's node data\n */\nexport interface CircleNode {\n  clipUid: string;\n  data: BubbleChartDataItem;\n  depth: number;\n  height: number;\n  leafUid: string;\n  parent: Node;\n  x: number;\n  y: number;\n  r: number;\n  value: number;\n  /** Dynamic data for internal logic */\n  _meta?: any;\n}\n\n/**\n * Interface for BubbleChartComponent's \"data\"\n *\n * @property containerId (required)\n * @property containerWidth (required)\n * @property containerHeight (required)\n * @property bubblesData (required)\n * @property isForceSimulationEnabled (required)\n * @property classes (optional)\n * @property reset (optional)\n*/\nexport interface BubbleChartData {\n  /**\n   * unique identifier for the bubble-chart\n   */\n  containerId: string;\n  /**\n   * total width for the bubble-chart\n   */\n  width: number;\n  /**\n   * total height for the bubble-chart\n   */\n  height: number;\n  /**\n   * data about the bubbles\n   */\n  data: BubbleChartDataItem[];\n  /**\n   * additional css classes\n   */\n  classes?: string;\n  /**\n   * list of ID's of selected items\n   * (selected items will render with a 'X' icon)\n   */\n  selected?: string[];\n  /**\n   * Specify a list of typeOfEntity types (domain),\n   * and a list of colors (range), which will be mapped together.\n   */\n  colorMatch?: {\n    domain: string[];\n    range: string[];\n  };\n  /**\n   * Define a max and min size for the bubbles\n   */\n  sizeRange: [number, number];\n  /**\n   * If defined, transition will be activated\n   * and the value will be used as duration\n   */\n  transition?: number;\n  /**\n   * If set to false, stops the shuffling of the data\n   * before rendering.\n   */\n  shuffle?: boolean;\n  /**\n   * Defines the font-family and the font-weight for each text group\n   */\n  fontRendering?: {\n    label?: {\n      family?: string;\n      weight?: string;\n    };\n    counter?: {\n      family?: string;\n      weight?: string;\n    };\n  };\n\n  /**\n   * expose the draw function outside of n7-frontend/components library\n   * this is needed to redraw bubble-chart-component on command\n   */\n  setDraw?: any;\n}\n\n@Component({\n  selector: 'n7-bubble-chart',\n  templateUrl: './bubble-chart.html'\n})\nexport class BubbleChartComponent implements AfterContentChecked {\n  @Input() data: BubbleChartData;\n\n  @Input() emit: any;\n\n  private d3;\n\n  private _loaded = false;\n\n  ngAfterContentChecked() {\n    /*\n     Waits for the dom to be loaded, then fires the draw function\n     that renders the chart.\n    */\n    if (this.data) {\n      if (this._loaded) return;\n      this._loaded = true;\n      setTimeout(() => {\n        import('d3').then((module) => {\n          this.d3 = module;\n          this.draw();\n          if (this.data && this.data.setDraw) {\n            this.data.setDraw(this.draw);\n          }\n        });\n      });\n    }\n  }\n\n  onClick(payload) {\n    if (!this.emit) return;\n    this.emit('click', payload);\n  }\n\n  /**\n   * Reference for much of the new text scaling code comes from:\n   * https://observablehq.com/@mbostock/fit-text-to-circle\n   */\n  measureWidth = (text) => {\n    const context = document.createElement('canvas').getContext('2d');\n    // measure text with the correct font family and weight\n    if (this.data?.fontRendering?.label?.family && this.data?.fontRendering?.label?.weight) {\n      context.font = `${this.data.fontRendering.label.weight} ${this.data.fontRendering.label.family}`;\n    }\n    // calculated width + padding\n    return context.measureText(text).width * 1.15;\n  };\n\n  isValidNumber = (value) => !Number.isNaN(Number.parseFloat(value));\n\n  removeUnneededNodes(svg) {\n    // select all nodes and rejoin data\n    const nodes = svg.selectAll('g')\n      .data(this.data);\n    // remove excess nodes\n    nodes.exit().remove();\n  }\n\n  draw = () => {\n    const { d3 } = this;\n    const {\n      containerId, data, height,\n      width, selected, transition,\n      colorMatch, shuffle,\n      fontRendering\n    } = this.data;\n    // SVG shape path for the close icon\n    const closeIconPath = 'M -50,40 L-40,50 0,10 40,50 50,40 10,0 50,-40 40,-50 0,-10 -40,-50 -50,-40 -10,0 -50,40';\n    const defaultLineHeight = 13;\n    // word count before truncating with ellipsis\n    const ellipsisThreshold = 4;\n    const textScalingFactor = 1;\n\n    if (!Array.isArray(data)) {\n      // Check if it is possible to draw with the current dataset\n      console.warn('(n7-bubble-chart) The data object is not in the expected format!');\n      return;\n    }\n\n    // animation settings used to render changes in the chart\n    let t = d3\n      .transition()\n      .duration(0);\n    if (typeof transition === 'number') {\n      t = d3\n        .transition()\n        .duration(transition)\n        .ease(d3.easeCubicInOut);\n    }\n\n    // calculate the bubble background color from it's type (domain)\n    const colorMap = d3.scaleOrdinal()\n      .domain(colorMatch ? colorMatch.domain : ['persona', 'luogo', 'organizzazione', 'cosa notevole'])\n      .range(colorMatch ? colorMatch.range : d3.schemeTableau10);\n\n    // calculate how big the radius of the bubble should be\n    const sizeScale = d3 // map entity count to bubble size\n      .scaleLinear()\n      .domain([0, d3.max(data, (d) => +d.count)])\n      .range([3, d3.max(data, (d) => +d.count)]);\n\n    // pack is a d3js method which calculates the bubble's x & y position in the chart\n    // circle packing reference: https://observablehq.com/@d3/pack\n    const pack = (children) => d3\n      .pack()\n      .size([width - 2, height - 2])\n      .padding(1.5)(\n        d3.hierarchy({ children }).sum((d) => sizeScale(d.count))\n      );\n\n    // the bubbles are packed in a single level tree, this is the root\n    // see circle packing reference: https://observablehq.com/@d3/pack\n    const root = () => { // if shuffle is undefined or true, shuffle the data\n      if (typeof shuffle === 'undefined' || shuffle) {\n        const shuffData = data.slice(); // do not modify the source data!\n        return pack(d3.shuffle(shuffData));\n      } // if shuffle is set to false, skip the data shuffle\n      return pack(data);\n    };\n\n    // svg canvas where all the bubbles are drawn\n    const svg = d3\n      .select(`#${containerId}`)\n      .attr('viewBox', [0, 0, width, height])\n      // .attr('font-family', 'Verdana, Geneva, sans-serif')\n      // .attr('font-size', '10px')\n      .style('font', '10px Verdana, Geneva, sans-serif')\n      .style('height', 'auto')\n      .style('max-width', '100%')\n      .attr('text-anchor', 'middle');\n\n    this.removeUnneededNodes(svg);\n\n    // a leaf of the \"pack tree\" corresponds to a bubble\n    const leaf = svg.selectAll('g').data(root().leaves(), (d) => d.data.entity.id);\n    leaf\n      .transition(t) // update transition on <g>\n      .attr('fill-opacity', 1)\n      .attr('transform', (d) => `translate(${d.x + 1},${d.y + 1})`);\n    // .attr('font-size', (d) => {\n    //   let size = d.r / 5.5;\n    //   size *= 1;\n    //   size += 1;\n    //   return `${Math.round(size)}px`;\n    // });\n\n    // clear all existing close icons from the bubbles\n    leaf.selectAll('.close-icon').remove();\n    if (selected) {\n      leaf // render only the necessary close icons\n        .filter((d) => selected.includes(d.data.entity.id))\n        .append('path')\n        .attr('class', 'close-icon')\n        .attr('d', closeIconPath)\n        .attr('fill', '#fff')\n        .attr('transform', (d) => {\n          if (d.r / 4 > 3) {\n            return `scale(.08) translate(0, ${d.r * 10 - 80})`;\n          }\n          return 'scale(.08)';\n        });\n    }\n\n    leaf\n      .select('circle')\n      .transition(t) // update transition on <circle>\n      .attr('fill-opacity', 1)\n      .attr('r', (d) => d.r);\n\n    // g represents a \"group of svg items\"\n    // items grouped together can be added / removed / scaled together\n    const g = leaf.enter().append('g');\n\n    g.attr('transform', (d) => `translate(${d.x + 1},${d.y + 1})`)\n      .attr('cursor', 'pointer')\n      .on('click', (event, d) => {\n        this.onClick(d.data.entity.id);\n      })\n      .attr('id', (d) => `g_${d.data.entity.id}`)\n      .append('circle')\n      .attr('id', (d) => { d.leafUid = d.data.entity.id; })\n      .attr('r', 0)\n      .transition(t) // enter() transition on <circle>\n      .attr('fill-opacity', 1)\n      .attr('fill', (d) => colorMap(d.data.entity.typeOfEntity))\n      .attr('r', (d) => d.r);\n\n    // prevents the text from overflowing the bubble\n    g.append('clipPath')\n      .attr('id', (d) => { d.clipUid = `Clip-${d.data.entity.id}`; })\n      .append('use')\n      .attr('xlink:href', (d) => d.leafUid.href);\n\n    /** NEW TEXT LOGIC */\n    g.append('text')\n      .attr('font-family', () => {\n        if (fontRendering && fontRendering.label && fontRendering.label.family) {\n          return fontRendering.label.family;\n        }\n        return 'inherit';\n      })\n      .attr('font-weight', () => {\n        if (fontRendering && fontRendering && fontRendering.label && fontRendering.label.weight) {\n          return fontRendering.label.weight;\n        }\n        return 'inherit';\n      })\n      .attr('fill', 'white')\n      .each((d: CircleNode) => {\n        // Capitalize the first letter of the label\n        d.data.entity.label = d.data.entity\n          .label.charAt(0).toUpperCase()\n          + d.data.entity.label.slice(1);\n        // 1. initialize meta object\n        if (!d._meta || typeof d._meta !== 'object') d._meta = {};\n        // 2. tokenize label & count into words\n        const words = d.data.entity.label.split(/[\\s]+/g); // To hyphenate: /\\s+|(?<=-)/\n        // Truncate with ellipsis if the label is longer than the threshold\n        if (words.length > ellipsisThreshold) {\n          words.splice(ellipsisThreshold, words.length - ellipsisThreshold);\n          words[ellipsisThreshold - 1] += '…';\n        }\n        // add counter\n        if (d.r / 4 > 2.5) {\n          // show text threshold\n          if (!words[words.length - 1]) words.pop();\n          if (!words[0]) words.shift();\n        } if (d.r / 4 > 4.5) {\n          // show number threshold\n          words.push(`${d.data.count}`);\n        }\n        d._meta.words = words;\n        d._meta.lineHeight = defaultLineHeight;\n        const targetWidth = Math.sqrt(\n          this.measureWidth(d._meta.words.join(' ').trim()) * defaultLineHeight\n        );\n        // 3. build lines of text\n        d._meta.lines = [];\n        let line;\n        let lineWidth0 = Infinity;\n        for (let i = 0, n = d._meta.words.length; i < n; i += 1) {\n          const lineText1 = (line ? `${line.text} ` : '') + words[i];\n          const lineWidth1 = this.measureWidth(lineText1);\n          if ((lineWidth0 + lineWidth1) / 2 < targetWidth && i !== n - 1) {\n            line.width = lineWidth0;\n            lineWidth0 = lineWidth1;\n            line.text = lineText1;\n          } else {\n            // if line is too long or this is the last line (counter), push to next line\n            lineWidth0 = this.measureWidth(words[i]);\n            line = { width: lineWidth0, text: words[i] };\n            d._meta.lines.push(line);\n          }\n        }\n        // 4. compute the bounding radius\n        let radius = 0;\n        for (let i = 0, n = d._meta.lines.length; i < n; i += 1) {\n          const dy = (Math.abs(i - n / 2) + 0.8) * d._meta.lineHeight;\n          const dx = d._meta.lines[i].width / 2;\n          radius = Math.max(radius, Math.sqrt(dx * dx + dy * dy));\n        }\n        d._meta.textRadius = radius;\n        return d;\n      })\n      .attr('transform', (d) => {\n        const scale = ((d.r * 0.8) / d._meta.textRadius) * textScalingFactor;\n        return `scale(${scale})`;\n      })\n      .filter((d) => (d.r / 4 > 2.5))\n      .selectAll('tspan')\n      .data((d) => d._meta.lines)\n      .enter()\n      .append('tspan')\n      .attr('x', 0)\n      .attr('y', (d, i, n) => (i - n.length / 2 + 0.8) * defaultLineHeight)\n      .attr('class', (d, i, n) => (\n        // if it's the last label and a valid number, mark as counter\n        i === n.length - 1 && this.isValidNumber(d.text) ? 'label-counter' : 'label-text'\n      ))\n      .text((d) => d.text)\n      .attr('fill-opacity', 0)\n      .transition(t) // enter() transition on <tspan>\n      .attr('fill-opacity', 1);\n\n    // custom style for the counter\n    g.selectAll('tspan.label-counter')\n      .attr('font-family', () => {\n        if (fontRendering && fontRendering.counter && fontRendering.counter.family) {\n          return fontRendering.counter.family;\n        }\n        return 'inherit';\n      })\n      .attr('font-weight', () => {\n        if (fontRendering && fontRendering.counter && fontRendering.counter.weight) {\n          return fontRendering.counter.weight;\n        }\n        return 'inherit';\n      });\n\n    leaf\n      .exit() // EXIT CYCLE\n      .remove();\n\n    if (selected) {\n      g.filter((d) => selected.includes(d.leafUid)) // append 'X' icon // only for selected bubbles\n        .append('path')\n        .attr('d', closeIconPath)\n        .attr('class', 'close-icon')\n        .attr('fill', (d) => {\n          if (d.r / 4 > 2.5) {\n            return '#fff';\n          }\n          return 'transparent';\n        })\n        .attr('transform', (d) => {\n          if (d.r / 4 > 3) {\n            return `scale(.08) translate(0, ${d.r * 10 - 80})`;\n          }\n          return 'scale(.08)';\n        });\n    }\n\n    // communicate end of draw\n    this.emit('d3end', data);\n  };\n}\n","<div *ngIf=\"data\" class=\"n7-bubble-chart {{ data.classes || '' }}\">\n    <svg #bubbleChart id=\"{{data.containerId}}\"></svg>\n</div>"]}