diff --git a/packages/extension-base/src/background/KoniTypes.ts b/packages/extension-base/src/background/KoniTypes.ts index bcf79904b7e..134b0eeffea 100644 --- a/packages/extension-base/src/background/KoniTypes.ts +++ b/packages/extension-base/src/background/KoniTypes.ts @@ -1075,7 +1075,15 @@ export type RequestSubscribeStaking = null export type RequestSubscribeStakingReward = null export enum ThemeNames { - LIGHT = 'light' + DEFAULT = 'default', + SKY = 'sky', + MORNING_SUNNY = 'morning_sunny', + SPRING = 'spring', + LAVENDER = 'lavender', + SUNNY = 'sunny', + BEGIE = 'begie', + CLOVE = 'clove', + AURORA = 'aurora', } export enum NETWORK_ERROR { diff --git a/packages/extension-base/src/koni/background/handlers/State.ts b/packages/extension-base/src/koni/background/handlers/State.ts index 596df767319..c2ec6c1091d 100644 --- a/packages/extension-base/src/koni/background/handlers/State.ts +++ b/packages/extension-base/src/koni/background/handlers/State.ts @@ -5,7 +5,7 @@ import { _AssetRef, _AssetType, _ChainAsset, _ChainInfo, _MultiChainAsset } from import { EvmProviderError } from '@subwallet/extension-base/background/errors/EvmProviderError'; import { withErrorLog } from '@subwallet/extension-base/background/handlers/helpers'; import { isSubscriptionRunning, unsubscribe } from '@subwallet/extension-base/background/handlers/subscriptions'; -import { AccountRefMap, AddTokenRequestExternal, AmountData, APIItemState, ApiMap, AuthRequestV2, BasicTxErrorType, ChainStakingMetadata, ChainType, ConfirmationsQueue, CrowdloanItem, CrowdloanJson, CurrentAccountInfo, EvmProviderErrorType, EvmSendTransactionParams, EvmSendTransactionRequest, EvmSignatureRequest, ExternalRequestPromise, ExternalRequestPromiseStatus, ExtrinsicType, MantaAuthorizationContext, MantaPayConfig, MantaPaySyncState, NftCollection, NftItem, NftJson, NominatorMetadata, RequestAccountExportPrivateKey, RequestCheckPublicAndSecretKey, RequestConfirmationComplete, RequestCrowdloanContributions, RequestSettingsType, ResponseAccountExportPrivateKey, ResponseCheckPublicAndSecretKey, ServiceInfo, SingleModeJson, StakingItem, StakingJson, StakingRewardItem, StakingRewardJson, StakingType, UiSettings } from '@subwallet/extension-base/background/KoniTypes'; +import { AccountRefMap, AddTokenRequestExternal, AmountData, APIItemState, ApiMap, AuthRequestV2, BasicTxErrorType, ChainStakingMetadata, ChainType, ConfirmationsQueue, CrowdloanItem, CrowdloanJson, CurrentAccountInfo, EvmProviderErrorType, EvmSendTransactionParams, EvmSendTransactionRequest, EvmSignatureRequest, ExternalRequestPromise, ExternalRequestPromiseStatus, ExtrinsicType, MantaAuthorizationContext, MantaPayConfig, MantaPaySyncState, NftCollection, NftItem, NftJson, NominatorMetadata, RequestAccountExportPrivateKey, RequestCheckPublicAndSecretKey, RequestConfirmationComplete, RequestCrowdloanContributions, RequestSettingsType, ResponseAccountExportPrivateKey, ResponseCheckPublicAndSecretKey, ServiceInfo, StakingItem, StakingJson, StakingRewardItem, StakingRewardJson, StakingType, UiSettings } from '@subwallet/extension-base/background/KoniTypes'; import { AccountJson, RequestAuthorizeTab, RequestRpcSend, RequestRpcSubscribe, RequestRpcUnsubscribe, RequestSign, ResponseRpcListProviders, ResponseSigning } from '@subwallet/extension-base/background/types'; import { ALL_ACCOUNT_KEY, ALL_GENESIS_HASH, MANTA_PAY_BALANCE_INTERVAL } from '@subwallet/extension-base/constants'; import { BalanceService } from '@subwallet/extension-base/services/balance-service'; @@ -13,7 +13,7 @@ import { ServiceStatus } from '@subwallet/extension-base/services/base/types'; import BuyService from '@subwallet/extension-base/services/buy-service'; import CampaignService from '@subwallet/extension-base/services/campaign-service'; import { ChainService } from '@subwallet/extension-base/services/chain-service'; -import { _DEFAULT_MANTA_ZK_CHAIN, _MANTA_ZK_CHAIN_GROUP, _PREDEFINED_SINGLE_MODES } from '@subwallet/extension-base/services/chain-service/constants'; +import { _DEFAULT_MANTA_ZK_CHAIN, _MANTA_ZK_CHAIN_GROUP } from '@subwallet/extension-base/services/chain-service/constants'; import { _ChainState, _NetworkUpsertParams, _ValidateCustomAssetRequest } from '@subwallet/extension-base/services/chain-service/types'; import { _getEvmChainId, _getSubstrateGenesisHash, _getTokenOnChainAssetId, _isAssetFungibleToken, _isChainEnabled, _isChainTestNet, _parseMetadataForSmartContractAsset } from '@subwallet/extension-base/services/chain-service/utils'; import { EventService } from '@subwallet/extension-base/services/event-service'; @@ -1242,16 +1242,6 @@ export default class KoniState { } } - findSingleMode (genesisHash: string): SingleModeJson | undefined { - const [networkKey] = this.findNetworkKeyByGenesisHash(genesisHash); - - if (!networkKey) { - return undefined; - } - - return (Object.values(_PREDEFINED_SINGLE_MODES)).find((item) => (item.networkKeys.includes(networkKey))); - } - public accountExportPrivateKey ({ address, password }: RequestAccountExportPrivateKey): ResponseAccountExportPrivateKey { // eslint-disable-next-line @typescript-eslint/no-unsafe-argument diff --git a/packages/extension-base/src/services/chain-service/constants.ts b/packages/extension-base/src/services/chain-service/constants.ts index 23d41c9d2e1..7684cc68189 100644 --- a/packages/extension-base/src/services/chain-service/constants.ts +++ b/packages/extension-base/src/services/chain-service/constants.ts @@ -3,7 +3,6 @@ import { _DEFAULT_CHAINS } from '@subwallet/chain-list'; import { _SubstrateChainType } from '@subwallet/chain-list/types'; -import { SingleModeJson, ThemeNames } from '@subwallet/extension-base/background/KoniTypes'; export const API_AUTO_CONNECT_MS = 3000; export const API_CONNECT_TIMEOUT = 30000; @@ -16,14 +15,6 @@ export const _API_OPTIONS_CHAIN_GROUP = { goldberg: ['goldberg_testnet'] }; -export const _PREDEFINED_SINGLE_MODES: Record = { - subspace: { - networkKeys: ['subspace_gemini_2a', 'subspace_test', 'subspace_gemini_3a'], - theme: ThemeNames.LIGHT, - autoTriggerDomain: 'subspace.network' - } -}; - export const _PURE_EVM_CHAINS = ['binance', 'binance_test', 'ethereum', 'ethereum_goerli', 'astarEvm', 'shidenEvm', 'shibuyaEvm', 'crabEvm', 'pangolinEvm', 'cloverEvm', 'boba_rinkeby', 'boba', 'bobabase', 'bobabeam', 'watr_network_evm']; // Get balance---------------------------------------------------------------------------------------------------------- diff --git a/packages/extension-base/src/services/setting-service/constants.ts b/packages/extension-base/src/services/setting-service/constants.ts index 2152d591b25..660baa0b349 100644 --- a/packages/extension-base/src/services/setting-service/constants.ts +++ b/packages/extension-base/src/services/setting-service/constants.ts @@ -4,7 +4,7 @@ import { BrowserConfirmationType, LanguageType, ThemeNames, UiSettings, WalletUnlockType } from '@subwallet/extension-base/background/KoniTypes'; import { TARGET_ENV } from '@subwallet/extension-base/utils'; -export const DEFAULT_THEME: ThemeNames = ThemeNames.LIGHT; +export const DEFAULT_THEME: ThemeNames = ThemeNames.DEFAULT; export const DEFAULT_NOTIFICATION_TYPE: BrowserConfirmationType = 'popup'; export const DEFAULT_AUTO_LOCK_TIME = 15; export const DEFAULT_UNLOCK_TYPE: WalletUnlockType = TARGET_ENV === 'extension' ? WalletUnlockType.ALWAYS_REQUIRED : WalletUnlockType.WHEN_NEEDED; diff --git a/packages/extension-koni-ui/src/Popup/Home/Games/index.tsx b/packages/extension-koni-ui/src/Popup/Home/Games/index.tsx index d863d5ce72f..9f2985b2303 100644 --- a/packages/extension-koni-ui/src/Popup/Home/Games/index.tsx +++ b/packages/extension-koni-ui/src/Popup/Home/Games/index.tsx @@ -1,16 +1,19 @@ // Copyright 2019-2022 @subwallet/extension-ui authors & contributors // SPDX-License-Identifier: Apache-2.0 +import { ShopModal } from '@subwallet/extension-koni-ui/components'; import GameAccount from '@subwallet/extension-koni-ui/components/Games/GameAccount'; import GameEnergy from '@subwallet/extension-koni-ui/components/Games/GameEnergy'; +import { ShopModalId } from '@subwallet/extension-koni-ui/components/Modal/Shop/ShopModal'; import { BookaSdk } from '@subwallet/extension-koni-ui/connector/booka/sdk'; -import { Game } from '@subwallet/extension-koni-ui/connector/booka/types'; +import { EnergyConfig, Game, GameInventoryItem, GameItem } from '@subwallet/extension-koni-ui/connector/booka/types'; import { useSetCurrentPage, useTranslation } from '@subwallet/extension-koni-ui/hooks'; import { GameApp } from '@subwallet/extension-koni-ui/Popup/Home/Games/gameSDK'; import { ThemeProps } from '@subwallet/extension-koni-ui/types'; -import { Button, Image, Typography } from '@subwallet/react-ui'; +import { Button, Icon, Image, ModalContext, Typography } from '@subwallet/react-ui'; import CN from 'classnames'; -import React, { useCallback, useEffect, useRef, useState } from 'react'; +import { ShoppingBag } from 'phosphor-react'; +import React, { useCallback, useContext, useEffect, useRef, useState } from 'react'; import styled from 'styled-components'; type Props = ThemeProps; @@ -28,10 +31,17 @@ function checkComingSoon (game: Game): boolean { return gameStartTime > Date.now(); } +const shopModalId = ShopModalId; + const Component = ({ className }: Props): React.ReactElement => { useSetCurrentPage('/home/games'); const gameIframe = useRef(null); const [gameList, setGameList] = useState(apiSDK.gameList); + const [energyConfig, setEnergyConfig] = useState(apiSDK.energyConfig); + const [gameItemMap, setGameItemMap] = useState>(apiSDK.gameItemMap); + const [gameInventoryItemList, setGameInventoryItemList] = useState(apiSDK.gameInventoryItemList); + const [currentGameShopId, setCurrentGameShopId] = useState(); + const { activeModal } = useContext(ModalContext); const [account, setAccount] = useState(apiSDK.account); const [currentGame, setCurrentGame] = useState(undefined); const { t } = useTranslation(); @@ -67,6 +77,13 @@ const Component = ({ className }: Props): React.ReactElement => { }; }, [exitGame]); + const onOpenShop = useCallback((gameId?: number) => { + return () => { + setCurrentGameShopId(gameId); + activeModal(shopModalId); + }; + }, [activeModal]); + useEffect(() => { const accountSub = apiSDK.subscribeAccount().subscribe((data) => { setAccount(data); @@ -76,9 +93,24 @@ const Component = ({ className }: Props): React.ReactElement => { setGameList(data); }); + const energyConfigSub = apiSDK.subscribeEnergyConfig().subscribe((data) => { + setEnergyConfig(data); + }); + + const gameItemMapSub = apiSDK.subscribeGameItemMap().subscribe((data) => { + setGameItemMap(data); + }); + + const gameInventoryItemListSub = apiSDK.subscribeGameInventoryItemList().subscribe((data) => { + setGameInventoryItemList(data); + }); + return () => { accountSub.unsubscribe(); + energyConfigSub.unsubscribe(); gameListSub.unsubscribe(); + gameItemMapSub.unsubscribe(); + gameInventoryItemListSub.unsubscribe(); }; }, []); @@ -92,8 +124,21 @@ const Component = ({ className }: Props): React.ReactElement => { /> + + +
+ +
+ { src={currentGame.url} /> } + + ; }; diff --git a/packages/extension-koni-ui/src/Popup/Home/Mission/TaskCategoryList.tsx b/packages/extension-koni-ui/src/Popup/Home/Mission/TaskCategoryList.tsx new file mode 100644 index 00000000000..faee880606a --- /dev/null +++ b/packages/extension-koni-ui/src/Popup/Home/Mission/TaskCategoryList.tsx @@ -0,0 +1,103 @@ +// Copyright 2019-2022 @subwallet/extension-ui authors & contributors +// SPDX-License-Identifier: Apache-2.0 + +import { TaskCategory, TaskCategoryInfo } from '@subwallet/extension-koni-ui/connector/booka/types'; +import { useTranslation } from '@subwallet/extension-koni-ui/hooks'; +import { ThemeProps } from '@subwallet/extension-koni-ui/types'; +import { formatInteger } from '@subwallet/extension-koni-ui/utils'; +import { Icon, Image, Typography } from '@subwallet/react-ui'; +import { CaretRight } from 'phosphor-react'; +import React, { useCallback, useMemo } from 'react'; +import styled from 'styled-components'; + +type Props = ThemeProps & { + taskCategoryList: TaskCategory[]; + taskCategoryInfoMap: Record; + onClickCategoryItem: (categoryId: number) => void; +}; + +const Component = ({ className, onClickCategoryItem, taskCategoryInfoMap, taskCategoryList }: Props): React.ReactElement => { + const { t } = useTranslation(); + + const filteredTaskCategoryList = useMemo(() => { + return taskCategoryList.filter((tc) => { + return taskCategoryInfoMap[tc.id] && taskCategoryInfoMap[tc.id].tasks.length; + }); + }, [taskCategoryInfoMap, taskCategoryList]); + + const onClickItem = useCallback((categoryId: number) => { + return () => { + onClickCategoryItem(categoryId); + }; + }, [onClickCategoryItem]); + + return ( +
+ + {t('Categories')} + + + { + filteredTaskCategoryList.map((tc) => ( +
+ +
+
{tc.name}
+ +
+ Min point can earn: {formatInteger(taskCategoryInfoMap[tc.id]?.minPoint || 0)} +
+
+
+ +
+
+ )) + } +
+ ); +}; + +export const TaskCategoryList = styled(Component)(({ theme: { extendToken, token } }: ThemeProps) => { + return { + '.task-category-banner': { + marginRight: token.marginSM + }, + + '.task-category-item': { + display: 'flex', + backgroundColor: token.colorBgSecondary, + minHeight: 50, + borderRadius: token.borderRadiusLG, + padding: token.padding, + cursor: 'pointer', + alignItems: 'center' + }, + + '.task-category-item-content': { + flex: 1 + }, + + '.task-category-item-caret-icon': { + minWidth: 40, + marginRight: -token.marginXS, + display: 'flex', + justifyContent: 'center' + }, + + '.task-category-item + .task-category-item': { + marginTop: token.marginXS + } + }; +}); diff --git a/packages/extension-koni-ui/src/Popup/Home/Mission/TaskItem.tsx b/packages/extension-koni-ui/src/Popup/Home/Mission/TaskItem.tsx index c27808af43b..16e14c76506 100644 --- a/packages/extension-koni-ui/src/Popup/Home/Mission/TaskItem.tsx +++ b/packages/extension-koni-ui/src/Popup/Home/Mission/TaskItem.tsx @@ -26,7 +26,6 @@ const _TaskItem = ({ className, task }: Props): React.ReactElement => { useSetCurrentPage('/home/mission'); const [taskLoading, setTaskLoading] = useState(false); const { t } = useTranslation(); - const [disabled, setDisabled] = useState(false); const completed = !!task.completedAt; const finishTask = useCallback(() => { @@ -39,56 +38,42 @@ const _TaskItem = ({ className, task }: Props): React.ReactElement => { setTaskLoading(false); }) .catch(console.error); + setTimeout(() => { - telegramConnector.openLink(task.url); + task.url && telegramConnector.openLink(task.url); }, 100); }, [task.id, task.url]); - const CountDownElement = useCallback(() => { - if (completed) { - return <>; - } - + const { endTime, + isDisabled, + isEnd, isInTimeRange, + isNotStarted, + startTime } = (() => { const now = Date.now(); - if (task.startTime) { - const startTime = new Date(task.startTime).getTime(); - - if (startTime > now) { - setDisabled(true); - - return ; - } - } - - if (task.endTime) { - const endTime = new Date(task.endTime).getTime(); - - if (endTime > now) { - return ; - } else { - setDisabled(true); - - return {t('Ended')}; - } - } - - return <>; - }, [completed, t, task.endTime, task.startTime]); + const startTime = task.startTime ? new Date(task.startTime).getTime() : undefined; + const endTime = task.endTime ? new Date(task.endTime).getTime() : undefined; + const isNotStarted = !completed && !!startTime && startTime > now; + const isInTimeRange = !completed && !!endTime && endTime > now; + const isEnd = !completed && !!endTime && endTime <= now; + + return { + startTime, + endTime, + isNotStarted, + isInTimeRange, + isEnd, + isDisabled: isNotStarted || isEnd + }; + })(); return
@@ -100,13 +85,32 @@ const _TaskItem = ({ className, task }: Props): React.ReactElement => { className={'__sub-title'} size={'sm'} > - - + + + { + isNotStarted && !!startTime && ( + + ) + } + { + isInTimeRange && !!endTime && ( + + ) + } + { + isEnd && ({t('Ended')}) + }
{!completed &&
+ + {sortedTaskList.map((task) => ( + + ))} + + ); +}; + +export const TaskList = styled(Component)(({ theme: { extendToken, token } }: ThemeProps) => { + return { + '.__list-header': { + display: 'flex', + alignItems: 'center', + marginBottom: token.marginXS, + + '.ant-typography': { + marginBottom: 0 + } + }, + + '.task-list': { + padding: token.padding, + + '.account-info': { + marginBottom: token.marginSM + } + } + }; +}); diff --git a/packages/extension-koni-ui/src/Popup/Home/Mission/index.tsx b/packages/extension-koni-ui/src/Popup/Home/Mission/index.tsx index f24a1d8d2cf..88d8ebe18c5 100644 --- a/packages/extension-koni-ui/src/Popup/Home/Mission/index.tsx +++ b/packages/extension-koni-ui/src/Popup/Home/Mission/index.tsx @@ -3,74 +3,138 @@ import GameAccount from '@subwallet/extension-koni-ui/components/Games/GameAccount'; import { BookaSdk } from '@subwallet/extension-koni-ui/connector/booka/sdk'; -import { Task } from '@subwallet/extension-koni-ui/connector/booka/types'; -import { useSetCurrentPage, useTranslation } from '@subwallet/extension-koni-ui/hooks'; -import TaskItem from '@subwallet/extension-koni-ui/Popup/Home/Mission/TaskItem'; +import { Task, TaskCategory, TaskCategoryInfo } from '@subwallet/extension-koni-ui/connector/booka/types'; +import { useSetCurrentPage } from '@subwallet/extension-koni-ui/hooks'; +import { TaskList } from '@subwallet/extension-koni-ui/Popup/Home/Mission/TaskList'; import { ThemeProps } from '@subwallet/extension-koni-ui/types'; -import { Typography } from '@subwallet/react-ui'; -import React, { useEffect, useMemo, useState } from 'react'; +import React, { useCallback, useEffect, useState } from 'react'; import styled from 'styled-components'; +import { TaskCategoryList } from './TaskCategoryList'; + type Props = ThemeProps; const apiSDK = BookaSdk.instance; +enum ViewMode { + CATEGORY_LIST = 'category_list', + TASK_LIST = 'task_list', +} + +function getTaskCategoryInfoMap (tasks: Task[]): Record { + const result: Record = {}; + const now = Date.now(); + + tasks.forEach((t) => { + if (!t.categoryId) { + return; + } + + if (!result[t.categoryId]) { + result[t.categoryId] = { + id: t.categoryId, + minPoint: t.pointReward || 0, + tasks: [t] + }; + } else { + result[t.categoryId].tasks.push(t); + + if (t.completedAt || t.status > 0) { + return; + } + + if (t.startTime && (now < new Date(t.startTime).getTime())) { + return; + } + + if (t.endTime && (now >= new Date(t.endTime).getTime())) { + return; + } + + result[t.categoryId].minPoint += (t.pointReward || 0); + } + }); + + return result; +} + const Component = ({ className }: Props): React.ReactElement => { useSetCurrentPage('/home/mission'); - const [taskList, setTaskList] = useState(apiSDK.taskList); + const [taskCategoryList, setTaskCategoryList] = useState(apiSDK.taskCategoryList); + const [taskCategoryInfoMap, setTaskCategoryInfoMap] = useState>(getTaskCategoryInfoMap(apiSDK.taskList)); const [account, setAccount] = useState(apiSDK.account); - const { t } = useTranslation(); + const [currentViewMode, setCurrentViewMode] = useState(ViewMode.CATEGORY_LIST); + const [currentTaskCategory, setCurrentTaskCategory] = useState(); useEffect(() => { const accountSub = apiSDK.subscribeAccount().subscribe((data) => { setAccount(data); }); + const taskCategoryListSub = apiSDK.subscribeTaskCategoryList().subscribe((data) => { + setTaskCategoryList(data); + }); + + let taskListUpdaterInterval: NodeJS.Timer; + const taskListSub = apiSDK.subscribeTaskList().subscribe((data) => { - setTaskList(data); + clearInterval(taskListUpdaterInterval); + + setTaskCategoryInfoMap(getTaskCategoryInfoMap(data)); + + taskListUpdaterInterval = setInterval(() => { + setTaskCategoryInfoMap(getTaskCategoryInfoMap(data)); + }, 10000); }); return () => { accountSub.unsubscribe(); + taskCategoryListSub.unsubscribe(); taskListSub.unsubscribe(); + clearInterval(taskListUpdaterInterval); }; }, []); - const sortedTaskList = useMemo(() => { - const now = Date.now(); - - return taskList.sort((a, b) => { - if (a.status < b.status) { - return -1; - } + const onClickCategoryItem = useCallback((categoryId: number) => { + setCurrentViewMode(ViewMode.TASK_LIST); + setCurrentTaskCategory(categoryId); + }, []); - const aDisabled = ((a.startTime && new Date(a.startTime).getTime() > now) || (a.endTime && new Date(a.endTime).getTime() < now)); - const bDisabled = ((b.startTime && new Date(b.startTime).getTime() > now) || (b.endTime && new Date(b.endTime).getTime() < now)); + const onBackToCategoryList = useCallback(() => { + setCurrentViewMode(ViewMode.CATEGORY_LIST); + setCurrentTaskCategory(undefined); + }, []); - if (aDisabled && !bDisabled) { - return 1; + return
+
+ {account && ( + + )} + + { + currentViewMode === ViewMode.CATEGORY_LIST && ( + + ) } - if (!aDisabled && bDisabled) { - return -1; + { + currentViewMode === ViewMode.TASK_LIST && ( + + ) } - - return 0; - }); - }, [taskList]); - - return
-
- {account && } - - {t('Missions')} - - {sortedTaskList.map((task) => ())}
; }; diff --git a/packages/extension-koni-ui/src/Popup/Settings/GeneralSetting.tsx b/packages/extension-koni-ui/src/Popup/Settings/GeneralSetting.tsx index 8aaf7ca72af..6c30add0475 100644 --- a/packages/extension-koni-ui/src/Popup/Settings/GeneralSetting.tsx +++ b/packages/extension-koni-ui/src/Popup/Settings/GeneralSetting.tsx @@ -11,7 +11,7 @@ import { Theme, ThemeProps } from '@subwallet/extension-koni-ui/types'; import { noop } from '@subwallet/extension-koni-ui/utils'; import { BackgroundIcon, Icon, SelectModal, SettingItem, SwIconProps } from '@subwallet/react-ui'; import CN from 'classnames'; -import { CaretRight, CheckCircle, GlobeHemisphereEast, Image, MoonStars, Sun } from 'phosphor-react'; +import { CaretRight, CheckCircle, GlobeHemisphereEast, Image, PaintBrush } from 'phosphor-react'; import React, { useCallback, useMemo, useState } from 'react'; import { useTranslation } from 'react-i18next'; import { useSelector } from 'react-redux'; @@ -33,24 +33,32 @@ function renderSelectionItem (item: SelectionItemType, _selected: boolean) { className={CN('__selection-item', { 'item-disabled': item.disabled })} key={item.key} leftItemIcon={ - + ( +
+ +
+ ) } name={item.title} rightItem={ _selected - ? + ? ( + + ) : null } /> @@ -88,8 +96,6 @@ type LoadingMap = { language: boolean; browserConfirmationType: boolean; }; -// "TODO: Will be shown when support for the LIGHT theme is implemented." -const isShowWalletTheme = false; function Component ({ className = '' }: Props): React.ReactElement { const { t } = useTranslation(); @@ -107,20 +113,61 @@ function Component ({ className = '' }: Props): React.ReactElement { const themeItems = useMemo(() => { return [ { - key: ThemeNames.LIGHT, - leftIcon: MoonStars, - leftIconBgColor: token.colorPrimary, - title: t('Dark theme') + key: ThemeNames.DEFAULT, + leftIcon: PaintBrush, + leftIconBgColor: '#CBF147', + title: t('Default') + }, + { + key: ThemeNames.SKY, + leftIcon: PaintBrush, + leftIconBgColor: '#C7F0FF', + title: t('Sky') + }, + { + key: ThemeNames.MORNING_SUNNY, + leftIcon: PaintBrush, + leftIconBgColor: '#FFF8C4', + title: t('Morning sunny') + }, + { + key: ThemeNames.SPRING, + leftIcon: PaintBrush, + leftIconBgColor: '#7EEC79', + title: t('Spring') + }, + { + key: ThemeNames.LAVENDER, + leftIcon: PaintBrush, + leftIconBgColor: '#BB9EFF', + title: t('Lavender') }, { - key: ThemeNames.LIGHT, - leftIcon: Sun, - leftIconBgColor: token.colorPrimary, - title: t('Light theme'), - disabled: true + key: ThemeNames.SUNNY, + leftIcon: PaintBrush, + leftIconBgColor: '#FBCE01', + title: t('Sunny') + }, + { + key: ThemeNames.BEGIE, + leftIcon: PaintBrush, + leftIconBgColor: '#EBD7C9', + title: t('Begie') + }, + { + key: ThemeNames.CLOVE, + leftIcon: PaintBrush, + leftIconBgColor: 'linear-gradient(117deg, #A2F6C1 9.05%, #CBF147 91.43%)', + title: t('Clove') + }, + { + key: ThemeNames.AURORA, + leftIcon: PaintBrush, + leftIconBgColor: 'linear-gradient(117deg, #A2F6C1 9.05%, #9FE3FF 91.43%)', + title: t('Aurora') } ]; - }, [t, token]); + }, [t]); const languageItems = useMemo(() => { return languageOptions.map((item) => ({ @@ -157,26 +204,25 @@ function Component ({ className = '' }: Props): React.ReactElement { title={t('General settings')} >
- {isShowWalletTheme && - } + (); const [currentEnergy, setCurrentEnergy] = useState(energy); const intervalRef = useRef(null); @@ -31,6 +31,10 @@ function _GameEnergy ({ className, energy, startTime }: GameEnergyProps) { }, [startTime]); const updateEnergy = useCallback(() => { + if (!maxEnergy) { + return; + } + const now = Date.now(); const diff = now - startRegen; const recovered = Math.floor(diff / regenTime); @@ -48,7 +52,7 @@ function _GameEnergy ({ className, energy, startTime }: GameEnergyProps) { setCurrentEnergy(recovered + energy); setCountdown(remainingTime); } - }, [energy, startRegen]); + }, [energy, maxEnergy, startRegen]); useEffect(() => { intervalRef.current = setInterval(updateEnergy, ONE_SECOND); @@ -60,6 +64,10 @@ function _GameEnergy ({ className, energy, startTime }: GameEnergyProps) { }; }, [updateEnergy]); + if (!maxEnergy) { + return null; + } + return
(({ theme: { token } }: LayoutBaseProps) => ({ +const Base = styled(Component)(({ theme: { extendToken, token } }: LayoutBaseProps) => ({ '.ant-sw-tab-bar-container': { padding: `${token.paddingXS}px ${token.paddingSM}px ${token.paddingSM}px`, borderRadius: 40, @@ -189,10 +187,22 @@ const Base = styled(Component)(({ theme: { token } }: LayoutBas width: `calc(100% - ${token.margin * 2}px)`, marginLeft: token.margin, marginBottom: token.marginLG, - backgroundColor: token.colorPrimary, + backgroundColor: extendToken.colorBgSecondary2, + + '.ant-sw-tab-bar-item': { + gap: token.sizeXXS + }, + + '.ant-sw-tab-bar-item.ant-sw-tab-bar-item-active': { + '.ant-sw-tab-bar-item-icon, .ant-sw-tab-bar-item-label': { + color: token.colorPrimary + } + }, '.ant-sw-tab-bar-item-label': { - textAlign: 'center' + textAlign: 'center', + fontSize: 10, + lineHeight: 1.6 } }, diff --git a/packages/extension-koni-ui/src/components/Layout/parts/SelectAccount/index.tsx b/packages/extension-koni-ui/src/components/Layout/parts/SelectAccount/index.tsx index a96deb710e6..204a01d9742 100644 --- a/packages/extension-koni-ui/src/components/Layout/parts/SelectAccount/index.tsx +++ b/packages/extension-koni-ui/src/components/Layout/parts/SelectAccount/index.tsx @@ -11,7 +11,7 @@ import { RootState } from '@subwallet/extension-koni-ui/stores'; import { Theme } from '@subwallet/extension-koni-ui/themes'; import { ThemeProps } from '@subwallet/extension-koni-ui/types'; import { copyToClipboard, findAccountByAddress, funcSortByName, isAccountAll, searchAccountFunction } from '@subwallet/extension-koni-ui/utils'; -import {Button, Icon, Image, ModalContext, SelectModal} from '@subwallet/react-ui'; +import { Button, Icon, Image, ModalContext, SelectModal } from '@subwallet/react-ui'; import CN from 'classnames'; import { Copy } from 'phosphor-react'; import React, { useCallback, useContext, useEffect, useMemo, useState } from 'react'; @@ -35,7 +35,7 @@ const rankIconMap: Record = { silver: '/images/ranks/silver.svg', gold: '/images/ranks/gold.svg', platinum: '/images/ranks/platinum.svg', - diamond: '/images/ranks/diamond.svg', + diamond: '/images/ranks/diamond.svg' }; function Component ({ className }: Props): React.ReactElement { @@ -48,7 +48,7 @@ function Component ({ className }: Props): React.ReactElement { const [gameAccount, setGameAccount] = useState(apiSDK.account); - const { accounts: _accounts, currentAccount, isAllAccount } = useSelector((state: RootState) => state.accountState); + const { accounts: _accounts, currentAccount } = useSelector((state: RootState) => state.accountState); const [selectedQrAddress, setSelectedQrAddress] = useState(); diff --git a/packages/extension-koni-ui/src/components/Modal/Shop/ShopModal.tsx b/packages/extension-koni-ui/src/components/Modal/Shop/ShopModal.tsx new file mode 100644 index 00000000000..cabd5ac7001 --- /dev/null +++ b/packages/extension-koni-ui/src/components/Modal/Shop/ShopModal.tsx @@ -0,0 +1,170 @@ +// Copyright 2019-2022 @subwallet/extension-koni-ui authors & contributors +// SPDX-License-Identifier: Apache-2.0 + +import { ShopItem } from '@subwallet/extension-koni-ui/components'; +import { BookaSdk } from '@subwallet/extension-koni-ui/connector/booka/sdk'; +import { EnergyConfig, GameInventoryItem, GameItem } from '@subwallet/extension-koni-ui/connector/booka/types'; +import useTranslation from '@subwallet/extension-koni-ui/hooks/common/useTranslation'; +import { ThemeProps } from '@subwallet/extension-koni-ui/types'; +import { ShopItemInfo } from '@subwallet/extension-koni-ui/types/shop'; +import { ModalContext, SwModal } from '@subwallet/react-ui'; +import React, { useCallback, useContext, useMemo, useState } from 'react'; +import styled from 'styled-components'; + +type Props = ThemeProps & { + gameId?: number; + energyConfig?: EnergyConfig; + gameItemMap: Record; + gameInventoryItemList: GameInventoryItem[]; +}; + +export const ShopModalId = 'ShopModalId'; +const apiSDK = BookaSdk.instance; + +function Component ({ className, energyConfig, + gameId, + gameInventoryItemList, gameItemMap }: Props): React.ReactElement { + const { t } = useTranslation(); + const [isLoading, setIsLoading] = useState(false); + + const { inactiveModal } = useContext(ModalContext); + + const onClose = useCallback(() => { + inactiveModal(ShopModalId); + }, [inactiveModal]); + + const items = useMemo(() => { + const result: ShopItemInfo[] = []; + + if (energyConfig) { + result.push({ + gameItemId: 'buy-energy-id', + name: 'Energy', + description: '', + price: energyConfig.energyPrice + }); + } + + const inventoryItemMapByGameItemId: Record = {}; + + gameInventoryItemList.forEach((i) => { + inventoryItemMapByGameItemId[i.gameItemId] = i; + }); + + const getShopItem = (gi: GameItem, disabled = false): ShopItemInfo => { + const limit = gi.maxBuy || undefined; + const inventoryQuantity = inventoryItemMapByGameItemId[gi.id]?.quantity || undefined; + + return { + gameItemId: `${gi.id}`, + name: gi.name, + gameId: gi.gameId, + limit, + description: gi.description, + inventoryQuantity, + itemGroup: gi.itemGroup, + itemGroupLevel: gi.itemGroupLevel, + price: gi.price, + disabled: disabled || (!!limit && limit > 0 && limit === inventoryQuantity) || (!!gi.itemGroup && inventoryQuantity === 1), + usable: !!inventoryQuantity && inventoryQuantity > 0 && inventoryItemMapByGameItemId[gi.id]?.usable + }; + }; + + [...Object.keys(gameItemMap)].forEach((groupKey) => { + if (groupKey !== 'NO_GROUP' && gameItemMap[groupKey][0]?.effectDuration === -1) { + const noQuantityItems = gameItemMap[groupKey].filter((gi) => !inventoryItemMapByGameItemId[gi.id]?.quantity); + + let itemPresentForGroup: GameItem; + + if (noQuantityItems.length) { + itemPresentForGroup = noQuantityItems.reduce((item, currentItem) => { + return currentItem.itemGroupLevel && item.itemGroupLevel && currentItem.itemGroupLevel < item.itemGroupLevel ? currentItem : item; + }, { itemGroupLevel: Number.POSITIVE_INFINITY } as GameItem); + + if (itemPresentForGroup.itemGroupLevel !== Number.POSITIVE_INFINITY) { + result.push(getShopItem(itemPresentForGroup)); + } + } else { + itemPresentForGroup = gameItemMap[groupKey] + .reduce((item, currentItem) => { + return currentItem.itemGroupLevel && item.itemGroupLevel && currentItem.itemGroupLevel > item.itemGroupLevel ? currentItem : item; + }, { itemGroupLevel: Number.NEGATIVE_INFINITY } as GameItem); + + if (itemPresentForGroup.itemGroupLevel !== Number.NEGATIVE_INFINITY) { + result.push(getShopItem(itemPresentForGroup, true)); + } + } + + return; + } + + gameItemMap[groupKey].forEach((gi) => { + if ((!gameId && !gi.gameId) || (gameId && gi.gameId === gameId)) { + result.push(getShopItem(gi)); + } + }); + }); + + return result; + }, [energyConfig, gameId, gameInventoryItemList, gameItemMap]); + + const onBuy = useCallback((gameItemId: string) => { + setIsLoading(true); + + if (gameItemId === 'buy-energy-id') { + apiSDK.buyEnergy().catch((e) => { + console.log('buyEnergy error', e); + }).finally(() => { + setIsLoading(false); + }); + } else { + apiSDK.buyItem(+gameItemId).catch((e) => { + console.log('buyItem error', e); + }).finally(() => { + setIsLoading(false); + }); + } + }, []); + + const onUse = useCallback((gameItemId: string) => { + setIsLoading(true); + + apiSDK.useInventoryItem(+gameItemId).catch((e) => { + console.log('onUse error', e); + }).finally(() => { + setIsLoading(false); + }); + }, []); + + return ( + + { + items.map((item) => ( + + )) + } + + ); +} + +const ShopModal = styled(Component)(({ theme: { token } }: Props) => { + return ({ + '.shop-item + .shop-item': { + marginTop: token.marginSM + } + }); +}); + +export default ShopModal; diff --git a/packages/extension-koni-ui/src/components/Modal/Shop/index.tsx b/packages/extension-koni-ui/src/components/Modal/Shop/index.tsx new file mode 100644 index 00000000000..aeaba09f14a --- /dev/null +++ b/packages/extension-koni-ui/src/components/Modal/Shop/index.tsx @@ -0,0 +1,4 @@ +// Copyright 2019-2022 @subwallet/extension-koni-ui authors & contributors +// SPDX-License-Identifier: Apache-2.0 + +export { default as ShopModal } from './ShopModal'; diff --git a/packages/extension-koni-ui/src/components/Modal/index.tsx b/packages/extension-koni-ui/src/components/Modal/index.tsx index da360635ee4..eb153e9170d 100644 --- a/packages/extension-koni-ui/src/components/Modal/index.tsx +++ b/packages/extension-koni-ui/src/components/Modal/index.tsx @@ -17,3 +17,4 @@ export * from './GlobalSearchTokenModal'; export * from './ReceiveModal'; export * from './Common'; export * from './Announcement'; +export * from './Shop'; diff --git a/packages/extension-koni-ui/src/components/Shop/ShopItem.tsx b/packages/extension-koni-ui/src/components/Shop/ShopItem.tsx new file mode 100644 index 00000000000..b3b6bdfd9b5 --- /dev/null +++ b/packages/extension-koni-ui/src/components/Shop/ShopItem.tsx @@ -0,0 +1,101 @@ +// Copyright 2019-2022 @polkadot/extension-ui authors & contributors +// SPDX-License-Identifier: Apache-2.0 + +import DefaultLogosMap from '@subwallet/extension-koni-ui/assets/logo'; +import { ThemeProps } from '@subwallet/extension-koni-ui/types'; +import { ShopItemInfo } from '@subwallet/extension-koni-ui/types/shop'; +import { Button, Image } from '@subwallet/react-ui'; +import CN from 'classnames'; +import React, { useCallback } from 'react'; +import styled from 'styled-components'; + +type Props = ThemeProps & ShopItemInfo & { + onBuy: (gameItemId: string, quantity?: number) => void + onUse: (gameItemId: string) => void +}; + +function Component (props: Props): React.ReactElement { + const { className = '', + description, + disabled, + gameItemId, + inventoryQuantity, + limit, + name, onBuy, onUse, price, usable } = props; + + const _onBuy = useCallback(() => { + onBuy(gameItemId, 1); + }, [gameItemId, onBuy]); + + const _onUse = useCallback(() => { + onUse(gameItemId); + }, [gameItemId, onUse]); + + return ( +
+ + +
+
{name}
+
description: {description}
+ { + !!limit && ( +
Limit: {limit}
+ ) + } + +
Price: {price}
+ + { + !!inventoryQuantity && ( +
Quantity: {inventoryQuantity}
+ ) + } + +
+ + { + usable && ( + + ) + } + + +
+ ); +} + +const ShopItem = styled(Component)(({ theme: { token } }: Props) => { + return ({ + display: 'flex', + backgroundColor: token.colorBgSecondary, + padding: token.paddingSM, + borderRadius: token.borderRadiusLG, + gap: token.sizeXS, + + '.__middle-part': { + flex: 1 + } + }); +}); + +export default ShopItem; diff --git a/packages/extension-koni-ui/src/components/Shop/index.ts b/packages/extension-koni-ui/src/components/Shop/index.ts new file mode 100644 index 00000000000..2632b841b40 --- /dev/null +++ b/packages/extension-koni-ui/src/components/Shop/index.ts @@ -0,0 +1,4 @@ +// Copyright 2019-2022 @polkadot/extension-ui authors & contributors +// SPDX-License-Identifier: Apache-2.0 + +export { default as ShopItem } from './ShopItem'; diff --git a/packages/extension-koni-ui/src/components/index.ts b/packages/extension-koni-ui/src/components/index.ts index 5b5a4a43039..57de0080d05 100644 --- a/packages/extension-koni-ui/src/components/index.ts +++ b/packages/extension-koni-ui/src/components/index.ts @@ -31,4 +31,4 @@ export * from './Setting'; export * from './StakingItem'; export * from './TokenItem'; export * from './WalletConnect'; -export * from './Crowdloan'; +export * from './Shop'; diff --git a/packages/extension-koni-ui/src/connector/booka/sdk.ts b/packages/extension-koni-ui/src/connector/booka/sdk.ts index d24769d8405..095856711ad 100644 --- a/packages/extension-koni-ui/src/connector/booka/sdk.ts +++ b/packages/extension-koni-ui/src/connector/booka/sdk.ts @@ -3,7 +3,7 @@ import { SWStorage } from '@subwallet/extension-base/storage'; import { createPromiseHandler } from '@subwallet/extension-base/utils'; -import { BookaAccount, Game, GamePlay, LeaderboardPerson, ReferralRecord, Task } from '@subwallet/extension-koni-ui/connector/booka/types'; +import { BookaAccount, EnergyConfig, Game, GameInventoryItem, GameItem, GamePlay, LeaderboardPerson, ReferralRecord, Task, TaskCategory } from '@subwallet/extension-koni-ui/connector/booka/types'; import { TelegramConnector } from '@subwallet/extension-koni-ui/connector/telegram'; import { signRaw } from '@subwallet/extension-koni-ui/messaging'; import fetch from 'cross-fetch'; @@ -16,21 +16,27 @@ const telegramConnector = TelegramConnector.instance; const CACHE_KEYS = { account: 'data--account-cache', + taskCategoryList: 'data--task-category-list-cache', taskList: 'data--task-list-cache', - gameList: 'data--game-list-cache' + gameList: 'data--game-list-cache', + energyConfig: 'data--energy-config' }; export class BookaSdk { private syncHandler = createPromiseHandler(); private accountSubject = new BehaviorSubject(undefined); private taskListSubject = new BehaviorSubject([]); + private taskCategoryListSubject = new BehaviorSubject([]); private gameListSubject = new BehaviorSubject([]); private currentGamePlaySubject = new BehaviorSubject(undefined); private leaderBoardSubject = new BehaviorSubject([]); private referralListSubject = new BehaviorSubject([]); + private gameItemMapSubject = new BehaviorSubject>({}); + private gameInventoryItemListSubject = new BehaviorSubject([]); + private energyConfigSubject = new BehaviorSubject(undefined); constructor () { - storage.getItems(Object.values(CACHE_KEYS)).then(([account, tasks, game]) => { + storage.getItems(Object.values(CACHE_KEYS)).then(([account, taskCategory, tasks, game, energyConfig]) => { if (account) { try { const accountData = JSON.parse(account) as BookaAccount; @@ -41,6 +47,16 @@ export class BookaSdk { } } + if (taskCategory) { + try { + const taskCategoryList = JSON.parse(taskCategory) as TaskCategory[]; + + this.taskCategoryListSubject.next(taskCategoryList); + } catch (e) { + console.error('Failed to parse task list', e); + } + } + if (tasks) { try { const taskList = JSON.parse(tasks) as Task[]; @@ -60,6 +76,16 @@ export class BookaSdk { console.error('Failed to parse game list', e); } } + + if (energyConfig) { + try { + const _energyConfig = JSON.parse(energyConfig) as EnergyConfig; + + this.energyConfigSubject.next(_energyConfig); + } catch (e) { + console.error('Failed to parse energy config', e); + } + } }).catch(console.error); } @@ -71,14 +97,30 @@ export class BookaSdk { return this.accountSubject.value; } + public get energyConfig () { + return this.energyConfigSubject.value; + } + public get taskList () { return this.taskListSubject.value; } + public get taskCategoryList () { + return this.taskCategoryListSubject.value; + } + public get gameList () { return this.gameListSubject.value; } + public get gameItemMap () { + return this.gameItemMapSubject.value; + } + + public get gameInventoryItemList () { + return this.gameInventoryItemListSubject.value; + } + public get leaderBoard () { return this.leaderBoardSubject.value; } @@ -146,6 +188,19 @@ export class BookaSdk { return this.accountSubject; } + async fetchEnergyConfig () { + const energyConfig = await this.getRequest(`${GAME_API_HOST}/api/shop/get-config-buy-energy`); + + if (energyConfig) { + this.energyConfigSubject.next(energyConfig); + storage.setItem(CACHE_KEYS.energyConfig, JSON.stringify(energyConfig)).catch(console.error); + } + } + + subscribeEnergyConfig () { + return this.energyConfigSubject; + } + async fetchGameList () { const gameList = await this.getRequest(`${GAME_API_HOST}/api/game/fetch`); @@ -159,6 +214,20 @@ export class BookaSdk { return this.gameListSubject; } + async fetchTaskCategoryList () { + await this.waitForSync; + const taskCategoryList = await this.getRequest(`${GAME_API_HOST}/api/task-category/fetch`); + + if (taskCategoryList) { + this.taskCategoryListSubject.next(taskCategoryList); + storage.setItem(CACHE_KEYS.taskCategoryList, JSON.stringify(taskCategoryList)).catch(console.error); + } + } + + subscribeTaskCategoryList () { + return this.taskCategoryListSubject; + } + async fetchTaskList () { await this.waitForSync; const taskList = await this.getRequest(`${GAME_API_HOST}/api/task/history`); @@ -176,6 +245,8 @@ export class BookaSdk { async finishTask (taskId: number) { await this.postRequest(`${GAME_API_HOST}/api/task/submit`, { taskId }); + await this.fetchTaskCategoryList(); + await this.fetchTaskList(); await this.reloadAccount(); @@ -230,7 +301,15 @@ export class BookaSdk { storage.setItem(CACHE_KEYS.account, JSON.stringify(account)).catch(console.error); this.syncHandler.resolve(); - await Promise.all([this.fetchGameList(), this.fetchTaskList(), this.fetchLeaderboard()]); + await Promise.all([ + this.fetchEnergyConfig(), + this.fetchGameList(), + this.fetchTaskCategoryList(), + this.fetchTaskList(), + this.fetchLeaderboard(), + this.fetchGameItemMap(), + this.fetchGameInventoryItemList() + ]); } else { throw new Error('Failed to sync account'); } @@ -298,6 +377,63 @@ export class BookaSdk { await Promise.all([this.reloadAccount()]); } + // --- shop + + async fetchGameItemMap () { + await this.waitForSync; + + const gameItemMap = await this.postRequest>(`${GAME_API_HOST}/api/shop/list-items`, {}); + + if (gameItemMap) { + this.gameItemMapSubject.next(gameItemMap); + } + } + + subscribeGameItemMap () { + return this.gameItemMapSubject; + } + + async fetchGameInventoryItemList () { + await this.waitForSync; + + const inventoryItemList = await this.getRequest(`${GAME_API_HOST}/api/shop/get-inventory`); + + if (inventoryItemList) { + this.gameInventoryItemListSubject.next(inventoryItemList); + } + } + + subscribeGameInventoryItemList () { + return this.gameInventoryItemListSubject; + } + + async buyItem (gameItemId: number, quantity = 1) { + await this.postRequest(`${GAME_API_HOST}/api/shop/buy-item`, { gameItemId, quantity }); + + await this.fetchGameInventoryItemList(); + + await this.fetchGameItemMap(); + + await this.reloadAccount(); + } + + async useInventoryItem (gameItemId: number) { + await this.postRequest(`${GAME_API_HOST}/api/shop/use-inventory-item`, { gameItemId }); + + await this.fetchGameInventoryItemList(); + + await this.fetchGameItemMap(); + + await this.reloadAccount(); + } + + async buyEnergy () { + await this.postRequest(`${GAME_API_HOST}/api/shop/buy-energy`, {}); + + await this.reloadAccount(); + } + // --- shop + async fetchLeaderboard () { await this.waitForSync; const leaderBoard = await this.getRequest(`${GAME_API_HOST}/api/game/leader-board`); diff --git a/packages/extension-koni-ui/src/connector/booka/types.ts b/packages/extension-koni-ui/src/connector/booka/types.ts index 58d61df8ac2..a0bad5ac7c8 100644 --- a/packages/extension-koni-ui/src/connector/booka/types.ts +++ b/packages/extension-koni-ui/src/connector/booka/types.ts @@ -7,6 +7,46 @@ export enum EventTypeEnum { EVENT = 'EVENT', } +export interface EnergyConfig { + energyPrice: number, + energyBuyLimit: number, + maxEnergy: number, + energyOneBuy: number +} + +export interface GameItem { + id: number, + contentId: number, + gameId: number, + slug: string, + name: string, + description: string, + price: number, + tokenPrice: number, + maxBuy?: number | null, + maxBuyDaily: number, + itemGroup: string, + itemGroupLevel: number, + effectDuration: number, +} + +export enum GameInventoryItemStatus { + INACTIVE = 'inactive', // After buy item request + ACTIVE = 'active', // After validate signature + USED = 'used', // After used item +} + +export interface GameInventoryItem { + id: number, + gameId: number, + accountId: number, + gameDataId: number, + gameItemId: number, + quantity: number, + usable: boolean, + itemId?: number | null +} + export interface Game { id: number; contentId: number; @@ -27,23 +67,41 @@ export interface Game { export interface Task { id: number; // id on db - gameId: number; contentId: number; - url: string; slug: string; - name: string; - description: string; - icon: string; - pointReward: number; - itemReward: number; - startTime?: string; - endTime?: string; - interval?: number; + gameId?: number | null; + categoryId?: number | null; + url?: string | null; + name?: string | null; + description?: string | null; + icon?: string | null; + pointReward?: number | null; + itemReward?: number | null; + startTime?: string | null; + endTime?: string | null; + interval?: number | null; status: number; completedAt?: string; } +export interface TaskCategory { + id: number; // id on db + contentId: number; + slug: string; + name?: string | null; + description?: string | null; + icon?: string | null; + active: boolean; + minPoint?: number; +} + +export type TaskCategoryInfo = { + id: number; + minPoint: number; + tasks: Task[]; +} + export interface GamePlay { id: number; // id on db gameId: number; diff --git a/packages/extension-koni-ui/src/contexts/DataContext.tsx b/packages/extension-koni-ui/src/contexts/DataContext.tsx index bbb3023ca4f..df0342c1dd2 100644 --- a/packages/extension-koni-ui/src/contexts/DataContext.tsx +++ b/packages/extension-koni-ui/src/contexts/DataContext.tsx @@ -1,12 +1,10 @@ // Copyright 2019-2022 @subwallet/extension-koni-ui authors & contributors // SPDX-License-Identifier: Apache-2.0 -import { useSelector } from '@subwallet/extension-koni-ui/hooks'; import { persistor, store, StoreName } from '@subwallet/extension-koni-ui/stores'; import { getMissionPoolData, subscribeAccountsData, subscribeAddressBook, subscribeAssetLogoMaps, subscribeAssetRegistry, subscribeAssetSettings, subscribeAuthorizeRequests, subscribeAuthUrls, subscribeBalance, subscribeBuyServices, subscribeBuyTokens, subscribeChainInfoMap, subscribeChainLogoMaps, subscribeChainStakingMetadata, subscribeChainStateMap, subscribeChainStatusMap, subscribeConfirmationRequests, subscribeConnectWCRequests, subscribeKeyringState, subscribeMantaPayConfig, subscribeMantaPaySyncingState, subscribeMetadataRequests, subscribeMultiChainAssetMap, subscribeNftCollections, subscribeNftItems, subscribePrice, subscribeProcessingCampaign, subscribeSigningRequests, subscribeStaking, subscribeStakingNominatorMetadata, subscribeStakingReward, subscribeTransactionRequests, subscribeTxHistory, subscribeUiSettings, subscribeWalletConnectSessions, subscribeWCNotSupportRequests, subscribeXcmRefMap } from '@subwallet/extension-koni-ui/stores/utils'; -import { isAccountAll } from '@subwallet/extension-koni-ui/utils'; import Bowser from 'bowser'; -import React, { useEffect, useRef } from 'react'; +import React from 'react'; import { Provider } from 'react-redux'; import { PersistGate } from 'redux-persist/integration/react'; diff --git a/packages/extension-koni-ui/src/contexts/ThemeContext.tsx b/packages/extension-koni-ui/src/contexts/ThemeContext.tsx index 745b2bd7fb3..a882b3b48c3 100644 --- a/packages/extension-koni-ui/src/contexts/ThemeContext.tsx +++ b/packages/extension-koni-ui/src/contexts/ThemeContext.tsx @@ -8,7 +8,7 @@ import { TelegramConnector } from '@subwallet/extension-koni-ui/connector/telegr import { DataContext } from '@subwallet/extension-koni-ui/contexts/DataContext'; import applyPreloadStyle from '@subwallet/extension-koni-ui/preloadStyle'; import { RootState } from '@subwallet/extension-koni-ui/stores'; -import { generateTheme, SW_THEME_CONFIGS, SwThemeConfig } from '@subwallet/extension-koni-ui/themes'; +import { generateTheme, getDefaultLogoMap, SW_THEME_CONFIGS, SwThemeConfig } from '@subwallet/extension-koni-ui/themes'; import { ConfigProvider, theme as reactUiTheme } from '@subwallet/react-ui'; import React, { useContext, useEffect, useMemo, useState } from 'react'; import { useSelector } from 'react-redux'; @@ -226,6 +226,48 @@ const GlobalStyle = createGlobalStyle(({ theme }) => { display: 'block', 'white-space': 'nowrap' } + }, + + // switcher + + '.ant-switch.ant-switch': { + background: extendToken.colorBgSecondary2 + }, + + '.ant-switch.ant-switch.ant-switch-checked': { + background: token.colorPrimary + }, + + // switcher + + '.ant-checkbox-checked.ant-checkbox-checked .ant-checkbox-inner': { + backgroundColor: extendToken.colorBgSecondary2, + borderColor: extendToken.colorBgSecondary2 + }, + + '.ant-checkbox-wrapper.ant-checkbox-wrapper:not(.ant-checkbox-wrapper-disabled):hover .ant-checkbox-checked:not(.ant-checkbox-disabled) .ant-checkbox-inner': { + backgroundColor: extendToken.colorBgSecondary2 + }, + + '.ant-checkbox-wrapper.ant-checkbox-wrapper:not(.ant-checkbox-wrapper-disabled):hover .ant-checkbox-checked:not(.ant-checkbox-disabled):after': { + borderColor: extendToken.colorBgSecondary2 + }, + + '.ant-checkbox-wrapper.ant-checkbox-wrapper:not(.ant-checkbox-wrapper-disabled):hover .ant-checkbox-inner, .ant-checkbox.ant-checkbox:not(.ant-checkbox-disabled):hover .ant-checkbox-inner': { + borderColor: extendToken.colorBgSecondary2 + }, + + '.ant-checkbox-wrapper-checked.ant-checkbox-wrapper-checked:not(.ant-checkbox-wrapper-disabled):hover .ant-checkbox-inner, .ant-checkbox-wrapper-checked.ant-checkbox-checked:not(.ant-checkbox-disabled):hover .ant-checkbox-inner': { + backgroundColor: extendToken.colorBgSecondary2, + borderColor: extendToken.colorBgSecondary2 + }, + + '.ant-checkbox-checked.ant-checkbox-checked:after': { + borderColor: extendToken.colorBgSecondary2 + }, + + '.ant-checkbox.ant-checkbox .ant-checkbox-inner:after': { + borderColor: token.colorPrimary } }); }); @@ -263,15 +305,20 @@ export function ThemeProvider ({ children }: ThemeProviderProps): React.ReactEle const dataContext = useContext(DataContext); const logoMaps = useSelector((state: RootState) => state.settings.logoMaps); const [themeReady, setThemeReady] = useState(false); + const themeName = useSelector((state: RootState) => state.settings.theme); const themeConfig = useMemo(() => { - const config = SW_THEME_CONFIGS[ThemeNames.LIGHT]; + const config = SW_THEME_CONFIGS[themeName] || SW_THEME_CONFIGS[ThemeNames.DEFAULT]; + + config.logoMap = getDefaultLogoMap(); Object.assign(config.logoMap.network, logoMaps.chainLogoMap); Object.assign(config.logoMap.symbol, logoMaps.assetLogoMap); + config.token = config.generateTokens(); + return config; - }, [logoMaps]); + }, [logoMaps.assetLogoMap, logoMaps.chainLogoMap, themeName]); useEffect(() => { dataContext.awaitStores(['settings']).then(() => { diff --git a/packages/extension-koni-ui/src/themes.ts b/packages/extension-koni-ui/src/themes.ts index 265e5dda5cf..086769fc5eb 100644 --- a/packages/extension-koni-ui/src/themes.ts +++ b/packages/extension-koni-ui/src/themes.ts @@ -18,7 +18,18 @@ export type GlobalToken = _GlobalToken; export interface ExtraToken { bodyBackgroundColor: string, logo: string, - defaultImagePlaceholder: string + defaultImagePlaceholder: string; + colorTextContent: string; + colorTextHeading: string; + colorTextDisabled: string; + colorTextHighLight: string; + colorBgSecondary1: string; + colorBgSecondary2: string; + colorBgSecondary3: string; + colorBgHover1: string; + colorBgHover2: string; + colorBgHover3: string; + colorBgGradient?: string; tokensScreenSuccessBackgroundColor: string, tokensScreenDangerBackgroundColor: string, tokensScreenInfoBackgroundColor: string, @@ -28,8 +39,6 @@ export type Theme = { id: ThemeNames; name: string; token: GlobalToken; - - // todo: add extend token later extendToken: ExtraToken, logoMap: Web3LogoMap, }; @@ -37,11 +46,9 @@ export type Theme = { export interface SwThemeConfig extends ThemeConfig { id: ThemeNames, name: string; - - generateExtraTokens: (token: AliasToken) => ExtraToken; - - customTokens: (token: AliasToken) => AliasToken; - logoMap: Web3LogoMap + generateTokens: () => Partial; + generateExtraTokens?: (token: AliasToken) => ExtraToken; + generateLogoMap?: () => Web3LogoMap } function genDefaultExtraTokens (token: AliasToken): ExtraToken { @@ -49,6 +56,16 @@ function genDefaultExtraTokens (token: AliasToken): ExtraToken { bodyBackgroundColor: token.colorBgBase, logo: subWalletLogo, defaultImagePlaceholder, + colorTextContent: token.colorTextDark3, + colorTextHeading: token.colorTextDark1, + colorTextDisabled: token.colorTextDark4, + colorTextHighLight: token.colorTextDark2, + colorBgSecondary1: '#fff', + colorBgSecondary2: '#1F1F23', + colorBgSecondary3: '#f0f0f0', + colorBgHover1: 'rgba(255, 255, 255, 0.85)', + colorBgHover2: 'rgba(31, 31, 35, 0.85)', + colorBgHover3: 'rgba(31, 31, 35, 0.06)', tokensScreenSuccessBackgroundColor: 'linear-gradient(180deg, rgba(76, 234, 172, 0.1) 16.47%, rgba(217, 217, 217, 0) 94.17%)', tokensScreenDangerBackgroundColor: 'linear-gradient(180deg, rgba(234, 76, 76, 0.1) 16.47%, rgba(217, 217, 217, 0) 94.17%)', tokensScreenInfoBackgroundColor: 'linear-gradient(180deg, rgba(0, 75, 255, 0.1) 16.47%, rgba(217, 217, 217, 0) 94.17%)' @@ -56,7 +73,7 @@ function genDefaultExtraTokens (token: AliasToken): ExtraToken { } // todo: will standardized logoMap later -const defaultLogoMap: Web3LogoMap = { +export const getDefaultLogoMap = (): Web3LogoMap => ({ ...logoMap, network: { ...IconMap, @@ -67,65 +84,173 @@ const defaultLogoMap: Web3LogoMap = { ...SwLogosMap }, default: SwLogosMap.default +}); + +const defaultToken: Partial = { + colorSecondary: '#44D5DE', + colorTextBase: '#1F1F23', + colorBgBase: '#ffffff', + colorBgSecondary: '#F0F0F0', + colorInfo: '#9FE3FF', + colorSuccess: '#0ACF88', + colorWarning: '#FFEC43', + colorError: '#FF2655', + colorText: '#1F1F23', + colorBgBorder: '#1F1F23', + colorBgDivider: 'rgba(31, 31, 35, 0.12)', + borderRadius: 8, + colorTextDark1: '#1F1F23', + colorTextDark2: 'rgba(31, 31, 35, 0.85)', + colorTextDark3: 'rgba(31, 31, 35, 0.65)', + colorTextDark4: 'rgba(31, 31, 35, 0.45)', + colorTextDark5: 'rgba(31, 31, 35, 0.35)', + colorTextDark6: 'rgba(31, 31, 35, 0.12)', + colorTextDark7: 'rgba(31, 31, 35, 0.06)', + colorTextLight1: '#ffffff', + colorTextLight2: 'rgba(255, 255, 255, 0.85)', + colorTextLight3: 'rgba(255, 255, 255, 0.65)', + colorTextLight4: 'rgba(255, 255, 255, 0.45)', + colorTextLight5: 'rgba(255, 255, 255, 0.30)', + colorTextLight6: 'rgba(255, 255, 255, 0.20)', + colorTextLight7: 'rgba(255, 255, 255, 0.12)', + 'gray-1': '#EEEEEE', + 'gray-2': '#DDDDDD', + 'gray-3': '#CCCCCC', + 'gray-4': '#BBBBBB', + 'gray-5': '#AAAAAA', + 'gray-6': '#999999' +}; + +const defaultThemeConfig = { + algorithm: SwReactUI.defaultAlgorithm, + components: { + SelectModal: { + disableAutoFocus: true + } + } }; // Todo: i18n for theme name // Implement theme from @subwallet/react-ui export const SW_THEME_CONFIGS: Record = { - [ThemeNames.LIGHT]: { - id: ThemeNames.LIGHT, - name: 'Dark', - algorithm: SwReactUI.defaultAlgorithm, - token: { - colorPrimary: '#3073F1', - colorSecondary: '#44D5DE', - colorTextBase: '#000000', - colorBgBase: '#ffffff', - colorBgSecondary: '#EEEEEE', - colorSuccess: '#4CEAAC', - colorWarning: '#FFD643', - colorError: '#F53861', - colorText: '#000000', - borderRadius: 8, - colorTextLight1: '#111', - colorTextLight2: '#222', - colorTextLight3: '#333', - colorTextLight4: '#444', - colorTextLight5: '#555', - colorTextLight6: '#666', - colorTextLight7: '#777', - colorTextLight8: '#888', - colorTextLight9: '#999', - 'gray-1': '#EEEEEE', - 'gray-2': '#DDDDDD', - 'gray-3': '#CCCCCC', - 'gray-4': '#BBBBBB', - 'gray-5': '#AAAAAA', - 'gray-6': '#999999' - }, - components: { - SelectModal: { - disableAutoFocus: true - } + [ThemeNames.DEFAULT]: { + ...defaultThemeConfig, + id: ThemeNames.DEFAULT, + name: 'Default', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#CBF147' + }; + } + }, + [ThemeNames.SKY]: { + ...defaultThemeConfig, + id: ThemeNames.SKY, + name: 'Sky', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#C7F0FF' + }; + } + }, + [ThemeNames.MORNING_SUNNY]: { + ...defaultThemeConfig, + id: ThemeNames.MORNING_SUNNY, + name: 'Morning sunny', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#FFF8C4' + }; + } + }, + [ThemeNames.SPRING]: { + ...defaultThemeConfig, + id: ThemeNames.SPRING, + name: 'Spring', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#7EEC79' + }; + } + }, + [ThemeNames.LAVENDER]: { + ...defaultThemeConfig, + id: ThemeNames.LAVENDER, + name: 'Lavender', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#BB9EFF' + }; + } + }, + [ThemeNames.SUNNY]: { + ...defaultThemeConfig, + id: ThemeNames.SUNNY, + name: 'Sunny', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#FBCE01' + }; + } + }, + [ThemeNames.BEGIE]: { + ...defaultThemeConfig, + id: ThemeNames.BEGIE, + name: 'Begie', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#EBD7C9' + }; + } + }, + [ThemeNames.CLOVE]: { + ...defaultThemeConfig, + id: ThemeNames.CLOVE, + name: 'Clove', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#CBF147' + }; }, - customTokens: (token) => (token), - generateExtraTokens: (token) => { - return { ...genDefaultExtraTokens(token) }; + generateExtraTokens: (token: AliasToken) => ({ + ...genDefaultExtraTokens(token), + colorBgGradient: 'linear-gradient(117deg, #A2F6C1 9.05%, #CBF147 91.43%)' + }) + }, + [ThemeNames.AURORA]: { + ...defaultThemeConfig, + id: ThemeNames.AURORA, + name: 'Aurora', + generateTokens: () => { + return { + ...defaultToken, + colorPrimary: '#C7F0FF' + }; }, - logoMap: defaultLogoMap + generateExtraTokens: (token: AliasToken) => ({ + ...genDefaultExtraTokens(token), + colorBgGradient: 'linear-gradient(117deg, #A2F6C1 9.05%, #9FE3FF 91.43%)' + }) } }; -export function generateTheme ({ customTokens, - generateExtraTokens, +export function generateTheme ({ generateExtraTokens, id, logoMap, name }: SwThemeConfig, token: GlobalToken): Theme { return { id, name, - token: customTokens(token), - extendToken: generateExtraTokens(token), - logoMap + token, + extendToken: generateExtraTokens?.(token) || genDefaultExtraTokens(token), + logoMap: logoMap || getDefaultLogoMap() } as Theme; } diff --git a/packages/extension-koni-ui/src/types/shop.ts b/packages/extension-koni-ui/src/types/shop.ts new file mode 100644 index 00000000000..a8a53763f77 --- /dev/null +++ b/packages/extension-koni-ui/src/types/shop.ts @@ -0,0 +1,17 @@ +// Copyright 2019-2022 @subwallet/extension-koni-ui authors & contributors +// SPDX-License-Identifier: Apache-2.0 + +export type ShopItemInfo = { + icon?: string; + gameItemId: string; + name: string; + gameId?: number; + limit?: number; + description?: string; + inventoryQuantity?: number; + itemGroup?: string; + itemGroupLevel?: number; + price: number; + disabled?: boolean; + usable?: boolean; +} diff --git a/packages/webapp/public/assets/reset.css b/packages/webapp/public/assets/reset.css index 3c5a5c4c79a..48081aba6e6 100644 --- a/packages/webapp/public/assets/reset.css +++ b/packages/webapp/public/assets/reset.css @@ -715,6 +715,36 @@ div#popup-container { border-radius: 100%; } +/* switcher */ + +.ant-switch.ant-switch { + width: 40px; + padding: 0; + min-width: 40px; + border-radius: 40px; + height: 24px; +} + +.ant-switch.ant-switch .ant-switch-handle { + width: 20px; + height: 20px; +} + +.ant-switch.ant-switch.ant-switch-checked .ant-switch-handle { + inset-inline-start: calc(100% - 22px); +} + +/* checkbox */ + +.ant-checkbox.ant-checkbox .ant-checkbox-inner { + border-radius: 4px; +} + +.ant-checkbox.ant-checkbox .ant-checkbox-inner:after { + top: 46%; + inset-inline-start: 25%; +} + /* slick */ .slick-list,.slick-slider,.slick-track{position:relative;display:block}.slick-loading .slick-slide,.slick-loading .slick-track{visibility:hidden}.slick-slider{box-sizing:border-box;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;-webkit-touch-callout:none;-khtml-user-select:none;-ms-touch-action:pan-y;touch-action:pan-y;-webkit-tap-highlight-color:transparent}.slick-list{overflow:hidden;margin:0;padding:0}.slick-list:focus{outline:0}.slick-list.dragging{cursor:pointer;cursor:hand}.slick-slider .slick-list,.slick-slider .slick-track{-webkit-transform:translate3d(0,0,0);-moz-transform:translate3d(0,0,0);-ms-transform:translate3d(0,0,0);-o-transform:translate3d(0,0,0);transform:translate3d(0,0,0)}.slick-track{top:0;left:0}.slick-track:after,.slick-track:before{display:table;content:''}.slick-track:after{clear:both}.slick-slide{display:none;float:left;height:100%;min-height:1px}[dir=rtl] .slick-slide{float:right}.slick-slide img{display:block}.slick-slide.slick-loading img{display:none}.slick-slide.dragging img{pointer-events:none}.slick-initialized .slick-slide{display:block}.slick-vertical .slick-slide{display:block;height:auto;border:1px solid transparent}.slick-arrow.slick-hidden{display:none}