feat(trading,governance,wallet): browser wallet integration (#4121)
This commit is contained in:
parent
5692d4e74c
commit
6a55319e04
@ -78,7 +78,7 @@ context(
|
|||||||
cy.getByTestId('connector-jsonRpc')
|
cy.getByTestId('connector-jsonRpc')
|
||||||
.should('be.visible')
|
.should('be.visible')
|
||||||
.and('have.text', 'Connect Vega wallet');
|
.and('have.text', 'Connect Vega wallet');
|
||||||
cy.getByTestId('connector-hosted')
|
cy.getByTestId('connector-rest')
|
||||||
.should('be.visible')
|
.should('be.visible')
|
||||||
.and('have.text', 'Hosted Fairground wallet');
|
.and('have.text', 'Hosted Fairground wallet');
|
||||||
});
|
});
|
||||||
@ -94,7 +94,7 @@ context(
|
|||||||
describe('when rest connector form opened', function () {
|
describe('when rest connector form opened', function () {
|
||||||
before('click hosted wallet app button', function () {
|
before('click hosted wallet app button', function () {
|
||||||
cy.getByTestId(connectorsList).within(() => {
|
cy.getByTestId(connectorsList).within(() => {
|
||||||
cy.getByTestId('connector-hosted').click();
|
cy.getByTestId('connector-rest').click();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -2,13 +2,8 @@ import { Button } from '@vegaprotocol/ui-toolkit';
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { useVegaWalletDialogStore } from '@vegaprotocol/wallet';
|
import { useVegaWalletDialogStore } from '@vegaprotocol/wallet';
|
||||||
import {
|
|
||||||
AppStateActionType,
|
|
||||||
useAppState,
|
|
||||||
} from '../../contexts/app-state/app-state-context';
|
|
||||||
|
|
||||||
export const ConnectToVega = () => {
|
export const ConnectToVega = () => {
|
||||||
const { appDispatch } = useAppState();
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
||||||
openVegaWalletDialog: store.openVegaWalletDialog,
|
openVegaWalletDialog: store.openVegaWalletDialog,
|
||||||
@ -16,10 +11,6 @@ export const ConnectToVega = () => {
|
|||||||
return (
|
return (
|
||||||
<Button
|
<Button
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
appDispatch({
|
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_OVERLAY,
|
|
||||||
isOpen: true,
|
|
||||||
});
|
|
||||||
openVegaWalletDialog();
|
openVegaWalletDialog();
|
||||||
}}
|
}}
|
||||||
data-testid="connect-to-vega-wallet-btn"
|
data-testid="connect-to-vega-wallet-btn"
|
||||||
|
@ -3,11 +3,6 @@ import { useVegaWallet, useVegaWalletDialogStore } from '@vegaprotocol/wallet';
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
|
||||||
import {
|
|
||||||
AppStateActionType,
|
|
||||||
useAppState,
|
|
||||||
} from '../../contexts/app-state/app-state-context';
|
|
||||||
|
|
||||||
interface VegaWalletContainerProps {
|
interface VegaWalletContainerProps {
|
||||||
children: (key: string) => React.ReactElement;
|
children: (key: string) => React.ReactElement;
|
||||||
}
|
}
|
||||||
@ -15,7 +10,6 @@ interface VegaWalletContainerProps {
|
|||||||
export const VegaWalletContainer = ({ children }: VegaWalletContainerProps) => {
|
export const VegaWalletContainer = ({ children }: VegaWalletContainerProps) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { pubKey } = useVegaWallet();
|
const { pubKey } = useVegaWallet();
|
||||||
const { appDispatch } = useAppState();
|
|
||||||
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
||||||
openVegaWalletDialog: store.openVegaWalletDialog,
|
openVegaWalletDialog: store.openVegaWalletDialog,
|
||||||
}));
|
}));
|
||||||
@ -25,10 +19,6 @@ export const VegaWalletContainer = ({ children }: VegaWalletContainerProps) => {
|
|||||||
<Button
|
<Button
|
||||||
data-testid="connect-to-vega-wallet-btn"
|
data-testid="connect-to-vega-wallet-btn"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
appDispatch({
|
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_OVERLAY,
|
|
||||||
isOpen: true,
|
|
||||||
});
|
|
||||||
openVegaWalletDialog();
|
openVegaWalletDialog();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
@ -13,12 +13,6 @@ export const VegaWalletDialogs = () => {
|
|||||||
<>
|
<>
|
||||||
<VegaConnectDialog
|
<VegaConnectDialog
|
||||||
connectors={Connectors}
|
connectors={Connectors}
|
||||||
onChangeOpen={(open) =>
|
|
||||||
appDispatch({
|
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_OVERLAY,
|
|
||||||
isOpen: open,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
riskMessage={<RiskMessage />}
|
riskMessage={<RiskMessage />}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
@ -71,7 +71,6 @@ export const VegaWallet = () => {
|
|||||||
|
|
||||||
const VegaWalletNotConnected = () => {
|
const VegaWalletNotConnected = () => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { appDispatch } = useAppState();
|
|
||||||
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
||||||
openVegaWalletDialog: store.openVegaWalletDialog,
|
openVegaWalletDialog: store.openVegaWalletDialog,
|
||||||
}));
|
}));
|
||||||
@ -79,10 +78,6 @@ const VegaWalletNotConnected = () => {
|
|||||||
<>
|
<>
|
||||||
<Button
|
<Button
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
appDispatch({
|
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_OVERLAY,
|
|
||||||
isOpen: true,
|
|
||||||
});
|
|
||||||
openVegaWalletDialog();
|
openVegaWalletDialog();
|
||||||
}}
|
}}
|
||||||
fill={true}
|
fill={true}
|
||||||
|
@ -28,9 +28,6 @@ export interface AppState {
|
|||||||
/** Total number of VEGA Tokens, both vesting and unlocked, associated for staking */
|
/** Total number of VEGA Tokens, both vesting and unlocked, associated for staking */
|
||||||
totalAssociated: BigNumber;
|
totalAssociated: BigNumber;
|
||||||
|
|
||||||
/** Whether or not the connect to VEGA wallet overlay is open */
|
|
||||||
vegaWalletOverlay: boolean;
|
|
||||||
|
|
||||||
/** Whether or not the manage VEGA wallet overlay is open */
|
/** Whether or not the manage VEGA wallet overlay is open */
|
||||||
vegaWalletManageOverlay: boolean;
|
vegaWalletManageOverlay: boolean;
|
||||||
|
|
||||||
@ -52,9 +49,7 @@ export enum AppStateActionType {
|
|||||||
SET_TOKEN,
|
SET_TOKEN,
|
||||||
SET_ALLOWANCE,
|
SET_ALLOWANCE,
|
||||||
REFRESH_BALANCES,
|
REFRESH_BALANCES,
|
||||||
SET_VEGA_WALLET_OVERLAY,
|
|
||||||
SET_VEGA_WALLET_MANAGE_OVERLAY,
|
SET_VEGA_WALLET_MANAGE_OVERLAY,
|
||||||
SET_DRAWER,
|
|
||||||
REFRESH_ASSOCIATED_BALANCES,
|
REFRESH_ASSOCIATED_BALANCES,
|
||||||
SET_ASSOCIATION_BREAKDOWN,
|
SET_ASSOCIATION_BREAKDOWN,
|
||||||
SET_TRANSACTION_OVERLAY,
|
SET_TRANSACTION_OVERLAY,
|
||||||
@ -69,18 +64,10 @@ export type AppStateAction =
|
|||||||
totalSupply: BigNumber;
|
totalSupply: BigNumber;
|
||||||
totalAssociated: BigNumber;
|
totalAssociated: BigNumber;
|
||||||
}
|
}
|
||||||
| {
|
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_OVERLAY;
|
|
||||||
isOpen: boolean;
|
|
||||||
}
|
|
||||||
| {
|
| {
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_MANAGE_OVERLAY;
|
type: AppStateActionType.SET_VEGA_WALLET_MANAGE_OVERLAY;
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
}
|
}
|
||||||
| {
|
|
||||||
type: AppStateActionType.SET_DRAWER;
|
|
||||||
isOpen: boolean;
|
|
||||||
}
|
|
||||||
| {
|
| {
|
||||||
type: AppStateActionType.SET_TRANSACTION_OVERLAY;
|
type: AppStateActionType.SET_TRANSACTION_OVERLAY;
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
|
@ -14,7 +14,6 @@ const initialAppState: AppState = {
|
|||||||
totalAssociated: new BigNumber(0),
|
totalAssociated: new BigNumber(0),
|
||||||
decimals: 0,
|
decimals: 0,
|
||||||
totalSupply: new BigNumber(0),
|
totalSupply: new BigNumber(0),
|
||||||
vegaWalletOverlay: false,
|
|
||||||
vegaWalletManageOverlay: false,
|
vegaWalletManageOverlay: false,
|
||||||
transactionOverlay: false,
|
transactionOverlay: false,
|
||||||
bannerMessage: '',
|
bannerMessage: '',
|
||||||
@ -31,23 +30,10 @@ function appStateReducer(state: AppState, action: AppStateAction): AppState {
|
|||||||
totalAssociated: action.totalAssociated,
|
totalAssociated: action.totalAssociated,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
case AppStateActionType.SET_VEGA_WALLET_OVERLAY: {
|
|
||||||
return {
|
|
||||||
...state,
|
|
||||||
vegaWalletOverlay: action.isOpen,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
case AppStateActionType.SET_VEGA_WALLET_MANAGE_OVERLAY: {
|
case AppStateActionType.SET_VEGA_WALLET_MANAGE_OVERLAY: {
|
||||||
return {
|
return {
|
||||||
...state,
|
...state,
|
||||||
vegaWalletManageOverlay: action.isOpen,
|
vegaWalletManageOverlay: action.isOpen,
|
||||||
vegaWalletOverlay: action.isOpen ? false : state.vegaWalletOverlay,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
case AppStateActionType.SET_DRAWER: {
|
|
||||||
return {
|
|
||||||
...state,
|
|
||||||
vegaWalletOverlay: false,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
case AppStateActionType.SET_TRANSACTION_OVERLAY: {
|
case AppStateActionType.SET_TRANSACTION_OVERLAY: {
|
||||||
|
@ -2,15 +2,18 @@ import {
|
|||||||
RestConnector,
|
RestConnector,
|
||||||
JsonRpcConnector,
|
JsonRpcConnector,
|
||||||
ViewConnector,
|
ViewConnector,
|
||||||
|
InjectedConnector,
|
||||||
} from '@vegaprotocol/wallet';
|
} from '@vegaprotocol/wallet';
|
||||||
|
|
||||||
const urlParams = new URLSearchParams(window.location.search);
|
const urlParams = new URLSearchParams(window.location.search);
|
||||||
|
|
||||||
|
export const injected = new InjectedConnector();
|
||||||
export const rest = new RestConnector();
|
export const rest = new RestConnector();
|
||||||
export const jsonRpc = new JsonRpcConnector();
|
export const jsonRpc = new JsonRpcConnector();
|
||||||
export const view = new ViewConnector(urlParams.get('address'));
|
export const view = new ViewConnector(urlParams.get('address'));
|
||||||
|
|
||||||
export const Connectors = {
|
export const Connectors = {
|
||||||
|
injected,
|
||||||
rest,
|
rest,
|
||||||
jsonRpc,
|
jsonRpc,
|
||||||
view,
|
view,
|
||||||
|
@ -10,10 +10,7 @@ import {
|
|||||||
} from '@vegaprotocol/ui-toolkit';
|
} from '@vegaprotocol/ui-toolkit';
|
||||||
import { addDecimal, toBigNum } from '@vegaprotocol/utils';
|
import { addDecimal, toBigNum } from '@vegaprotocol/utils';
|
||||||
import { ProposalState, VoteValue } from '@vegaprotocol/types';
|
import { ProposalState, VoteValue } from '@vegaprotocol/types';
|
||||||
import {
|
import { useAppState } from '../../../../contexts/app-state/app-state-context';
|
||||||
AppStateActionType,
|
|
||||||
useAppState,
|
|
||||||
} from '../../../../contexts/app-state/app-state-context';
|
|
||||||
import { BigNumber } from '../../../../lib/bignumber';
|
import { BigNumber } from '../../../../lib/bignumber';
|
||||||
import { DATE_FORMAT_LONG } from '../../../../lib/date-formats';
|
import { DATE_FORMAT_LONG } from '../../../../lib/date-formats';
|
||||||
import { VoteState } from './use-user-vote';
|
import { VoteState } from './use-user-vote';
|
||||||
@ -73,7 +70,6 @@ export const VoteButtons = ({
|
|||||||
dialog: Dialog,
|
dialog: Dialog,
|
||||||
}: VoteButtonsProps) => {
|
}: VoteButtonsProps) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { appDispatch } = useAppState();
|
|
||||||
const { pubKey } = useVegaWallet();
|
const { pubKey } = useVegaWallet();
|
||||||
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
||||||
openVegaWalletDialog: store.openVegaWalletDialog,
|
openVegaWalletDialog: store.openVegaWalletDialog,
|
||||||
@ -98,10 +94,6 @@ export const VoteButtons = ({
|
|||||||
<div data-testid="connect-wallet">
|
<div data-testid="connect-wallet">
|
||||||
<ButtonLink
|
<ButtonLink
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
appDispatch({
|
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_OVERLAY,
|
|
||||||
isOpen: true,
|
|
||||||
});
|
|
||||||
openVegaWalletDialog();
|
openVegaWalletDialog();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@ -142,7 +134,6 @@ export const VoteButtons = ({
|
|||||||
minVoterBalance,
|
minVoterBalance,
|
||||||
spamProtectionMinTokens,
|
spamProtectionMinTokens,
|
||||||
t,
|
t,
|
||||||
appDispatch,
|
|
||||||
openVegaWalletDialog,
|
openVegaWalletDialog,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
@ -14,7 +14,6 @@ const mockAppState: AppState = {
|
|||||||
totalAssociated: new BigNumber('50063005'),
|
totalAssociated: new BigNumber('50063005'),
|
||||||
decimals: 18,
|
decimals: 18,
|
||||||
totalSupply: mockTotalSupply,
|
totalSupply: mockTotalSupply,
|
||||||
vegaWalletOverlay: false,
|
|
||||||
vegaWalletManageOverlay: false,
|
vegaWalletManageOverlay: false,
|
||||||
transactionOverlay: false,
|
transactionOverlay: false,
|
||||||
bannerMessage: '',
|
bannerMessage: '',
|
||||||
|
@ -2,14 +2,9 @@ import classNames from 'classnames';
|
|||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { useVegaWalletDialogStore } from '@vegaprotocol/wallet';
|
import { useVegaWalletDialogStore } from '@vegaprotocol/wallet';
|
||||||
import { Button } from '@vegaprotocol/ui-toolkit';
|
import { Button } from '@vegaprotocol/ui-toolkit';
|
||||||
import {
|
|
||||||
AppStateActionType,
|
|
||||||
useAppState,
|
|
||||||
} from '../../contexts/app-state/app-state-context';
|
|
||||||
import { SubHeading } from '../../components/heading';
|
import { SubHeading } from '../../components/heading';
|
||||||
|
|
||||||
export const ConnectToSeeRewards = () => {
|
export const ConnectToSeeRewards = () => {
|
||||||
const { appDispatch } = useAppState();
|
|
||||||
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
const { openVegaWalletDialog } = useVegaWalletDialogStore((store) => ({
|
||||||
openVegaWalletDialog: store.openVegaWalletDialog,
|
openVegaWalletDialog: store.openVegaWalletDialog,
|
||||||
}));
|
}));
|
||||||
@ -26,10 +21,6 @@ export const ConnectToSeeRewards = () => {
|
|||||||
<Button
|
<Button
|
||||||
data-testid="connect-to-vega-wallet-btn"
|
data-testid="connect-to-vega-wallet-btn"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
appDispatch({
|
|
||||||
type: AppStateActionType.SET_VEGA_WALLET_OVERLAY,
|
|
||||||
isOpen: true,
|
|
||||||
});
|
|
||||||
openVegaWalletDialog();
|
openVegaWalletDialog();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
@ -63,7 +63,7 @@ describe(
|
|||||||
cy.contains('Hosted Fairground wallet');
|
cy.contains('Hosted Fairground wallet');
|
||||||
|
|
||||||
cy.getByTestId('connectors-list')
|
cy.getByTestId('connectors-list')
|
||||||
.find('[data-testid="connector-hosted"]')
|
.find('[data-testid="connector-rest"]')
|
||||||
.click();
|
.click();
|
||||||
cy.getByTestId(form).find('#wallet').click().type('user');
|
cy.getByTestId(form).find('#wallet').click().type('user');
|
||||||
cy.getByTestId(form).find('#passphrase').click().type('pass');
|
cy.getByTestId(form).find('#passphrase').click().type('pass');
|
||||||
@ -89,7 +89,7 @@ describe(
|
|||||||
);
|
);
|
||||||
cy.getByTestId(connectVegaBtn).click();
|
cy.getByTestId(connectVegaBtn).click();
|
||||||
cy.getByTestId('connectors-list')
|
cy.getByTestId('connectors-list')
|
||||||
.find('[data-testid="connector-hosted"]')
|
.find('[data-testid="connector-rest"]')
|
||||||
.click();
|
.click();
|
||||||
cy.getByTestId(form).find('#wallet').click().type('invalid name');
|
cy.getByTestId(form).find('#wallet').click().type('invalid name');
|
||||||
cy.getByTestId(form).find('#passphrase').click().type('invalid password');
|
cy.getByTestId(form).find('#passphrase').click().type('invalid password');
|
||||||
@ -100,7 +100,7 @@ describe(
|
|||||||
it('doesnt connect with empty fields', () => {
|
it('doesnt connect with empty fields', () => {
|
||||||
cy.getByTestId(connectVegaBtn).click();
|
cy.getByTestId(connectVegaBtn).click();
|
||||||
cy.getByTestId('connectors-list')
|
cy.getByTestId('connectors-list')
|
||||||
.find('[data-testid="connector-hosted"]')
|
.find('[data-testid="connector-rest"]')
|
||||||
.click();
|
.click();
|
||||||
|
|
||||||
cy.getByTestId('rest-connector-form').find('button[type=submit]').click();
|
cy.getByTestId('rest-connector-form').find('button[type=submit]').click();
|
||||||
|
@ -190,6 +190,9 @@ export const VegaWalletConnectButton = () => {
|
|||||||
>
|
>
|
||||||
<DropdownMenuContent
|
<DropdownMenuContent
|
||||||
onInteractOutside={() => setDropdownOpen(false)}
|
onInteractOutside={() => setDropdownOpen(false)}
|
||||||
|
sideOffset={20}
|
||||||
|
side="bottom"
|
||||||
|
align="end"
|
||||||
>
|
>
|
||||||
<div className="min-w-[340px]" data-testid="keypair-list">
|
<div className="min-w-[340px]" data-testid="keypair-list">
|
||||||
<DropdownMenuRadioGroup
|
<DropdownMenuRadioGroup
|
||||||
|
@ -2,10 +2,12 @@ import {
|
|||||||
RestConnector,
|
RestConnector,
|
||||||
JsonRpcConnector,
|
JsonRpcConnector,
|
||||||
ViewConnector,
|
ViewConnector,
|
||||||
|
InjectedConnector,
|
||||||
} from '@vegaprotocol/wallet';
|
} from '@vegaprotocol/wallet';
|
||||||
|
|
||||||
export const rest = new RestConnector();
|
export const rest = new RestConnector();
|
||||||
export const jsonRpc = new JsonRpcConnector();
|
export const jsonRpc = new JsonRpcConnector();
|
||||||
|
export const injected = new InjectedConnector();
|
||||||
|
|
||||||
let view: ViewConnector;
|
let view: ViewConnector;
|
||||||
if (typeof window !== 'undefined') {
|
if (typeof window !== 'undefined') {
|
||||||
@ -16,6 +18,7 @@ if (typeof window !== 'undefined') {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const Connectors = {
|
export const Connectors = {
|
||||||
|
injected,
|
||||||
rest,
|
rest,
|
||||||
jsonRpc,
|
jsonRpc,
|
||||||
view,
|
view,
|
||||||
|
@ -45,7 +45,7 @@ export function Dialog({
|
|||||||
'dark:bg-black bg-white dark:text-white',
|
'dark:bg-black bg-white dark:text-white',
|
||||||
getIntentBorder(intent),
|
getIntentBorder(intent),
|
||||||
{
|
{
|
||||||
'w-[620px]': size === 'small',
|
'w-[520px]': size === 'small',
|
||||||
'w-[720px] lg:w-[940px]': size === 'medium',
|
'w-[720px] lg:w-[940px]': size === 'medium',
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -77,7 +77,7 @@ export function Dialog({
|
|||||||
className="absolute p-2 top-0 right-0 md:top-2 md:right-2"
|
className="absolute p-2 top-0 right-0 md:top-2 md:right-2"
|
||||||
data-testid="dialog-close"
|
data-testid="dialog-close"
|
||||||
>
|
>
|
||||||
<VegaIcon name={VegaIconNames.CROSS} />
|
<VegaIcon name={VegaIconNames.CROSS} size={24} />
|
||||||
</DialogPrimitives.Close>
|
</DialogPrimitives.Close>
|
||||||
)}
|
)}
|
||||||
<div className="flex gap-4 max-w-full">
|
<div className="flex gap-4 max-w-full">
|
||||||
|
@ -74,11 +74,11 @@ export const DropdownMenuContent = forwardRef<
|
|||||||
React.ComponentProps<typeof DropdownMenuPrimitive.Content>
|
React.ComponentProps<typeof DropdownMenuPrimitive.Content>
|
||||||
>(({ className, ...contentProps }, forwardedRef) => (
|
>(({ className, ...contentProps }, forwardedRef) => (
|
||||||
<DropdownMenuPrimitive.Content
|
<DropdownMenuPrimitive.Content
|
||||||
{...contentProps}
|
|
||||||
ref={forwardedRef}
|
ref={forwardedRef}
|
||||||
|
sideOffset={10}
|
||||||
className="min-w-[290px] bg-vega-light-100 dark:bg-vega-dark-100 p-2 rounded z-20 text-black dark:text-white border border-vega-light-200 dark:border-vega-dark-200"
|
className="min-w-[290px] bg-vega-light-100 dark:bg-vega-dark-100 p-2 rounded z-20 text-black dark:text-white border border-vega-light-200 dark:border-vega-dark-200"
|
||||||
align="start"
|
align="start"
|
||||||
sideOffset={10}
|
{...contentProps}
|
||||||
/>
|
/>
|
||||||
));
|
));
|
||||||
|
|
||||||
|
@ -0,0 +1,7 @@
|
|||||||
|
export const IconChevronLeft = ({ size = 16 }: { size: number }) => {
|
||||||
|
return (
|
||||||
|
<svg width={size} height={size} viewBox="0 0 16 16">
|
||||||
|
<path d="M10.38 1.62L11.13 2.38L5.5 8L11.13 13.62L10.38 14.38L4 8L10.38 1.62Z" />
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
};
|
@ -2,6 +2,7 @@ import { IconArrowDown } from './svg-icons/icon-arrow-down';
|
|||||||
import { IconArrowRight } from './svg-icons/icon-arrow-right';
|
import { IconArrowRight } from './svg-icons/icon-arrow-right';
|
||||||
import { IconBreakdown } from './svg-icons/icon-breakdown';
|
import { IconBreakdown } from './svg-icons/icon-breakdown';
|
||||||
import { IconChevronDown } from './svg-icons/icon-chevron-down';
|
import { IconChevronDown } from './svg-icons/icon-chevron-down';
|
||||||
|
import { IconChevronLeft } from './svg-icons/icon-chevron-left';
|
||||||
import { IconChevronUp } from './svg-icons/icon-chevron-up';
|
import { IconChevronUp } from './svg-icons/icon-chevron-up';
|
||||||
import { IconCopy } from './svg-icons/icon-copy';
|
import { IconCopy } from './svg-icons/icon-copy';
|
||||||
import { IconCross } from './svg-icons/icon-cross';
|
import { IconCross } from './svg-icons/icon-cross';
|
||||||
@ -26,6 +27,7 @@ export enum VegaIconNames {
|
|||||||
ARROW_RIGHT = 'arrow-right',
|
ARROW_RIGHT = 'arrow-right',
|
||||||
BREAKDOWN = 'breakdown',
|
BREAKDOWN = 'breakdown',
|
||||||
CHEVRON_DOWN = 'chevron-down',
|
CHEVRON_DOWN = 'chevron-down',
|
||||||
|
CHEVRON_LEFT = 'chevron-left',
|
||||||
CHEVRON_UP = 'chevron-up',
|
CHEVRON_UP = 'chevron-up',
|
||||||
COPY = 'copy',
|
COPY = 'copy',
|
||||||
CROSS = 'cross',
|
CROSS = 'cross',
|
||||||
@ -53,6 +55,7 @@ export const VegaIconNameMap: Record<
|
|||||||
'arrow-down': IconArrowDown,
|
'arrow-down': IconArrowDown,
|
||||||
'arrow-right': IconArrowRight,
|
'arrow-right': IconArrowRight,
|
||||||
'chevron-down': IconChevronDown,
|
'chevron-down': IconChevronDown,
|
||||||
|
'chevron-left': IconChevronLeft,
|
||||||
'chevron-up': IconChevronUp,
|
'chevron-up': IconChevronUp,
|
||||||
'open-external': IconOpenExternal,
|
'open-external': IconOpenExternal,
|
||||||
'question-mark': IconQuestionMark,
|
'question-mark': IconQuestionMark,
|
||||||
|
@ -1,7 +1,10 @@
|
|||||||
import { DocsLinks, ExternalLinks } from '@vegaprotocol/environment';
|
import { DocsLinks, ExternalLinks } from '@vegaprotocol/environment';
|
||||||
import { t } from '@vegaprotocol/i18n';
|
import { t } from '@vegaprotocol/i18n';
|
||||||
import { Link } from '@vegaprotocol/ui-toolkit';
|
import { Link } from '@vegaprotocol/ui-toolkit';
|
||||||
|
import classNames from 'classnames';
|
||||||
import type { ReactNode } from 'react';
|
import type { ReactNode } from 'react';
|
||||||
|
import type { VegaConnector } from '../connectors';
|
||||||
|
import { RestConnector } from '../connectors';
|
||||||
|
|
||||||
export const ConnectDialogTitle = ({ children }: { children: ReactNode }) => {
|
export const ConnectDialogTitle = ({ children }: { children: ReactNode }) => {
|
||||||
return (
|
return (
|
||||||
@ -18,11 +21,35 @@ export const ConnectDialogContent = ({ children }: { children: ReactNode }) => {
|
|||||||
return <div>{children}</div>;
|
return <div>{children}</div>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const ConnectDialogFooter = ({ children }: { children?: ReactNode }) => {
|
export const ConnectDialogFooter = ({
|
||||||
|
connector,
|
||||||
|
}: {
|
||||||
|
connector: VegaConnector | undefined;
|
||||||
|
}) => {
|
||||||
|
const wrapperClasses = classNames(
|
||||||
|
'flex justify-center gap-4',
|
||||||
|
'px-4 md:px-8 pt-4 md:pt-6',
|
||||||
|
'border-t border-vega-light-200 dark:border-vega-dark-200',
|
||||||
|
'text-vega-light-400 dark:text-vega-dark-400'
|
||||||
|
);
|
||||||
|
const isHostedWalletSelected = connector instanceof RestConnector;
|
||||||
return (
|
return (
|
||||||
<footer className="flex justify-center gap-4 px-4 md:px-8 pt-4 md:pt-6 -mx-4 md:-mx-8 border-t border-neutral-500 text-neutral-500 dark:text-neutral-400 mt-6">
|
<footer className={wrapperClasses}>
|
||||||
{children ? (
|
{isHostedWalletSelected ? (
|
||||||
children
|
<p className="text-center">
|
||||||
|
{t('For demo purposes get a ')}
|
||||||
|
<Link
|
||||||
|
href={ExternalLinks.VEGA_WALLET_HOSTED_URL}
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
>
|
||||||
|
{t('hosted wallet')}
|
||||||
|
</Link>
|
||||||
|
{t(', or for the real experience create a wallet in the ')}
|
||||||
|
<Link href={ExternalLinks.VEGA_WALLET_URL}>
|
||||||
|
{t('Vega wallet app')}
|
||||||
|
</Link>
|
||||||
|
</p>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
<Link href={ExternalLinks.VEGA_WALLET_URL}>
|
<Link href={ExternalLinks.VEGA_WALLET_URL}>
|
||||||
|
@ -16,6 +16,7 @@ import {
|
|||||||
import type { VegaConnectDialogProps } from '..';
|
import type { VegaConnectDialogProps } from '..';
|
||||||
import {
|
import {
|
||||||
ClientErrors,
|
ClientErrors,
|
||||||
|
InjectedConnector,
|
||||||
JsonRpcConnector,
|
JsonRpcConnector,
|
||||||
RestConnector,
|
RestConnector,
|
||||||
ViewConnector,
|
ViewConnector,
|
||||||
@ -24,6 +25,12 @@ import {
|
|||||||
import { useEnvironment } from '@vegaprotocol/environment';
|
import { useEnvironment } from '@vegaprotocol/environment';
|
||||||
import type { ChainIdQuery } from './__generated__/ChainId';
|
import type { ChainIdQuery } from './__generated__/ChainId';
|
||||||
import { ChainIdDocument } from './__generated__/ChainId';
|
import { ChainIdDocument } from './__generated__/ChainId';
|
||||||
|
import {
|
||||||
|
mockBrowserWallet,
|
||||||
|
clearBrowserWallet,
|
||||||
|
delayedReject,
|
||||||
|
delayedResolve,
|
||||||
|
} from '../test-helpers';
|
||||||
|
|
||||||
const mockUpdateDialogOpen = jest.fn();
|
const mockUpdateDialogOpen = jest.fn();
|
||||||
const mockCloseVegaDialog = jest.fn();
|
const mockCloseVegaDialog = jest.fn();
|
||||||
@ -49,10 +56,12 @@ const INITIAL_KEY = 'some-key';
|
|||||||
const rest = new RestConnector();
|
const rest = new RestConnector();
|
||||||
const jsonRpc = new JsonRpcConnector();
|
const jsonRpc = new JsonRpcConnector();
|
||||||
const view = new ViewConnector(INITIAL_KEY);
|
const view = new ViewConnector(INITIAL_KEY);
|
||||||
|
const injected = new InjectedConnector();
|
||||||
const connectors = {
|
const connectors = {
|
||||||
rest,
|
rest,
|
||||||
jsonRpc,
|
jsonRpc,
|
||||||
view,
|
view,
|
||||||
|
injected,
|
||||||
};
|
};
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
jest.clearAllMocks();
|
jest.clearAllMocks();
|
||||||
@ -105,7 +114,7 @@ describe('VegaConnectDialog', () => {
|
|||||||
expect(screen.getByTestId('connector-jsonRpc')).toHaveTextContent(
|
expect(screen.getByTestId('connector-jsonRpc')).toHaveTextContent(
|
||||||
'Connect Vega wallet'
|
'Connect Vega wallet'
|
||||||
);
|
);
|
||||||
expect(screen.getByTestId('connector-hosted')).toHaveTextContent(
|
expect(screen.getByTestId('connector-rest')).toHaveTextContent(
|
||||||
'Hosted Fairground wallet'
|
'Hosted Fairground wallet'
|
||||||
);
|
);
|
||||||
expect(screen.getByTestId('connector-view')).toHaveTextContent(
|
expect(screen.getByTestId('connector-view')).toHaveTextContent(
|
||||||
@ -113,6 +122,17 @@ describe('VegaConnectDialog', () => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('displays browser wallet option if detected on window object', async () => {
|
||||||
|
mockBrowserWallet();
|
||||||
|
render(generateJSX());
|
||||||
|
const list = await screen.findByTestId('connectors-list');
|
||||||
|
expect(list.children).toHaveLength(4);
|
||||||
|
expect(screen.getByTestId('connector-injected')).toHaveTextContent(
|
||||||
|
'Connect Web wallet'
|
||||||
|
);
|
||||||
|
clearBrowserWallet();
|
||||||
|
});
|
||||||
|
|
||||||
describe('RestConnector', () => {
|
describe('RestConnector', () => {
|
||||||
it('connects', async () => {
|
it('connects', async () => {
|
||||||
const spy = jest
|
const spy = jest
|
||||||
@ -229,17 +249,19 @@ describe('VegaConnectDialog', () => {
|
|||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
spyOnCheckCompat = jest
|
spyOnCheckCompat = jest
|
||||||
.spyOn(connectors.jsonRpc, 'checkCompat')
|
.spyOn(connectors.jsonRpc, 'checkCompat')
|
||||||
.mockImplementation(() => delayedResolve(true));
|
.mockImplementation(() => delayedResolve(true, delay));
|
||||||
spyOnGetChainId = jest
|
spyOnGetChainId = jest
|
||||||
.spyOn(connectors.jsonRpc, 'getChainId')
|
.spyOn(connectors.jsonRpc, 'getChainId')
|
||||||
.mockImplementation(() => delayedResolve({ chainID: mockChainId }));
|
.mockImplementation(() =>
|
||||||
|
delayedResolve({ chainID: mockChainId }, delay)
|
||||||
|
);
|
||||||
spyOnConnectWallet = jest
|
spyOnConnectWallet = jest
|
||||||
.spyOn(connectors.jsonRpc, 'connectWallet')
|
.spyOn(connectors.jsonRpc, 'connectWallet')
|
||||||
.mockImplementation(() => delayedResolve(null));
|
.mockImplementation(() => delayedResolve(null, delay));
|
||||||
spyOnConnect = jest
|
spyOnConnect = jest
|
||||||
.spyOn(connectors.jsonRpc, 'connect')
|
.spyOn(connectors.jsonRpc, 'connect')
|
||||||
.mockImplementation(() =>
|
.mockImplementation(() =>
|
||||||
delayedResolve([{ publicKey: 'pubkey', name: 'test key 1' }])
|
delayedResolve([{ publicKey: 'pubkey', name: 'test key 1' }], delay)
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -351,18 +373,6 @@ describe('VegaConnectDialog', () => {
|
|||||||
expect(screen.getByText('An unknown error occurred')).toBeInTheDocument();
|
expect(screen.getByText('An unknown error occurred')).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
function delayedResolve<T>(result: T): Promise<T> {
|
|
||||||
return new Promise((resolve) => {
|
|
||||||
setTimeout(() => resolve(result), delay);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function delayedReject<T>(result: T): Promise<T> {
|
|
||||||
return new Promise((_, reject) => {
|
|
||||||
setTimeout(() => reject(result), delay);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async function selectJsonRpc() {
|
async function selectJsonRpc() {
|
||||||
expect(await screen.findByRole('dialog')).toBeInTheDocument();
|
expect(await screen.findByRole('dialog')).toBeInTheDocument();
|
||||||
fireEvent.click(await screen.findByTestId('connector-jsonRpc'));
|
fireEvent.click(await screen.findByTestId('connector-jsonRpc'));
|
||||||
@ -439,4 +449,109 @@ describe('VegaConnectDialog', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('InjectedConnector', () => {
|
||||||
|
beforeAll(() => {
|
||||||
|
jest.useFakeTimers();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
jest.useRealTimers();
|
||||||
|
localStorage.clear();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
clearBrowserWallet();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('connects', async () => {
|
||||||
|
const delay = 100;
|
||||||
|
const vegaWindow = {
|
||||||
|
getChainId: jest.fn(() =>
|
||||||
|
delayedResolve({ chainID: mockChainId }, delay)
|
||||||
|
),
|
||||||
|
connectWallet: jest.fn(() => delayedResolve(null, delay)),
|
||||||
|
disconnectWallet: jest.fn(() => delayedResolve(undefined, delay)),
|
||||||
|
listKeys: jest.fn(() =>
|
||||||
|
delayedResolve(
|
||||||
|
{
|
||||||
|
keys: [{ name: 'test key', publicKey: '0x123' }],
|
||||||
|
},
|
||||||
|
100
|
||||||
|
)
|
||||||
|
),
|
||||||
|
};
|
||||||
|
mockBrowserWallet(vegaWindow);
|
||||||
|
render(generateJSX());
|
||||||
|
await selectInjected();
|
||||||
|
|
||||||
|
// Chain check
|
||||||
|
expect(screen.getByText('Verifying chain')).toBeInTheDocument();
|
||||||
|
expect(vegaWindow.getChainId).toHaveBeenCalled();
|
||||||
|
await act(async () => {
|
||||||
|
jest.advanceTimersByTime(delay);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Await user connect
|
||||||
|
expect(screen.getByText('Connecting...')).toBeInTheDocument();
|
||||||
|
expect(vegaWindow.connectWallet).toHaveBeenCalled();
|
||||||
|
await act(async () => {
|
||||||
|
jest.advanceTimersByTime(delay);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Connect (list keys)
|
||||||
|
expect(vegaWindow.listKeys).toHaveBeenCalled();
|
||||||
|
await act(async () => {
|
||||||
|
jest.advanceTimersByTime(delay);
|
||||||
|
});
|
||||||
|
expect(screen.getByText('Successfully connected')).toBeInTheDocument();
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
jest.advanceTimersByTime(CLOSE_DELAY);
|
||||||
|
});
|
||||||
|
expect(mockCloseVegaDialog).toHaveBeenCalledWith();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('handles invalid chain', async () => {
|
||||||
|
const delay = 100;
|
||||||
|
const invalidChain = 'invalid chain';
|
||||||
|
const vegaWindow = {
|
||||||
|
getChainId: jest.fn(() =>
|
||||||
|
delayedResolve({ chainID: invalidChain }, delay)
|
||||||
|
),
|
||||||
|
connectWallet: jest.fn(() => delayedResolve(null, delay)),
|
||||||
|
disconnectWallet: jest.fn(() => delayedResolve(undefined, delay)),
|
||||||
|
listKeys: jest.fn(() =>
|
||||||
|
delayedResolve(
|
||||||
|
{
|
||||||
|
keys: [{ name: 'test key', publicKey: '0x123' }],
|
||||||
|
},
|
||||||
|
100
|
||||||
|
)
|
||||||
|
),
|
||||||
|
};
|
||||||
|
mockBrowserWallet(vegaWindow);
|
||||||
|
render(generateJSX());
|
||||||
|
await selectInjected();
|
||||||
|
|
||||||
|
// Chain check
|
||||||
|
expect(screen.getByText('Verifying chain')).toBeInTheDocument();
|
||||||
|
expect(vegaWindow.getChainId).toHaveBeenCalled();
|
||||||
|
await act(async () => {
|
||||||
|
jest.advanceTimersByTime(delay);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(screen.getByText('Wrong network')).toBeInTheDocument();
|
||||||
|
expect(
|
||||||
|
screen.getByText(
|
||||||
|
new RegExp(`set your wallet network in your app to "${mockChainId}"`)
|
||||||
|
)
|
||||||
|
).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
async function selectInjected() {
|
||||||
|
expect(await screen.findByRole('dialog')).toBeInTheDocument();
|
||||||
|
fireEvent.click(await screen.findByTestId('connector-injected'));
|
||||||
|
}
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -3,45 +3,50 @@ import {
|
|||||||
Button,
|
Button,
|
||||||
Dialog,
|
Dialog,
|
||||||
FormGroup,
|
FormGroup,
|
||||||
Icon,
|
|
||||||
Input,
|
Input,
|
||||||
Link,
|
VegaIcon,
|
||||||
Loader,
|
VegaIconNames,
|
||||||
} from '@vegaprotocol/ui-toolkit';
|
} from '@vegaprotocol/ui-toolkit';
|
||||||
import { useCallback, useState } from 'react';
|
import { useCallback, useState } from 'react';
|
||||||
import type { WalletClientError } from '@vegaprotocol/wallet-client';
|
import type { WalletClientError } from '@vegaprotocol/wallet-client';
|
||||||
import { t } from '@vegaprotocol/i18n';
|
import { t } from '@vegaprotocol/i18n';
|
||||||
import type { VegaConnector } from '../connectors';
|
import type { VegaConnector } from '../connectors';
|
||||||
|
import { InjectedConnector } from '../connectors';
|
||||||
import { ViewConnector } from '../connectors';
|
import { ViewConnector } from '../connectors';
|
||||||
import { JsonRpcConnector, RestConnector } from '../connectors';
|
import { JsonRpcConnector, RestConnector } from '../connectors';
|
||||||
import { RestConnectorForm } from './rest-connector-form';
|
import { RestConnectorForm } from './rest-connector-form';
|
||||||
import { JsonRpcConnectorForm } from './json-rpc-connector-form';
|
import { JsonRpcConnectorForm } from './json-rpc-connector-form';
|
||||||
import {
|
import { Networks, useEnvironment } from '@vegaprotocol/environment';
|
||||||
Networks,
|
|
||||||
useEnvironment,
|
|
||||||
ExternalLinks,
|
|
||||||
} from '@vegaprotocol/environment';
|
|
||||||
import {
|
import {
|
||||||
ConnectDialogContent,
|
ConnectDialogContent,
|
||||||
ConnectDialogFooter,
|
ConnectDialogFooter,
|
||||||
ConnectDialogTitle,
|
ConnectDialogTitle,
|
||||||
} from './connect-dialog-elements';
|
} from './connect-dialog-elements';
|
||||||
import type { Status } from '../use-json-rpc-connect';
|
import type { Status as JsonRpcStatus } from '../use-json-rpc-connect';
|
||||||
|
import type { Status as InjectedStatus } from '../use-injected-connector';
|
||||||
import { useJsonRpcConnect } from '../use-json-rpc-connect';
|
import { useJsonRpcConnect } from '../use-json-rpc-connect';
|
||||||
import { ViewConnectorForm } from './view-connector-form';
|
import { ViewConnectorForm } from './view-connector-form';
|
||||||
import { useChainIdQuery } from './__generated__/ChainId';
|
import { useChainIdQuery } from './__generated__/ChainId';
|
||||||
import { useVegaWallet } from '../use-vega-wallet';
|
import { useVegaWallet } from '../use-vega-wallet';
|
||||||
|
import { useInjectedConnector } from '../use-injected-connector';
|
||||||
|
import { InjectedConnectorForm } from './injected-connector-form';
|
||||||
|
|
||||||
export const CLOSE_DELAY = 1700;
|
export const CLOSE_DELAY = 1700;
|
||||||
type Connectors = { [key: string]: VegaConnector };
|
type Connectors = { [key: string]: VegaConnector };
|
||||||
type WalletType = 'jsonRpc' | 'hosted' | 'view';
|
export type WalletType = 'injected' | 'jsonRpc' | 'rest' | 'view';
|
||||||
|
|
||||||
export interface VegaConnectDialogProps {
|
export interface VegaConnectDialogProps {
|
||||||
connectors: Connectors;
|
connectors: Connectors;
|
||||||
onChangeOpen?: (open: boolean) => void;
|
|
||||||
riskMessage?: React.ReactNode;
|
riskMessage?: React.ReactNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface VegaWalletDialogStore {
|
||||||
|
vegaWalletDialogOpen: boolean;
|
||||||
|
updateVegaWalletDialog: (open: boolean) => void;
|
||||||
|
openVegaWalletDialog: () => void;
|
||||||
|
closeVegaWalletDialog: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
export const useVegaWalletDialogStore = create<VegaWalletDialogStore>()(
|
export const useVegaWalletDialogStore = create<VegaWalletDialogStore>()(
|
||||||
(set) => ({
|
(set) => ({
|
||||||
vegaWalletDialogOpen: false,
|
vegaWalletDialogOpen: false,
|
||||||
@ -52,32 +57,20 @@ export const useVegaWalletDialogStore = create<VegaWalletDialogStore>()(
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
export interface VegaWalletDialogStore {
|
|
||||||
vegaWalletDialogOpen: boolean;
|
|
||||||
updateVegaWalletDialog: (open: boolean) => void;
|
|
||||||
openVegaWalletDialog: () => void;
|
|
||||||
closeVegaWalletDialog: () => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const VegaConnectDialog = ({
|
export const VegaConnectDialog = ({
|
||||||
connectors,
|
connectors,
|
||||||
onChangeOpen,
|
|
||||||
riskMessage,
|
riskMessage,
|
||||||
}: VegaConnectDialogProps) => {
|
}: VegaConnectDialogProps) => {
|
||||||
|
const { disconnect, acknowledgeNeeded } = useVegaWallet();
|
||||||
const vegaWalletDialogOpen = useVegaWalletDialogStore(
|
const vegaWalletDialogOpen = useVegaWalletDialogStore(
|
||||||
(store) => store.vegaWalletDialogOpen
|
(store) => store.vegaWalletDialogOpen
|
||||||
);
|
);
|
||||||
const updateVegaWalletDialog = useVegaWalletDialogStore(
|
const updateVegaWalletDialog = useVegaWalletDialogStore(
|
||||||
(store) => (open: boolean) => {
|
(store) => (open: boolean) => {
|
||||||
store.updateVegaWalletDialog(open);
|
store.updateVegaWalletDialog(open);
|
||||||
onChangeOpen?.(open);
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
const closeVegaWalletDialog = useVegaWalletDialogStore((store) => () => {
|
|
||||||
store.closeVegaWalletDialog();
|
|
||||||
onChangeOpen?.(false);
|
|
||||||
});
|
|
||||||
const { disconnect, acknowledgeNeeded } = useVegaWallet();
|
|
||||||
const onVegaWalletDialogChange = useCallback(
|
const onVegaWalletDialogChange = useCallback(
|
||||||
(open: boolean) => {
|
(open: boolean) => {
|
||||||
updateVegaWalletDialog(open);
|
updateVegaWalletDialog(open);
|
||||||
@ -88,41 +81,9 @@ export const VegaConnectDialog = ({
|
|||||||
[updateVegaWalletDialog, acknowledgeNeeded, disconnect]
|
[updateVegaWalletDialog, acknowledgeNeeded, disconnect]
|
||||||
);
|
);
|
||||||
|
|
||||||
const { data, error, loading } = useChainIdQuery();
|
// Ensure we have a chain Id so we can compare with wallet chain id.
|
||||||
|
// This value will already be in the cache, if it failed the app wont render
|
||||||
const renderContent = () => {
|
const { data } = useChainIdQuery();
|
||||||
if (error) {
|
|
||||||
return (
|
|
||||||
<ConnectDialogContent>
|
|
||||||
<ConnectDialogTitle>
|
|
||||||
{t('Could not retrieve chain id')}
|
|
||||||
</ConnectDialogTitle>
|
|
||||||
<ConnectDialogFooter />
|
|
||||||
</ConnectDialogContent>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (loading || !data) {
|
|
||||||
return (
|
|
||||||
<ConnectDialogContent>
|
|
||||||
<ConnectDialogTitle>{t('Fetching chain ID')}</ConnectDialogTitle>
|
|
||||||
<div className="flex justify-center items-center my-6">
|
|
||||||
<Loader />
|
|
||||||
</div>
|
|
||||||
<ConnectDialogFooter />
|
|
||||||
</ConnectDialogContent>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ConnectDialogContainer
|
|
||||||
connectors={connectors}
|
|
||||||
closeDialog={closeVegaWalletDialog}
|
|
||||||
appChainId={data.statistics.chainId}
|
|
||||||
riskMessage={riskMessage}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
@ -130,29 +91,35 @@ export const VegaConnectDialog = ({
|
|||||||
size="small"
|
size="small"
|
||||||
onChange={onVegaWalletDialogChange}
|
onChange={onVegaWalletDialogChange}
|
||||||
>
|
>
|
||||||
{renderContent()}
|
{data && (
|
||||||
|
<ConnectDialogContainer
|
||||||
|
connectors={connectors}
|
||||||
|
appChainId={data.statistics.chainId}
|
||||||
|
riskMessage={riskMessage}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</Dialog>
|
</Dialog>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const ConnectDialogContainer = ({
|
const ConnectDialogContainer = ({
|
||||||
connectors,
|
connectors,
|
||||||
closeDialog,
|
|
||||||
appChainId,
|
appChainId,
|
||||||
riskMessage,
|
riskMessage,
|
||||||
}: {
|
}: {
|
||||||
connectors: Connectors;
|
connectors: Connectors;
|
||||||
closeDialog: () => void;
|
|
||||||
appChainId: string;
|
appChainId: string;
|
||||||
riskMessage?: React.ReactNode;
|
riskMessage?: React.ReactNode;
|
||||||
}) => {
|
}) => {
|
||||||
const { VEGA_WALLET_URL, VEGA_ENV, HOSTED_WALLET_URL } = useEnvironment();
|
const { VEGA_WALLET_URL, VEGA_ENV, HOSTED_WALLET_URL } = useEnvironment();
|
||||||
|
const closeDialog = useVegaWalletDialogStore(
|
||||||
|
(store) => store.closeVegaWalletDialog
|
||||||
|
);
|
||||||
const [selectedConnector, setSelectedConnector] = useState<VegaConnector>();
|
const [selectedConnector, setSelectedConnector] = useState<VegaConnector>();
|
||||||
const [walletUrl, setWalletUrl] = useState(VEGA_WALLET_URL || '');
|
const [walletUrl, setWalletUrl] = useState(VEGA_WALLET_URL || '');
|
||||||
const [walletType, setWalletType] = useState<WalletType>();
|
|
||||||
const reset = useCallback(() => {
|
const reset = useCallback(() => {
|
||||||
setSelectedConnector(undefined);
|
setSelectedConnector(undefined);
|
||||||
setWalletType(undefined);
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const delayedOnConnect = useCallback(() => {
|
const delayedOnConnect = useCallback(() => {
|
||||||
@ -161,52 +128,59 @@ const ConnectDialogContainer = ({
|
|||||||
}, CLOSE_DELAY);
|
}, CLOSE_DELAY);
|
||||||
}, [closeDialog]);
|
}, [closeDialog]);
|
||||||
|
|
||||||
const { connect, ...jsonRpcState } = useJsonRpcConnect(delayedOnConnect);
|
const { connect: jsonRpcConnect, ...jsonRpcState } =
|
||||||
|
useJsonRpcConnect(delayedOnConnect);
|
||||||
|
const { connect: injectedConnect, ...injectedState } =
|
||||||
|
useInjectedConnector(delayedOnConnect);
|
||||||
|
|
||||||
const handleSelect = (type: WalletType, isHosted = false) => {
|
const handleSelect = (type: WalletType) => {
|
||||||
let connector;
|
const connector = connectors[type];
|
||||||
|
|
||||||
if (isHosted) {
|
// If type is rest user has selected the hosted wallet option. So here
|
||||||
// If the user has selected hosted wallet ensure that we are connecting to https://vega-hosted-wallet.on.fleek.co/
|
// we ensure that we are connecting to https://vega-hosted-wallet.on.fleek.co/
|
||||||
// otherwise use the default walletUrl or what has been put in the input
|
// otherwise use walletUrl which defaults to the localhost:1789
|
||||||
connector = connectors['rest'];
|
connector.url = type === 'rest' ? HOSTED_WALLET_URL : walletUrl;
|
||||||
connector.url = HOSTED_WALLET_URL || walletUrl;
|
|
||||||
} else {
|
|
||||||
connector = connectors[type];
|
|
||||||
connector.url = walletUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!connector) {
|
if (!connector) {
|
||||||
|
// we should never get here unless connectors are not configured correctly
|
||||||
throw new Error(`Connector type: ${type} not configured`);
|
throw new Error(`Connector type: ${type} not configured`);
|
||||||
}
|
}
|
||||||
|
|
||||||
setSelectedConnector(connector);
|
setSelectedConnector(connector);
|
||||||
setWalletType(type);
|
|
||||||
|
|
||||||
// Immediately connect on selection if jsonRpc is selected, we can't do this
|
// Immediately connect on selection if jsonRpc is selected, we can't do this
|
||||||
// for rest because we need to show an authentication form
|
// for rest because we need to show an authentication form
|
||||||
if (connector instanceof JsonRpcConnector) {
|
if (connector instanceof JsonRpcConnector) {
|
||||||
connect(connector, appChainId);
|
jsonRpcConnect(connector, appChainId);
|
||||||
|
} else if (connector instanceof InjectedConnector) {
|
||||||
|
injectedConnect(connector, appChainId);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return selectedConnector !== undefined && walletType !== undefined ? (
|
return (
|
||||||
<SelectedForm
|
<>
|
||||||
type={walletType}
|
<ConnectDialogContent>
|
||||||
connector={selectedConnector}
|
{selectedConnector !== undefined ? (
|
||||||
jsonRpcState={jsonRpcState}
|
<SelectedForm
|
||||||
onConnect={closeDialog}
|
connector={selectedConnector}
|
||||||
appChainId={appChainId}
|
jsonRpcState={jsonRpcState}
|
||||||
reset={reset}
|
injectedState={injectedState}
|
||||||
riskMessage={riskMessage}
|
onConnect={closeDialog}
|
||||||
/>
|
appChainId={appChainId}
|
||||||
) : (
|
reset={reset}
|
||||||
<ConnectorList
|
riskMessage={riskMessage}
|
||||||
walletUrl={walletUrl}
|
/>
|
||||||
setWalletUrl={setWalletUrl}
|
) : (
|
||||||
onSelect={handleSelect}
|
<ConnectorList
|
||||||
isMainnet={VEGA_ENV === Networks.MAINNET}
|
walletUrl={walletUrl}
|
||||||
/>
|
setWalletUrl={setWalletUrl}
|
||||||
|
onSelect={handleSelect}
|
||||||
|
isMainnet={VEGA_ENV === Networks.MAINNET}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</ConnectDialogContent>
|
||||||
|
<ConnectDialogFooter connector={selectedConnector} />
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -216,136 +190,129 @@ const ConnectorList = ({
|
|||||||
setWalletUrl,
|
setWalletUrl,
|
||||||
isMainnet,
|
isMainnet,
|
||||||
}: {
|
}: {
|
||||||
onSelect: (type: WalletType, isHosted?: boolean) => void;
|
onSelect: (type: WalletType) => void;
|
||||||
walletUrl: string;
|
walletUrl: string;
|
||||||
setWalletUrl: (value: string) => void;
|
setWalletUrl: (value: string) => void;
|
||||||
isMainnet: boolean;
|
isMainnet: boolean;
|
||||||
}) => {
|
}) => {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ConnectDialogContent>
|
<ConnectDialogTitle>{t('Connect')}</ConnectDialogTitle>
|
||||||
<ConnectDialogTitle>{t('Connect')}</ConnectDialogTitle>
|
<CustomUrlInput walletUrl={walletUrl} setWalletUrl={setWalletUrl} />
|
||||||
<CustomUrlInput walletUrl={walletUrl} setWalletUrl={setWalletUrl} />
|
<ul data-testid="connectors-list" className="mb-6">
|
||||||
<ul data-testid="connectors-list" className="mb-6">
|
<li className="mb-4 last:mb-0">
|
||||||
|
<ConnectionOption
|
||||||
|
type="jsonRpc"
|
||||||
|
text={t('Connect Vega wallet')}
|
||||||
|
onClick={() => onSelect('jsonRpc')}
|
||||||
|
/>
|
||||||
|
</li>
|
||||||
|
{'vega' in window && (
|
||||||
<li className="mb-4 last:mb-0">
|
<li className="mb-4 last:mb-0">
|
||||||
<ConnectionOption
|
<ConnectionOption
|
||||||
type="jsonRpc"
|
type="injected"
|
||||||
text={t('Connect Vega wallet')}
|
text={t('Connect Web wallet')}
|
||||||
onClick={() => onSelect('jsonRpc')}
|
onClick={() => onSelect('injected')}
|
||||||
/>
|
/>
|
||||||
</li>
|
</li>
|
||||||
{!isMainnet && (
|
)}
|
||||||
<li className="mb-4 last:mb-0">
|
{!isMainnet && (
|
||||||
<ConnectionOption
|
|
||||||
type="hosted"
|
|
||||||
text={t('Hosted Fairground wallet')}
|
|
||||||
onClick={() => onSelect('hosted', true)}
|
|
||||||
/>
|
|
||||||
</li>
|
|
||||||
)}
|
|
||||||
<li className="mb-4 last:mb-0">
|
<li className="mb-4 last:mb-0">
|
||||||
<div className="my-4 text-center text-vega-dark-400">{t('OR')}</div>
|
|
||||||
<ConnectionOption
|
<ConnectionOption
|
||||||
type="view"
|
type="rest"
|
||||||
text={t('View as vega user')}
|
text={t('Hosted Fairground wallet')}
|
||||||
onClick={() => onSelect('view')}
|
onClick={() => onSelect('rest')}
|
||||||
/>
|
/>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
)}
|
||||||
</ConnectDialogContent>
|
<li className="mb-4 last:mb-0">
|
||||||
<ConnectDialogFooter />
|
<div className="my-4 text-center">{t('OR')}</div>
|
||||||
|
<ConnectionOption
|
||||||
|
type="view"
|
||||||
|
text={t('View as vega user')}
|
||||||
|
onClick={() => onSelect('view')}
|
||||||
|
/>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const SelectedForm = ({
|
const SelectedForm = ({
|
||||||
type,
|
|
||||||
connector,
|
connector,
|
||||||
appChainId,
|
appChainId,
|
||||||
jsonRpcState,
|
jsonRpcState,
|
||||||
|
injectedState,
|
||||||
reset,
|
reset,
|
||||||
onConnect,
|
onConnect,
|
||||||
riskMessage,
|
riskMessage,
|
||||||
}: {
|
}: {
|
||||||
type: WalletType;
|
|
||||||
connector: VegaConnector;
|
connector: VegaConnector;
|
||||||
appChainId: string;
|
appChainId: string;
|
||||||
jsonRpcState: {
|
jsonRpcState: {
|
||||||
status: Status;
|
status: JsonRpcStatus;
|
||||||
error: WalletClientError | null;
|
error: WalletClientError | null;
|
||||||
};
|
};
|
||||||
|
injectedState: {
|
||||||
|
status: InjectedStatus;
|
||||||
|
error: Error | null;
|
||||||
|
};
|
||||||
reset: () => void;
|
reset: () => void;
|
||||||
onConnect: () => void;
|
onConnect: () => void;
|
||||||
riskMessage?: React.ReactNode;
|
riskMessage?: React.ReactNode;
|
||||||
}) => {
|
}) => {
|
||||||
|
if (connector instanceof InjectedConnector) {
|
||||||
|
return (
|
||||||
|
<InjectedConnectorForm
|
||||||
|
status={injectedState.status}
|
||||||
|
error={injectedState.error}
|
||||||
|
onConnect={onConnect}
|
||||||
|
appChainId={appChainId}
|
||||||
|
reset={reset}
|
||||||
|
riskMessage={riskMessage}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (connector instanceof RestConnector) {
|
if (connector instanceof RestConnector) {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ConnectDialogContent>
|
<button
|
||||||
<button
|
onClick={reset}
|
||||||
onClick={reset}
|
className="absolute p-2 top-0 left-0 md:top-2 md:left-2"
|
||||||
className="absolute p-2 top-0 left-0 md:top-2 md:left-2"
|
data-testid="back-button"
|
||||||
data-testid="back-button"
|
>
|
||||||
>
|
<VegaIcon name={VegaIconNames.CHEVRON_LEFT} />
|
||||||
<Icon name={'chevron-left'} ariaLabel="back" size={4} />
|
</button>
|
||||||
</button>
|
<ConnectDialogTitle>{t('Connect')}</ConnectDialogTitle>
|
||||||
<ConnectDialogTitle>{t('Connect')}</ConnectDialogTitle>
|
<div className="mb-2">
|
||||||
<div className="mb-2">
|
<RestConnectorForm connector={connector} onConnect={onConnect} />
|
||||||
<RestConnectorForm connector={connector} onConnect={onConnect} />
|
</div>
|
||||||
</div>
|
|
||||||
</ConnectDialogContent>
|
|
||||||
{type === 'hosted' ? (
|
|
||||||
<ConnectDialogFooter>
|
|
||||||
<p className="text-center">
|
|
||||||
{t('For demo purposes get a ')}
|
|
||||||
<Link
|
|
||||||
href={ExternalLinks.VEGA_WALLET_HOSTED_URL}
|
|
||||||
target="_blank"
|
|
||||||
rel="noreferrer"
|
|
||||||
>
|
|
||||||
{t('hosted wallet')}
|
|
||||||
</Link>
|
|
||||||
{t(', or for the real experience create a wallet in the ')}
|
|
||||||
<Link href={ExternalLinks.VEGA_WALLET_URL}>
|
|
||||||
{t('Vega wallet app')}
|
|
||||||
</Link>
|
|
||||||
</p>
|
|
||||||
</ConnectDialogFooter>
|
|
||||||
) : (
|
|
||||||
<ConnectDialogFooter />
|
|
||||||
)}
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (connector instanceof JsonRpcConnector) {
|
if (connector instanceof JsonRpcConnector) {
|
||||||
return (
|
return (
|
||||||
<ConnectDialogContent>
|
<JsonRpcConnectorForm
|
||||||
<JsonRpcConnectorForm
|
connector={connector}
|
||||||
connector={connector}
|
status={jsonRpcState.status}
|
||||||
status={jsonRpcState.status}
|
error={jsonRpcState.error}
|
||||||
error={jsonRpcState.error}
|
onConnect={onConnect}
|
||||||
onConnect={onConnect}
|
appChainId={appChainId}
|
||||||
appChainId={appChainId}
|
reset={reset}
|
||||||
reset={reset}
|
riskMessage={riskMessage}
|
||||||
riskMessage={riskMessage}
|
/>
|
||||||
/>
|
|
||||||
</ConnectDialogContent>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (connector instanceof ViewConnector) {
|
if (connector instanceof ViewConnector) {
|
||||||
return (
|
return (
|
||||||
<>
|
<ViewConnectorForm
|
||||||
<ConnectDialogContent>
|
connector={connector}
|
||||||
<ViewConnectorForm
|
onConnect={onConnect}
|
||||||
connector={connector}
|
reset={reset}
|
||||||
onConnect={onConnect}
|
/>
|
||||||
reset={reset}
|
|
||||||
/>
|
|
||||||
</ConnectDialogContent>
|
|
||||||
<ConnectDialogFooter />
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -366,12 +333,12 @@ const ConnectionOption = ({
|
|||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
size="lg"
|
size="lg"
|
||||||
fill={true}
|
fill={true}
|
||||||
variant={['hosted', 'view'].includes(type) ? 'default' : 'primary'}
|
variant={['rest', 'view'].includes(type) ? 'default' : 'primary'}
|
||||||
data-testid={`connector-${type}`}
|
data-testid={`connector-${type}`}
|
||||||
>
|
>
|
||||||
<span className="-mx-6 flex text-left justify-between items-center">
|
<span className="-mx-10 flex text-left justify-between items-center">
|
||||||
{text}
|
{text}
|
||||||
<Icon name="chevron-right" />
|
<VegaIcon name={VegaIconNames.ARROW_RIGHT} />
|
||||||
</span>
|
</span>
|
||||||
</Button>
|
</Button>
|
||||||
);
|
);
|
||||||
@ -387,9 +354,7 @@ const CustomUrlInput = ({
|
|||||||
const [urlInputExpanded, setUrlInputExpanded] = useState(false);
|
const [urlInputExpanded, setUrlInputExpanded] = useState(false);
|
||||||
return urlInputExpanded ? (
|
return urlInputExpanded ? (
|
||||||
<>
|
<>
|
||||||
<p className="mb-2 text-neutral-600 dark:text-neutral-400">
|
<p className="mb-2">{t('Custom wallet location')}</p>
|
||||||
{t('Custom wallet location')}
|
|
||||||
</p>
|
|
||||||
<FormGroup
|
<FormGroup
|
||||||
labelFor="wallet-url"
|
labelFor="wallet-url"
|
||||||
label={t('Custom wallet location')}
|
label={t('Custom wallet location')}
|
||||||
@ -401,12 +366,10 @@ const CustomUrlInput = ({
|
|||||||
name="wallet-url"
|
name="wallet-url"
|
||||||
/>
|
/>
|
||||||
</FormGroup>
|
</FormGroup>
|
||||||
<p className="mb-2 text-neutral-600 dark:text-neutral-400">
|
<p className="mb-2">{t('Choose wallet app to connect')}</p>
|
||||||
{t('Choose wallet app to connect')}
|
|
||||||
</p>
|
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<p className="mb-6 text-neutral-600 dark:text-neutral-400">
|
<p className="mb-6">
|
||||||
{t(
|
{t(
|
||||||
'Choose wallet app to connect, or to change port or server URL enter a '
|
'Choose wallet app to connect, or to change port or server URL enter a '
|
||||||
)}
|
)}
|
||||||
|
153
libs/wallet/src/connect-dialog/injected-connector-form.tsx
Normal file
153
libs/wallet/src/connect-dialog/injected-connector-form.tsx
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
import { t } from '@vegaprotocol/i18n';
|
||||||
|
import { Status } from '../use-injected-connector';
|
||||||
|
import { ConnectDialogTitle } from './connect-dialog-elements';
|
||||||
|
import type { ReactNode } from 'react';
|
||||||
|
import {
|
||||||
|
Button,
|
||||||
|
ButtonLink,
|
||||||
|
Diamond,
|
||||||
|
Loader,
|
||||||
|
Tick,
|
||||||
|
} from '@vegaprotocol/ui-toolkit';
|
||||||
|
import { setAcknowledged } from '../storage';
|
||||||
|
import { useVegaWallet } from '../use-vega-wallet';
|
||||||
|
|
||||||
|
export const InjectedConnectorForm = ({
|
||||||
|
status,
|
||||||
|
onConnect,
|
||||||
|
riskMessage,
|
||||||
|
appChainId,
|
||||||
|
reset,
|
||||||
|
error,
|
||||||
|
}: {
|
||||||
|
// connector: JsonRpcConnector;
|
||||||
|
appChainId: string;
|
||||||
|
status: Status;
|
||||||
|
error: Error | null;
|
||||||
|
onConnect: () => void;
|
||||||
|
reset: () => void;
|
||||||
|
riskMessage?: React.ReactNode;
|
||||||
|
}) => {
|
||||||
|
const { disconnect } = useVegaWallet();
|
||||||
|
|
||||||
|
if (status === Status.Idle) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status === Status.Error) {
|
||||||
|
return <Error error={error} appChainId={appChainId} onTryAgain={reset} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status === Status.GettingChainId) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ConnectDialogTitle>{t('Verifying chain')}</ConnectDialogTitle>
|
||||||
|
<Center>
|
||||||
|
<Loader />
|
||||||
|
</Center>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status === Status.Connected) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ConnectDialogTitle>{t('Successfully connected')}</ConnectDialogTitle>
|
||||||
|
<Center>
|
||||||
|
<Tick />
|
||||||
|
</Center>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status === Status.Connecting) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ConnectDialogTitle>{t('Connecting...')}</ConnectDialogTitle>
|
||||||
|
<Center>
|
||||||
|
<Diamond />
|
||||||
|
</Center>
|
||||||
|
<p className="text-center">
|
||||||
|
{t(
|
||||||
|
"Approve the connection from your Vega wallet app. If you have multiple wallets you'll need to choose which to connect with."
|
||||||
|
)}
|
||||||
|
</p>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status === Status.AcknowledgeNeeded) {
|
||||||
|
const setConnection = () => {
|
||||||
|
setAcknowledged();
|
||||||
|
onConnect();
|
||||||
|
};
|
||||||
|
const handleDisagree = () => {
|
||||||
|
disconnect();
|
||||||
|
onConnect(); // this is dialog closing
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ConnectDialogTitle>{t('Understand the risk')}</ConnectDialogTitle>
|
||||||
|
{riskMessage}
|
||||||
|
<div className="grid grid-cols-2 gap-5">
|
||||||
|
<div>
|
||||||
|
<Button onClick={handleDisagree} fill>
|
||||||
|
{t('Cancel')}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<Button onClick={setConnection} variant="primary" fill>
|
||||||
|
{t('I agree')}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const Center = ({ children }: { children: ReactNode }) => {
|
||||||
|
return (
|
||||||
|
<div className="flex justify-center items-center my-6">{children}</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const Error = ({
|
||||||
|
error,
|
||||||
|
appChainId,
|
||||||
|
onTryAgain,
|
||||||
|
}: {
|
||||||
|
error: Error | null;
|
||||||
|
appChainId: string;
|
||||||
|
onTryAgain: () => void;
|
||||||
|
}) => {
|
||||||
|
let title = t('Something went wrong');
|
||||||
|
let text: ReactNode | undefined = t('An unknown error occurred');
|
||||||
|
const tryAgain: ReactNode | null = (
|
||||||
|
<p className="text-center">
|
||||||
|
<ButtonLink onClick={onTryAgain}>{t('Try again')}</ButtonLink>
|
||||||
|
</p>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
if (error.message === 'Invalid chain') {
|
||||||
|
title = t('Wrong network');
|
||||||
|
text = t(
|
||||||
|
'To complete your wallet connection, set your wallet network in your app to "%s".',
|
||||||
|
appChainId
|
||||||
|
);
|
||||||
|
} else if (error.message === 'window.vega not found') {
|
||||||
|
title = t('No wallet detected');
|
||||||
|
text = t('Vega browser extension not installed');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ConnectDialogTitle>{title}</ConnectDialogTitle>
|
||||||
|
<p className="text-center mb-2 first-letter:uppercase">{text}</p>
|
||||||
|
{tryAgain}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
@ -1,21 +1,83 @@
|
|||||||
import type { VegaConnector } from './vega-connector';
|
import { clearConfig, setConfig } from '../storage';
|
||||||
|
import type { Transaction, VegaConnector } from './vega-connector';
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface Vega {
|
||||||
|
getChainId: () => Promise<{ chainID: string }>;
|
||||||
|
connectWallet: () => Promise<null>;
|
||||||
|
disconnectWallet: () => Promise<void>;
|
||||||
|
listKeys: () => Promise<{
|
||||||
|
keys: Array<{ name: string; publicKey: string }>;
|
||||||
|
}>;
|
||||||
|
sendTransaction: (params: {
|
||||||
|
publicKey: string;
|
||||||
|
transaction: Transaction;
|
||||||
|
sendingMode: 'TYPE_SYNC';
|
||||||
|
}) => Promise<{
|
||||||
|
receivedAt: string;
|
||||||
|
sentAt: string;
|
||||||
|
transaction: {
|
||||||
|
from: {
|
||||||
|
pubKey: string;
|
||||||
|
};
|
||||||
|
inputData: string;
|
||||||
|
pow: {
|
||||||
|
tid: string;
|
||||||
|
nonce: string;
|
||||||
|
};
|
||||||
|
signature: {
|
||||||
|
algo: string;
|
||||||
|
value: string;
|
||||||
|
version: number;
|
||||||
|
};
|
||||||
|
version: number;
|
||||||
|
};
|
||||||
|
transactionHash: string;
|
||||||
|
}>;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Window {
|
||||||
|
vega: Vega;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Dummy injected connector that we may use when browser wallet is implemented
|
|
||||||
*/
|
|
||||||
export class InjectedConnector implements VegaConnector {
|
export class InjectedConnector implements VegaConnector {
|
||||||
description = 'Connects using the Vega wallet browser extension';
|
description = 'Connects using the Vega wallet browser extension';
|
||||||
|
|
||||||
|
async getChainId() {
|
||||||
|
return window.vega.getChainId();
|
||||||
|
}
|
||||||
|
|
||||||
|
connectWallet() {
|
||||||
|
return window.vega.connectWallet();
|
||||||
|
}
|
||||||
|
|
||||||
async connect() {
|
async connect() {
|
||||||
return [{ publicKey: '0x123', name: 'text key' }];
|
const res = await window.vega.listKeys();
|
||||||
|
setConfig({
|
||||||
|
connector: 'injected',
|
||||||
|
token: null, // no token required for injected
|
||||||
|
url: null, // no url for injected
|
||||||
|
});
|
||||||
|
return res.keys;
|
||||||
}
|
}
|
||||||
|
|
||||||
async disconnect() {
|
disconnect() {
|
||||||
return;
|
clearConfig();
|
||||||
|
return window.vega.disconnectWallet();
|
||||||
}
|
}
|
||||||
|
|
||||||
// @ts-ignore injected connector is not implemented
|
async sendTx(pubKey: string, transaction: Transaction) {
|
||||||
sendTx() {
|
const result = await window.vega.sendTransaction({
|
||||||
throw new Error('Not implemented');
|
publicKey: pubKey,
|
||||||
|
transaction,
|
||||||
|
sendingMode: 'TYPE_SYNC' as const,
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
transactionHash: result.transactionHash,
|
||||||
|
receivedAt: result.receivedAt,
|
||||||
|
sentAt: result.sentAt,
|
||||||
|
signature: result.transaction.signature.value,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -411,7 +411,7 @@ export interface PubKey {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface VegaConnector {
|
export interface VegaConnector {
|
||||||
url: string | null;
|
url?: string | null;
|
||||||
|
|
||||||
/** Connect to wallet and return keys */
|
/** Connect to wallet and return keys */
|
||||||
connect(): Promise<PubKey[] | null>;
|
connect(): Promise<PubKey[] | null>;
|
||||||
|
@ -106,7 +106,6 @@ export const VegaWalletProvider = ({ children }: VegaWalletProviderProps) => {
|
|||||||
if (!connector.current) {
|
if (!connector.current) {
|
||||||
throw new Error('No connector');
|
throw new Error('No connector');
|
||||||
}
|
}
|
||||||
|
|
||||||
return connector.current.sendTx(pubkey, transaction);
|
return connector.current.sendTx(pubkey, transaction);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
@ -2,7 +2,7 @@ import { LocalStorage } from '@vegaprotocol/utils';
|
|||||||
|
|
||||||
interface ConnectorConfig {
|
interface ConnectorConfig {
|
||||||
token: string | null;
|
token: string | null;
|
||||||
connector: 'rest' | 'jsonRpc' | 'view';
|
connector: 'injected' | 'rest' | 'jsonRpc' | 'view';
|
||||||
url: string | null;
|
url: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
39
libs/wallet/src/test-helpers.ts
Normal file
39
libs/wallet/src/test-helpers.ts
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
export function mockBrowserWallet(overrides?: Partial<Vega>) {
|
||||||
|
const vega: Vega = {
|
||||||
|
getChainId: jest.fn().mockReturnValue(Promise.resolve({ chainID: '1' })),
|
||||||
|
connectWallet: jest.fn().mockReturnValue(Promise.resolve(null)),
|
||||||
|
disconnectWallet: jest.fn().mockReturnValue(Promise.resolve()),
|
||||||
|
listKeys: jest
|
||||||
|
.fn()
|
||||||
|
.mockReturnValue({ keys: [{ name: 'test key', publicKey: '0x123' }] }),
|
||||||
|
sendTransaction: jest.fn().mockReturnValue({
|
||||||
|
code: 1,
|
||||||
|
data: '',
|
||||||
|
height: '1',
|
||||||
|
log: '',
|
||||||
|
success: true,
|
||||||
|
txHash: '0x123',
|
||||||
|
}),
|
||||||
|
...overrides,
|
||||||
|
};
|
||||||
|
// @ts-ignore globalThis has no index signature
|
||||||
|
globalThis.vega = vega;
|
||||||
|
return vega;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function clearBrowserWallet() {
|
||||||
|
// @ts-ignore no index signature on globalThis
|
||||||
|
delete globalThis['vega'];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function delayedResolve<T>(result: T, delay = 0): Promise<T> {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
setTimeout(() => resolve(result), delay);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function delayedReject<T>(result: T, delay = 0): Promise<T> {
|
||||||
|
return new Promise((_, reject) => {
|
||||||
|
setTimeout(() => reject(result), delay);
|
||||||
|
});
|
||||||
|
}
|
104
libs/wallet/src/use-injected-connector.spec.tsx
Normal file
104
libs/wallet/src/use-injected-connector.spec.tsx
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
import { act, renderHook, waitFor } from '@testing-library/react';
|
||||||
|
import { Status, useInjectedConnector } from './use-injected-connector';
|
||||||
|
import type { ReactNode } from 'react';
|
||||||
|
import { VegaWalletProvider } from './provider';
|
||||||
|
import { InjectedConnector } from './connectors';
|
||||||
|
import { mockBrowserWallet } from './test-helpers';
|
||||||
|
import { useEnvironment } from '@vegaprotocol/environment';
|
||||||
|
import { Networks } from '@vegaprotocol/environment';
|
||||||
|
|
||||||
|
jest.mock('@vegaprotocol/environment');
|
||||||
|
|
||||||
|
const setup = (callback = jest.fn()) => {
|
||||||
|
const wrapper = ({ children }: { children: ReactNode }) => (
|
||||||
|
<VegaWalletProvider>{children}</VegaWalletProvider>
|
||||||
|
);
|
||||||
|
return renderHook(() => useInjectedConnector(callback), { wrapper });
|
||||||
|
};
|
||||||
|
|
||||||
|
const injected = new InjectedConnector();
|
||||||
|
|
||||||
|
describe('useInjectedConnector', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
// @ts-ignore useEnvironment has been mocked
|
||||||
|
useEnvironment.mockImplementation(() => ({ VEGA_ENV: Networks.TESTNET }));
|
||||||
|
});
|
||||||
|
it('attempts connection', async () => {
|
||||||
|
const { result } = setup();
|
||||||
|
expect(typeof result.current.connect).toBe('function');
|
||||||
|
expect(result.current.status).toBe(Status.Idle);
|
||||||
|
expect(result.current.error).toBe(null);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('errors if vega not injected', async () => {
|
||||||
|
const { result } = setup();
|
||||||
|
await act(async () => {
|
||||||
|
result.current.connect(injected, '1');
|
||||||
|
});
|
||||||
|
expect(result.current.error?.message).toBe('window.vega not found');
|
||||||
|
expect(result.current.status).toBe(Status.Error);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('errors if chain ids dont match', async () => {
|
||||||
|
mockBrowserWallet();
|
||||||
|
const { result } = setup();
|
||||||
|
await act(async () => {
|
||||||
|
result.current.connect(injected, '2'); // default mock chainId is '1'
|
||||||
|
});
|
||||||
|
expect(result.current.error?.message).toBe('Invalid chain');
|
||||||
|
expect(result.current.status).toBe(Status.Error);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('errors if connection throws', async () => {
|
||||||
|
const callback = jest.fn();
|
||||||
|
mockBrowserWallet({
|
||||||
|
getChainId: () => Promise.reject('failed'),
|
||||||
|
});
|
||||||
|
const { result } = setup(callback);
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
result.current.connect(injected, '1'); // default mock chainId is '1'
|
||||||
|
});
|
||||||
|
expect(result.current.status).toBe(Status.Error);
|
||||||
|
expect(result.current.error?.message).toBe('injected connection failed');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('connects', async () => {
|
||||||
|
const callback = jest.fn();
|
||||||
|
const vega = mockBrowserWallet();
|
||||||
|
const { result } = setup(callback);
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
result.current.connect(injected, '1'); // default mock chainId is '1'
|
||||||
|
});
|
||||||
|
expect(result.current.status).toBe(Status.GettingChainId);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(vega.connectWallet).toHaveBeenCalled();
|
||||||
|
expect(vega.listKeys).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.current.status).toBe(Status.Connected);
|
||||||
|
expect(callback).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('connects when aknowledgement required', async () => {
|
||||||
|
const callback = jest.fn();
|
||||||
|
// @ts-ignore useEnvironment has been mocked
|
||||||
|
useEnvironment.mockImplementation(() => ({ VEGA_ENV: Networks.MAINNET }));
|
||||||
|
|
||||||
|
const vega = mockBrowserWallet();
|
||||||
|
const { result } = setup(callback);
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
result.current.connect(injected, '1'); // default mock chainId is '1'
|
||||||
|
});
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(vega.listKeys).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.current.status).toBe(Status.AcknowledgeNeeded);
|
||||||
|
expect(callback).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
61
libs/wallet/src/use-injected-connector.ts
Normal file
61
libs/wallet/src/use-injected-connector.ts
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
import { useCallback, useState } from 'react';
|
||||||
|
import type { InjectedConnector } from './connectors';
|
||||||
|
import { useVegaWallet } from './use-vega-wallet';
|
||||||
|
|
||||||
|
export enum Status {
|
||||||
|
Idle = 'Idle',
|
||||||
|
GettingChainId = 'GettingChainId',
|
||||||
|
Connecting = 'Connecting',
|
||||||
|
Connected = 'Connected',
|
||||||
|
Error = 'Error',
|
||||||
|
AcknowledgeNeeded = 'AcknowledgeNeeded',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useInjectedConnector = (onConnect: () => void) => {
|
||||||
|
const { connect, acknowledgeNeeded } = useVegaWallet();
|
||||||
|
const [status, setStatus] = useState(Status.Idle);
|
||||||
|
const [error, setError] = useState<Error | null>(null);
|
||||||
|
|
||||||
|
const attemptConnect = useCallback(
|
||||||
|
async (connector: InjectedConnector, appChainId: string) => {
|
||||||
|
try {
|
||||||
|
if (!('vega' in window)) {
|
||||||
|
throw new Error('window.vega not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
setStatus(Status.GettingChainId);
|
||||||
|
|
||||||
|
const { chainID } = await connector.getChainId();
|
||||||
|
|
||||||
|
if (chainID !== appChainId) {
|
||||||
|
throw new Error('Invalid chain');
|
||||||
|
}
|
||||||
|
|
||||||
|
setStatus(Status.Connecting);
|
||||||
|
await connector.connectWallet(); // authorize wallet
|
||||||
|
await connect(connector); // connect with keys
|
||||||
|
|
||||||
|
if (acknowledgeNeeded) {
|
||||||
|
setStatus(Status.AcknowledgeNeeded);
|
||||||
|
} else {
|
||||||
|
setStatus(Status.Connected);
|
||||||
|
onConnect();
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
if (err instanceof Error) {
|
||||||
|
setError(err);
|
||||||
|
} else {
|
||||||
|
setError(new Error('injected connection failed'));
|
||||||
|
}
|
||||||
|
setStatus(Status.Error);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[acknowledgeNeeded, connect, onConnect]
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
status,
|
||||||
|
error,
|
||||||
|
connect: attemptConnect,
|
||||||
|
};
|
||||||
|
};
|
@ -11,7 +11,6 @@ export enum Status {
|
|||||||
GettingChainId = 'GettingChainId',
|
GettingChainId = 'GettingChainId',
|
||||||
Connecting = 'Connecting',
|
Connecting = 'Connecting',
|
||||||
GettingPerms = 'GettingPerms',
|
GettingPerms = 'GettingPerms',
|
||||||
ListingKeys = 'ListingKeys',
|
|
||||||
Connected = 'Connected',
|
Connected = 'Connected',
|
||||||
Error = 'Error',
|
Error = 'Error',
|
||||||
AcknowledgeNeeded = 'AcknowledgeNeeded',
|
AcknowledgeNeeded = 'AcknowledgeNeeded',
|
||||||
|
Loading…
Reference in New Issue
Block a user