diff --git a/.changeset/rotten-kangaroos-train.md b/.changeset/rotten-kangaroos-train.md new file mode 100644 index 000000000..e16b9748f --- /dev/null +++ b/.changeset/rotten-kangaroos-train.md @@ -0,0 +1,5 @@ +--- +'@siafoundation/design-system': patch +--- + +Add use client / RSC support to Table. diff --git a/.changeset/stupid-fishes-brake.md b/.changeset/stupid-fishes-brake.md new file mode 100644 index 000000000..1c284bb10 --- /dev/null +++ b/.changeset/stupid-fishes-brake.md @@ -0,0 +1,5 @@ +--- +'explorer': patch +--- + +Add detailed labels to contract proof outputs explaining what each output is for. diff --git a/.changeset/weak-tips-raise.md b/.changeset/weak-tips-raise.md new file mode 100644 index 000000000..bd20530a0 --- /dev/null +++ b/.changeset/weak-tips-raise.md @@ -0,0 +1,5 @@ +--- +'@siafoundation/design-system': patch +--- + +Fix flex layout and add a title tooltip to EntityListItem. diff --git a/apps/explorer/components/Contract/index.tsx b/apps/explorer/components/Contract/index.tsx index 846ca20fa..6bdfc04c8 100644 --- a/apps/explorer/components/Contract/index.tsx +++ b/apps/explorer/components/Contract/index.tsx @@ -5,6 +5,7 @@ import BigNumber from 'bignumber.js' import { SiaCentralContract, SiaCentralExchangeRates, + SiaCentralPartialSiacoinOutput, } from '@siafoundation/sia-central' import { humanBytes, humanDate, humanSiacoin } from '@siafoundation/units' import { EntityList, EntityListItemProps } from '@siafoundation/design-system' @@ -148,34 +149,90 @@ export function Contract({ contract, rates, renewedFrom, renewedTo }: Props) { ] as DatumProps[] }, [contract, exchange]) - const missedProofOutputs = useMemo(() => { - if (!contract) { - return [] + const validProofOutputs = useMemo(() => { + if (isProperlyFormedNewContract(contract)) { + const { renterPayoutValid, hostPayoutValid } = + getNewContractFormattedOutputs(contract) + return [ + { + label: 'renter payout: remaining renter allowance', + initials: 'r', + sc: new BigNumber(renterPayoutValid.value), + hash: renterPayoutValid.output_id, + }, + { + label: `host payout`, + initials: 'h', + sc: new BigNumber(hostPayoutValid.value), + hash: hostPayoutValid.output_id, + }, + ] as EntityListItemProps[] + } + if (isProperlyFormedRenewedContract(contract)) { + const { renterPayoutValid, hostPayoutValid } = + getRenewedContractFormattedOutputs(contract) + return [ + { + label: 'renter payout: remaining renter allowance', + initials: 'r', + sc: new BigNumber(renterPayoutValid.value), + hash: renterPayoutValid.output_id, + }, + { + label: `host payout: payout minus risked collateral and storage revenue`, + initials: 'h', + sc: new BigNumber(hostPayoutValid.value), + hash: hostPayoutValid.output_id, + }, + ] as EntityListItemProps[] } - const list: EntityListItemProps[] = [] - contract.missed_proof_outputs?.forEach((o) => { - list.push({ - label: o.source ? o.source.replace(/_/g, ' ') : 'output', - sc: new BigNumber(o.value), - hash: o.output_id, - }) - }) - return list + return contract?.valid_proof_outputs?.map(genericOutputListItem) || [] }, [contract]) - const validProofOutputs = useMemo(() => { - if (!contract) { - return [] + const missedProofOutputs = useMemo(() => { + if (isProperlyFormedNewContract(contract)) { + const { renterPayoutMissed, hostPayoutMissed, hostBurned } = + getNewContractFormattedOutputs(contract) + return [ + { + label: 'renter payout: remaining renter allowance', + initials: 'r', + sc: new BigNumber(renterPayoutMissed.value), + hash: renterPayoutMissed.output_id, + }, + { + label: `host payout: payout minus risked collateral and storage revenue`, + initials: 'h', + sc: new BigNumber(hostPayoutMissed.value), + hash: hostPayoutMissed.output_id, + }, + { + label: 'host burn: host revenue plus risked collateral', + initials: 'b', + sc: new BigNumber(hostBurned.value), + hash: hostBurned.output_id, + }, + ] as EntityListItemProps[] + } + if (isProperlyFormedRenewedContract(contract)) { + const { renterPayoutMissed, hostPayoutMissed } = + getRenewedContractFormattedOutputs(contract) + return [ + { + label: 'renter payout: remaining renter allowance', + initials: 'r', + sc: new BigNumber(renterPayoutMissed.value), + hash: renterPayoutMissed.output_id, + }, + { + label: `host payout: payout minus risked collateral and storage revenue`, + initials: 'h', + sc: new BigNumber(hostPayoutMissed.value), + hash: hostPayoutMissed.output_id, + }, + ] as EntityListItemProps[] } - const list: EntityListItemProps[] = [] - contract.valid_proof_outputs?.forEach((o) => { - list.push({ - label: o.source ? o.source.replace(/_/g, ' ') : 'output', - sc: new BigNumber(o.value), - hash: o.output_id, - }) - }) - return list + return contract?.missed_proof_outputs?.map(genericOutputListItem) || [] }, [contract]) return ( @@ -216,3 +273,93 @@ export function Contract({ contract, rates, renewedFrom, renewedTo }: Props) { ) } + +function isProperlyFormedNewContract(contract: SiaCentralContract) { + // renter payout, host payout + if (contract.valid_proof_outputs?.length !== 2) { + return false + } + // renter payout, host payout, and host burned + if (contract.missed_proof_outputs?.length !== 3) { + return false + } + + const { renterPayoutValid, renterPayoutMissed } = + getNewContractFormattedOutputs(contract) + + // renter payout valid and missed should be the same + if (renterPayoutValid.value !== renterPayoutMissed.value) { + return false + } + + // math.MaxUint64 with lost precision + const mathMaxUint64 = 18446744073709552000 + if (contract.revision_number >= mathMaxUint64) { + return false + } + return true +} + +function isProperlyFormedRenewedContract(contract: SiaCentralContract) { + // renter payout, host payout + if (contract.valid_proof_outputs?.length !== 2) { + return false + } + // renter payout, host payout + if (contract.missed_proof_outputs?.length !== 2) { + return false + } + + const { + renterPayoutValid, + renterPayoutMissed, + hostPayoutValid, + hostPayoutMissed, + } = getRenewedContractFormattedOutputs(contract) + + // renter payout valid and missed should be the same + if (renterPayoutValid.value !== renterPayoutMissed.value) { + return false + } + + // host payout valid and missed should be the same + if (hostPayoutValid.value !== hostPayoutMissed.value) { + return false + } + + // math.MaxUint64 with lost precision + const mathMaxUint64 = 18446744073709552000 + if (contract.revision_number !== mathMaxUint64) { + return false + } + return true +} + +function getNewContractFormattedOutputs(contract: SiaCentralContract) { + return { + renterPayoutValid: contract.valid_proof_outputs[0], + renterPayoutMissed: contract.missed_proof_outputs[0], + hostPayoutValid: contract.valid_proof_outputs[1], + hostPayoutMissed: contract.missed_proof_outputs[1], + hostBurned: contract.missed_proof_outputs[2], + } +} + +function getRenewedContractFormattedOutputs(contract: SiaCentralContract) { + return { + renterPayoutValid: contract.valid_proof_outputs[0], + renterPayoutMissed: contract.missed_proof_outputs[0], + hostPayoutValid: contract.valid_proof_outputs[1], + hostPayoutMissed: contract.missed_proof_outputs[1], + } +} + +function genericOutputListItem( + o: SiaCentralPartialSiacoinOutput +): EntityListItemProps { + return { + label: o.source ? o.source.replace(/_/g, ' ') : 'output', + sc: new BigNumber(o.value), + hash: o.output_id, + } +} diff --git a/apps/explorer/components/ExplorerDatum.tsx b/apps/explorer/components/ExplorerDatum.tsx index 78454266b..78220b2f7 100644 --- a/apps/explorer/components/ExplorerDatum.tsx +++ b/apps/explorer/components/ExplorerDatum.tsx @@ -32,7 +32,7 @@ export function ExplorerDatum({ comment, }: DatumProps) { return ( -
+
{upperFirst(label)} diff --git a/libs/design-system/src/components/EntityListItem.tsx b/libs/design-system/src/components/EntityListItem.tsx index 924b37cbd..8df484f8f 100644 --- a/libs/design-system/src/components/EntityListItem.tsx +++ b/libs/design-system/src/components/EntityListItem.tsx @@ -17,6 +17,7 @@ import BigNumber from 'bignumber.js' import { DotMark16 } from '@siafoundation/react-icons' import { EntityListItemLayout } from './EntityListItemLayout' import { ValueScFiat } from './ValueScFiat' +import { Tooltip } from '../core/Tooltip' export type EntityListItemProps = { label?: string @@ -62,9 +63,9 @@ export function EntityListItem(entity: EntityListItemProps) { const title = isValidUrl(label) ? label : upperFirst(label) return ( -
+
-
+
{entity.height && entity.blockHref && ( @@ -72,11 +73,23 @@ export function EntityListItem(entity: EntityListItemProps) { )} - {title || truncHashEl} + {title ? ( + + + {title} + + + ) : ( + + {truncHashEl} + + )}
- {!!sc && } - {!!sf && } +
+ {!!sc && } + {!!sf && } +
{!!title && truncHashEl}
diff --git a/libs/design-system/src/components/Table/TableRow.tsx b/libs/design-system/src/components/Table/TableRow.tsx index 527cf0a67..2ff5775f5 100644 --- a/libs/design-system/src/components/Table/TableRow.tsx +++ b/libs/design-system/src/components/Table/TableRow.tsx @@ -1,3 +1,5 @@ +'use client' + import { CSSProperties, forwardRef, useMemo } from 'react' import { cx } from 'class-variance-authority' import { useDroppable, useDraggable } from '@dnd-kit/core' diff --git a/libs/design-system/src/components/Table/index.tsx b/libs/design-system/src/components/Table/index.tsx index d873ad301..7cf496dfa 100644 --- a/libs/design-system/src/components/Table/index.tsx +++ b/libs/design-system/src/components/Table/index.tsx @@ -1,3 +1,5 @@ +'use client' + import { Tooltip } from '../../core/Tooltip' import { Panel } from '../../core/Panel' import { Text } from '../../core/Text'