mirror of
https://github.com/pezkuwichain/pezkuwi-apps.git
synced 2026-04-22 08:58:00 +00:00
7a4bbeac25
- Update @pezkuwi/extension-inject to ^0.62.13 with proper /types exports - Update @pezkuwi/extension-dapp to ^0.62.13 - Update @pezkuwi/extension-compat-metamask to ^0.62.13 - Fix IconTheme type to include 'bizinikiwi' and 'pezkuwi' themes - Fix endpoint array issues (getTeleports -> direct array references) - Add type assertions for external package compatibility (acala, moonbeam, parallel) - Fix subspace.ts dynamic class typing - Fix conviction type in page-referenda - Update Pallet type names to Pezpallet prefix across codebase - Define InjectedExtension types locally for module resolution - Add styled-components DefaultTheme augmentation - Add react-copy-to-clipboard type declaration for React 18 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
212 lines
7.2 KiB
TypeScript
212 lines
7.2 KiB
TypeScript
// Copyright 2017-2026 @pezkuwi/react-hooks authors & contributors
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
import type { Dispatch, SetStateAction } from 'react';
|
|
import type { ApiPromise } from '@pezkuwi/api';
|
|
import type { DeriveEraPoints, DeriveEraRewards, DeriveStakerReward } from '@pezkuwi/api-derive/types';
|
|
import type { u32, Vec } from '@pezkuwi/types';
|
|
import type { EraIndex } from '@pezkuwi/types/interfaces';
|
|
import type { PezpalletStakingStakingLedger } from '@pezkuwi/types/lookup';
|
|
import type { StakerState } from './types.js';
|
|
|
|
import { useCallback, useEffect, useState } from 'react';
|
|
|
|
import { BN_ZERO } from '@pezkuwi/util';
|
|
|
|
import { createNamedHook } from './createNamedHook.js';
|
|
import { useApi } from './useApi.js';
|
|
import { useCall } from './useCall.js';
|
|
import { useEventTrigger } from './useEventTrigger.js';
|
|
import { useIsMountedRef } from './useIsMountedRef.js';
|
|
import { useOwnStashIds } from './useOwnStashes.js';
|
|
|
|
interface State {
|
|
allRewards?: Record<string, DeriveStakerReward[]> | null;
|
|
isLoadingRewards: boolean;
|
|
rewardCount: number;
|
|
}
|
|
|
|
interface ValidatorWithEras {
|
|
eras: EraIndex[];
|
|
stashId: string;
|
|
}
|
|
|
|
interface Filtered {
|
|
filteredEras: EraIndex[];
|
|
validatorEras: ValidatorWithEras[];
|
|
}
|
|
|
|
const EMPTY_FILTERED: Filtered = {
|
|
filteredEras: [],
|
|
validatorEras: []
|
|
};
|
|
|
|
const EMPTY_STATE: State = {
|
|
isLoadingRewards: true,
|
|
rewardCount: 0
|
|
};
|
|
|
|
function getLegacyRewards (ledger: PezpalletStakingStakingLedger, claimedRewardsEras?: Vec<u32>): u32[] {
|
|
const legacyRewards = ledger.legacyClaimedRewards || (ledger as unknown as { claimedRewards: u32[] }).claimedRewards || [];
|
|
|
|
return legacyRewards.concat(claimedRewardsEras?.toArray() || []);
|
|
}
|
|
|
|
function getRewards ([[stashIds], available]: [[string[], EraIndex[]], DeriveStakerReward[][]]): State {
|
|
const allRewards: Record<string, DeriveStakerReward[]> = {};
|
|
|
|
stashIds.forEach((stashId, index): void => {
|
|
allRewards[stashId] = available[index].filter(({ eraReward }) => !eraReward.isZero());
|
|
});
|
|
|
|
return {
|
|
allRewards,
|
|
isLoadingRewards: false,
|
|
rewardCount: Object.values(allRewards).filter((rewards) => rewards.length !== 0).length
|
|
};
|
|
}
|
|
|
|
function getValRewards (api: ApiPromise, validatorEras: ValidatorWithEras[], erasPoints: DeriveEraPoints[], erasRewards: DeriveEraRewards[]): State {
|
|
const allRewards: Record<string, DeriveStakerReward[]> = {};
|
|
|
|
validatorEras.forEach(({ eras, stashId }): void => {
|
|
eras.forEach((era): void => {
|
|
const eraPoints = erasPoints.find((p) => p.era.eq(era));
|
|
const eraRewards = erasRewards.find((r) => r.era.eq(era));
|
|
|
|
if (eraPoints?.eraPoints.gt(BN_ZERO) && eraPoints?.validators[stashId] && eraRewards) {
|
|
const reward = eraPoints.validators[stashId].mul(eraRewards.eraReward).div(eraPoints.eraPoints);
|
|
|
|
if (!reward.isZero()) {
|
|
const total = api.createType('Balance', reward);
|
|
|
|
if (!allRewards[stashId]) {
|
|
allRewards[stashId] = [];
|
|
}
|
|
|
|
allRewards[stashId].push({
|
|
era,
|
|
eraReward: eraRewards.eraReward,
|
|
isClaimed: false,
|
|
isEmpty: false,
|
|
isValidator: true,
|
|
nominating: [],
|
|
validators: {
|
|
[stashId]: {
|
|
total,
|
|
value: total
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
});
|
|
});
|
|
|
|
return {
|
|
allRewards,
|
|
isLoadingRewards: false,
|
|
rewardCount: Object.values(allRewards).filter((rewards) => rewards.length !== 0).length
|
|
};
|
|
}
|
|
|
|
function useStakerRewards (filteredEras: EraIndex[], setState: Dispatch<SetStateAction<State>>, ownValidators?: StakerState[], additional?: string[]) {
|
|
const { api } = useApi();
|
|
|
|
const stashIds = useOwnStashIds(additional);
|
|
const trigger = useEventTrigger([api.events.staking?.PayoutStarted, api.events.staking?.Rewarded]);
|
|
const [stakerRewards, setStakerRewards] = useState<[[string[], EraIndex[]], DeriveStakerReward[][]]>();
|
|
|
|
const onFetchStakeRewards = useCallback(async () => {
|
|
if (!ownValidators?.length && !!filteredEras.length && stashIds) {
|
|
setState((e) => ({ ...e, isLoadingRewards: true }));
|
|
|
|
const stakerRewards = await api.derive.staking?.stakerRewardsMultiEras(stashIds, filteredEras);
|
|
|
|
setStakerRewards([[stashIds, filteredEras], stakerRewards]);
|
|
}
|
|
}, [api.derive.staking, filteredEras, ownValidators?.length, setState, stashIds]);
|
|
|
|
useEffect(() => {
|
|
onFetchStakeRewards().catch(console.error);
|
|
}, [onFetchStakeRewards]);
|
|
|
|
useEffect(() => {
|
|
if (trigger.blockHash) {
|
|
onFetchStakeRewards().catch(console.error);
|
|
}
|
|
}, [onFetchStakeRewards, trigger.blockHash]);
|
|
|
|
return stakerRewards;
|
|
}
|
|
|
|
function useOwnEraRewardsImpl (maxEras?: number, ownValidators?: StakerState[], additional?: string[]): State {
|
|
const { api } = useApi();
|
|
const mountedRef = useIsMountedRef();
|
|
|
|
const allEras = useCall<EraIndex[]>(api.derive.staking?.erasHistoric);
|
|
const [{ filteredEras, validatorEras }, setFiltered] = useState<Filtered>(EMPTY_FILTERED);
|
|
const [state, setState] = useState<State>(EMPTY_STATE);
|
|
|
|
const stakerRewards = useStakerRewards(filteredEras, setState, ownValidators, additional);
|
|
const erasPoints = useCall<DeriveEraPoints[]>(!!validatorEras.length && !!filteredEras.length && api.derive.staking._erasPoints, [filteredEras, false]);
|
|
const erasRewards = useCall<DeriveEraRewards[]>(!!validatorEras.length && !!filteredEras.length && api.derive.staking._erasRewards, [filteredEras, false]);
|
|
|
|
useEffect((): void => {
|
|
setState({ allRewards: null, isLoadingRewards: true, rewardCount: 0 });
|
|
}, [maxEras, ownValidators]);
|
|
|
|
useEffect((): void => {
|
|
if (allEras && maxEras) {
|
|
const filteredEras = allEras.slice(-1 * maxEras);
|
|
const validatorEras: ValidatorWithEras[] = [];
|
|
|
|
if (allEras.length === 0) {
|
|
setState({
|
|
allRewards: {},
|
|
isLoadingRewards: false,
|
|
rewardCount: 0
|
|
});
|
|
setFiltered({ filteredEras, validatorEras });
|
|
} else if (ownValidators?.length) {
|
|
ownValidators.forEach(({ claimedRewardsEras, stakingLedger, stashId }): void => {
|
|
if (stakingLedger) {
|
|
const eras = filteredEras.filter((era) => !getLegacyRewards(stakingLedger, claimedRewardsEras).some((c) => era.eq(c)));
|
|
|
|
if (eras.length) {
|
|
validatorEras.push({ eras, stashId });
|
|
}
|
|
}
|
|
});
|
|
|
|
// When we have just claimed, we have filtered eras, but no validator eras - set accordingly
|
|
if (filteredEras.length && !validatorEras.length) {
|
|
setState({
|
|
allRewards: {},
|
|
isLoadingRewards: false,
|
|
rewardCount: 0
|
|
});
|
|
}
|
|
}
|
|
|
|
setFiltered({ filteredEras, validatorEras });
|
|
}
|
|
}, [allEras, maxEras, ownValidators]);
|
|
|
|
useEffect((): void => {
|
|
mountedRef.current && stakerRewards && !ownValidators && setState(
|
|
getRewards(stakerRewards)
|
|
);
|
|
}, [mountedRef, ownValidators, stakerRewards]);
|
|
|
|
useEffect((): void => {
|
|
mountedRef && erasPoints && erasRewards && ownValidators && setState(
|
|
getValRewards(api, validatorEras, erasPoints, erasRewards)
|
|
);
|
|
}, [api, erasPoints, erasRewards, mountedRef, ownValidators, validatorEras]);
|
|
|
|
return state;
|
|
}
|
|
|
|
export const useOwnEraRewards = createNamedHook('useOwnEraRewards', useOwnEraRewardsImpl);
|