@orbcharts/plugin-basic 4.0.0-pre-alpha.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.
Files changed (208) hide show
  1. package/LICENSE +201 -0
  2. package/dist/orbcharts-plugin-basic.es.js +25335 -0
  3. package/dist/orbcharts-plugin-basic.umd.js +341 -0
  4. package/dist/plugin-basic/src/baseLayers/BaseBars.d.ts +38 -0
  5. package/dist/plugin-basic/src/baseLayers/BaseBarsTriangle.d.ts +37 -0
  6. package/dist/plugin-basic/src/baseLayers/BaseCategoryAxis.d.ts +42 -0
  7. package/dist/plugin-basic/src/baseLayers/BaseDots.d.ts +38 -0
  8. package/dist/plugin-basic/src/baseLayers/BaseLegend.d.ts +31 -0
  9. package/dist/plugin-basic/src/baseLayers/BaseLineAreas.d.ts +36 -0
  10. package/dist/plugin-basic/src/baseLayers/BaseLines.d.ts +36 -0
  11. package/dist/plugin-basic/src/baseLayers/BaseStackedBars.d.ts +41 -0
  12. package/dist/plugin-basic/src/baseLayers/BaseTooltip.d.ts +47 -0
  13. package/dist/plugin-basic/src/baseLayers/BaseValueAxis.d.ts +38 -0
  14. package/dist/plugin-basic/src/baseLayers/BaseXAxis.d.ts +25 -0
  15. package/dist/plugin-basic/src/baseLayers/BaseXZoom.d.ts +22 -0
  16. package/dist/plugin-basic/src/baseLayers/BaseYAxis.d.ts +23 -0
  17. package/dist/plugin-basic/src/baseLayers/types.d.ts +171 -0
  18. package/dist/plugin-basic/src/const/layerIndex.d.ts +10 -0
  19. package/dist/plugin-basic/src/const/sharedPluginParams.d.ts +6 -0
  20. package/dist/plugin-basic/src/index.d.ts +2 -0
  21. package/dist/plugin-basic/src/plugins/CompositionPlot/CompositionPlot.d.ts +22 -0
  22. package/dist/plugin-basic/src/plugins/CompositionPlot/contextObservables.d.ts +40 -0
  23. package/dist/plugin-basic/src/plugins/CompositionPlot/defaults.d.ts +10 -0
  24. package/dist/plugin-basic/src/plugins/CompositionPlot/index.d.ts +3 -0
  25. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Bubbles.d.ts +16 -0
  26. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Indicator.d.ts +0 -0
  27. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Pie.d.ts +16 -0
  28. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/PieEventTexts.d.ts +16 -0
  29. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/PieLabels.d.ts +16 -0
  30. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Rose.d.ts +16 -0
  31. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/RoseLabels.d.ts +16 -0
  32. package/dist/plugin-basic/src/plugins/CompositionPlot/layers/Waffle.d.ts +0 -0
  33. package/dist/plugin-basic/src/plugins/CompositionPlot/types.d.ts +110 -0
  34. package/dist/plugin-basic/src/plugins/CompositionPlot/utils.d.ts +19 -0
  35. package/dist/plugin-basic/src/plugins/HierarchyPlot/HierarchyPlot.d.ts +22 -0
  36. package/dist/plugin-basic/src/plugins/HierarchyPlot/contextObservables.d.ts +16 -0
  37. package/dist/plugin-basic/src/plugins/HierarchyPlot/defaults.d.ts +4 -0
  38. package/dist/plugin-basic/src/plugins/HierarchyPlot/index.d.ts +3 -0
  39. package/dist/plugin-basic/src/plugins/HierarchyPlot/layers/TreeMap.d.ts +16 -0
  40. package/dist/plugin-basic/src/plugins/HierarchyPlot/types.d.ts +29 -0
  41. package/dist/plugin-basic/src/plugins/Legend/Legend.d.ts +22 -0
  42. package/dist/plugin-basic/src/plugins/Legend/contextObservables.d.ts +9 -0
  43. package/dist/plugin-basic/src/plugins/Legend/defaults.d.ts +4 -0
  44. package/dist/plugin-basic/src/plugins/Legend/index.d.ts +3 -0
  45. package/dist/plugin-basic/src/plugins/Legend/layers/Legend.d.ts +16 -0
  46. package/dist/plugin-basic/src/plugins/Legend/types.d.ts +31 -0
  47. package/dist/plugin-basic/src/plugins/Legend/utils.d.ts +19 -0
  48. package/dist/plugin-basic/src/plugins/NetworkPlot/NetworkPlot.d.ts +22 -0
  49. package/dist/plugin-basic/src/plugins/NetworkPlot/contextObservables.d.ts +19 -0
  50. package/dist/plugin-basic/src/plugins/NetworkPlot/defaults.d.ts +5 -0
  51. package/dist/plugin-basic/src/plugins/NetworkPlot/index.d.ts +3 -0
  52. package/dist/plugin-basic/src/plugins/NetworkPlot/layers/ForceDirected.d.ts +16 -0
  53. package/dist/plugin-basic/src/plugins/NetworkPlot/layers/ForceDirectedBubbles.d.ts +16 -0
  54. package/dist/plugin-basic/src/plugins/NetworkPlot/types.d.ts +117 -0
  55. package/dist/plugin-basic/src/plugins/ScatterPlot/ScatterPlot.d.ts +22 -0
  56. package/dist/plugin-basic/src/plugins/ScatterPlot/contextObservables.d.ts +140 -0
  57. package/dist/plugin-basic/src/plugins/ScatterPlot/defaults.d.ts +8 -0
  58. package/dist/plugin-basic/src/plugins/ScatterPlot/index.d.ts +3 -0
  59. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/Scatter.d.ts +16 -0
  60. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/ScatterBubbles.d.ts +16 -0
  61. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/XYAux.d.ts +16 -0
  62. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/XYAxes.d.ts +16 -0
  63. package/dist/plugin-basic/src/plugins/ScatterPlot/layers/XZoom.d.ts +16 -0
  64. package/dist/plugin-basic/src/plugins/ScatterPlot/types.d.ts +146 -0
  65. package/dist/plugin-basic/src/plugins/SeriesPlot/SeriesPlot.d.ts +22 -0
  66. package/dist/plugin-basic/src/plugins/SeriesPlot/contextObservables.d.ts +77 -0
  67. package/dist/plugin-basic/src/plugins/SeriesPlot/defaults.d.ts +15 -0
  68. package/dist/plugin-basic/src/plugins/SeriesPlot/index.d.ts +3 -0
  69. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/Bars.d.ts +16 -0
  70. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/BarsPN.d.ts +16 -0
  71. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/BarsTriangle.d.ts +16 -0
  72. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/CategoryAux.d.ts +16 -0
  73. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/CategoryAxis.d.ts +16 -0
  74. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/CategoryZoom.d.ts +16 -0
  75. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/Dots.d.ts +16 -0
  76. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/LineAreas.d.ts +16 -0
  77. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/Lines.d.ts +16 -0
  78. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/StackedBars.d.ts +16 -0
  79. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/StackedValueAxis.d.ts +16 -0
  80. package/dist/plugin-basic/src/plugins/SeriesPlot/layers/ValueAxis.d.ts +16 -0
  81. package/dist/plugin-basic/src/plugins/SeriesPlot/types.d.ts +140 -0
  82. package/dist/plugin-basic/src/plugins/Tooltip/Tooltip.d.ts +22 -0
  83. package/dist/plugin-basic/src/plugins/Tooltip/contextObservables.d.ts +9 -0
  84. package/dist/plugin-basic/src/plugins/Tooltip/defaults.d.ts +4 -0
  85. package/dist/plugin-basic/src/plugins/Tooltip/index.d.ts +3 -0
  86. package/dist/plugin-basic/src/plugins/Tooltip/layers/Tooltip.d.ts +16 -0
  87. package/dist/plugin-basic/src/plugins/Tooltip/types.d.ts +35 -0
  88. package/dist/plugin-basic/src/plugins/Tooltip/utils.d.ts +19 -0
  89. package/dist/plugin-basic/src/plugins/index.d.ts +7 -0
  90. package/dist/plugin-basic/src/types/BaseLayer.d.ts +3 -0
  91. package/dist/plugin-basic/src/types/Common.d.ts +14 -0
  92. package/dist/plugin-basic/src/types/ComputedData.d.ts +27 -0
  93. package/dist/plugin-basic/src/types/PluginParams.d.ts +66 -0
  94. package/dist/plugin-basic/src/types/index.d.ts +3 -0
  95. package/dist/plugin-basic/src/utils/commonUtils.d.ts +3 -0
  96. package/dist/plugin-basic/src/utils/d3Graphics.d.ts +24 -0
  97. package/dist/plugin-basic/src/utils/d3Scale.d.ts +28 -0
  98. package/dist/plugin-basic/src/utils/d3Utils.d.ts +14 -0
  99. package/dist/plugin-basic/src/utils/graphObservables.d.ts +0 -0
  100. package/dist/plugin-basic/src/utils/gridObservables.d.ts +51 -0
  101. package/dist/plugin-basic/src/utils/multivariateObservables.d.ts +74 -0
  102. package/dist/plugin-basic/src/utils/observables.d.ts +34 -0
  103. package/dist/plugin-basic/src/utils/orbchartsUtils.d.ts +26 -0
  104. package/dist/plugin-basic/src/utils/seriesObservables.d.ts +22 -0
  105. package/dist/plugin-basic/vite.config.d.ts +2 -0
  106. package/dist/src/index.d.ts +1 -0
  107. package/package.json +62 -0
  108. package/src/baseLayers/BaseBars.ts +783 -0
  109. package/src/baseLayers/BaseBarsTriangle.ts +692 -0
  110. package/src/baseLayers/BaseCategoryAxis.ts +708 -0
  111. package/src/baseLayers/BaseDots.ts +495 -0
  112. package/src/baseLayers/BaseLegend.ts +684 -0
  113. package/src/baseLayers/BaseLineAreas.ts +644 -0
  114. package/src/baseLayers/BaseLines.ts +721 -0
  115. package/src/baseLayers/BaseStackedBars.ts +818 -0
  116. package/src/baseLayers/BaseTooltip.ts +435 -0
  117. package/src/baseLayers/BaseValueAxis.ts +612 -0
  118. package/src/baseLayers/BaseXAxis.ts +412 -0
  119. package/src/baseLayers/BaseXZoom.ts +250 -0
  120. package/src/baseLayers/BaseYAxis.ts +371 -0
  121. package/src/baseLayers/types.ts +174 -0
  122. package/src/const/layerIndex.ts +36 -0
  123. package/src/const/sharedPluginParams.ts +29 -0
  124. package/src/index.ts +3 -0
  125. package/src/plugins/CompositionPlot/CompositionPlot.ts +308 -0
  126. package/src/plugins/CompositionPlot/contextObservables.ts +251 -0
  127. package/src/plugins/CompositionPlot/defaults.ts +162 -0
  128. package/src/plugins/CompositionPlot/index.ts +3 -0
  129. package/src/plugins/CompositionPlot/layers/Bubbles.ts +808 -0
  130. package/src/plugins/CompositionPlot/layers/Indicator.ts +0 -0
  131. package/src/plugins/CompositionPlot/layers/Pie.ts +776 -0
  132. package/src/plugins/CompositionPlot/layers/PieEventTexts.ts +326 -0
  133. package/src/plugins/CompositionPlot/layers/PieLabels.ts +651 -0
  134. package/src/plugins/CompositionPlot/layers/Rose.ts +546 -0
  135. package/src/plugins/CompositionPlot/layers/RoseLabels.ts +616 -0
  136. package/src/plugins/CompositionPlot/layers/Waffle.ts +0 -0
  137. package/src/plugins/CompositionPlot/types.ts +129 -0
  138. package/src/plugins/CompositionPlot/utils.ts +53 -0
  139. package/src/plugins/HierarchyPlot/HierarchyPlot.ts +190 -0
  140. package/src/plugins/HierarchyPlot/contextObservables.ts +136 -0
  141. package/src/plugins/HierarchyPlot/defaults.ts +31 -0
  142. package/src/plugins/HierarchyPlot/index.ts +3 -0
  143. package/src/plugins/HierarchyPlot/layers/TreeMap.ts +371 -0
  144. package/src/plugins/HierarchyPlot/types.ts +36 -0
  145. package/src/plugins/Legend/Legend.ts +151 -0
  146. package/src/plugins/Legend/contextObservables.ts +55 -0
  147. package/src/plugins/Legend/defaults.ts +37 -0
  148. package/src/plugins/Legend/index.ts +3 -0
  149. package/src/plugins/Legend/layers/Legend.ts +114 -0
  150. package/src/plugins/Legend/types.ts +45 -0
  151. package/src/plugins/Legend/utils.ts +53 -0
  152. package/src/plugins/NetworkPlot/NetworkPlot.ts +228 -0
  153. package/src/plugins/NetworkPlot/contextObservables.ts +123 -0
  154. package/src/plugins/NetworkPlot/defaults.ts +147 -0
  155. package/src/plugins/NetworkPlot/index.ts +3 -0
  156. package/src/plugins/NetworkPlot/layers/ForceDirected.ts +1048 -0
  157. package/src/plugins/NetworkPlot/layers/ForceDirectedBubbles.ts +1318 -0
  158. package/src/plugins/NetworkPlot/types.ts +146 -0
  159. package/src/plugins/ScatterPlot/ScatterPlot.ts +569 -0
  160. package/src/plugins/ScatterPlot/contextObservables.ts +901 -0
  161. package/src/plugins/ScatterPlot/defaults.ts +212 -0
  162. package/src/plugins/ScatterPlot/index.ts +3 -0
  163. package/src/plugins/ScatterPlot/layers/Scatter.ts +518 -0
  164. package/src/plugins/ScatterPlot/layers/ScatterBubbles.ts +670 -0
  165. package/src/plugins/ScatterPlot/layers/XYAux.ts +686 -0
  166. package/src/plugins/ScatterPlot/layers/XYAxes.ts +205 -0
  167. package/src/plugins/ScatterPlot/layers/XZoom.ts +48 -0
  168. package/src/plugins/ScatterPlot/types.ts +179 -0
  169. package/src/plugins/SeriesPlot/SeriesPlot.ts +494 -0
  170. package/src/plugins/SeriesPlot/contextObservables.ts +726 -0
  171. package/src/plugins/SeriesPlot/defaults.ts +142 -0
  172. package/src/plugins/SeriesPlot/index.ts +3 -0
  173. package/src/plugins/SeriesPlot/layers/Bars.ts +84 -0
  174. package/src/plugins/SeriesPlot/layers/BarsPN.ts +85 -0
  175. package/src/plugins/SeriesPlot/layers/BarsTriangle.ts +89 -0
  176. package/src/plugins/SeriesPlot/layers/CategoryAux.ts +1131 -0
  177. package/src/plugins/SeriesPlot/layers/CategoryAxis.ts +92 -0
  178. package/src/plugins/SeriesPlot/layers/CategoryZoom.ts +233 -0
  179. package/src/plugins/SeriesPlot/layers/Dots.ts +91 -0
  180. package/src/plugins/SeriesPlot/layers/LineAreas.ts +82 -0
  181. package/src/plugins/SeriesPlot/layers/Lines.ts +75 -0
  182. package/src/plugins/SeriesPlot/layers/StackedBars.ts +85 -0
  183. package/src/plugins/SeriesPlot/layers/StackedValueAxis.ts +111 -0
  184. package/src/plugins/SeriesPlot/layers/ValueAxis.ts +111 -0
  185. package/src/plugins/SeriesPlot/types.ts +201 -0
  186. package/src/plugins/Tooltip/Tooltip.ts +159 -0
  187. package/src/plugins/Tooltip/contextObservables.ts +55 -0
  188. package/src/plugins/Tooltip/defaults.ts +458 -0
  189. package/src/plugins/Tooltip/index.ts +3 -0
  190. package/src/plugins/Tooltip/layers/Tooltip.ts +90 -0
  191. package/src/plugins/Tooltip/types.ts +55 -0
  192. package/src/plugins/Tooltip/utils.ts +53 -0
  193. package/src/plugins/index.ts +8 -0
  194. package/src/types/BaseLayer.ts +3 -0
  195. package/src/types/Common.ts +20 -0
  196. package/src/types/ComputedData.ts +55 -0
  197. package/src/types/PluginParams.ts +81 -0
  198. package/src/types/index.ts +3 -0
  199. package/src/utils/commonUtils.ts +31 -0
  200. package/src/utils/d3Graphics.ts +177 -0
  201. package/src/utils/d3Scale.ts +198 -0
  202. package/src/utils/d3Utils.ts +92 -0
  203. package/src/utils/graphObservables.ts +0 -0
  204. package/src/utils/gridObservables.ts +637 -0
  205. package/src/utils/multivariateObservables.ts +790 -0
  206. package/src/utils/observables.ts +357 -0
  207. package/src/utils/orbchartsUtils.ts +335 -0
  208. package/src/utils/seriesObservables.ts +172 -0
@@ -0,0 +1,435 @@
1
+ import * as d3 from 'd3'
2
+ import {
3
+ combineLatest,
4
+ map,
5
+ merge,
6
+ filter,
7
+ switchMap,
8
+ debounceTime,
9
+ takeUntil,
10
+ Subject,
11
+ Observable,
12
+ distinctUntilChanged } from 'rxjs'
13
+ import type { ColorType, EventData, RenderDatumBase, Theme } from '@orbcharts/core'
14
+ import type { BaseLayerFn } from '../types/BaseLayer'
15
+ import type { Layout } from '../types/PluginParams'
16
+ import type { ComputedDatum } from '../types'
17
+ import { fontSizePxObservable } from '../utils/observables'
18
+ import { getSvgGElementSize, appendSvg } from '../utils/d3Utils'
19
+ import { getColor, createClassName } from '../utils/orbchartsUtils'
20
+ import { measureTextWidth, toCurrency } from '../utils/commonUtils'
21
+
22
+ export interface BaseTooltipStyle {
23
+ backgroundColor: string;
24
+ backgroundOpacity: number;
25
+ strokeColor: string;
26
+ offset: [number, number];
27
+ padding: number;
28
+ textColor: string;
29
+ textSize: number | string;
30
+ textSizePx: number;
31
+ // seriesColors: string[];
32
+ }
33
+ export interface BaseTooltipUtils {
34
+ toCurrency: (num: number | null) => string;
35
+ measureTextWidth(text: string, size?: number): number;
36
+ }
37
+ export type BaseTooltipParams = {
38
+ backgroundColorType: ColorType;
39
+ backgroundOpacity: number;
40
+ strokeColorType: ColorType;
41
+ textColorType: ColorType;
42
+ offset: [number, number];
43
+ padding: number;
44
+ renderFn: ((eventData: EventData<any>, context: {
45
+ styles: BaseTooltipStyle;
46
+ utils: BaseTooltipUtils;
47
+ seriesData: ComputedDatum<any>[]
48
+ categoryData: ComputedDatum<any>[]
49
+ }) => string[] | string);
50
+ };
51
+
52
+ interface BaseTooltipContext {
53
+ pluginName: string
54
+ layerName: string
55
+ rootSelection: d3.Selection<any, unknown, any, unknown>
56
+ baseTooltipParams$: Observable<BaseTooltipParams>
57
+ theme$: Observable<Theme>
58
+ layout$: Observable<Layout>
59
+ event$: Observable<EventData<any>>
60
+ SeriesDataMap$: Observable<Map<string, ComputedDatum<any>[]>>
61
+ CategoryDataMap$: Observable<Map<string, ComputedDatum<any>[]>>
62
+ }
63
+
64
+ function textToSvg (_textArr: string[] | string | null | undefined, textStyle: BaseTooltipStyle) {
65
+ const lineHeight = textStyle.textSizePx * 1.5
66
+
67
+ const textArr = _textArr == null
68
+ ? []
69
+ : Array.isArray(_textArr)
70
+ ? _textArr
71
+ : typeof _textArr === 'string'
72
+ ? _textArr.split('\n')
73
+ : [_textArr]
74
+
75
+ const tspan = textArr
76
+ .filter(d => d != '')
77
+ .map((text, i) => {
78
+ const top = i * lineHeight
79
+ return `<tspan x="0" y="${top}">${text}</tspan>`
80
+ })
81
+ .join('')
82
+
83
+ if (tspan) {
84
+ return `<text font-size="${textStyle.textSize}" fill="${textStyle.textColor}" x="0" y="0" style="dominant-baseline:text-before-edge">
85
+ ${tspan}
86
+ </text>`
87
+ } else {
88
+ return ''
89
+ }
90
+ }
91
+
92
+ function renderTooltip ({ rootSelection, pluginName, layerName, gClassName, boxClassName, rootWidth, rootHeight, svgString, tooltipStyle, event }: {
93
+ rootSelection: d3.Selection<any, unknown, any, unknown>
94
+ pluginName: string
95
+ layerName: string
96
+ gClassName: string
97
+ boxClassName: string
98
+ rootWidth: number
99
+ rootHeight: number
100
+ svgString: string
101
+ tooltipStyle: BaseTooltipStyle
102
+ event: MouseEvent
103
+ }) {
104
+ // if (!svgString) {
105
+ // return
106
+ // }
107
+ // const rootSelection = d3.select('svg.bpcharts__root')
108
+ // console.log('tooltip', { rootSelection, pluginName, gClassName, boxClassName, rootWidth, rootHeight, svgString, tooltipStyle, event })
109
+ rootSelection.interrupt('fadeout')
110
+ const radius = 5
111
+
112
+ // data(svg string無值則為空陣列)
113
+ const contentData = svgString ? [svgString] : []
114
+ const styleData = svgString ? [tooltipStyle] : []
115
+ // tooltipG <g>
116
+ const tooltipG = rootSelection
117
+ .selectAll<SVGGElement, string>(`g.${gClassName}`)
118
+ .data(contentData)
119
+ .join(
120
+ enter => {
121
+ return enter
122
+ .append('g')
123
+ .classed(gClassName, true)
124
+ .attr('pointer-events', 'none')
125
+ },
126
+ update => {
127
+ return update
128
+ },
129
+ exit => {
130
+ return exit
131
+ // .transition('fadeout')
132
+ // .duration(0)
133
+ // .delay(500)
134
+ .style('opacity', 0)
135
+ .remove()
136
+ }
137
+ )
138
+ .attr('transform', () => `translate(${event.offsetX}, ${event.offsetY})`)
139
+
140
+ // tooltipBox <g><g>
141
+ const tooltipBox = tooltipG
142
+ .selectAll<SVGGElement, string>(`g.${boxClassName}`)
143
+ .data(styleData)
144
+ .join(
145
+ enter => {
146
+ return enter
147
+ .append('g')
148
+ .classed(createClassName(pluginName, layerName, 'box'), true)
149
+ },
150
+ )
151
+
152
+ // rect <g><g><rect>
153
+ const rect = tooltipBox
154
+ .selectAll<SVGRectElement, string>('rect')
155
+ .data(styleData)
156
+ .join(
157
+ enter => {
158
+ return enter
159
+ .append('rect')
160
+ .attr('rx', radius)
161
+ .attr('ry', radius)
162
+ }
163
+ )
164
+ .attr('fill', d => d.backgroundColor)
165
+ .attr('stroke', d => d.strokeColor)
166
+ .attr('opacity', d => d.backgroundOpacity)
167
+
168
+ // text <g><g><g>
169
+ const contentG = tooltipBox
170
+ .selectAll<SVGGElement, string>('g')
171
+ .data(contentData)
172
+ .join(
173
+ enter => {
174
+ return enter
175
+ .append('g')
176
+ .classed(createClassName(pluginName, layerName, 'content'), true)
177
+ .attr('transform', () => `translate(${tooltipStyle.padding}, ${tooltipStyle.padding})`)
178
+ }
179
+ )
180
+ // 使用字串加入svg
181
+ if (contentData.length) {
182
+ appendSvg(contentG, contentData[0])
183
+ }
184
+ const contentSize = (contentG?.node()) ? getSvgGElementSize(contentG!) : { width: 0, height: 0 }
185
+
186
+ // rect size
187
+ rect
188
+ .attr('width', contentSize.width + tooltipStyle.padding * 2)
189
+ .attr('height', contentSize.height + tooltipStyle.padding * 2)
190
+
191
+ // -- tooltipG --
192
+ // 取得tooltip <g>的尺寸
193
+ const tooltipSize = (tooltipBox?.node()) ? getSvgGElementSize(tooltipBox!) : { width: 0, height: 0 }
194
+ // const minX = 0
195
+ const maxX = rootWidth - tooltipSize.width
196
+ // const minY = 0
197
+ const maxY = rootHeight - tooltipSize.height
198
+
199
+ // -- 相對游標位置的offset --
200
+ const offsetX = (event.offsetX + tooltipStyle.offset[0]) > maxX ? maxX - event.offsetX : tooltipStyle.offset[0]
201
+ const offsetY = (event.offsetY + tooltipStyle.offset[1]) > maxY ? maxY - event.offsetY : tooltipStyle.offset[1]
202
+ tooltipBox.attr('transform', d => `translate(${offsetX}, ${offsetY})`)
203
+ tooltipBox.attr('transform', d => `translate(${offsetX}, ${offsetY})`)
204
+
205
+ // const minX = containerSize.x
206
+ // const maxX = containerSize.x + containerSize.width - tooltipSize.width
207
+ // const minY = containerSize.y
208
+ // const maxY = containerSize.y + containerSize.height - tooltipSize.height
209
+
210
+ // .style('position', 'absolute')
211
+ // .style('z-index', 10000)
212
+ // .style('left', (d) => {
213
+ // const x = d.x + this.baseTooltipParams.offsetX! - containerSize.x
214
+ // if (x < minX) {
215
+ // return `${minX}px`
216
+ // } else if (x > maxX) {
217
+ // return `${maxX}px`
218
+ // }
219
+ // return `${x}px`
220
+ // })
221
+ // .style('top', (d) => {
222
+ // const y = d.y + this.baseTooltipParams.offsetY! - containerSize.y
223
+ // if (y < minY) {
224
+ // return `${minY}px`
225
+ // } else if (y > maxY) {
226
+ // return `${maxY}px`
227
+ // }
228
+ // return `${y}px`
229
+ // })
230
+ // .html((d) => d.contentHtml)
231
+
232
+ }
233
+
234
+ function removeTooltip (gClassName: string) {
235
+ d3.selectAll(`g.${gClassName}`)
236
+ .remove()
237
+ }
238
+
239
+ export const createBaseTooltip: BaseLayerFn<BaseTooltipContext> = ({
240
+ pluginName,
241
+ layerName,
242
+ rootSelection,
243
+ baseTooltipParams$,
244
+ theme$,
245
+ layout$,
246
+ event$,
247
+ SeriesDataMap$,
248
+ CategoryDataMap$
249
+ }) => {
250
+
251
+ const destroy$ = new Subject()
252
+
253
+ const gClassName = createClassName(pluginName, layerName, 'g')
254
+ const boxClassName = createClassName(pluginName, layerName, 'box')
255
+
256
+ // 事件觸發
257
+ const eventMouseover$: Observable<EventData<any>> = event$.pipe(
258
+ takeUntil(destroy$),
259
+ filter(d => d.eventName === 'mouseover' || d.eventName === 'mousemove'),
260
+ // distinctUntilChanged((prev, current) => prev.eventName === current.eventName)
261
+ // map(d => d as EventData<typeof chartType>)
262
+ )
263
+ const eventMouseout$: Observable<EventData<any>> = event$.pipe(
264
+ takeUntil(destroy$),
265
+ filter(d => d.eventName === 'mouseout'),
266
+ // distinctUntilChanged((prev, current) => prev.eventName === current.eventName)
267
+ // map(d => d as EventData<typeof chartType>)
268
+ )
269
+
270
+ const fontSizePx$ = fontSizePxObservable(theme$)
271
+
272
+ const tooltipStyle$: Observable<BaseTooltipStyle> = combineLatest({
273
+ theme: theme$,
274
+ fontSizePx: fontSizePx$,
275
+ baseTooltipParams: baseTooltipParams$,
276
+ }).pipe(
277
+ takeUntil(destroy$),
278
+ debounceTime(0),
279
+ map(data => {
280
+ return {
281
+ backgroundColor: getColor(data.baseTooltipParams.backgroundColorType, data.theme),
282
+ backgroundOpacity: data.baseTooltipParams.backgroundOpacity,
283
+ strokeColor: getColor(data.baseTooltipParams.strokeColorType, data.theme),
284
+ offset: data.baseTooltipParams.offset,
285
+ padding: data.baseTooltipParams.padding,
286
+ textSize: data.theme.fontSize,
287
+ textSizePx: data.fontSizePx,
288
+ textColor: getColor(data.baseTooltipParams.textColorType, data.theme),
289
+ // seriesColors: data.theme.colors[data.theme.colorScheme].data
290
+ }
291
+ })
292
+ )
293
+
294
+ const contentRenderFn$: Observable<((eventData: EventData<any>) => string)> = combineLatest({
295
+ baseTooltipParams: baseTooltipParams$,
296
+ tooltipStyle: tooltipStyle$,
297
+ SeriesDataMap: SeriesDataMap$,
298
+ CategoryDataMap: CategoryDataMap$
299
+ }).pipe(
300
+ takeUntil(destroy$),
301
+ debounceTime(0),
302
+ map(data => {
303
+ // if (data.baseTooltipParams.svgRenderFn) {
304
+ // return data.baseTooltipParams.svgRenderFn
305
+ // }
306
+ // // 將textRenderFn回傳的資料使用<text>包裝起來
307
+ // return (eventData: EventData<any>) => {
308
+ // const textArr: string | string[] | null = data.baseTooltipParams.textRenderFn
309
+ // ? data.baseTooltipParams.textRenderFn(eventData as any)
310
+ // : null
311
+ // return textToSvg(textArr, data.tooltipStyle)
312
+ // }
313
+ return (eventData: EventData) => {
314
+ const renderText: string | string[] = data.baseTooltipParams.renderFn(
315
+ // mouseover事件的資料
316
+ eventData,
317
+ // context
318
+ {
319
+ utils: {
320
+ toCurrency,
321
+ measureTextWidth
322
+ },
323
+ styles: data.tooltipStyle,
324
+ seriesData: eventData.target.series ? data.SeriesDataMap.get(eventData.target.series) : [],
325
+ categoryData: (eventData.target as RenderDatumBase<'grid'>).category ? data.CategoryDataMap.get((eventData.target as RenderDatumBase<'grid'>).category) : []
326
+ }
327
+ )
328
+ // string
329
+ if (typeof renderText === 'string') {
330
+ const trimText = renderText.trim()
331
+ const isSvgText = trimText.slice(0, 1) === '<' && trimText.slice(trimText.length - 1, trimText.length) === '>'
332
+
333
+ if (isSvgText) {
334
+ return renderText // svg字串
335
+ } else {
336
+ const textArr = renderText.split('\n')
337
+ return textToSvg(textArr, data.tooltipStyle) // 多行文字轉svg字串
338
+ }
339
+ }
340
+ // string[]
341
+ else if (Array.isArray(renderText)) {
342
+ return textToSvg(renderText, data.tooltipStyle) // 多行文字轉svg字串
343
+ }
344
+ return ''
345
+ }
346
+ })
347
+ )
348
+
349
+ const mouseoverTooltipSvg$ = combineLatest({
350
+ event: eventMouseover$,
351
+ contentRenderFn: contentRenderFn$
352
+ }).pipe(
353
+ takeUntil(destroy$),
354
+ debounceTime(0),
355
+ map(d => d.contentRenderFn(d.event))
356
+ )
357
+
358
+ const mouseoutTooltipSvg$ = eventMouseout$.pipe(
359
+ takeUntil(destroy$),
360
+ map(d => '')
361
+ )
362
+
363
+ const svgString$ = merge(mouseoverTooltipSvg$, mouseoutTooltipSvg$)
364
+ .pipe(
365
+ takeUntil(destroy$),
366
+ distinctUntilChanged((a, b) => a === b)
367
+ )
368
+
369
+ const eventTooltip$ = merge(eventMouseover$, eventMouseout$)
370
+ .pipe(
371
+ takeUntil(destroy$),
372
+ // filter(d => {
373
+ // return (d.eventName === 'mouseover' || d.eventName === 'mousemove' || d.eventName === 'mouseout')
374
+ // && d.event != undefined
375
+ // }),
376
+ // map(d => d.event!),
377
+ )
378
+
379
+ combineLatest({
380
+ svgString: svgString$,
381
+ layout: layout$,
382
+ tooltipStyle: tooltipStyle$,
383
+ }).pipe(
384
+ takeUntil(destroy$),
385
+ debounceTime(0),
386
+ switchMap(data => {
387
+ // 只當有event時才產生資料流
388
+ return eventTooltip$.pipe(
389
+ map(eventTooltip => {
390
+ return {
391
+ svgString: data.svgString,
392
+ layout: data.layout,
393
+ tooltipStyle: data.tooltipStyle,
394
+ eventTooltip: eventTooltip
395
+ }
396
+ })
397
+ )
398
+ })
399
+ ).subscribe(data => {
400
+ if (data.eventTooltip.eventName === 'mouseout') {
401
+ removeTooltip(gClassName)
402
+ return
403
+ }
404
+ renderTooltip({
405
+ rootSelection,
406
+ pluginName,
407
+ layerName,
408
+ gClassName,
409
+ boxClassName,
410
+ rootWidth: data.layout.rootWidth,
411
+ rootHeight: data.layout.rootHeight,
412
+ svgString: data.svgString,
413
+ tooltipStyle: data.tooltipStyle,
414
+ event: data.eventTooltip.event as MouseEvent
415
+ })
416
+ })
417
+
418
+
419
+
420
+ // const chartType$ = eventMouseover$.pipe(
421
+ // takeUntil(destroy$),
422
+ // map(d => d.type),
423
+ // distinctUntilChanged()
424
+ // )
425
+
426
+
427
+ // eventMouseover$.subscribe(event => {
428
+
429
+ // })
430
+
431
+
432
+ return () => {
433
+ destroy$.next(undefined)
434
+ }
435
+ }