mango-v4-ui/components/modals/SharePositionModal.tsx

185 lines
6.0 KiB
TypeScript
Raw Normal View History

2023-02-26 19:09:31 -08:00
import { Group, PerpPosition } from '@blockworks-foundation/mango-v4'
import { TwitterIcon } from '@components/icons/TwitterIcon'
import Button from '@components/shared/Button'
import { Dialog } from '@headlessui/react'
import { DocumentDuplicateIcon } from '@heroicons/react/20/solid'
import { useScreenshot } from 'hooks/useScreenshot'
import { useTranslation } from 'next-i18next'
import { createRef, useEffect, useMemo, useState } from 'react'
import { ModalProps } from 'types/modal'
import { ttCommons, ttCommonsExpanded, ttCommonsMono } from 'utils/fonts'
2023-02-26 19:09:31 -08:00
import { formatNumericValue, getDecimalCount } from 'utils/numbers'
interface SharePositionModalProps {
group: Group
position: PerpPosition
}
type ModalCombinedProps = SharePositionModalProps & ModalProps
2023-02-26 19:30:58 -08:00
async function copyToClipboard(image: HTMLCanvasElement) {
2023-02-26 19:09:31 -08:00
try {
2023-02-26 19:30:58 -08:00
image.toBlob((blob: Blob | null) => {
if (blob) {
navigator.clipboard.write([new ClipboardItem({ 'image/png': blob })])
}
2023-02-26 19:09:31 -08:00
}, 'image/png')
} catch (error) {
console.error(error)
}
}
const SharePositionModal = ({
group,
isOpen,
onClose,
position,
}: ModalCombinedProps) => {
const { t } = useTranslation('trade')
const ref = createRef<HTMLDivElement>()
const [copied, setCopied] = useState(false)
const [showButton, setShowButton] = useState(true)
const [image, takeScreenshot] = useScreenshot()
const market = group.getPerpMarketByMarketIndex(position.marketIndex)
const basePosition = position.getBasePositionUi(market)
const entryPrice = position.getAverageEntryPriceUi(market)
const side = basePosition > 0 ? 'long' : 'short'
const roi = useMemo(() => {
if (!market) return 0
2023-03-29 03:38:26 -07:00
let roi
2023-02-26 19:09:31 -08:00
const indexPrice = market.uiPrice
2023-03-29 03:38:26 -07:00
if (basePosition > 0) {
roi = (indexPrice / entryPrice - 1) * 100
} else {
roi = (indexPrice / entryPrice - 1) * -100
}
2023-02-26 19:09:31 -08:00
return roi
2023-03-29 03:38:26 -07:00
}, [basePosition, entryPrice, market])
2023-02-26 19:09:31 -08:00
useEffect(() => {
if (image) {
copyToClipboard(image)
setCopied(true)
setShowButton(true)
}
}, [image])
useEffect(() => {
// if the button is hidden we are taking a screenshot
if (!showButton) {
takeScreenshot(ref.current as HTMLElement)
}
}, [showButton])
const handleCopyToClipboard = () => {
setShowButton(false)
}
return (
<Dialog
open={isOpen}
onClose={onClose}
className="relative z-40 overflow-y-auto"
>
<div
className="fixed inset-0 backdrop-brightness-[0.3]"
aria-hidden="true"
/>
<div className="fixed inset-0 flex flex-col items-center justify-center text-center">
<Dialog.Panel
className={`relative flex flex-col items-center ${ttCommons.variable} ${ttCommonsExpanded.variable} ${ttCommonsMono.variable}`}
>
2023-02-26 19:09:31 -08:00
<div
2023-02-26 20:44:59 -08:00
className="relative h-[338px] w-[600px] overflow-hidden border border-slate-700 bg-gradient-to-b from-slate-900 to-black pt-6"
2023-02-26 19:09:31 -08:00
ref={ref}
>
2023-02-26 20:44:59 -08:00
<div className="w-1/2 text-left">
<div className="px-8">
<img
className="mb-8 h-7 w-auto shrink-0"
2023-02-26 20:44:59 -08:00
src="/logos/logo-with-text.svg"
alt="Mango"
/>
<div className="mb-4 flex items-center">
<p
className={`font-display text-base uppercase text-th-fgd-1 ${
side === 'long' ? 'text-th-up' : 'text-th-down'
2023-02-26 19:09:31 -08:00
}`}
>
2023-02-26 20:44:59 -08:00
{side}
</p>
<span className="mx-2 text-base text-th-fgd-4">|</span>
<p className="font-display text-base text-white">
{market.name}
</p>
</div>
<div
className={`mb-4 font-display text-5xl ${
roi >= 0 ? 'text-th-up' : 'text-th-down'
}`}
>
{roi >= 0 ? '+' : ''}
{roi.toFixed(2)}%
</div>
<div className="flex justify-between">
<p className="mb-2 text-base text-gray-500">
2023-06-27 20:48:25 -07:00
{t('avg-entry-price')}
2023-02-26 20:44:59 -08:00
</p>
<p className="ml-2 font-mono text-base text-white">
{formatNumericValue(
entryPrice,
2023-07-21 11:47:53 -07:00
getDecimalCount(market.tickSize),
2023-02-26 20:44:59 -08:00
)}
</p>
</div>
<div className="flex justify-between">
<p className="text-base text-gray-500">
{t('current-price')}
</p>
<p className="ml-2 font-mono text-base text-white">
{formatNumericValue(
market.uiPrice,
2023-07-21 11:47:53 -07:00
getDecimalCount(market.tickSize),
2023-02-26 20:44:59 -08:00
)}
</p>
2023-02-26 19:09:31 -08:00
</div>
</div>
</div>
<div className="absolute left-0 top-0">
2023-02-26 19:09:31 -08:00
<img
src={roi >= 0 ? '/images/space.svg' : '/images/underwater.svg'}
alt="Share Background"
/>
</div>
</div>
{copied ? (
<a
href={`https://twitter.com/intent/tweet?text=I'm ${side.toUpperCase()} %24${
market.name
} on %40mangomarkets%0A[PASTE IMAGE HERE]`}
target="_blank"
rel="noreferrer"
>
<Button className="mt-6 flex items-center">
<TwitterIcon className="mr-2 h-5 w-5 text-th-fgd-1" />
{t('tweet-position')}
</Button>
</a>
) : (
<Button
className="mt-6 flex items-center"
onClick={handleCopyToClipboard}
>
<DocumentDuplicateIcon className="mr-2 h-5 w-5 text-th-fgd-1" />
{t('copy-and-share')}
</Button>
)}
</Dialog.Panel>
</div>
</Dialog>
)
}
export default SharePositionModal