From 6231f6b6e45425e7a09fecc31b540537a7ce86ae Mon Sep 17 00:00:00 2001 From: Diego Medina Date: Fri, 13 May 2022 10:14:30 -0300 Subject: [PATCH 1/4] fix: Support the Clipboard API in modern browsers --- .../src/components/CopyToClipboard/index.jsx | 6 +- .../components/menu/ShareMenuItems/index.tsx | 3 +- .../useExploreAdditionalActionsMenu/index.jsx | 3 +- superset-frontend/src/utils/copy.ts | 93 +++++++++++++------ .../SyntaxHighlighterCopy/index.tsx | 2 +- .../CRUD/data/savedquery/SavedQueryList.tsx | 6 +- superset-frontend/src/views/CRUD/hooks.ts | 6 +- 7 files changed, 77 insertions(+), 42 deletions(-) diff --git a/superset-frontend/src/components/CopyToClipboard/index.jsx b/superset-frontend/src/components/CopyToClipboard/index.jsx index 2047cf4b0fbdb..95b6cdfdc0f70 100644 --- a/superset-frontend/src/components/CopyToClipboard/index.jsx +++ b/superset-frontend/src/components/CopyToClipboard/index.jsx @@ -57,10 +57,10 @@ class CopyToClipboard extends React.Component { onClick() { if (this.props.getText) { this.props.getText(d => { - this.copyToClipboard(d); + this.copyToClipboard(Promise.resolve(d)); }); } else { - this.copyToClipboard(this.props.text); + this.copyToClipboard(Promise.resolve(this.props.text)); } } @@ -72,7 +72,7 @@ class CopyToClipboard extends React.Component { } copyToClipboard(textToCopy) { - copyTextToClipboard(textToCopy) + copyTextToClipboard(() => textToCopy) .then(() => { this.props.addSuccessToast(t('Copied to clipboard!')); }) diff --git a/superset-frontend/src/dashboard/components/menu/ShareMenuItems/index.tsx b/superset-frontend/src/dashboard/components/menu/ShareMenuItems/index.tsx index 92e5665aa01b5..f9016e5263c02 100644 --- a/superset-frontend/src/dashboard/components/menu/ShareMenuItems/index.tsx +++ b/superset-frontend/src/dashboard/components/menu/ShareMenuItems/index.tsx @@ -64,8 +64,7 @@ const ShareMenuItems = (props: ShareMenuItemProps) => { async function onCopyLink() { try { - const url = await generateUrl(); - await copyTextToClipboard(url); + await copyTextToClipboard(generateUrl); addSuccessToast(t('Copied to clipboard!')); } catch (error) { logging.error(error); diff --git a/superset-frontend/src/explore/components/useExploreAdditionalActionsMenu/index.jsx b/superset-frontend/src/explore/components/useExploreAdditionalActionsMenu/index.jsx index 652347326778f..bebe5be269774 100644 --- a/superset-frontend/src/explore/components/useExploreAdditionalActionsMenu/index.jsx +++ b/superset-frontend/src/explore/components/useExploreAdditionalActionsMenu/index.jsx @@ -168,8 +168,7 @@ export const useExploreAdditionalActionsMenu = ( if (!latestQueryFormData) { throw new Error(); } - const url = await getChartPermalink(latestQueryFormData); - await copyTextToClipboard(url); + await copyTextToClipboard(() => getChartPermalink(latestQueryFormData)); addSuccessToast(t('Copied to clipboard!')); } catch (error) { addDangerToast(t('Sorry, something went wrong. Try again later.')); diff --git a/superset-frontend/src/utils/copy.ts b/superset-frontend/src/utils/copy.ts index 7db289c04037d..1aa1aca8a26ec 100644 --- a/superset-frontend/src/utils/copy.ts +++ b/superset-frontend/src/utils/copy.ts @@ -17,40 +17,73 @@ * under the License. */ -const copyTextToClipboard = async (text: string) => - new Promise((resolve, reject) => { - const selection: Selection | null = document.getSelection(); - if (selection) { - selection.removeAllRanges(); - const range = document.createRange(); - const span = document.createElement('span'); - span.textContent = text; - span.style.position = 'fixed'; - span.style.top = '0'; - span.style.clip = 'rect(0, 0, 0, 0)'; - span.style.whiteSpace = 'pre'; - - document.body.appendChild(span); - range.selectNode(span); - selection.addRange(range); - - try { - if (!document.execCommand('copy')) { +// Use the new Clipboard API if the browser supports it +const copyTextWithClipboardApi = async (getText: () => Promise) => { + try { + // Safari (WebKit) does not support delayed generation of clipboard. + // This means that writing to the clipboard, from the moment the user + // interacts with the app, must be instantaneous. + // However, neither writeText nor write accepts a Promise, so + // we need to create a ClipboardItem that accepts said Promise to + // delay the text generation, as needed. + // Source: https://bugs.webkit.org/show_bug.cgi?id=222262 + const clipboardItem = new ClipboardItem({ + 'text/plain': getText(), + }); + await navigator.clipboard.write([clipboardItem]); + } catch { + // For Blink, the above method won't work, but we can use the + // default (intended) API, since the delayed generation of the + // clipboard is now supported. + // Source: https://bugs.chromium.org/p/chromium/issues/detail?id=1014310 + const text = await getText(); + await navigator.clipboard.writeText(text); + } +}; + +const copyTextToClipboard = async (getText: () => Promise) => { + try { + await copyTextWithClipboardApi(getText); + } catch { + // If the Clipboard API is not supported, fallback to the older method. + const text = await getText(); + const copyPromise = new Promise((resolve, reject) => { + const selection: Selection | null = document.getSelection(); + if (selection) { + selection.removeAllRanges(); + const range = document.createRange(); + const span = document.createElement('span'); + span.textContent = text; + span.style.position = 'fixed'; + span.style.top = '0'; + span.style.clip = 'rect(0, 0, 0, 0)'; + span.style.whiteSpace = 'pre'; + + document.body.appendChild(span); + range.selectNode(span); + selection.addRange(range); + + try { + if (!document.execCommand('copy')) { + reject(); + } + } catch (err) { reject(); } - } catch (err) { - reject(); - } - document.body.removeChild(span); - if (selection.removeRange) { - selection.removeRange(range); - } else { - selection.removeAllRanges(); + document.body.removeChild(span); + if (selection.removeRange) { + selection.removeRange(range); + } else { + selection.removeAllRanges(); + } } - } - resolve(); - }); + resolve(); + }); + + await copyPromise; + } +}; export default copyTextToClipboard; diff --git a/superset-frontend/src/views/CRUD/data/components/SyntaxHighlighterCopy/index.tsx b/superset-frontend/src/views/CRUD/data/components/SyntaxHighlighterCopy/index.tsx index 73a0f8e9cc080..8e96b95f0dc84 100644 --- a/superset-frontend/src/views/CRUD/data/components/SyntaxHighlighterCopy/index.tsx +++ b/superset-frontend/src/views/CRUD/data/components/SyntaxHighlighterCopy/index.tsx @@ -65,7 +65,7 @@ export default function SyntaxHighlighterCopy({ language: 'sql' | 'markdown' | 'html' | 'json'; }) { function copyToClipboard(textToCopy: string) { - copyTextToClipboard(textToCopy) + copyTextToClipboard(() => Promise.resolve(textToCopy)) .then(() => { if (addSuccessToast) { addSuccessToast(t('SQL Copied!')); diff --git a/superset-frontend/src/views/CRUD/data/savedquery/SavedQueryList.tsx b/superset-frontend/src/views/CRUD/data/savedquery/SavedQueryList.tsx index df3f16a858c13..d2dc6aff9c3f7 100644 --- a/superset-frontend/src/views/CRUD/data/savedquery/SavedQueryList.tsx +++ b/superset-frontend/src/views/CRUD/data/savedquery/SavedQueryList.tsx @@ -210,8 +210,10 @@ function SavedQueryList({ const copyQueryLink = useCallback( (id: number) => { - copyTextToClipboard( - `${window.location.origin}/superset/sqllab?savedQueryId=${id}`, + copyTextToClipboard(() => + Promise.resolve( + `${window.location.origin}/superset/sqllab?savedQueryId=${id}`, + ), ) .then(() => { addSuccessToast(t('Link Copied!')); diff --git a/superset-frontend/src/views/CRUD/hooks.ts b/superset-frontend/src/views/CRUD/hooks.ts index 18349b305c212..ed49da1e8cfec 100644 --- a/superset-frontend/src/views/CRUD/hooks.ts +++ b/superset-frontend/src/views/CRUD/hooks.ts @@ -611,8 +611,10 @@ export const copyQueryLink = ( addDangerToast: (arg0: string) => void, addSuccessToast: (arg0: string) => void, ) => { - copyTextToClipboard( - `${window.location.origin}/superset/sqllab?savedQueryId=${id}`, + copyTextToClipboard(() => + Promise.resolve( + `${window.location.origin}/superset/sqllab?savedQueryId=${id}`, + ), ) .then(() => { addSuccessToast(t('Link Copied!')); From a545d7e101421a99f1f732f4f57c14401e68969d Mon Sep 17 00:00:00 2001 From: Diego Medina Date: Fri, 13 May 2022 16:10:11 -0300 Subject: [PATCH 2/4] fix tests --- .../ShareMenuItems/ShareMenuItems.test.tsx | 10 +++++---- .../CopyToClipboardButton.test.tsx | 22 ++++++++++++++----- .../DataTablesPane/DataTablesPane.test.tsx | 6 ++--- superset-frontend/src/utils/common.js | 2 +- 4 files changed, 27 insertions(+), 13 deletions(-) diff --git a/superset-frontend/src/dashboard/components/menu/ShareMenuItems/ShareMenuItems.test.tsx b/superset-frontend/src/dashboard/components/menu/ShareMenuItems/ShareMenuItems.test.tsx index 579f9d4b69077..498009224a5e7 100644 --- a/superset-frontend/src/dashboard/components/menu/ShareMenuItems/ShareMenuItems.test.tsx +++ b/superset-frontend/src/dashboard/components/menu/ShareMenuItems/ShareMenuItems.test.tsx @@ -102,9 +102,10 @@ test('Click on "Copy dashboard URL" and succeed', async () => { userEvent.click(screen.getByRole('button', { name: 'Copy dashboard URL' })); - await waitFor(() => { + await waitFor(async () => { expect(spy).toBeCalledTimes(1); - expect(spy).toBeCalledWith('http://localhost/superset/dashboard/p/123/'); + const value = await spy.mock.calls[0][0](); + expect(value).toBe('http://localhost/superset/dashboard/p/123/'); expect(props.addSuccessToast).toBeCalledTimes(1); expect(props.addSuccessToast).toBeCalledWith('Copied to clipboard!'); expect(props.addDangerToast).toBeCalledTimes(0); @@ -128,9 +129,10 @@ test('Click on "Copy dashboard URL" and fail', async () => { userEvent.click(screen.getByRole('button', { name: 'Copy dashboard URL' })); - await waitFor(() => { + await waitFor(async () => { expect(spy).toBeCalledTimes(1); - expect(spy).toBeCalledWith('http://localhost/superset/dashboard/p/123/'); + const value = await spy.mock.calls[0][0](); + expect(value).toBe('http://localhost/superset/dashboard/p/123/'); expect(props.addSuccessToast).toBeCalledTimes(0); expect(props.addDangerToast).toBeCalledTimes(1); expect(props.addDangerToast).toBeCalledWith( diff --git a/superset-frontend/src/explore/components/DataTableControl/CopyToClipboardButton.test.tsx b/superset-frontend/src/explore/components/DataTableControl/CopyToClipboardButton.test.tsx index 2ce91590b9890..a158bfd7ed518 100644 --- a/superset-frontend/src/explore/components/DataTableControl/CopyToClipboardButton.test.tsx +++ b/superset-frontend/src/explore/components/DataTableControl/CopyToClipboardButton.test.tsx @@ -18,7 +18,7 @@ */ import userEvent from '@testing-library/user-event'; import React from 'react'; -import { render, screen } from 'spec/helpers/testing-library'; +import { render, screen, waitFor } from 'spec/helpers/testing-library'; import { CopyToClipboardButton } from '.'; test('Render a button', () => { @@ -28,14 +28,26 @@ test('Render a button', () => { expect(screen.getByRole('button')).toBeInTheDocument(); }); -test('Should copy to clipboard', () => { - document.execCommand = jest.fn(); +test('Should copy to clipboard', async () => { + const callback = jest.fn(); + document.execCommand = callback; + + const originalClipboard = { ...global.navigator.clipboard }; + // @ts-ignore + global.navigator.clipboard = { write: callback, writeText: callback }; render(, { useRedux: true, }); - expect(document.execCommand).toHaveBeenCalledTimes(0); + expect(callback).toHaveBeenCalledTimes(0); userEvent.click(screen.getByRole('button')); - expect(document.execCommand).toHaveBeenCalledWith('copy'); + + await waitFor(() => { + expect(callback).toHaveBeenCalled(); + }); + + jest.resetAllMocks(); + // @ts-ignore + global.navigator.clipboard = originalClipboard; }); diff --git a/superset-frontend/src/explore/components/DataTablesPane/DataTablesPane.test.tsx b/superset-frontend/src/explore/components/DataTablesPane/DataTablesPane.test.tsx index 76d4b6951d8ca..57d599ee82b9a 100644 --- a/superset-frontend/src/explore/components/DataTablesPane/DataTablesPane.test.tsx +++ b/superset-frontend/src/explore/components/DataTablesPane/DataTablesPane.test.tsx @@ -175,9 +175,9 @@ describe('DataTablesPane', () => { expect(await screen.findByText('1 row')).toBeVisible(); userEvent.click(screen.getByLabelText('Copy')); - expect(copyToClipboardSpy).toHaveBeenCalledWith( - '2009-01-01 00:00:00\tAction\n', - ); + expect(copyToClipboardSpy).toHaveBeenCalledTimes(1); + const value = await copyToClipboardSpy.mock.calls[0][0](); + expect(value).toBe('2009-01-01 00:00:00\tAction\n'); copyToClipboardSpy.mockRestore(); fetchMock.restore(); }); diff --git a/superset-frontend/src/utils/common.js b/superset-frontend/src/utils/common.js index 4efdb205e5a9f..9c98739d225c4 100644 --- a/superset-frontend/src/utils/common.js +++ b/superset-frontend/src/utils/common.js @@ -94,7 +94,7 @@ export function prepareCopyToClipboardTabularData(data, columns) { for (let i = 0; i < data.length; i += 1) { const row = {}; for (let j = 0; j < columns.length; j += 1) { - // JavaScript does not mantain the order of a mixed set of keys (i.e integers and strings) + // JavaScript does not maintain the order of a mixed set of keys (i.e integers and strings) // the below function orders the keys based on the column names. const key = columns[j].name || columns[j]; if (data[i][key]) { From c8f4de98ccae139ca028772bcce6593e2e8bca2a Mon Sep 17 00:00:00 2001 From: Diego Medina Date: Mon, 16 May 2022 16:54:32 -0300 Subject: [PATCH 3/4] PR comment --- superset-frontend/src/utils/copy.ts | 76 ++++++++++++++--------------- 1 file changed, 37 insertions(+), 39 deletions(-) diff --git a/superset-frontend/src/utils/copy.ts b/superset-frontend/src/utils/copy.ts index 1aa1aca8a26ec..f6895c8a7f339 100644 --- a/superset-frontend/src/utils/copy.ts +++ b/superset-frontend/src/utils/copy.ts @@ -41,49 +41,47 @@ const copyTextWithClipboardApi = async (getText: () => Promise) => { } }; -const copyTextToClipboard = async (getText: () => Promise) => { - try { - await copyTextWithClipboardApi(getText); - } catch { +const copyTextToClipboard = (getText: () => Promise) => + copyTextWithClipboardApi(getText) // If the Clipboard API is not supported, fallback to the older method. - const text = await getText(); - const copyPromise = new Promise((resolve, reject) => { - const selection: Selection | null = document.getSelection(); - if (selection) { - selection.removeAllRanges(); - const range = document.createRange(); - const span = document.createElement('span'); - span.textContent = text; - span.style.position = 'fixed'; - span.style.top = '0'; - span.style.clip = 'rect(0, 0, 0, 0)'; - span.style.whiteSpace = 'pre'; + .catch(() => + getText().then( + text => + new Promise((resolve, reject) => { + const selection: Selection | null = document.getSelection(); + if (selection) { + selection.removeAllRanges(); + const range = document.createRange(); + const span = document.createElement('span'); + span.textContent = text; + span.style.position = 'fixed'; + span.style.top = '0'; + span.style.clip = 'rect(0, 0, 0, 0)'; + span.style.whiteSpace = 'pre'; - document.body.appendChild(span); - range.selectNode(span); - selection.addRange(range); + document.body.appendChild(span); + range.selectNode(span); + selection.addRange(range); - try { - if (!document.execCommand('copy')) { - reject(); - } - } catch (err) { - reject(); - } + try { + if (!document.execCommand('copy')) { + reject(); + } + } catch (err) { + reject(); + } - document.body.removeChild(span); - if (selection.removeRange) { - selection.removeRange(range); - } else { - selection.removeAllRanges(); - } - } + document.body.removeChild(span); + if (selection.removeRange) { + selection.removeRange(range); + } else { + selection.removeAllRanges(); + } + } - resolve(); - }); - - await copyPromise; - } -}; + resolve(); + }), + ), + ); export default copyTextToClipboard; From f02bf0be384c5bc849850d355456d08dd360f420 Mon Sep 17 00:00:00 2001 From: Diego Medina Date: Thu, 2 Jun 2022 13:54:31 -0300 Subject: [PATCH 4/4] Improvements --- superset-frontend/src/utils/common.js | 6 +++++ superset-frontend/src/utils/copy.ts | 34 +++++++++++++++++---------- 2 files changed, 27 insertions(+), 13 deletions(-) diff --git a/superset-frontend/src/utils/common.js b/superset-frontend/src/utils/common.js index 9c98739d225c4..603ec7c54992d 100644 --- a/superset-frontend/src/utils/common.js +++ b/superset-frontend/src/utils/common.js @@ -145,4 +145,10 @@ export const detectOS = () => { return 'Unknown OS'; }; +export const isSafari = () => { + const { userAgent } = navigator; + + return userAgent && /^((?!chrome|android).)*safari/i.test(userAgent); +}; + export const isNullish = value => value === null || value === undefined; diff --git a/superset-frontend/src/utils/copy.ts b/superset-frontend/src/utils/copy.ts index f6895c8a7f339..0980f2ab17079 100644 --- a/superset-frontend/src/utils/copy.ts +++ b/superset-frontend/src/utils/copy.ts @@ -17,21 +17,29 @@ * under the License. */ +import { isSafari } from './common'; + // Use the new Clipboard API if the browser supports it const copyTextWithClipboardApi = async (getText: () => Promise) => { - try { - // Safari (WebKit) does not support delayed generation of clipboard. - // This means that writing to the clipboard, from the moment the user - // interacts with the app, must be instantaneous. - // However, neither writeText nor write accepts a Promise, so - // we need to create a ClipboardItem that accepts said Promise to - // delay the text generation, as needed. - // Source: https://bugs.webkit.org/show_bug.cgi?id=222262 - const clipboardItem = new ClipboardItem({ - 'text/plain': getText(), - }); - await navigator.clipboard.write([clipboardItem]); - } catch { + // Safari (WebKit) does not support delayed generation of clipboard. + // This means that writing to the clipboard, from the moment the user + // interacts with the app, must be instantaneous. + // However, neither writeText nor write accepts a Promise, so + // we need to create a ClipboardItem that accepts said Promise to + // delay the text generation, as needed. + // Source: https://bugs.webkit.org/show_bug.cgi?id=222262P + if (isSafari()) { + try { + const clipboardItem = new ClipboardItem({ + 'text/plain': getText(), + }); + await navigator.clipboard.write([clipboardItem]); + } catch { + // Fallback to default clipboard API implementation + const text = await getText(); + await navigator.clipboard.writeText(text); + } + } else { // For Blink, the above method won't work, but we can use the // default (intended) API, since the delayed generation of the // clipboard is now supported.