diff --git a/public/images/holders-square.png b/public/images/holders-square.png new file mode 100644 index 000000000..a9d64bbf0 Binary files /dev/null and b/public/images/holders-square.png differ diff --git a/src/app/communities/NftHoldersSection.tsx b/src/app/communities/NftHoldersSection.tsx new file mode 100644 index 000000000..0cfdf6069 --- /dev/null +++ b/src/app/communities/NftHoldersSection.tsx @@ -0,0 +1,153 @@ +import { Address } from 'viem' +import { useFetchNftHolders } from '@/shared/hooks/useFetchNftHolders' +import { HeaderTitle, Paragraph, Span } from '@/components/Typography' +import { Table } from '@/components/Table' +import { LoadingSpinner } from '@/components/LoadingSpinner' +import { EXPLORER_URL } from '@/lib/constants' +import { RxExternalLink, RxViewGrid } from 'react-icons/rx' +import Image from 'next/image' +import { truncateMiddle } from '@/lib/utils' +import { useState } from 'react' +import { FaTable } from 'react-icons/fa' + +interface HolderColumnProps { + address: string + rns?: string + image?: string +} +const HolderColumn = ({ address, rns, image }: HolderColumnProps) => { + return ( + + Holders Image + + {rns || address} + + + + ) +} + +interface IdNumberColumnProps { + id: string + image?: string +} +const IdNumberColumn = ({ id, image }: IdNumberColumnProps) => { + return ( +
+ Holders Image Square + #{id} +
+ ) +} + +interface HoldersSectionProps { + address: Address +} + +const CardHolderParagraph = ({ address }: { address: string }) => ( + + + HOLDER + + Holders Image + + {truncateMiddle(address, 5, 5)} + + + +) + +interface CardProps { + image: string + id: string + holderAddress: string +} + +const Card = ({ image, id, holderAddress }: CardProps) => { + return ( +
+ NFT +
+ + ID# {id} + + +
+
+ ) +} + +interface CardViewProps { + nfts: { image_url: string; id: string; owner: string }[] +} + +const CardView = ({ nfts }: CardViewProps) => ( +
+ {nfts.map(({ image_url, id, owner }) => ( + + ))} +
+) + +type ViewState = 'images' | 'table' + +const ViewIconHandler = ({ + view, + onChangeView, +}: { + view: ViewState + onChangeView: (view: ViewState) => void +}) => ( + +
onChangeView('table')} + > + +
+
onChangeView('images')} + > + +
+
+) + +export const NftHoldersSection = ({ address }: HoldersSectionProps) => { + const { currentResults, paginationElement, isLoading } = useFetchNftHolders(address) + + const [view, setView] = useState('images') + + const onChangeView = (selectedView: ViewState) => { + setView(selectedView) + } + + const holders = currentResults.map(({ owner, ens_domain_name, id, image_url }) => ({ + holder: , + 'ID Number': , + })) + + return ( +
+ + Holders + + + {view === 'table' && holders && holders?.length > 0 && } + {view === 'images' && currentResults && currentResults?.length > 0 && ( + + )} + {isLoading && } +
{paginationElement}
+ + ) +} diff --git a/src/app/user/Balances/actions.ts b/src/app/user/Balances/actions.ts index b634270cc..fc665830a 100644 --- a/src/app/user/Balances/actions.ts +++ b/src/app/user/Balances/actions.ts @@ -5,13 +5,19 @@ import { fetchNFTsOwnedByAddressAndNftAddress, fetchPricesEndpoint, fetchProposalsCreatedByGovernorAddress, + getNftHolders, getNftInfo, getTokenHoldersOfAddress, } from '@/lib/endpoints' import { tokenContracts, GovernorAddress } from '@/lib/contracts' import { NftMeta } from '@/shared/types' import { ipfsGateways } from '@/config' -import { NextPageParams, ServerResponseV2, TokenHoldersResponse } from '@/app/user/Balances/types' +import { + NextPageParams, + NftHolderItem, + ServerResponseV2, + TokenHoldersResponse, +} from '@/app/user/Balances/types' export const fetchAddressTokens = (address: string, chainId = 31) => axiosInstance @@ -126,6 +132,17 @@ export async function fetchIpfsUri( export const fetchNftInfo = (address: string) => axiosInstance.get(getNftInfo.replace('{{nftAddress}}', address)) +export const fetchNftHoldersOfAddress = async (address: string, params: NextPageParams | null) => { + const { data } = await axiosInstance.get>( + getNftHolders.replace('{{address}}', address), + { params }, + ) + if (data.error) { + throw new Error(data.error) + } + return data +} + export const fetchTokenHoldersOfAddress = async (address: string, nextParams: NextPageParams | null) => { const { data } = await axiosInstance.get>( getTokenHoldersOfAddress.replace('{{address}}', address), diff --git a/src/app/user/Balances/types.ts b/src/app/user/Balances/types.ts index 704cf7a47..d25a5e252 100644 --- a/src/app/user/Balances/types.ts +++ b/src/app/user/Balances/types.ts @@ -53,3 +53,19 @@ export interface ServerResponseV2 { next_page_params: NextPageParams | null error?: string } + +export type NftHolderItem = { + owner: string + ens_domain_name: null + id: string + image_url: string + metadata: Metadata +} + +export type Metadata = { + creator: string + description: string + external_url: string + image: string + name: string +} diff --git a/src/lib/endpoints.ts b/src/lib/endpoints.ts index fabf5a60c..e4da2eb90 100644 --- a/src/lib/endpoints.ts +++ b/src/lib/endpoints.ts @@ -20,3 +20,5 @@ export const getNftInfo = process.env.NEXT_PUBLIC_API_RWS_NFT_INFO || `/nfts/{{nftAddress}}?chainId=${CHAIN_ID}` export const getTokenHoldersOfAddress = `/address/{{address}}/holders?chainId=${CHAIN_ID}` + +export const getNftHolders = `/nfts/{{address}}/holders?chainId=${CHAIN_ID}` diff --git a/src/pages/communities/nft/[address].tsx b/src/pages/communities/nft/[address].tsx index f28897de6..7af985b1e 100644 --- a/src/pages/communities/nft/[address].tsx +++ b/src/pages/communities/nft/[address].tsx @@ -13,6 +13,7 @@ import { useAccount } from 'wagmi' import { useCommunity } from '@/shared/hooks/useCommunity' import { useStRif } from '@/shared/hooks/useStRIf' import { CopyButton } from '@/components/CopyButton' +import { NftHoldersSection } from '@/app/communities/NftHoldersSection' /** * Name of the local storage variable with information about whether the token was added to the wallet @@ -324,6 +325,8 @@ export default function Page() { + {/* Holders list */} + ) } diff --git a/src/shared/hooks/useFetchNftHolders.ts b/src/shared/hooks/useFetchNftHolders.ts new file mode 100644 index 000000000..b68f388a0 --- /dev/null +++ b/src/shared/hooks/useFetchNftHolders.ts @@ -0,0 +1,19 @@ +import { Address } from 'viem' +import { fetchNftHoldersOfAddress } from '@/app/user/Balances/actions' +import { usePagination } from '@/shared/hooks/usePagination' +import { usePaginationUi } from '@/shared/hooks/usePaginationUi' + +export const useFetchNftHolders = (address: Address) => { + const query = usePagination({ + queryKey: ['nft_holders'], + queryFn: ({ pageParam }) => fetchNftHoldersOfAddress(address, pageParam), + initialPageParam: null, + resultsPerTablePage: 10, + hasMorePagesProperty: 'next_page_params', + getNextPageParam: lastPage => lastPage.next_page_params, + }) + + const ui = usePaginationUi(query) + + return { ...query, ...ui } +}