diff --git a/src/locales/messages.pot b/src/locales/messages.pot
index 911a89df0c..b2e94cd120 100644
--- a/src/locales/messages.pot
+++ b/src/locales/messages.pot
@@ -1229,9 +1229,6 @@ msgstr ""
msgid "Issue as ERC-20"
msgstr ""
-msgid "Pay {primaryAmount}"
-msgstr ""
-
msgid "Active"
msgstr ""
diff --git a/src/packages/v4/components/ProjectDashboard/ReduxProjectCartProvider.tsx b/src/packages/v4/components/ProjectDashboard/ReduxProjectCartProvider.tsx
index 02816d4d59..e84f41f073 100644
--- a/src/packages/v4/components/ProjectDashboard/ReduxProjectCartProvider.tsx
+++ b/src/packages/v4/components/ProjectDashboard/ReduxProjectCartProvider.tsx
@@ -1,8 +1,9 @@
-import { useWallet } from 'hooks/Wallet'
+import { useV4NftRewards } from 'packages/v4/contexts/V4NftRewards/V4NftRewardsProvider'
+import { useV4UserNftCredits } from 'packages/v4/contexts/V4UserNftCreditsProvider'
import { V4CurrencyOption } from 'packages/v4/models/v4CurrencyOption'
import React from 'react'
import { useProjectDispatch } from './redux/hooks'
-// import { projectCartActions } from './redux/projectCartSlice'
+import { projectCartActions } from './redux/projectCartSlice'
export type ProjectCartCurrencyAmount = {
amount: number
@@ -20,26 +21,22 @@ export const ReduxProjectCartProvider = ({
}: {
children: React.ReactNode
}) => {
- // const { rewardTiers } = useContext(NftRewardsContext).nftRewards
-
- const { userAddress } = useWallet()
- // const userNftCredits = useNftCredits(userAddress)
-
+ const {
+ nftRewards: { rewardTiers },
+ } = useV4NftRewards()
+ const nftCredits = useV4UserNftCredits()
const dispatch = useProjectDispatch()
// Set the nfts on load
- // useEffect(() => {
- // dispatch(projectCartActions.setAllNftRewards(rewardTiers ?? []))
- // }, [dispatch, rewardTiers])
+ React.useEffect(() => {
+ dispatch(projectCartActions.setAllNftRewards(rewardTiers ?? []))
+ }, [dispatch, rewardTiers])
// Set the user's NFT credits on load
- // useEffect(() => {
- // dispatch(
- // projectCartActions.setUserNftCredits(
- // userNftCredits.data?.toBigInt() ?? 0n,
- // ),
- // )
- // }, [dispatch, userNftCredits.data])
+ React.useEffect(() => {
+ if (nftCredits.isLoading) return
+ dispatch(projectCartActions.setUserNftCredits(nftCredits.data ?? 0n))
+ }, [dispatch, nftCredits.isLoading, nftCredits.data])
return <>{children}>
}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/NftReward.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/NftReward.tsx
deleted file mode 100644
index ac0f817e24..0000000000
--- a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/NftReward.tsx
+++ /dev/null
@@ -1,93 +0,0 @@
-// const NftReward: React.FC<{
-// nft: ProjectCartNftReward
-// className?: string
-// }> = ({ nft, className }) => {
-// const {
-// price,
-// name,
-// quantity,
-// fileUrl,
-// removeNft,
-// increaseQuantity,
-// decreaseQuantity,
-// } = useNftCartItem(nft)
-
-// const handleRemove = useCallback(() => {
-// emitConfirmationDeletionModal({
-// onConfirm: removeNft,
-// title: t`Remove NFT`,
-// description: t`Are you sure you want to remove this NFT?`,
-// })
-// }, [removeNft])
-
-// const handleDecreaseQuantity = useCallback(() => {
-// if (quantity - 1 <= 0) {
-// handleRemove()
-// } else {
-// decreaseQuantity()
-// }
-// }, [decreaseQuantity, handleRemove, quantity])
-
-// const priceText = price === null ? '-' : formatCurrencyAmount(price)
-
-// return (
-//
-//
-//
-//
-//
-//
-// NFT
-//
-
-//
{priceText}
-//
-//
-
-//
-//
-//
-//
-//
-// )
-// }
-
-// const RemoveIcon: React.FC<{ onClick: () => void }> = ({ onClick }) => (
-//
-// )
-
-// const QuantityControl: React.FC<{
-// quantity: number
-// onIncrease: () => void
-// onDecrease: () => void
-// }> = ({ quantity, onIncrease, onDecrease }) => {
-// return (
-//
-//
-//
-//
-// {quantity}
-//
-//
-//
-//
-// )
-// }
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/PayProjectModal.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/PayProjectModal.tsx
index 466aede953..0fe15b45fa 100644
--- a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/PayProjectModal.tsx
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/PayProjectModal.tsx
@@ -3,11 +3,13 @@ import EtherscanLink from 'components/EtherscanLink'
import ExternalLink from 'components/ExternalLink'
import { JuiceModal } from 'components/modals/JuiceModal'
import { Formik } from 'formik'
-import Image from "next/legacy/image"
+import Image from 'next/legacy/image'
+import { useV4UserNftCredits } from 'packages/v4/contexts/V4UserNftCreditsProvider'
import { twMerge } from 'tailwind-merge'
import { helpPagePath } from 'utils/helpPagePath'
import { MessageSection } from './components/MessageSection'
import { ReceiveSection } from './components/ReceiveSection'
+import { usePayAmounts } from './hooks/usePayAmounts'
import {
PayProjectModalFormValues,
usePayProjectModal,
@@ -16,8 +18,6 @@ import {
export const PayProjectModal: React.FC = () => {
const {
open,
- primaryAmount,
- secondaryAmount,
validationSchema,
isTransactionPending,
isTransactionConfirmed,
@@ -27,6 +27,7 @@ export const PayProjectModal: React.FC = () => {
setOpen,
onPaySubmit,
} = usePayProjectModal()
+ const { formattedTotalAmount } = usePayAmounts()
return (
@@ -50,7 +51,7 @@ export const PayProjectModal: React.FC = () => {
position="top"
okLoading={props.isSubmitting || isTransactionPending}
okButtonForm="PayProjectModalForm"
- okText={t`Pay ${primaryAmount}`}
+ okText={t`Pay ${formattedTotalAmount.primaryAmount}`}
cancelText={
isTransactionPending || isTransactionConfirmed
? t`Close`
@@ -97,19 +98,7 @@ export const PayProjectModal: React.FC = () => {
) : (
<>
-
-
- Total amount
-
-
- {primaryAmount} {' '}
- {secondaryAmount && (
-
- ({secondaryAmount})
-
- )}
-
-
+
@@ -172,3 +161,60 @@ export const PayProjectModal: React.FC = () => {
)
}
+
+const AmountSection = () => {
+ const { data: nftCredits } = useV4UserNftCredits()
+ const { formattedAmount, formattedNftCredits, formattedTotalAmount } =
+ usePayAmounts()
+
+ const RowData = ({
+ label,
+ primaryAmount,
+ secondaryAmount,
+ }: {
+ label: React.ReactNode
+ primaryAmount: React.ReactNode
+ secondaryAmount: React.ReactNode
+ }) => (
+
+
{label}
+
+ {primaryAmount} {' '}
+ {secondaryAmount && (
+
+ ({secondaryAmount})
+
+ )}
+
+
+ )
+
+ if (!nftCredits || nftCredits <= 0n || !formattedNftCredits)
+ return (
+
+ )
+
+ return (
+
+
+
+
+
+ )
+}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveNftItem.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveNftItem.tsx
new file mode 100644
index 0000000000..d1072d7a1c
--- /dev/null
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveNftItem.tsx
@@ -0,0 +1,37 @@
+import { Trans } from '@lingui/macro'
+import { CartItemBadge } from 'components/CartItemBadge'
+import { SmallNftSquare } from 'components/NftRewards/SmallNftSquare'
+import { useNftCartItem } from 'packages/v4/hooks/useNftCartItem'
+import { twMerge } from 'tailwind-merge'
+import { ProjectCartNftReward } from '../../../ReduxProjectCartProvider'
+
+export const ReceiveNftItem = ({
+ className,
+ nftReward,
+}: {
+ className?: string
+ nftReward: ProjectCartNftReward
+}) => {
+ const { fileUrl, name, quantity } = useNftCartItem(nftReward)
+
+ return (
+
+
+
+
+ {name}
+
+ NFT
+
+
+
{quantity}
+
+
+ )
+}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveSection.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveSection.tsx
index 4d0a946216..fdb45559e5 100644
--- a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveSection.tsx
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveSection.tsx
@@ -6,6 +6,7 @@ import {
} from '../hooks/usePayProjectModal/usePayProjectModal'
import { useProjectPaymentTokens } from '../hooks/useProjectPaymentTokens'
import { EditRewardBeneficiary } from './EditRewardBeneficiary'
+import { ReceiveNftItem } from './ReceiveNftItem'
import { ReceiveTokensItem } from './ReceiveTokensItem'
export const ReceiveSection = ({ className }: { className?: string }) => {
@@ -36,6 +37,13 @@ export const ReceiveSection = ({ className }: { className?: string }) => {
+ {nftRewards.map(nftReward => (
+
+ ))}
)
}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveTokensItem.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveTokensItem.tsx
index 3aa82bbce0..99a24668d8 100644
--- a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveTokensItem.tsx
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/components/ReceiveTokensItem.tsx
@@ -2,7 +2,6 @@ import { Trans } from '@lingui/macro'
import { CartItemBadge } from 'components/CartItemBadge'
import { ProjectHeaderLogo } from 'components/Project/ProjectHeader/ProjectHeaderLogo'
import { twMerge } from 'tailwind-merge'
-// import { useProjectHasErc20Token } from 'packages/v2v3/components/V2V3Project/ProjectDashboard/hooks/useProjectHasErc20Token'
// import { BUYBACK_DELEGATE_ENABLED_PROJECT_IDS } from 'packages/v2v3/constants/buybackDelegateEnabledProjectIds'
import { useProjectHasErc20Token } from 'packages/v4/hooks/useProjectHasErc20Token'
import { useProjectPaymentTokens } from '../hooks/useProjectPaymentTokens'
@@ -11,7 +10,6 @@ export const ReceiveTokensItem = ({ className }: { className?: string }) => {
const { receivedTickets, receivedTokenSymbolText } = useProjectPaymentTokens()
const projectHasErc20Token = useProjectHasErc20Token()
-
if (receivedTickets === '0') {
return null
}
@@ -24,11 +22,11 @@ export const ReceiveTokensItem = ({ className }: { className?: string }) => {
{receivedTokenSymbolText}
- { projectHasErc20Token ?
+ {projectHasErc20Token ? (
ERC-20
- : null}
+ ) : null}
{receivedTickets}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/hooks/usePayAmounts.ts b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/hooks/usePayAmounts.ts
new file mode 100644
index 0000000000..2ea6141cc6
--- /dev/null
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/hooks/usePayAmounts.ts
@@ -0,0 +1,132 @@
+import { useCurrencyConverter } from 'hooks/useCurrencyConverter'
+import { useV4UserNftCredits } from 'packages/v4/contexts/V4UserNftCreditsProvider'
+import { V4_CURRENCY_ETH, V4_CURRENCY_USD } from 'packages/v4/utils/currency'
+import { formatCurrencyAmount } from 'packages/v4/utils/formatCurrencyAmount'
+import React from 'react'
+import { fromWad, parseWad } from 'utils/format/formatNumber'
+import { useProjectSelector } from '../../../redux/hooks'
+import { usePayProjectModal } from './usePayProjectModal/usePayProjectModal'
+
+export const usePayAmounts = () => {
+ const converter = useCurrencyConverter()
+ const { payAmount } = useProjectSelector(state => state.projectCart)
+ const { primaryAmount, secondaryAmount } = usePayProjectModal()
+ const { data: nftCreditsData } = useV4UserNftCredits()
+
+ const payAmountRaw = React.useMemo(() => {
+ if (!payAmount) return
+
+ switch (payAmount.currency) {
+ case V4_CURRENCY_ETH:
+ return {
+ eth: parseWad(payAmount.amount),
+ usd: converter.weiToUsd(parseWad(payAmount.amount))!,
+ }
+ case V4_CURRENCY_USD:
+ return {
+ eth: converter.usdToWei(payAmount.amount),
+ usd: parseWad(payAmount.amount),
+ }
+ }
+ }, [converter, payAmount])
+
+ const appliedNFTCreditsRaw = React.useMemo(() => {
+ if (!payAmountRaw || !nftCreditsData) return
+
+ const nftCreditsApplied = payAmountRaw.eth.lt(nftCreditsData)
+ ? payAmountRaw.eth
+ : nftCreditsData
+
+ const eth = nftCreditsApplied
+ const usd = parseWad(converter.weiToUsd(nftCreditsApplied))!
+
+ return {
+ eth,
+ usd,
+ }
+ }, [converter, nftCreditsData, payAmountRaw])
+
+ const formattedNftCredits = React.useMemo(() => {
+ if (!appliedNFTCreditsRaw || !payAmount) return
+
+ switch (payAmount.currency) {
+ case V4_CURRENCY_ETH:
+ return {
+ primaryAmount: formatCurrencyAmount({
+ amount: fromWad(appliedNFTCreditsRaw.eth),
+ currency: V4_CURRENCY_ETH,
+ }),
+ secondaryAmount: formatCurrencyAmount({
+ amount: fromWad(appliedNFTCreditsRaw.usd),
+ currency: V4_CURRENCY_USD,
+ }),
+ }
+ case V4_CURRENCY_USD:
+ return {
+ primaryAmount: formatCurrencyAmount({
+ amount: fromWad(appliedNFTCreditsRaw.usd),
+ currency: V4_CURRENCY_USD,
+ }),
+ secondaryAmount: formatCurrencyAmount({
+ amount: fromWad(appliedNFTCreditsRaw.eth),
+ currency: V4_CURRENCY_ETH,
+ }),
+ }
+ }
+ }, [appliedNFTCreditsRaw, payAmount])
+
+ const formattedTotalAmount = React.useMemo(() => {
+ if (!payAmountRaw || !payAmount) return
+
+ if (!appliedNFTCreditsRaw) {
+ return {
+ primaryAmount: primaryAmount,
+ secondaryAmount: secondaryAmount,
+ }
+ }
+
+ const totalEth = payAmountRaw.eth.sub(appliedNFTCreditsRaw.eth)
+ const totalUsd = converter.weiToUsd(parseWad(totalEth))
+
+ const formattedEth = formatCurrencyAmount({
+ amount: fromWad(totalEth),
+ currency: V4_CURRENCY_ETH,
+ })
+ const formattedUsd = formatCurrencyAmount({
+ amount: fromWad(totalUsd),
+ currency: V4_CURRENCY_USD,
+ })
+
+ switch (payAmount?.currency) {
+ case V4_CURRENCY_ETH:
+ return {
+ primaryAmount: formattedEth,
+ secondaryAmount: formattedUsd,
+ }
+ case V4_CURRENCY_USD:
+ return {
+ primaryAmount: formattedUsd,
+ secondaryAmount: formattedEth,
+ }
+ }
+ }, [
+ appliedNFTCreditsRaw,
+ converter,
+ payAmount,
+ payAmountRaw,
+ primaryAmount,
+ secondaryAmount,
+ ])
+
+ return {
+ formattedAmount: { primaryAmount, secondaryAmount },
+ formattedNftCredits: {
+ primaryAmount: formattedNftCredits?.primaryAmount,
+ secondaryAmount: formattedNftCredits?.secondaryAmount,
+ },
+ formattedTotalAmount: {
+ primaryAmount: formattedTotalAmount?.primaryAmount,
+ secondaryAmount: formattedTotalAmount?.secondaryAmount,
+ },
+ }
+}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/hooks/usePayProjectModal/usePayProjectTx.ts b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/hooks/usePayProjectModal/usePayProjectTx.ts
index 16005cb97f..7116194971 100644
--- a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/hooks/usePayProjectModal/usePayProjectTx.ts
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayProjectModal/hooks/usePayProjectModal/usePayProjectTx.ts
@@ -1,19 +1,21 @@
+import { waitForTransactionReceipt } from '@wagmi/core'
+import { TxHistoryContext } from 'contexts/Transaction/TxHistoryContext'
import { FormikHelpers } from 'formik'
import { useWallet } from 'hooks/Wallet'
import { useCurrencyConverter } from 'hooks/useCurrencyConverter'
-import { useProjectSelector } from 'packages/v4/components/ProjectDashboard/redux/hooks'
-import { ProjectPayReceipt } from 'packages/v4/views/V4ProjectDashboard/hooks/useProjectPageQueries'
-// import { NftRewardsContext } from 'packages/v4/contexts/NftRewards/NftRewardsContext'
-// import { useProjectHasErc20 } from 'packages/v4/hooks/useProjectHasErc20'
-import { waitForTransactionReceipt } from '@wagmi/core'
-import { TxHistoryContext } from 'contexts/Transaction/TxHistoryContext'
-import { NATIVE_TOKEN } from 'juice-sdk-core'
+import { DEFAULT_METADATA, NATIVE_TOKEN } from 'juice-sdk-core'
import {
useJBContractContext,
+ useJBRulesetContext,
+ usePreparePayMetadata,
useWriteJbMultiTerminalPay,
} from 'juice-sdk-react'
-// import { useProjectHasErc20 } from 'packages/v2v3/hooks/useProjectHasErc20'
+import { useProjectSelector } from 'packages/v4/components/ProjectDashboard/redux/hooks'
+import { useV4NftRewards } from 'packages/v4/contexts/V4NftRewards/V4NftRewardsProvider'
+import { useV4UserNftCredits } from 'packages/v4/contexts/V4UserNftCreditsProvider'
+import { useProjectHasErc20Token } from 'packages/v4/hooks/useProjectHasErc20Token'
import { V4_CURRENCY_ETH } from 'packages/v4/utils/currency'
+import { ProjectPayReceipt } from 'packages/v4/views/V4ProjectDashboard/hooks/useProjectPageQueries'
import { wagmiConfig } from 'packages/v4/wagmiConfig'
import { useCallback, useContext, useMemo } from 'react'
import { buildPaymentMemo } from 'utils/buildPaymentMemo'
@@ -39,15 +41,17 @@ export const usePayProjectTx = ({
) => void
}) => {
const { userAddress } = useWallet()
+ const { data: nftCredits } = useV4UserNftCredits()
const { payAmount, chosenNftRewards } = useProjectSelector(
state => state.projectCart,
)
- // const {
- // nftRewards: { rewardTiers },
- // } = useContext(NftRewardsContext)
+ const {
+ nftRewards: { rewardTiers },
+ } = useV4NftRewards()
const converter = useCurrencyConverter()
const { receivedTickets } = useProjectPaymentTokens()
- // const projectHasErc20 = useProjectHasErc20()
+ // TODO: is this needed for preferClaimedTokens?
+ const projectHasErc20 = useProjectHasErc20Token()
const buildPayReceipt = useCallback(
(txHash: Hash): ProjectPayReceipt => {
@@ -69,17 +73,37 @@ export const usePayProjectTx = ({
const weiAmount = useMemo(() => {
if (!payAmount) {
return 0n
- } else if (payAmount.currency === V4_CURRENCY_ETH) {
- return parseEther(payAmount.amount.toString())
- } else {
- return converter.usdToWei(payAmount.amount).toBigInt()
}
- }, [payAmount, converter])
+ let weiAmount =
+ payAmount.currency === V4_CURRENCY_ETH
+ ? parseEther(payAmount.amount.toString())
+ : converter.usdToWei(payAmount.amount).toBigInt()
+ if (nftCredits) {
+ if (nftCredits >= weiAmount) {
+ weiAmount = 0n
+ } else {
+ weiAmount -= nftCredits
+ }
+ }
- // const prepareDelegateMetadata = usePrepareDelegatePayMetadata(weiAmount, {
- // nftRewards: chosenNftRewards,
- // receivedTickets,
- // })
+ return weiAmount
+ }, [converter, nftCredits, payAmount])
+
+ const {
+ rulesetMetadata: { data: rulesetMetadata },
+ } = useJBRulesetContext()
+ const metadata = usePreparePayMetadata(
+ rulesetMetadata?.dataHook
+ ? {
+ jb721Hook: {
+ dataHookAddress: rulesetMetadata.dataHook,
+ tierIdsToMint: chosenNftRewards
+ .map(({ id, quantity }) => Array(quantity).fill(BigInt(id)))
+ .flat(),
+ },
+ }
+ : undefined,
+ )
const { writeContractAsync: writePay } = useWriteJbMultiTerminalPay()
const { contracts, projectId } = useJBContractContext()
@@ -91,7 +115,6 @@ export const usePayProjectTx = ({
formikHelpers: FormikHelpers,
) => {
if (
- !weiAmount ||
!contracts.primaryNativeTerminal.data ||
!userAddress ||
!values.userAcceptsTerms
@@ -103,13 +126,13 @@ export const usePayProjectTx = ({
const memo = buildPaymentMemo({
text: messageString,
imageUrl: attachedUrl,
- // nftUrls: chosenNftRewards
- // .map(
- // ({ id }) =>
- // (rewardTiers ?? []).find(({ id: tierId }) => tierId === id)
- // ?.fileUrl,
- // )
- // .filter((url): url is string => !!url),
+ nftUrls: chosenNftRewards
+ .map(
+ ({ id }) =>
+ (rewardTiers ?? []).find(({ id: tierId }) => tierId === id)
+ ?.fileUrl,
+ )
+ .filter((url): url is string => !!url),
})
const beneficiary = (values.beneficiaryAddress ?? userAddress) as Address
const args = [
@@ -119,9 +142,15 @@ export const usePayProjectTx = ({
beneficiary,
0n,
memo,
- '0x0',
+ metadata ?? DEFAULT_METADATA,
] as const
+ // SIMULATE TRANSACTION:
+ // const encodedData = encodeFunctionData({
+ // abi: jbMultiTerminalAbi, // ABI of the contract
+ // functionName: 'pay',
+ // args,
+ // })
try {
const hash = await writePay({
address: contracts.primaryNativeTerminal.data,
@@ -147,21 +176,19 @@ export const usePayProjectTx = ({
}
},
[
- // projectHasErc20,
- buildPayReceipt,
- // chosenNftRewards,
- onTransactionConfirmedCallback,
- onTransactionErrorCallback,
- onTransactionPendingCallback,
- // payProjectTx,
- // rewardTiers,
weiAmount,
+ contracts.primaryNativeTerminal.data,
userAddress,
- // prepareDelegateMetadata,
+ chosenNftRewards,
projectId,
+ metadata,
+ rewardTiers,
writePay,
- contracts.primaryNativeTerminal.data,
+ onTransactionPendingCallback,
addTransaction,
+ onTransactionConfirmedCallback,
+ buildPayReceipt,
+ onTransactionErrorCallback,
],
)
}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayRedeemInput.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayRedeemInput.tsx
index 7afc137db8..681536befa 100644
--- a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayRedeemInput.tsx
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/PayRedeemInput.tsx
@@ -1,10 +1,21 @@
-import { ArrowDownIcon } from '@heroicons/react/24/outline'
+import {
+ ArrowDownIcon,
+ MinusIcon,
+ PlusIcon,
+ TrashIcon,
+} from '@heroicons/react/24/outline'
import { t } from '@lingui/macro'
import { Tooltip } from 'antd'
+import { CartItemBadge } from 'components/CartItemBadge'
+import { SmallNftSquare } from 'components/NftRewards/SmallNftSquare'
+import { TruncatedText } from 'components/TruncatedText'
+import { emitConfirmationDeletionModal } from 'hooks/emitConfirmationDeletionModal'
import { useCurrencyConverter } from 'hooks/useCurrencyConverter'
+import { useNftCartItem } from 'packages/v4/hooks/useNftCartItem'
import { V4_CURRENCY_USD } from 'packages/v4/utils/currency'
import { formatCurrencyAmount } from 'packages/v4/utils/formatCurrencyAmount'
-import { ReactNode, useMemo } from 'react'
+import { useProjectPageQueries } from 'packages/v4/views/V4ProjectDashboard/hooks/useProjectPageQueries'
+import React, { ReactNode, useMemo } from 'react'
import { twMerge } from 'tailwind-merge'
import { formatAmount } from 'utils/format/formatAmount'
import { ProjectCartNftReward } from '../ReduxProjectCartProvider'
@@ -134,14 +145,13 @@ export const PayRedeemInput = ({
)}
- {/* {nfts && nfts?.length > 0 && (
+ {nfts && nfts?.length > 0 && (
{nfts.map((nft, i) => (
))}
)}
- */}
{downArrow && (
@@ -208,3 +218,109 @@ const DownArrow = ({ className }: { className?: string }) => {
)
}
+
+const NftReward: React.FC<{
+ nft: ProjectCartNftReward
+ className?: string
+}> = ({ nft, className }) => {
+ const {
+ price,
+ name,
+ quantity,
+ fileUrl,
+ removeNft,
+ increaseQuantity,
+ decreaseQuantity,
+ } = useNftCartItem(nft)
+ const { setProjectPageTab } = useProjectPageQueries()
+
+ const handleRemove = React.useCallback(() => {
+ emitConfirmationDeletionModal({
+ onConfirm: removeNft,
+ title: t`Remove NFT`,
+ description: t`Are you sure you want to remove this NFT?`,
+ })
+ }, [removeNft])
+
+ const handleDecreaseQuantity = React.useCallback(() => {
+ if (quantity - 1 <= 0) {
+ handleRemove()
+ } else {
+ decreaseQuantity()
+ }
+ }, [decreaseQuantity, handleRemove, quantity])
+
+ const priceText = useMemo(() => {
+ if (price === null) {
+ return '-'
+ }
+ return formatCurrencyAmount(price)
+ }, [price])
+
+ return (
+
+
+
+
+
setProjectPageTab('nft_rewards')}
+ >
+
+ NFT
+
+
+
{priceText}
+
+
+
+
+
+
+
+
+ )
+}
+
+const RemoveIcon: React.FC<{ onClick: () => void }> = ({ onClick }) => (
+
+)
+
+const QuantityControl: React.FC<{
+ quantity: number
+ onIncrease: () => void
+ onDecrease: () => void
+}> = ({ quantity, onIncrease, onDecrease }) => {
+ return (
+
+
+
+
+ {quantity}
+
+
+
+
+ )
+}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/V4NftCreditsCallouts.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/V4NftCreditsCallouts.tsx
new file mode 100644
index 0000000000..787d11e527
--- /dev/null
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/V4NftCreditsCallouts.tsx
@@ -0,0 +1,41 @@
+import { CubeIcon } from '@heroicons/react/24/outline'
+import { Trans } from '@lingui/macro'
+import { Button } from 'antd'
+import { formatEther } from 'juice-sdk-core'
+import { useV4UserNftCredits } from 'packages/v4/contexts/V4UserNftCreditsProvider'
+import { useProjectPageQueries } from 'packages/v4/views/V4ProjectDashboard/hooks/useProjectPageQueries'
+
+export function V4NftCreditsCallouts() {
+ const { setProjectPageTab } = useProjectPageQueries()
+ const { data: nftCredits } = useV4UserNftCredits()
+
+ if (!nftCredits || nftCredits <= 0n) {
+ return null
+ }
+
+ return (
+
+
+
+
+
+
+ You have{' '}
+ {formatEther(nftCredits)} ETH {' '}
+ of unclaimed NFT credits
+
+
+
setProjectPageTab('nft_rewards')}
+ >
+ Browse NFTs
+
+
+ )
+}
diff --git a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/V4PayRedeemCard.tsx b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/V4PayRedeemCard.tsx
index fb63e4426d..68012a30a1 100644
--- a/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/V4PayRedeemCard.tsx
+++ b/src/packages/v4/components/ProjectDashboard/V4PayRedeemCard/V4PayRedeemCard.tsx
@@ -3,15 +3,19 @@ import { Trans, t } from '@lingui/macro'
import { Tooltip } from 'antd'
import { Callout } from 'components/Callout/Callout'
import { useJBRulesetContext } from 'juice-sdk-react'
+import { useV4NftRewards } from 'packages/v4/contexts/V4NftRewards/V4NftRewardsProvider'
import { usePayoutLimit } from 'packages/v4/hooks/usePayoutLimit'
+import { useProjectHasErc20Token } from 'packages/v4/hooks/useProjectHasErc20Token'
import { MAX_PAYOUT_LIMIT } from 'packages/v4/utils/math'
-import { ReactNode } from 'react'
+import { useV4TokensPanel } from 'packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4TokensPanel/hooks/useV4TokensPanel'
+import React, { ReactNode } from 'react'
import { twMerge } from 'tailwind-merge'
import { useProjectDispatch, useProjectSelector } from '../redux/hooks'
import { payRedeemActions } from '../redux/payRedeemSlice'
import { PayConfiguration } from './PayConfiguration'
import { PayProjectModal } from './PayProjectModal/PayProjectModal'
import { RedeemConfiguration } from './RedeemConfiguration'
+import { V4NftCreditsCallouts } from './V4NftCreditsCallouts'
type PayRedeemCardProps = {
className?: string
@@ -22,18 +26,18 @@ export const V4PayRedeemCard: React.FC = ({
}) => {
const { ruleset, rulesetMetadata } = useJBRulesetContext()
const state = useProjectSelector(state => state.payRedeem.cardState)
- // const { value: hasNfts, loading: hasNftsLoading } = useHasNftRewards()
+ const nftRewards = useV4NftRewards()
const { data: payoutLimit } = usePayoutLimit()
const dispatch = useProjectDispatch()
- const projectHasErc20Token = false // TODO
+ const projectHasErc20Token = useProjectHasErc20Token()
// TODO: We should probably break out tokens panel hook into reusable module
- // const { userTokenBalance: panelBalance } = useTokensPanel()
- // const tokenBalance = panelBalance
- // ? parseFloat(panelBalance.replaceAll(',', ''))
- // : undefined
- const tokenBalance = 0 // TODO
+ const { userTokenBalance: panelBalance } = useV4TokensPanel()
+ const tokenBalance = React.useMemo(() => {
+ if (!panelBalance) return undefined
+ return panelBalance.toFloat()
+ }, [panelBalance])
const redeems = {
loading: ruleset.isLoading,
enabled:
@@ -41,15 +45,25 @@ export const V4PayRedeemCard: React.FC = ({
rulesetMetadata.data.redemptionRate.value > 0n,
}
- const weight = ruleset.data?.weight
- const isIssuingTokens = Boolean(weight && weight.value > 0n)
- // const showNfts = hasNfts && !hasNftsLoading
- const noticeText = isIssuingTokens
- ? // showNfts
- // ? t`Project isn't currently issuing tokens, but is issuing NFTs`
- // :
- t`Project isn't currently issuing tokens`
- : undefined
+ const isIssuingTokens = React.useMemo(() => {
+ const weight = ruleset.data?.weight
+ return Boolean(weight && weight.value > 0n)
+ }, [ruleset.data?.weight])
+
+ const noticeText = React.useMemo(() => {
+ if (!isIssuingTokens) {
+ return undefined
+ }
+ const showNfts =
+ !nftRewards.loading &&
+ (nftRewards.nftRewards.rewardTiers ?? []).length > 0
+
+ if (showNfts) {
+ return t`Project isn't currently issuing tokens, but is issuing NFTs`
+ }
+
+ return t`Project isn't currently issuing tokens`
+ }, [isIssuingTokens, nftRewards.loading, nftRewards.nftRewards.rewardTiers])
const redeemDisabled =
!rulesetMetadata.data?.redemptionRate ||
@@ -111,9 +125,10 @@ export const V4PayRedeemCard: React.FC = ({
)}
- {/* */}
- {/*
- {projectHasErc20Token && unclaimedTokenBalance?.gt(0) && (
+
+
+ {/* TODO */}
+ {/* {projectHasErc20Token && unclaimedTokenBalance?.gt(0) && (
)} */}
diff --git a/src/packages/v4/components/ProjectDashboard/components/SuccessPayView/SuccessPayView.tsx b/src/packages/v4/components/ProjectDashboard/components/SuccessPayView/SuccessPayView.tsx
index abbee4066b..4a3ebc13cd 100644
--- a/src/packages/v4/components/ProjectDashboard/components/SuccessPayView/SuccessPayView.tsx
+++ b/src/packages/v4/components/ProjectDashboard/components/SuccessPayView/SuccessPayView.tsx
@@ -5,6 +5,7 @@ import dynamic from 'next/dynamic'
import Link from 'next/link'
import { v4ProjectRoute } from 'packages/v4/utils/routes'
import { useChainId } from 'wagmi'
+import { SuccessNftItem } from './components/SuccessNftItem'
import { SuccessPayCard } from './components/SuccessPayCard'
import { SuccessTokensItem } from './components/SuccessTokensItem'
import { useSuccessPayView } from './hooks/useSuccessPayView'
@@ -78,6 +79,9 @@ export const SuccessPayView = () => {
Your NFTs & Rewards
+ {projectPayReceipt?.nfts.map(({ id }) => (
+
+ ))}
diff --git a/src/packages/v4/components/ProjectDashboard/components/SuccessPayView/components/SuccessNftItem.tsx b/src/packages/v4/components/ProjectDashboard/components/SuccessPayView/components/SuccessNftItem.tsx
new file mode 100644
index 0000000000..c494bb7108
--- /dev/null
+++ b/src/packages/v4/components/ProjectDashboard/components/SuccessPayView/components/SuccessNftItem.tsx
@@ -0,0 +1,48 @@
+import { Trans } from '@lingui/macro'
+import { CartItemBadge } from 'components/CartItemBadge'
+import { NftPreview } from 'components/NftRewards/NftPreview'
+import { SmallNftSquare } from 'components/NftRewards/SmallNftSquare'
+import { useV4NftRewards } from 'packages/v4/contexts/V4NftRewards/V4NftRewardsProvider'
+import { useMemo, useState } from 'react'
+
+export const SuccessNftItem = ({ id }: { id: number }) => {
+ const {
+ nftRewards: { rewardTiers },
+ } = useV4NftRewards()
+ const [previewVisible, setPreviewVisible] = useState(false)
+
+ const openPreview = () => setPreviewVisible(true)
+
+ const rewardTier = useMemo(() => {
+ if (!rewardTiers) return undefined
+ const nftReward = rewardTiers.find(reward => reward.id === id)
+ return nftReward
+ }, [id, rewardTiers])
+
+ return (
+ <>
+
+
+
+ {rewardTier?.name ?? ''}
+
+
+ NFT
+
+
+ {rewardTier && (
+
+ )}
+ >
+ )
+}
diff --git a/src/packages/v4/contexts/V4NftRewards/V4NftRewardsProvider.tsx b/src/packages/v4/contexts/V4NftRewards/V4NftRewardsProvider.tsx
new file mode 100644
index 0000000000..44a6d1778c
--- /dev/null
+++ b/src/packages/v4/contexts/V4NftRewards/V4NftRewardsProvider.tsx
@@ -0,0 +1,126 @@
+import {
+ jb721TiersHookStoreAbi,
+ useJBRulesetContext,
+ useReadJb721TiersHookPricingContext,
+ useReadJb721TiersHookStoreAddress,
+ useReadJb721TiersHookStoreFlagsOf,
+ useReadJb721TiersHookStoreTiersOf,
+} from 'juice-sdk-react'
+import { JB721GovernanceType } from 'models/nftRewards'
+import { V2V3CurrencyOption } from 'packages/v2v3/models/currencyOption'
+import React, { createContext } from 'react'
+import { DEFAULT_NFT_PRICING } from 'redux/slices/editingV2Project'
+import { NftRewardsData } from 'redux/slices/shared/v2ProjectTypes'
+import { CIDsOfNftRewardTiersResponse } from 'utils/nftRewards'
+import { ContractFunctionReturnType } from 'viem'
+import { useNftRewards } from './useNftRewards'
+
+const DEFAULT_NFT_FLAGS = {
+ noNewTiersWithReserves: false,
+ noNewTiersWithVotes: false,
+ noNewTiersWithOwnerMinting: false,
+ preventOverspending: false,
+}
+
+// TODO: This should be imported from the SDK
+export type JB721TierV4 = ContractFunctionReturnType<
+ typeof jb721TiersHookStoreAbi,
+ 'view',
+ 'tiersOf'
+>[0]
+
+type NftRewardsContextType = {
+ // nftRewards: is useReadJb721TiersHookStoreTiersOf.data returned
+ nftRewards: Omit<
+ NftRewardsData,
+ 'flags' | 'collectionMetadata' | 'postPayModal'
+ >
+ loading: boolean | undefined
+}
+
+export const V4NftRewardsContext = createContext({
+ nftRewards: {
+ CIDs: undefined,
+ rewardTiers: undefined,
+ // postPayModal: undefined,
+ // collectionMetadata: EMPTY_NFT_COLLECTION_METADATA,
+ // flags: DEFAULT_NFT_FLAGS,
+ governanceType: JB721GovernanceType.NONE,
+ pricing: DEFAULT_NFT_PRICING,
+ },
+ loading: false,
+})
+
+export const V4NftRewardsProvider: React.FC<
+ React.PropsWithChildren
+> = ({ children }) => {
+ const jbRuleSet = useJBRulesetContext()
+ const dataHookAddress = jbRuleSet.rulesetMetadata.data?.dataHook
+
+ const storeAddress = useReadJb721TiersHookStoreAddress({
+ address: dataHookAddress,
+ })
+
+ const tiersOf = useReadJb721TiersHookStoreTiersOf({
+ address: storeAddress.data,
+ args: [
+ dataHookAddress ?? `0x${'0'.repeat(40)}`,
+ [], // _categories
+ false, // _includeResolvedUri, return in each tier a result from a tokenUriResolver if one is included in the delegate
+ 0n, // _startingId
+ 10n, // limit
+ ],
+ })
+
+ const { data: loadedRewardTiers, isLoading: nftRewardTiersLoading } =
+ useNftRewards(tiersOf.data ?? [], 4, storeAddress.data)
+
+ const loadedCIDs = CIDsOfNftRewardTiersResponse(tiersOf.data ?? [])
+
+ const p = useReadJb721TiersHookPricingContext()
+ const currency = Number(p.data ? p.data[0] : 0) as V2V3CurrencyOption
+
+ const flags = useReadJb721TiersHookStoreFlagsOf({
+ address: '0x7b1F4Ba6312A104E645B06Ab97e4CaA1ef0F773f',
+ })
+
+ const loading = React.useMemo(
+ () =>
+ storeAddress.isLoading ||
+ tiersOf.isLoading ||
+ nftRewardTiersLoading ||
+ p.isLoading ||
+ flags.isLoading,
+ [
+ storeAddress.isLoading,
+ tiersOf.isLoading,
+ nftRewardTiersLoading,
+ p.isLoading,
+ flags.isLoading,
+ ],
+ )
+
+ const ctx = {
+ nftRewards: {
+ CIDs: loadedCIDs,
+ rewardTiers: loadedRewardTiers,
+ pricing: { currency },
+ governanceType: JB721GovernanceType.NONE,
+ // collectionMetadata: {
+ // ...EMPTY_NFT_COLLECTION_METADATA,
+ // uri: collection
+ // }
+ // postPayModal: projectMetadata?.nftPaymentSuccessModal,
+ // flags: flags.data ?? DEFAULT_NFT_FLAGS,
+ },
+ loading,
+ }
+
+ return (
+
+ {children}
+
+ )
+}
+
+export const useV4NftRewards = () => React.useContext(V4NftRewardsContext)
diff --git a/src/packages/v4/contexts/V4NftRewards/useNftRewards.ts b/src/packages/v4/contexts/V4NftRewards/useNftRewards.ts
new file mode 100644
index 0000000000..463806752b
--- /dev/null
+++ b/src/packages/v4/contexts/V4NftRewards/useNftRewards.ts
@@ -0,0 +1,59 @@
+import { useQuery, UseQueryResult } from '@tanstack/react-query'
+import axios from 'axios'
+import { formatEther } from 'juice-sdk-core'
+import { IPFSNftRewardTier, NftRewardTier } from 'models/nftRewards'
+import { withHttps } from 'utils/externalLink'
+import { cidFromUrl, decodeEncodedIpfsUri, ipfsGatewayUrl } from 'utils/ipfs'
+import { JB721TierV4 } from './V4NftRewardsProvider'
+
+async function fetchRewardTierMetadata({ tier }: { tier: JB721TierV4 }) {
+ const tierCid = decodeEncodedIpfsUri(tier.encodedIPFSUri)
+ const url = ipfsGatewayUrl(tierCid)
+
+ const response = await axios.get(url)
+ const tierMetadata: IPFSNftRewardTier = response.data
+
+ const maxSupply = tier.initialSupply
+
+ // Some projects have image links hard-coded to the old IPFS gateway.
+ const pinataRegex = /^(https?:\/\/jbx\.mypinata\.cloud)/
+ if (tierMetadata?.image && pinataRegex.test(tierMetadata.image)) {
+ const imageUrlCid = cidFromUrl(tierMetadata.image)
+ tierMetadata.image = ipfsGatewayUrl(imageUrlCid)
+ }
+
+ const rawContributionFloor = tier.price
+
+ return {
+ id: tier.id,
+ name: tierMetadata.name,
+ description: tierMetadata.description,
+ externalLink: withHttps(tierMetadata.externalLink),
+ // convert rawContributionFloor bigint to a number
+ contributionFloor: formatEther(rawContributionFloor),
+ maxSupply,
+ remainingSupply: tier.remainingSupply,
+ fileUrl: tierMetadata.image,
+ beneficiary: tier.reserveBeneficiary,
+ reservedRate: tier.reserveFrequency,
+ votingWeight: tier.votingUnits,
+ }
+}
+
+export const useNftRewards = (
+ tiers: readonly JB721TierV4[],
+ projectId: number | undefined,
+ dataSourceAddress: string | undefined,
+): UseQueryResult => {
+ const enabled = Boolean(tiers?.length)
+
+ return useQuery({
+ queryKey: ['nftRewards', projectId, dataSourceAddress],
+ enabled,
+ queryFn: async () => {
+ return await Promise.all(
+ tiers.map(tier => fetchRewardTierMetadata({ tier })),
+ )
+ },
+ })
+}
diff --git a/src/packages/v4/contexts/V4UserNftCreditsProvider.tsx b/src/packages/v4/contexts/V4UserNftCreditsProvider.tsx
new file mode 100644
index 0000000000..d9bcaebb05
--- /dev/null
+++ b/src/packages/v4/contexts/V4UserNftCreditsProvider.tsx
@@ -0,0 +1,37 @@
+import { useWallet } from 'hooks/Wallet'
+import {
+ useJBRulesetContext,
+ useReadJb721TiersHookPayCreditsOf,
+} from 'juice-sdk-react'
+import React, { PropsWithChildren } from 'react'
+
+const V4UserNftCreditsContext = React.createContext<{
+ data: bigint | undefined
+ isLoading: boolean
+}>({
+ data: undefined,
+ isLoading: false,
+})
+
+export const V4UserNftCreditsProvider: React.FC = ({
+ children,
+}) => {
+ const { userAddress } = useWallet()
+ const {
+ rulesetMetadata: { data: rulesetMetadata },
+ } = useJBRulesetContext()
+ const creds = useReadJb721TiersHookPayCreditsOf({
+ address: rulesetMetadata?.dataHook,
+ args: userAddress ? [userAddress] : undefined,
+ })
+
+ return (
+
+ {children}
+
+ )
+}
+
+export const useV4UserNftCredits = () => {
+ return React.useContext(V4UserNftCreditsContext)
+}
diff --git a/src/packages/v4/hooks/JB721Delegate/transactor/useLaunchProjectWithNftsTx.ts b/src/packages/v4/hooks/JB721Delegate/transactor/useLaunchProjectWithNftsTx.ts
index f105666072..04608418c7 100644
--- a/src/packages/v4/hooks/JB721Delegate/transactor/useLaunchProjectWithNftsTx.ts
+++ b/src/packages/v4/hooks/JB721Delegate/transactor/useLaunchProjectWithNftsTx.ts
@@ -1,3 +1,7 @@
+import { waitForTransactionReceipt } from '@wagmi/core'
+import { JUICEBOX_MONEY_PROJECT_METADATA_DOMAIN } from 'constants/metadataDomain'
+import { TxHistoryContext } from 'contexts/Transaction/TxHistoryContext'
+import { useWallet } from 'hooks/Wallet'
import {
DEFAULT_MEMO,
NATIVE_TOKEN,
@@ -9,26 +13,21 @@ import {
useReadJb721TiersHookStoreTiersOf,
useWriteJb721TiersHookProjectDeployerLaunchProjectFor,
} from 'juice-sdk-react'
+import { isValidMustStartAtOrAfter } from 'packages/v2v3/utils/fundingCycle'
import {
JBDeploy721TiersHookConfig,
LaunchProjectWithNftsTxArgs,
} from 'packages/v4/models/nfts'
-import { Address, WaitForTransactionReceiptReturnType, zeroAddress } from 'viem'
-import {
- LaunchV2V3ProjectArgs,
- transformV2V3CreateArgsToV4,
-} from '../../../utils/launchProjectTransformers'
-
-import { waitForTransactionReceipt } from '@wagmi/core'
-import { JUICEBOX_MONEY_PROJECT_METADATA_DOMAIN } from 'constants/metadataDomain'
-import { TxHistoryContext } from 'contexts/Transaction/TxHistoryContext'
-import { useWallet } from 'hooks/Wallet'
-import { isValidMustStartAtOrAfter } from 'packages/v2v3/utils/fundingCycle'
import { wagmiConfig } from 'packages/v4/wagmiConfig'
import { useContext } from 'react'
import { DEFAULT_MUST_START_AT_OR_AFTER } from 'redux/slices/shared/v2ProjectDefaultState'
import { ipfsUri } from 'utils/ipfs'
+import { Address, WaitForTransactionReceiptReturnType, zeroAddress } from 'viem'
import { useChainId } from 'wagmi'
+import {
+ LaunchV2V3ProjectArgs,
+ transformV2V3CreateArgsToV4,
+} from '../../../utils/launchProjectTransformers'
import {
LaunchTxOpts,
SUPPORTED_JB_CONTROLLER_ADDRESS,
diff --git a/src/packages/v4/hooks/useNftCartItem.ts b/src/packages/v4/hooks/useNftCartItem.ts
new file mode 100644
index 0000000000..6f9b23766f
--- /dev/null
+++ b/src/packages/v4/hooks/useNftCartItem.ts
@@ -0,0 +1,53 @@
+import { useProjectDispatch } from 'packages/v2v3/components/V2V3Project/ProjectDashboard/redux/hooks'
+import React from 'react'
+import { projectCartActions } from '../components/ProjectDashboard/redux/projectCartSlice'
+import { ProjectCartNftReward } from '../components/ProjectDashboard/ReduxProjectCartProvider'
+import { useV4NftRewards } from '../contexts/V4NftRewards/V4NftRewardsProvider'
+import { V4_CURRENCY_ETH } from '../utils/currency'
+
+export const useNftCartItem = ({ id, quantity }: ProjectCartNftReward) => {
+ const dispatch = useProjectDispatch()
+ const { nftRewards } = useV4NftRewards()
+ const rewardTiers = React.useMemo(
+ () => nftRewards.rewardTiers ?? [],
+ [nftRewards.rewardTiers],
+ )
+
+ const rewardTier = React.useMemo(
+ () => rewardTiers.find(tier => tier.id === id),
+ [rewardTiers, id],
+ )
+
+ const price = React.useMemo(
+ () => ({
+ amount: (rewardTier?.contributionFloor ?? 0) * quantity,
+ currency: V4_CURRENCY_ETH,
+ }),
+ [quantity, rewardTier?.contributionFloor],
+ )
+
+ const removeNft = React.useCallback(
+ () => dispatch(projectCartActions.removeNftReward({ id })),
+ [dispatch, id],
+ )
+
+ const increaseQuantity = React.useCallback(
+ () => dispatch(projectCartActions.increaseNftRewardQuantity({ id })),
+ [dispatch, id],
+ )
+
+ const decreaseQuantity = React.useCallback(
+ () => dispatch(projectCartActions.decreaseNftRewardQuantity({ id })),
+ [dispatch, id],
+ )
+
+ return {
+ name: rewardTier?.name,
+ fileUrl: rewardTier?.fileUrl,
+ quantity,
+ price,
+ removeNft,
+ increaseQuantity,
+ decreaseQuantity,
+ }
+}
diff --git a/src/packages/v4/hooks/useNftRewardsEnabledForPay.ts b/src/packages/v4/hooks/useNftRewardsEnabledForPay.ts
new file mode 100644
index 0000000000..5a18cbfa54
--- /dev/null
+++ b/src/packages/v4/hooks/useNftRewardsEnabledForPay.ts
@@ -0,0 +1,25 @@
+import { JBRulesetContext, useJBRulesetContext } from 'juice-sdk-react'
+import React from 'react'
+import { zeroAddress } from 'viem'
+import { useV4NftRewards } from '../contexts/V4NftRewards/V4NftRewardsProvider'
+
+type RulesetMetadata = JBRulesetContext['rulesetMetadata']['data']
+
+export function useNftRewardsEnabledForPay() {
+ const jbRuleset = useJBRulesetContext()
+ const { nftRewards } = useV4NftRewards()
+
+ const hasNftRewards = React.useMemo(
+ () => nftRewards.rewardTiers?.length !== 0,
+ [nftRewards.rewardTiers],
+ )
+
+ return hasNftRewards && hasDataSourceForPay(jbRuleset.rulesetMetadata.data)
+}
+
+const hasDataSourceForPay = (rulesetMetadata: RulesetMetadata) => {
+ return (
+ rulesetMetadata?.dataHook !== zeroAddress &&
+ !!rulesetMetadata?.useDataHookForPay
+ )
+}
diff --git a/src/packages/v4/utils/editRuleset.ts b/src/packages/v4/utils/editRuleset.ts
index ca46980f54..38f28dbd09 100644
--- a/src/packages/v4/utils/editRuleset.ts
+++ b/src/packages/v4/utils/editRuleset.ts
@@ -1,9 +1,9 @@
-import { NATIVE_TOKEN } from "juice-sdk-core";
-import round from "lodash/round";
-import { issuanceRateFrom } from "packages/v2v3/utils/math";
-import { parseWad } from "utils/format/formatNumber";
-import { otherUnitToSeconds } from "utils/format/formatTime";
-import { EditCycleFormFields } from "../views/V4ProjectSettings/EditCyclePage/EditCycleFormFields";
+import { NATIVE_TOKEN } from 'juice-sdk-core'
+import round from 'lodash/round'
+import { issuanceRateFrom } from 'packages/v2v3/utils/math'
+import { parseWad } from 'utils/format/formatNumber'
+import { otherUnitToSeconds } from 'utils/format/formatTime'
+import { EditCycleFormFields } from '../views/V4ProjectSettings/EditCyclePage/EditCycleFormFields'
export function transformEditCycleFormFieldsToTxArgs({
formValues,
@@ -11,21 +11,21 @@ export function transformEditCycleFormFieldsToTxArgs({
tokenAddress,
projectId,
}: {
- formValues: EditCycleFormFields;
- primaryNativeTerminal: `0x${string}`;
- tokenAddress: `0x${string}`;
- projectId: bigint;
+ formValues: EditCycleFormFields
+ primaryNativeTerminal: `0x${string}`
+ tokenAddress: `0x${string}`
+ projectId: bigint
}) {
- const now = round(new Date().getTime() / 1000);
- const mustStartAtOrAfter = now;
+ const now = round(new Date().getTime() / 1000)
+ const mustStartAtOrAfter = now
const duration = otherUnitToSeconds({
duration: formValues.duration,
unit: formValues.durationUnit.value,
})
- const weight = BigInt(issuanceRateFrom(formValues.issuanceRate.toString()));
- const decayPercent = round(formValues.decayPercent * 10000000);
- const approvalHook = formValues.approvalHook;
+ const weight = BigInt(issuanceRateFrom(formValues.issuanceRate.toString()))
+ const decayPercent = round(formValues.decayPercent * 10000000)
+ const approvalHook = formValues.approvalHook
const rulesetConfigurations = [
{
@@ -54,15 +54,15 @@ export function transformEditCycleFormFieldsToTxArgs({
useTotalSurplusForRedemptions: false, // Defaulting to false as it's not in formValues
useDataHookForPay: false, // Defaulting to false as it's not in formValues
useDataHookForRedeem: false, // Defaulting to false as it's not in formValues
- dataHook: "0x0000000000000000000000000000000000000000" as `0x${string}`, // Defaulting to a null address
+ dataHook: '0x0000000000000000000000000000000000000000' as `0x${string}`, // Defaulting to a null address
metadata: 0, // Assuming no additional metadata is provided
- allowCrosschainSuckerExtension: false
+ allowCrosschainSuckerExtension: false,
},
splitGroups: [
{
groupId: BigInt(NATIVE_TOKEN),
- splits: formValues.payoutSplits.map((split) => ({
+ splits: formValues.payoutSplits.map(split => ({
preferAddToBalance: Boolean(split.preferAddToBalance),
percent: Number(split.percent.value),
projectId: BigInt(split.projectId),
@@ -73,7 +73,7 @@ export function transformEditCycleFormFieldsToTxArgs({
},
{
groupId: BigInt(1),
- splits: formValues.reservedTokensSplits.map((split) => ({
+ splits: formValues.reservedTokensSplits.map(split => ({
preferAddToBalance: Boolean(split.preferAddToBalance),
percent: Number(split.percent.value),
projectId: BigInt(split.projectId),
@@ -103,11 +103,7 @@ export function transformEditCycleFormFieldsToTxArgs({
},
],
},
- ];
+ ]
- return [
- projectId,
- rulesetConfigurations,
- formValues.memo ?? "",
- ] as const;
+ return [projectId, rulesetConfigurations, formValues.memo ?? ''] as const
}
diff --git a/src/packages/v4/utils/launchProjectTransformers.ts b/src/packages/v4/utils/launchProjectTransformers.ts
index 066ea2d678..0effb3028b 100644
--- a/src/packages/v4/utils/launchProjectTransformers.ts
+++ b/src/packages/v4/utils/launchProjectTransformers.ts
@@ -128,7 +128,7 @@ type LaunchProjectJBSplit = Omit & { percent: number }
export type LaunchV4ProjectGroupedSplit = Omit<
V4GroupedSplits,
'splits' | 'groupId'
-> & { splits: LaunchProjectJBSplit[], groupId: bigint }
+> & { splits: LaunchProjectJBSplit[]; groupId: bigint }
export function transformV2V3SplitsToV4({
v2v3Splits,
@@ -136,10 +136,7 @@ export function transformV2V3SplitsToV4({
v2v3Splits: V2V3GroupedSplits[]
}): LaunchV4ProjectGroupedSplit[] {
return v2v3Splits.map(group => ({
- groupId:
- group.group === SplitGroup.ETHPayout
- ? BigInt(NATIVE_TOKEN)
- : 1n, // TODO dont hardcode reserved token group as 1n
+ groupId: group.group === SplitGroup.ETHPayout ? BigInt(NATIVE_TOKEN) : 1n, // TODO dont hardcode reserved token group as 1n
splits: group.splits.map(split => ({
preferAddToBalance: Boolean(split.preferClaimed),
percent: split.percent,
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/V4ActivityList.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/V4ActivityList.tsx
index 27db6ece2d..a6f83fd979 100644
--- a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/V4ActivityList.tsx
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/V4ActivityList.tsx
@@ -1,38 +1,38 @@
import { t } from '@lingui/macro'
import Loading from 'components/Loading'
-import {
- NativeTokenValue,
- useJBContractContext,
- useJBTokenContext,
-} from 'juice-sdk-react'
+import RichNote from 'components/RichNote/RichNote'
+import { NativeTokenValue, useJBContractContext } from 'juice-sdk-react'
import {
OrderDirection,
PayEvent_OrderBy,
PayEventsDocument,
} from 'packages/v4/graphql/client/graphql'
import { useSubgraphQuery } from 'packages/v4/graphql/useSubgraphQuery'
+import React from 'react'
import { ActivityEvent } from './activityEventElems/ActivityElement'
import { ActivityOptions } from './ActivityOptions'
import { PayEvent } from './models/ActivityEvents'
import { transformPayEventsRes } from './utils/transformEventsData'
export function V4ActivityList() {
- const { token } = useJBTokenContext()
const { projectId } = useJBContractContext()
// TODO: pageSize (pagination)
const { data: payEventsData, isLoading } = useSubgraphQuery({
- document: PayEventsDocument,
+ document: PayEventsDocument,
variables: {
orderBy: PayEvent_OrderBy.timestamp,
orderDirection: OrderDirection.desc,
where: {
projectId: Number(projectId),
},
- }
+ },
})
- const payEvents = transformPayEventsRes(payEventsData) ?? []
+ const payEvents = React.useMemo(
+ () => transformPayEventsRes(payEventsData) ?? [],
+ [payEventsData],
+ )
return (
@@ -62,15 +62,10 @@ export function V4ActivityList() {
header={t`Paid`}
subject={
-
-
- }
- extra={
-
- bought {event.beneficiaryTokenCount?.format(6)}{' '}
- {token.data?.symbol ?? 'tokens'}
+
}
+ extra={ }
/>
)
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/models/ActivityEvents.ts b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/models/ActivityEvents.ts
index f960855f98..3e3c623afb 100644
--- a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/models/ActivityEvents.ts
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/models/ActivityEvents.ts
@@ -7,6 +7,7 @@ export type PayEvent = {
amountUSD: Ether | undefined
beneficiary: Address
beneficiaryTokenCount?: JBProjectToken
+ note: string
timestamp: number
txHash: string
}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/utils/transformEventsData.ts b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/utils/transformEventsData.ts
index 243a00239d..4083ca7de4 100644
--- a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/utils/transformEventsData.ts
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ActivityPanel/utils/transformEventsData.ts
@@ -9,11 +9,14 @@ export function transformPayEventsRes(
return {
id: event.id,
amount: new Ether(BigInt(event.amount)),
- amountUSD: event.amountUSD ? new Ether(BigInt(event.amountUSD)) : undefined,
+ amountUSD: event.amountUSD
+ ? new Ether(BigInt(event.amountUSD))
+ : undefined,
beneficiary: event.beneficiary,
beneficiaryTokenCount: new JBProjectToken(
BigInt(event.beneficiaryTokenCount),
),
+ note: event.note,
timestamp: event.timestamp,
txHash: event.txHash,
}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/AddNftButton.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/AddNftButton.tsx
new file mode 100644
index 0000000000..8bac6cf1af
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/AddNftButton.tsx
@@ -0,0 +1,25 @@
+import { PlusIcon } from '@heroicons/react/24/solid'
+import { Trans } from '@lingui/macro'
+import { stopPropagation } from 'react-stop-propagation'
+import { twMerge } from 'tailwind-merge'
+
+export const nftHoverButtonClasses =
+ 'absolute bottom-0 flex h-12 w-full items-center justify-center rounded-b-lg text-base font-medium text-white opacity-0 transition-all duration-200 ease-in-out group-hover:opacity-100'
+
+// Button that appears when hovering an NFT reward card
+export function AddNftButton({ onClick }: { onClick: VoidFunction }) {
+ return (
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftDetails.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftDetails.tsx
new file mode 100644
index 0000000000..053d5d1fdd
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftDetails.tsx
@@ -0,0 +1,67 @@
+import { Skeleton } from 'antd'
+import { TruncatedText } from 'components/TruncatedText'
+import ETHAmount from 'components/currency/ETHAmount'
+import { NftRewardTier } from 'models/nftRewards'
+import { twMerge } from 'tailwind-merge'
+import { parseWad } from 'utils/format/formatNumber'
+
+export function NftDetails({
+ rewardTier,
+ loading,
+ hideAttributes,
+ remainingSupplyText,
+}: {
+ rewardTier: NftRewardTier | undefined
+ loading: boolean | undefined
+ hideAttributes?: boolean
+ remainingSupplyText: string
+}) {
+ return (
+
+
+
+
+ {!hideAttributes ? (
+
+ {rewardTier?.contributionFloor ? (
+
+
+
+
+
+ ) : null}
+
+
+ {remainingSupplyText}
+
+
+
+ ) : null}
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftReward.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftReward.tsx
new file mode 100644
index 0000000000..e17b50d006
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftReward.tsx
@@ -0,0 +1,157 @@
+import { t } from '@lingui/macro'
+import { Tooltip } from 'antd'
+import { NftPreview } from 'components/NftRewards/NftPreview'
+import { NftRewardTier } from 'models/nftRewards'
+import { DEFAULT_NFT_MAX_SUPPLY } from 'packages/v2v3/constants/nftRewards'
+import { usePayProjectDisabled } from 'packages/v2v3/hooks/usePayProjectDisabled'
+import { useProjectSelector } from 'packages/v4/components/ProjectDashboard/redux/hooks'
+import { useNftRewardsEnabledForPay } from 'packages/v4/hooks/useNftRewardsEnabledForPay'
+import { useMemo, useState } from 'react'
+import { twMerge } from 'tailwind-merge'
+import { ipfsUriToGatewayUrl } from 'utils/ipfs'
+import { AddNftButton } from './AddNftButton'
+import { NftDetails } from './NftDetails'
+import { NftThumbnail } from './NftThumbnail'
+import { PreviewAddRemoveNftButton } from './PreviewAddRemoveNftButton'
+import { RemoveNftButton } from './RemoveNftButton'
+
+type NftRewardProps = {
+ className?: string
+ rewardTier?: NftRewardTier
+ loading?: boolean
+ onSelect: (quantity?: number) => void
+ onDeselect: VoidFunction
+ previewDisabled?: boolean
+ hideAttributes?: boolean
+}
+
+export function NftReward({
+ className,
+ loading,
+ rewardTier,
+ previewDisabled,
+ onSelect,
+ onDeselect,
+ hideAttributes,
+}: NftRewardProps) {
+ const [previewVisible, setPreviewVisible] = useState(false)
+ const chosenNftRewards = useProjectSelector(
+ state => state.projectCart.chosenNftRewards,
+ )
+
+ const nftsEnabledForPay = useNftRewardsEnabledForPay()
+ const {
+ payDisabled,
+ message,
+ loading: payDisabledLoading,
+ } = usePayProjectDisabled()
+
+ const quantitySelected = useMemo(
+ () =>
+ chosenNftRewards.find(nft => nft.id === rewardTier?.id)?.quantity ?? 0,
+ [chosenNftRewards, rewardTier?.id],
+ )
+ const isSelected = quantitySelected > 0
+
+ const fileUrl = useMemo(
+ () =>
+ rewardTier?.fileUrl ? ipfsUriToGatewayUrl(rewardTier.fileUrl) : undefined,
+ [rewardTier?.fileUrl],
+ )
+
+ const remainingSupply = rewardTier?.remainingSupply
+ const hasRemainingSupply = remainingSupply && remainingSupply > 0
+ const remainingSupplyText = !hasRemainingSupply
+ ? t`SOLD OUT`
+ : rewardTier.maxSupply === DEFAULT_NFT_MAX_SUPPLY
+ ? t`Unlimited`
+ : t`${rewardTier?.remainingSupply} remaining`
+
+ const disabled = Boolean(
+ !hasRemainingSupply || !nftsEnabledForPay || payDisabled,
+ )
+ const disabledReason = useMemo(() => {
+ if (!hasRemainingSupply) return t`Sold out`
+ if (!nftsEnabledForPay) return t`NFTs are not enabled for pay`
+ if (payDisabled) return message
+ }, [nftsEnabledForPay, hasRemainingSupply, payDisabled, message])
+
+ const openPreview = () => {
+ setPreviewVisible(true)
+ }
+
+ return (
+ <>
+
+
+
+
+ {!disabled &&
+ (isSelected ? (
+
onDeselect()} />
+ ) : (
+ onSelect(1)} />
+ ))}
+
+
+
+ {rewardTier && !previewDisabled && previewVisible ? (
+ onSelect(1)}
+ onDeselect={onDeselect}
+ isSelected={isSelected}
+ />
+ }
+ />
+ ) : null}
+ >
+ )
+}
+
+export const NftRewardSkeleton = () => (
+
+)
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftThumbnail.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftThumbnail.tsx
new file mode 100644
index 0000000000..4a26f9d577
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/NftThumbnail.tsx
@@ -0,0 +1,32 @@
+import { JuiceVideoThumbnailOrImage } from 'components/JuiceVideo/JuiceVideoThumbnailOrImage'
+import { NftRewardTier } from 'models/nftRewards'
+import { twMerge } from 'tailwind-merge'
+
+export function NftThumbnail({
+ fileUrl,
+ isSelected,
+ rewardTier,
+}: {
+ fileUrl: string | undefined
+ isSelected: boolean
+ rewardTier: NftRewardTier | undefined
+}) {
+ return (
+
+ {fileUrl ? (
+
+ ) : null}
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/PreviewAddRemoveNftButton.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/PreviewAddRemoveNftButton.tsx
new file mode 100644
index 0000000000..dd3d5a37c9
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/PreviewAddRemoveNftButton.tsx
@@ -0,0 +1,58 @@
+import { MinusIcon, PlusIcon } from '@heroicons/react/24/solid'
+import { Trans, t } from '@lingui/macro'
+import { Button } from 'antd'
+import { emitConfirmationDeletionModal } from 'hooks/emitConfirmationDeletionModal'
+import { useCallback } from 'react'
+import { twMerge } from 'tailwind-merge'
+
+const iconClasses = 'mr-1 h-6 w-6'
+const containerClasses = 'flex items-center justify-center'
+
+export function PreviewAddRemoveNftButton({
+ className,
+ isSelected,
+ onSelect,
+ onDeselect,
+}: {
+ className?: string
+ isSelected?: boolean
+ onSelect: VoidFunction
+ onDeselect: VoidFunction
+}) {
+ const buttonContents = isSelected ? (
+
+
+
+ Remove NFT
+
+
+ ) : (
+
+ )
+
+ const handleDeselect = useCallback(() => {
+ emitConfirmationDeletionModal({
+ onConfirm: onDeselect,
+ title: t`Remove NFT`,
+ description: t`Are you sure you want to remove this NFT?`,
+ })
+ }, [onDeselect])
+
+ return (
+
+ {buttonContents}
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/RemoveNftButton.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/RemoveNftButton.tsx
new file mode 100644
index 0000000000..b218c69098
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/NftReward/RemoveNftButton.tsx
@@ -0,0 +1,33 @@
+import { TrashIcon } from '@heroicons/react/24/solid'
+import { Trans, t } from '@lingui/macro'
+import { emitConfirmationDeletionModal } from 'hooks/emitConfirmationDeletionModal'
+import { useCallback } from 'react'
+import { stopPropagation } from 'react-stop-propagation'
+import { twMerge } from 'tailwind-merge'
+import { nftHoverButtonClasses } from './AddNftButton'
+
+// Button that appears when hovering an NFT reward card
+export function RemoveNftButton({ onClick }: { onClick: VoidFunction }) {
+ const handleDeselect = useCallback(() => {
+ emitConfirmationDeletionModal({
+ onConfirm: onClick,
+ title: t`Remove NFT`,
+ description: t`Are you sure you want to remove this NFT?`,
+ })
+ }, [onClick])
+
+ return (
+
+
+
+ Remove NFT
+
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/NftCreditsSection.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/NftCreditsSection.tsx
new file mode 100644
index 0000000000..22f2b12fb4
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/NftCreditsSection.tsx
@@ -0,0 +1,26 @@
+import { Trans } from '@lingui/macro'
+import TooltipIcon from 'components/TooltipIcon'
+import ETHAmount from 'components/currency/ETHAmount'
+import { BigNumber } from 'ethers'
+
+export function NftCreditsSection({ credits }: { credits: bigint }) {
+ return (
+ <>
+
+ Your credits
+
+
+ {/* // TODO: make ETHAmount take BigInts */}
+ credits{' '}
+
+ You have NFT credits from previous payments. Select NFTs to mint
+ and use your credits.
+
+ }
+ />
+
+ >
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftTile.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftTile.tsx
new file mode 100644
index 0000000000..ba3881d138
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftTile.tsx
@@ -0,0 +1,32 @@
+import { Tooltip } from 'antd'
+import { JuiceVideoThumbnailOrImage } from 'components/JuiceVideo/JuiceVideoThumbnailOrImage'
+import { NftRewardTier } from 'models/nftRewards'
+import { useMemo } from 'react'
+import { pinataToGatewayUrl } from 'utils/ipfs'
+
+export function RedeemNftTile({
+ rewardTier,
+ tokenId,
+}: {
+ rewardTier: NftRewardTier | undefined
+ tokenId: string
+}) {
+ const _name = rewardTier?.name ?? `NFT ${tokenId}`
+ const fileUrl = useMemo(() => {
+ if (!rewardTier?.fileUrl) return
+ return pinataToGatewayUrl(rewardTier.fileUrl)
+ }, [rewardTier?.fileUrl])
+ return (
+
+
+
+
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftTiles.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftTiles.tsx
new file mode 100644
index 0000000000..a5cf4c8391
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftTiles.tsx
@@ -0,0 +1,34 @@
+import Loading from 'components/Loading'
+import { NfTsQuery } from 'generated/graphql'
+import { useJB721DelegateTokenToNftReward } from '../hooks/useJB721DelegateTokenToNftReward'
+import { RedeemNftTile } from './RedeemNftTile'
+
+function RedeemNftTileLoader({ nft }: { nft: NfTsQuery['nfts'][number] }) {
+ const tokenId = nft.tokenId.toHexString()
+ const _nft = {
+ ...nft,
+ tokenId,
+ }
+ const { data: rewardTier } = useJB721DelegateTokenToNftReward(_nft)
+ if (!rewardTier)
+ return (
+
+
+
+ )
+ return
+}
+
+export function RedeemNftTiles({
+ nftAccountBalance,
+}: {
+ nftAccountBalance: NfTsQuery | undefined
+}) {
+ return (
+
+ {nftAccountBalance?.nfts.map((nft, i) => (
+
+ ))}
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftsSection.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftsSection.tsx
new file mode 100644
index 0000000000..a436a4aead
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/RedeemNftsSection/RedeemNftsSection.tsx
@@ -0,0 +1,74 @@
+import { useV4UserNftCredits } from 'packages/v4/contexts/V4UserNftCreditsProvider'
+import { NftCreditsSection } from './NftCreditsSection'
+
+export function RedeemNftsSection() {
+ // const [redeemNftsModalVisible, setRedeemNftsModalVisible] = useState(false)
+
+ const nftCredits = useV4UserNftCredits()
+
+ // TODO: This needs to be implemented
+ // const { fundingCycleMetadata, primaryTerminalCurrentOverflow } =
+ // useContext(V2V3ProjectContext)
+ // const { data, loading } = useNftAccountBalance({
+ // accountAddress: userAddress,
+ // dataSourceAddress: fundingCycleMetadata?.dataSource,
+ // })
+ // const { data: credits, loading: loadingCredits } = useNftCredits(userAddress)
+
+ // const hasOverflow = primaryTerminalCurrentOverflow?.gt(0)
+ // const hasRedemptionRate = fundingCycleMetadata?.redemptionRate.gt(0)
+ // const canRedeem =
+ // hasOverflow &&
+ // hasRedemptionRate &&
+ // fundingCycleMetadata?.useDataSourceForRedeem
+
+ // const hasRedeemableNfts = (data?.nfts?.length ?? 0) > 0
+
+ // const showRedeemSection = !loading && hasRedeemableNfts && !!userAddress
+ const showCreditSection =
+ !nftCredits.isLoading && nftCredits.data && nftCredits.data > 0n
+
+ // if (!showRedeemSection && !showCreditSection) return null
+
+ return (
+
+ {showCreditSection ? (
+
+
+
+ ) : null}
+
+ {/* TODO: Redeem nft section */}
+ {/* {showRedeemSection ? (
+
+
+ Your NFTs
+
+
+
+
+
+ setRedeemNftsModalVisible(true)}
+ >
+ {canRedeem ? (
+ Redeem NFTs
+ ) : (
+ Burn NFTs
+ )}
+
+
+
+ {redeemNftsModalVisible && (
+
setRedeemNftsModalVisible(false)}
+ onConfirmed={() => setRedeemNftsModalVisible(false)}
+ />
+ )}
+
+ ) : null} */}
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/V4NftRewardsPanel.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/V4NftRewardsPanel.tsx
new file mode 100644
index 0000000000..70de871008
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/V4NftRewardsPanel.tsx
@@ -0,0 +1,47 @@
+import { Trans, t } from '@lingui/macro'
+import { EmptyScreen } from 'components/Project/ProjectTabs/EmptyScreen'
+import { NftReward, NftRewardSkeleton } from './NftReward/NftReward'
+import { RedeemNftsSection } from './RedeemNftsSection/RedeemNftsSection'
+import { useNftRewardsPanel } from './hooks/useNftRewardsPanel'
+
+export const V4NftRewardsPanel = () => {
+ const {
+ rewardTiers,
+ handleTierSelect,
+ handleTierDeselect,
+ loading: nftsLoading,
+ } = useNftRewardsPanel()
+
+ return (
+
+
+ NFTs
+
+
+
+ {!nftsLoading && rewardTiers?.length ? (
+
+ {rewardTiers?.map((tier, i) => (
+
+ handleTierSelect(tier.id, quantity)}
+ onDeselect={() => handleTierDeselect(tier.id)}
+ />
+
+ ))}
+
+ ) : nftsLoading ? (
+
+ {[...Array(6)].map((_, i) => (
+
+ ))}
+
+ ) : (
+
+ )}
+
+ )
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/hooks/useJB721DelegateTokenToNftReward.ts b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/hooks/useJB721DelegateTokenToNftReward.ts
new file mode 100644
index 0000000000..26542c7ece
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/hooks/useJB721DelegateTokenToNftReward.ts
@@ -0,0 +1,37 @@
+import { Nft } from 'generated/graphql'
+import { NftRewardTier } from 'models/nftRewards'
+import {
+ NFT_METADATA_CONTRIBUTION_FLOOR_ATTRIBUTES_INDEX,
+ useJB721DelegateTokenMetadata,
+} from 'packages/v2v3/components/V2V3Project/ManageNftsSection/RedeemNftsModal/RedeemNftCard'
+
+export type RedeemingNft = Pick & {
+ tokenId: string
+}
+
+export function useJB721DelegateTokenToNftReward(nft: RedeemingNft): {
+ data: NftRewardTier | undefined
+} {
+ const { data: tierData } = useJB721DelegateTokenMetadata(nft.tokenUri)
+ const contributionFloor =
+ tierData?.attributes[NFT_METADATA_CONTRIBUTION_FLOOR_ATTRIBUTES_INDEX]
+ .value ?? 0
+
+ return {
+ data: tierData
+ ? {
+ name: tierData.name,
+ contributionFloor,
+ id: parseInt(nft.tokenId),
+ remainingSupply: undefined,
+ maxSupply: undefined,
+ fileUrl: tierData.image,
+ externalLink: undefined,
+ description: undefined,
+ beneficiary: undefined,
+ reservedRate: undefined,
+ votingWeight: undefined,
+ }
+ : undefined,
+ }
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/hooks/useNftRewardsPanel.ts b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/hooks/useNftRewardsPanel.ts
new file mode 100644
index 0000000000..5184ab5a59
--- /dev/null
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4NftRewardsPanel/hooks/useNftRewardsPanel.ts
@@ -0,0 +1,36 @@
+import { useProjectDispatch } from 'packages/v4/components/ProjectDashboard/redux/hooks'
+import { payRedeemActions } from 'packages/v4/components/ProjectDashboard/redux/payRedeemSlice'
+import { projectCartActions } from 'packages/v4/components/ProjectDashboard/redux/projectCartSlice'
+import { V4NftRewardsContext } from 'packages/v4/contexts/V4NftRewards/V4NftRewardsProvider'
+import { useCallback, useContext } from 'react'
+
+export const useNftRewardsPanel = () => {
+ const dispatch = useProjectDispatch()
+ const {
+ nftRewards: { rewardTiers },
+ loading,
+ } = useContext(V4NftRewardsContext)
+
+ const handleTierSelect = useCallback(
+ (tierId: number, quantity: number) => {
+ dispatch(payRedeemActions.changeToPay())
+ dispatch(projectCartActions.upsertNftReward({ id: tierId, quantity }))
+ },
+ [dispatch],
+ )
+
+ const handleTierDeselect = useCallback(
+ (tierId: number) => {
+ dispatch(payRedeemActions.changeToPay())
+ dispatch(projectCartActions.removeNftReward({ id: tierId }))
+ },
+ [dispatch],
+ )
+
+ return {
+ rewardTiers,
+ loading,
+ handleTierSelect,
+ handleTierDeselect,
+ }
+}
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ProjectTabs.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ProjectTabs.tsx
index cc99f1436c..0f0534b016 100644
--- a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ProjectTabs.tsx
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4ProjectTabs.tsx
@@ -1,14 +1,14 @@
-import { Fragment, useEffect, useMemo, useRef, useState } from 'react'
-
import { Tab } from '@headlessui/react'
import { t } from '@lingui/macro'
import { ProjectTab } from 'components/Project/ProjectTabs/ProjectTab'
import { useOnScreen } from 'hooks/useOnScreen'
+import { Fragment, useEffect, useMemo, useRef, useState } from 'react'
import { twMerge } from 'tailwind-merge'
import { useProjectPageQueries } from '../hooks/useProjectPageQueries'
import V4AboutPanel from './V4AboutPanel'
import { V4ActivityPanel } from './V4ActivityPanel/V4ActivityPanel'
import { V4CyclesPayoutsPanel } from './V4CyclesPayoutsPanel/V4CyclesPayoutsPanel'
+import { V4NftRewardsPanel } from './V4NftRewardsPanel/V4NftRewardsPanel'
import { V4TokensPanel } from './V4TokensPanel/V4TokensPanel'
type ProjectTabConfig = {
@@ -21,6 +21,8 @@ type ProjectTabConfig = {
export const V4ProjectTabs = ({ className }: { className?: string }) => {
const { projectPageTab, setProjectPageTab } = useProjectPageQueries()
+ const showNftRewards = true
+
const containerRef = useRef(null)
const panelRef = useRef(null)
const isPanelVisible = useOnScreen(panelRef)
@@ -48,12 +50,12 @@ export const V4ProjectTabs = ({ className }: { className?: string }) => {
() => [
{ id: 'activity', name: t`Activity`, panel: },
{ id: 'about', name: t`About`, panel: },
- // {
- // id: 'nft_rewards',
- // name: t`NFTs`,
- // panel: ,
- // hideTab: !showNftRewards,
- // },
+ {
+ id: 'nft_rewards',
+ name: t`NFTs`,
+ panel: ,
+ hideTab: !showNftRewards,
+ },
{
id: 'cycle_payouts',
name: t`Cycles & Payouts`,
@@ -61,7 +63,7 @@ export const V4ProjectTabs = ({ className }: { className?: string }) => {
},
{ id: 'tokens', name: t`Tokens`, panel: },
],
- [],
+ [showNftRewards],
)
const selectedTabIndex = useMemo(() => {
diff --git a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4TokensPanel/V4TokensPanel.tsx b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4TokensPanel/V4TokensPanel.tsx
index de2f7b7234..65281c7549 100644
--- a/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4TokensPanel/V4TokensPanel.tsx
+++ b/src/packages/v4/views/V4ProjectDashboard/V4ProjectTabs/V4TokensPanel/V4TokensPanel.tsx
@@ -13,6 +13,7 @@ import { AddTokenToMetamaskButton } from 'components/buttons/AddTokenToMetamaskB
import { ISSUE_ERC20_EXPLANATION } from 'components/strings'
import { useJBContractContext } from 'juice-sdk-react'
import { V4TokenHoldersModal } from 'packages/v4/components/modals/V4TokenHoldersModal/V4TokenHoldersModal'
+import { useProjectHasErc20Token } from 'packages/v4/hooks/useProjectHasErc20Token'
import { v4ProjectRoute } from 'packages/v4/utils/routes'
import { useCallback, useState } from 'react'
import { reloadWindow } from 'utils/windowUtils'
@@ -34,6 +35,7 @@ export const V4TokensPanel = () => {
projectToken,
totalSupply,
} = useV4TokensPanel()
+ const projectHasErc20Token = useProjectHasErc20Token()
const { canMintTokens } = useV4BalanceMenuItemsUserFlags()
@@ -76,7 +78,7 @@ export const V4TokensPanel = () => {
{userTokenBalance.format(8)} tokens
- {/* {projectHasErc20Token && (
+ {projectHasErc20Token && (
{
>
Claim ERC-20 token
- )} */}
+ )}
{/* {
const chainId = useChainId()
const { projectId: projectIdBig } = useJBContractContext()
const projectId = Number(projectIdBig)
-
+
const {
projectToken,
projectTokenAddress,
@@ -203,25 +205,26 @@ const ProjectTokenCard = () => {
{projectTokenAddress && projectHasErc20Token && (
)}
{canCreateErc20Token ? (
-
- }
- type='link'
- >
+
+ } type="link">
Create ERC-20 Token
- ): null}
+ ) : null}
>
}
/>
diff --git a/src/packages/v4/views/V4ProjectSettings/ProjectSettingsDashboard.tsx b/src/packages/v4/views/V4ProjectSettings/ProjectSettingsDashboard.tsx
index 67382b8867..5c5e9e016d 100644
--- a/src/packages/v4/views/V4ProjectSettings/ProjectSettingsDashboard.tsx
+++ b/src/packages/v4/views/V4ProjectSettings/ProjectSettingsDashboard.tsx
@@ -2,8 +2,13 @@ import { Trans } from '@lingui/macro'
import { Button } from 'antd'
import EthereumAddress from 'components/EthereumAddress'
import Loading from 'components/Loading'
-import { NativeTokenValue, useJBContractContext, useJBProjectMetadataContext } from 'juice-sdk-react'
+import {
+ NativeTokenValue,
+ useJBContractContext,
+ useJBProjectMetadataContext,
+} from 'juice-sdk-react'
import Link from 'next/link'
+import { useProjectHasErc20Token } from 'packages/v4/hooks/useProjectHasErc20Token'
import { useV4BalanceOfNativeTerminal } from 'packages/v4/hooks/useV4BalanceOfNativeTerminal'
import useProjectOwnerOf from 'packages/v4/hooks/useV4ProjectOwnerOf'
import { useV4WalletHasPermission } from 'packages/v4/hooks/useV4WalletHasPermission'
@@ -58,7 +63,7 @@ export function ProjectSettingsDashboard() {
const { metadata } = useJBProjectMetadataContext()
const { distributableAmount } = useV4DistributableAmount()
- const projectHasErc20Token = false // @v4TODO
+ const projectHasErc20Token = useProjectHasErc20Token()
const hasIssueTicketsPermission = useV4WalletHasPermission(
V4OperatorPermission.MINT_TOKENS,
)
@@ -113,9 +118,7 @@ export function ProjectSettingsDashboard() {
{!loading ? (
-
+
) : (
)}
diff --git a/src/pages/v4/[chainName]/p/[projectId]/index.tsx b/src/pages/v4/[chainName]/p/[projectId]/index.tsx
index 0bf163c12e..07a3dd836b 100644
--- a/src/pages/v4/[chainName]/p/[projectId]/index.tsx
+++ b/src/pages/v4/[chainName]/p/[projectId]/index.tsx
@@ -5,7 +5,9 @@ import { JBChainId, JBProjectProvider } from 'juice-sdk-react'
import { useRouter } from 'next/router'
import { ReduxProjectCartProvider } from 'packages/v4/components/ProjectDashboard/ReduxProjectCartProvider'
import store from 'packages/v4/components/ProjectDashboard/redux/store'
+import { V4NftRewardsProvider } from 'packages/v4/contexts/V4NftRewards/V4NftRewardsProvider'
import V4ProjectMetadataProvider from 'packages/v4/contexts/V4ProjectMetadataProvider'
+import { V4UserNftCreditsProvider } from 'packages/v4/contexts/V4UserNftCreditsProvider'
import { useCurrentRouteChainId } from 'packages/v4/hooks/useCurrentRouteChainId'
import { V4ProjectDashboard } from 'packages/v4/views/V4ProjectDashboard/V4ProjectDashboard'
import { wagmiConfig } from 'packages/v4/wagmiConfig'
@@ -71,7 +73,13 @@ const Providers: React.FC<
>
- {children}
+
+
+
+ {children}
+
+
+
diff --git a/src/utils/nftRewards.ts b/src/utils/nftRewards.ts
index 628d98d2ee..6ba1ee9218 100644
--- a/src/utils/nftRewards.ts
+++ b/src/utils/nftRewards.ts
@@ -23,6 +23,7 @@ import {
import { DEFAULT_NFT_MAX_SUPPLY } from 'packages/v2v3/constants/nftRewards'
import { JB721DelegateVersion } from 'packages/v2v3/models/contracts'
import { V2V3CurrencyOption } from 'packages/v2v3/models/currencyOption'
+import { JB721TierV4 } from 'packages/v4/contexts/V4NftRewards/V4NftRewardsProvider'
import { decodeEncodedIpfsUri, encodeIpfsUri, ipfsUri } from 'utils/ipfs'
export function sortNftsByContributionFloor(
@@ -67,7 +68,11 @@ export function getNftRewardOfFloor({
// returns an array of CIDs from a given array of RewardTier obj's
export function CIDsOfNftRewardTiersResponse(
- nftRewardTiersResponse: JB721TierV3[] | JB_721_TIER_V3_2[] | undefined,
+ nftRewardTiersResponse:
+ | JB721TierV3[]
+ | JB_721_TIER_V3_2[]
+ | readonly JB721TierV4[]
+ | undefined,
): string[] {
const cids =
nftRewardTiersResponse