From 0230b0c73f9c3e199ae7da0f42efdb24084afc16 Mon Sep 17 00:00:00 2001 From: Frenkie Nguyen Date: Wed, 4 Dec 2024 11:24:54 +0700 Subject: [PATCH] [Issue-219-220][Story Odyssey] Add popup notification for users who failed to mint --- .../src/components/Layout/base/Base.tsx | 82 +------------ .../src/components/Layout/base/Home.tsx | 111 ++++++++++++++---- .../components/Mint/MintNftDetail/index.tsx | 8 +- .../components/Modal/Common/AlertModal.tsx | 10 +- 4 files changed, 108 insertions(+), 103 deletions(-) diff --git a/packages/extension-koni-ui/src/components/Layout/base/Base.tsx b/packages/extension-koni-ui/src/components/Layout/base/Base.tsx index 19afce2020..e162d88de4 100644 --- a/packages/extension-koni-ui/src/components/Layout/base/Base.tsx +++ b/packages/extension-koni-ui/src/components/Layout/base/Base.tsx @@ -3,20 +3,16 @@ import { LanguageType } from '@subwallet/extension-base/background/KoniTypes'; import DefaultLogosMap from '@subwallet/extension-koni-ui/assets/logo'; -import { BookaSdk } from '@subwallet/extension-koni-ui/connector/booka/sdk'; -import { NftMintingLog } from '@subwallet/extension-koni-ui/connector/booka/types'; -import { CONFIRM_SHOW_MINTING_FAILED_MODAL } from '@subwallet/extension-koni-ui/constants'; -import { useConfirmModal, useDefaultNavigate, useSelector } from '@subwallet/extension-koni-ui/hooks'; -import { LayoutBackgroundImages, LayoutBackgroundStyle, Theme, ThemeProps } from '@subwallet/extension-koni-ui/types'; -import { Icon, SwModalFuncProps, SwScreenLayout, SwScreenLayoutProps } from '@subwallet/react-ui'; +import { useDefaultNavigate, useSelector } from '@subwallet/extension-koni-ui/hooks'; +import { LayoutBackgroundImages, LayoutBackgroundStyle, ThemeProps } from '@subwallet/extension-koni-ui/types'; +import { Icon, SwScreenLayout, SwScreenLayoutProps } from '@subwallet/react-ui'; import { SwTabBarItem } from '@subwallet/react-ui/es/sw-tab-bar'; import CN from 'classnames'; -import { ArrowLeft, ChartBar, CheckCircle, Gift, House, Target, UserCirclePlus } from 'phosphor-react'; -import React, { useCallback, useEffect, useMemo, useState } from 'react'; +import { ArrowLeft, ChartBar, Gift, House, Target, UserCirclePlus } from 'phosphor-react'; +import React, { useCallback, useEffect, useMemo } from 'react'; import { useTranslation } from 'react-i18next'; import { useLocation, useNavigate } from 'react-router-dom'; -import styled, { useTheme } from 'styled-components'; -import { useLocalStorage } from 'usehooks-ts'; +import styled from 'styled-components'; import SelectAccount from '../parts/SelectAccount'; @@ -33,7 +29,6 @@ SwScreenLayoutProps, const specialLanguages: Array = ['ja', 'ru']; type TabItemType = Omit & { url: string }; -const apiSDK = BookaSdk.instance; const Component = ({ backgroundImages, backgroundStyle, children, className, headerIcons, onBack, onTabSelected, ...props }: LayoutBaseProps) => { const navigate = useNavigate(); @@ -42,11 +37,6 @@ const Component = ({ backgroundImages, backgroundStyle, children, className, hea const { t } = useTranslation(); const { language } = useSelector((state) => state.settings); - const { token } = useTheme() as Theme; - - const [mintingLog, setMintingLog] = useState(); - const [isShowPopupMintFailed, setIsShowPopupMintFailed] = useLocalStorage(CONFIRM_SHOW_MINTING_FAILED_MODAL, 'nonConfirmed'); - const tabBarItems = useMemo((): TabItemType[] => ([ // { // icon: { @@ -203,66 +193,6 @@ const Component = ({ backgroundImages, backgroundStyle, children, className, hea goHome(); }, [goHome]); - const mintingFailedModalProps = useMemo((): Partial => ({ - id: 'alert-minting-failed', - className: CN('general-confirmation-modal', className), - title: t('Badge minting failed'), - okText: t('I understand'), - content: ( -
-
{t('Mint your badge again')}
-
{t('Due to technical issues, your badge wasn’t minted in Phase 1. Click the Mint tab to mint your badge again on December 6')}
-
- ), - icon: ( -
- -
- ), - closable: true, - maskClosable: true, - okCancel: false, - okButtonProps: { - icon: ( - - ), - shape: 'round' - } - }), [className, t, token.colorIconHover]); - - const { handleSimpleConfirmModal: handleMintingFailedModal } = useConfirmModal(mintingFailedModalProps); - - useEffect(() => { - const fetchMintingLog = async () => { - try { - const mintingLog = await apiSDK.getNftMintingLog(); - - setMintingLog(mintingLog); - } catch (error) { - console.error('Error fetching minting log:', error); - } - }; - - fetchMintingLog().catch(console.error); - }, []); - - useEffect(() => { - if (mintingLog?.notify && isShowPopupMintFailed.includes('nonConfirmed')) { - setIsShowPopupMintFailed('confirmed'); - handleMintingFailedModal().then().catch(console.error); - } - }, [handleMintingFailedModal, isShowPopupMintFailed, mintingLog?.notify, navigate, setIsShowPopupMintFailed]); - useEffect(() => { onTabSelected?.(selectedTab); }, [onTabSelected, selectedTab]); diff --git a/packages/extension-koni-ui/src/components/Layout/base/Home.tsx b/packages/extension-koni-ui/src/components/Layout/base/Home.tsx index 07a3fce5a7..0fa39a9b8c 100644 --- a/packages/extension-koni-ui/src/components/Layout/base/Home.tsx +++ b/packages/extension-koni-ui/src/components/Layout/base/Home.tsx @@ -2,17 +2,19 @@ // SPDX-License-Identifier: Apache-2.0 import { AccountJson } from '@subwallet/extension-base/background/types'; -import { Layout, WalletConnect } from '@subwallet/extension-koni-ui/components'; +import { AlertModal, Layout, WalletConnect } from '@subwallet/extension-koni-ui/components'; import { LayoutBaseProps } from '@subwallet/extension-koni-ui/components/Layout/base/Base'; +import { BookaSdk } from '@subwallet/extension-koni-ui/connector/booka/sdk'; +import { NftMintingLog } from '@subwallet/extension-koni-ui/connector/booka/types'; import { TelegramConnector } from '@subwallet/extension-koni-ui/connector/telegram'; -import { VISIT_INVITATION_SCREEN_FLAG } from '@subwallet/extension-koni-ui/constants'; +import { CONFIRM_SHOW_MINTING_FAILED_MODAL, VISIT_INVITATION_SCREEN_FLAG } from '@subwallet/extension-koni-ui/constants'; import { CUSTOMIZE_MODAL } from '@subwallet/extension-koni-ui/constants/modal'; import { WalletConnectContext } from '@subwallet/extension-koni-ui/contexts/WalletConnectContext'; -import { useNotification, useSelector } from '@subwallet/extension-koni-ui/hooks'; +import { useAlert, useNotification, useSelector } from '@subwallet/extension-koni-ui/hooks'; import { noop } from '@subwallet/extension-koni-ui/utils'; import { ButtonProps, Icon, ModalContext, Tooltip } from '@subwallet/react-ui'; -import { Export, FadersHorizontal, MagnifyingGlass } from 'phosphor-react'; -import React, { useCallback, useContext, useMemo, useState } from 'react'; +import { CheckCircle, Export, FadersHorizontal, Gift, MagnifyingGlass } from 'phosphor-react'; +import React, { useCallback, useContext, useEffect, useMemo, useState } from 'react'; import { useTranslation } from 'react-i18next'; import { useNavigate } from 'react-router-dom'; import styled from 'styled-components'; @@ -34,6 +36,9 @@ type Props = { }; const telegramConnector = TelegramConnector.instance; +const apiSDK = BookaSdk.instance; + +const alertModalId = 'alert-minting-failed-modal'; const Component = (props: Props) => { const { backgroundImages, backgroundStyle, children, className, onClickFilterIcon, onClickSearchIcon, onTabSelected, showConnectIcon = true, showFilterIcon, showGiftIcon, showSearchIcon, showTabBar } = props; @@ -46,6 +51,10 @@ const Component = (props: Props) => { const { activeModal } = useContext(ModalContext); const { connectWC, disconnectWC } = useContext(WalletConnectContext); + const [mintingLog, setMintingLog] = useState(); + const [isShowPopupMintFailed, setIsShowPopupMintFailed] = useLocalStorage(CONFIRM_SHOW_MINTING_FAILED_MODAL, 'nonConfirmed'); + const { alertProps, closeAlert, openAlert } = useAlert(alertModalId); + const { wcAccount } = useSelector((state) => state.accountState); const notify = useNotification(); @@ -181,23 +190,83 @@ const Component = (props: Props) => { navigate('/settings/list'); }, [navigate]); + const handleMintingFailedModal = useCallback(() => { + openAlert({ + className: 'general-confirmation-modal modal-revert-header', + title: t('Badge minting failed'), + iconProps: { + phosphorIcon: Gift, + weight: 'fill' + }, + contentTitle: t('Mint your badge again'), + content: ( + t('Due to technical issues, your badge wasn’t minted in Phase 1. Click the Mint tab to mint your badge again on December 6') + ), + okButton: { + icon: CheckCircle, + iconWeight: 'fill', + text: t('I understand'), + onClick: () => { + setIsShowPopupMintFailed('confirmed'); + closeAlert(); + } + } + }); + }, [closeAlert, openAlert, setIsShowPopupMintFailed, t]); + + useEffect(() => { + const fetchMintingLog = async () => { + try { + const mintingLog = await apiSDK.getNftMintingLog(); + + setMintingLog(mintingLog); + } catch (error) { + console.error('Error fetching minting log:', error); + } + }; + + fetchMintingLog().catch(console.error); + }, []); + + const onCancel = useCallback(() => { + setIsShowPopupMintFailed('confirmed'); + }, [setIsShowPopupMintFailed]); + + useEffect(() => { + if (mintingLog?.notify && isShowPopupMintFailed.includes('nonConfirmed')) { + handleMintingFailedModal(); + } + }, [handleMintingFailedModal, isShowPopupMintFailed, mintingLog?.notify, navigate, setIsShowPopupMintFailed]); + return ( - - {children} - + <> + + {children} + + + { + !!alertProps && ( + + ) + } + ); }; diff --git a/packages/extension-koni-ui/src/components/Mint/MintNftDetail/index.tsx b/packages/extension-koni-ui/src/components/Mint/MintNftDetail/index.tsx index 43cf3e9e0d..3553c7a62b 100644 --- a/packages/extension-koni-ui/src/components/Mint/MintNftDetail/index.tsx +++ b/packages/extension-koni-ui/src/components/Mint/MintNftDetail/index.tsx @@ -82,7 +82,7 @@ const Component: React.FC = (props: Props) => { const notifyIneligibleProps = useMemo((): Partial => ({ id: 'alert-ineligible-mint', - className: CN('general-confirmation-modal', className), + className: CN('general-confirmation-modal modal-revert-header', className), title: t('Ineligible to mint'), okText: t('Back to home'), content: ( @@ -119,7 +119,7 @@ const Component: React.FC = (props: Props) => { const failedToMintProps = useMemo((): Partial => ({ id: 'failed_to_mint', - className: CN('general-confirmation-modal', className), + className: CN('general-confirmation-modal modal-revert-header', className), title: t('Failed to mint'), okText: t('Back to home'), okCancel: false, @@ -156,7 +156,7 @@ const Component: React.FC = (props: Props) => { const badgeAlreadyMintedProps = useMemo((): Partial => ({ id: 'badge_already_minted', - className: CN('general-confirmation-modal', className), + className: CN('general-confirmation-modal modal-revert-header', className), title: t('Failed to mint'), okText: t('Got it'), okCancel: false, @@ -193,7 +193,7 @@ const Component: React.FC = (props: Props) => { const inSufficientBalanceProps = useMemo((): Partial => ({ id: 'in_sufficient_balance', - className: CN('general-confirmation-modal', className), + className: CN('general-confirmation-modal modal-revert-header', className), title: t('Failed to mint'), okText: t('Got it'), okCancel: false, diff --git a/packages/extension-koni-ui/src/components/Modal/Common/AlertModal.tsx b/packages/extension-koni-ui/src/components/Modal/Common/AlertModal.tsx index 2bdfb33e2b..05be4ff5a3 100644 --- a/packages/extension-koni-ui/src/components/Modal/Common/AlertModal.tsx +++ b/packages/extension-koni-ui/src/components/Modal/Common/AlertModal.tsx @@ -11,7 +11,8 @@ import React, { useCallback, useContext } from 'react'; import styled from 'styled-components'; type Props = ThemeProps & AlertDialogProps & { - modalId: string + modalId: string, + _onCancel?: () => void, } const alertTypeAndIconMap = { @@ -41,13 +42,18 @@ const Component: React.FC = (props: Props) => { modalId, okButton, title, + _onCancel, type = NotificationType.INFO, iconProps, contentTitle } = props; const { inactiveModal } = useContext(ModalContext); const onCancel = useCallback(() => { + if (_onCancel) { + _onCancel(); + } + inactiveModal(modalId); - }, [inactiveModal, modalId]); + }, [_onCancel, inactiveModal, modalId]); return ( <>