From c7aa8e5267ace5cb0b280b27ff237472b22ba331 Mon Sep 17 00:00:00 2001 From: Panteleymonchuk Date: Wed, 4 Dec 2024 17:01:58 +0200 Subject: [PATCH] feat(wallet-dashboard): style selected visual Assets (#4085) * feat(wallet-dashboard): style selected visual Assets. * refactor(core): destructure metaKeys and metaValues from attributes * refactor(wallet): move Collapsible component to core. * feat(dashboard): integrate useAssetsDialog for asset details view * fix(assets): update import path and enhance text styling in DetailsView * refactor(wallet-dashboard): move state to page * refactor(wallet-dashboard): rename handler functions for consistency and clarity * refactor(dashboard): update state for asset view, improve code * fix(wallet-dashboard): unify asset transfer success and error handling in AssetDialog * refactor(dashboard): adjust z-index for dialog and notifications; * feat(dashboard): add refetch functionality after asset transfered * refactor(dashboard): rename callbacks for clarity in AssetDialog * refactor(dashboard, cove): rename hooks, remove duplication. * refactor(dashboard): remove unused asset details page --------- Co-authored-by: Bran <52735957+brancoder@users.noreply.github.com> Co-authored-by: Marc Espin --- .../components/collapsible/Collapsible.tsx} | 0 apps/core/src/components/collapsible/index.ts | 4 + apps/core/src/components/index.ts | 2 +- apps/core/src/hooks/index.ts | 5 +- .../src}/hooks/useFileExtensionType.ts | 0 .../{useGetNFTMeta.ts => useGetNFTDisplay.ts} | 4 +- .../ui/app => core/src}/hooks/useMediaUrl.ts | 0 .../app => core/src}/hooks/useNFTBasicData.ts | 2 +- apps/core/src/hooks/useNftDetails.ts | 98 +++++++++ .../ui/app => core/src}/hooks/useOwnedNFT.ts | 2 +- apps/core/src/utils/index.ts | 1 + .../src/utils}/truncateString.ts | 0 .../organisms/accordion/Accordion.tsx | 2 +- .../components/organisms/dialog/Dialog.tsx | 2 +- .../(protected)/assets/[objectId]/page.tsx | 44 ---- .../app/(protected)/assets/page.tsx | 10 + .../components/AssetsList.tsx | 9 +- .../components/Dialogs/Assets/AssetDialog.tsx | 81 ++++++++ .../Assets/constants/AssetsDialogView.ts | 7 + .../Dialogs/Assets/constants/index.ts | 4 + .../components/Dialogs/Assets/index.ts | 5 + .../Dialogs/Assets/views/DetailsView.tsx | 189 ++++++++++++++++++ .../Dialogs/Assets/views/SendView.tsx | 70 +++++++ .../components/Dialogs/Assets/views/index.ts | 5 + .../Notifications/Notifications.tsx | 2 +- .../components/tiles/AssetTileLink.tsx | 36 +--- .../components/tiles/VisualAssetTile.tsx | 4 +- .../hooks/useCreateSendAssetTransaction.ts | 25 +-- .../providers/AppProviders.tsx | 43 ++-- .../ui/app/components/nft-display/index.tsx | 4 +- .../src/ui/app/helpers/formatAccountName.ts | 2 +- apps/wallet/src/ui/app/helpers/index.ts | 1 - apps/wallet/src/ui/app/hooks/index.ts | 4 - .../pages/accounts/manage/AccountGroup.tsx | 3 +- .../transaction-request/GasFees.tsx | 3 +- .../TransactionDetails/Command.tsx | 2 +- .../TransactionDetails/index.tsx | 2 +- .../ui/app/pages/home/kiosk-details/index.tsx | 3 +- .../ui/app/pages/home/nft-details/index.tsx | 126 ++++-------- .../ui/app/pages/home/nft-transfer/index.tsx | 3 +- .../ui/app/pages/home/nfts/HiddenAsset.tsx | 4 +- .../ui/app/pages/home/tokens/TokenList.tsx | 2 +- .../cards/ObjectChanges.tsx | 2 +- 43 files changed, 587 insertions(+), 230 deletions(-) rename apps/{wallet/src/ui/app/shared/collapse/index.tsx => core/src/components/collapsible/Collapsible.tsx} (100%) create mode 100644 apps/core/src/components/collapsible/index.ts rename apps/{wallet/src/ui/app => core/src}/hooks/useFileExtensionType.ts (100%) rename apps/core/src/hooks/{useGetNFTMeta.ts => useGetNFTDisplay.ts} (91%) rename apps/{wallet/src/ui/app => core/src}/hooks/useMediaUrl.ts (100%) rename apps/{wallet/src/ui/app => core/src}/hooks/useNFTBasicData.ts (92%) create mode 100644 apps/core/src/hooks/useNftDetails.ts rename apps/{wallet/src/ui/app => core/src}/hooks/useOwnedNFT.ts (94%) rename apps/{wallet/src/ui/app/helpers => core/src/utils}/truncateString.ts (100%) delete mode 100644 apps/wallet-dashboard/app/(protected)/assets/[objectId]/page.tsx create mode 100644 apps/wallet-dashboard/components/Dialogs/Assets/AssetDialog.tsx create mode 100644 apps/wallet-dashboard/components/Dialogs/Assets/constants/AssetsDialogView.ts create mode 100644 apps/wallet-dashboard/components/Dialogs/Assets/constants/index.ts create mode 100644 apps/wallet-dashboard/components/Dialogs/Assets/index.ts create mode 100644 apps/wallet-dashboard/components/Dialogs/Assets/views/DetailsView.tsx create mode 100644 apps/wallet-dashboard/components/Dialogs/Assets/views/SendView.tsx create mode 100644 apps/wallet-dashboard/components/Dialogs/Assets/views/index.ts diff --git a/apps/wallet/src/ui/app/shared/collapse/index.tsx b/apps/core/src/components/collapsible/Collapsible.tsx similarity index 100% rename from apps/wallet/src/ui/app/shared/collapse/index.tsx rename to apps/core/src/components/collapsible/Collapsible.tsx diff --git a/apps/core/src/components/collapsible/index.ts b/apps/core/src/components/collapsible/index.ts new file mode 100644 index 00000000000..dbcea5060fa --- /dev/null +++ b/apps/core/src/components/collapsible/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './Collapsible'; diff --git a/apps/core/src/components/index.ts b/apps/core/src/components/index.ts index a2aa837dfc6..9f005714aa5 100644 --- a/apps/core/src/components/index.ts +++ b/apps/core/src/components/index.ts @@ -5,5 +5,5 @@ export * from './coin'; export * from './icon'; export * from './Inputs'; export * from './QR'; - +export * from './collapsible'; export * from './providers'; diff --git a/apps/core/src/hooks/index.ts b/apps/core/src/hooks/index.ts index a508b67233b..13c49c813b6 100644 --- a/apps/core/src/hooks/index.ts +++ b/apps/core/src/hooks/index.ts @@ -32,7 +32,7 @@ export * from './useQueryTransactionsByAddress'; export * from './useGetTransaction'; export * from './useExtendedTransactionSummary'; export * from './useSortedCoinsByCategories'; -export * from './useGetNFTMeta'; +export * from './useGetNFTDisplay'; export * from './useIotaAddressValidation'; export * from './useUnlockTimelockedObjectsTransaction'; export * from './useGetAllOwnedObjects'; @@ -43,6 +43,9 @@ export * from './useTransactionData'; export * from './useGetStakingValidatorDetails'; export * from './useCursorPagination'; export * from './useTheme'; +export * from './useNFTBasicData'; +export * from './useOwnedNFT'; +export * from './useNftDetails'; export * from './useCountdownByTimestamp'; export * from './stake'; diff --git a/apps/wallet/src/ui/app/hooks/useFileExtensionType.ts b/apps/core/src/hooks/useFileExtensionType.ts similarity index 100% rename from apps/wallet/src/ui/app/hooks/useFileExtensionType.ts rename to apps/core/src/hooks/useFileExtensionType.ts diff --git a/apps/core/src/hooks/useGetNFTMeta.ts b/apps/core/src/hooks/useGetNFTDisplay.ts similarity index 91% rename from apps/core/src/hooks/useGetNFTMeta.ts rename to apps/core/src/hooks/useGetNFTDisplay.ts index b63df42ce5c..0819a967f20 100644 --- a/apps/core/src/hooks/useGetNFTMeta.ts +++ b/apps/core/src/hooks/useGetNFTDisplay.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useGetObject } from './'; +import { useGetObject } from '.'; import { useMemo } from 'react'; export type NFTMetadata = { @@ -11,7 +11,7 @@ export type NFTMetadata = { url: string; }; -export function useGetNFTMeta(objectID: string) { +export function useGetNFTDisplay(objectID: string) { const resp = useGetObject(objectID); const nftMeta = useMemo(() => { if (!resp.data) return null; diff --git a/apps/wallet/src/ui/app/hooks/useMediaUrl.ts b/apps/core/src/hooks/useMediaUrl.ts similarity index 100% rename from apps/wallet/src/ui/app/hooks/useMediaUrl.ts rename to apps/core/src/hooks/useMediaUrl.ts diff --git a/apps/wallet/src/ui/app/hooks/useNFTBasicData.ts b/apps/core/src/hooks/useNFTBasicData.ts similarity index 92% rename from apps/wallet/src/ui/app/hooks/useNFTBasicData.ts rename to apps/core/src/hooks/useNFTBasicData.ts index 2b9360bf94a..227a9707de1 100644 --- a/apps/wallet/src/ui/app/hooks/useNFTBasicData.ts +++ b/apps/core/src/hooks/useNFTBasicData.ts @@ -7,7 +7,7 @@ import type { IotaObjectData } from '@iota/iota-sdk/client'; import useFileExtensionType from './useFileExtensionType'; import useMediaUrl from './useMediaUrl'; -export default function useNFTBasicData(nftObj: IotaObjectData | null) { +export function useNFTBasicData(nftObj: IotaObjectData | null) { const nftObjectID = nftObj?.objectId || null; const filePath = useMediaUrl(nftObj?.content || null); let objType = null; diff --git a/apps/core/src/hooks/useNftDetails.ts b/apps/core/src/hooks/useNftDetails.ts new file mode 100644 index 00000000000..f76001b0273 --- /dev/null +++ b/apps/core/src/hooks/useNftDetails.ts @@ -0,0 +1,98 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 +import { + useGetNFTDisplay, + useOwnedNFT, + useNFTBasicData, + useGetKioskContents, + useIsAssetTransferable, +} from './'; +import { formatAddress } from '@iota/iota-sdk/utils'; +import { truncateString } from '../utils'; + +type NftField = { keys: string[]; values: string[] }; + +type NftFields = { + metadata?: { fields?: { attributes?: { fields?: NftField } } }; +}; + +export function useNftDetails(nftId: string, accountAddress: string | null) { + const { data: objectData, isPending: isNftLoading } = useOwnedNFT(nftId || '', accountAddress); + const { data } = useGetKioskContents(accountAddress); + + const isContainedInKiosk = data?.lookup.get(nftId!); + const kioskItem = data?.list.find((k) => k.data?.objectId === nftId); + + const { data: isAssetTransferable, isLoading: isCheckingAssetTransferability } = + useIsAssetTransferable(objectData); + + const { nftFields } = useNFTBasicData(objectData); + + const { data: nftDisplayData, isPending: isPendingNftDislpay } = useGetNFTDisplay(nftId); + + const nftName = nftDisplayData?.name || formatAddress(nftId); + const nftImageUrl = nftDisplayData?.imageUrl || ''; + + // Extract either the attributes, or use the top-level NFT fields: + const { keys: metaKeys, values: metaValues } = + (nftFields as NftFields)?.metadata?.fields?.attributes?.fields || + Object.entries(nftFields ?? {}) + .filter(([key]) => key !== 'id') + .reduce( + (acc, [key, value]) => { + acc.keys.push(key); + acc.values.push(value as string); + return acc; + }, + { keys: [], values: [] }, + ); + + const ownerAddress = + (objectData?.owner && + typeof objectData?.owner === 'object' && + 'AddressOwner' in objectData.owner && + objectData.owner.AddressOwner) || + ''; + + function formatMetaValue(value: string | object) { + if (typeof value === 'object') { + return { + value: JSON.stringify(value), + valueLink: undefined, + }; + } else { + if (value.includes('http')) { + return { + value: value.startsWith('http') + ? truncateString(value, 20, 8) + : formatAddress(value), + valueLink: value, + }; + } + return { + value: value, + valueLink: undefined, + }; + } + } + + const isLoading = isNftLoading || isCheckingAssetTransferability || isPendingNftDislpay; + + return { + isLoading, + objectData, + isNftLoading, + nftName, + nftImageUrl, + ownerAddress, + isCheckingAssetTransferability, + isAssetTransferable, + metaKeys, + metaValues, + formatMetaValue, + isContainedInKiosk, + kioskItem, + nftDisplayData, + isPendingNftDislpay, + }; +} diff --git a/apps/wallet/src/ui/app/hooks/useOwnedNFT.ts b/apps/core/src/hooks/useOwnedNFT.ts similarity index 94% rename from apps/wallet/src/ui/app/hooks/useOwnedNFT.ts rename to apps/core/src/hooks/useOwnedNFT.ts index 0e308a484ce..a7f96ac25d2 100644 --- a/apps/wallet/src/ui/app/hooks/useOwnedNFT.ts +++ b/apps/core/src/hooks/useOwnedNFT.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useGetKioskContents, useGetObject } from '@iota/core'; +import { useGetKioskContents, useGetObject } from './'; import { useMemo } from 'react'; export function useOwnedNFT(nftObjectId: string | null, address: string | null) { diff --git a/apps/core/src/utils/index.ts b/apps/core/src/utils/index.ts index 4d57564fa92..9973f811e10 100644 --- a/apps/core/src/utils/index.ts +++ b/apps/core/src/utils/index.ts @@ -20,6 +20,7 @@ export * from './getDelegationDataByStakeId'; export * from './api-env'; export * from './getExplorerPaths'; export * from './getExplorerLink'; +export * from './truncateString'; export * from './stake'; export * from './transaction'; diff --git a/apps/wallet/src/ui/app/helpers/truncateString.ts b/apps/core/src/utils/truncateString.ts similarity index 100% rename from apps/wallet/src/ui/app/helpers/truncateString.ts rename to apps/core/src/utils/truncateString.ts diff --git a/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx b/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx index ce4d80f97c6..1489c52c1d5 100644 --- a/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx +++ b/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx @@ -7,7 +7,7 @@ import { ArrowDown } from '@iota/ui-icons'; import { Button, ButtonType } from '@/lib'; import { ICON_STYLE } from './accordion.classes'; -interface AccordionHeaderProps { +export interface AccordionHeaderProps { /** * Flag for show/hide content */ diff --git a/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx b/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx index 58575ba3dcb..5bd43a1b7ba 100644 --- a/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx +++ b/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx @@ -74,7 +74,7 @@ const DialogContent = React.forwardRef< { - const params = useParams(); - const objectId = params.objectId as string; - const { data: asset } = useGetObject(objectId); - const { data: isAssetTransferable } = useIsAssetTransferable(asset?.data); - const activeAccount = useCurrentAccount(); - - const { openPopup, closePopup } = usePopups(); - - const showSendAssetPopup = useCallback(() => { - if (asset?.data) { - openPopup(); - } - }, [asset, openPopup, closePopup]); - - return ( -
- - {asset?.data ? ( - - ) : ( -
Asset not found
- )} - {isAssetTransferable && activeAccount ? ( - - ) : null} -
- ); -}; - -export default VisualAssetDetailPage; diff --git a/apps/wallet-dashboard/app/(protected)/assets/page.tsx b/apps/wallet-dashboard/app/(protected)/assets/page.tsx index 4ff62fed3f8..ed96fe0a519 100644 --- a/apps/wallet-dashboard/app/(protected)/assets/page.tsx +++ b/apps/wallet-dashboard/app/(protected)/assets/page.tsx @@ -10,6 +10,7 @@ import { IotaObjectData } from '@iota/iota-sdk/client'; import { useState } from 'react'; import { AssetCategory } from '@/lib/enums'; import { AssetList } from '@/components/AssetsList'; +import { AssetDialog } from '@/components/Dialogs/Assets'; const OBJECTS_PER_REQ = 50; @@ -25,6 +26,7 @@ const ASSET_CATEGORIES: { label: string; value: AssetCategory }[] = [ ]; export default function AssetsDashboardPage(): React.JSX.Element { + const [selectedAsset, setSelectedAsset] = useState(null); const [selectedCategory, setSelectedCategory] = useState(AssetCategory.Visual); const account = useCurrentAccount(); const { data, isFetching, fetchNextPage, hasNextPage } = useGetOwnedObjects( @@ -49,6 +51,10 @@ export default function AssetsDashboardPage(): React.JSX.Element { } } + function onAssetClick(asset: IotaObjectData) { + setSelectedAsset(asset); + } + return ( @@ -67,10 +73,14 @@ export default function AssetsDashboardPage(): React.JSX.Element { <AssetList assets={assets} selectedCategory={selectedCategory} + onClick={onAssetClick} hasNextPage={hasNextPage} isFetchingNextPage={isFetching} fetchNextPage={fetchNextPage} /> + {selectedAsset && ( + <AssetDialog onClose={() => setSelectedAsset(null)} asset={selectedAsset} /> + )} </div> </Panel> ); diff --git a/apps/wallet-dashboard/components/AssetsList.tsx b/apps/wallet-dashboard/components/AssetsList.tsx index 0835f0a98e5..adb9e251581 100644 --- a/apps/wallet-dashboard/components/AssetsList.tsx +++ b/apps/wallet-dashboard/components/AssetsList.tsx @@ -15,6 +15,7 @@ interface AssetListProps { hasNextPage: boolean; isFetchingNextPage: boolean; fetchNextPage: () => void; + onClick: (asset: IotaObjectData) => void; } const ASSET_LAYOUT: Record<AssetCategory, string> = { @@ -29,6 +30,7 @@ export function AssetList({ hasNextPage, isFetchingNextPage, fetchNextPage, + onClick, }: AssetListProps): React.JSX.Element { const observerElem = useRef<HTMLDivElement | null>(null); const { isIntersecting } = useOnScreen(observerElem); @@ -43,7 +45,12 @@ export function AssetList({ return ( <div className={cl('max-h-[600px]', ASSET_LAYOUT[selectedCategory])}> {assets.map((asset) => ( - <AssetTileLink key={asset.digest} asset={asset} type={selectedCategory} /> + <AssetTileLink + key={asset.digest} + asset={asset} + type={selectedCategory} + onClick={onClick} + /> ))} <div ref={observerElem}> {isSpinnerVisible ? ( diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/AssetDialog.tsx b/apps/wallet-dashboard/components/Dialogs/Assets/AssetDialog.tsx new file mode 100644 index 00000000000..ec3d8d38d11 --- /dev/null +++ b/apps/wallet-dashboard/components/Dialogs/Assets/AssetDialog.tsx @@ -0,0 +1,81 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import React, { useState } from 'react'; +import { Dialog } from '@iota/apps-ui-kit'; +import { FormikProvider, useFormik } from 'formik'; +import { useCurrentAccount } from '@iota/dapp-kit'; +import { createNftSendValidationSchema } from '@iota/core'; +import { DetailsView, SendView } from './views'; +import { IotaObjectData } from '@iota/iota-sdk/client'; +import { AssetsDialogView } from './constants'; +import { useCreateSendAssetTransaction, useNotifications } from '@/hooks'; +import { NotificationType } from '@/stores/notificationStore'; + +interface AssetsDialogProps { + onClose: () => void; + asset: IotaObjectData; +} + +interface FormValues { + to: string; +} + +const INITIAL_VALUES: FormValues = { + to: '', +}; + +export function AssetDialog({ onClose, asset }: AssetsDialogProps): JSX.Element { + const [view, setView] = useState<AssetsDialogView>(AssetsDialogView.Details); + const account = useCurrentAccount(); + const activeAddress = account?.address ?? ''; + const objectId = asset?.objectId ?? ''; + const { addNotification } = useNotifications(); + const validationSchema = createNftSendValidationSchema(activeAddress, objectId); + + const { mutation: sendAsset } = useCreateSendAssetTransaction(objectId); + + const formik = useFormik<FormValues>({ + initialValues: INITIAL_VALUES, + validationSchema: validationSchema, + onSubmit: onSubmit, + validateOnChange: true, + }); + + async function onSubmit(values: FormValues) { + try { + await sendAsset.mutateAsync(values.to); + addNotification('Transfer transaction successful', NotificationType.Success); + onClose(); + setView(AssetsDialogView.Details); + } catch { + addNotification('Transfer transaction failed', NotificationType.Error); + } + } + + function onDetailsSend() { + setView(AssetsDialogView.Send); + } + + function onSendViewBack() { + setView(AssetsDialogView.Details); + } + function onOpenChange() { + setView(AssetsDialogView.Details); + onClose(); + } + return ( + <Dialog open onOpenChange={onOpenChange}> + <FormikProvider value={formik}> + <> + {view === AssetsDialogView.Details && ( + <DetailsView asset={asset} onClose={onOpenChange} onSend={onDetailsSend} /> + )} + {view === AssetsDialogView.Send && ( + <SendView asset={asset} onClose={onOpenChange} onBack={onSendViewBack} /> + )} + </> + </FormikProvider> + </Dialog> + ); +} diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/constants/AssetsDialogView.ts b/apps/wallet-dashboard/components/Dialogs/Assets/constants/AssetsDialogView.ts new file mode 100644 index 00000000000..88cb34c15b1 --- /dev/null +++ b/apps/wallet-dashboard/components/Dialogs/Assets/constants/AssetsDialogView.ts @@ -0,0 +1,7 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export enum AssetsDialogView { + Details = 'Details', + Send = 'Send', +} diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/constants/index.ts b/apps/wallet-dashboard/components/Dialogs/Assets/constants/index.ts new file mode 100644 index 00000000000..76b42558333 --- /dev/null +++ b/apps/wallet-dashboard/components/Dialogs/Assets/constants/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './AssetsDialogView'; diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/index.ts b/apps/wallet-dashboard/components/Dialogs/Assets/index.ts new file mode 100644 index 00000000000..3a6df1b0b55 --- /dev/null +++ b/apps/wallet-dashboard/components/Dialogs/Assets/index.ts @@ -0,0 +1,5 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './AssetDialog'; +export * from './constants'; diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/views/DetailsView.tsx b/apps/wallet-dashboard/components/Dialogs/Assets/views/DetailsView.tsx new file mode 100644 index 00000000000..6ca3452ffa6 --- /dev/null +++ b/apps/wallet-dashboard/components/Dialogs/Assets/views/DetailsView.tsx @@ -0,0 +1,189 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import React from 'react'; +import { ExplorerLinkType, useNftDetails, Collapsible } from '@iota/core'; +import { + Button, + ButtonType, + Header, + KeyValueInfo, + VisualAssetCard, + VisualAssetType, +} from '@iota/apps-ui-kit'; +import Link from 'next/link'; +import { formatAddress } from '@iota/iota-sdk/utils'; +import { Layout, LayoutBody, LayoutFooter } from '../../Staking/views/Layout'; +import { IotaObjectData } from '@iota/iota-sdk/client'; +import { ExplorerLink } from '@/components/ExplorerLink'; +import { useCurrentAccount } from '@iota/dapp-kit'; + +interface DetailsViewProps { + asset: IotaObjectData; + onClose: () => void; + onSend: () => void; +} + +export function DetailsView({ onClose, asset, onSend }: DetailsViewProps) { + const account = useCurrentAccount(); + + const senderAddress = account?.address ?? ''; + const objectId = asset.objectId; + + const { + nftName, + nftImageUrl, + nftDisplayData, + ownerAddress, + isAssetTransferable, + metaKeys, + metaValues, + formatMetaValue, + isContainedInKiosk, + kioskItem, + } = useNftDetails(objectId, senderAddress); + + function handleMoreAboutKiosk() { + window.open('https://docs.iota.org/references/ts-sdk/kiosk/', '_blank'); + } + + function handleMarketplace() { + // TODO: https://github.com/iotaledger/iota/issues/4024 + window.open('https://docs.iota.org/references/ts-sdk/kiosk/', '_blank'); + } + + return ( + <Layout> + <Header title="Asset" onClose={onClose} titleCentered /> + <LayoutBody> + <div className="flex w-full flex-col items-center justify-center gap-xs"> + <div className="w-[172px]"> + <VisualAssetCard + assetSrc={nftImageUrl} + assetTitle={nftName} + assetType={VisualAssetType.Image} + altText={nftName || 'NFT'} + isHoverable={false} + /> + </div> + <ExplorerLink type={ExplorerLinkType.Object} objectID={objectId}> + <Button type={ButtonType.Ghost} text="View on Explorer" /> + </ExplorerLink> + <div className="flex w-full flex-col gap-md"> + <div className="flex flex-col gap-xxxs"> + <span className="text-title-lg text-neutral-10 dark:text-neutral-92"> + {nftDisplayData?.name} + </span> + {nftDisplayData?.description ? ( + <span className="text-body-md text-neutral-60"> + {nftDisplayData?.description} + </span> + ) : null} + </div> + + {(nftDisplayData?.projectUrl || !!nftDisplayData?.creator) && ( + <div className="flex flex-col gap-xs"> + {nftDisplayData?.projectUrl && ( + <KeyValueInfo + keyText="Website" + value={ + <Link href={nftDisplayData?.projectUrl}> + {nftDisplayData?.projectUrl} + </Link> + } + fullwidth + /> + )} + {nftDisplayData?.creator && ( + <KeyValueInfo + keyText="Creator" + value={nftDisplayData?.creator ?? '-'} + fullwidth + /> + )} + </div> + )} + + <Collapsible defaultOpen title="Details"> + <div className="flex flex-col gap-xs px-md pb-xs pt-sm"> + {ownerAddress && ( + <KeyValueInfo + keyText="Owner" + value={ + <ExplorerLink + type={ExplorerLinkType.Address} + address={ownerAddress} + > + {formatAddress(ownerAddress)} + </ExplorerLink> + } + fullwidth + /> + )} + {objectId && ( + <KeyValueInfo + keyText="Object ID" + value={formatAddress(objectId)} + fullwidth + /> + )} + </div> + </Collapsible> + {metaKeys.length ? ( + <Collapsible defaultOpen title="Attributes"> + <div className="flex flex-col gap-xs px-md pb-xs pt-sm"> + {metaKeys.map((aKey, idx) => { + const { value, valueLink } = formatMetaValue( + metaValues[idx], + ); + return ( + <KeyValueInfo + key={idx} + keyText={aKey} + value={ + valueLink ? ( + <Link key={aKey} href={valueLink || ''}> + {value} + </Link> + ) : ( + value + ) + } + fullwidth + /> + ); + })} + </div> + </Collapsible> + ) : null} + </div> + </div> + </LayoutBody> + <LayoutFooter> + <div className="flex flex-col"> + {isContainedInKiosk && kioskItem?.isLocked ? ( + <div className="flex flex-col gap-2"> + <Button + type={ButtonType.Secondary} + onClick={handleMoreAboutKiosk} + text="Learn more about Kiosks" + /> + <Button + type={ButtonType.Primary} + onClick={handleMarketplace} + text="Marketplace" + /> + </div> + ) : ( + <Button + disabled={!isAssetTransferable} + onClick={onSend} + text="Send" + fullWidth + /> + )} + </div> + </LayoutFooter> + </Layout> + ); +} diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/views/SendView.tsx b/apps/wallet-dashboard/components/Dialogs/Assets/views/SendView.tsx new file mode 100644 index 00000000000..973d0407fd9 --- /dev/null +++ b/apps/wallet-dashboard/components/Dialogs/Assets/views/SendView.tsx @@ -0,0 +1,70 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import React from 'react'; +import { AddressInput, useNftDetails } from '@iota/core'; +import { useFormikContext } from 'formik'; +import { Layout, LayoutBody, LayoutFooter } from '../../Staking/views/Layout'; +import { + Button, + ButtonHtmlType, + Header, + VisualAssetCard, + VisualAssetType, + Title, +} from '@iota/apps-ui-kit'; +import { Loader } from '@iota/ui-icons'; +import { useCurrentAccount } from '@iota/dapp-kit'; +import { IotaObjectData } from '@iota/iota-sdk/client'; + +interface SendViewProps { + asset: IotaObjectData; + onClose: () => void; + onBack: () => void; +} + +export function SendView({ asset, onClose, onBack }: SendViewProps) { + const { isValid, dirty, isSubmitting, submitForm } = useFormikContext(); + + const account = useCurrentAccount(); + + const senderAddress = account?.address ?? ''; + const objectId = asset?.objectId || ''; + + const { nftName, nftImageUrl } = useNftDetails(objectId, senderAddress); + return ( + <Layout> + <Header title="Send asset" onClose={onClose} titleCentered onBack={onBack} /> + <LayoutBody> + <div className="flex w-full flex-col items-center justify-center gap-xs"> + <div className="w-[172px]"> + <VisualAssetCard + assetSrc={nftImageUrl} + assetTitle={nftName} + assetType={VisualAssetType.Image} + altText={nftName || 'NFT'} + isHoverable={false} + /> + </div> + <div className="flex w-full flex-col gap-md"> + <div className="flex flex-col items-center gap-xxxs"> + <Title title={nftName} /> + </div> + <AddressInput name="to" placeholder="Enter Address" /> + </div> + </div> + </LayoutBody> + <LayoutFooter> + <Button + fullWidth + htmlType={ButtonHtmlType.Submit} + disabled={!(isValid && dirty) || isSubmitting} + text="Send" + icon={isSubmitting ? <Loader className="animate-spin" /> : undefined} + iconAfterText + onClick={submitForm} + /> + </LayoutFooter> + </Layout> + ); +} diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/views/index.ts b/apps/wallet-dashboard/components/Dialogs/Assets/views/index.ts new file mode 100644 index 00000000000..fcc3856cac3 --- /dev/null +++ b/apps/wallet-dashboard/components/Dialogs/Assets/views/index.ts @@ -0,0 +1,5 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './DetailsView'; +export * from './SendView'; diff --git a/apps/wallet-dashboard/components/Notifications/Notifications.tsx b/apps/wallet-dashboard/components/Notifications/Notifications.tsx index 3e0b0aad6d5..77493b164d9 100644 --- a/apps/wallet-dashboard/components/Notifications/Notifications.tsx +++ b/apps/wallet-dashboard/components/Notifications/Notifications.tsx @@ -55,7 +55,7 @@ function Notification(props: { notification: NotificationData }): JSX.Element { export default function Notifications(): JSX.Element { const notifications = useNotificationStore((state) => state.notifications); return ( - <div className="absolute right-2 top-1 z-50"> + <div className="fixed right-2 top-1 z-[99999]"> {notifications.map((notification) => ( <Notification key={`${notification.index}`} notification={notification} /> ))} diff --git a/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx b/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx index 3bf06075604..64c4d805513 100644 --- a/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx +++ b/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx @@ -3,50 +3,30 @@ 'use client'; -import { ASSETS_ROUTE } from '@/lib/constants/routes.constants'; import { AssetCategory } from '@/lib/enums'; import { VisibilityOff } from '@iota/ui-icons'; import { VisualAssetTile } from '.'; import { IotaObjectData } from '@iota/iota-sdk/client'; import { NonVisualAssetCard } from './NonVisualAssetTile'; -import { useExplorerLinkGetter } from '@/hooks'; -import Link from 'next/link'; -import { ExplorerLinkType } from '@iota/core'; interface AssetTileLinkProps { asset: IotaObjectData; type: AssetCategory; + onClick: (asset: IotaObjectData) => void; } -export function AssetTileLink({ asset, type }: AssetTileLinkProps): React.JSX.Element { - const getExplorerLink = useExplorerLinkGetter(); - const linkProps = getAssetLinkProps(asset); - - function getAssetLinkProps(asset: IotaObjectData): React.ComponentProps<typeof Link> { - if (type === AssetCategory.Visual) { - return { href: ASSETS_ROUTE.path + `/${asset.objectId}` }; - } else { - const explorerLink = - getExplorerLink({ - type: ExplorerLinkType.Object, - objectID: asset.objectId, - }) ?? ''; - - return { - href: explorerLink, - target: '_blank', - rel: 'noopener noreferrer', - }; - } +export function AssetTileLink({ asset, type, onClick }: AssetTileLinkProps): React.JSX.Element { + function handleClick() { + onClick(asset); } return ( - <Link {...linkProps}> + <> {type === AssetCategory.Visual ? ( - <VisualAssetTile asset={asset} icon={<VisibilityOff />} /> + <VisualAssetTile asset={asset} icon={<VisibilityOff />} onClick={handleClick} /> ) : ( - <NonVisualAssetCard asset={asset} /> + <NonVisualAssetCard asset={asset} onClick={handleClick} /> )} - </Link> + </> ); } diff --git a/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx b/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx index 6fbc03801c4..642d06daff0 100644 --- a/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx +++ b/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx @@ -5,7 +5,7 @@ import { IotaObjectData } from '@iota/iota-sdk/client'; import React from 'react'; -import { useGetNFTMeta } from '@iota/core'; +import { useGetNFTDisplay } from '@iota/core'; import { FlexDirection } from '@/lib/ui/enums'; import { VisualAssetCard, VisualAssetType, type VisualAssetCardProps } from '@iota/apps-ui-kit'; @@ -20,7 +20,7 @@ export function VisualAssetTile({ onIconClick, icon, }: AssetCardProps): React.JSX.Element | null { - const { data: nftMeta } = useGetNFTMeta(asset.objectId); + const { data: nftMeta } = useGetNFTDisplay(asset.objectId); if (!asset.display || !nftMeta || !nftMeta.imageUrl) { return null; diff --git a/apps/wallet-dashboard/hooks/useCreateSendAssetTransaction.ts b/apps/wallet-dashboard/hooks/useCreateSendAssetTransaction.ts index 82dd2b5fd6d..aa50c4fac7c 100644 --- a/apps/wallet-dashboard/hooks/useCreateSendAssetTransaction.ts +++ b/apps/wallet-dashboard/hooks/useCreateSendAssetTransaction.ts @@ -1,28 +1,16 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useIotaClient, useSignAndExecuteTransaction } from '@iota/dapp-kit'; +import { useSignAndExecuteTransaction } from '@iota/dapp-kit'; import { Transaction } from '@iota/iota-sdk/transactions'; import { useMutation } from '@tanstack/react-query'; export function useCreateSendAssetTransaction( objectId: string, - onSuccess: () => void, - onError: (error: unknown) => void, + onSuccess?: () => void, + onError?: (error: unknown) => void, ) { - const iotaClient = useIotaClient(); - const { mutateAsync: signAndExecuteTransaction } = useSignAndExecuteTransaction({ - execute: async ({ bytes, signature }) => - await iotaClient.executeTransactionBlock({ - transactionBlock: bytes, - signature, - options: { - showEffects: true, - showEvents: true, - showInput: true, - }, - }), - }); + const { mutateAsync: signAndExecuteTransaction } = useSignAndExecuteTransaction(); const mutation = useMutation({ mutationFn: async (to: string) => { if (!to) { @@ -34,6 +22,11 @@ export function useCreateSendAssetTransaction( return signAndExecuteTransaction({ transaction: tx, + options: { + showEffects: true, + showEvents: true, + showInput: true, + }, }); }, onSuccess, diff --git a/apps/wallet-dashboard/providers/AppProviders.tsx b/apps/wallet-dashboard/providers/AppProviders.tsx index f6712288174..27744398f86 100644 --- a/apps/wallet-dashboard/providers/AppProviders.tsx +++ b/apps/wallet-dashboard/providers/AppProviders.tsx @@ -9,6 +9,7 @@ import { IotaClientProvider, lightTheme, darkTheme, WalletProvider } from '@iota import { getAllNetworks, getDefaultNetwork } from '@iota/iota-sdk/client'; import { QueryClient, QueryClientProvider } from '@tanstack/react-query'; import { useState } from 'react'; +import { KioskClientProvider } from '@iota/core'; import { growthbook } from '@/lib/utils'; import { Popup } from '@/components/Popup'; import { ThemeProvider } from '@iota/core'; @@ -24,26 +25,28 @@ export function AppProviders({ children }: React.PropsWithChildren) { <GrowthBookProvider growthbook={growthbook}> <QueryClientProvider client={queryClient}> <IotaClientProvider networks={allNetworks} defaultNetwork={defaultNetwork}> - <WalletProvider - autoConnect={true} - theme={[ - { - variables: lightTheme, - }, - { - selector: '.dark', - variables: darkTheme, - }, - ]} - > - <ThemeProvider appId="iota-dashboard"> - <PopupProvider> - {children} - <Toaster /> - <Popup /> - </PopupProvider> - </ThemeProvider> - </WalletProvider> + <KioskClientProvider> + <WalletProvider + autoConnect={true} + theme={[ + { + variables: lightTheme, + }, + { + selector: '.dark', + variables: darkTheme, + }, + ]} + > + <ThemeProvider appId="iota-dashboard"> + <PopupProvider> + {children} + <Toaster /> + <Popup /> + </PopupProvider> + </ThemeProvider> + </WalletProvider> + </KioskClientProvider> </IotaClientProvider> </QueryClientProvider> </GrowthBookProvider> diff --git a/apps/wallet/src/ui/app/components/nft-display/index.tsx b/apps/wallet/src/ui/app/components/nft-display/index.tsx index dec1953fc94..779afa87e49 100644 --- a/apps/wallet/src/ui/app/components/nft-display/index.tsx +++ b/apps/wallet/src/ui/app/components/nft-display/index.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Loading, NftImage } from '_components'; -import { isKioskOwnerToken, useGetNFTMeta, useGetObject, useKioskClient } from '@iota/core'; +import { isKioskOwnerToken, useGetNFTDisplay, useGetObject, useKioskClient } from '@iota/core'; import { formatAddress } from '@iota/iota-sdk/utils'; import { cva } from 'class-variance-authority'; import type { VariantProps } from 'class-variance-authority'; @@ -42,7 +42,7 @@ export function NFTDisplayCard({ onIconClick, }: NFTDisplayCardProps) { const { data: objectData } = useGetObject(objectId); - const { data: nftMeta, isPending } = useGetNFTMeta(objectId); + const { data: nftMeta, isPending } = useGetNFTDisplay(objectId); const nftName = nftMeta?.name || formatAddress(objectId); const nftImageUrl = nftMeta?.imageUrl || ''; const video = useResolveVideo(objectData); diff --git a/apps/wallet/src/ui/app/helpers/formatAccountName.ts b/apps/wallet/src/ui/app/helpers/formatAccountName.ts index dff1c73ba80..c512a363841 100644 --- a/apps/wallet/src/ui/app/helpers/formatAccountName.ts +++ b/apps/wallet/src/ui/app/helpers/formatAccountName.ts @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import { formatAddress } from '@iota/iota-sdk/utils'; -import { truncateString } from './truncateString'; +import { truncateString } from '@iota/core'; export function formatAccountName( nickname: string | undefined | null, diff --git a/apps/wallet/src/ui/app/helpers/index.ts b/apps/wallet/src/ui/app/helpers/index.ts index aa14d5f7511..fde8b2c4f27 100644 --- a/apps/wallet/src/ui/app/helpers/index.ts +++ b/apps/wallet/src/ui/app/helpers/index.ts @@ -7,5 +7,4 @@ export { default as notEmpty } from './notEmptyCheck'; // export { getEventsSummary } from './getEventsSummary'; export { getAmount } from './getAmount'; export { checkStakingTxn } from './checkStakingTxn'; -export { truncateString } from './truncateString'; export { formatAccountName } from './formatAccountName'; diff --git a/apps/wallet/src/ui/app/hooks/index.ts b/apps/wallet/src/ui/app/hooks/index.ts index 33244c68262..f788f5dce44 100644 --- a/apps/wallet/src/ui/app/hooks/index.ts +++ b/apps/wallet/src/ui/app/hooks/index.ts @@ -6,15 +6,11 @@ export { default as useAppDispatch } from './useAppDispatch'; export { default as useAppSelector } from './useAppSelector'; export { default as useInitializedGuard } from './useInitializedGuard'; export { default as useFullscreenGuard } from './useFullscreenGuard'; -export { default as useMediaUrl } from './useMediaUrl'; export { default as useOnClickOutside } from './useOnClickOutside'; export { default as useOnKeyboardEvent } from './useOnKeyboardEvent'; -export { default as useFileExtensionType } from './useFileExtensionType'; -export { default as useNFTBasicData } from './useNFTBasicData'; export { useTransactionDryRun } from './useTransactionDryRun'; export { useGetTxnRecipientAddress } from './useGetTxnRecipientAddress'; export { useGetTransferAmount } from './useGetTransferAmount'; -export { useOwnedNFT } from './useOwnedNFT'; export { useCopyToClipboard } from './useCopyToClipboard'; export * from './useExplorerLink'; diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx index b6cd64ae3b6..c1a63ae97cd 100644 --- a/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx @@ -14,9 +14,8 @@ import { Button, ButtonSize, ButtonType, Dropdown, ListItem } from '@iota/apps-u import { Add, MoreHoriz, TriangleDown } from '@iota/ui-icons'; import { OutsideClickHandler } from '_components/OutsideClickHandler'; import { AccountGroupItem } from '_pages/accounts/manage/AccountGroupItem'; -import { Collapsible } from '_app/shared/collapse'; import { useFeature } from '@growthbook/growthbook-react'; -import { Feature } from '@iota/core'; +import { Feature, Collapsible } from '@iota/core'; import { useActiveAccount } from '_app/hooks/useActiveAccount'; const ACCOUNT_TYPE_TO_LABEL: Record<AccountType, string> = { diff --git a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/GasFees.tsx b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/GasFees.tsx index 2460016d0fb..31351c29c2e 100644 --- a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/GasFees.tsx +++ b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/GasFees.tsx @@ -3,9 +3,8 @@ // SPDX-License-Identifier: Apache-2.0 import { TitleSize, Badge, BadgeType, Title, Panel } from '@iota/apps-ui-kit'; -import { Collapsible } from '_src/ui/app/shared/collapse'; import { GasSummary } from '_src/ui/app/shared/transaction-summary/cards/GasSummary'; -import { type GasSummaryType } from '@iota/core'; +import { type GasSummaryType, Collapsible } from '@iota/core'; interface GasFeesProps { sender?: string; diff --git a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Command.tsx b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Command.tsx index 72075262870..f3106a47831 100644 --- a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Command.tsx +++ b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Command.tsx @@ -5,7 +5,7 @@ import { TypeTagSerializer, type TypeTag } from '@iota/iota-sdk/bcs'; import { type TransactionArgument, type Commands } from '@iota/iota-sdk/transactions/'; import { formatAddress, normalizeIotaAddress, toB64 } from '@iota/iota-sdk/utils'; -import { Collapsible } from '_src/ui/app/shared/collapse'; +import { Collapsible } from '@iota/core'; import { TitleSize } from '@iota/apps-ui-kit'; type TransactionType = ReturnType<(typeof Commands)[keyof typeof Commands]>; diff --git a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/index.tsx b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/index.tsx index 368174dc241..518a0929061 100644 --- a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/index.tsx +++ b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/index.tsx @@ -6,7 +6,7 @@ import { useTransactionData } from '_src/ui/app/hooks'; import { type Transaction } from '@iota/iota-sdk/transactions'; import { Command } from './Command'; import { Input } from './Input'; -import { Collapsible } from '_src/ui/app/shared/collapse'; +import { Collapsible } from '@iota/core'; import { ButtonSegment, ButtonSegmentType, diff --git a/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx b/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx index 1d3d10dc9fe..826dc5e2317 100644 --- a/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx @@ -12,8 +12,7 @@ import { PageTemplate, } from '_components'; import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; -import { Collapsible } from '_src/ui/app/shared/collapse'; -import { useGetKioskContents } from '@iota/core'; +import { useGetKioskContents, Collapsible } from '@iota/core'; import { formatAddress } from '@iota/iota-sdk/utils'; import { Link, useSearchParams, useNavigate } from 'react-router-dom'; import cl from 'clsx'; diff --git a/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx b/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx index 3879f492cea..062577a10b1 100644 --- a/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx @@ -3,61 +3,35 @@ // SPDX-License-Identifier: Apache-2.0 import { useActiveAddress } from '_app/hooks/useActiveAddress'; -import { Collapsible } from '_app/shared/collapse'; import { ExplorerLink, ExplorerLinkType, Loading, NFTDisplayCard, PageTemplate } from '_components'; -import { useNFTBasicData, useOwnedNFT } from '_hooks'; import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; -import { useIsAssetTransferable, useGetKioskContents, useGetNFTMeta } from '@iota/core'; +import { useNFTBasicData, useNftDetails, Collapsible } from '@iota/core'; import { formatAddress } from '@iota/iota-sdk/utils'; import cl from 'clsx'; import { Link, Navigate, useNavigate, useSearchParams } from 'react-router-dom'; import { Button, ButtonType, KeyValueInfo } from '@iota/apps-ui-kit'; -import { truncateString } from '_src/ui/app/helpers'; - -type NftFields = { - metadata?: { fields?: { attributes?: { fields?: { keys: string[]; values: string[] } } } }; -}; function NFTDetailsPage() { const navigate = useNavigate(); const [searchParams] = useSearchParams(); const nftId = searchParams.get('objectId'); const accountAddress = useActiveAddress(); - const { data: objectData, isPending: isNftLoading } = useOwnedNFT(nftId || '', accountAddress); - const { data: isAssetTransferable, isLoading: isCheckingAssetTransferability } = - useIsAssetTransferable(objectData); - const { nftFields, fileExtensionType, filePath } = useNFTBasicData(objectData); - const address = useActiveAddress(); - const { data } = useGetKioskContents(address); - - const isContainedInKiosk = data?.lookup.get(nftId!); - const kioskItem = data?.list.find((k) => k.data?.objectId === nftId); + const { + nftDisplayData, + isLoading, + ownerAddress, + objectData, + metaKeys, + metaValues, + formatMetaValue, + isContainedInKiosk, + kioskItem, + isAssetTransferable, + } = useNftDetails(nftId || '', accountAddress); + const { fileExtensionType, filePath } = useNFTBasicData(objectData); - // Extract either the attributes, or use the top-level NFT fields: - const metaFields = - (nftFields as NftFields)?.metadata?.fields?.attributes?.fields || - Object.entries(nftFields ?? {}) - .filter(([key]) => key !== 'id') - .reduce( - (acc, [key, value]) => { - acc.keys.push(key); - acc.values.push(value as string); - return acc; - }, - { keys: [] as string[], values: [] as string[] }, - ); - const metaKeys: string[] = metaFields ? metaFields.keys : []; - const metaValues = metaFields ? metaFields.values : []; - const { data: nftDisplayData, isPending: isPendingDisplay } = useGetNFTMeta(nftId || ''); - const ownerAddress = - (objectData?.owner && - typeof objectData?.owner === 'object' && - 'AddressOwner' in objectData.owner && - objectData.owner.AddressOwner) || - ''; const isGuardLoading = useUnlockedGuard(); - const isPending = - isNftLoading || isPendingDisplay || isGuardLoading || isCheckingAssetTransferability; + const isPending = isLoading || isGuardLoading; function handleMoreAboutKiosk() { window.open('https://docs.iota.org/references/ts-sdk/kiosk/', '_blank'); @@ -72,28 +46,6 @@ function NFTDetailsPage() { navigate(`/nft-transfer/${nftId}`); } - function formatMetaValue(value: string | object) { - if (typeof value === 'object') { - return { - value: JSON.stringify(value), - valueLink: undefined, - }; - } else { - if (value.includes('http')) { - return { - value: value.startsWith('http') - ? truncateString(value, 20, 8) - : formatAddress(value), - valueLink: value, - }; - } - return { - value: value, - valueLink: undefined, - }; - } - } - return ( <PageTemplate title="Visual Asset" @@ -136,31 +88,29 @@ function NFTDetailsPage() { </span> ) : null} </div> - {nftDisplayData?.projectUrl || - (nftDisplayData?.creator && ( - <div className="flex flex-col gap-xs"> - {nftDisplayData?.projectUrl && ( - <KeyValueInfo - keyText="Website" - value={ - <Link - to={nftDisplayData?.projectUrl} - > - {nftDisplayData?.projectUrl} - </Link> - } - fullwidth - /> - )} - {nftDisplayData?.creator && ( - <KeyValueInfo - keyText="Creator" - value={nftDisplayData?.creator ?? '-'} - fullwidth - /> - )} - </div> - ))} + {(nftDisplayData?.projectUrl || + nftDisplayData?.creator) && ( + <div className="flex flex-col gap-xs"> + {nftDisplayData?.projectUrl && ( + <KeyValueInfo + keyText="Website" + value={ + <Link to={nftDisplayData?.projectUrl}> + {nftDisplayData?.projectUrl} + </Link> + } + fullwidth + /> + )} + {nftDisplayData?.creator && ( + <KeyValueInfo + keyText="Creator" + value={nftDisplayData?.creator ?? '-'} + fullwidth + /> + )} + </div> + )} </div> <div className="flex flex-col gap-md"> <Collapsible defaultOpen title="Details"> diff --git a/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx b/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx index bae076ae8a2..8d4c920ec38 100644 --- a/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx @@ -4,11 +4,10 @@ import { useActiveAddress } from '_app/hooks/useActiveAddress'; import { Loading, NFTDisplayCard, Overlay } from '_components'; -import { useOwnedNFT } from '_hooks'; import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; import { Navigate, useNavigate, useParams } from 'react-router-dom'; import { TransferNFTForm } from './TransferNFTForm'; -import { useIsAssetTransferable } from '@iota/core'; +import { useOwnedNFT, useIsAssetTransferable } from '@iota/core'; function NftTransferPage() { const { nftId } = useParams(); diff --git a/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx b/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx index 4651fc56e7a..92ac60bc862 100644 --- a/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx +++ b/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx @@ -10,7 +10,7 @@ import { useHiddenAssets } from '../assets/HiddenAssetsProvider'; import { getKioskIdFromOwnerCap, isKioskOwnerToken, - useGetNFTMeta, + useGetNFTDisplay, useGetObject, useKioskClient, } from '@iota/core'; @@ -44,7 +44,7 @@ export default function HiddenAsset(item: HiddenAssetProps) { const navigate = useNavigate(); const { objectId, type } = item.data!; const { data: objectData } = useGetObject(objectId); - const { data: nftMeta } = useGetNFTMeta(objectId); + const { data: nftMeta } = useGetNFTDisplay(objectId); const nftName = nftMeta?.name || formatAddress(objectId); const nftImageUrl = nftMeta?.imageUrl || ''; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/TokenList.tsx b/apps/wallet/src/ui/app/pages/home/tokens/TokenList.tsx index 7952e1c8db2..d2b1a8dba3b 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/TokenList.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/TokenList.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { Collapsible } from '_src/ui/app/shared/collapse'; +import { Collapsible } from '@iota/core'; import { type ReactNode } from 'react'; type TokenListProps = { diff --git a/apps/wallet/src/ui/app/shared/transaction-summary/cards/ObjectChanges.tsx b/apps/wallet/src/ui/app/shared/transaction-summary/cards/ObjectChanges.tsx index 86febdbe570..0f1e92eddfc 100644 --- a/apps/wallet/src/ui/app/shared/transaction-summary/cards/ObjectChanges.tsx +++ b/apps/wallet/src/ui/app/shared/transaction-summary/cards/ObjectChanges.tsx @@ -8,12 +8,12 @@ import { type ObjectChangeSummary, type IotaObjectChangeTypes, type IotaObjectChangeWithDisplay, + Collapsible, } from '@iota/core'; import { formatAddress } from '@iota/iota-sdk/utils'; import cx from 'clsx'; import { ExpandableList } from '../../ExpandableList'; import { ObjectChangeDisplay } from './objectSummary/ObjectChangeDisplay'; -import { Collapsible } from '../../collapse'; import { Badge, BadgeType,