billboard.js
Version:
Re-usable easy interface JavaScript chart library, based on D3 v4+
701 lines (577 loc) • 18.4 kB
text/typescript
/**
* Copyright (c) 2017 ~ present NAVER Corp.
* billboard.js project is licensed under the MIT license
*/
import type {d3Selection} from "../../../types";
import {$COMMON, $EVENT, $SHAPE} from "../../config/classes";
import {getPointer, getScrollPosition, isBoolean, isFunction} from "../../module/util";
export default {
/**
* Initialize the area that detects the event.
* Add a container for the zone that detects the event.
* @private
*/
initEventRect(): void {
const $$ = this;
$$.$el.main.select(`.${$COMMON.chart}`)
.append("g")
.attr("class", $EVENT.eventRects)
.style("fill-opacity", "0");
},
/**
* Redraws the area that detects the event.
* @private
*/
redrawEventRect(): void {
const $$ = this;
const {config, state, $el} = $$;
const isMultipleX = $$.isMultipleX();
const isInverted = config.axis_x_inverted;
if ($el.eventRect) {
$$.updateEventRect($el.eventRect, true);
// do not initialize eventRect when data is empty
} else if ($$.data.targets.length) {
const eventRects = $$.$el.main.select(`.${$EVENT.eventRects}`)
.style("cursor", config.zoom_enabled && config.zoom_type !== "drag" ?
(
config.axis_rotated ? "ns-resize" : "ew-resize"
) :
null)
.classed($EVENT.eventRectsMultiple, isMultipleX)
.classed($EVENT.eventRectsSingle, !isMultipleX);
// append event <rect>
const eventRectUpdate = eventRects.selectAll(`.${$EVENT.eventRect}`)
.data([0])
.enter()
.append("rect");
$$.updateEventRect(eventRectUpdate);
// bind event to <rect> element
$$.updateEventType(eventRectUpdate);
// bind draggable selection
eventRectUpdate.call($$.getDraggableSelection());
$el.eventRect = eventRectUpdate;
if (
$$.state.inputType === "touch" && !$el.svg.on("touchstart.eventRect") &&
!$$.hasArcType()
) {
$$.bindTouchOnEventRect();
}
// when initilazed with empty data and data loaded later, need to update eventRect
state.rendered && $$.updateEventRect($el.eventRect, true);
}
if (!isMultipleX) {
// Set data and update eventReceiver.data
const xAxisTickValues = $$.getMaxDataCountTarget();
if (!config.data_xSort || isInverted) {
xAxisTickValues.sort((a, b) => (isInverted ? b.x - a.x : a.x - b.x));
}
// update data's index value to be alinged with the x Axis
$$.updateDataIndexByX(xAxisTickValues);
$$.updateXs(xAxisTickValues);
$$.updatePointClass?.(true);
state.eventReceiver.data = xAxisTickValues;
}
$$.updateEventRectData();
},
bindTouchOnEventRect(): void {
const $$ = this;
const {config, state, $el: {eventRect, svg}} = $$;
const selectRect = context => {
if ($$.isMultipleX()) {
$$.selectRectForMultipleXs(context);
} else {
const index = $$.getDataIndexFromEvent(state.event);
$$.callOverOutForTouch(index);
index === -1 ? $$.unselectRect() : $$.selectRectForSingle(context, index);
}
};
const unselectRect = () => {
$$.unselectRect();
$$.callOverOutForTouch();
};
// call event.preventDefault()
// according 'interaction.inputType.touch.preventDefault' option
const preventDefault = config.interaction_inputType_touch.preventDefault;
const isPrevented = (isBoolean(preventDefault) && preventDefault) || false;
const preventThreshold = (!isNaN(preventDefault) && preventDefault) || null;
let startPx;
const preventEvent = event => {
const eventType = event.type;
const touch = event.changedTouches[0];
const currentXY = touch[`client${config.axis_rotated ? "Y" : "X"}`];
// prevent document scrolling
if (eventType === "touchstart") {
if (isPrevented) {
event.preventDefault();
} else if (preventThreshold !== null) {
startPx = currentXY;
}
} else if (eventType === "touchmove") {
if (
isPrevented || startPx === true || (
preventThreshold !== null &&
Math.abs(startPx - currentXY) >= preventThreshold
)
) {
// once prevented, keep prevented during whole 'touchmove' context
startPx = true;
event.preventDefault();
}
}
};
// bind touch events
eventRect
.on("touchstart", event => {
state.event = event;
$$.updateEventRect();
})
.on("touchstart.eventRect touchmove.eventRect", event => {
state.event = event;
if (!eventRect.empty() && eventRect.classed($EVENT.eventRect)) {
// if touch points are > 1, means doing zooming interaction. In this case do not execute tooltip codes.
if (
state.dragging || state.flowing || $$.hasArcType() ||
event.touches.length > 1
) {
return;
}
preventEvent(event);
selectRect(eventRect.node());
} else {
unselectRect();
}
}, true)
.on("touchend.eventRect", event => {
state.event = event;
if (!eventRect.empty() && eventRect.classed($EVENT.eventRect)) {
if ($$.hasArcType() || !$$.toggleShape || state.cancelClick) {
state.cancelClick && (state.cancelClick = false);
}
}
}, true);
svg.on("touchstart", event => {
state.event = event;
const {target} = event;
if (target && target !== eventRect.node()) {
unselectRect();
}
});
},
/**
* Update event rect size
* @param {d3Selection} eventRect Event <rect> element
* @param {boolean} force Force to update
* @private
*/
updateEventRect(eventRect?, force = false): void {
const $$ = this;
const {state, $el} = $$;
const {eventReceiver, width, height, rendered, resizing} = state;
const rectElement = eventRect || $el.eventRect;
const updateClientRect = (): void => {
if (eventReceiver) {
const scrollPos = getScrollPosition($el.chart.node());
eventReceiver.rect = rectElement.node()
.getBoundingClientRect()
.toJSON();
eventReceiver.rect.top += scrollPos.y;
eventReceiver.rect.left += scrollPos.x;
}
};
if (!rendered || resizing || force) {
rectElement
.attr("x", 0)
.attr("y", 0)
.attr("width", width)
.attr("height", height);
// only for init
if (!rendered || force) {
rectElement.classed($EVENT.eventRect, true);
}
}
updateClientRect();
},
/**
* Update event type (single or multiple x)
* @param {d3Selection | boolean} target Target element or boolean to rebind event
*/
updateEventType(target: d3Selection | boolean): void {
const $$ = this;
const isRebindCall = isBoolean(target);
const eventRect = isRebindCall ? $$.$el.eventRect : target;
const unbindEvent = isRebindCall ? target !== eventRect?.datum().multipleX : false;
if (eventRect) {
// release previous event listeners
unbindEvent && eventRect?.on("mouseover mousemove mouseout click", null);
$$.isMultipleX() ?
$$.generateEventRectsForMultipleXs(eventRect) :
$$.generateEventRectsForSingleX(eventRect);
}
},
/**
* Updates the location and size of the eventRect.
* @private
*/
updateEventRectData(): void {
const $$ = this;
const {config, scale, state} = $$;
const xScale = scale.zoom || scale.x;
const isRotated = config.axis_rotated;
const isMultipleX = $$.isMultipleX();
let x;
let y;
let w;
let h;
$$.updateEventType(isMultipleX);
if (isMultipleX) {
// TODO: rotated not supported yet
x = 0;
y = 0;
w = state.width;
h = state.height;
} else {
let rectW;
let rectX;
if ($$.axis.isCategorized()) {
rectW = $$.getEventRectWidth();
rectX = d => xScale(d.x) - (rectW / 2);
} else {
const getPrevNextX = ({index}) => ({
prev: $$.getPrevX(index),
next: $$.getNextX(index)
});
rectW = (d): number => {
const x = getPrevNextX(d);
const xDomain = xScale.domain();
let val: number;
// if there this is a single data point make the eventRect full width (or height)
if (x.prev === null && x.next === null) {
val = isRotated ? state.height : state.width;
} else if (x.prev === null) {
val = (xScale(x.next) + xScale(d.x)) / 2;
} else if (x.next === null) {
val = xScale(xDomain[1]) - (
(xScale(x.prev) + xScale(d.x)) / 2
);
} else {
Object.keys(x).forEach((key, i) => {
x[key] = x[key] ?? xDomain[i];
});
val = Math.max(0, (xScale(x.next) - xScale(x.prev)) / 2);
}
return val;
};
rectX = (d): number => {
const x = getPrevNextX(d);
let val: number;
// if there this is a single data point position the eventRect at 0
if (x.prev === null && x.next === null) {
val = 0;
} else if (x.prev === null) {
val = xScale(xScale.domain()[0]);
} else {
val = (xScale(d.x) + xScale(x.prev)) / 2;
}
return val;
};
}
x = isRotated ? 0 : rectX;
y = isRotated ? rectX : 0;
w = isRotated ? state.width : rectW;
h = isRotated ? rectW : state.height;
}
const {eventReceiver} = state;
const call: any = (fn, v) => (isFunction(fn) ? fn(v) : fn);
// reset for possible remains coords data before the data loading
eventReceiver.coords.splice(eventReceiver.data.length);
eventReceiver.data.forEach((d, i) => {
eventReceiver.coords[i] = {
x: call(x, d),
y: call(y, d),
w: call(w, d),
h: call(h, d)
};
});
},
/**
* Seletct rect for single x value
* @param {d3Selection} context Event rect element
* @param {number} index x Axis index
* @private
*/
selectRectForSingle(context: SVGRectElement, index: number): void {
const $$ = this;
const {config, $el: {main, circle}} = $$;
const isSelectionEnabled = config.data_selection_enabled;
const isSelectionGrouped = config.data_selection_grouped;
const isSelectable = config.data_selection_isselectable;
const isTooltipGrouped = config.tooltip_grouped;
const selectedData = $$.getAllValuesOnIndex(index);
if (isTooltipGrouped) {
$$.showTooltip(selectedData, context);
$$.showGridFocus?.(selectedData);
if (!isSelectionEnabled || isSelectionGrouped) {
return;
}
}
// remove possible previous focused state
!circle &&
main.selectAll(`.${$COMMON.EXPANDED}:not(.${$SHAPE.shape}-${index})`).classed(
$COMMON.EXPANDED,
false
);
const shapeAtIndex = main.selectAll(`.${$SHAPE.shape}-${index}`)
.classed($COMMON.EXPANDED, true)
.style("cursor", isSelectable ? "pointer" : null)
.filter(function(d) {
return $$.isWithinShape(this, d);
});
if (shapeAtIndex.empty() && !isTooltipGrouped && config.interaction_onout) {
$$.hideGridFocus?.();
$$.hideTooltip();
!isSelectionGrouped && $$.setExpand(index);
}
shapeAtIndex
.call(selected => {
const d = selected.data();
if (
isSelectionEnabled &&
(isSelectionGrouped || isSelectable?.bind($$.api)(d))
) {
context.style.cursor = "pointer";
}
if (!isTooltipGrouped) {
$$.showTooltip(d, context);
$$.showGridFocus?.(d);
$$.unexpandCircles?.();
selected.each(d => $$.setExpand(index, d.id));
}
});
},
/**
* Select rect for multiple x values
* @param {d3Selection} context Event rect element
* @param {boolean} [triggerEvent=true] Whether trigger event or not
* @private
*/
selectRectForMultipleXs(context: SVGRectElement, triggerEvent = true): void {
const $$ = this;
const {config, state} = $$;
const targetsToShow = $$.filterTargetsToShow($$.data.targets);
// do nothing when dragging
if (state.dragging || $$.hasArcType(targetsToShow)) {
return;
}
const mouse = getPointer(state.event, context);
const closest = $$.findClosestFromTargets(targetsToShow, mouse);
if (triggerEvent && state.mouseover && (!closest || closest.id !== state.mouseover.id)) {
config.data_onout.call($$.api, state.mouseover);
state.mouseover = undefined;
}
if (!closest) {
$$.unselectRect();
return;
}
const sameXData = (
$$.isBubbleType(closest) || $$.isScatterType(closest) || !config.tooltip_grouped
) ?
[closest] :
$$.filterByX(targetsToShow, closest.x);
// show tooltip when cursor is close to some point
const selectedData = sameXData.map(d => $$.addName(d));
$$.showTooltip(selectedData, context);
// expand points
$$.setExpand(closest.index, closest.id, true);
// Show xgrid focus line
$$.showGridFocus(selectedData);
const dist = $$.dist(closest, mouse);
// Show cursor as pointer if point is close to mouse position
if ($$.isBarType(closest.id) || dist < $$.getPointSensitivity(closest)) {
$$.$el.svg.select(`.${$EVENT.eventRect}`).style("cursor", "pointer");
if (triggerEvent && !state.mouseover) {
config.data_onover.call($$.api, closest);
state.mouseover = closest;
}
}
},
/**
* Unselect EventRect.
* @private
*/
unselectRect(): void {
const $$ = this;
const {$el: {circle, tooltip}} = $$;
$$.$el.svg.select(`.${$EVENT.eventRect}`).style("cursor", null);
$$.hideGridFocus();
if (tooltip) {
$$.hideTooltip();
$$._handleLinkedCharts(false);
}
circle && !$$.isPointFocusOnly() && $$.unexpandCircles();
$$.expandBarTypeShapes(false);
},
/**
* Create eventRect for each data on the x-axis.
* Register touch and drag events.
* @param {object} eventRectEnter d3.select($EVENT.eventRects) object.
* @returns {object} d3.select($EVENT.eventRects) object.
* @private
*/
generateEventRectsForSingleX(eventRectEnter) {
const $$ = this;
const {config, state} = $$;
const {eventReceiver} = state;
const rect = eventRectEnter
.style("cursor",
config.data_selection_enabled && config.data_selection_grouped ? "pointer" : null)
.on("click", function(event) {
state.event = event;
const {currentIdx, data} = eventReceiver;
const d = data[
currentIdx === -1 ? $$.getDataIndexFromEvent(event) : currentIdx
];
$$.clickHandlerForSingleX.bind(this)(d, $$);
})
.datum({multipleX: false});
if (state.inputType === "mouse") {
const getData = event => {
const index = event ? $$.getDataIndexFromEvent(event) : eventReceiver.currentIdx;
return index > -1 ? eventReceiver.data[index] : null;
};
rect
.on("mouseover", event => {
state.event = event;
$$.updateEventRect();
Object.values($$.$el.axisTooltip)
.forEach((v: d3Selection) => v?.style("display", null));
})
.on("mousemove", function(event) {
const d = getData(event);
state.event = event;
if (!d) {
return;
}
let {index} = d;
const stepType = config.line_step_type;
// tooltip position match for step-before & step-after
if (
config.line_step_tooltipMatch && $$.hasType("step") &&
/^step\-(before|after)$/.test(stepType)
) {
const scale = $$.scale.zoom || $$.scale.x;
const xs = $$.axis.xs[index];
const inverted = scale.invert(getPointer(event, this)[0]);
if (stepType === "step-after" && inverted < xs) {
index -= 1;
} else if (stepType === "step-before" && inverted > xs) {
index += 1;
}
}
$$.showAxisGridFocus();
const eventOnSameIdx = config.tooltip_grouped &&
index === eventReceiver.currentIdx;
// do nothing while dragging/flowing
if (state.dragging || state.flowing || $$.hasArcType() || eventOnSameIdx) {
config.tooltip_show && eventOnSameIdx && $$.setTooltipPosition();
return;
}
if (index !== eventReceiver.currentIdx) {
$$.setOverOut(false, eventReceiver.currentIdx);
eventReceiver.currentIdx = index;
}
index === -1 ? $$.unselectRect() : $$.selectRectForSingle(this, index);
// As of individual data point(or <path>) element can't bind mouseover/out event
// to determine current interacting element, so use 'mousemove' event instead.
$$.setOverOut(index !== -1, index);
})
.on("mouseout", event => {
state.event = event;
// chart is destroyed
if (
!config || $$.hasArcType() || eventReceiver.currentIdx === -1 ||
!config.interaction_onout
) {
return;
}
$$.hideAxisGridFocus();
$$.unselectRect();
$$.setOverOut(false, eventReceiver.currentIdx);
// reset the event current index
eventReceiver.currentIdx = -1;
});
}
return rect;
},
clickHandlerForSingleX(d, ctx): void {
const $$ = ctx;
const {config, state, $el: {main}} = $$;
if (!d || $$.hasArcType() || state.cancelClick) {
state.cancelClick && (state.cancelClick = false);
return;
}
const {index} = d;
main.selectAll(`.${$SHAPE.shape}-${index}`)
.each(function(d2) {
if (config.data_selection_grouped || $$.isWithinShape(this, d2)) {
$$.toggleShape?.(this, d2, index);
config.data_onclick.bind($$.api)(d2, this);
}
});
},
/**
* Create an eventRect,
* Register touch and drag events.
* @param {object} eventRectEnter d3.select($EVENT.eventRects) object.
* @private
*/
generateEventRectsForMultipleXs(eventRectEnter): void {
const $$ = this;
const {config, state} = $$;
eventRectEnter
.on("click", function(event) {
state.event = event;
$$.clickHandlerForMultipleXS.bind(this)($$);
})
.datum({multipleX: true});
if (state.inputType === "mouse") {
eventRectEnter
.on("mouseover mousemove", function(event) {
state.event = event;
$$.selectRectForMultipleXs(this);
})
.on("mouseout", event => {
state.event = event;
// chart is destroyed
if (!$$.config || $$.hasArcType() || !config.interaction_onout) {
return;
}
$$.unselectRect();
});
}
},
clickHandlerForMultipleXS(ctx): void {
const $$ = ctx;
const {config, state} = $$;
const targetsToShow = $$.filterTargetsToShow($$.data.targets);
if ($$.hasArcType(targetsToShow)) {
return;
}
const mouse = getPointer(state.event, this);
const closest = $$.findClosestFromTargets(targetsToShow, mouse);
const sensitivity = $$.getPointSensitivity(closest);
if (!closest) {
return;
}
// select if selection enabled
if ($$.isBarType(closest.id) || $$.dist(closest, mouse) < sensitivity) {
$$.$el.main.selectAll(`.${$SHAPE.shapes}${$$.getTargetSelectorSuffix(closest.id)}`)
.selectAll(`.${$SHAPE.shape}-${closest.index}`)
.each(function() {
if (config.data_selection_grouped || $$.isWithinShape(this, closest)) {
$$.toggleShape?.(this, closest, closest.index);
config.data_onclick.bind($$.api)(closest, this);
}
});
}
}
};