2023-12-08 09:14:59 +00:00
|
|
|
import { useEffect } from 'react';
|
2023-12-08 14:48:12 +00:00
|
|
|
import { matchPath, useLocation } from 'react-router-dom';
|
2023-09-14 19:10:33 +00:00
|
|
|
import { Dialog, Intent } from '@vegaprotocol/ui-toolkit';
|
2024-03-01 14:25:56 +00:00
|
|
|
import {
|
|
|
|
ConnectionStatus,
|
|
|
|
useConnect,
|
|
|
|
useWallet,
|
|
|
|
Links,
|
|
|
|
RiskAck,
|
|
|
|
ConnectionOptionRecord,
|
|
|
|
ConnectionOptionDefault,
|
|
|
|
} from '@vegaprotocol/wallet-react';
|
|
|
|
import { ConnectorErrors } from '@vegaprotocol/wallet';
|
|
|
|
import { ensureSuffix } from '@vegaprotocol/utils';
|
2023-11-16 03:10:39 +00:00
|
|
|
import { useT } from '../../lib/use-t';
|
2023-12-08 14:48:12 +00:00
|
|
|
import { Routes } from '../../lib/links';
|
2023-12-08 09:14:59 +00:00
|
|
|
import { WelcomeDialogContent } from './welcome-dialog-content';
|
2024-03-01 14:25:56 +00:00
|
|
|
import {
|
|
|
|
useOnboardingStore,
|
|
|
|
type OnboardingDialog,
|
|
|
|
} from '../../stores/onboarding';
|
|
|
|
import { RiskAckContent } from '../risk-ack-content';
|
2023-12-08 14:48:12 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A list of paths on which the welcome dialog should be omitted.
|
|
|
|
*/
|
|
|
|
const OMIT_ON_LIST = [ensureSuffix(Routes.REFERRALS, '/*')];
|
2022-12-13 13:31:28 +00:00
|
|
|
|
|
|
|
export const WelcomeDialog = () => {
|
2023-12-08 14:48:12 +00:00
|
|
|
const { pathname } = useLocation();
|
2023-09-01 09:00:20 +00:00
|
|
|
const dismissed = useOnboardingStore((store) => store.dismissed);
|
2024-03-01 14:25:56 +00:00
|
|
|
const dialog = useOnboardingStore((store) => store.dialog);
|
2023-09-14 19:10:33 +00:00
|
|
|
const dismiss = useOnboardingStore((store) => store.dismiss);
|
2024-03-01 14:25:56 +00:00
|
|
|
const setDialog = useOnboardingStore((store) => store.setDialog);
|
2023-09-22 09:04:02 +00:00
|
|
|
|
2023-12-08 09:14:59 +00:00
|
|
|
useEffect(() => {
|
2023-12-08 14:48:12 +00:00
|
|
|
const shouldOmit = OMIT_ON_LIST.map((path) =>
|
|
|
|
matchPath(path, pathname)
|
|
|
|
).some((m) => !!m);
|
|
|
|
|
|
|
|
if (dismissed || shouldOmit) return;
|
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
setDialog('intro');
|
|
|
|
}, [dismissed, pathname, setDialog]);
|
2023-09-22 09:04:02 +00:00
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
const onClose = () => {
|
|
|
|
if (dialog === 'connect' || dialog === 'risk') {
|
|
|
|
setDialog('intro');
|
|
|
|
} else {
|
|
|
|
setDialog('inactive');
|
|
|
|
dismiss();
|
|
|
|
}
|
|
|
|
};
|
2023-01-02 16:01:06 +00:00
|
|
|
|
2023-09-14 19:10:33 +00:00
|
|
|
return (
|
2022-12-13 13:31:28 +00:00
|
|
|
<Dialog
|
2024-03-01 14:25:56 +00:00
|
|
|
open={dismissed ? false : dialog !== 'inactive'}
|
2023-08-02 15:34:04 +00:00
|
|
|
size="medium"
|
2023-09-22 09:04:02 +00:00
|
|
|
onChange={onClose}
|
2023-08-02 15:34:04 +00:00
|
|
|
intent={Intent.None}
|
|
|
|
dataTestId="welcome-dialog"
|
2022-12-13 13:31:28 +00:00
|
|
|
>
|
2024-03-01 14:25:56 +00:00
|
|
|
<DialogStepSwitch
|
|
|
|
dialog={dialog}
|
|
|
|
onConnect={() => {
|
|
|
|
setTimeout(() => setDialog('intro'), 1000);
|
|
|
|
}}
|
|
|
|
/>
|
2022-12-13 13:31:28 +00:00
|
|
|
</Dialog>
|
2023-09-14 19:10:33 +00:00
|
|
|
);
|
2022-12-13 13:31:28 +00:00
|
|
|
};
|
2024-03-01 14:25:56 +00:00
|
|
|
|
|
|
|
const DialogStepSwitch = ({
|
|
|
|
dialog,
|
|
|
|
onConnect,
|
|
|
|
}: {
|
|
|
|
dialog: OnboardingDialog;
|
|
|
|
onConnect: () => void;
|
|
|
|
}) => {
|
|
|
|
const accept = useOnboardingStore((store) => store.acceptRisk);
|
|
|
|
const reject = useOnboardingStore((store) => store.rejectRisk);
|
|
|
|
const setDialog = useOnboardingStore((store) => store.setDialog);
|
|
|
|
|
|
|
|
if (dialog === 'intro') {
|
|
|
|
return <WelcomeDialogContent />;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dialog === 'connect') {
|
|
|
|
return <OnboardConnectionOptions onConnect={onConnect} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dialog === 'risk') {
|
|
|
|
return (
|
|
|
|
<RiskAck
|
|
|
|
onAccept={() => {
|
|
|
|
accept();
|
|
|
|
setDialog('connect');
|
|
|
|
}}
|
|
|
|
onReject={() => {
|
|
|
|
reject();
|
|
|
|
setDialog('intro');
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<RiskAckContent />
|
|
|
|
</RiskAck>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const OnboardConnectionOptions = ({ onConnect }: { onConnect: () => void }) => {
|
|
|
|
const t = useT();
|
|
|
|
const error = useWallet((store) => store.error);
|
|
|
|
const { connect, connectors } = useConnect();
|
|
|
|
const status = useWallet((store) => store.status);
|
|
|
|
|
|
|
|
if (status === 'disconnected') {
|
|
|
|
return (
|
|
|
|
<div className="flex flex-col gap-2 md:gap-4">
|
|
|
|
<h2>{t('Connect to Vega')}</h2>
|
|
|
|
<div className="flex flex-col items-start gap-6">
|
|
|
|
<ul className="flex flex-col -mx-4 -mb-4">
|
|
|
|
{connectors.map((c) => {
|
|
|
|
const ConnectionOption = ConnectionOptionRecord[c.id];
|
|
|
|
const props = {
|
|
|
|
id: c.id,
|
|
|
|
name: c.name,
|
|
|
|
description: c.description,
|
|
|
|
showDescription: true,
|
|
|
|
onClick: async () => {
|
|
|
|
const res = await connect(c.id);
|
|
|
|
if (res.status === 'connected') {
|
|
|
|
onConnect();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
if (ConnectionOption) {
|
|
|
|
return (
|
|
|
|
<li key={c.id}>
|
|
|
|
<ConnectionOption {...props} />
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<li key={c.id}>
|
|
|
|
<ConnectionOptionDefault {...props} />
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</ul>
|
|
|
|
{error && error.code !== ConnectorErrors.userRejected.code && (
|
|
|
|
<p className="text-danger text-sm first-letter:uppercase">
|
|
|
|
{error.message}
|
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
<a
|
|
|
|
href={Links.walletOverview}
|
|
|
|
target="_blank"
|
|
|
|
rel="noreferrer"
|
|
|
|
className="text-sm underline underline-offset-4"
|
|
|
|
>
|
|
|
|
Dont have a wallet?
|
|
|
|
</a>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return <ConnectionStatus status={status} />;
|
|
|
|
};
|