@orbcharts/plugins-basic 3.0.0-alpha.78 → 3.0.0-beta.1
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.
- package/dist/orbcharts-plugins-basic.es.js +7969 -7502
- package/dist/orbcharts-plugins-basic.umd.js +124 -19
- package/dist/src/base/BaseTooltip.d.ts +14 -0
- package/dist/src/grid/defaults.d.ts +3 -2
- package/dist/src/grid/index.d.ts +2 -1
- package/dist/src/grid/plugins/GridTooltip.d.ts +1 -0
- package/dist/src/grid/plugins/GridZoom.d.ts +1 -0
- package/dist/src/multiGrid/defaults.d.ts +2 -1
- package/dist/src/multiGrid/index.d.ts +1 -0
- package/dist/src/multiGrid/plugins/MultiGridTooltip.d.ts +1 -0
- package/dist/src/noneData/index.d.ts +0 -1
- package/dist/src/series/defaults.d.ts +2 -1
- package/dist/src/series/index.d.ts +1 -0
- package/dist/src/series/plugins/SeriesTooltip.d.ts +1 -0
- package/dist/src/tree/defaults.d.ts +2 -1
- package/dist/src/tree/index.d.ts +1 -0
- package/dist/src/tree/plugins/TreeTooltip.d.ts +1 -0
- package/package.json +4 -4
- package/src/base/BaseDots.ts +4 -4
- package/src/base/BaseLegend.ts +75 -39
- package/src/base/BaseTooltip.ts +383 -0
- package/src/grid/defaults.ts +122 -5
- package/src/grid/index.ts +2 -1
- package/src/grid/plugins/GridLegend.ts +11 -0
- package/src/grid/plugins/GridTooltip.ts +66 -0
- package/src/grid/plugins/{ScalingArea.ts → GridZoom.ts} +5 -5
- package/src/multiGrid/defaults.ts +69 -3
- package/src/multiGrid/index.ts +1 -0
- package/src/multiGrid/plugins/MultiGridLegend.ts +11 -0
- package/src/multiGrid/plugins/MultiGridTooltip.ts +66 -0
- package/src/noneData/index.ts +1 -1
- package/src/series/defaults.ts +61 -3
- package/src/series/index.ts +1 -0
- package/src/series/plugins/SeriesLegend.ts +11 -0
- package/src/series/plugins/SeriesTooltip.ts +66 -0
- package/src/tree/defaults.ts +59 -3
- package/src/tree/index.ts +2 -1
- package/src/tree/plugins/TreeLegend.ts +11 -0
- package/src/tree/plugins/TreeTooltip.ts +66 -0
- package/dist/src/grid/plugins/ScalingArea.d.ts +0 -1
@@ -0,0 +1,383 @@
|
|
1
|
+
import * as d3 from 'd3'
|
2
|
+
import {
|
3
|
+
combineLatest,
|
4
|
+
map,
|
5
|
+
merge,
|
6
|
+
filter,
|
7
|
+
switchMap,
|
8
|
+
first,
|
9
|
+
iif,
|
10
|
+
takeUntil,
|
11
|
+
Subject,
|
12
|
+
Observable,
|
13
|
+
distinctUntilChanged } from 'rxjs'
|
14
|
+
import type {
|
15
|
+
EventTypeMap,
|
16
|
+
PluginConstructor,
|
17
|
+
ChartType,
|
18
|
+
ChartParams,
|
19
|
+
Layout
|
20
|
+
} from '../../lib/core-types'
|
21
|
+
import type { BaseTooltipParams, BaseTooltipStyle } from '../../lib/plugins-basic-types'
|
22
|
+
import type { BasePluginFn } from './types'
|
23
|
+
import { defineNoneDataPlugin, textSizePxObservable } from '../../lib/core'
|
24
|
+
import { getSvgGElementSize, appendSvg } from '../utils/d3Utils'
|
25
|
+
import { getColor, getClassName } from '../utils/orbchartsUtils'
|
26
|
+
|
27
|
+
|
28
|
+
interface BaseTooltipContext {
|
29
|
+
rootSelection: d3.Selection<any, unknown, any, unknown>
|
30
|
+
fullParams$: Observable<BaseTooltipParams>
|
31
|
+
fullChartParams$: Observable<ChartParams>
|
32
|
+
layout$: Observable<Layout>
|
33
|
+
event$: Subject<EventTypeMap<any>>
|
34
|
+
}
|
35
|
+
|
36
|
+
// export interface BaseTooltipStyle {
|
37
|
+
// backgroundColor: string
|
38
|
+
// backgroundOpacity: number
|
39
|
+
// strokeColor: string
|
40
|
+
// offset: [number, number]
|
41
|
+
// padding: number
|
42
|
+
// textColor: string
|
43
|
+
// textSize: number | string // chartParams上的設定
|
44
|
+
// textSizePx: number
|
45
|
+
// }
|
46
|
+
|
47
|
+
function textToSvg (_textArr: string[] | string | null | undefined, textStyle: BaseTooltipStyle) {
|
48
|
+
const lineHeight = textStyle.textSizePx * 1.5
|
49
|
+
|
50
|
+
const textArr = _textArr == null
|
51
|
+
? []
|
52
|
+
: Array.isArray(_textArr)
|
53
|
+
? _textArr
|
54
|
+
: typeof _textArr === 'string'
|
55
|
+
? _textArr.split('\n')
|
56
|
+
: [_textArr]
|
57
|
+
|
58
|
+
const tspan = textArr
|
59
|
+
.filter(d => d != '')
|
60
|
+
.map((text, i) => {
|
61
|
+
const top = i * lineHeight
|
62
|
+
return `<tspan x="0" y="${top}">${text}</tspan>`
|
63
|
+
})
|
64
|
+
.join('')
|
65
|
+
|
66
|
+
if (tspan) {
|
67
|
+
return `<text font-size="${textStyle.textSize}" fill="${textStyle.textColor}" x="0" y="0" style="dominant-baseline:text-before-edge">
|
68
|
+
${tspan}
|
69
|
+
</text>`
|
70
|
+
} else {
|
71
|
+
return ''
|
72
|
+
}
|
73
|
+
}
|
74
|
+
|
75
|
+
function renderTooltip ({ rootSelection, pluginName, gClassName, boxClassName, rootWidth, rootHeight, svgString, tooltipStyle, event }: {
|
76
|
+
rootSelection: d3.Selection<any, unknown, any, unknown>
|
77
|
+
pluginName: string
|
78
|
+
gClassName: string
|
79
|
+
boxClassName: string
|
80
|
+
rootWidth: number
|
81
|
+
rootHeight: number
|
82
|
+
svgString: string
|
83
|
+
tooltipStyle: BaseTooltipStyle
|
84
|
+
event: MouseEvent
|
85
|
+
}) {
|
86
|
+
// if (!svgString) {
|
87
|
+
// return
|
88
|
+
// }
|
89
|
+
// const rootSelection = d3.select('svg.bpcharts__root')
|
90
|
+
// console.log('tooltip', { selection, rootWidth, rootHeight, svgString, event })
|
91
|
+
rootSelection.interrupt('fadeout')
|
92
|
+
const radius = 5
|
93
|
+
|
94
|
+
// data(svg string無值則為空陣列)
|
95
|
+
const contentData = svgString ? [svgString] : []
|
96
|
+
const styleData = svgString ? [tooltipStyle] : []
|
97
|
+
// tooltipG <g>
|
98
|
+
const tooltipG = rootSelection
|
99
|
+
.selectAll<SVGGElement, string>(`g.${gClassName}`)
|
100
|
+
.data(contentData)
|
101
|
+
.join(
|
102
|
+
enter => {
|
103
|
+
return enter
|
104
|
+
.append('g')
|
105
|
+
.classed(gClassName, true)
|
106
|
+
.attr('pointer-events', 'none')
|
107
|
+
},
|
108
|
+
update => {
|
109
|
+
return update
|
110
|
+
},
|
111
|
+
exit => {
|
112
|
+
return exit
|
113
|
+
// .transition('fadeout')
|
114
|
+
// .duration(0)
|
115
|
+
// .delay(500)
|
116
|
+
.style('opacity', 0)
|
117
|
+
.remove()
|
118
|
+
}
|
119
|
+
)
|
120
|
+
.attr('transform', () => `translate(${event.offsetX}, ${event.offsetY})`)
|
121
|
+
|
122
|
+
// tooltipBox <g><g>
|
123
|
+
const tooltipBox = tooltipG
|
124
|
+
.selectAll<SVGGElement, string>(`g.${boxClassName}`)
|
125
|
+
.data(styleData)
|
126
|
+
.join(
|
127
|
+
enter => {
|
128
|
+
return enter
|
129
|
+
.append('g')
|
130
|
+
.classed(getClassName(pluginName, 'box'), true)
|
131
|
+
},
|
132
|
+
)
|
133
|
+
|
134
|
+
// rect <g><g><rect>
|
135
|
+
const rect = tooltipBox
|
136
|
+
.selectAll<SVGRectElement, string>('rect')
|
137
|
+
.data(styleData)
|
138
|
+
.join(
|
139
|
+
enter => {
|
140
|
+
return enter
|
141
|
+
.append('rect')
|
142
|
+
.attr('rx', radius)
|
143
|
+
.attr('ry', radius)
|
144
|
+
}
|
145
|
+
)
|
146
|
+
.attr('fill', d => d.backgroundColor)
|
147
|
+
.attr('stroke', d => d.strokeColor)
|
148
|
+
.attr('opacity', d => d.backgroundOpacity)
|
149
|
+
|
150
|
+
// text <g><g><g>
|
151
|
+
const contentG = tooltipBox
|
152
|
+
.selectAll<SVGGElement, string>('g')
|
153
|
+
.data(contentData)
|
154
|
+
.join(
|
155
|
+
enter => {
|
156
|
+
return enter
|
157
|
+
.append('g')
|
158
|
+
.classed(getClassName(pluginName, 'content'), true)
|
159
|
+
.attr('transform', () => `translate(${tooltipStyle.padding}, ${tooltipStyle.padding})`)
|
160
|
+
}
|
161
|
+
)
|
162
|
+
// 使用字串加入svg
|
163
|
+
if (contentData.length) {
|
164
|
+
appendSvg(contentG, contentData[0])
|
165
|
+
}
|
166
|
+
const contentSize = (contentG?.node()) ? getSvgGElementSize(contentG!) : { width: 0, height: 0 }
|
167
|
+
|
168
|
+
// rect size
|
169
|
+
rect
|
170
|
+
.attr('width', contentSize.width + tooltipStyle.padding * 2)
|
171
|
+
.attr('height', contentSize.height + tooltipStyle.padding * 2)
|
172
|
+
|
173
|
+
// -- tooltipG --
|
174
|
+
// 取得tooltip <g>的尺寸
|
175
|
+
const tooltipSize = (tooltipBox?.node()) ? getSvgGElementSize(tooltipBox!) : { width: 0, height: 0 }
|
176
|
+
// const minX = 0
|
177
|
+
const maxX = rootWidth - tooltipSize.width
|
178
|
+
// const minY = 0
|
179
|
+
const maxY = rootHeight - tooltipSize.height
|
180
|
+
|
181
|
+
// -- 相對游標位置的offset --
|
182
|
+
const offsetX = (event.offsetX + tooltipStyle.offset[0]) > maxX ? maxX - event.offsetX : tooltipStyle.offset[0]
|
183
|
+
const offsetY = (event.offsetY + tooltipStyle.offset[1]) > maxY ? maxY - event.offsetY : tooltipStyle.offset[1]
|
184
|
+
tooltipBox.attr('transform', d => `translate(${offsetX}, ${offsetY})`)
|
185
|
+
tooltipBox.attr('transform', d => `translate(${offsetX}, ${offsetY})`)
|
186
|
+
|
187
|
+
// const minX = containerSize.x
|
188
|
+
// const maxX = containerSize.x + containerSize.width - tooltipSize.width
|
189
|
+
// const minY = containerSize.y
|
190
|
+
// const maxY = containerSize.y + containerSize.height - tooltipSize.height
|
191
|
+
|
192
|
+
// .style('position', 'absolute')
|
193
|
+
// .style('z-index', 10000)
|
194
|
+
// .style('left', (d) => {
|
195
|
+
// const x = d.x + this.fullParams.offsetX! - containerSize.x
|
196
|
+
// if (x < minX) {
|
197
|
+
// return `${minX}px`
|
198
|
+
// } else if (x > maxX) {
|
199
|
+
// return `${maxX}px`
|
200
|
+
// }
|
201
|
+
// return `${x}px`
|
202
|
+
// })
|
203
|
+
// .style('top', (d) => {
|
204
|
+
// const y = d.y + this.fullParams.offsetY! - containerSize.y
|
205
|
+
// if (y < minY) {
|
206
|
+
// return `${minY}px`
|
207
|
+
// } else if (y > maxY) {
|
208
|
+
// return `${maxY}px`
|
209
|
+
// }
|
210
|
+
// return `${y}px`
|
211
|
+
// })
|
212
|
+
// .html((d) => d.contentHtml)
|
213
|
+
|
214
|
+
}
|
215
|
+
|
216
|
+
export const createBaseTooltip: BasePluginFn<BaseTooltipContext> = (pluginName: string, {
|
217
|
+
rootSelection,
|
218
|
+
fullParams$,
|
219
|
+
fullChartParams$,
|
220
|
+
layout$,
|
221
|
+
event$
|
222
|
+
}) => {
|
223
|
+
|
224
|
+
const destroy$ = new Subject()
|
225
|
+
|
226
|
+
const gClassName = getClassName(pluginName, 'g')
|
227
|
+
const boxClassName = getClassName(pluginName, 'box')
|
228
|
+
|
229
|
+
// 事件觸發
|
230
|
+
const eventMouseover$: Observable<EventTypeMap<any>> = event$.pipe(
|
231
|
+
takeUntil(destroy$),
|
232
|
+
filter(d => d.eventName === 'mouseover' || d.eventName === 'mousemove'),
|
233
|
+
// distinctUntilChanged((prev, current) => prev.eventName === current.eventName)
|
234
|
+
// map(d => d as EventTypeMap<typeof chartType>)
|
235
|
+
)
|
236
|
+
const eventMouseout$: Observable<EventTypeMap<any>> = event$.pipe(
|
237
|
+
takeUntil(destroy$),
|
238
|
+
filter(d => d.eventName === 'mouseout'),
|
239
|
+
// distinctUntilChanged((prev, current) => prev.eventName === current.eventName)
|
240
|
+
// map(d => d as EventTypeMap<typeof chartType>)
|
241
|
+
)
|
242
|
+
|
243
|
+
const textSizePx$ = textSizePxObservable(fullChartParams$)
|
244
|
+
|
245
|
+
const tooltipStyle$: Observable<BaseTooltipStyle> = combineLatest({
|
246
|
+
fullChartParams: fullChartParams$,
|
247
|
+
fullParams: fullParams$,
|
248
|
+
textSizePx: textSizePx$
|
249
|
+
}).pipe(
|
250
|
+
takeUntil(destroy$),
|
251
|
+
switchMap(async d => d),
|
252
|
+
map(data => {
|
253
|
+
return {
|
254
|
+
backgroundColor: getColor(data.fullParams.backgroundColorType, data.fullChartParams),
|
255
|
+
backgroundOpacity: data.fullParams.backgroundOpacity,
|
256
|
+
strokeColor: getColor(data.fullParams.strokeColorType, data.fullChartParams),
|
257
|
+
offset: data.fullParams.offset,
|
258
|
+
padding: data.fullParams.padding,
|
259
|
+
textSize: data.fullChartParams.styles.textSize,
|
260
|
+
textSizePx: data.textSizePx,
|
261
|
+
textColor: getColor(data.fullParams.textColorType, data.fullChartParams),
|
262
|
+
seriesColors: data.fullChartParams.colors[data.fullChartParams.colorScheme].series
|
263
|
+
}
|
264
|
+
})
|
265
|
+
)
|
266
|
+
|
267
|
+
const contentRenderFn$: Observable<((eventData: EventTypeMap<any>) => string)> = combineLatest({
|
268
|
+
fullParams: fullParams$,
|
269
|
+
tooltipStyle: tooltipStyle$
|
270
|
+
}).pipe(
|
271
|
+
takeUntil(destroy$),
|
272
|
+
switchMap(async d => d),
|
273
|
+
map(data => {
|
274
|
+
// if (data.fullParams.svgRenderFn) {
|
275
|
+
// return data.fullParams.svgRenderFn
|
276
|
+
// }
|
277
|
+
// // 將textRenderFn回傳的資料使用<text>包裝起來
|
278
|
+
// return (eventData: EventTypeMap<any>) => {
|
279
|
+
// const textArr: string | string[] | null = data.fullParams.textRenderFn
|
280
|
+
// ? data.fullParams.textRenderFn(eventData as any)
|
281
|
+
// : null
|
282
|
+
// return textToSvg(textArr, data.tooltipStyle)
|
283
|
+
// }
|
284
|
+
return (eventData: EventTypeMap<any>) => {
|
285
|
+
const renderText: string | string[] = data.fullParams.renderFn(
|
286
|
+
// mouseover事件的資料
|
287
|
+
eventData,
|
288
|
+
// context
|
289
|
+
{
|
290
|
+
styles: data.tooltipStyle
|
291
|
+
}
|
292
|
+
)
|
293
|
+
// string
|
294
|
+
if (typeof renderText === 'string') {
|
295
|
+
const trimText = renderText.trim()
|
296
|
+
const isSvgText = trimText.slice(0, 1) === '<' && trimText.slice(trimText.length - 1, trimText.length) === '>'
|
297
|
+
|
298
|
+
if (isSvgText) {
|
299
|
+
return renderText // svg字串
|
300
|
+
} else {
|
301
|
+
const textArr = renderText.split('\n')
|
302
|
+
return textToSvg(textArr, data.tooltipStyle) // 多行文字轉svg字串
|
303
|
+
}
|
304
|
+
}
|
305
|
+
// string[]
|
306
|
+
else if (Array.isArray(renderText)) {
|
307
|
+
return textToSvg(renderText, data.tooltipStyle) // 多行文字轉svg字串
|
308
|
+
}
|
309
|
+
return ''
|
310
|
+
}
|
311
|
+
})
|
312
|
+
)
|
313
|
+
|
314
|
+
const mouseoverTooltipSvg$ = combineLatest({
|
315
|
+
event: eventMouseover$,
|
316
|
+
contentRenderFn: contentRenderFn$
|
317
|
+
}).pipe(
|
318
|
+
takeUntil(destroy$),
|
319
|
+
switchMap(async d => d),
|
320
|
+
map(d => d.contentRenderFn(d.event))
|
321
|
+
)
|
322
|
+
|
323
|
+
const mouseoutTooltipSvg$ = eventMouseout$.pipe(
|
324
|
+
takeUntil(destroy$),
|
325
|
+
map(d => '')
|
326
|
+
)
|
327
|
+
|
328
|
+
const svgString$ = merge(mouseoverTooltipSvg$, mouseoutTooltipSvg$)
|
329
|
+
.pipe(
|
330
|
+
takeUntil(destroy$),
|
331
|
+
distinctUntilChanged((a, b) => a === b)
|
332
|
+
)
|
333
|
+
|
334
|
+
const eventTooltip$ = merge(eventMouseover$, eventMouseout$)
|
335
|
+
.pipe(
|
336
|
+
takeUntil(destroy$),
|
337
|
+
// filter(d => {
|
338
|
+
// return (d.eventName === 'mouseover' || d.eventName === 'mousemove' || d.eventName === 'mouseout')
|
339
|
+
// && d.event != undefined
|
340
|
+
// }),
|
341
|
+
map(d => d.event!),
|
342
|
+
)
|
343
|
+
|
344
|
+
combineLatest({
|
345
|
+
svgString: svgString$,
|
346
|
+
eventTooltip: eventTooltip$,
|
347
|
+
layout: layout$,
|
348
|
+
tooltipStyle: tooltipStyle$,
|
349
|
+
}).pipe(
|
350
|
+
takeUntil(destroy$),
|
351
|
+
switchMap(async d => d),
|
352
|
+
).subscribe(data => {
|
353
|
+
renderTooltip({
|
354
|
+
rootSelection,
|
355
|
+
pluginName,
|
356
|
+
gClassName,
|
357
|
+
boxClassName,
|
358
|
+
rootWidth: data.layout.rootWidth,
|
359
|
+
rootHeight: data.layout.rootHeight,
|
360
|
+
svgString: data.svgString,
|
361
|
+
tooltipStyle: data.tooltipStyle,
|
362
|
+
event: data.eventTooltip
|
363
|
+
})
|
364
|
+
})
|
365
|
+
|
366
|
+
|
367
|
+
|
368
|
+
// const chartType$ = eventMouseover$.pipe(
|
369
|
+
// takeUntil(destroy$),
|
370
|
+
// map(d => d.type),
|
371
|
+
// distinctUntilChanged()
|
372
|
+
// )
|
373
|
+
|
374
|
+
|
375
|
+
// eventMouseover$.subscribe(event => {
|
376
|
+
|
377
|
+
// })
|
378
|
+
|
379
|
+
|
380
|
+
return () => {
|
381
|
+
destroy$.next(undefined)
|
382
|
+
}
|
383
|
+
}
|
package/src/grid/defaults.ts
CHANGED
@@ -9,8 +9,11 @@ import type {
|
|
9
9
|
GroupAxisParams,
|
10
10
|
ValueAxisParams,
|
11
11
|
ValueStackAxisParams,
|
12
|
-
|
13
|
-
|
12
|
+
GridTooltipParams,
|
13
|
+
GridZoomParams,
|
14
|
+
GridLegendParams
|
15
|
+
} from '../../lib/plugins-basic-types'
|
16
|
+
import { measureTextWidth } from '../utils/commonUtils'
|
14
17
|
|
15
18
|
export const DEFAULT_LINES_PARAMS: LinesParams = {
|
16
19
|
lineCurve: 'curveLinear',
|
@@ -108,13 +111,14 @@ export const DEFAULT_VALUE_STACK_AXIS_PARAMS: ValueStackAxisParams = {
|
|
108
111
|
...DEFAULT_VALUE_AXIS_PARAMS
|
109
112
|
}
|
110
113
|
|
111
|
-
export const
|
114
|
+
export const DEFAULT_GRID_ZOOM_PARAMS: GridZoomParams = {
|
112
115
|
|
113
116
|
}
|
114
117
|
|
115
118
|
export const DEFAULT_GRID_LEGEND_PARAMS: GridLegendParams = {
|
116
|
-
position: 'right',
|
117
|
-
justify: 'end',
|
119
|
+
// position: 'right',
|
120
|
+
// justify: 'end',
|
121
|
+
placement: 'bottom',
|
118
122
|
padding: 28,
|
119
123
|
// offset: [0, 0],
|
120
124
|
backgroundFill: 'none',
|
@@ -126,3 +130,116 @@ export const DEFAULT_GRID_LEGEND_PARAMS: GridLegendParams = {
|
|
126
130
|
// highlightEvent: false
|
127
131
|
textColorType: 'primary'
|
128
132
|
}
|
133
|
+
|
134
|
+
export const DEFAULT_GRID_TOOLTIP_PARAMS: GridTooltipParams = {
|
135
|
+
backgroundColorType: 'background',
|
136
|
+
strokeColorType: 'primary',
|
137
|
+
backgroundOpacity: 0.8,
|
138
|
+
textColorType: 'primary',
|
139
|
+
offset: [20, 5],
|
140
|
+
padding: 10,
|
141
|
+
// renderFn: (eventData, { styles }) => {
|
142
|
+
// const bulletWidth = styles.textSizePx * 0.7
|
143
|
+
// const offset = (styles.textSizePx / 2) - (bulletWidth / 2)
|
144
|
+
// if (eventData.highlightTarget === 'group') {
|
145
|
+
// const titleSvg = `<g><text dominant-baseline="hanging" font-size="${styles.textSizePx}">${eventData.groupLabel}</text></g>`
|
146
|
+
// const maxLengthText = eventData.groups.reduce((acc, group) => {
|
147
|
+
// const text = `${group.seriesLabel}${group.value}`
|
148
|
+
// return text.length > acc.length ? text : acc
|
149
|
+
// }, '')
|
150
|
+
// const maxTextWidth = measureTextWidth(maxLengthText, styles.textSizePx)
|
151
|
+
// const lineEndX = maxTextWidth + styles.textSizePx * 2
|
152
|
+
// const contentSvg = eventData.groups
|
153
|
+
// .map((group, i) => {
|
154
|
+
// const y = i * styles.textSizePx * 1.5
|
155
|
+
// return `<g transform="translate(0, ${styles.textSizePx * 2})">
|
156
|
+
// <rect width="${bulletWidth}" height="${bulletWidth}" x="${offset}" y="${y + offset}" rx="${bulletWidth / 2}" fill="${group.color}"></rect>
|
157
|
+
// <text x="${styles.textSizePx * 1.5}" y="${y}" font-size="${styles.textSizePx}" dominant-baseline="hanging" fill="${styles.textColor}">
|
158
|
+
// <tspan>${group.seriesLabel}</tspan>
|
159
|
+
// <tspan font-weight="bold" text-anchor="end" x="${lineEndX}">${group.value}</tspan>
|
160
|
+
// </text>
|
161
|
+
// </g>`
|
162
|
+
// })
|
163
|
+
// .join('')
|
164
|
+
// return `${titleSvg}
|
165
|
+
// ${contentSvg}`
|
166
|
+
// } else if (eventData.highlightTarget === 'series') {
|
167
|
+
// const titleSvg = `<g><text dominant-baseline="hanging" font-size="${styles.textSizePx}">${eventData.groupLabel}</text></g>`
|
168
|
+
// const textWidth = measureTextWidth(`${eventData.seriesLabel}${eventData.datum.value}`, styles.textSizePx)
|
169
|
+
// const lineEndX = textWidth + styles.textSizePx * 2
|
170
|
+
// const contentSvg = `<g transform="translate(0, ${styles.textSizePx * 2})">
|
171
|
+
// <rect width="${bulletWidth}" height="${bulletWidth}" x="${offset}" y="${offset}" rx="${bulletWidth / 2}" fill="${eventData.datum.color}"></rect>
|
172
|
+
// <text x="${styles.textSizePx * 1.5}" font-size="${styles.textSizePx}" dominant-baseline="hanging" fill="${styles.textColor}">
|
173
|
+
// <tspan>${eventData.seriesLabel}</tspan>
|
174
|
+
// <tspan font-weight="bold" text-anchor="end" x="${lineEndX}">${eventData.datum.value}</tspan>
|
175
|
+
// </text>
|
176
|
+
// </g>`
|
177
|
+
// return `${titleSvg}
|
178
|
+
// ${contentSvg}`
|
179
|
+
// } else {
|
180
|
+
// const textWidth = measureTextWidth(`${eventData.datum.label}${eventData.datum.value}`, styles.textSizePx)
|
181
|
+
// const lineEndX = textWidth + styles.textSizePx * 2
|
182
|
+
// return `<g>
|
183
|
+
// <rect width="${bulletWidth}" height="${bulletWidth}" x="${offset}" y="${offset}" rx="${bulletWidth / 2}" fill="${eventData.datum.color}"></rect>
|
184
|
+
// <text x="${styles.textSizePx * 1.5}" font-size="${styles.textSizePx}" dominant-baseline="hanging" fill="${styles.textColor}">
|
185
|
+
// <tspan>${eventData.datum.label}</tspan>
|
186
|
+
// <tspan font-weight="bold" text-anchor="end" x="${lineEndX}">${eventData.datum.value}</tspan>
|
187
|
+
// </text>
|
188
|
+
// </g>`
|
189
|
+
// }
|
190
|
+
// },
|
191
|
+
renderFn: (eventData, { styles }) => {
|
192
|
+
const bulletWidth = styles.textSizePx * 0.7
|
193
|
+
const offset = (styles.textSizePx / 2) - (bulletWidth / 2)
|
194
|
+
|
195
|
+
const titleSvg = `<g><text dominant-baseline="hanging" font-size="${styles.textSizePx}">${eventData.groupLabel}</text></g>`
|
196
|
+
const maxLengthText = eventData.groups.reduce((acc, group) => {
|
197
|
+
const text = `${group.seriesLabel}${group.value}`
|
198
|
+
return text.length > acc.length ? text : acc
|
199
|
+
}, '')
|
200
|
+
const maxTextWidth = measureTextWidth(maxLengthText, styles.textSizePx)
|
201
|
+
const lineEndX = maxTextWidth + styles.textSizePx * 2
|
202
|
+
const contentSvg = eventData.groups
|
203
|
+
.map((group, i) => {
|
204
|
+
const y = i * styles.textSizePx * 1.5
|
205
|
+
const isHighlight = group.id === (eventData.datum && eventData.datum.id)
|
206
|
+
return `<g transform="translate(0, ${styles.textSizePx * 2})">
|
207
|
+
<rect width="${bulletWidth}" height="${bulletWidth}" x="${offset}" y="${y + offset}" rx="${bulletWidth / 2}" fill="${group.color}"></rect>
|
208
|
+
<text x="${styles.textSizePx * 1.5}" y="${y}" font-size="${styles.textSizePx}" dominant-baseline="hanging" fill="${styles.textColor}">
|
209
|
+
<tspan font-weight="${isHighlight ? 'bold' : ''}">${group.seriesLabel}</tspan>
|
210
|
+
<tspan font-weight="bold" text-anchor="end" x="${lineEndX}">${group.value}</tspan>
|
211
|
+
</text>
|
212
|
+
</g>`
|
213
|
+
})
|
214
|
+
.join('')
|
215
|
+
return `${titleSvg}
|
216
|
+
${contentSvg}`
|
217
|
+
}
|
218
|
+
}
|
219
|
+
DEFAULT_GRID_TOOLTIP_PARAMS.renderFn.toString = () => `(eventData, { styles }) => {
|
220
|
+
const bulletWidth = styles.textSizePx * 0.7
|
221
|
+
const offset = (styles.textSizePx / 2) - (bulletWidth / 2)
|
222
|
+
|
223
|
+
const titleSvg = \`<g><text dominant-baseline="hanging" font-size="\${styles.textSizePx}">\${eventData.groupLabel}</text></g>\`
|
224
|
+
const maxLengthText = eventData.groups.reduce((acc, group) => {
|
225
|
+
const text = \`\${group.seriesLabel}\${group.value}\`
|
226
|
+
return text.length > acc.length ? text : acc
|
227
|
+
}, '')
|
228
|
+
const maxTextWidth = measureTextWidth(maxLengthText, styles.textSizePx)
|
229
|
+
const lineEndX = maxTextWidth + styles.textSizePx * 2
|
230
|
+
const contentSvg = eventData.groups
|
231
|
+
.map((group, i) => {
|
232
|
+
const y = i * styles.textSizePx * 1.5
|
233
|
+
const isHighlight = group.id === (eventData.datum && eventData.datum.id)
|
234
|
+
return \`<g transform="translate(0, \${styles.textSizePx * 2})">
|
235
|
+
<rect width="\${bulletWidth}" height="\${bulletWidth}" x="\${offset}" y="\${y + offset}" rx="\${bulletWidth / 2}" fill="\${group.color}"></rect>
|
236
|
+
<text x="\${styles.textSizePx * 1.5}" y="\${y}" font-size="\${styles.textSizePx}" dominant-baseline="hanging" fill="\${styles.textColor}">
|
237
|
+
<tspan font-weight="\${isHighlight ? 'bold' : ''}">\${group.seriesLabel}</tspan>
|
238
|
+
<tspan font-weight="bold" text-anchor="end" x="\${lineEndX}">\${group.value}</tspan>
|
239
|
+
</text>
|
240
|
+
</g>\`
|
241
|
+
})
|
242
|
+
.join('')
|
243
|
+
return \`\${titleSvg}
|
244
|
+
\${contentSvg}\`
|
245
|
+
}`
|
package/src/grid/index.ts
CHANGED
@@ -11,5 +11,6 @@ export { GridLegend } from './plugins/GridLegend'
|
|
11
11
|
export { GroupAxis } from './plugins/GroupAxis'
|
12
12
|
export { ValueAxis } from './plugins/ValueAxis'
|
13
13
|
export { ValueStackAxis } from './plugins/ValueStackAxis'
|
14
|
-
export {
|
14
|
+
export { GridTooltip } from './plugins/GridTooltip'
|
15
|
+
export { GridZoom } from './plugins/GridZoom'
|
15
16
|
export { GroupAux } from './plugins/GroupAux'
|
@@ -21,6 +21,17 @@ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_GRID_LE
|
|
21
21
|
layerIndex: LAYER_INDEX_OF_INFO,
|
22
22
|
validator: (params, { validateColumns }) => {
|
23
23
|
const result = validateColumns(params, {
|
24
|
+
placement: {
|
25
|
+
toBe: '"top" | "top-start" | "top-end" | "bottom" | "bottom-start" | "bottom-end" | "left" | "left-start" | "left-end" | "right" | "right-start" | "right-end"',
|
26
|
+
test: (value) => {
|
27
|
+
return [
|
28
|
+
'top', 'top-start', 'top-end',
|
29
|
+
'bottom', 'bottom-start', 'bottom-end',
|
30
|
+
'left', 'left-start', 'left-end',
|
31
|
+
'right', 'right-start', 'right-end'
|
32
|
+
].includes(value)
|
33
|
+
}
|
34
|
+
},
|
24
35
|
padding: {
|
25
36
|
toBeTypes: ['number']
|
26
37
|
},
|
@@ -0,0 +1,66 @@
|
|
1
|
+
import {
|
2
|
+
Subject,
|
3
|
+
} from 'rxjs'
|
4
|
+
import type { DefinePluginConfig } from '../../../lib/core-types'
|
5
|
+
import {
|
6
|
+
defineGridPlugin } from '../../../lib/core'
|
7
|
+
import { DEFAULT_GRID_TOOLTIP_PARAMS } from '../defaults'
|
8
|
+
import { LAYER_INDEX_OF_TOOLTIP } from '../../const'
|
9
|
+
import { createBaseTooltip } from '../../base/BaseTooltip'
|
10
|
+
|
11
|
+
const pluginName = 'GridTooltip'
|
12
|
+
|
13
|
+
const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_GRID_TOOLTIP_PARAMS> = {
|
14
|
+
name: pluginName,
|
15
|
+
defaultParams: DEFAULT_GRID_TOOLTIP_PARAMS,
|
16
|
+
layerIndex: LAYER_INDEX_OF_TOOLTIP,
|
17
|
+
validator: (params, { validateColumns }) => {
|
18
|
+
const result = validateColumns(params, {
|
19
|
+
backgroundColorType: {
|
20
|
+
toBeOption: 'ColorType',
|
21
|
+
},
|
22
|
+
backgroundOpacity: {
|
23
|
+
toBeTypes: ['number']
|
24
|
+
},
|
25
|
+
strokeColorType: {
|
26
|
+
toBeOption: 'ColorType',
|
27
|
+
},
|
28
|
+
offset: {
|
29
|
+
toBe: '[number, number]',
|
30
|
+
test: (value: any) => {
|
31
|
+
return Array.isArray(value)
|
32
|
+
&& value.length === 2
|
33
|
+
&& typeof value[0] === 'number'
|
34
|
+
&& typeof value[1] === 'number'
|
35
|
+
}
|
36
|
+
},
|
37
|
+
padding: {
|
38
|
+
toBeTypes: ['number']
|
39
|
+
},
|
40
|
+
textColorType: {
|
41
|
+
toBeOption: 'ColorType',
|
42
|
+
},
|
43
|
+
renderFn: {
|
44
|
+
toBeTypes: ['Function']
|
45
|
+
},
|
46
|
+
})
|
47
|
+
return result
|
48
|
+
}
|
49
|
+
}
|
50
|
+
|
51
|
+
export const GridTooltip = defineGridPlugin(pluginConfig)(({ selection, rootSelection, name, subject, observer }) => {
|
52
|
+
const destroy$ = new Subject()
|
53
|
+
|
54
|
+
const unsubscribeTooltip = createBaseTooltip(pluginName, {
|
55
|
+
rootSelection,
|
56
|
+
fullParams$: observer.fullParams$,
|
57
|
+
fullChartParams$: observer.fullChartParams$,
|
58
|
+
layout$: observer.layout$,
|
59
|
+
event$: subject.event$,
|
60
|
+
})
|
61
|
+
|
62
|
+
return () => {
|
63
|
+
destroy$.next(undefined)
|
64
|
+
unsubscribeTooltip()
|
65
|
+
}
|
66
|
+
})
|
@@ -13,16 +13,16 @@ import type { DefinePluginConfig } from '../../../lib/core-types'
|
|
13
13
|
import type { DataFormatterGrid } from '../../../lib/core-types'
|
14
14
|
import {
|
15
15
|
defineGridPlugin, createAxisLinearScale } from '../../../lib/core'
|
16
|
-
import {
|
16
|
+
import { DEFAULT_GRID_ZOOM_PARAMS } from '../defaults'
|
17
17
|
import { getClassName, getUniID } from '../../utils/orbchartsUtils'
|
18
18
|
import { LAYER_INDEX_OF_ROOT } from '../../const'
|
19
19
|
|
20
|
-
const pluginName = '
|
20
|
+
const pluginName = 'GridZoom'
|
21
21
|
const rectClassName = getClassName(pluginName, 'rect')
|
22
22
|
|
23
|
-
const pluginConfig: DefinePluginConfig<typeof pluginName, typeof
|
23
|
+
const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_GRID_ZOOM_PARAMS> = {
|
24
24
|
name: pluginName,
|
25
|
-
defaultParams:
|
25
|
+
defaultParams: DEFAULT_GRID_ZOOM_PARAMS,
|
26
26
|
layerIndex: LAYER_INDEX_OF_ROOT,
|
27
27
|
validator: (params, { validateColumns }) => {
|
28
28
|
return {
|
@@ -33,7 +33,7 @@ const pluginConfig: DefinePluginConfig<typeof pluginName, typeof DEFAULT_SCALING
|
|
33
33
|
}
|
34
34
|
}
|
35
35
|
|
36
|
-
export const
|
36
|
+
export const GridZoom = defineGridPlugin(pluginConfig)(({ selection, rootSelection, name, observer, subject }) => {
|
37
37
|
|
38
38
|
const destroy$ = new Subject()
|
39
39
|
|