UNPKG

recharts

Version:
116 lines (110 loc) 5.13 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.mouseMoveMiddleware = exports.mouseMoveAction = exports.mouseClickMiddleware = exports.mouseClickAction = void 0; var _toolkit = require("@reduxjs/toolkit"); var _tooltipSlice = require("./tooltipSlice"); var _selectActivePropsFromChartPointer = require("./selectors/selectActivePropsFromChartPointer"); var _selectTooltipEventType = require("./selectors/selectTooltipEventType"); var _getRelativeCoordinate = require("../util/getRelativeCoordinate"); var mouseClickAction = exports.mouseClickAction = (0, _toolkit.createAction)('mouseClick'); var mouseClickMiddleware = exports.mouseClickMiddleware = (0, _toolkit.createListenerMiddleware)(); // TODO: there's a bug here when you click the chart the activeIndex resets to zero mouseClickMiddleware.startListening({ actionCreator: mouseClickAction, effect: (action, listenerApi) => { var mousePointer = action.payload; var activeProps = (0, _selectActivePropsFromChartPointer.selectActivePropsFromChartPointer)(listenerApi.getState(), (0, _getRelativeCoordinate.getRelativeCoordinate)(mousePointer)); if ((activeProps === null || activeProps === void 0 ? void 0 : activeProps.activeIndex) != null) { listenerApi.dispatch((0, _tooltipSlice.setMouseClickAxisIndex)({ activeIndex: activeProps.activeIndex, activeDataKey: undefined, activeCoordinate: activeProps.activeCoordinate })); } } }); var mouseMoveAction = exports.mouseMoveAction = (0, _toolkit.createAction)('mouseMove'); var mouseMoveMiddleware = exports.mouseMoveMiddleware = (0, _toolkit.createListenerMiddleware)(); /* * This single rafId is safe because: * 1. Each chart has its own Redux store instance with its own middleware * 2. mouseMoveAction only fires from one DOM element (the chart wrapper) * 3. Rapid mousemove events from the same element SHOULD debounce - we only care about the latest position * This is different from externalEventsMiddleware which handles multiple event types * (click, mouseenter, mouseleave, etc.) that should NOT cancel each other. */ var rafId = null; var timeoutId = null; var latestChartPointer = null; mouseMoveMiddleware.startListening({ actionCreator: mouseMoveAction, effect: (action, listenerApi) => { var mousePointer = action.payload; var state = listenerApi.getState(); var { throttleDelay, throttledEvents } = state.eventSettings; var isThrottled = throttledEvents === 'all' || (throttledEvents === null || throttledEvents === void 0 ? void 0 : throttledEvents.includes('mousemove')); // Cancel any pending execution if (rafId !== null) { cancelAnimationFrame(rafId); rafId = null; } if (timeoutId !== null && (typeof throttleDelay !== 'number' || !isThrottled)) { clearTimeout(timeoutId); timeoutId = null; } /* * Here it is important to resolve the chart pointer _before_ the callback, * because once we leave the current event loop, the mousePointer event object will lose * reference to currentTarget which getRelativeCoordinate uses. */ latestChartPointer = (0, _getRelativeCoordinate.getRelativeCoordinate)(mousePointer); var callback = () => { /* * Here we read a fresh state again inside the callback to ensure we have the latest state values * after any potential actions that may have been dispatched between the original event and this callback. */ var currentState = listenerApi.getState(); var tooltipEventType = (0, _selectTooltipEventType.selectTooltipEventType)(currentState, currentState.tooltip.settings.shared); if (!latestChartPointer) { rafId = null; timeoutId = null; return; } /* * This functionality only applies to charts that have axes. * Graphical items have its own mouse events handling mechanism where they attach events directly to the items. */ if (tooltipEventType === 'axis') { var activeProps = (0, _selectActivePropsFromChartPointer.selectActivePropsFromChartPointer)(currentState, latestChartPointer); if ((activeProps === null || activeProps === void 0 ? void 0 : activeProps.activeIndex) != null) { listenerApi.dispatch((0, _tooltipSlice.setMouseOverAxisIndex)({ activeIndex: activeProps.activeIndex, activeDataKey: undefined, activeCoordinate: activeProps.activeCoordinate })); } else { // this is needed to clear tooltip state when the mouse moves out of the inRange (svg - offset) function, but not yet out of the svg listenerApi.dispatch((0, _tooltipSlice.mouseLeaveChart)()); } } rafId = null; timeoutId = null; }; if (!isThrottled) { callback(); return; } if (throttleDelay === 'raf') { rafId = requestAnimationFrame(callback); } else if (typeof throttleDelay === 'number') { if (timeoutId === null) { timeoutId = setTimeout(callback, throttleDelay); } } } });