From 02e31d0ae5d4af454ef7d765cd518eff6844fe17 Mon Sep 17 00:00:00 2001 From: Cee Chen <549407+cee-chen@users.noreply.github.com> Date: Thu, 12 Jan 2023 11:19:19 -0800 Subject: [PATCH] [EuiToolTip] Enforce only one visible tooltip at a time (#6520) * [misc cleanup] Group relative imports by general concept - keep parent services together, keep tooltip-specific imports together * Add tooltip manager that hides all other tooltips when a new tooltip is shown * Write Cypress E2E tests for multiple tooltip behavior * Changelog --- src/components/tool_tip/tool_tip.spec.tsx | 60 +++++++++++++++++++ src/components/tool_tip/tool_tip.tsx | 17 ++++-- .../tool_tip/tool_tip_manager.test.ts | 43 +++++++++++++ src/components/tool_tip/tool_tip_manager.ts | 32 ++++++++++ upcoming_changelogs/6520.md | 3 + 5 files changed, 149 insertions(+), 6 deletions(-) create mode 100644 src/components/tool_tip/tool_tip.spec.tsx create mode 100644 src/components/tool_tip/tool_tip_manager.test.ts create mode 100644 src/components/tool_tip/tool_tip_manager.ts create mode 100644 upcoming_changelogs/6520.md diff --git a/src/components/tool_tip/tool_tip.spec.tsx b/src/components/tool_tip/tool_tip.spec.tsx new file mode 100644 index 00000000000..7ad209eee70 --- /dev/null +++ b/src/components/tool_tip/tool_tip.spec.tsx @@ -0,0 +1,60 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License + * 2.0 and the Server Side Public License, v 1; you may not use this file except + * in compliance with, at your election, the Elastic License 2.0 or the Server + * Side Public License, v 1. + */ + +/// + +import React from 'react'; + +import { EuiButton } from '../../components'; +import { EuiToolTip } from './tool_tip'; + +describe('EuiToolTip', () => { + it('shows the tooltip on hover', () => { + cy.mount( + + Show tooltip + + ); + cy.get('[data-test-subj="tooltip"]').should('not.exist'); + cy.get('[data-test-subj="toggleToolTip"]').trigger('mouseover'); + cy.get('[data-test-subj="tooltip"]').should('exist'); + }); + + it('shows the tooltip on keyboard focus', () => { + cy.mount( + + Show tooltip + + ); + cy.get('[data-test-subj="tooltip"]').should('not.exist'); + cy.get('[data-test-subj="toggleToolTip"]').focus(); + cy.get('[data-test-subj="tooltip"]').should('exist'); + }); + + it('does not show multiple tooltips if one tooltip toggle is focused and another tooltip toggle is hovered', () => { + cy.mount( + <> + + Show tooltip A + + + Show tooltip B + + + ); + cy.get('[data-test-subj="tooltip"]').should('not.exist'); + + cy.get('[data-test-subj="toggleToolTipA"]').focus(); + cy.contains('Tooltip A').should('exist'); + cy.contains('Tooltip B').should('not.exist'); + + cy.get('[data-test-subj="toggleToolTipB"]').trigger('mouseover'); + cy.contains('Tooltip B').should('exist'); + cy.contains('Tooltip A').should('not.exist'); + }); +}); diff --git a/src/components/tool_tip/tool_tip.tsx b/src/components/tool_tip/tool_tip.tsx index e385271e2c2..26fbcb0e84c 100644 --- a/src/components/tool_tip/tool_tip.tsx +++ b/src/components/tool_tip/tool_tip.tsx @@ -16,14 +16,15 @@ import React, { import classNames from 'classnames'; import { CommonProps, keysOf } from '../common'; +import { findPopoverPosition, htmlIdGenerator } from '../../services'; +import { enqueueStateChange } from '../../services/react'; +import { EuiResizeObserver } from '../observer/resize_observer'; import { EuiPortal } from '../portal'; + +import { EuiToolTipPopover, ToolTipPositions } from './tool_tip_popover'; import { EuiToolTipAnchor } from './tool_tip_anchor'; import { EuiToolTipArrow } from './tool_tip_arrow'; -import { EuiToolTipPopover, ToolTipPositions } from './tool_tip_popover'; -import { enqueueStateChange } from '../../services/react'; -import { findPopoverPosition, htmlIdGenerator } from '../../services'; - -import { EuiResizeObserver } from '../observer/resize_observer'; +import { toolTipManager } from './tool_tip_manager'; const positionsToClassNameMap: { [key in ToolTipPositions]: string } = { top: 'euiToolTip--top', @@ -209,7 +210,10 @@ export class EuiToolTip extends Component { showToolTip = () => { if (!this.timeoutId) { this.timeoutId = setTimeout(() => { - enqueueStateChange(() => this.setState({ visible: true })); + enqueueStateChange(() => { + this.setState({ visible: true }); + toolTipManager.registerTooltip(this.hideToolTip); + }); }, delayToMsMap[this.props.delay]); } }; @@ -267,6 +271,7 @@ export class EuiToolTip extends Component { toolTipStyles: DEFAULT_TOOLTIP_STYLES, arrowStyles: undefined, }); + toolTipManager.deregisterToolTip(this.hideToolTip); } }); }; diff --git a/src/components/tool_tip/tool_tip_manager.test.ts b/src/components/tool_tip/tool_tip_manager.test.ts new file mode 100644 index 00000000000..2f32aa74ba2 --- /dev/null +++ b/src/components/tool_tip/tool_tip_manager.test.ts @@ -0,0 +1,43 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License + * 2.0 and the Server Side Public License, v 1; you may not use this file except + * in compliance with, at your election, the Elastic License 2.0 or the Server + * Side Public License, v 1. + */ + +import { toolTipManager } from './tool_tip_manager'; + +describe('ToolTipManager', () => { + describe('registerToolTip', () => { + const hideToolTip = jest.fn(); + + it('stores the passed hideToolTip callback', () => { + toolTipManager.registerTooltip(hideToolTip); + + expect(toolTipManager.toolTipsToHide.has(hideToolTip)).toBeTruthy(); + }); + + it('calls the previously stored hideToolTip callback and removes it from storage', () => { + toolTipManager.registerTooltip(() => {}); + + expect(hideToolTip).toHaveBeenCalledTimes(1); + expect(toolTipManager.toolTipsToHide.has(hideToolTip)).toBeFalsy(); + }); + }); + + describe('deregisterToolTip', () => { + // If the current tooltip is already hidden before the next tooltip is visible, + // there's no need to re-hide it, so we deregister the callback + const deregisteredHide = jest.fn(); + + it('removes the hide callback from storage', () => { + toolTipManager.registerTooltip(deregisteredHide); + toolTipManager.deregisterToolTip(deregisteredHide); + toolTipManager.registerTooltip(() => {}); + + expect(deregisteredHide).toHaveBeenCalledTimes(0); + expect(toolTipManager.toolTipsToHide.has(deregisteredHide)).toBeFalsy(); + }); + }); +}); diff --git a/src/components/tool_tip/tool_tip_manager.ts b/src/components/tool_tip/tool_tip_manager.ts new file mode 100644 index 00000000000..be522e71c57 --- /dev/null +++ b/src/components/tool_tip/tool_tip_manager.ts @@ -0,0 +1,32 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License + * 2.0 and the Server Side Public License, v 1; you may not use this file except + * in compliance with, at your election, the Elastic License 2.0 or the Server + * Side Public License, v 1. + */ + +/** + * Manager utility that ensures only one tooltip is visible at a time + * + * UX rationale (primarily for mouse-only users): + * @see https://github.com/elastic/kibana/issues/144482 + * @see https://github.com/elastic/eui/issues/5883 + */ +class ToolTipManager { + // We use a set instead of a single var just in case + // multiple tooltips are registered via async shenanigans + toolTipsToHide = new Set(); + + registerTooltip = (hideCallback: Function) => { + this.toolTipsToHide.forEach((hide) => hide()); + this.toolTipsToHide.clear(); + this.toolTipsToHide.add(hideCallback); + }; + + deregisterToolTip = (hideCallback: Function) => { + this.toolTipsToHide.delete(hideCallback); + }; +} + +export const toolTipManager = new ToolTipManager(); diff --git a/upcoming_changelogs/6520.md b/upcoming_changelogs/6520.md new file mode 100644 index 00000000000..22e93fe4611 --- /dev/null +++ b/upcoming_changelogs/6520.md @@ -0,0 +1,3 @@ +**Breaking changes** + +- `EuiToolTip`s now internally enforce only showing **one** tooltip at a time (the most recently triggered tooltip). This primarily affects scenarios where users are focused on a tooltip toggle via click, and then hover onto another tooltip toggle.