2022-09-20 22:31:54 -07:00
|
|
|
import { useState, useEffect } from 'react'
|
|
|
|
import mangoStore from '@store/mangoStore'
|
|
|
|
import { useWallet } from '@solana/wallet-adapter-react'
|
|
|
|
import { ArrowLeftIcon, PhotoIcon } from '@heroicons/react/20/solid'
|
|
|
|
import { useTranslation } from 'next-i18next'
|
2022-09-24 05:07:10 -07:00
|
|
|
import Button, { LinkButton } from '../shared/Button'
|
2022-09-20 22:31:54 -07:00
|
|
|
import { bs58 } from '@project-serum/anchor/dist/cjs/utils/bytes'
|
|
|
|
import { notify } from 'utils/notifications'
|
2023-01-18 05:13:29 -08:00
|
|
|
import { MANGO_DATA_API_URL } from 'utils/constants'
|
2022-09-20 22:31:54 -07:00
|
|
|
|
2023-02-25 16:22:39 -08:00
|
|
|
const ImgWithLoader = (props: { className: string; src: string }) => {
|
2022-09-20 22:31:54 -07:00
|
|
|
const [isLoading, setIsLoading] = useState(true)
|
|
|
|
return (
|
|
|
|
<div className="relative">
|
|
|
|
{isLoading && (
|
|
|
|
<PhotoIcon className="absolute left-1/2 top-1/2 z-10 h-1/4 w-1/4 -translate-x-1/2 -translate-y-1/2 animate-pulse text-th-fgd-4" />
|
|
|
|
)}
|
|
|
|
<img {...props} onLoad={() => setIsLoading(false)} alt="" />
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
const EditNftProfilePic = ({ onClose }: { onClose: () => void }) => {
|
|
|
|
const { t } = useTranslation(['common', 'profile'])
|
|
|
|
const { publicKey, signMessage } = useWallet()
|
|
|
|
const nfts = mangoStore((s) => s.wallet.nfts.data)
|
|
|
|
const nftsLoading = mangoStore((s) => s.wallet.nfts.loading)
|
|
|
|
const connection = mangoStore((s) => s.connection)
|
|
|
|
const [selectedProfile, setSelectedProfile] = useState<string>('')
|
2023-01-03 14:20:00 -08:00
|
|
|
const actions = mangoStore.getState().actions
|
2022-09-20 22:31:54 -07:00
|
|
|
const profile = mangoStore((s) => s.profile.details)
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (publicKey) {
|
|
|
|
actions.fetchNfts(connection, publicKey)
|
|
|
|
}
|
|
|
|
}, [publicKey])
|
|
|
|
|
|
|
|
useEffect(() => {
|
2022-12-23 10:53:57 -08:00
|
|
|
if (profile?.profile_image_url) {
|
2022-09-20 22:31:54 -07:00
|
|
|
setSelectedProfile(profile.profile_image_url)
|
|
|
|
}
|
|
|
|
}, [profile])
|
|
|
|
|
|
|
|
const saveProfileImage = async () => {
|
2022-12-23 10:53:57 -08:00
|
|
|
const name = profile?.profile_name.toLowerCase()
|
|
|
|
const traderCategory = profile?.trader_category
|
2022-09-20 22:31:54 -07:00
|
|
|
try {
|
|
|
|
if (!publicKey) throw new Error('Wallet not connected!')
|
|
|
|
if (!signMessage)
|
|
|
|
throw new Error('Wallet does not support message signing!')
|
|
|
|
|
|
|
|
const messageString = JSON.stringify({
|
|
|
|
profile_name: name,
|
|
|
|
trader_category: traderCategory,
|
|
|
|
profile_image_url: selectedProfile,
|
|
|
|
})
|
|
|
|
const message = new TextEncoder().encode(messageString)
|
|
|
|
const signature = await signMessage(message)
|
|
|
|
|
|
|
|
const requestOptions = {
|
|
|
|
method: 'POST',
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
body: JSON.stringify({
|
|
|
|
wallet_pk: publicKey.toString(),
|
|
|
|
message: messageString,
|
|
|
|
signature: bs58.encode(signature),
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
const response = await fetch(
|
2023-01-18 05:13:29 -08:00
|
|
|
`${MANGO_DATA_API_URL}/user-data/profile-details`,
|
2022-09-20 22:31:54 -07:00
|
|
|
requestOptions
|
|
|
|
)
|
|
|
|
if (response.status === 200) {
|
|
|
|
await actions.fetchProfileDetails(publicKey.toString())
|
|
|
|
notify({
|
|
|
|
type: 'success',
|
|
|
|
title: t('profile:profile-pic-success'),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
notify({
|
|
|
|
type: 'success',
|
|
|
|
title: t('profile:profile:profile-pic-failure'),
|
|
|
|
})
|
|
|
|
} finally {
|
|
|
|
onClose()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const removeProfileImage = async () => {
|
2022-12-23 10:53:57 -08:00
|
|
|
const name = profile?.profile_name.toLowerCase()
|
|
|
|
const traderCategory = profile?.trader_category
|
2022-09-20 22:31:54 -07:00
|
|
|
try {
|
|
|
|
if (!publicKey) throw new Error('Wallet not connected!')
|
|
|
|
if (!signMessage)
|
|
|
|
throw new Error('Wallet does not support message signing!')
|
|
|
|
|
|
|
|
const messageString = JSON.stringify({
|
|
|
|
profile_name: name,
|
|
|
|
trader_category: traderCategory,
|
|
|
|
profile_image_url: '',
|
|
|
|
})
|
|
|
|
const message = new TextEncoder().encode(messageString)
|
|
|
|
const signature = await signMessage(message)
|
|
|
|
|
|
|
|
const requestOptions = {
|
|
|
|
method: 'POST',
|
|
|
|
headers: { 'Content-Type': 'application/json' },
|
|
|
|
body: JSON.stringify({
|
|
|
|
wallet_pk: publicKey.toString(),
|
|
|
|
message: messageString,
|
|
|
|
signature: bs58.encode(signature),
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
const response = await fetch(
|
2023-01-18 05:13:29 -08:00
|
|
|
`${MANGO_DATA_API_URL}/user-data/profile-details`,
|
2022-09-20 22:31:54 -07:00
|
|
|
requestOptions
|
|
|
|
)
|
|
|
|
if (response.status === 200) {
|
|
|
|
await actions.fetchProfileDetails(publicKey.toString())
|
|
|
|
notify({
|
|
|
|
type: 'success',
|
|
|
|
title: t('profile:profile-pic-remove-success'),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
notify({
|
|
|
|
type: 'success',
|
|
|
|
title: t('profile:profile-pic-remove-failure'),
|
|
|
|
})
|
|
|
|
} finally {
|
|
|
|
onClose()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<div className="mb-3 flex w-full flex-col items-center sm:mt-3 sm:flex-row sm:justify-between">
|
|
|
|
<button
|
|
|
|
onClick={onClose}
|
2022-12-15 19:16:05 -08:00
|
|
|
className={`absolute left-2 top-3 z-40 text-th-fgd-4 focus:outline-none md:hover:text-th-active`}
|
2022-09-20 22:31:54 -07:00
|
|
|
>
|
|
|
|
<ArrowLeftIcon className={`h-5 w-5`} />
|
|
|
|
</button>
|
|
|
|
<h2>{t('profile:choose-profile')}</h2>
|
|
|
|
<div className="mt-3 flex items-center space-x-4 sm:mt-0">
|
|
|
|
<Button disabled={!selectedProfile} onClick={saveProfileImage}>
|
|
|
|
{t('save')}
|
|
|
|
</Button>
|
2022-12-23 10:53:57 -08:00
|
|
|
{profile?.profile_image_url ? (
|
2022-11-02 09:16:25 -07:00
|
|
|
<LinkButton className="text-sm" onClick={removeProfileImage}>
|
2022-09-20 22:31:54 -07:00
|
|
|
{t('profile:remove')}
|
|
|
|
</LinkButton>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{nfts.length > 0 ? (
|
|
|
|
<div className="flex flex-col items-center">
|
2022-11-02 09:16:25 -07:00
|
|
|
<div className="mb-4 grid w-full grid-flow-row grid-cols-3 gap-3">
|
2022-09-20 22:31:54 -07:00
|
|
|
{nfts.map((n) => (
|
|
|
|
<button
|
2022-11-02 09:16:25 -07:00
|
|
|
className={`default-transition col-span-1 flex items-center justify-center rounded-md border bg-th-bkg-2 py-3 sm:py-4 md:hover:bg-th-bkg-3 ${
|
2022-09-20 22:31:54 -07:00
|
|
|
selectedProfile === n.image
|
2022-11-30 19:32:32 -08:00
|
|
|
? 'border-th-active'
|
2022-09-20 22:31:54 -07:00
|
|
|
: 'border-th-bkg-3'
|
|
|
|
}`}
|
|
|
|
key={n.image}
|
|
|
|
onClick={() => setSelectedProfile(n.image)}
|
|
|
|
>
|
|
|
|
<ImgWithLoader
|
|
|
|
className="h-16 w-16 flex-shrink-0 rounded-full sm:h-20 sm:w-20"
|
|
|
|
src={n.image}
|
|
|
|
/>
|
|
|
|
</button>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
) : nftsLoading ? (
|
|
|
|
<div className="mb-4 grid w-full grid-flow-row grid-cols-3 gap-4">
|
|
|
|
{[...Array(9)].map((i) => (
|
|
|
|
<div
|
|
|
|
className="col-span-1 h-[90px] animate-pulse rounded-md bg-th-bkg-3 sm:h-28"
|
|
|
|
key={i}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
<div className="flex h-60 items-center justify-center">
|
|
|
|
<p>{t('profile:no-nfts')}</p>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default EditNftProfilePic
|