Skip to content

Commit

Permalink
feat: initial implementation #2025
Browse files Browse the repository at this point in the history
  • Loading branch information
marek-mihok committed Jul 31, 2023
1 parent e4fa263 commit 376383f
Show file tree
Hide file tree
Showing 2 changed files with 81 additions and 40 deletions.
101 changes: 63 additions & 38 deletions ui/src/copyable_text.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,17 +6,21 @@ import { clas, cssVar, pc } from './theme'

const
css = stylesheet({
btnMultiline: {
animate: {
opacity: 0,
transition: 'opacity .5s'
},
visible: {
opacity: 1,
},
btn: {
minWidth: 'initial',
position: 'absolute',
position: 'fixed',
width: 24,
height: 24,
right: 0,
transform: 'translate(-4px, 4px)',
// TODO: Compute dynamically based on anchor element.
// right: 0,
// transform: 'translate(-4px, 4px)',
zIndex: 1,
},
copiedBtn: {
Expand Down Expand Up @@ -54,19 +58,17 @@ export interface CopyableText {
height?: S
}

export const XCopyableText = ({ model }: { model: CopyableText }) => {
export const ClipboardCopyButton = ({ value, anchorElementRef, showOnHover }: { value: S, anchorElementRef: any, showOnHover: B }) => {
const
{ name, multiline, label, value, height } = model,
heightStyle = multiline && height === '1' ? fullHeightStyle : undefined,
ref = React.useRef<Fluent.ITextField>(null),
timeoutRef = React.useRef<U>(),
[copied, setCopied] = React.useState(false),
[visible, setVisible] = React.useState(!showOnHover),
onClick = async () => {
const el = ref.current
const el = anchorElementRef.current
if (!el) return
try {
if (document.queryCommandSupported('copy')) {
el.select()
el.select() // TODO: Test, replace with componentRef
document.execCommand('copy')
window.getSelection()?.removeAllRanges()
}
Expand All @@ -78,36 +80,59 @@ export const XCopyableText = ({ model }: { model: CopyableText }) => {
timeoutRef.current = window.setTimeout(() => setCopied(false), 2000)
}

React.useEffect(() => {
const el = anchorElementRef.current
if (el) {
console.log(anchorElementRef.current.getBoundingClientRect())
// textFieldMultiline: multiline ? { '&:hover button': { opacity: 1 } } : undefined // TODO: Re-implement with pure CSS: '&hover childId'
el.addEventListener('mouseenter', () => { setVisible(true) })
el.addEventListener('mouseleave', () => { setVisible(false) })
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [])

React.useEffect(() => () => window.clearTimeout(timeoutRef.current), [])

return (<Fluent.PrimaryButton
title='Copy to clipboard'
onClick={onClick}
iconProps={{ iconName: copied ? 'CheckMark' : 'Copy' }}
className={clas(css.btn, copied ? css.copiedBtn : '', showOnHover ? css.animate : '', visible ? css.visible : '')}
/>)
}

export const XCopyableText = ({ model }: { model: CopyableText }) => {
const
{ name, multiline, label, value, height } = model,
heightStyle = multiline && height === '1' ? fullHeightStyle : undefined,
ref = React.useRef<Fluent.ITextField>(null),
domRef = React.useRef<HTMLDivElement>(null)

return (
<Fluent.TextField
data-test={name}
componentRef={ref}
value={value}
multiline={multiline}
onRenderLabel={() =>
<div className={css.labelContainer}>
<Fluent.Label>{label}</Fluent.Label>
<Fluent.PrimaryButton
title='Copy to clipboard'
onClick={onClick}
iconProps={{ iconName: copied ? 'CheckMark' : 'Copy' }}
className={clas(css.btn, copied ? css.copiedBtn : '', multiline ? css.btnMultiline : '')}
/>
</div>
}
styles={{
root: {
...heightStyle,
textFieldRoot: { position: 'relative', width: pc(100) },
textFieldMultiline: multiline ? { '&:hover button': { opacity: 1 } } : undefined
},
wrapper: heightStyle,
fieldGroup: heightStyle || { minHeight: height },
field: { ...heightStyle, height, resize: multiline ? 'vertical' : 'none', },
}}
readOnly
/>
<>
<Fluent.TextField
data-test={name}
componentRef={ref}
elementRef={domRef} // Temporary solution which will be replaced with ref once TextField is converted to a function component.
value={value}
multiline={multiline}
onRenderLabel={() =>
<div className={css.labelContainer}>
<Fluent.Label>{label}</Fluent.Label>
</div>
}
styles={{
root: {
...heightStyle,
textFieldRoot: { position: 'relative', width: pc(100) },
},
wrapper: heightStyle,
fieldGroup: heightStyle || { minHeight: height },
field: { ...heightStyle, height, resize: multiline ? 'vertical' : 'none', },
}}
readOnly
/>
<ClipboardCopyButton anchorElementRef={domRef} showOnHover={!!multiline} value={value} />
</>
)
}
20 changes: 18 additions & 2 deletions ui/src/markdown.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,8 @@ import { stylesheet } from 'typestyle'
import { cards, grid, substitute } from './layout'
import { border, clas, cssVar, padding, pc } from './theme'
import { bond } from './ui'
import { ClipboardCopyButton } from './copyable_text'
import ReactDOM from 'react-dom'

const
css = stylesheet({
Expand Down Expand Up @@ -89,7 +91,16 @@ const highlightSyntax = async (str: S, language: S, codeBlockId: S) => {
const highlightedCode = language
? hljs.highlight(str, { language, ignoreIllegals: true }).value
: hljs.highlightAuto(str).value
codeBlock.outerHTML = `<code class="hljs">${highlightedCode}</code>`
codeBlock.outerHTML = `<code id="xxxid" class="hljs">${highlightedCode}</code>`

// TODO: Use React to render the copy button.
const cb = document.getElementById('xxxid')

const copyButton = document.createElement("div")
if (copyButton && cb) {
ReactDOM.render(<ClipboardCopyButton value={str} anchorElementRef={cb} showOnHover />, copyButton)
cb.innerHTML += copyButton.innerHTML
}
}

export const
Expand All @@ -103,6 +114,7 @@ export const
}
}),
Markdown = ({ source }: { source: S }) => {
const componentRef = React.useRef<HTMLDivElement>(null)
const onClick = (e: React.MouseEvent<HTMLDivElement>) => {
const hrefAttr = (e.target as HTMLAnchorElement).getAttribute('href')
if (e.target instanceof HTMLAnchorElement && hrefAttr?.startsWith('?')) {
Expand All @@ -114,7 +126,11 @@ export const
return false
}
}
return <div onClick={onClick} className={clas(css.markdown, 'wave-markdown')} dangerouslySetInnerHTML={{ __html: markdown.render(source) }} />
return <>
<div ref={componentRef} onClick={onClick} className={clas(css.markdown, 'wave-markdown')} dangerouslySetInnerHTML={{ __html: markdown.render(source) }} />
{/* TODO: Fix anchoring. */}
<ClipboardCopyButton value={'aaa'} anchorElementRef={componentRef} showOnHover />
</>
}

/**
Expand Down

0 comments on commit 376383f

Please sign in to comment.