Skip to content

Commit

Permalink
Merge pull request #988 from oraichain/feat/sort-validator
Browse files Browse the repository at this point in the history
Feat/sort validator
  • Loading branch information
haunv3 authored Jun 18, 2024
2 parents 46431e9 + 83f62ff commit 02be6b3
Show file tree
Hide file tree
Showing 3 changed files with 141 additions and 121 deletions.
230 changes: 116 additions & 114 deletions src/components/ValidatorList/ValidatorCardList/ValidatorCardList.js
Original file line number Diff line number Diff line change
@@ -1,13 +1,13 @@
// @ts-nocheck
import React, {memo, useMemo} from "react";
import {NavLink} from "react-router-dom";
import React, { memo, useMemo } from "react";
import { NavLink } from "react-router-dom";
import classNames from "classnames/bind";
import consts from "src/constants/consts";
import {formatPercentage, formatInteger, formatOrai} from "src/helpers/helper";
import {computeTotalVotingPower} from "src/components/ValidatorList/ValidatorTable/ValidatorTable";
import { formatPercentage, formatInteger, formatOrai, groupAndShuffle } from "src/helpers/helper";
import { computeTotalVotingPower } from "src/components/ValidatorList/ValidatorTable/ValidatorTable";
import aiIcon from "src/assets/common/ai_ic.svg";
import {logoBrand} from "src/constants/logoBrand";
import {Progress} from "antd";
import { logoBrand } from "src/constants/logoBrand";
import { Progress } from "antd";
import Delegate from "src/components/common/Delegate";
import styles from "./ValidatorCardList.module.scss";
import "./style.css";
Expand All @@ -22,136 +22,138 @@ const getCumulativeShareCell = (previousValue, currentValue, totalValue) => {
return (
<div className={cx("cumulative-share-data-cell")}>
<div className={cx("graph")}>
<div className={cx("graph-previous")} style={{width: previousPercent + "%"}}></div>
<div className={cx("graph-current")} style={{left: previousPercent + "%", width: currentPercent + "%"}}></div>
<div className={cx("graph-previous")} style={{ width: previousPercent + "%" }}></div>
<div className={cx("graph-current")} style={{ left: previousPercent + "%", width: currentPercent + "%" }}></div>
</div>
<div className={cx("total-value")}>{totalPercent} %</div>
</div>
);
};

const ValidatorCardList = memo(({data = []}) => {
const ValidatorCardList = memo(({ data = [] }) => {
const totalVotingPower = useMemo(() => computeTotalVotingPower(data), [data]);
let previousVotingPower = 0;
return (
<div className='validator-card-list'>
{data?.map((item, index) => {
let currentVotingPower = 0;
let votingPowerDataCell = (
<div className={cx("voting-power-data-cell")}>
<div>-</div>
<div>- %</div>
</div>
);
if (item?.voting_power && totalVotingPower > 0) {
currentVotingPower = parseFloat(item.voting_power);
votingPowerDataCell = (
{groupAndShuffle(data, 5)
?.flat()
?.map((item, index) => {
let currentVotingPower = 0;
let votingPowerDataCell = (
<div className={cx("voting-power-data-cell")}>
<span>{formatInteger(currentVotingPower)}</span>
<span>{formatPercentage(currentVotingPower / totalVotingPower, 2)}%</span>
<div>-</div>
<div>- %</div>
</div>
);
}
if (item?.voting_power && totalVotingPower > 0) {
currentVotingPower = parseFloat(item.voting_power);
votingPowerDataCell = (
<div className={cx("voting-power-data-cell")}>
<span>{formatInteger(currentVotingPower)}</span>
<span>{formatPercentage(currentVotingPower / totalVotingPower, 2)}%</span>
</div>
);
}

const cumulativeShareDataCell = getCumulativeShareCell(previousVotingPower, currentVotingPower, totalVotingPower);
previousVotingPower += currentVotingPower;
const estAPR = item?.apr.toFixed(2);
const cumulativeShareDataCell = getCumulativeShareCell(previousVotingPower, currentVotingPower, totalVotingPower);
previousVotingPower += currentVotingPower;
const estAPR = item?.apr.toFixed(2);

const logoItem = logoBrand.find(it => item.operator_address === it.operatorAddress);
const logo = item?.image ? item?.image : logoItem;
const logoURL = logo ? (typeof logo === "object" ? logo.logo : logo) : false;
const validatorCardListItem = (
<div className={cx("validator-card-list-item")} key={"validator-card-list-item-" + index}>
<table>
<tbody>
<tr>
<td>
<div className={cx("item-title")}>Rank </div>
</td>
<td>
<div className={cx("item-link")}>{item?.rank ?? "-"}</div>
</td>
</tr>
const logoItem = logoBrand.find(it => item.operator_address === it.operatorAddress);
const logo = item?.image ? item?.image : logoItem;
const logoURL = logo ? (typeof logo === "object" ? logo.logo : logo) : false;
const validatorCardListItem = (
<div className={cx("validator-card-list-item")} key={"validator-card-list-item-" + index}>
<table>
<tbody>
<tr>
<td>
<div className={cx("item-title")}>Rank </div>
</td>
<td>
<div className={cx("item-link")}>{item?.rank ?? "-"}</div>
</td>
</tr>

<tr>
<td>
<div className={cx("item-title")}>Proposer</div>
</td>
<td>
{item?.moniker ? (
<NavLink className={cx("validator-data-cell")} to={`${consts.PATH.VALIDATORS}/${item.operator_address}`}>
{logoURL && <img alt='/' src={logoURL} width={32} height={32} className={cx("logo")} />}
{!logoURL && <div className={cx("logo-custom")}> {item.moniker.substring(0, 3).toUpperCase()} </div>}
{item.moniker}
</NavLink>
) : (
<div className={cx("item-link")}>-</div>
)}
</td>
</tr>
<tr>
<td>
<div className={cx("item-title")}>Proposer</div>
</td>
<td>
{item?.moniker ? (
<NavLink className={cx("validator-data-cell")} to={`${consts.PATH.VALIDATORS}/${item.operator_address}`}>
{logoURL && <img alt='/' src={logoURL} width={32} height={32} className={cx("logo")} />}
{!logoURL && <div className={cx("logo-custom")}> {item.moniker.substring(0, 3).toUpperCase()} </div>}
{item.moniker}
</NavLink>
) : (
<div className={cx("item-link")}>-</div>
)}
</td>
</tr>

<tr>
<td colSpan={2}>
<div className={cx("item-title")}>Voting power</div>
{votingPowerDataCell}
</td>
</tr>
<tr>
<td colSpan={2}>
<div className={cx("item-title")}>Voting power</div>
{votingPowerDataCell}
</td>
</tr>

<tr>
<td colSpan={2}>
<div className={cx("item-title")}>Cumulative Share %</div>
{cumulativeShareDataCell}
</td>
</tr>
<tr>
<td colSpan={2}>
<div className={cx("item-title")}>Cumulative Share %</div>
{cumulativeShareDataCell}
</td>
</tr>

<tr>
<td>
<div className={cx("item-title")}>Uptime</div>
<div className={cx("item-text")}>
<div>{item?.uptime ? formatPercentage(item.uptime, 2) + "%" : "-"}</div>
<div>
{item?.uptime && (
<Progress
percent={formatPercentage(item.uptime, 2)}
showInfo={false}
strokeColor={formatPercentage(item.uptime, 2) === 100 ? "#37cc6e" : "#7664E4"}
trailColor='#bfbfbf'
/>
)}
<tr>
<td>
<div className={cx("item-title")}>Uptime</div>
<div className={cx("item-text")}>
<div>{item?.uptime ? formatPercentage(item.uptime, 2) + "%" : "-"}</div>
<div>
{item?.uptime && (
<Progress
percent={formatPercentage(item.uptime, 2)}
showInfo={false}
strokeColor={formatPercentage(item.uptime, 2) === 100 ? "#37cc6e" : "#7664E4"}
trailColor='#bfbfbf'
/>
)}
</div>
</div>
</div>
</td>
<td>
<div className={cx("item-title")}>Commisson</div>
<div className={cx("item-text")}>{item?.commission_rate ? formatPercentage(item.commission_rate, 2) + "%" : "-"}</div>
</td>
</tr>
</td>
<td>
<div className={cx("item-title")}>Commisson</div>
<div className={cx("item-text")}>{item?.commission_rate ? formatPercentage(item.commission_rate, 2) + "%" : "-"}</div>
</td>
</tr>

<tr>
<td>
<div className={cx("item-title")}>Self Bonded</div>
<div className={cx("voting-power-data-cell")}>
<span>{formatOrai(item?.self_bonded)}</span>
<span>ORAI</span>
</div>
</td>
<td>
<Delegate
a={item}
operatorAddress={item.operator_address}
openButtonText='Delegate'
delegateText={`Delegate for "${item?.moniker}"`}
estAPR={estAPR / 100}
/>
</td>
</tr>
</tbody>
</table>
</div>
);
<tr>
<td>
<div className={cx("item-title")}>Self Bonded</div>
<div className={cx("voting-power-data-cell")}>
<span>{formatOrai(item?.self_bonded)}</span>
<span>ORAI</span>
</div>
</td>
<td>
<Delegate
a={item}
operatorAddress={item.operator_address}
openButtonText='Delegate'
delegateText={`Delegate for "${item?.moniker}"`}
estAPR={estAPR / 100}
/>
</td>
</tr>
</tbody>
</table>
</div>
);

return validatorCardListItem;
})}
return validatorCardListItem;
})}
</div>
);
});
Expand Down
10 changes: 3 additions & 7 deletions src/components/ValidatorList/ValidatorTable/ValidatorTable.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import { _ } from "src/lib/scripts";
import { tableThemes } from "src/constants/tableThemes";
import { sortDirections } from "src/constants/sortDirections";
import consts from "src/constants/consts";
import { formatPercentage, formatInteger, formatOrai } from "src/helpers/helper";
import { formatPercentage, formatInteger, formatOrai, groupAndShuffle } from "src/helpers/helper";
import { compareTwoValues } from "src/helpers/compare";
import Delegate from "src/components/common/Delegate";
import ThemedTable from "src/components/common/ThemedTable";
Expand Down Expand Up @@ -214,13 +214,9 @@ const ValidatorTable = memo(({ data = [] }) => {

const sortData = (data, extraSortField = sortFields.RANK) => {
if (!data) return [];

if (isFirstSort) {
return [...data]
.map(e => {
return { ...e, votingPowerMixUpTime: e.voting_power * e.uptime };
})
.sort((a, b) => 0.5 - Math.random());
const groupSize = 10;
return groupAndShuffle(data, groupSize).flat();
}

if (canSort) {
Expand Down
22 changes: 22 additions & 0 deletions src/helpers/helper.js
Original file line number Diff line number Diff line change
Expand Up @@ -376,3 +376,25 @@ export const calculateTallyProposal = ({ totalVote, bonded, tally }) => {
vote_percentage: votePercentage,
};
};

export function shuffleArray(array) {
for (let i = array.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[array[i], array[j]] = [array[j], array[i]];
}
return array;
}

export function groupAndShuffle(array, groupSize) {
const sortedArray = array
.map(e => {
return { ...e, votingPowerMixUpTime: e.voting_power * e.uptime };
})
.sort((a, b) => b.votingPowerMixUpTime - a.votingPowerMixUpTime);
const groups = [];
for (let i = 0; i < sortedArray.length; i += groupSize) {
groups.push(sortedArray.slice(i, i + groupSize));
}
const shuffledGroups = groups.map(group => shuffleArray(group));
return shuffledGroups;
}

0 comments on commit 02be6b3

Please sign in to comment.