diff --git a/CHANGELOG.md b/CHANGELOG.md index 259ae4d28dc..7c8485c1d2f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,7 +8,7 @@ - Moved all `EuiHeader` SASS variables to `global_styles` ([#3592](https://github.com/elastic/eui/pull/3592)) - Added `side` prop to `EuiGlobalToastList` for choosing which window side to display toasts ([#3600](https://github.com/elastic/eui/pull/3600)) - Default `titleSize` get's implicitly set to 'm' for `EuiEmptyPrompt` ([#3598](https://github.com/elastic/eui/pull/3598)) -- Updated `logoElastic` to meet brand guidelines ([#3613](https://github.com/elastic/eui/pull/3613)) +- Updated `logoElastic` to meet brand guidelines ([#3613](https://github.com/elastic/eui/pull/3613)) **Bug fixes** @@ -18,6 +18,7 @@ - Fixed `EuiComboBox`'s options list `zIndex` positioning when nested in other `zIndex` contexts ([#3551](https://github.com/elastic/eui/pull/3551)) - Fixed `euiHeaderAffordForFixed` mixin's use of header SASS variable ([#3592](https://github.com/elastic/eui/pull/3592)) - Included `onClick` as a valid prop for `EuiControlBar` **icon** controls ([#3581](https://github.com/elastic/eui/pull/3581)) +- Fixed poor performance of `EuiToolTip` during frequent mouesover/mouseout events ([#3596](https://github.com/elastic/eui/pull/3596)) **Breaking changes** diff --git a/src/components/tool_tip/_tool_tip.scss b/src/components/tool_tip/_tool_tip.scss index 0a06935b378..eb6b1e038a3 100644 --- a/src/components/tool_tip/_tool_tip.scss +++ b/src/components/tool_tip/_tool_tip.scss @@ -8,11 +8,6 @@ position: absolute; opacity: 0; - // Animation delays - &.euiToolTip--delayLong { - animation-delay: $euiAnimSpeedNormal * 5; - } - // Custom sizing $arrowSize: $euiSizeM; $arrowPlusSize: (($arrowSize/2) + 1px) * -1; /* 1 */ diff --git a/src/components/tool_tip/tool_tip.test.tsx b/src/components/tool_tip/tool_tip.test.tsx index bf861b2aa29..d55a0e6f26d 100644 --- a/src/components/tool_tip/tool_tip.test.tsx +++ b/src/components/tool_tip/tool_tip.test.tsx @@ -23,6 +23,7 @@ import { requiredProps, findTestSubject, takeMountedSnapshot, + sleep, } from '../../test'; import { EuiToolTip } from './tool_tip'; @@ -41,7 +42,7 @@ describe('EuiToolTip', () => { expect(component).toMatchSnapshot(); }); - test('shows tooltip on focus', () => { + test('shows tooltip on focus', async () => { const component = mount( @@ -50,6 +51,7 @@ describe('EuiToolTip', () => { const trigger = findTestSubject(component, 'trigger'); trigger.simulate('focus'); + await sleep(260); // wait for showToolTip setTimout expect(takeMountedSnapshot(component)).toMatchSnapshot(); }); }); diff --git a/src/components/tool_tip/tool_tip.tsx b/src/components/tool_tip/tool_tip.tsx index 885cf7ff7de..34ad953f67f 100644 --- a/src/components/tool_tip/tool_tip.tsx +++ b/src/components/tool_tip/tool_tip.tsx @@ -32,6 +32,7 @@ import { keysOf } from '../common'; import { EuiPortal } from '../portal'; import { EuiToolTipPopover } from './tool_tip_popover'; import { findPopoverPosition, htmlIdGenerator, keys } from '../../services'; +import { enqueueStateChange } from '../../services/react'; import { EuiResizeObserver } from '../observer/resize_observer'; @@ -48,13 +49,11 @@ export const POSITIONS = keysOf(positionsToClassNameMap); export type ToolTipDelay = 'regular' | 'long'; -const delayToClassNameMap: { [key in ToolTipDelay]: string | null } = { - regular: null, - long: 'euiToolTip--delayLong', +const delayToMsMap: { [key in ToolTipDelay]: number } = { + regular: 250, + long: 250 * 5, }; -export const DELAY = keysOf(delayToClassNameMap); - interface ToolTipStyles { top: number; left: number | 'auto'; @@ -126,6 +125,7 @@ export class EuiToolTip extends Component { _isMounted = false; anchor: null | HTMLElement = null; popover: null | HTMLElement = null; + private timeoutId?: ReturnType; state: State = { visible: false, @@ -140,11 +140,18 @@ export class EuiToolTip extends Component { delay: 'regular', }; + clearAnimationTimeout = () => { + if (this.timeoutId) { + this.timeoutId = clearTimeout(this.timeoutId) as undefined; + } + }; + componentDidMount() { this._isMounted = true; } componentWillUnmount() { + this.clearAnimationTimeout(); this._isMounted = false; window.removeEventListener('mousemove', this.hasFocusMouseMoveListener); } @@ -184,7 +191,11 @@ export class EuiToolTip extends Component { }; showToolTip = () => { - this.setState({ visible: true }); + if (!this.timeoutId) { + this.timeoutId = setTimeout(() => { + enqueueStateChange(() => this.setState({ visible: true })); + }, delayToMsMap[this.props.delay]); + } }; positionToolTip = () => { @@ -232,8 +243,9 @@ export class EuiToolTip extends Component { }; hideToolTip = () => { + this.clearAnimationTimeout(); if (this._isMounted) { - this.setState({ visible: false }); + enqueueStateChange(() => this.setState({ visible: false })); } }; @@ -280,7 +292,6 @@ export class EuiToolTip extends Component { const classes = classNames( 'euiToolTip', positionsToClassNameMap[this.state.calculatedPosition], - delayToClassNameMap[delay], className ); diff --git a/src/components/tool_tip/tool_tip_popover.tsx b/src/components/tool_tip/tool_tip_popover.tsx index 663c9bb31d4..bc50e04661f 100644 --- a/src/components/tool_tip/tool_tip_popover.tsx +++ b/src/components/tool_tip/tool_tip_popover.tsx @@ -23,7 +23,7 @@ import { CommonProps } from '../common'; type Props = CommonProps & Omit, 'title'> & { - positionToolTip: (rect: ClientRect | DOMRect) => void; + positionToolTip: () => void; children?: ReactNode; title?: ReactNode; popoverRef?: (ref: HTMLDivElement) => void; @@ -36,7 +36,7 @@ export class EuiToolTipPopover extends Component { requestAnimationFrame(() => { // Because of this delay, sometimes `positionToolTip` becomes unavailable. if (this.popover) { - this.props.positionToolTip(this.popover.getBoundingClientRect()); + this.props.positionToolTip(); } }); }; @@ -50,8 +50,6 @@ export class EuiToolTipPopover extends Component { componentDidMount() { document.body.classList.add('euiBody-hasPortalContent'); - - this.updateDimensions(); window.addEventListener('resize', this.updateDimensions); } diff --git a/src/global_styling/mixins/_tool_tip.scss b/src/global_styling/mixins/_tool_tip.scss index 710f58fe1b4..665b3bf0fec 100644 --- a/src/global_styling/mixins/_tool_tip.scss +++ b/src/global_styling/mixins/_tool_tip.scss @@ -23,6 +23,6 @@ margin-bottom: $euiSizeXS; } -@mixin euiToolTipAnimation($side: 'top', $delay: $euiAnimSpeedNormal) { - animation: #{map-get($euiTooltipAnimations, $side)} $euiAnimSpeedSlow ease-out $euiAnimSpeedNormal forwards; +@mixin euiToolTipAnimation($side: 'top') { + animation: #{map-get($euiTooltipAnimations, $side)} $euiAnimSpeedSlow ease-out 0s forwards; }