From fff09ab03be1e125e1ed0ef270d7340692a8ded4 Mon Sep 17 00:00:00 2001 From: Marek Mihok Date: Thu, 17 Aug 2023 11:20:15 +0200 Subject: [PATCH] chore: use text input as anchor element, refactor and cleanup, ref working only on second render #2025 --- ui/src/copyable_text.tsx | 44 +++++++++++++++++++--------------------- 1 file changed, 21 insertions(+), 23 deletions(-) diff --git a/ui/src/copyable_text.tsx b/ui/src/copyable_text.tsx index 591b35fb6a1..4ba6018c1d4 100644 --- a/ui/src/copyable_text.tsx +++ b/ui/src/copyable_text.tsx @@ -7,7 +7,7 @@ import { clas, cssVar, pc } from './theme' const BUTTON_HEIGHT = 24, BUTTON_WIDTH = 34, - CORNER_OFFSET = 6 + CORNER_OFFSET = 4 const css = stylesheet({ @@ -33,9 +33,6 @@ const } } }, - labelContainer: { - position: 'relative' - } }), fullHeightStyle = { display: 'flex', @@ -60,18 +57,19 @@ export interface CopyableText { height?: S } -export const ClipboardCopyButton = ({ value, anchorElementRef, showOnHover }: { value: S, anchorElementRef: any, showOnHover: B }) => { +type CopyButton = { value: S, anchorElement: HTMLElement | undefined, showOnHoverOnly?: B } + +export const ClipboardCopyButton = ({ value, anchorElement, showOnHoverOnly = false }: CopyButton) => { const timeoutRef = React.useRef(), [copied, setCopied] = React.useState(false), - [visible, setVisible] = React.useState(!showOnHover), + [visible, setVisible] = React.useState(!showOnHoverOnly), [position, setPosition] = React.useState({ x: 0, y: 0 }), onClick = async () => { - const el = anchorElementRef.current - if (!el) return + if (!anchorElement) return try { - if (document.queryCommandSupported('copy')) { - el.select() // TODO: Test, replace with componentRef + if (document.queryCommandSupported('copy') && anchorElement.tagName === 'input') { + (anchorElement as HTMLInputElement).select() document.execCommand('copy') window.getSelection()?.removeAllRanges() } @@ -84,12 +82,11 @@ export const ClipboardCopyButton = ({ value, anchorElementRef, showOnHover }: { } React.useEffect(() => { - const el = anchorElementRef.current - if (el) { - const rect = anchorElementRef.current.getBoundingClientRect() + if (anchorElement) { + const rect = anchorElement.getBoundingClientRect() setPosition({ x: rect.left + rect.width, y: rect.top }) - el.addEventListener('mouseenter', () => setVisible(true)) - el.addEventListener('mouseleave', (ev: MouseEvent) => { + anchorElement.addEventListener('mouseenter', () => setVisible(true)) + anchorElement.addEventListener('mouseleave', (ev: MouseEvent) => { if ((ev.relatedTarget as HTMLElement)?.id === 'copybutton') return setVisible(false) }) @@ -105,7 +102,7 @@ export const ClipboardCopyButton = ({ value, anchorElementRef, showOnHover }: { onClick={onClick} iconProps={{ iconName: copied ? 'CheckMark' : 'Copy' }} style={{ left: position.x - BUTTON_WIDTH - CORNER_OFFSET, top: position.y + CORNER_OFFSET }} - className={clas(css.btn, copied ? css.copiedBtn : '', showOnHover ? css.animate : '', visible ? css.visible : '')} + className={clas(css.btn, copied ? css.copiedBtn : '', showOnHoverOnly ? css.animate : '', visible ? css.visible : '')} />) } @@ -121,14 +118,11 @@ export const XCopyableText = ({ model }: { model: CopyableText }) => { -
- {label} -
- } + label={label} styles={{ root: { ...heightStyle, @@ -140,7 +134,11 @@ export const XCopyableText = ({ model }: { model: CopyableText }) => { }} readOnly /> - + ) } \ No newline at end of file