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 (
+
+
+
+ {rns || address}
+
+
+
+ )
+}
+
+interface IdNumberColumnProps {
+ id: string
+ image?: string
+}
+const IdNumberColumn = ({ id, image }: IdNumberColumnProps) => {
+ return (
+
+
+
#{id}
+
+ )
+}
+
+interface HoldersSectionProps {
+ address: Address
+}
+
+const CardHolderParagraph = ({ address }: { address: string }) => (
+
+
+ HOLDER
+
+
+
+ {truncateMiddle(address, 5, 5)}
+
+
+
+)
+
+interface CardProps {
+ image: string
+ id: string
+ holderAddress: string
+}
+
+const Card = ({ image, id, holderAddress }: CardProps) => {
+ return (
+
+
+
+
+ )
+}
+
+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 }
+}