From 2e09c89a6b7767bde561fb8fad72b3f4f2c5b734 Mon Sep 17 00:00:00 2001 From: Emily Xiong Date: Fri, 14 Jun 2024 14:48:04 -0400 Subject: [PATCH] feat(graph): add copy button for entire target configuration --- .../ui-code-block/src/lib/json-code-block.tsx | 47 ++++---------- graph/ui-components/src/index.ts | 1 + .../lib/copy-to-clipboard-button.stories.tsx | 20 ++++++ .../src/lib/copy-to-clipboard-button.tsx | 61 +++++++++++++++++++ .../copy-to-clipboard.stories.tsx | 17 ------ .../copy-to-clipboard/copy-to-clipboard.tsx | 36 ----------- .../lib/project-details/project-details.tsx | 35 +++++++---- .../target-configuration-details-header.tsx | 19 +++--- .../target-configuration-details.tsx | 60 +++++++----------- .../target-executor/target-executor-title.tsx | 28 +++++---- package.json | 2 +- pnpm-lock.yaml | 2 +- 12 files changed, 167 insertions(+), 161 deletions(-) create mode 100644 graph/ui-components/src/lib/copy-to-clipboard-button.stories.tsx create mode 100644 graph/ui-components/src/lib/copy-to-clipboard-button.tsx delete mode 100644 graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.stories.tsx delete mode 100644 graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.tsx diff --git a/graph/ui-code-block/src/lib/json-code-block.tsx b/graph/ui-code-block/src/lib/json-code-block.tsx index 1142025c501570..4f7c7c5077c98b 100644 --- a/graph/ui-code-block/src/lib/json-code-block.tsx +++ b/graph/ui-code-block/src/lib/json-code-block.tsx @@ -1,13 +1,8 @@ -import { - ClipboardDocumentCheckIcon, - ClipboardDocumentIcon, -} from '@heroicons/react/24/outline'; -// @ts-ignore -import { CopyToClipboard } from 'react-copy-to-clipboard'; // @ts-ignore import SyntaxHighlighter, { createElement } from 'react-syntax-highlighter'; -import { JSX, ReactNode, useEffect, useMemo, useState } from 'react'; +import { JSX, ReactNode, useMemo } from 'react'; import { twMerge } from 'tailwind-merge'; +import { CopyToClipboardButton } from '@nx/graph/ui-components'; export function JsonCodeBlockPreTag({ children, @@ -30,45 +25,27 @@ export function JsonCodeBlockPreTag({ export interface JsonCodeBlockProps { data: any; renderSource: (propertyName: string) => ReactNode; + copyTooltipText: string; } export function JsonCodeBlock(props: JsonCodeBlockProps): JSX.Element { - const [copied, setCopied] = useState(false); const jsonString = useMemo( () => JSON.stringify(props.data, null, 2), [props.data] ); - useEffect(() => { - if (!copied) return; - const t = setTimeout(() => { - setCopied(false); - }, 3000); - return () => clearTimeout(t); - }, [copied]); return (
- { - setCopied(true); - }} - > - - + tooltipAlignment="right" + tooltipText={props.copyTooltipText} + className={twMerge( + 'not-prose flex', + 'border border-slate-200 bg-slate-50/50 p-2 dark:border-slate-700 dark:bg-slate-800/60', + 'opacity-0 transition-opacity group-hover:opacity-100' + )} + />
= { + component: CopyToClipboardButton, + title: 'CopyToClipboardButton', +}; +export default meta; + +type Story = StoryObj; + +export const Simple: Story = { + args: { + text: 'Hello, world!', + tooltipAlignment: 'left', + } as CopyToClipboardButtonProps, +}; diff --git a/graph/ui-components/src/lib/copy-to-clipboard-button.tsx b/graph/ui-components/src/lib/copy-to-clipboard-button.tsx new file mode 100644 index 00000000000000..0f7b6675084b01 --- /dev/null +++ b/graph/ui-components/src/lib/copy-to-clipboard-button.tsx @@ -0,0 +1,61 @@ +// @ts-ignore +import { CopyToClipboard } from 'react-copy-to-clipboard'; +import { JSX, ReactNode, useEffect, useState } from 'react'; +import { + ClipboardDocumentCheckIcon, + ClipboardDocumentIcon, +} from '@heroicons/react/24/outline'; + +export interface CopyToClipboardButtonProps { + text: string; + tooltipText?: string; + tooltipAlignment?: 'left' | 'right'; + className?: string; + children?: ReactNode; +} + +export function CopyToClipboardButton({ + text, + tooltipAlignment, + tooltipText, + className, + children, +}: CopyToClipboardButtonProps) { + const [copied, setCopied] = useState(false); + + useEffect(() => { + if (!copied) return; + const t = setTimeout(() => { + setCopied(false); + }, 3000); + return () => clearTimeout(t); + }, [copied]); + + return ( + { + setCopied(true); + }} + > + + + ); +} diff --git a/graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.stories.tsx b/graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.stories.tsx deleted file mode 100644 index 8101be70cfbea3..00000000000000 --- a/graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.stories.tsx +++ /dev/null @@ -1,17 +0,0 @@ -import type { Meta, StoryObj } from '@storybook/react'; -import { CopyToClipboard } from './copy-to-clipboard'; - -const meta: Meta = { - component: CopyToClipboard, - title: 'CopyToClipboard', -}; -export default meta; - -type Story = StoryObj; - -export const Simple: Story = { - args: { - onCopy: () => {}, - tooltipAlignment: 'left', - }, -}; diff --git a/graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.tsx b/graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.tsx deleted file mode 100644 index e7cbfdd7ffc38e..00000000000000 --- a/graph/ui-project-details/src/lib/copy-to-clipboard/copy-to-clipboard.tsx +++ /dev/null @@ -1,36 +0,0 @@ -import { ClipboardIcon } from '@heroicons/react/24/outline'; -import { JSX, useEffect, useState } from 'react'; - -interface CopyToClipboardProps { - onCopy: () => void; - tooltipAlignment?: 'left' | 'right'; -} - -export function CopyToClipboard(props: CopyToClipboardProps): JSX.Element { - const [copied, setCopied] = useState(false); - useEffect(() => { - if (copied) { - const timeout = setTimeout(() => { - setCopied(false); - }, 3000); - return () => clearTimeout(timeout); - } - }); - return ( - - { - e.stopPropagation(); - setCopied(true); - props.onCopy(); - }} - > - - ); -} diff --git a/graph/ui-project-details/src/lib/project-details/project-details.tsx b/graph/ui-project-details/src/lib/project-details/project-details.tsx index dd2b4cc532d8c6..62e5a604069430 100644 --- a/graph/ui-project-details/src/lib/project-details/project-details.tsx +++ b/graph/ui-project-details/src/lib/project-details/project-details.tsx @@ -8,6 +8,7 @@ import { GraphError } from 'nx/src/command-line/graph/graph'; /* eslint-enable @nx/enforce-module-boundaries */ import { EyeIcon } from '@heroicons/react/24/outline'; import { PropertyInfoTooltip, Tooltip } from '@nx/graph/ui-tooltips'; +import { CopyToClipboardButton } from '@nx/graph/ui-components'; import { TooltipTriggerText } from '../target-configuration-details/tooltip-trigger-text'; import { twMerge } from 'tailwind-merge'; import { Pill } from '../pill'; @@ -67,7 +68,7 @@ export const ProjectDetails = ({ >
@@ -86,17 +87,23 @@ export const ProjectDetails = ({ className="h-6 w-6" />
- - {onViewInProjectGraph && viewInProjectGraphPosition === 'top' && ( + {onViewInProjectGraph && viewInProjectGraphPosition === 'top' && ( +
+ + Copy Project + onViewInProjectGraph({ projectName: project.name }) } /> - )}{' '} - +
+ )}
-
+
{projectData.metadata?.description ? (

@@ -129,16 +136,22 @@ export const ProjectDetails = ({ ) : null}

- - {onViewInProjectGraph && - viewInProjectGraphPosition === 'bottom' && ( + {onViewInProjectGraph && + viewInProjectGraphPosition === 'bottom' && ( +
+ + Copy Project + onViewInProjectGraph({ projectName: project.name }) } /> - )}{' '} - +
+ )}
diff --git a/graph/ui-project-details/src/lib/target-configuration-details-header/target-configuration-details-header.tsx b/graph/ui-project-details/src/lib/target-configuration-details-header/target-configuration-details-header.tsx index 9fe0174e60388d..0a5aaa3a287551 100644 --- a/graph/ui-project-details/src/lib/target-configuration-details-header/target-configuration-details-header.tsx +++ b/graph/ui-project-details/src/lib/target-configuration-details-header/target-configuration-details-header.tsx @@ -1,6 +1,7 @@ /* eslint-disable @nx/enforce-module-boundaries */ // nx-ignore-next-line import type { TargetConfiguration } from '@nx/devkit'; +import { CopyToClipboardButton } from '@nx/graph/ui-components'; import { ChevronDownIcon, ChevronUpIcon, @@ -13,7 +14,6 @@ import { twMerge } from 'tailwind-merge'; import { Pill } from '../pill'; import { TargetTechnologies } from '../target-technologies/target-technologies'; import { SourceInfo } from '../source-info/source-info'; -import { CopyToClipboard } from '../copy-to-clipboard/copy-to-clipboard'; import { getDisplayHeaderFromTargetConfiguration } from '../utils/get-display-header-from-target-configuration'; import { TargetExecutor } from '../target-executor/target-executor'; @@ -45,10 +45,6 @@ export const TargetConfigurationDetailsHeader = ({ onRunTarget, onViewInTaskGraph, }: TargetConfigurationDetailsHeaderProps) => { - const handleCopyClick = async (copyText: string) => { - await window.navigator.clipboard.writeText(copyText); - }; - if (!collapsable) { // when collapsable is false, isCollasped should be false isCollasped = false; @@ -123,6 +119,12 @@ export const TargetConfigurationDetailsHeader = ({
+ {onViewInTaskGraph && (