From 1758aac21eb8c47b8c1e0da596273da8d406bad6 Mon Sep 17 00:00:00 2001 From: Lucas Werey Date: Tue, 10 Sep 2024 10:12:25 +0200 Subject: [PATCH] :sparkles:feat(ui): add simplehash calls for ordis --- .changeset/dry-parents-breathe.md | 7 + .../Ordinals/components/Error.tsx | 21 +++ .../Ordinals/components/Icons.tsx | 103 ++++++++++-- .../components/Inscriptions/helpers.ts | 43 +++++ .../components/Inscriptions/index.tsx | 76 +++++---- .../Inscriptions/useInscriptionsModel.tsx | 26 +-- .../Ordinals/components/Loader.tsx | 18 +++ .../Ordinals/components/RareSats/Item.tsx | 11 +- .../components/RareSats/RowLayout.tsx | 5 - .../components/RareSats/TableHeader.tsx | 2 - .../Ordinals/components/RareSats/helpers.ts | 153 +++++++----------- .../Ordinals/components/RareSats/index.tsx | 57 ++++--- .../components/RareSats/useRareSatsModel.ts | 20 +-- .../Ordinals/screens/Account/index.tsx | 13 +- .../components/Collection/ShowMore.tsx | 5 +- .../Collection/TableRow/IconContainer.tsx | 29 ---- .../TableRow/IconContainer/ToolTip.tsx | 18 +++ .../TableRow/IconContainer/index.tsx | 40 +++++ .../Collectibles/hooks/useFetchOrdinals.ts | 18 +++ .../Collectibles/types/Collectibles.ts | 8 +- .../features/Collectibles/types/Ordinals.ts | 21 ++- .../features/Collectibles/types/RareSats.ts | 8 - .../Collectibles/types/enum/Collectibles.ts | 8 + .../Collectibles/types/enum/DetailDrawer.ts | 1 + .../src/renderer/screens/account/index.tsx | 3 +- .../static/i18n/en/app.json | 33 ++++ .../src/hooks/helpers/ordinals.ts | 90 +++++++++++ libs/live-nft-react/src/hooks/types.ts | 49 ++++++ .../src/hooks/useFetchOrdinals.ts | 49 ++++++ libs/live-nft-react/src/index.ts | 2 + libs/live-nft-react/src/queryKeys.ts | 1 + libs/live-nft/src/api/types.ts | 62 +++++++ 32 files changed, 750 insertions(+), 250 deletions(-) create mode 100644 .changeset/dry-parents-breathe.md create mode 100644 apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Error.tsx create mode 100644 apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/helpers.ts create mode 100644 apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Loader.tsx delete mode 100644 apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer.tsx create mode 100644 apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/ToolTip.tsx create mode 100644 apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/index.tsx create mode 100644 apps/ledger-live-desktop/src/newArch/features/Collectibles/hooks/useFetchOrdinals.ts create mode 100644 libs/live-nft-react/src/hooks/helpers/ordinals.ts create mode 100644 libs/live-nft-react/src/hooks/useFetchOrdinals.ts diff --git a/.changeset/dry-parents-breathe.md b/.changeset/dry-parents-breathe.md new file mode 100644 index 000000000000..42c1cc409974 --- /dev/null +++ b/.changeset/dry-parents-breathe.md @@ -0,0 +1,7 @@ +--- +"ledger-live-desktop": patch +"@ledgerhq/live-nft-react": patch +"@ledgerhq/live-nft": patch +--- + +Plug the front with simplehash api for the rare sats table and inscriptions table diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Error.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Error.tsx new file mode 100644 index 000000000000..ad5a7f3d72b5 --- /dev/null +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Error.tsx @@ -0,0 +1,21 @@ +import React from "react"; +import { Flex, Icons, Text } from "@ledgerhq/react-ui"; +import { useTranslation } from "react-i18next"; + +type Props = { + hasError: boolean; +}; + +const Error: React.FC = ({ hasError }) => { + const { t } = useTranslation(); + if (!hasError) return null; + + return ( + + + {t("crash.title")} + + ); +}; + +export default Error; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Icons.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Icons.tsx index feb7ec041fe5..745bf825fc87 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Icons.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Icons.tsx @@ -3,71 +3,144 @@ import { Icons } from "@ledgerhq/react-ui"; import { IconProps } from "../../types/Collection"; export const mappingKeysWithIconAndName = { - alpha: { icon: (props: IconProps) => , name: "Alpha" }, + alpha: { + icon: (props: IconProps) => , + name: "Alpha", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.alpha", + }, black_epic: { icon: (props: IconProps) => , name: "Black Epic", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.black_epic", }, black_legendary: { icon: (props: IconProps) => , name: "Black Legendary", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.black_legendary", }, black_mythic: { icon: (props: IconProps) => , name: "Black Mythic", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.black_mythic", }, black_rare: { icon: (props: IconProps) => , name: "Black Rare", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.black_rare", }, black_uncommon: { icon: (props: IconProps) => , name: "Black Uncommon", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.black_uncommon", + }, + block_9: { + icon: (props: IconProps) => , + name: "Block 9", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.block_9", }, - block_9: { icon: (props: IconProps) => , name: "Block 9" }, block_9_450x: { icon: (props: IconProps) => , name: "Block 9 450x", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.block_9_450x", + }, + block_78: { + icon: (props: IconProps) => , + name: "Block 78", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.block_78", }, - block_78: { icon: (props: IconProps) => , name: "Block 78" }, block_286: { icon: (props: IconProps) => , name: "Block 286", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.block_286", }, block_666: { icon: (props: IconProps) => , name: "Block 666", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.block_666", + }, + common: { + icon: (props: IconProps) => , + name: "Common", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.common", + }, + epic: { + icon: (props: IconProps) => , + name: "Epic", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.epic", }, - common: { icon: (props: IconProps) => , name: "Common" }, - epic: { icon: (props: IconProps) => , name: "Epic" }, first_tx: { icon: (props: IconProps) => , name: "First Transaction", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.first_tx", + }, + hitman: { + icon: (props: IconProps) => , + name: "Hitman", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.hitman", + }, + jpeg: { + icon: (props: IconProps) => , + name: "JPEG", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.jpeg", + }, + legacy: { + icon: (props: IconProps) => , + name: "Legacy", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.legacy", }, - hitman: { icon: (props: IconProps) => , name: "Hitman" }, - jpeg: { icon: (props: IconProps) => , name: "JPEG" }, - legacy: { icon: (props: IconProps) => , name: "Legacy" }, legendary: { icon: (props: IconProps) => , name: "Legendary", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.legendary", + }, + mythic: { + icon: (props: IconProps) => , + name: "Mythic", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.mythic", + }, + nakamoto: { + icon: (props: IconProps) => , + name: "Nakamoto", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.nakamoto", + }, + omega: { + icon: (props: IconProps) => , + name: "Omega", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.omega", }, - mythic: { icon: (props: IconProps) => , name: "Mythic" }, - nakamoto: { icon: (props: IconProps) => , name: "Nakamoto" }, - omega: { icon: (props: IconProps) => , name: "Omega" }, paliblock: { icon: (props: IconProps) => , name: "PaliBlock Palindrome", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.paliblock", }, palindrome: { icon: (props: IconProps) => , name: "Palindrome", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.palindrome", }, palinception: { icon: (props: IconProps) => , name: "Palinception", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.palinception", + }, + pizza: { + icon: (props: IconProps) => , + name: "Pizza", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.pizza", + }, + rare: { + icon: (props: IconProps) => , + name: "Rare", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.rare", + }, + uncommon: { + icon: (props: IconProps) => , + name: "Uncommon", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.uncommon", + }, + vintage: { + icon: (props: IconProps) => , + name: "Vintage", + descriptionTranslationKey: "ordinals.rareSats.rareSat.description.vintage", }, - pizza: { icon: (props: IconProps) => , name: "Pizza" }, - rare: { icon: (props: IconProps) => , name: "Rare" }, - uncommon: { icon: (props: IconProps) => , name: "Uncommon" }, - vintage: { icon: (props: IconProps) => , name: "Vintage" }, }; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/helpers.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/helpers.ts new file mode 100644 index 000000000000..0a49f3a29d17 --- /dev/null +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/helpers.ts @@ -0,0 +1,43 @@ +import { SimpleHashNft } from "@ledgerhq/live-nft/api/types"; +import { IconProps } from "LLD/features/Collectibles/types/Collection"; +import { mappingKeysWithIconAndName } from "../Icons"; + +function matchCorrespondingIcon( + rareSats: SimpleHashNft[], +): Array JSX.Element> }> { + return rareSats.map(rareSat => { + const iconKeys: string[] = []; + const rarity = rareSat.extra_metadata?.ordinal_details?.sat_rarity?.toLowerCase(); + + if (rarity && rarity !== "common") { + iconKeys.push(rarity.replace(" ", "_")); + } + + const icons = iconKeys + .map( + iconKey => + mappingKeysWithIconAndName[iconKey as keyof typeof mappingKeysWithIconAndName]?.icon, + ) + .filter(Boolean) as Array<({ size, color, style }: IconProps) => JSX.Element>; + + return { ...rareSat, icons }; + }); +} + +export function getInscriptionsData(inscriptions: SimpleHashNft[]) { + const inscriptionsWithIcons = matchCorrespondingIcon(inscriptions); + return inscriptionsWithIcons.map(item => ({ + tokenName: item.name || item.contract.name || "", + collectionName: item.collection.name, + tokenIcons: item.icons, + rareSatName: item.extra_metadata?.ordinal_details?.sat_name, + media: { + uri: item.image_url || item.previews?.image_small_url, + isLoading: false, + useFallback: true, + contentType: item.extra_metadata?.ordinal_details?.content_type, + mediaType: "image", + }, + onClick: () => {}, + })); +} diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/index.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/index.tsx index 4691340a2152..49795be99ea7 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/index.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/index.tsx @@ -1,5 +1,4 @@ import React from "react"; -import { Account } from "@ledgerhq/types-live"; import { Box, Flex } from "@ledgerhq/react-ui"; import { useInscriptionsModel } from "./useInscriptionsModel"; import TableContainer from "~/renderer/components/TableContainer"; @@ -8,15 +7,20 @@ import { OrdinalsRowProps, TableHeaderTitleKey } from "LLD/features/Collectibles import TableRow from "LLD/features/Collectibles/components/Collection/TableRow"; import ShowMore from "LLD/features/Collectibles/components/Collection/ShowMore"; import { MediaProps } from "LLD/features/Collectibles/types/Media"; +import { SimpleHashNft } from "@ledgerhq/live-nft/api/types"; +import { Status } from "LLD/features/Collectibles/types/Collectibles"; +import { TanStackStatus } from "LLD/features/Collectibles/types/enum/Collectibles"; +import Loader from "../Loader"; +import Error from "../Error"; -type ViewProps = ReturnType; +type ViewProps = ReturnType & { status: Status }; type Props = { - account: Account; + inscriptions: SimpleHashNft[]; + status: Status; }; export type InscriptionsItemProps = { - isLoading: boolean; tokenName: string; collectionName: string; tokenIcons: OrdinalsRowProps["tokenIcons"]; @@ -24,57 +28,65 @@ export type InscriptionsItemProps = { onClick: () => void; }; -const Item: React.FC = ({ +type ItemProps = { + isLoading: boolean; +} & InscriptionsItemProps; + +const Item: React.FC = ({ isLoading, tokenName, collectionName, tokenIcons, media, onClick, -}) => { - return ( - - ); -}; +}) => ( + +); + +const View: React.FC = ({ displayShowMore, status, inscriptions, onShowMore }) => { + const isLoading = status === TanStackStatus.Pending; + const hasError = status === TanStackStatus.Error; + const hasInscriptions = inscriptions.length > 0 && !hasError; + const nothingToShow = !hasInscriptions && !isLoading && !hasError; -function View({ displayedObjects, displayShowMore, onShowMore }: ViewProps) { return ( - + - {/** titlekey doesn't need to be translated so we keep it hard coded */} - {displayedObjects ? ( - displayedObjects.map((item, index) => ( + + + {hasInscriptions && + inscriptions.map((item, index) => ( - )) - ) : ( - - {"NOTHING TO SHOW"} + ))} + {nothingToShow && ( + + {"NOTHING TO SHOW WAITING FOR DESIGN"} )} - {displayShowMore && } + {displayShowMore && !hasError && } ); -} - -const Inscriptions: React.FC = ({ account }: Props) => { - return ; }; +const Inscriptions: React.FC = ({ inscriptions, status }) => ( + +); + export default Inscriptions; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/useInscriptionsModel.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/useInscriptionsModel.tsx index 05900b0790e9..9fb172a73be1 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/useInscriptionsModel.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Inscriptions/useInscriptionsModel.tsx @@ -1,32 +1,36 @@ import { useEffect, useMemo, useState } from "react"; -import { Account } from "@ledgerhq/types-live"; import { InscriptionsItemProps } from "./index"; -import { mockedItems as InscriptionsMocked } from "LLD/features/Collectibles/__integration__/mockedInscriptions"; +import { SimpleHashNft } from "@ledgerhq/live-nft/api/types"; +import { getInscriptionsData } from "./helpers"; + type Props = { - account: Account; + inscriptions: SimpleHashNft[]; }; -export const useInscriptionsModel = ({ account }: Props) => { +export const useInscriptionsModel = ({ inscriptions }: Props) => { const [displayShowMore, setDisplayShowMore] = useState(false); const [displayedObjects, setDisplayedObjects] = useState([]); - const mockedItems: InscriptionsItemProps[] = useMemo(() => [...InscriptionsMocked], []); + const items: InscriptionsItemProps[] = useMemo( + () => getInscriptionsData(inscriptions), + [inscriptions], + ); useEffect(() => { - if (mockedItems.length > 3) setDisplayShowMore(true); - setDisplayedObjects(mockedItems.slice(0, 3)); - }, [mockedItems]); + if (items.length > 3) setDisplayShowMore(true); + setDisplayedObjects(items.slice(0, 3)); + }, [items]); const onShowMore = () => { setDisplayedObjects(prevDisplayedObjects => { const newDisplayedObjects = [ ...prevDisplayedObjects, - ...mockedItems.slice(prevDisplayedObjects.length, prevDisplayedObjects.length + 3), + ...items.slice(prevDisplayedObjects.length, prevDisplayedObjects.length + 3), ]; - if (newDisplayedObjects.length === mockedItems.length) setDisplayShowMore(false); + if (newDisplayedObjects.length === items.length) setDisplayShowMore(false); return newDisplayedObjects; }); }; - return { account, displayedObjects, displayShowMore, onShowMore }; + return { inscriptions: displayedObjects, displayShowMore, onShowMore }; }; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Loader.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Loader.tsx new file mode 100644 index 000000000000..b49a1336d6f8 --- /dev/null +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/Loader.tsx @@ -0,0 +1,18 @@ +import { Flex, InfiniteLoader } from "@ledgerhq/react-ui"; +import React from "react"; + +type Props = { + isLoading: boolean; +}; + +const Loader: React.FC = ({ isLoading }) => { + if (!isLoading) return null; + + return ( + + + + ); +}; + +export default Loader; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/Item.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/Item.tsx index 6845c31536f4..c2124150008f 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/Item.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/Item.tsx @@ -2,21 +2,22 @@ import React from "react"; import RowLayout from "LLD/features/Collectibles/Ordinals/components/RareSats/RowLayout"; import IconContainer from "LLD/features/Collectibles/components/Collection/TableRow/IconContainer"; import TokenTitle from "LLD/features/Collectibles/components/Collection/TableRow/TokenTitle"; -import { RareSat } from "LLD/features/Collectibles/types/Ordinals"; import { Text, Flex } from "@ledgerhq/react-ui"; +import { RareSat } from "LLD/features/Collectibles/types/Ordinals"; const Item = ({ icons, - name, + names, + displayed_names, year, count, utxo_size, isMultipleRow, -}: RareSat & { isMultipleRow: boolean }) => { +}: RareSat) => { const firstColumn = ( - {icons && } - + {icons && } + ); const secondColumn = ( diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/RowLayout.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/RowLayout.tsx index 31449ce9ec39..a17d3342df9f 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/RowLayout.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/RowLayout.tsx @@ -5,7 +5,6 @@ import styled from "styled-components"; type Props = { firstColumnElement: JSX.Element; secondColumnElement: JSX.Element; - thirdColumnElement?: JSX.Element; bgColor?: string; isMultipleRow?: boolean; }; @@ -23,7 +22,6 @@ const Container = styled(Flex)` const RowLayout: React.FC = ({ firstColumnElement, secondColumnElement, - thirdColumnElement, bgColor, isMultipleRow, }) => { @@ -42,9 +40,6 @@ const RowLayout: React.FC = ({ {secondColumnElement} - - {thirdColumnElement} - ); diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/TableHeader.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/TableHeader.tsx index 8ce914bf20f9..e38ccead5355 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/TableHeader.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/TableHeader.tsx @@ -14,13 +14,11 @@ export const TableHeader = () => { const firstColumn = Column("ordinals.rareSats.table.type"); const secondColumn = Column("ordinals.rareSats.table.year"); - const thirdColumn = Column("ordinals.rareSats.table.utxo"); return ( ); diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/helpers.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/helpers.ts index 23e7080c2dea..9a7423bbe53b 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/helpers.ts +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/helpers.ts @@ -1,107 +1,62 @@ import { mappingKeysWithIconAndName } from "../Icons"; -import { MappingKeys } from "LLD/features/Collectibles/types/Ordinals"; -import { IconProps } from "LLD/features/Collectibles/types/Collection"; -import { RareSat } from "LLD/features/Collectibles/types/Ordinals"; +import { SimpleHashNft } from "@ledgerhq/live-nft/api/types"; import { - Satributes, - Subrange, - SatRange, - MockedRareSat, - Sat, - Icons, -} from "LLD/features/Collectibles/types/RareSats"; - -export const processSatType = ( - type: string, - satributes: Satributes, - icons: Icons, - displayNames: string[], - totalCount: number, -) => { - const attribute = satributes[type as MappingKeys]; - if (attribute && attribute.count) { - displayNames.push(type); - if (mappingKeysWithIconAndName[type as MappingKeys]) { - icons[type] = mappingKeysWithIconAndName[type as MappingKeys].icon; + SimpleHashNftWithIcons, + RareSat, + MappingKeys, +} from "LLD/features/Collectibles/types/Ordinals"; + +export function matchCorrespondingIcon(rareSats: SimpleHashNft[]): SimpleHashNftWithIcons[] { + return rareSats.map(rareSat => { + const iconKeys: string[] = []; + if (rareSat.name) { + iconKeys.push(rareSat.name.toLowerCase().replace(" ", "_")); } - totalCount = attribute.count; - } - return { displayNames, totalCount }; -}; - -export const processSatTypes = (satTypes: string[], satributes: Satributes) => { - let displayNames: string[] = []; - let totalCount = 0; - const icons: { [key: string]: ({ size, color, style }: IconProps) => JSX.Element } = {}; - - satTypes.forEach(type => { - const result = processSatType(type, satributes, icons, displayNames, totalCount); - displayNames = result.displayNames; - totalCount = result.totalCount; - }); - - return { displayNames, totalCount, icons }; -}; -export const processSubrange = ( - subrange: Subrange, - satributes: Satributes, - year: string, - value: number, -) => { - const { sat_types } = subrange; - const { displayNames, totalCount, icons } = processSatTypes(sat_types, satributes); - - const name = displayNames - .map(dn => mappingKeysWithIconAndName[dn.toLowerCase().replace(" ", "_") as MappingKeys]?.name) - .filter(Boolean) - .join(" / "); + if (rareSat.extra_metadata?.utxo_details?.satributes) { + iconKeys.push(...Object.keys(rareSat.extra_metadata.utxo_details.satributes)); + } - return { - count: totalCount.toString() + (totalCount === 1 ? " sat" : " sats"), - display_name: displayNames.join(" / "), - year, - utxo_size: value.toString(), - icons, - name, - }; -}; + const icons = iconKeys + .map( + iconKey => + mappingKeysWithIconAndName[iconKey as keyof typeof mappingKeysWithIconAndName]?.icon, + ) + .filter(Boolean) as RareSat["icons"]; -export const processSatRanges = (satRanges: SatRange[], satributes: Satributes) => { - return satRanges.flatMap(range => { - const { year, value, subranges } = range; - return subranges.flatMap(subrange => processSubrange(subrange, satributes, year, value)); + return { ...rareSat, icons }; }); -}; - -export const processRareSat = (sat: Sat) => { - const { extra_metadata } = sat; - const satributes = extra_metadata.utxo_details.satributes as Satributes; - const satRanges = extra_metadata.utxo_details.sat_ranges; - return processSatRanges(satRanges, satributes); -}; - -export const processRareSats = (rareSats: MockedRareSat[]) => { - return rareSats.flatMap(item => item.nfts.flatMap(processRareSat)); -}; - -export const groupRareSats = (processedRareSats: RareSat[]) => { - return processedRareSats.reduce( - (acc, sat) => { - if (!acc[sat.utxo_size]) { - acc[sat.utxo_size] = []; - } - acc[sat.utxo_size].push(sat); - return acc; - }, - {} as Record, - ); -}; +} + +export function createRareSatObject( + rareSats: Record, +): Record { + const result: Record = {}; + + for (const [key, value] of Object.entries(rareSats)) { + result[key] = value.map(rareSat => { + const { icons, extra_metadata } = rareSat; + const year = extra_metadata?.utxo_details?.sat_ranges?.[0]?.year || ""; + const displayed_names = + Object.values(extra_metadata?.utxo_details?.satributes || {}) + .map(attr => attr.display_name) + .join(" / ") || ""; + const names = rareSat.extra_metadata?.utxo_details?.satributes + ? (Object.keys(rareSat.extra_metadata.utxo_details.satributes) as MappingKeys[]) + : []; + const count = extra_metadata?.utxo_details?.value || 0; + const isMultipleRow = value.length > 1; + + return { + year, + displayed_names, + names, + count: `${count} ${count > 1 ? "sats" : "sat"}`, + isMultipleRow, + icons, + }; + }); + } -export const finalizeRareSats = (groupedRareSats: Record) => { - return Object.entries(groupedRareSats).map(([utxo_size, sats]) => ({ - utxo_size, - sats, - isMultipleRow: sats.length > 1, - })); -}; + return result; +} diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/index.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/index.tsx index 838d4a0c957f..40f1bdd35932 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/index.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/index.tsx @@ -6,38 +6,57 @@ import Item from "./Item"; import { TableHeaderTitleKey } from "LLD/features/Collectibles/types/Collection"; import { Box, Flex } from "@ledgerhq/react-ui"; import { TableHeader as TableHeaderContainer } from "./TableHeader"; +import { SimpleHashNft } from "@ledgerhq/live-nft/api/types"; +import { Status } from "LLD/features/Collectibles/types/Collectibles"; +import { TanStackStatus } from "LLD/features/Collectibles/types/enum/Collectibles"; +import Loader from "../Loader"; +import Error from "../Error"; -type ViewProps = ReturnType; +type ViewProps = ReturnType & { + status: Status; +}; + +type Props = { + rareSats: SimpleHashNft[]; + status: Status; +}; + +function View({ rareSats, status }: ViewProps) { + const isLoading = status === TanStackStatus.Pending; + const hasError = status === TanStackStatus.Error; + const isLoaded = status === TanStackStatus.Success; + const hasRareSats = Object.values(rareSats).length > 0; + const dataReady = isLoaded && hasRareSats; -function View({ rareSats }: ViewProps) { return ( - + + + {dataReady && } - {rareSats - ? rareSats.map(rareSatGroup => ( - - {rareSatGroup.sats.map(rareSat => ( - - ))} - - )) - : null} - {/** wait for design */} + {dataReady && + Object.entries(rareSats).map(([key, rareSatGroup]) => ( + + {rareSatGroup.map((rareSat, index) => ( + + ))} + + ))} + {isLoaded && !hasRareSats && ( + + {"NOTHING TO SHOW WAITING FOR DESIGN"} + + )} ); } -const RareSats = () => { - return ; +const RareSats = ({ rareSats, status }: Props) => { + return ; }; export default RareSats; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/useRareSatsModel.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/useRareSatsModel.ts index 931c590a56d8..a59b1eafc054 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/useRareSatsModel.ts +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/components/RareSats/useRareSatsModel.ts @@ -1,12 +1,14 @@ -import { mockedRareSats } from "LLD/features/Collectibles/__integration__/mockedRareSats"; -import { processRareSats, groupRareSats, finalizeRareSats } from "./helpers"; +import { matchCorrespondingIcon, createRareSatObject } from "./helpers"; +import { SimpleHashNft } from "@ledgerhq/live-nft/api/types"; +import { regroupRareSatsByContractAddress } from "@ledgerhq/live-nft-react"; -type RareSatsProps = {}; - -export const useRareSatsModel = (_props: RareSatsProps) => { - const processedRareSats = processRareSats(mockedRareSats); - const groupedRareSats = groupRareSats(processedRareSats); - const finalRareSats = finalizeRareSats(groupedRareSats); +type Props = { + rareSats: SimpleHashNft[]; +}; - return { rareSats: finalRareSats }; +export const useRareSatsModel = ({ rareSats }: Props) => { + const matchedRareSats = matchCorrespondingIcon(rareSats); + const regroupedRareSats = regroupRareSatsByContractAddress(matchedRareSats); + const rareSatsCreated = createRareSatObject(regroupedRareSats); + return { rareSats: rareSatsCreated }; }; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/screens/Account/index.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/screens/Account/index.tsx index 30aed15cccc5..3dcd7da4c148 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/screens/Account/index.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/Ordinals/screens/Account/index.tsx @@ -1,18 +1,23 @@ import React from "react"; -import { Account } from "@ledgerhq/types-live"; import Inscriptions from "../../components/Inscriptions"; import RareSats from "../../components/RareSats"; import { Flex } from "@ledgerhq/react-ui"; +import useFetchOrdinals from "LLD/features/Collectibles/hooks/useFetchOrdinals"; +import { BitcoinAccount } from "@ledgerhq/coin-bitcoin/lib/types"; type Props = { - account: Account; + account: BitcoinAccount; }; const OrdinalsAccount: React.FC = ({ account }) => { + const { rareSats, inscriptions, status } = useFetchOrdinals({ + account, + }); + return ( - - + + ); }; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/ShowMore.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/ShowMore.tsx index 1e7e844d7da3..52cf18fadd3d 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/ShowMore.tsx +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/ShowMore.tsx @@ -7,15 +7,16 @@ import { useTranslation } from "react-i18next"; type Props = { onShowMore: () => void; + isInscriptions?: boolean; }; -const ShowMore: React.FC = ({ onShowMore }) => { +const ShowMore: React.FC = ({ isInscriptions = false, onShowMore }) => { const { t } = useTranslation(); return ( - {t("NFT.collections.seeMore")} + {isInscriptions ? t("ordinals.inscriptions.seeMore") : t("NFT.collections.seeMore")} diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer.tsx deleted file mode 100644 index 79bd9ee94c49..000000000000 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer.tsx +++ /dev/null @@ -1,29 +0,0 @@ -import React from "react"; -import { Flex } from "@ledgerhq/react-ui"; -import { IconProps } from "LLD/features/Collectibles/types/Collection"; - -type Props = { - icons: (({ size, color, style }: IconProps) => JSX.Element)[]; -}; - -const IconContainer: React.FC = ({ icons }) => { - return ( - - {icons.map((Icon, index) => ( - - ))} - - ); -}; - -export default IconContainer; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/ToolTip.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/ToolTip.tsx new file mode 100644 index 000000000000..077069133b16 --- /dev/null +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/ToolTip.tsx @@ -0,0 +1,18 @@ +import React from "react"; +import Tooltip from "~/renderer/components/Tooltip"; +import { mappingKeysWithIconAndName } from "LLD/features/Collectibles/Ordinals/components/Icons"; +import { useTranslation } from "react-i18next"; + +type ToolTipProps = { + content: keyof typeof mappingKeysWithIconAndName; + children: React.ReactNode; +}; + +const RareSatToolTip: React.FC = ({ content, children }) => { + const { t } = useTranslation(); + + const tooltipContent = mappingKeysWithIconAndName[content]?.descriptionTranslationKey || content; + return {children}; +}; + +export default RareSatToolTip; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/index.tsx b/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/index.tsx new file mode 100644 index 000000000000..e6f15e06809b --- /dev/null +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/components/Collection/TableRow/IconContainer/index.tsx @@ -0,0 +1,40 @@ +import React from "react"; +import { Flex } from "@ledgerhq/react-ui"; +import { IconProps } from "LLD/features/Collectibles/types/Collection"; +import RareSatToolTip from "./ToolTip"; +import { mappingKeysWithIconAndName } from "LLD/features/Collectibles/Ordinals/components/Icons"; + +type IconNameKeys = keyof typeof mappingKeysWithIconAndName; + +type Props = { + icons: (({ size, color, style }: IconProps) => JSX.Element)[]; + iconNames?: IconNameKeys[]; +}; + +const IconContainer: React.FC = ({ icons, iconNames }) => { + return ( + + {icons.map((Icon, index) => + iconNames ? ( + + + + ) : ( + + ), + )} + + ); +}; + +export default IconContainer; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/hooks/useFetchOrdinals.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/hooks/useFetchOrdinals.ts new file mode 100644 index 000000000000..d203e233e420 --- /dev/null +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/hooks/useFetchOrdinals.ts @@ -0,0 +1,18 @@ +import { useFetchOrdinals as fetchOrdinalsFromSimpleHash } from "@ledgerhq/live-nft-react"; +import { BitcoinAccount } from "@ledgerhq/coin-bitcoin/lib/types"; + +type Props = { + account: BitcoinAccount; +}; + +const useFetchOrdinals = ({ account }: Props) => { + const utxosAddresses = account.bitcoinResources?.utxos?.map(utxo => utxo.address).join(",") || ""; + const { rareSats, inscriptions, status } = fetchOrdinalsFromSimpleHash({ + addresses: utxosAddresses, + threshold: 0, + }); + + return { rareSats, inscriptions, status }; +}; + +export default useFetchOrdinals; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Collectibles.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Collectibles.ts index aa557daae387..140e179b2da2 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Collectibles.ts +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Collectibles.ts @@ -1,9 +1,15 @@ import { ProtoNFT, NFT, Account } from "@ledgerhq/types-live"; import { CollectibleTypeEnum } from "./enum/Collectibles"; -export type CollectibleType = CollectibleTypeEnum.NFT | CollectibleTypeEnum.Ordinal; +export type CollectibleType = + | CollectibleTypeEnum.NFT + | CollectibleTypeEnum.Ordinal + | CollectibleTypeEnum.RareSat + | CollectibleTypeEnum.Inscriptions; export type BaseNftsProps = { nfts: (ProtoNFT | NFT)[]; account: Account; }; + +export type Status = "error" | "success" | "pending"; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Ordinals.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Ordinals.ts index 38ce93b53a2a..2902f8a4bdae 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Ordinals.ts +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/Ordinals.ts @@ -1,14 +1,19 @@ +import { SimpleHashNft } from "@ledgerhq/live-nft/api/types"; import { mappingKeysWithIconAndName } from "../Ordinals/components/Icons"; import { IconProps } from "./Collection"; export type MappingKeys = keyof typeof mappingKeysWithIconAndName; -export type RareSat = { - count: string; - display_name: string | string[]; +type Icon = ({ size, color, style }: IconProps) => JSX.Element; + +export interface RareSat { + displayed_names: string; + icons?: Icon[]; year: string; - utxo_size: string; - icons?: { [key: string]: ({ size, color, style }: IconProps) => JSX.Element }; - name: string; - isDoubleRow?: boolean; -}; + count: string; + names: MappingKeys[]; + isMultipleRow: boolean; +} +export interface SimpleHashNftWithIcons extends SimpleHashNft { + icons?: Icon[]; +} diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/RareSats.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/RareSats.ts index 0aa47984190e..ad620f3b5af6 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/RareSats.ts +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/RareSats.ts @@ -52,11 +52,3 @@ export type ExtraMetadata = { animation_original_url: string | null; metadata_original_url: string | null; }; - -export type Sat = { - extra_metadata: ExtraMetadata; -}; - -export type MockedRareSat = { - nfts: Sat[]; -}; diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/Collectibles.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/Collectibles.ts index a4c730ef9630..db234075d289 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/Collectibles.ts +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/Collectibles.ts @@ -1,4 +1,12 @@ export enum CollectibleTypeEnum { NFT = "NFT", Ordinal = "Ordinal", + RareSat = "RareSat", + Inscriptions = "Inscriptions", +} + +export enum TanStackStatus { + Error = "error", + Success = "success", + Pending = "pending", } diff --git a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/DetailDrawer.ts b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/DetailDrawer.ts index da05e93321d9..e5bfbe7fbd4b 100644 --- a/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/DetailDrawer.ts +++ b/apps/ledger-live-desktop/src/newArch/features/Collectibles/types/enum/DetailDrawer.ts @@ -5,6 +5,7 @@ export enum FieldStatus { NoData = "nodata", Queued = "queued", } + export enum ItemType { Separator = "separator", ExternalLink = "external", diff --git a/apps/ledger-live-desktop/src/renderer/screens/account/index.tsx b/apps/ledger-live-desktop/src/renderer/screens/account/index.tsx index 26c74695c8c6..f5c1b1b850e1 100644 --- a/apps/ledger-live-desktop/src/renderer/screens/account/index.tsx +++ b/apps/ledger-live-desktop/src/renderer/screens/account/index.tsx @@ -44,6 +44,7 @@ import { useLocalizedUrl } from "~/renderer/hooks/useLocalizedUrls"; import { urls } from "~/config/urls"; import { CurrencyConfig } from "@ledgerhq/coin-framework/config"; import { useFeature } from "@ledgerhq/live-common/featureFlags/index"; +import { BitcoinAccount } from "@ledgerhq/coin-bitcoin/lib/types"; type Params = { id: string; @@ -220,7 +221,7 @@ const AccountPage = ({ ) ) : null} {isOrdinalsEnabled && account.type === "Account" && account.currency.id === "bitcoin" ? ( - + ) : null} {account.type === "Account" ? : null} { + if (nft.chain === OrdinalsChainsEnum.INSCRIPTIONS) accumulator.inscriptions.push(nft); + else accumulator.rareSats.push(nft); + return accumulator; + }, initialAccumulator); +} + +/** + * Processes an array of rareSats by removing the common rarity. + * @param rareSats - An array of rareSats to process. + * @returns An array of rareSats with the common rarity removed. + */ +export function removeCommonRareSats(rareSats: SimpleHashNft[]): SimpleHashNft[] { + return rareSats + .map(nft => { + const utxoDetails = nft.extra_metadata?.utxo_details; + if (utxoDetails) { + const { common, ...restSatributes } = utxoDetails.satributes; + return { + ...nft, + extra_metadata: { + ...nft.extra_metadata, + utxo_details: { + ...utxoDetails, + satributes: restSatributes, + }, + }, + }; + } + return nft; + }) + .filter(nft => { + const utxoDetails = nft.extra_metadata?.utxo_details; + if (utxoDetails) { + const keys = Object.keys(utxoDetails.satributes); + return keys.length > 0; + } + return true; + }); +} +/** + * Processes an array of rareSats by regrouping them by Contract Address. + * @param rareSats - An array of rareSats to process. + * @returns An array of rareSats with the common rarity removed. + */ +export function regroupRareSatsByContractAddress( + rareSats: SimpleHashNft[], +): Record { + return rareSats.reduce>((acc, sat) => { + const { contract_address } = sat; + acc[contract_address] = acc[contract_address] || []; + acc[contract_address].push(sat); + return acc; + }, {}); +} + +/** + * Processes the NFTs by restructuring them. + * @param nfts - The array of NFTs to process. + * @returns An object containing two arrays: rareSats and inscriptions. + */ +export function processOrdinals(nfts: SimpleHashNft[]): { + rareSats: SimpleHashNft[]; + inscriptions: SimpleHashNft[]; +} { + const { rareSats, inscriptions } = categorizeNftsByChain(nfts); + const rareSatsWithoutCommonSats = removeCommonRareSats(rareSats); + + return { + rareSats: rareSatsWithoutCommonSats, + inscriptions, + }; +} diff --git a/libs/live-nft-react/src/hooks/types.ts b/libs/live-nft-react/src/hooks/types.ts index efa0467d75ed..7321a50548be 100644 --- a/libs/live-nft-react/src/hooks/types.ts +++ b/libs/live-nft-react/src/hooks/types.ts @@ -50,3 +50,52 @@ export type RefreshMetadataResult = UseMutationResult< // Check Spam Score Contract or NFT export type CheckSpamScoreResult = UseQueryResult; + +// Fetch Ordinals from SimpleHash +export enum OrdinalsChainsEnum { + RARESATS = "utxo", + INSCRIPTIONS = "bitcoin", +} +export type OrdinalsStandard = "raresats" | "inscriptions"; +export type FetchNftsProps = { + addresses: string; + threshold: number; +}; +export enum RareSatRarity { + ALPHA = "alpha", + BLACK_EPIC = "black_epic", + BLACK_LEGENDARY = "black_legendary", + BLACK_MYTHIC = "black_mythic", + BLACK_RARE = "black_rare", + BLACK_UNCOMMON = "black_uncommon", + BLOCK_9 = "block_9", + BLOCK_9_450X = "block_9_450x", + BLOCK_78 = "block_78", + BLOCK_286 = "block_286", + BLOCK_666 = "block_666", + COMMON = "common", + EPIC = "epic", + FIRST_TX = "first_tx", + HITMAN = "hitman", + JPEG = "jpeg", + LEGACY = "legacy", + LEGENDARY = "legendary", + MYTHIC = "mythic", + NAKAMOTO = "nakamoto", + OMEGA = "omega", + PALIBLOCK = "paliblock", + PALINDROME = "palindrome", + PALINCEPTION = "palinception", + PIZZA = "pizza", + RARE = "rare", + UNCOMMON = "uncommon", + VINTAGE = "vintage", + LOW_SERIAL_NUMBER = "low_serial_number", + SPECIAL_TRANSACTION = "special_transaction", + COINBASE_REWARD = "coinbase_reward", + DUST = "dust", + UNIQUE_PATTERN = "unique_pattern", + COLORED_COIN = "colored_coin", + HISTORICAL_EVENT = "historical_event", + NON_STANDARD_SCRIPT = "non_standard_script", +} diff --git a/libs/live-nft-react/src/hooks/useFetchOrdinals.ts b/libs/live-nft-react/src/hooks/useFetchOrdinals.ts new file mode 100644 index 000000000000..0464912acbba --- /dev/null +++ b/libs/live-nft-react/src/hooks/useFetchOrdinals.ts @@ -0,0 +1,49 @@ +import { useEffect, useMemo } from "react"; +import { fetchNftsFromSimpleHash } from "@ledgerhq/live-nft/api/simplehash"; +import { InfiniteData, useInfiniteQuery, UseInfiniteQueryResult } from "@tanstack/react-query"; +import { SimpleHashResponse, SimpleHashNft } from "@ledgerhq/live-nft/api/types"; +import { FetchNftsProps, OrdinalsChainsEnum } from "./types"; +import { NFTS_QUERY_KEY } from "../queryKeys"; +import { processOrdinals } from "./helpers/ordinals"; + +type Result = UseInfiniteQueryResult, Error> & { + rareSats: SimpleHashNft[]; + inscriptions: SimpleHashNft[]; +}; + +export function useFetchOrdinals({ addresses, threshold }: FetchNftsProps): Result { + const chains = [OrdinalsChainsEnum.INSCRIPTIONS, OrdinalsChainsEnum.RARESATS]; + const addressString = Array.isArray(addresses) ? addresses.join(",") : addresses; + const queryResult = useInfiniteQuery({ + queryKey: [NFTS_QUERY_KEY.FetchOrdinals, addresses, chains], + queryFn: ({ pageParam }: { pageParam: string | undefined }) => + fetchNftsFromSimpleHash({ + addresses: addressString, + chains, + cursor: pageParam, + threshold, + }), + initialPageParam: undefined, + getNextPageParam: lastPage => lastPage.next_cursor, + enabled: addresses.length > 0, + }); + + const nfts = useMemo( + () => queryResult.data?.pages.flatMap(page => page.nfts) ?? [], + [queryResult.data], + ); + + const { rareSats, inscriptions } = useMemo(() => processOrdinals(nfts), [nfts]); + + useEffect(() => { + if (queryResult.hasNextPage && !queryResult.isFetchingNextPage) { + queryResult.fetchNextPage(); + } + }, [queryResult, queryResult.hasNextPage, queryResult.isFetchingNextPage]); + + return { + ...queryResult, + rareSats, + inscriptions, + }; +} diff --git a/libs/live-nft-react/src/index.ts b/libs/live-nft-react/src/index.ts index 3d8c50c23a88..db82f03c1f90 100644 --- a/libs/live-nft-react/src/index.ts +++ b/libs/live-nft-react/src/index.ts @@ -4,3 +4,5 @@ export * from "./hooks/useSpamReportNft"; export * from "./hooks/useNftFloorPrice"; export * from "./hooks/useRefreshMetadata"; export * from "./hooks/useCheckSpamScore"; +export * from "./hooks/useFetchOrdinals"; +export * from "./hooks/helpers/ordinals"; diff --git a/libs/live-nft-react/src/queryKeys.ts b/libs/live-nft-react/src/queryKeys.ts index 9405a157801a..78730a322ce0 100644 --- a/libs/live-nft-react/src/queryKeys.ts +++ b/libs/live-nft-react/src/queryKeys.ts @@ -3,4 +3,5 @@ export const NFTS_QUERY_KEY = { SpamReport: "SpamReport", FloorPrice: "FloorPrice", CheckSpamScore: "CheckSpamScore", + FetchOrdinals: "FetchOrdinals", }; diff --git a/libs/live-nft/src/api/types.ts b/libs/live-nft/src/api/types.ts index e2804a382464..4491f38c75da 100644 --- a/libs/live-nft/src/api/types.ts +++ b/libs/live-nft/src/api/types.ts @@ -11,6 +11,63 @@ export interface SimpleHashRefreshResponse { readonly message: string; } +export interface UtxoDetails { + readonly distinct_rare_sats: number; + readonly satributes: { + readonly [key: string]: { + readonly count: number; + readonly display_name: string; + readonly description: string; + readonly icon: string; + }; + }; + readonly sat_ranges: { + readonly starting_sat: number; + readonly value: number; + readonly distinct_rare_sats: number; + readonly year: string; + readonly subranges: { + readonly starting_sat: number; + readonly value: number; + readonly sat_types: string[]; + }[]; + }[]; + readonly block_number: number; + readonly value: number; + readonly script_pub_key: { + readonly asm: string; + readonly desc: string; + readonly hex: string; + readonly address: string; + readonly type: string; + }; +} + +export interface ordinal_details { + readonly charms: string | null; + readonly content_length: number; + readonly content_type: string; + readonly inscription_id: string; + readonly inscription_number: number; + readonly location: string; + readonly output_value: number; + readonly parents: string | null; + readonly protocol_content: string | null; + readonly protocol_name: string | null; + readonly sat_name: string; + readonly sat_number: number; + readonly sat_rarity: string; +} + +export interface preview { + readonly blurhash: string; + readonly image_large_url: string; + readonly image_medium_url: string; + readonly image_opengraph_url: string; + readonly image_small_url: string; + readonly predominate_color: string; +} + export interface SimpleHashNft { readonly nft_id: string; readonly chain: string; @@ -20,17 +77,22 @@ export interface SimpleHashNft { readonly name: string; readonly description: string; readonly token_count: number; + readonly previews?: preview; + readonly other_url?: string; readonly collection: { readonly name: string; readonly spam_score: number; }; readonly contract: { readonly type: string; + readonly name?: string; }; readonly extra_metadata?: { readonly ledger_metadata?: { readonly ledger_stax_image: string; }; + readonly utxo_details?: UtxoDetails; + readonly ordinal_details?: ordinal_details; readonly image_original_url: string; readonly animation_original_url: string; };