|
|
@ -1,35 +1,34 @@ |
|
|
|
import { ChartwerkPod, VueChartwerkPodMixin, TickOrientation, TimeFormat, yAxisOrientation, CrosshairOrientation } from '@chartwerk/core'; |
|
|
|
import { ChartwerkPod, VueChartwerkPodMixin, TimeFormat, yAxisOrientation, CrosshairOrientation } from '@chartwerk/core'; |
|
|
|
import { ScatterData, ScatterOptions, PointType, LineType, Datapoint } from './types'; |
|
|
|
import { ScatterData, ScatterOptions, PointType, LineType, HighlightedData, MouseMoveEvent, DelaunayDataRow } from './types'; |
|
|
|
|
|
|
|
|
|
|
|
import { DelaunayDiagram } from './delaunay'; |
|
|
|
import { DelaunayDiagram } from './models/delaunay'; |
|
|
|
|
|
|
|
import { ScatterSeries } from './models/scatter_series'; |
|
|
|
|
|
|
|
|
|
|
|
import * as d3 from 'd3'; |
|
|
|
import * as d3 from 'd3'; |
|
|
|
import * as _ from 'lodash'; |
|
|
|
import * as _ from 'lodash'; |
|
|
|
|
|
|
|
|
|
|
|
// TODO: use pod state with defaults
|
|
|
|
|
|
|
|
const DEFAULT_POINT_SIZE = 4; |
|
|
|
|
|
|
|
const POINT_HIGHLIGHT_DIAMETER = 4; |
|
|
|
const POINT_HIGHLIGHT_DIAMETER = 4; |
|
|
|
const CROSSHAIR_BACKGROUND_OPACITY = 0.3; |
|
|
|
const CROSSHAIR_BACKGROUND_OPACITY = 0.3; |
|
|
|
const DEFAULT_POINT_TYPE = PointType.CIRCLE; |
|
|
|
|
|
|
|
const DEFAULT_LINE_TYPE = LineType.NONE; |
|
|
|
|
|
|
|
const DEFAULT_LINE_DASHED_AMOUNT = 4; |
|
|
|
const DEFAULT_LINE_DASHED_AMOUNT = 4; |
|
|
|
|
|
|
|
|
|
|
|
export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOptions> { |
|
|
|
export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOptions> { |
|
|
|
metricContainer: any; |
|
|
|
metricContainer: any; |
|
|
|
_delaunayDiagram: DelaunayDiagram; |
|
|
|
_delaunayDiagram: DelaunayDiagram; |
|
|
|
|
|
|
|
series: ScatterSeries; |
|
|
|
|
|
|
|
|
|
|
|
constructor(el: HTMLElement, _series: ScatterData[] = [], _options: ScatterOptions = {}) { |
|
|
|
constructor(el: HTMLElement, _series: ScatterData[] = [], _options: ScatterOptions = {}) { |
|
|
|
super(el, _series, _options); |
|
|
|
super(el, _series, _options); |
|
|
|
|
|
|
|
this.series = new ScatterSeries(_series); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
renderMetrics(): void { |
|
|
|
renderMetrics(): void { |
|
|
|
if(this.series.length === 0) { |
|
|
|
if(!this.series.isSeriesAvailable) { |
|
|
|
this.renderNoDataPointsMessage(); |
|
|
|
this.renderNoDataPointsMessage(); |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
this.updateCrosshair(); |
|
|
|
this.updateCrosshair(); |
|
|
|
|
|
|
|
|
|
|
|
this._delaunayDiagram = new DelaunayDiagram(this.series, this.xScale, this.getYScale.bind(this)); |
|
|
|
this._delaunayDiagram = new DelaunayDiagram(this.series, this.state.xScale, this.getYScale.bind(this)); |
|
|
|
|
|
|
|
|
|
|
|
this.renderLines(); |
|
|
|
this.renderLines(); |
|
|
|
this.renderPoints(); |
|
|
|
this.renderPoints(); |
|
|
@ -47,21 +46,20 @@ export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOption |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
appendCrosshairPoints(): void { |
|
|
|
appendCrosshairPoints(): void { |
|
|
|
this.series.forEach((serie: ScatterData, serieIdx: number) => { |
|
|
|
this.series.visibleSeries.forEach((serie: ScatterData) => { |
|
|
|
this.appendCrosshairPoint(serieIdx); |
|
|
|
this.appendCrosshairPoint(serie.idx, serie.pointType, serie.pointSize); |
|
|
|
}); |
|
|
|
}); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
protected appendCrosshairPoint(serieIdx: number): void { |
|
|
|
protected appendCrosshairPoint(serieIdx: number, pointType: PointType, size: number): void { |
|
|
|
// TODO: add Crosshair type options
|
|
|
|
// TODO: add Crosshair type options
|
|
|
|
const pointType = this.series[serieIdx].pointType || DEFAULT_POINT_TYPE; |
|
|
|
|
|
|
|
switch(pointType) { |
|
|
|
switch(pointType) { |
|
|
|
case PointType.NONE: |
|
|
|
case PointType.NONE: |
|
|
|
return; |
|
|
|
return; |
|
|
|
case PointType.CIRCLE: |
|
|
|
case PointType.CIRCLE: |
|
|
|
this.crosshair.append('circle') |
|
|
|
this.crosshair.append('circle') |
|
|
|
.attr('class', `crosshair-point crosshair-point-${serieIdx} crosshair-background`) |
|
|
|
.attr('class', `crosshair-point crosshair-point-${serieIdx} crosshair-background`) |
|
|
|
.attr('r', this.getCrosshairCircleBackgroundSize(serieIdx)) |
|
|
|
.attr('r', this.getCrosshairCircleBackgroundSize(size, pointType)) |
|
|
|
.attr('clip-path', `url(#${this.rectClipId})`) |
|
|
|
.attr('clip-path', `url(#${this.rectClipId})`) |
|
|
|
.style('opacity', CROSSHAIR_BACKGROUND_OPACITY) |
|
|
|
.style('opacity', CROSSHAIR_BACKGROUND_OPACITY) |
|
|
|
.style('pointer-events', 'none') |
|
|
|
.style('pointer-events', 'none') |
|
|
@ -70,8 +68,8 @@ export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOption |
|
|
|
case PointType.RECTANGLE: |
|
|
|
case PointType.RECTANGLE: |
|
|
|
this.crosshair.append('rect') |
|
|
|
this.crosshair.append('rect') |
|
|
|
.attr('class', `crosshair-point crosshair-point-${serieIdx} crosshair-background`) |
|
|
|
.attr('class', `crosshair-point crosshair-point-${serieIdx} crosshair-background`) |
|
|
|
.attr('width', this.getCrosshairCircleBackgroundSize(serieIdx)) |
|
|
|
.attr('width', this.getCrosshairCircleBackgroundSize(size, pointType)) |
|
|
|
.attr('height', this.getCrosshairCircleBackgroundSize(serieIdx)) |
|
|
|
.attr('height', this.getCrosshairCircleBackgroundSize(size, pointType)) |
|
|
|
.attr('clip-path', `url(#${this.rectClipId})`) |
|
|
|
.attr('clip-path', `url(#${this.rectClipId})`) |
|
|
|
.style('opacity', CROSSHAIR_BACKGROUND_OPACITY) |
|
|
|
.style('opacity', CROSSHAIR_BACKGROUND_OPACITY) |
|
|
|
.style('pointer-events', 'none') |
|
|
|
.style('pointer-events', 'none') |
|
|
@ -83,134 +81,137 @@ export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOption |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
protected renderLines(): void { |
|
|
|
protected renderLines(): void { |
|
|
|
this.series.forEach((serie, serieIdx) => { |
|
|
|
this.series.visibleSeries.forEach(serie => { |
|
|
|
if(serie.visible === false) { |
|
|
|
this.renderLine(serie); |
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
const lineType = serie.lineType || DEFAULT_LINE_TYPE; |
|
|
|
|
|
|
|
this.renderLine(serie.datapoints, lineType, this.getSerieColor(serieIdx), serie.yOrientation); |
|
|
|
|
|
|
|
}); |
|
|
|
}); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
renderLine(datapoints: number[][], lineType: LineType, color: string, orientation: yAxisOrientation): void { |
|
|
|
renderLine(serie: ScatterData): void { |
|
|
|
if(lineType === LineType.NONE) { |
|
|
|
if(serie.lineType === LineType.NONE) { |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
let strokeDasharray; |
|
|
|
let strokeDasharray; |
|
|
|
// TODO: move to option
|
|
|
|
// TODO: move to option
|
|
|
|
if(lineType === LineType.DASHED) { |
|
|
|
if(serie.lineType === LineType.DASHED) { |
|
|
|
strokeDasharray = DEFAULT_LINE_DASHED_AMOUNT; |
|
|
|
strokeDasharray = DEFAULT_LINE_DASHED_AMOUNT; |
|
|
|
} |
|
|
|
} |
|
|
|
const lineGenerator = d3.line() |
|
|
|
const lineGenerator = d3.line() |
|
|
|
.x((d: [number, number]) => this.xScale(d[0])) |
|
|
|
.x((d: [number, number]) => this.state.xScale(d[0])) |
|
|
|
.y((d: [number, number]) => this.getYScale(orientation)(d[1])); |
|
|
|
.y((d: [number, number]) => this.getYScale(serie.yOrientation)(d[1])); |
|
|
|
|
|
|
|
|
|
|
|
this.metricContainer |
|
|
|
this.metricContainer |
|
|
|
.append('path') |
|
|
|
.append('path') |
|
|
|
.datum(datapoints) |
|
|
|
.datum(serie.datapoints) |
|
|
|
.attr('class', 'metric-path') |
|
|
|
.attr('class', 'metric-path') |
|
|
|
|
|
|
|
.attr('d', lineGenerator) |
|
|
|
.attr('fill', 'none') |
|
|
|
.attr('fill', 'none') |
|
|
|
.style('pointer-events', 'none') |
|
|
|
.attr('stroke', serie.color) |
|
|
|
.attr('stroke', color) |
|
|
|
|
|
|
|
.attr('stroke-width', 1) |
|
|
|
.attr('stroke-width', 1) |
|
|
|
.attr('stroke-opacity', 0.7) |
|
|
|
.attr('stroke-opacity', 0.7) |
|
|
|
.attr('stroke-dasharray', strokeDasharray) |
|
|
|
.attr('stroke-dasharray', strokeDasharray) |
|
|
|
.attr('d', lineGenerator); |
|
|
|
.style('pointer-events', serie.clickCallback ? 'auto' : 'none') |
|
|
|
|
|
|
|
.style('cursor', serie.clickCallback ? 'pointer' : 'crosshair') |
|
|
|
|
|
|
|
.on('click', () => { serie.clickCallback({ target: serie.target, class: serie.class, alias: serie.alias }) }); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
protected renderPoints(): void { |
|
|
|
protected renderPoints(): void { |
|
|
|
if(!this._delaunayDiagram.data) { |
|
|
|
if(!this._delaunayDiagram.data) { |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
this.metricContainer.selectAll(null) |
|
|
|
this.metricContainer.selectAll(null) |
|
|
|
.data(this._delaunayDiagram.data) |
|
|
|
.data(this._delaunayDiagram.data) |
|
|
|
.enter() |
|
|
|
.enter() |
|
|
|
.append('circle') |
|
|
|
.append('circle') |
|
|
|
.filter((d: number[]) => this.series[_.last(d)].pointType !== PointType.RECTANGLE) |
|
|
|
.filter((d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.pointType !== PointType.RECTANGLE) |
|
|
|
.attr('class', (d, i: number) => `metric-element metric-circle point-${i}`) |
|
|
|
.attr('class', (d, i: number) => `metric-element metric-circle point-${i}`) |
|
|
|
.attr('r', (d: number[]) => this.series[_.last(d)].pointSize || DEFAULT_POINT_SIZE) |
|
|
|
.attr('r', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.pointSize) |
|
|
|
.style('fill', (d: number[], i: number) => this.getSeriesColorFromDataRow(d, i)) |
|
|
|
.attr('cx', (d: DelaunayDataRow) => this.state.xScale(d[0])) |
|
|
|
.style('pointer-events', 'none') |
|
|
|
.attr('cy', (d: DelaunayDataRow) => this.getYScale(this.series.getSerieByTarget(d[4])?.yOrientation)(d[1])) |
|
|
|
.attr('cx', (d: any[]) => this.xScale(d[0])) |
|
|
|
.style('fill', (d: DelaunayDataRow, i: number) => this.getSeriesColorFromDataRow(d, i)) |
|
|
|
.attr('cy', (d: any[]) => this.getYScale(this.series[_.last(d)].yOrientation)(d[1])); |
|
|
|
.style('pointer-events', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.clickCallback ? 'auto' : 'none') |
|
|
|
|
|
|
|
.style('cursor', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.clickCallback ? 'pointer' : 'crosshair') |
|
|
|
|
|
|
|
.on('click', (d: DelaunayDataRow) => { |
|
|
|
|
|
|
|
d3.event.stopPropagation(); |
|
|
|
|
|
|
|
const serie = this.series.getSerieByTarget(d[4]); |
|
|
|
|
|
|
|
const serieData = { target: serie?.target, class: serie?.class, alias: serie?.alias }; |
|
|
|
|
|
|
|
serie?.clickCallback(serieData, d); |
|
|
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
this.metricContainer.selectAll(null) |
|
|
|
this.metricContainer.selectAll(null) |
|
|
|
.data(this._delaunayDiagram.data) |
|
|
|
.data(this._delaunayDiagram.data) |
|
|
|
.enter() |
|
|
|
.enter() |
|
|
|
.append('rect') |
|
|
|
.append('rect') |
|
|
|
.filter((d: number[]) => this.series[_.last(d)].pointType === PointType.RECTANGLE) |
|
|
|
.filter((d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.pointType === PointType.RECTANGLE) |
|
|
|
.attr('class', (d, i: number) => `metric-element metric-circle point-${i}`) |
|
|
|
.attr('class', (d, i: number) => `metric-element metric-circle point-${i}`) |
|
|
|
.attr('r', (d: number[]) => this.series[_.last(d)].pointSize || DEFAULT_POINT_SIZE) |
|
|
|
.attr('r', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.pointSize) |
|
|
|
.style('fill', (d: number[]) => this.getSerieColor(_.last(d))) |
|
|
|
.attr('x', (d: DelaunayDataRow) => this.state.xScale(d[0]) - this.series.getSerieByTarget(d[4])?.pointSize / 2) |
|
|
|
.style('pointer-events', 'none') |
|
|
|
.attr('y', (d: DelaunayDataRow) => this.getYScale(this.series.getSerieByTarget(d[4])?.yOrientation)(d[1]) - this.series.getSerieByTarget(d[4])?.pointSize / 2) |
|
|
|
.attr('x', (d: number[]) => this.xScale(d[0]) - (this.series[_.last(d)].pointSize || DEFAULT_POINT_SIZE) / 2) |
|
|
|
.attr('width', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.pointSize) |
|
|
|
.attr('y', (d: number[]) => this.getYScale(this.series[_.last(d)].yOrientation)(d[1]) - (this.series[_.last(d)].pointSize || DEFAULT_POINT_SIZE) / 2) |
|
|
|
.attr('height', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.pointSize) |
|
|
|
.attr('width', (d: number[]) => this.series[_.last(d)].pointSize || DEFAULT_POINT_SIZE) |
|
|
|
.style('fill', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.color) |
|
|
|
.attr('height', (d: number[]) => this.series[_.last(d)].pointSize || DEFAULT_POINT_SIZE); |
|
|
|
.style('pointer-events', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.clickCallback ? 'auto' : 'none') |
|
|
|
|
|
|
|
.style('cursor', (d: DelaunayDataRow) => this.series.getSerieByTarget(d[4])?.clickCallback ? 'pointer' : 'crosshair') |
|
|
|
|
|
|
|
.on('click', (d: DelaunayDataRow) => { |
|
|
|
|
|
|
|
d3.event.stopPropagation(); |
|
|
|
|
|
|
|
const serie = this.series.getSerieByTarget(d[4]); |
|
|
|
|
|
|
|
const serieData = { target: serie?.target, class: serie?.class, alias: serie?.alias }; |
|
|
|
|
|
|
|
serie?.clickCallback(serieData, d); |
|
|
|
|
|
|
|
}); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
getSeriesColorFromDataRow(datapoint: Datapoint): string { |
|
|
|
getSeriesColorFromDataRow(values: DelaunayDataRow, rowIdx: number): string { |
|
|
|
const serieIdx = datapoint.serieIdx; |
|
|
|
const serie = this.series.getSerieByTarget(values[4]); |
|
|
|
if(this.series[serieIdx].colorFormatter) { |
|
|
|
if(serie?.colorFormatter) { |
|
|
|
return this.series[serieIdx].colorFormatter(datapoint); |
|
|
|
return serie.colorFormatter(values, rowIdx); |
|
|
|
} |
|
|
|
} |
|
|
|
return this.getSerieColor(serieIdx); |
|
|
|
return serie.color; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
onPanningEnd(): void { |
|
|
|
onPanningEnd(): void { |
|
|
|
this.isPanning = false; |
|
|
|
this.isPanning = false; |
|
|
|
this.onMouseOut(); |
|
|
|
this.onMouseOut(); |
|
|
|
this._delaunayDiagram.setDelaunayDiagram(this.xScale, this.getYScale.bind(this)); |
|
|
|
this._delaunayDiagram.setDelaunayDiagram(this.state.xScale, this.getYScale.bind(this)); |
|
|
|
if(this.options.eventsCallbacks !== undefined && this.options.eventsCallbacks.panningEnd !== undefined) { |
|
|
|
this.options.callbackPanningEnd([this.state.xValueRange, this.state.yValueRange, this.state.y1ValueRange]); |
|
|
|
this.options.eventsCallbacks.panningEnd([this.state.xValueRange, this.state.yValueRange, this.state.y1ValueRange]); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
console.log('on panning end, but there is no callback'); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
unhighlight(): void { |
|
|
|
unhighlight(): void { |
|
|
|
this.crosshair.selectAll('.crosshair-point').style('display', 'none'); |
|
|
|
this.crosshair.selectAll('.crosshair-point').style('display', 'none'); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
highlight(datapoint: Datapoint): void { |
|
|
|
highlight(pointIdx: number): void { |
|
|
|
this.unhighlight(); |
|
|
|
this.unhighlight(); |
|
|
|
|
|
|
|
|
|
|
|
const serieIdx = datapoint.serieIdx; |
|
|
|
const row = this._delaunayDiagram.getDataRowByIndex(pointIdx); |
|
|
|
const serieOrientation = this.series[serieIdx].yOrientation; |
|
|
|
if(row === undefined || row === null) { |
|
|
|
const size = this.getCrosshairCircleBackgroundSize(serieIdx); |
|
|
|
return; |
|
|
|
this.crosshair.selectAll(`.crosshair-point-${serieIdx}`) |
|
|
|
} |
|
|
|
.attr('cx', this.xScale(datapoint.x)) |
|
|
|
const serie = this.series.getSerieByTarget(row[4]); |
|
|
|
.attr('cy', this.getYScale(serieOrientation)(datapoint.y)) |
|
|
|
const size = this.getCrosshairCircleBackgroundSize(serie.pointSize, serie.pointType); |
|
|
|
.attr('x', this.xScale(datapoint.x - size / 2)) |
|
|
|
this.crosshair.selectAll(`.crosshair-point-${serie.idx}`) |
|
|
|
.attr('y', this.getYScale(serieOrientation)(datapoint.y) - size / 2) |
|
|
|
.attr('cx', this.state.xScale(row[0])) |
|
|
|
.attr('fill', this.getSeriesColorFromDataRow(datapoint)) |
|
|
|
.attr('cy', this.getYScale(serie.yOrientation)(row[1])) |
|
|
|
|
|
|
|
.attr('x', this.state.xScale(row[0]) - size / 2) |
|
|
|
|
|
|
|
.attr('y', this.getYScale(serie.yOrientation)(row[1]) - size / 2) |
|
|
|
|
|
|
|
.attr('fill', this.getSeriesColorFromDataRow(row, pointIdx)) |
|
|
|
.style('display', null); |
|
|
|
.style('display', null); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
protected getCrosshairCircleBackgroundSize(serieIdx: number): number { |
|
|
|
protected getCrosshairCircleBackgroundSize(pointSize: number, pointType: PointType): number { |
|
|
|
const seriePointSize = this.series[serieIdx].pointSize || DEFAULT_POINT_SIZE; |
|
|
|
|
|
|
|
const pointType = this.series[serieIdx].pointType || DEFAULT_POINT_TYPE; |
|
|
|
|
|
|
|
let highlightDiameter = POINT_HIGHLIGHT_DIAMETER; |
|
|
|
let highlightDiameter = POINT_HIGHLIGHT_DIAMETER; |
|
|
|
if(pointType === PointType.RECTANGLE) { |
|
|
|
if(pointType === PointType.RECTANGLE) { |
|
|
|
highlightDiameter = highlightDiameter * 2; |
|
|
|
highlightDiameter = highlightDiameter * 2; |
|
|
|
} |
|
|
|
} |
|
|
|
return seriePointSize + highlightDiameter; |
|
|
|
return pointSize + highlightDiameter; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
public renderSharedCrosshair(values: { x?: number, y?: number }): void { |
|
|
|
public renderSharedCrosshair(values: { x?: number, y?: number }): void { |
|
|
|
this.onMouseOver(); // TODO: refactor to use it once
|
|
|
|
this.onMouseOver(); // TODO: refactor to use it once
|
|
|
|
const eventX = this.xScale(values.x); |
|
|
|
const eventX = this.state.xScale(values.x); |
|
|
|
const eventY = this.yScale(values.y); |
|
|
|
const eventY = this.state.yScale(values.y); |
|
|
|
this.moveCrosshairLine(eventX, eventY); |
|
|
|
this.moveCrosshairLine(eventX, eventY); |
|
|
|
const datapoints = this.findDatapoint(values.x, values.y); |
|
|
|
const datapoints = this.findAndHighlightDatapoints(values.x, values.y); |
|
|
|
|
|
|
|
|
|
|
|
if(this.options.eventsCallbacks === undefined || this.options.eventsCallbacks.sharedCrosshairMove === undefined) { |
|
|
|
this.options.callbackSharedCrosshairMove({ |
|
|
|
console.log('Shared crosshair move, but there is no callback'); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
this.options.eventsCallbacks.sharedCrosshairMove({ |
|
|
|
|
|
|
|
datapoints, |
|
|
|
datapoints, |
|
|
|
eventX, eventY |
|
|
|
eventX, eventY |
|
|
|
}); |
|
|
|
}); |
|
|
@ -241,48 +242,44 @@ export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOption |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
highlightDatapoint(datapoint?: Datapoint): void { |
|
|
|
findAndHighlightDatapoints(eventX: number, eventY: number): { |
|
|
|
if(_.isNil(datapoint)) { |
|
|
|
xValue: number, yValue: number, customValue: number, |
|
|
|
return; |
|
|
|
pointIdx: number, totalPointIdx: number, |
|
|
|
} |
|
|
|
serieInfo: { target: string, alias?: string, class?: string, idx?: number } |
|
|
|
this.highlight(datapoint);
|
|
|
|
} | null { |
|
|
|
} |
|
|
|
if(!this.series.isSeriesAvailable) { |
|
|
|
|
|
|
|
|
|
|
|
findDatapoint(eventX: number, eventY: number): Datapoint | null { |
|
|
|
|
|
|
|
if(this.series === undefined || this.series.length === 0) { |
|
|
|
|
|
|
|
return null; |
|
|
|
return null; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
const pointIndex = this._delaunayDiagram.findPointIndex(eventX, eventY); |
|
|
|
const pointIndex = this._delaunayDiagram.findPointIndex(eventX, eventY); |
|
|
|
const datapoint = this._delaunayDiagram.getDataRowByIndex(pointIndex); |
|
|
|
if(pointIndex === undefined) { |
|
|
|
if(_.isNil(datapoint)) { |
|
|
|
this.unhighlight(); |
|
|
|
// TODO: maybe throw an error?
|
|
|
|
|
|
|
|
return null; |
|
|
|
return null; |
|
|
|
} |
|
|
|
} |
|
|
|
const serieIdx = _.last(datapoint); |
|
|
|
this.highlight(pointIndex); |
|
|
|
let seriePointIdx = pointIndex |
|
|
|
const row = this._delaunayDiagram.data[pointIndex]; |
|
|
|
for(let i = 0; i < serieIdx; i++) { |
|
|
|
const serie = this.series.getSerieByTarget(row[4]); |
|
|
|
seriePointIdx -= this.series[serieIdx].datapoints.length; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return { |
|
|
|
return { |
|
|
|
x: datapoint[0], |
|
|
|
xValue: row[0], |
|
|
|
y: datapoint[1], |
|
|
|
yValue: row[1], |
|
|
|
rawValues: datapoint, |
|
|
|
customValue: row[2], |
|
|
|
serieIdx, |
|
|
|
pointIdx: row[3], |
|
|
|
pointIdx: seriePointIdx, |
|
|
|
serieInfo: { |
|
|
|
|
|
|
|
target: serie.target, |
|
|
|
|
|
|
|
alias: serie.alias, |
|
|
|
|
|
|
|
class: serie.class, |
|
|
|
|
|
|
|
idx: serie.idx, |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
totalPointIdx: pointIndex, |
|
|
|
}; |
|
|
|
}; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
protected getYScale(orientation: yAxisOrientation): d3.ScaleLinear<number, number> { |
|
|
|
protected getYScale(orientation: yAxisOrientation): d3.ScaleLinear<number, number> { |
|
|
|
if(orientation === undefined || orientation === yAxisOrientation.BOTH) { |
|
|
|
|
|
|
|
return this.yScale; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
switch(orientation) { |
|
|
|
switch(orientation) { |
|
|
|
case yAxisOrientation.LEFT: |
|
|
|
case yAxisOrientation.LEFT: |
|
|
|
return this.yScale; |
|
|
|
return this.state.yScale; |
|
|
|
case yAxisOrientation.RIGHT: |
|
|
|
case yAxisOrientation.RIGHT: |
|
|
|
return this.y1Scale; |
|
|
|
return this.state.y1Scale; |
|
|
|
default: |
|
|
|
default: |
|
|
|
throw new Error(`Unknown type of y axis orientation: ${orientation}`)
|
|
|
|
throw new Error(`Unknown type of y axis orientation: ${orientation}`)
|
|
|
|
} |
|
|
|
} |
|
|
@ -292,14 +289,12 @@ export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOption |
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
onMouseMove(): void { |
|
|
|
onMouseMove(d: any): void { |
|
|
|
|
|
|
|
const mousePosition = d3.mouse(this.chartContainer.node()); |
|
|
|
const mousePosition = d3.mouse(this.chartContainer.node()); |
|
|
|
const eventX = mousePosition[0]; |
|
|
|
const eventX = mousePosition[0]; |
|
|
|
const eventY = mousePosition[1]; |
|
|
|
const eventY = mousePosition[1]; |
|
|
|
|
|
|
|
|
|
|
|
// TODO: seems isOutOfChart is deprecated (check clippath correctness)
|
|
|
|
if(this.isPanning === true || this.isBrushing === true) { |
|
|
|
if(this.isOutOfChart() === true || this.isPanning === true || this.isBrushing === true) { |
|
|
|
|
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
return; |
|
|
|
return; |
|
|
|
} else { |
|
|
|
} else { |
|
|
@ -309,26 +304,26 @@ export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOption |
|
|
|
this.moveCrosshairLine(eventX, eventY); |
|
|
|
this.moveCrosshairLine(eventX, eventY); |
|
|
|
|
|
|
|
|
|
|
|
// TOOD: it should be two different methods
|
|
|
|
// TOOD: it should be two different methods
|
|
|
|
const datapoint = this.findDatapoint(eventX, eventY); |
|
|
|
const highlighted = this.findAndHighlightDatapoints(eventX, eventY); |
|
|
|
this.highlightDatapoint(datapoint); |
|
|
|
this.options.callbackMouseMove({ |
|
|
|
if(this.options.eventsCallbacks === undefined || this.options.eventsCallbacks.mouseMove === undefined) { |
|
|
|
bbox: { |
|
|
|
console.log('Mouse move, but there is no callback'); |
|
|
|
clientX: d3.event.clientX, |
|
|
|
return; |
|
|
|
clientY: d3.event.clientY, |
|
|
|
} |
|
|
|
x: eventX, |
|
|
|
// TODO: group fields
|
|
|
|
y: eventY, |
|
|
|
this.options.eventsCallbacks.mouseMove({ |
|
|
|
chartWidth: this.width, |
|
|
|
x: d3.event.clientX, |
|
|
|
chartHeight: this.height, |
|
|
|
y: d3.event.clientY, |
|
|
|
}, |
|
|
|
xval: this.xScale.invert(eventX), |
|
|
|
data: { |
|
|
|
yval: this.xScale.invert(eventY), |
|
|
|
xval: this.state.xScale.invert(eventX), |
|
|
|
highlighted: datapoint, |
|
|
|
yval: this.state.xScale.invert(eventY), |
|
|
|
chartX: eventX, |
|
|
|
highlighted, |
|
|
|
chartWidth: this.width |
|
|
|
} |
|
|
|
}); |
|
|
|
}); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
onMouseOver(): void { |
|
|
|
onMouseOver(): void { |
|
|
|
if(this.isOutOfChart() === true || this.isPanning === true || this.isBrushing === true) { |
|
|
|
if(this.isPanning === true || this.isBrushing === true) { |
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
@ -336,9 +331,7 @@ export class ChartwerkScatterPod extends ChartwerkPod<ScatterData, ScatterOption |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
onMouseOut(): void { |
|
|
|
onMouseOut(): void { |
|
|
|
if(this.options.eventsCallbacks !== undefined && this.options.eventsCallbacks.mouseOut !== undefined) { |
|
|
|
this.options.callbackMouseOut(); |
|
|
|
this.options.eventsCallbacks.mouseOut(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
this.crosshair.style('display', 'none'); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
@ -368,4 +361,4 @@ export const VueChartwerkScatterPodObject = { |
|
|
|
} |
|
|
|
} |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
export { ScatterData, ScatterOptions, TickOrientation, TimeFormat, PointType, LineType }; |
|
|
|
export { ScatterData, ScatterOptions, TimeFormat, PointType, LineType, HighlightedData, MouseMoveEvent }; |
|
|
|