|
|
|
import React from 'react'
|
|
|
|
import { withRouter } from 'next/router'
|
|
|
|
import { useCopyTextHandler, useOnline, useKeyboardListener } from 'actionsack'
|
|
|
|
|
|
|
|
import { COLORS, EXPORT_SIZES } from '../lib/constants'
|
|
|
|
import Button from './Button'
|
|
|
|
import Input from './Input'
|
|
|
|
import Popout, { managePopout } from './Popout'
|
|
|
|
|
|
|
|
const toIFrame = url =>
|
|
|
|
`<iframe
|
|
|
|
src="${location.origin}/embed${url}"
|
|
|
|
style="transform:scale(0.7); width:1024px; height:473px; border:0; overflow:hidden;"
|
|
|
|
sandbox="allow-scripts allow-same-origin">
|
|
|
|
</iframe>
|
|
|
|
`
|
|
|
|
|
|
|
|
const toURL = url => encodeURI(`${location.origin}${url}`)
|
|
|
|
|
|
|
|
const MAX_PAYLOAD_SIZE = 5e6 // bytes
|
|
|
|
function verifyPayloadSize(str) {
|
|
|
|
if (typeof str !== 'string') return true
|
|
|
|
|
|
|
|
if (typeof window !== 'undefined') {
|
|
|
|
return new Blob([str]).size < MAX_PAYLOAD_SIZE
|
|
|
|
}
|
|
|
|
|
|
|
|
return Buffer.byteLength(str, 'utf8')
|
|
|
|
}
|
|
|
|
|
|
|
|
const CopyEmbed = withRouter(({ router: { asPath }, mapper, title, margin }) => {
|
|
|
|
const text = React.useMemo(() => mapper(asPath), [mapper, asPath])
|
|
|
|
const { onClick, copied } = useCopyTextHandler(text)
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
onClick={onClick}
|
|
|
|
center
|
|
|
|
hoverColor={COLORS.PURPLE}
|
|
|
|
color={COLORS.DARK_PURPLE}
|
|
|
|
margin={margin}
|
|
|
|
style={{ minWidth: 48 }}
|
|
|
|
>
|
|
|
|
{copied ? 'Copied!' : title}
|
|
|
|
</Button>
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
const popoutStyle = { width: '280px', right: 0 }
|
|
|
|
|
|
|
|
function useSafari() {
|
|
|
|
const [isSafari, setSafari] = React.useState(false)
|
|
|
|
React.useEffect(() => {
|
|
|
|
setSafari(
|
|
|
|
window.navigator &&
|
|
|
|
window.navigator.userAgent.indexOf('Safari') !== -1 &&
|
|
|
|
window.navigator.userAgent.indexOf('Chrome') === -1
|
|
|
|
)
|
|
|
|
}, [])
|
|
|
|
|
|
|
|
return isSafari
|
|
|
|
}
|
|
|
|
|
|
|
|
function ExportMenu({
|
|
|
|
backgroundImage,
|
|
|
|
onChange,
|
|
|
|
exportSize,
|
|
|
|
isVisible,
|
|
|
|
toggleVisibility,
|
|
|
|
export: exportImage
|
|
|
|
}) {
|
|
|
|
const tooLarge = React.useMemo(() => !verifyPayloadSize(backgroundImage), [backgroundImage])
|
|
|
|
const online = useOnline()
|
|
|
|
const isSafari = useSafari()
|
|
|
|
|
|
|
|
useKeyboardListener('⌘-⇧-e', () => exportImage())
|
|
|
|
|
|
|
|
const disablePNG = isSafari && (tooLarge || !online)
|
|
|
|
|
|
|
|
const input = React.useRef()
|
|
|
|
|
|
|
|
const handleExportSizeChange = selectedSize => () => onChange('exportSize', selectedSize)
|
|
|
|
|
|
|
|
const handleExport = format => () =>
|
|
|
|
exportImage(format, {
|
|
|
|
filename: input.current.value
|
|
|
|
})
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="export-menu-container" id="export-menu">
|
|
|
|
<div className="flex">
|
|
|
|
<Button
|
|
|
|
border
|
|
|
|
large
|
|
|
|
center
|
|
|
|
color={COLORS.PURPLE}
|
|
|
|
padding="0 16px"
|
|
|
|
selected={isVisible}
|
|
|
|
onClick={toggleVisibility}
|
|
|
|
data-cy="export-button"
|
|
|
|
>
|
|
|
|
Export
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
<Popout
|
|
|
|
hidden={!isVisible}
|
|
|
|
borderColor={COLORS.PURPLE}
|
|
|
|
pointerRight="28px"
|
|
|
|
style={popoutStyle}
|
|
|
|
>
|
|
|
|
<div className="export-row">
|
|
|
|
<span className="filename">File name</span>
|
|
|
|
<Input ref={input} title="filename" placeholder="carbon" color={COLORS.PURPLE} />
|
|
|
|
</div>
|
|
|
|
<div className="export-row">
|
|
|
|
<span>Size</span>
|
|
|
|
<div className="flex">
|
|
|
|
{EXPORT_SIZES.map(({ name }, i) => (
|
|
|
|
<Button
|
|
|
|
center
|
|
|
|
key={name}
|
|
|
|
hoverColor={COLORS.PURPLE}
|
|
|
|
margin={i === EXPORT_SIZES.length - 1 ? 0 : '0 10px 0 0'}
|
|
|
|
color={exportSize === name ? COLORS.PURPLE : COLORS.DARK_PURPLE}
|
|
|
|
onClick={handleExportSizeChange(name)}
|
|
|
|
>
|
|
|
|
{name}
|
|
|
|
</Button>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="export-row">
|
|
|
|
<Button center color={COLORS.PURPLE} onClick={handleExport('open')}>
|
|
|
|
Open
|
|
|
|
</Button>
|
|
|
|
<div className="save-container">
|
|
|
|
<span>Copy embed</span>
|
|
|
|
<div>
|
|
|
|
<CopyEmbed title="URL" mapper={toURL} margin="0 4px 0 0" />
|
|
|
|
<CopyEmbed title="IFrame" mapper={toIFrame} margin="0 0 0 4px" />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="save-container">
|
|
|
|
<span>Export as</span>
|
|
|
|
<div>
|
|
|
|
{!disablePNG && (
|
|
|
|
<Button
|
|
|
|
center
|
|
|
|
margin="0 8px 0 0"
|
|
|
|
hoverColor={COLORS.PURPLE}
|
|
|
|
color={COLORS.DARK_PURPLE}
|
|
|
|
onClick={handleExport('png')}
|
|
|
|
id="export-png"
|
|
|
|
>
|
|
|
|
PNG
|
|
|
|
</Button>
|
|
|
|
)}
|
|
|
|
<Button
|
|
|
|
center
|
|
|
|
hoverColor={COLORS.PURPLE}
|
|
|
|
color={COLORS.DARK_PURPLE}
|
|
|
|
onClick={handleExport('svg')}
|
|
|
|
id="export-svg"
|
|
|
|
>
|
|
|
|
SVG
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Popout>
|
|
|
|
<style jsx>
|
|
|
|
{`
|
|
|
|
.export-menu-container {
|
|
|
|
position: relative;
|
|
|
|
color: ${COLORS.PURPLE};
|
|
|
|
}
|
|
|
|
|
|
|
|
.flex {
|
|
|
|
display: flex;
|
|
|
|
height: 100%;
|
|
|
|
}
|
|
|
|
|
|
|
|
.export-row {
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
justify-content: space-between;
|
|
|
|
padding: 8px 16px;
|
|
|
|
border-bottom: 1px solid ${COLORS.PURPLE};
|
|
|
|
}
|
|
|
|
.export-row > :global(button) {
|
|
|
|
border-right: 1px solid ${COLORS.PURPLE};
|
|
|
|
}
|
|
|
|
.export-row:last-child {
|
|
|
|
border-bottom: none;
|
|
|
|
padding: 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
.filename {
|
|
|
|
flex-basis: 72px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.save-container {
|
|
|
|
display: flex;
|
|
|
|
flex: 1;
|
|
|
|
flex-direction: column;
|
|
|
|
justify-content: center;
|
|
|
|
align-items: center;
|
|
|
|
padding: 12px 16px;
|
|
|
|
}
|
|
|
|
.save-container > div {
|
|
|
|
margin-top: 6px;
|
|
|
|
display: flex;
|
|
|
|
flex: 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
.save-container:first-of-type {
|
|
|
|
padding: 12px 12px;
|
|
|
|
border-right: 1px solid ${COLORS.PURPLE};
|
|
|
|
}
|
|
|
|
`}
|
|
|
|
</style>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default managePopout(React.memo(ExportMenu))
|