2023-07-31 16:08:55 +00:00
|
|
|
import { useCopyTimeout } from '@vegaprotocol/react-helpers';
|
|
|
|
import {
|
|
|
|
TradingButton as Button,
|
|
|
|
VegaIcon,
|
|
|
|
VegaIconNames,
|
|
|
|
} from '@vegaprotocol/ui-toolkit';
|
|
|
|
import { truncateByChars } from '@vegaprotocol/utils';
|
2024-03-01 14:25:56 +00:00
|
|
|
import { type Key } from '@vegaprotocol/wallet';
|
|
|
|
import { useVegaWallet } from '@vegaprotocol/wallet-react';
|
2023-07-31 16:08:55 +00:00
|
|
|
import { useCallback, useMemo } from 'react';
|
|
|
|
import CopyToClipboard from 'react-copy-to-clipboard';
|
|
|
|
import { ViewType, useSidebar } from '../sidebar';
|
2023-09-15 12:36:08 +00:00
|
|
|
import { useGetCurrentRouteId } from '../../lib/hooks/use-get-current-route-id';
|
2023-11-16 03:10:39 +00:00
|
|
|
import { useT } from '../../lib/use-t';
|
2024-03-08 17:29:41 +00:00
|
|
|
import { usePartyProfilesQuery } from '../vega-wallet-connect-button/__generated__/PartyProfiles';
|
|
|
|
import { useProfileDialogStore } from '../../stores/profile-dialog-store';
|
2023-07-31 16:08:55 +00:00
|
|
|
|
|
|
|
export const VegaWalletMenu = ({
|
|
|
|
setMenu,
|
|
|
|
}: {
|
|
|
|
setMenu: (open: 'nav' | 'wallet' | null) => void;
|
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-07-31 16:08:55 +00:00
|
|
|
const { pubKey, pubKeys, selectPubKey, disconnect } = useVegaWallet();
|
2023-09-15 12:36:08 +00:00
|
|
|
const currentRouteId = useGetCurrentRouteId();
|
|
|
|
const setViews = useSidebar((store) => store.setViews);
|
2023-07-31 16:08:55 +00:00
|
|
|
|
2024-03-08 17:29:41 +00:00
|
|
|
const { data } = usePartyProfilesQuery({
|
|
|
|
variables: { partyIds: pubKeys.map((pk) => pk.publicKey) },
|
|
|
|
skip: pubKeys.length <= 0,
|
|
|
|
fetchPolicy: 'cache-and-network',
|
|
|
|
});
|
|
|
|
|
2023-07-31 16:08:55 +00:00
|
|
|
const activeKey = useMemo(() => {
|
|
|
|
return pubKeys?.find((pk) => pk.publicKey === pubKey);
|
|
|
|
}, [pubKey, pubKeys]);
|
|
|
|
|
|
|
|
const onSelectItem = useCallback(
|
|
|
|
(pubkey: string) => {
|
|
|
|
selectPubKey(pubkey);
|
|
|
|
},
|
|
|
|
[selectPubKey]
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div>
|
|
|
|
<div className="grow my-4" role="list">
|
2024-03-08 17:29:41 +00:00
|
|
|
{(pubKeys || []).map((pk) => {
|
|
|
|
const profile = data?.partiesProfilesConnection?.edges.find(
|
|
|
|
(e) => e.node.partyId === pk.publicKey
|
|
|
|
);
|
|
|
|
return (
|
|
|
|
<KeypairListItem
|
|
|
|
key={pk.publicKey}
|
|
|
|
pk={pk}
|
|
|
|
isActive={activeKey?.publicKey === pk.publicKey}
|
|
|
|
onSelectItem={onSelectItem}
|
|
|
|
alias={profile?.node.alias}
|
|
|
|
setMenu={setMenu}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
})}
|
2023-07-31 16:08:55 +00:00
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className="flex flex-col gap-2 m-4">
|
|
|
|
<Button
|
|
|
|
onClick={() => {
|
2023-09-15 12:36:08 +00:00
|
|
|
setViews({ type: ViewType.Transfer }, currentRouteId);
|
2023-07-31 16:08:55 +00:00
|
|
|
setMenu(null);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{t('Transfer')}
|
|
|
|
</Button>
|
|
|
|
<Button
|
|
|
|
onClick={async () => {
|
|
|
|
await disconnect();
|
|
|
|
setMenu(null);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{t('Disconnect')}
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const KeypairListItem = ({
|
|
|
|
pk,
|
|
|
|
isActive,
|
2024-03-08 17:29:41 +00:00
|
|
|
alias,
|
2023-07-31 16:08:55 +00:00
|
|
|
onSelectItem,
|
2024-03-08 17:29:41 +00:00
|
|
|
setMenu,
|
2023-07-31 16:08:55 +00:00
|
|
|
}: {
|
2024-03-01 14:25:56 +00:00
|
|
|
pk: Key;
|
2023-07-31 16:08:55 +00:00
|
|
|
isActive: boolean;
|
2024-03-08 17:29:41 +00:00
|
|
|
alias: string | undefined;
|
2023-07-31 16:08:55 +00:00
|
|
|
onSelectItem: (pk: string) => void;
|
2024-03-08 17:29:41 +00:00
|
|
|
setMenu: (open: 'nav' | 'wallet' | null) => void;
|
2023-07-31 16:08:55 +00:00
|
|
|
}) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-07-31 16:08:55 +00:00
|
|
|
const [copied, setCopied] = useCopyTimeout();
|
2024-03-08 17:29:41 +00:00
|
|
|
const setOpen = useProfileDialogStore((store) => store.setOpen);
|
2023-07-31 16:08:55 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<div
|
2024-03-08 17:29:41 +00:00
|
|
|
className="flex flex-col w-full px-4 mb-4"
|
2023-07-31 16:08:55 +00:00
|
|
|
data-testid={`key-${pk.publicKey}-mobile`}
|
|
|
|
>
|
|
|
|
<span className="flex gap-2 items-center mr-2">
|
|
|
|
<button type="button" onClick={() => onSelectItem(pk.publicKey)}>
|
|
|
|
<span className="uppercase">{pk.name}</span>
|
|
|
|
</button>
|
|
|
|
{isActive && <VegaIcon name={VegaIconNames.TICK} />}
|
|
|
|
</span>
|
|
|
|
<span className="flex gap-2 items-center">
|
|
|
|
{truncateByChars(pk.publicKey)}{' '}
|
|
|
|
<CopyToClipboard text={pk.publicKey} onCopy={() => setCopied(true)}>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
data-testid="copy-vega-public-key"
|
|
|
|
onClick={(e) => e.stopPropagation()}
|
|
|
|
>
|
|
|
|
<span className="sr-only">{t('Copy')}</span>
|
|
|
|
<VegaIcon name={VegaIconNames.COPY} />
|
|
|
|
</button>
|
|
|
|
</CopyToClipboard>
|
|
|
|
{copied && <span className="text-xs">{t('Copied')}</span>}
|
|
|
|
</span>
|
2024-03-08 17:29:41 +00:00
|
|
|
<span
|
|
|
|
className="flex gap-2 items-center"
|
|
|
|
data-testid={`key-${pk.publicKey}`}
|
|
|
|
>
|
|
|
|
<span className="truncate">{alias ? alias : t('No alias')}</span>
|
|
|
|
{isActive && (
|
|
|
|
<button
|
|
|
|
data-testid="alias"
|
|
|
|
onClick={() => {
|
|
|
|
setOpen(pk.publicKey);
|
|
|
|
setMenu(null);
|
|
|
|
}}
|
|
|
|
className="flex items-center gap-1"
|
|
|
|
>
|
|
|
|
<VegaIcon name={VegaIconNames.EDIT} />
|
|
|
|
</button>
|
|
|
|
)}
|
|
|
|
</span>
|
2023-07-31 16:08:55 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|