2023-02-16 03:52:54 +00:00
|
|
|
import type { InMemoryCacheConfig } from '@apollo/client';
|
|
|
|
import {
|
2023-10-13 14:02:33 +00:00
|
|
|
AppLoader,
|
2023-02-16 03:52:54 +00:00
|
|
|
NetworkLoader,
|
|
|
|
useEnvironment,
|
2024-03-08 16:12:11 +00:00
|
|
|
useNodeSwitcherStore,
|
2023-02-16 03:52:54 +00:00
|
|
|
} from '@vegaprotocol/environment';
|
2024-03-08 16:12:11 +00:00
|
|
|
import { useEffect, type ReactNode, useState } from 'react';
|
2023-02-16 03:52:54 +00:00
|
|
|
import { Web3Provider } from './web3-provider';
|
2023-11-16 03:10:39 +00:00
|
|
|
import { useT } from '../../lib/use-t';
|
2024-01-05 11:16:59 +00:00
|
|
|
import { DataLoader } from './data-loader';
|
2024-03-01 14:25:56 +00:00
|
|
|
import { WalletProvider } from '@vegaprotocol/wallet-react';
|
|
|
|
import { useVegaWalletConfig } from '../../lib/hooks/use-vega-wallet-config';
|
2024-03-08 16:12:11 +00:00
|
|
|
import { Trans } from 'react-i18next';
|
|
|
|
import { Button, Loader, Splash, VLogo } from '@vegaprotocol/ui-toolkit';
|
|
|
|
|
|
|
|
const Failure = ({ reason }: { reason?: ReactNode }) => {
|
|
|
|
const t = useT();
|
|
|
|
const setNodeSwitcher = useNodeSwitcherStore((store) => store.setDialogOpen);
|
|
|
|
return (
|
|
|
|
<Splash>
|
|
|
|
<div className="border border-vega-red m-10 mx-auto w-4/5 max-w-3xl rounded-lg overflow-hidden animate-shake">
|
|
|
|
<div className="bg-vega-red text-white px-2 py-2 flex gap-1 items-center font-alpha calt uppercase">
|
|
|
|
<VLogo className="h-4" />
|
|
|
|
<span className="text-lg">{t('Failed to initialize the app')}</span>
|
|
|
|
</div>
|
|
|
|
<div className="p-4 text-left text-sm">
|
|
|
|
<p className="mb-4">{reason}</p>
|
|
|
|
<div className="text-center">
|
|
|
|
<Button className="border-2" onClick={() => setNodeSwitcher(true)}>
|
|
|
|
{t('Change node')}
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Splash>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const Loading = () => {
|
|
|
|
const [showSlowNotification, setShowSlowNotification] = useState(false);
|
|
|
|
const t = useT();
|
|
|
|
const setNodeSwitcher = useNodeSwitcherStore((store) => store.setDialogOpen);
|
|
|
|
useEffect(() => {
|
|
|
|
const to = setTimeout(() => {
|
|
|
|
if (!showSlowNotification) setShowSlowNotification(true);
|
|
|
|
}, 5000);
|
|
|
|
return () => {
|
|
|
|
clearTimeout(to);
|
|
|
|
};
|
|
|
|
}, [showSlowNotification]);
|
|
|
|
return (
|
|
|
|
<Splash>
|
|
|
|
<div className="border border-transparent m-10 mx-auto w-4/5 max-w-3xl rounded-lg overflow-hidden">
|
|
|
|
<div className="mt-11 p-4 text-left text-sm">
|
|
|
|
<Loader />
|
|
|
|
{showSlowNotification && (
|
|
|
|
<>
|
|
|
|
<p className="mt-4 text-center">
|
|
|
|
{t(
|
|
|
|
"It looks like you're connection is slow, try switching to another node."
|
|
|
|
)}
|
|
|
|
</p>
|
|
|
|
<div className="mt-4 text-center">
|
|
|
|
<Button
|
|
|
|
className="border-2"
|
|
|
|
onClick={() => setNodeSwitcher(true)}
|
|
|
|
>
|
|
|
|
{t('Change node')}
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Splash>
|
|
|
|
);
|
|
|
|
};
|
2023-02-16 03:52:54 +00:00
|
|
|
|
2023-10-13 14:02:33 +00:00
|
|
|
export const Bootstrapper = ({ children }: { children: ReactNode }) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2024-03-08 16:12:11 +00:00
|
|
|
|
|
|
|
const { error, VEGA_URL } = useEnvironment((state) => ({
|
|
|
|
error: state.error,
|
|
|
|
VEGA_URL: state.VEGA_URL,
|
|
|
|
}));
|
2024-03-01 14:25:56 +00:00
|
|
|
const config = useVegaWalletConfig();
|
2023-02-16 03:52:54 +00:00
|
|
|
|
2024-03-01 14:25:56 +00:00
|
|
|
if (!config) {
|
2023-10-16 14:49:23 +00:00
|
|
|
return <AppLoader />;
|
2023-08-31 02:40:04 +00:00
|
|
|
}
|
|
|
|
|
2024-03-08 16:12:11 +00:00
|
|
|
const ERR_DATA_LOADER = (
|
|
|
|
<Trans
|
|
|
|
i18nKey="It appears that the connection to the node <0>{{VEGA_URL}}</0> does not return necessary data, try switching to another node."
|
|
|
|
components={[
|
|
|
|
<span key="vega" className="text-muted">
|
|
|
|
{VEGA_URL}
|
|
|
|
</span>,
|
|
|
|
]}
|
|
|
|
values={{
|
|
|
|
VEGA_URL,
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
2023-02-16 03:52:54 +00:00
|
|
|
return (
|
2023-11-16 03:10:39 +00:00
|
|
|
<NetworkLoader
|
|
|
|
cache={cacheConfig}
|
2024-03-08 16:12:11 +00:00
|
|
|
skeleton={<Loading />}
|
|
|
|
failure={<Failure reason={error} />}
|
2023-11-16 03:10:39 +00:00
|
|
|
>
|
2024-03-08 16:12:11 +00:00
|
|
|
<DataLoader
|
|
|
|
skeleton={<Loading />}
|
|
|
|
failure={<Failure reason={ERR_DATA_LOADER} />}
|
2023-02-16 03:52:54 +00:00
|
|
|
>
|
2024-03-08 16:12:11 +00:00
|
|
|
<Web3Provider
|
|
|
|
skeleton={<Loading />}
|
|
|
|
failure={<Failure reason={t('Could not configure web3 provider')} />}
|
2023-10-16 14:49:23 +00:00
|
|
|
>
|
2024-03-08 16:12:11 +00:00
|
|
|
<WalletProvider config={config}>{children}</WalletProvider>
|
|
|
|
</Web3Provider>
|
|
|
|
</DataLoader>
|
2023-11-16 03:10:39 +00:00
|
|
|
</NetworkLoader>
|
2023-02-16 03:52:54 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const cacheConfig: InMemoryCacheConfig = {
|
|
|
|
typePolicies: {
|
2023-06-16 08:50:22 +00:00
|
|
|
Statistics: {
|
|
|
|
merge: true,
|
|
|
|
},
|
2023-02-16 03:52:54 +00:00
|
|
|
Account: {
|
|
|
|
keyFields: false,
|
|
|
|
fields: {
|
|
|
|
balanceFormatted: {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Instrument: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
|
|
|
TradableInstrument: {
|
2023-05-31 15:38:49 +00:00
|
|
|
keyFields: false,
|
2023-02-16 03:52:54 +00:00
|
|
|
},
|
|
|
|
Product: {
|
|
|
|
keyFields: ['settlementAsset', ['id']],
|
|
|
|
},
|
|
|
|
MarketData: {
|
|
|
|
keyFields: ['market', ['id']],
|
|
|
|
},
|
|
|
|
Node: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
|
|
|
Withdrawal: {
|
|
|
|
fields: {
|
|
|
|
pendingOnForeignChain: {
|
|
|
|
read: (isPending = false) => isPending,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ERC20: {
|
|
|
|
keyFields: ['contractAddress'],
|
|
|
|
},
|
|
|
|
Party: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
2023-05-25 10:59:08 +00:00
|
|
|
Position: {
|
|
|
|
keyFields: ['market', ['id'], 'party', ['id']],
|
|
|
|
},
|
2023-02-16 03:52:54 +00:00
|
|
|
Fees: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
2023-06-16 08:50:22 +00:00
|
|
|
// The folling types are cached by the data provider and not by apollo
|
|
|
|
PositionUpdate: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
|
|
|
TradeUpdate: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
|
|
|
AccountUpdate: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
2023-05-31 15:38:49 +00:00
|
|
|
OrderUpdate: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
2024-02-12 14:03:21 +00:00
|
|
|
Game: {
|
|
|
|
keyFields: false,
|
|
|
|
},
|
2023-02-16 03:52:54 +00:00
|
|
|
},
|
|
|
|
};
|