Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(wallet-dashboard): add staking confirmation screen #4034

Closed
wants to merge 48 commits into from
Closed
Show file tree
Hide file tree
Changes from 44 commits
Commits
Show all changes
48 commits
Select commit Hold shift + click to select a range
c925a99
feat(tooling-dashboard): style selected stake
panteleymonchuk Oct 31, 2024
1d8887e
feat(tooling-dashboard): add data to stake details page
panteleymonchuk Oct 31, 2024
a852fab
feat(wallet-dashboard): remove extra memo.
panteleymonchuk Nov 1, 2024
8325b36
Merge remote-tracking branch 'origin/develop' into tooling-dashboard/…
panteleymonchuk Nov 7, 2024
388058c
feat(staking): refactor StakeDialog. Add new Layout component.
panteleymonchuk Nov 7, 2024
d8d03ac
feat(tooling-core): add clsx dependency to package.json and update pn…
panteleymonchuk Nov 7, 2024
1ff40ea
Merge remote-tracking branch 'origin/develop' into tooling-dashboard/…
panteleymonchuk Nov 7, 2024
2c1adc9
Merge remote-tracking branch 'origin/tooling-epic/dashboard-styling' …
panteleymonchuk Nov 13, 2024
4d93394
feat(wallet-dashboard): manage view for dialog outside.
panteleymonchuk Nov 13, 2024
8c17341
feat(wallet-dashboard): join changes from PR 3854
panteleymonchuk Nov 13, 2024
7f66cd0
feat(wallet-dashboard): join enter amount screen from PR 3874
panteleymonchuk Nov 13, 2024
5932ed5
feat(tooling-core): move validation schema
panteleymonchuk Nov 14, 2024
1507c81
feat(wallet-dashboard): add staking confirmation screen
VmMad Nov 14, 2024
9bdb887
feat(wallet-dashboard): integrate Formik
panteleymonchuk Nov 14, 2024
499e4a6
fix: build error
VmMad Nov 14, 2024
14a2786
feat(wallet-dashboard): enhance StakeDialog and EnterAmountView with …
panteleymonchuk Nov 14, 2024
c1341dc
feat(wallet-dashboard): update StakeDialog to support selectedValidat…
panteleymonchuk Nov 14, 2024
c4329f4
feat(wallet-dashboard): refactor StakedInfo and Validator components …
panteleymonchuk Nov 14, 2024
3a4ebb7
feat(wallet-dashboard): move useStakeTxnInfo hook to the core
panteleymonchuk Nov 15, 2024
3818f2c
Merge branch 'tooling-dashboard/style-selected-stake' into tooling-da…
VmMad Nov 15, 2024
54b1b1f
fix(tooling-core): downgrade bignumber.js to 9.1.1 and yup to 1.1.1
panteleymonchuk Nov 15, 2024
4777432
feat: add new layout
VmMad Nov 15, 2024
b13ac91
Merge branch 'tooling-dashboard/style-selected-stake' into tooling-da…
VmMad Nov 15, 2024
8242185
refactor(tooling-dashboard): change export to default and clean up co…
panteleymonchuk Nov 15, 2024
cc8b8c6
Merge branch 'tooling-dashboard/style-selected-stake' into tooling-da…
VmMad Nov 18, 2024
d5ee9a1
fix: import
VmMad Nov 18, 2024
47cbd98
refactor: rename button and view
VmMad Nov 19, 2024
371b2fd
Merge branch 'tooling-epic/dashboard-styling' into tooling-dashboard/…
brancoder Nov 19, 2024
be3b938
refactor(wallet-dashboard): simplify validator info retrieval and add…
panteleymonchuk Nov 19, 2024
80049ac
refactor(wallet-dashboard): streamline stake calculations and integra…
panteleymonchuk Nov 19, 2024
24144b3
Merge remote-tracking branch 'origin/tooling-epic/dashboard-styling' …
panteleymonchuk Nov 20, 2024
0cfd637
refactor(wallet, core): update import paths for consistency and clarity
panteleymonchuk Nov 20, 2024
dc99beb
refactor(wallet-dashboard): integrate FormikProvider. Polish interfaces.
panteleymonchuk Nov 20, 2024
8b2df8f
Merge branch 'tooling-dashboard/style-selected-stake' into tooling-da…
VmMad Nov 21, 2024
873a5aa
refactor: update component name
VmMad Nov 21, 2024
a6e664b
Merge remote-tracking branch 'origin/develop' into tooling-dashboard/…
panteleymonchuk Nov 21, 2024
2afa4d8
Merge branch 'develop' into tooling-dashboard/style-selected-stake
brancoder Nov 22, 2024
796bf46
refactor: improve props
VmMad Nov 22, 2024
bcbdebd
feat(wallet-dashboard): refactor staking dialog management for home
panteleymonchuk Nov 22, 2024
b964e75
feat(wallet-dashboard): move constants to another folder.
panteleymonchuk Nov 22, 2024
c8bbc90
feat(wallet-dashboard): enhance transaction handling and improve user…
panteleymonchuk Nov 22, 2024
af7f7dc
fix(wallet-dashboard): update onBack handler for improved navigation
panteleymonchuk Nov 22, 2024
69bcf98
Merge branch 'develop' into tooling-dashboard/style-selected-stake
VmMad Nov 22, 2024
ce38b84
Merge branch 'tooling-dashboard/style-selected-stake' into tooling-da…
VmMad Nov 25, 2024
24ea08c
Merge branch 'develop' into tooling-dashboard/style-confirmation-screen
VmMad Nov 25, 2024
8c14776
fix: update coin image
VmMad Nov 25, 2024
9f7fe1d
refactor: remove debris
VmMad Nov 26, 2024
026ee47
Merge branch 'develop' into tooling-dashboard/style-confirmation-screen
VmMad Nov 26, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 1 addition & 2 deletions apps/core/src/components/coin/CoinIcon.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,9 @@
// SPDX-License-Identifier: Apache-2.0

import React from 'react';
import { useCoinMetadata } from '../../hooks';
import { useCoinMetadata, ImageIcon, ImageIconSize } from '../../index';
import { IotaLogoMark } from '@iota/ui-icons';
import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils';
import { ImageIcon, ImageIconSize } from '../icon';
import cx from 'clsx';

interface NonIotaCoinProps {
Expand Down
3 changes: 3 additions & 0 deletions apps/core/src/constants/staking.constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,3 +6,6 @@ export const UNSTAKING_REQUEST_EVENT = '0x3::validator::UnstakingRequestEvent';

export const DELEGATED_STAKES_QUERY_STALE_TIME = 10_000;
export const DELEGATED_STAKES_QUERY_REFETCH_INTERVAL = 30_000;

export const NUM_OF_EPOCH_BEFORE_STAKING_REWARDS_REDEEMABLE = 2;
export const NUM_OF_EPOCH_BEFORE_STAKING_REWARDS_STARTS = 1;
2 changes: 2 additions & 0 deletions apps/core/src/hooks/stake/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,3 +4,5 @@
export * from './useGetDelegatedStake';
export * from './useTotalDelegatedRewards';
export * from './useTotalDelegatedStake';
export * from './useValidatorInfo';
export * from './useStakeTxnInfo';
52 changes: 52 additions & 0 deletions apps/core/src/hooks/stake/useStakeTxnInfo.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
// Copyright (c) 2024 IOTA Stiftung
// SPDX-License-Identifier: Apache-2.0
import {
useGetTimeBeforeEpochNumber,
useTimeAgo,
TimeUnit,
NUM_OF_EPOCH_BEFORE_STAKING_REWARDS_REDEEMABLE,
NUM_OF_EPOCH_BEFORE_STAKING_REWARDS_STARTS,
} from '../../index';

export function useStakeTxnInfo(startEpoch?: string | number) {
const startEarningRewardsEpoch =
Number(startEpoch || 0) + NUM_OF_EPOCH_BEFORE_STAKING_REWARDS_STARTS;

const redeemableRewardsEpoch =
Number(startEpoch || 0) + NUM_OF_EPOCH_BEFORE_STAKING_REWARDS_REDEEMABLE;

const { data: timeBeforeStakeRewardsStarts } =
useGetTimeBeforeEpochNumber(startEarningRewardsEpoch);
const timeBeforeStakeRewardsStartsAgo = useTimeAgo({
timeFrom: timeBeforeStakeRewardsStarts,
shortedTimeLabel: false,
shouldEnd: true,
maxTimeUnit: TimeUnit.ONE_HOUR,
});
const stakedRewardsStartEpoch =
timeBeforeStakeRewardsStarts > 0
? `in ${timeBeforeStakeRewardsStartsAgo}`
: startEpoch
? `Epoch #${Number(startEarningRewardsEpoch)}`
: '--';

const { data: timeBeforeStakeRewardsRedeemable } =
useGetTimeBeforeEpochNumber(redeemableRewardsEpoch);
const timeBeforeStakeRewardsRedeemableAgo = useTimeAgo({
timeFrom: timeBeforeStakeRewardsRedeemable,
shortedTimeLabel: false,
shouldEnd: true,
maxTimeUnit: TimeUnit.ONE_HOUR,
});
const timeBeforeStakeRewardsRedeemableAgoDisplay =
timeBeforeStakeRewardsRedeemable > 0
? `in ${timeBeforeStakeRewardsRedeemableAgo}`
: startEpoch
? `Epoch #${Number(redeemableRewardsEpoch)}`
: '--';

return {
stakedRewardsStartEpoch,
timeBeforeStakeRewardsRedeemableAgoDisplay,
};
}
54 changes: 54 additions & 0 deletions apps/core/src/hooks/stake/useValidatorInfo.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
// Copyright (c) 2024 IOTA Stiftung
// SPDX-License-Identifier: Apache-2.0
import { useIotaClientQuery } from '@iota/dapp-kit';
import { useGetValidatorsApy } from '../';
import { getTotalValidatorStake } from '../../utils';

export function useValidatorInfo({ validatorAddress }: { validatorAddress: string }) {
const {
data: system,
isPending: isPendingValidators,
isError: errorValidators,
} = useIotaClientQuery('getLatestIotaSystemState');
const { data: rollingAverageApys } = useGetValidatorsApy();

const validatorSummary =
system?.activeValidators.find((validator) => validator.iotaAddress === validatorAddress) ||
null;

const currentEpoch = Number(system?.epoch || 0);

const totalValidatorStake = getTotalValidatorStake(validatorSummary);

const stakingPoolActivationEpoch = Number(validatorSummary?.stakingPoolActivationEpoch || 0);

// flag as new validator if the validator was activated in the last epoch
// for genesis validators, this will be false
const newValidator = currentEpoch - stakingPoolActivationEpoch <= 1 && currentEpoch !== 0;

// flag if the validator is at risk of being removed from the active set
const isAtRisk = system?.atRiskValidators.some((item) => item[0] === validatorAddress);

const { apy, isApyApproxZero } = rollingAverageApys?.[validatorAddress] ?? {
apy: null,
};

const commission = validatorSummary ? Number(validatorSummary.commissionRate) / 100 : 0;

return {
system,
isPendingValidators,
errorValidators,

currentEpoch,
validatorSummary,
name: validatorSummary?.name || '',
stakingPoolActivationEpoch,
commission,
newValidator,
isAtRisk,
apy,
isApyApproxZero,
totalValidatorStake,
};
}
10 changes: 6 additions & 4 deletions apps/core/src/hooks/useGetValidatorsApy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,11 +13,13 @@ import { roundFloat } from '../utils/roundFloat';

const DEFAULT_APY_DECIMALS = 2;

export interface ValidatorApyData {
apy: number;
isApyApproxZero: boolean;
}

export interface ApyByValidator {
[validatorAddress: string]: {
apy: number;
isApyApproxZero: boolean;
};
[validatorAddress: string]: ValidatorApyData;
}
// For small APY, show ~0% instead of 0%
// If APY falls below 0.001, show ~0% instead of 0% since we round to 2 decimal places
Expand Down
6 changes: 6 additions & 0 deletions apps/core/src/utils/formatApy.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
// Copyright (c) 2024 IOTA Stiftung
// SPDX-License-Identifier: Apache-2.0

export function formatApy(apy: number, isApyApproxZero: boolean = false): string {
return isApyApproxZero ? '~0%' : `${apy}%`;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
return isApyApproxZero ? '~0%' : `${apy}%`;
return isApyApproxZero ? '~ 0%' : `${apy}%`;

}
1 change: 1 addition & 0 deletions apps/core/src/utils/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ export * from './getDelegationDataByStakeId';
export * from './api-env';
export * from './getExplorerPaths';
export * from './getExplorerLink';
export * from './formatApy';

export * from './stake';
export * from './transaction';
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
// Modifications Copyright (c) 2024 IOTA Stiftung
// SPDX-License-Identifier: Apache-2.0

import { CoinFormat, formatBalance } from '@iota/core';
import { CoinFormat, formatBalance } from '../../index';
import BigNumber from 'bignumber.js';
import { mixed, object } from 'yup';

Expand Down
9 changes: 9 additions & 0 deletions apps/core/src/utils/stake/getTotalValidatorStake.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
// Copyright (c) 2024 IOTA Stiftung
// SPDX-License-Identifier: Apache-2.0

import { IotaValidatorSummary } from '@iota/iota-sdk/client';

//TODO: verify this is the correct validator stake balance
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

comment unnecessary

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this was in the code previously, are you sure we should delete it?

export function getTotalValidatorStake(validatorSummary: IotaValidatorSummary | null) {
return validatorSummary?.stakingPoolIotaBalance || 0;
}
2 changes: 2 additions & 0 deletions apps/core/src/utils/stake/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,3 +6,5 @@ export * from './formatDelegatedStake';
export * from './createStakeTransaction';
export * from './createTimelockedUnstakeTransaction';
export * from './createTimelockedStakeTransaction';
export * from './createValidationSchema';
export * from './getTotalValidatorStake';
39 changes: 25 additions & 14 deletions apps/wallet-dashboard/app/(protected)/staking/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,8 @@
'use client';

import { AmountBox, Box, StakeCard, StakeDialog, Button } from '@/components';
import { StakeDetailsDialog } from '@/components/Dialogs';
import { useStakeDialog } from '@/components/Dialogs/Staking/hooks/useStakeDialog';

import {
ExtendedDelegatedStake,
formatDelegatedStake,
Expand All @@ -17,12 +18,22 @@ import {
} from '@iota/core';
import { useCurrentAccount } from '@iota/dapp-kit';
import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils';
import { useState } from 'react';
import { StakeDialogView } from '@/components/Dialogs/Staking/StakeDialog';

function StakingDashboardPage(): JSX.Element {
const account = useCurrentAccount();
const [isDialogStakeOpen, setIsDialogStakeOpen] = useState(false);
const [selectedStake, setSelectedStake] = useState<ExtendedDelegatedStake | null>(null);

const {
isDialogStakeOpen,
stakeDialogView,
setStakeDialogView,
selectedStake,
setSelectedStake,
selectedValidator,
setSelectedValidator,
handleCloseStakeDialog,
handleNewStake,
} = useStakeDialog();

const { data: delegatedStakeData } = useGetDelegatedStake({
address: account?.address || '',
Expand All @@ -43,13 +54,10 @@ function StakingDashboardPage(): JSX.Element {
);

const viewStakeDetails = (extendedStake: ExtendedDelegatedStake) => {
setStakeDialogView(StakeDialogView.Details);
setSelectedStake(extendedStake);
};

function handleNewStake() {
setIsDialogStakeOpen(true);
}

return (
<>
<div className="flex flex-col items-center justify-center gap-4 pt-12">
Expand Down Expand Up @@ -79,12 +87,15 @@ function StakingDashboardPage(): JSX.Element {
</Box>
<Button onClick={handleNewStake}>New Stake</Button>
</div>
<StakeDialog isOpen={isDialogStakeOpen} setOpen={setIsDialogStakeOpen} />;
{selectedStake && (
<StakeDetailsDialog
extendedStake={selectedStake}
handleClose={() => setSelectedStake(null)}
showActiveStatus
{isDialogStakeOpen && stakeDialogView && (
<StakeDialog
stakedDetails={selectedStake}
isOpen={isDialogStakeOpen}
handleClose={handleCloseStakeDialog}
view={stakeDialogView}
setView={setStakeDialogView}
selectedValidator={selectedValidator}
setSelectedValidator={setSelectedValidator}
/>
)}
</>
Expand Down

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

Loading
Loading