UNPKG

billboard.js

Version:

Re-usable easy interface JavaScript chart library, based on D3 v4+

701 lines (577 loc) 18.4 kB
/** * 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); } }); } } };