@orbcharts/plugins-basic 3.0.0-alpha.68 → 3.0.0-alpha.70

Sign up to get free protection for your applications and to get access to all the features.
Files changed (79) hide show
  1. package/LICENSE +200 -200
  2. package/dist/orbcharts-plugins-basic.es.js +4838 -4760
  3. package/dist/orbcharts-plugins-basic.umd.js +15 -14
  4. package/dist/src/utils/d3Graphics.d.ts +10 -0
  5. package/package.json +42 -42
  6. package/src/base/BaseBarStack.ts +779 -779
  7. package/src/base/BaseBars.ts +764 -764
  8. package/src/base/BaseBarsTriangle.ts +672 -672
  9. package/src/base/BaseDots.ts +513 -513
  10. package/src/base/BaseGroupAxis.ts +675 -652
  11. package/src/base/BaseLegend.ts +642 -642
  12. package/src/base/BaseLineAreas.ts +628 -628
  13. package/src/base/BaseLines.ts +704 -704
  14. package/src/base/BaseValueAxis.ts +578 -578
  15. package/src/base/types.ts +2 -2
  16. package/src/grid/defaults.ts +128 -128
  17. package/src/grid/gridObservables.ts +543 -543
  18. package/src/grid/index.ts +15 -15
  19. package/src/grid/plugins/BarStack.ts +43 -43
  20. package/src/grid/plugins/Bars.ts +44 -44
  21. package/src/grid/plugins/BarsPN.ts +41 -41
  22. package/src/grid/plugins/BarsTriangle.ts +42 -42
  23. package/src/grid/plugins/Dots.ts +37 -37
  24. package/src/grid/plugins/GridLegend.ts +59 -59
  25. package/src/grid/plugins/GroupAux.ts +1014 -991
  26. package/src/grid/plugins/GroupAxis.ts +36 -36
  27. package/src/grid/plugins/LineAreas.ts +40 -40
  28. package/src/grid/plugins/Lines.ts +40 -40
  29. package/src/grid/plugins/ScalingArea.ts +176 -174
  30. package/src/grid/plugins/ValueAxis.ts +36 -36
  31. package/src/grid/plugins/ValueStackAxis.ts +38 -38
  32. package/src/grid/types.ts +123 -123
  33. package/src/index.ts +9 -9
  34. package/src/multiGrid/defaults.ts +158 -158
  35. package/src/multiGrid/index.ts +13 -13
  36. package/src/multiGrid/multiGridObservables.ts +49 -49
  37. package/src/multiGrid/plugins/MultiBarStack.ts +78 -78
  38. package/src/multiGrid/plugins/MultiBars.ts +77 -77
  39. package/src/multiGrid/plugins/MultiBarsTriangle.ts +77 -77
  40. package/src/multiGrid/plugins/MultiDots.ts +65 -65
  41. package/src/multiGrid/plugins/MultiGridLegend.ts +89 -89
  42. package/src/multiGrid/plugins/MultiGroupAxis.ts +70 -70
  43. package/src/multiGrid/plugins/MultiLineAreas.ts +77 -77
  44. package/src/multiGrid/plugins/MultiLines.ts +77 -77
  45. package/src/multiGrid/plugins/MultiValueAxis.ts +69 -69
  46. package/src/multiGrid/plugins/MultiValueStackAxis.ts +69 -69
  47. package/src/multiGrid/plugins/OverlappingValueAxes.ts +170 -170
  48. package/src/multiGrid/plugins/OverlappingValueStackAxes.ts +169 -169
  49. package/src/multiGrid/types.ts +72 -72
  50. package/src/noneData/defaults.ts +102 -102
  51. package/src/noneData/index.ts +3 -3
  52. package/src/noneData/plugins/Container.ts +10 -10
  53. package/src/noneData/plugins/Tooltip.ts +327 -327
  54. package/src/noneData/types.ts +26 -26
  55. package/src/series/defaults.ts +149 -149
  56. package/src/series/index.ts +9 -9
  57. package/src/series/plugins/Bubbles.ts +545 -545
  58. package/src/series/plugins/Pie.ts +584 -584
  59. package/src/series/plugins/PieEventTexts.ts +262 -262
  60. package/src/series/plugins/PieLabels.ts +604 -598
  61. package/src/series/plugins/Rose.ts +481 -481
  62. package/src/series/plugins/RoseLabels.ts +571 -565
  63. package/src/series/plugins/SeriesLegend.ts +59 -59
  64. package/src/series/seriesObservables.ts +145 -145
  65. package/src/series/seriesUtils.ts +51 -51
  66. package/src/series/types.ts +87 -87
  67. package/src/tree/defaults.ts +23 -23
  68. package/src/tree/index.ts +3 -3
  69. package/src/tree/plugins/TreeLegend.ts +59 -59
  70. package/src/tree/plugins/TreeMap.ts +305 -305
  71. package/src/tree/types.ts +23 -23
  72. package/src/utils/commonUtils.ts +21 -21
  73. package/src/utils/d3Graphics.ts +174 -124
  74. package/src/utils/d3Utils.ts +73 -73
  75. package/src/utils/observables.ts +14 -14
  76. package/src/utils/orbchartsUtils.ts +100 -100
  77. package/tsconfig.base.json +13 -13
  78. package/tsconfig.json +2 -2
  79. package/vite.config.js +22 -22
@@ -1,22 +1,22 @@
1
- // 取得文字寬度
2
- export function measureTextWidth (text: string, size: number = 10) {
3
- const context = document.createElement("canvas").getContext("2d")
4
- let width = context?.measureText(text)?.width ?? 0
5
- return width * size / 10 // 以10為基準
6
- }
7
-
8
- // 取得最小及最大值 - 數字陣列
9
- export function getMinAndMax (data: number[]): [number, number] {
10
- const defaultMinAndMax: [number, number] = [0, 0] // default
11
- if (!data.length) {
12
- return defaultMinAndMax
13
- }
14
- const minAndMax: [number, number] = data.reduce((prev, current) => {
15
- // [min, max]
16
- return [
17
- current < prev[0] ? current : prev[0],
18
- current > prev[1] ? current : prev[1]
19
- ]
20
- }, [data[0], data[0]])
21
- return minAndMax
1
+ // 取得文字寬度
2
+ export function measureTextWidth (text: string, size: number = 10) {
3
+ const context = document.createElement("canvas").getContext("2d")
4
+ let width = context?.measureText(text)?.width ?? 0
5
+ return width * size / 10 // 以10為基準
6
+ }
7
+
8
+ // 取得最小及最大值 - 數字陣列
9
+ export function getMinAndMax (data: number[]): [number, number] {
10
+ const defaultMinAndMax: [number, number] = [0, 0] // default
11
+ if (!data.length) {
12
+ return defaultMinAndMax
13
+ }
14
+ const minAndMax: [number, number] = data.reduce((prev, current) => {
15
+ // [min, max]
16
+ return [
17
+ current < prev[0] ? current : prev[0],
18
+ current > prev[1] ? current : prev[1]
19
+ ]
20
+ }, [data[0], data[0]])
21
+ return minAndMax
22
22
  }
@@ -1,125 +1,175 @@
1
-
2
- type RenderCircleTextParams = {
3
- text: string,
4
- radius: number,
5
- lineHeight: number,
6
- isBreakAll: boolean,
7
- limit?:number
8
- }
9
-
10
- type Line = { width: number; text: string }
11
-
12
- export function renderCircleText (selection: d3.Selection<any, any, any, any>, {
13
- text,
14
- radius,
15
- lineHeight,
16
- isBreakAll = false,
17
- limit = 0
18
- }: RenderCircleTextParams): d3.Selection<SVGTSpanElement, Line, SVGTextElement, any> | undefined {
19
- if (selection == null || text == null) {
20
- console.error("selection or text is not defined")
21
- return
22
- }
23
- if (radius == null) {
24
- const getBox = selection.node().getBBox()
25
- radius = getBox.width / 2
26
- }
27
-
28
- function getWords (text: string) {
29
- let words
30
- if (isBreakAll) {
31
- words = text.split('')
32
- } else {
33
- words = text.split(/\s+/g) // To hyphenate: /\s+|(?<=-)/
34
- }
35
- if (!words[words.length - 1]) words.pop()
36
- if (!words[0]) words.shift()
37
- return words
38
- }
39
-
40
- // 省略文章字數
41
- function ellipisText (text:string, limit:number) {
42
- if (text && limit) {
43
- if (text.length > limit) {
44
- text = text.substring(0, limit) + "..."; // 超過字數以"..."取代
45
- }
46
- }
47
- return text;
48
- }
49
-
50
- function measureWidth (text: string) {
51
- const context = document.createElement("canvas").getContext("2d")
52
- // return text => context.measureText(text).width
53
- return context?.measureText(text)?.width ?? 0
54
- }
55
-
56
- function getTargetWidth (text: string) {
57
- const m = measureWidth(text.trim())
58
- const result = Math.sqrt(m * lineHeight)
59
- return result
60
- // return(
61
- // Math.sqrt(measureWidth(text.trim()) * lineHeight)
62
- // )
63
- }
64
-
65
- function getLines (words: string[], targetWidth: number) {
66
- let line: Line = { width: 0, text: '' }
67
- let lineWidth0 = Infinity
68
- const lines: Array<Line> = []
69
- let space = " "
70
- if (isBreakAll) {
71
- space = ""
72
- }
73
- for (let i = 0, n = words.length; i < n; ++i) {
74
- const lineText1 = (line.text ? line.text + space : '') + words[i]
75
- const lineWidth1 = measureWidth(lineText1)
76
- if ((lineWidth0 + lineWidth1) / 2 < targetWidth) {
77
- line.width = lineWidth0 = lineWidth1
78
- line.text = lineText1
79
- } else {
80
- lineWidth0 = measureWidth(words[i])
81
- line = {width: lineWidth0, text: words[i]}
82
- lines.push(line)
83
- }
84
- }
85
- return lines
86
- }
87
-
88
- function getTextRadius (lines: Array<Line>) {
89
- let radius = 0
90
- for (let i = 0, n = lines.length; i < n; ++i) {
91
- const dy: number = (Math.abs(i - n / 2 + 0.5) + 0.5) * lineHeight
92
- const dx: number = lines[i].width / 2
93
- radius = Math.max(radius, Math.sqrt(dx ** 2 + dy ** 2))
94
- }
95
- return radius
96
- }
97
-
98
- function draw (selection: d3.Selection<any, any, any, any>, text: string) {
99
- if(limit > 0) text = ellipisText(text,limit)
100
- const words = getWords(text)
101
- const targetWidth = getTargetWidth(text)
102
- const lines = getLines(words, targetWidth)
103
- const textRadius = getTextRadius(lines)
104
-
105
- let t = selection.select<SVGTextElement>("text")
106
- if (!t.size()) {
107
- t = selection.append("text")
108
- }
109
- t.attr("transform", `translate(${0},${0}) scale(${radius / textRadius})`)
110
- const tspanUpdate = t.selectAll<SVGTSpanElement, Line>("tspan")
111
- .data(lines)
112
- const tspanEnter = tspanUpdate.enter()
113
- .append<SVGTSpanElement>("tspan")
114
- .attr("x", 0)
115
- .merge(tspanUpdate as d3.Selection<SVGTSpanElement, Line, SVGTextElement, undefined>)
116
- .attr("y", (d: Line, i: number) => (i - lines.length / 2 + 0.8) * lineHeight)
117
- .text((d: Line) => d.text)
118
- tspanUpdate.exit().remove()
119
-
120
- // return selection.node()
121
- return tspanUpdate.merge(tspanEnter)
122
- }
123
-
124
- return draw(selection, text)
1
+
2
+ type RenderCircleTextParams = {
3
+ text: string,
4
+ radius: number,
5
+ lineHeight: number,
6
+ isBreakAll: boolean,
7
+ limit?:number
8
+ }
9
+
10
+ type Line = { width: number; text: string }
11
+
12
+ export function renderCircleText (selection: d3.Selection<any, any, any, any>, {
13
+ text,
14
+ radius,
15
+ lineHeight,
16
+ isBreakAll = false,
17
+ limit = 0
18
+ }: RenderCircleTextParams): d3.Selection<SVGTSpanElement, Line, SVGTextElement, any> | undefined {
19
+ if (selection == null || text == null) {
20
+ console.error("selection or text is not defined")
21
+ return
22
+ }
23
+ if (radius == null) {
24
+ const getBox = selection.node().getBBox()
25
+ radius = getBox.width / 2
26
+ }
27
+
28
+ function getWords (text: string) {
29
+ let words
30
+ if (isBreakAll) {
31
+ words = text.split('')
32
+ } else {
33
+ words = text.split(/\s+/g) // To hyphenate: /\s+|(?<=-)/
34
+ }
35
+ if (!words[words.length - 1]) words.pop()
36
+ if (!words[0]) words.shift()
37
+ return words
38
+ }
39
+
40
+ // 省略文章字數
41
+ function ellipisText (text:string, limit:number) {
42
+ if (text && limit) {
43
+ if (text.length > limit) {
44
+ text = text.substring(0, limit) + "..."; // 超過字數以"..."取代
45
+ }
46
+ }
47
+ return text;
48
+ }
49
+
50
+ function measureWidth (text: string) {
51
+ const context = document.createElement("canvas").getContext("2d")
52
+ // return text => context.measureText(text).width
53
+ return context?.measureText(text)?.width ?? 0
54
+ }
55
+
56
+ function getTargetWidth (text: string) {
57
+ const m = measureWidth(text.trim())
58
+ const result = Math.sqrt(m * lineHeight)
59
+ return result
60
+ // return(
61
+ // Math.sqrt(measureWidth(text.trim()) * lineHeight)
62
+ // )
63
+ }
64
+
65
+ function getLines (words: string[], targetWidth: number) {
66
+ let line: Line = { width: 0, text: '' }
67
+ let lineWidth0 = Infinity
68
+ const lines: Array<Line> = []
69
+ let space = " "
70
+ if (isBreakAll) {
71
+ space = ""
72
+ }
73
+ for (let i = 0, n = words.length; i < n; ++i) {
74
+ const lineText1 = (line.text ? line.text + space : '') + words[i]
75
+ const lineWidth1 = measureWidth(lineText1)
76
+ if ((lineWidth0 + lineWidth1) / 2 < targetWidth) {
77
+ line.width = lineWidth0 = lineWidth1
78
+ line.text = lineText1
79
+ } else {
80
+ lineWidth0 = measureWidth(words[i])
81
+ line = {width: lineWidth0, text: words[i]}
82
+ lines.push(line)
83
+ }
84
+ }
85
+ return lines
86
+ }
87
+
88
+ function getTextRadius (lines: Array<Line>) {
89
+ let radius = 0
90
+ for (let i = 0, n = lines.length; i < n; ++i) {
91
+ const dy: number = (Math.abs(i - n / 2 + 0.5) + 0.5) * lineHeight
92
+ const dx: number = lines[i].width / 2
93
+ radius = Math.max(radius, Math.sqrt(dx ** 2 + dy ** 2))
94
+ }
95
+ return radius
96
+ }
97
+
98
+ function draw (selection: d3.Selection<any, any, any, any>, text: string) {
99
+ if(limit > 0) text = ellipisText(text,limit)
100
+ const words = getWords(text)
101
+ const targetWidth = getTargetWidth(text)
102
+ const lines = getLines(words, targetWidth)
103
+ const textRadius = getTextRadius(lines)
104
+
105
+ let t = selection.select<SVGTextElement>("text")
106
+ if (!t.size()) {
107
+ t = selection.append("text")
108
+ }
109
+ t.attr("transform", `translate(${0},${0}) scale(${radius / textRadius})`)
110
+ const tspanUpdate = t.selectAll<SVGTSpanElement, Line>("tspan")
111
+ .data(lines)
112
+ const tspanEnter = tspanUpdate.enter()
113
+ .append<SVGTSpanElement>("tspan")
114
+ .attr("x", 0)
115
+ .merge(tspanUpdate as d3.Selection<SVGTSpanElement, Line, SVGTextElement, undefined>)
116
+ .attr("y", (d: Line, i: number) => (i - lines.length / 2 + 0.8) * lineHeight)
117
+ .text((d: Line) => d.text)
118
+ tspanUpdate.exit().remove()
119
+
120
+ // return selection.node()
121
+ return tspanUpdate.merge(tspanEnter)
122
+ }
123
+
124
+ return draw(selection, text)
125
+ }
126
+
127
+ // 圖軸上的多行tspan
128
+ export function renderTspansOnAxis (textSelection: d3.Selection<d3.BaseType, unknown, null, undefined>, {
129
+ textArr,
130
+ textSizePx,
131
+ groupAxisPosition
132
+ }: {
133
+ textArr: string[]
134
+ textSizePx: number
135
+ groupAxisPosition: 'top' | 'right' | 'bottom' | 'left'
136
+ }) {
137
+ // -- 將原本單行文字改為多行文字 --
138
+ textSelection.text(null) // 先清空原本的 text
139
+
140
+ const textX = Number(textSelection.attr('x'))
141
+ let textY = Number(textSelection.attr('y'))
142
+ if (groupAxisPosition === 'top') {
143
+ // 當文字在上方時,要往上偏移第一行的高度
144
+ textY -= (textArr.length - 1) * textSizePx
145
+ }
146
+
147
+ textSelection
148
+ .selectAll('tspan')
149
+ .data(textArr)
150
+ .join('tspan')
151
+ .attr('x', textX)
152
+ .attr('y', (_d, _i) => textY + _i * textSizePx)
153
+ .text(d => d)
154
+ }
155
+
156
+ // 四象限上的多行tspan
157
+ export function renderTspansOnQuadrant (textSelection: d3.Selection<d3.BaseType, unknown, null, undefined>, {
158
+ textArr,
159
+ textSizePx,
160
+ quadrant
161
+ }: {
162
+ textArr: string[]
163
+ textSizePx: number
164
+ quadrant: number
165
+ }) {
166
+ textSelection
167
+ .selectAll('tspan')
168
+ .data(textArr)
169
+ .join('tspan')
170
+ .attr('x', 0)
171
+ .attr('y', (_d, _i) => quadrant == 1 || quadrant == 2
172
+ ? - (textArr.length - 1 - _i) * textSizePx
173
+ : _i * textSizePx)
174
+ .text(d => d)
125
175
  }
@@ -1,73 +1,73 @@
1
- import * as d3 from 'd3'
2
-
3
- export function getSvgGElementSize (selection: d3.Selection<SVGGElement, any, any, any>): DOMRect {
4
- try {
5
- return selection.node()!.getBBox()
6
- } catch (e: any) {
7
- throw new Error(e)
8
- }
9
- }
10
-
11
- // 使用字串加入svg
12
- export function appendSvg (selection: d3.Selection<any, any, any, any>, svgString: string): void {
13
- function parseSvg (svgString: string) {
14
- const div = document.createElementNS('http://www.w3.org/1999/xhtml', 'div');
15
- div.innerHTML= '<svg xmlns="http://www.w3.org/2000/svg">'+ svgString +'</svg>';
16
- const frag = document.createDocumentFragment()
17
- while (frag && div?.firstChild?.firstChild)
18
- frag.appendChild(div.firstChild.firstChild);
19
- return frag;
20
- }
21
- // 刪除現有子節點
22
- const node = selection.node()
23
- while(node.hasChildNodes())
24
- {
25
- node.removeChild(node.firstChild);
26
- }
27
- // 加入dom
28
- selection.node().appendChild(parseSvg(svgString))
29
- }
30
-
31
- export function getD3TransitionEase (easeName: string) {
32
- if (easeName.substring(0, 4) !== 'ease') {
33
- return d3.easeCubic
34
- }
35
- return (d3 as any)[easeName] ?? d3.easeCubic
36
- }
37
-
38
- export function makeD3Arc ({ axisWidth, innerRadius, outerRadius, padAngle, cornerRadius }: {
39
- axisWidth: number
40
- innerRadius: number
41
- outerRadius: number
42
- padAngle: number
43
- cornerRadius: number
44
- }): d3.Arc<any, d3.DefaultArcObject> {
45
- const arcScale = d3.scaleLinear()
46
- .domain([0, 1])
47
- .range([0, axisWidth / 2])
48
-
49
- const _outerRadius = arcScale(outerRadius)!
50
-
51
- return d3.arc()
52
- .innerRadius(arcScale(innerRadius)!)
53
- .outerRadius(_outerRadius)
54
- .padAngle(padAngle)
55
- .padRadius(_outerRadius)
56
- .cornerRadius(cornerRadius)
57
- }
58
-
59
- export const parseTickFormatValue = (value: any, tickFormat: string | ((text: d3.NumberValue) => string)) => {
60
- if (tickFormat! instanceof Function == true) {
61
- return (tickFormat as ((text: d3.NumberValue) => string))(value)
62
- }
63
- return d3.format(tickFormat as string)!(value)
64
- }
65
-
66
- export const parseDateTickFormatValue = (value: any, tickFormat: string | ((text: d3.NumberValue) => string)) => {
67
- if (tickFormat! instanceof Function == true) {
68
- return (tickFormat as ((text: d3.NumberValue) => string))(value)
69
- }
70
- return d3.timeFormat(tickFormat as string)!(value)
71
- }
72
-
73
-
1
+ import * as d3 from 'd3'
2
+
3
+ export function getSvgGElementSize (selection: d3.Selection<SVGGElement, any, any, any>): DOMRect {
4
+ try {
5
+ return selection.node()!.getBBox()
6
+ } catch (e: any) {
7
+ throw new Error(e)
8
+ }
9
+ }
10
+
11
+ // 使用字串加入svg
12
+ export function appendSvg (selection: d3.Selection<any, any, any, any>, svgString: string): void {
13
+ function parseSvg (svgString: string) {
14
+ const div = document.createElementNS('http://www.w3.org/1999/xhtml', 'div');
15
+ div.innerHTML= '<svg xmlns="http://www.w3.org/2000/svg">'+ svgString +'</svg>';
16
+ const frag = document.createDocumentFragment()
17
+ while (frag && div?.firstChild?.firstChild)
18
+ frag.appendChild(div.firstChild.firstChild);
19
+ return frag;
20
+ }
21
+ // 刪除現有子節點
22
+ const node = selection.node()
23
+ while(node.hasChildNodes())
24
+ {
25
+ node.removeChild(node.firstChild);
26
+ }
27
+ // 加入dom
28
+ selection.node().appendChild(parseSvg(svgString))
29
+ }
30
+
31
+ export function getD3TransitionEase (easeName: string) {
32
+ if (easeName.substring(0, 4) !== 'ease') {
33
+ return d3.easeCubic
34
+ }
35
+ return (d3 as any)[easeName] ?? d3.easeCubic
36
+ }
37
+
38
+ export function makeD3Arc ({ axisWidth, innerRadius, outerRadius, padAngle, cornerRadius }: {
39
+ axisWidth: number
40
+ innerRadius: number
41
+ outerRadius: number
42
+ padAngle: number
43
+ cornerRadius: number
44
+ }): d3.Arc<any, d3.DefaultArcObject> {
45
+ const arcScale = d3.scaleLinear()
46
+ .domain([0, 1])
47
+ .range([0, axisWidth / 2])
48
+
49
+ const _outerRadius = arcScale(outerRadius)!
50
+
51
+ return d3.arc()
52
+ .innerRadius(arcScale(innerRadius)!)
53
+ .outerRadius(_outerRadius)
54
+ .padAngle(padAngle)
55
+ .padRadius(_outerRadius)
56
+ .cornerRadius(cornerRadius)
57
+ }
58
+
59
+ export const parseTickFormatValue = (value: any, tickFormat: string | ((text: d3.NumberValue) => string)) => {
60
+ if (tickFormat! instanceof Function == true) {
61
+ return (tickFormat as ((text: d3.NumberValue) => string))(value)
62
+ }
63
+ return d3.format(tickFormat as string)!(value)
64
+ }
65
+
66
+ export const parseDateTickFormatValue = (value: any, tickFormat: string | ((text: d3.NumberValue) => string)) => {
67
+ if (tickFormat! instanceof Function == true) {
68
+ return (tickFormat as ((text: d3.NumberValue) => string))(value)
69
+ }
70
+ return d3.timeFormat(tickFormat as string)!(value)
71
+ }
72
+
73
+
@@ -1,14 +1,14 @@
1
- import * as d3 from 'd3'
2
- import { Observable, merge, distinctUntilChanged, fromEvent } from 'rxjs'
3
-
4
- export function d3EventObservable(selection: d3.Selection<any, any, any, any>, event: any) {
5
- // Start with an observable that will never emit
6
- let obs = new Observable(() => {});
7
- selection.each(function () {
8
- // Create observables from each of the elements
9
- const events = fromEvent(this as any, event);
10
- // Merge the observables into one
11
- obs = merge(obs, events);
12
- });
13
- return obs;
14
- }
1
+ import * as d3 from 'd3'
2
+ import { Observable, merge, distinctUntilChanged, fromEvent } from 'rxjs'
3
+
4
+ export function d3EventObservable(selection: d3.Selection<any, any, any, any>, event: any) {
5
+ // Start with an observable that will never emit
6
+ let obs = new Observable(() => {});
7
+ selection.each(function () {
8
+ // Create observables from each of the elements
9
+ const events = fromEvent(this as any, event);
10
+ // Merge the observables into one
11
+ obs = merge(obs, events);
12
+ });
13
+ return obs;
14
+ }