@broxus/tvm-connect
Version:
TypeScript SDK for connecting to Nekoton-compatible wallets using a unified interface.
98 lines (97 loc) • 7.13 kB
JavaScript
import { error, formattedTokenAmount, getUserAgent } from '@broxus/js-utils';
import { ExplorerAccountLink, Icon, Skeleton, TokenIcon, WalletAccount } from '@broxus/react-components';
import { Button, Dropdown, Flex, Grid, List, Text, useConfig, } from '@broxus/react-uikit';
import classNames from 'classnames';
import { observer } from 'mobx-react-lite';
import * as React from 'react';
import { useIntl } from 'react-intl';
import { TvmConnectButton } from '../components/TvmConnectButton';
import { TVM_RECENT_CONNECTION } from '../constants';
import { useSharedParams, useTvmConnectService } from '../context';
import messages from '../intl';
import { getRecentConnectionMeta, isNekotonWebview, storeRecentConnectionMeta } from '../utils';
export const TvmConnector = observer(({ className, connectButtonTrigger, connectButtonType = 'primary', popupType, showDropMenu = true, showSubIcon = true, standalone = isNekotonWebview(getUserAgent()), suffix, warnUnsupportedNetwork = true, ...props }) => {
const intl = useIntl();
const config = useConfig();
const service = useTvmConnectService();
const params = useSharedParams();
const drop = React.useRef(null);
const { changeAccountButtonText = intl.formatMessage(messages.TVM_CONNECT_CHANGE_ACCOUNT_BTN_TEXT), changeWalletButtonText = intl.formatMessage(messages.TVM_CONNECT_CHANGE_WALLET_BTN_TEXT), connectButtonShape = config.button?.shape ?? 'circle', connectButtonText = intl.formatMessage(messages.TVM_CONNECT_CONNECT_BTN_TEXT), disconnectButtonText = intl.formatMessage(messages.TVM_CONNECT_DISCONNECT_BTN_TEXT), network = service.network, } = props;
const changeAccount = async () => {
if (service.hasProvider) {
drop.current?.close();
try {
await service.provider?.changeAccount();
}
catch (e) {
error(e);
}
}
};
const changeWallet = () => {
drop.current?.close();
};
const disconnect = async () => {
if (service.hasProvider) {
drop.current?.close();
await service.disconnect();
const storageKey = params.recentMetaStorageKey ?? TVM_RECENT_CONNECTION;
const recentMeta = getRecentConnectionMeta(storageKey);
if (recentMeta) {
storeRecentConnectionMeta({ ...recentMeta, disconnected: true }, storageKey);
}
}
};
const isInitializing = service.isInitializing || service.isInitializing == null;
const isDisconnected = !service.isConnected || service.address == null;
let subTitle = intl.formatMessage(messages.TVM_CONNECT_NOT_CONNECTED_HINT);
if (isInitializing) {
subTitle = intl.formatMessage(messages.TVM_CONNECT_INITIALIZING_HINT);
}
else if (service.isConnecting) {
subTitle = intl.formatMessage(messages.TVM_CONNECT_CONNECTING_HINT);
}
else if (service.isUnsupportedNetwork && !isDisconnected && warnUnsupportedNetwork) {
subTitle = (React.createElement(Text, { color: "danger" }, intl.formatMessage(messages.TVM_CONNECT_UNSUPPORTED_NETWORK_WARNING)));
}
else if (service.isSyncing) {
subTitle = React.createElement(Skeleton, { width: 80 });
}
else if (service.isReady) {
subTitle = `${formattedTokenAmount(service.balance, network?.currency.decimals ?? service.currency.decimals)} ${network?.currency.symbol || service.currency.symbol}`;
}
let networkIcon = React.createElement("div", { className: "tvm-connect-connector-icon" }), providerIcon;
if (network?.icon) {
networkIcon = React.createElement(TokenIcon, { className: "uk-text-middle", iconUrl: network.icon, size: 32 });
}
else if (service.network?.icon) {
networkIcon = React.createElement(TokenIcon, { className: "uk-text-middle", iconUrl: service.network.icon, size: 32 });
}
if (service.providerInfo?.icon) {
providerIcon = React.createElement(TokenIcon, { iconUrl: service.providerInfo.icon, size: 16 });
}
if (service.isUnsupportedNetwork && !isDisconnected && warnUnsupportedNetwork) {
networkIcon = React.createElement(Icon, { icon: "danger", ratio: 1.6 });
}
const networkName = network?.shortName ?? service.network?.shortName;
return (React.createElement(Flex, { alignItems: "center", className: classNames('tvm-connect-connector', className, {
[service.providerId]: service.providerId != null,
}), component: Grid },
React.createElement("div", null,
React.createElement(WalletAccount, { address: service.address?.toString(), icon: networkIcon, subIcon: !isDisconnected && showSubIcon ? providerIcon : undefined, subTitle: (React.createElement(Text, { loading: service.isConnecting || isInitializing || service.isSyncing }, subTitle)), title: isDisconnected
? networkName ?? intl.formatMessage(messages.TVM_CONNECT_CONNECTOR_BLOCKCHAIN_NAME)
: (React.createElement(ExplorerAccountLink, { address: service.address?.toString(), baseUrl: network?.explorer.baseUrl || service.network?.explorer.baseUrl, className: "tvm-connect-connector-address", copyable: true, subPath: network?.explorer.accountsSubPath || service.network?.explorer.accountsSubPath, tooltip: intl.formatMessage(messages.TVM_CONNECT_CONNECTOR_EXPLORER_HINT, {
explorerTitle: network?.explorer.title || service.network?.explorer.title || '',
}) })) })),
React.createElement(Flex, { alignItems: "center", className: "tvm-connect-connector-suffix" },
suffix,
!isDisconnected && showDropMenu && (React.createElement(Dropdown, { ref: drop, action: ['click'], hideAction: ['click'], overlay: (React.createElement(List, { className: "uk-margin-remove", size: "large" },
React.createElement(List.Item, null,
React.createElement(Button, { className: "tvm-connect-change-account", type: "link", onClick: changeAccount }, changeAccountButtonText)),
!standalone && (React.createElement(List.Item, { key: "connect-wallet" },
React.createElement(TvmConnectButton, { className: "tvm-connect-change-wallet", network: network, popupType: popupType, type: "link", onOpen: changeWallet }, changeWalletButtonText))),
React.createElement(List.Item, null,
React.createElement(Button, { "aria-disabled": service.isDisconnecting, className: "tvm-connect-logout", disabled: service.isDisconnecting, type: "link", onClick: disconnect }, disconnectButtonText)))), overlayClassName: "tvm-connect-dropdown", placement: "bottom-right" },
React.createElement(Icon, { className: "tvm-connect-dropdown-trigger", icon: "ellipsisVertical", ratio: 0.8 }))),
isDisconnected && (React.createElement(TvmConnectButton, { key: "connect", network: network, popupType: popupType, shape: connectButtonShape, standalone: standalone, trigger: connectButtonTrigger, type: connectButtonType }, connectButtonText)))));
});