Files
pezkuwi-sdk-ui/packages/page-explorer/src/NodeInfo/Peers.tsx
T
pezkuwichain d949863789 Initial commit: Pezkuwi SDK UI
Comprehensive web interface for interacting with Pezkuwi blockchain.

Features:
- Blockchain explorer
- Wallet management
- Staking interface
- Governance participation
- Developer tools

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-19 13:55:36 +03:00

78 lines
1.9 KiB
TypeScript

// Copyright 2017-2026 @pezkuwi/app-explorer authors & contributors
// SPDX-License-Identifier: Apache-2.0
import type { PeerInfo } from '@pezkuwi/types/interfaces';
import React, { useMemo, useRef } from 'react';
import { styled, Table } from '@pezkuwi/react-components';
import { formatNumber, stringPascalCase } from '@pezkuwi/util';
import { useTranslation } from '../translate.js';
interface Props {
className?: string;
peers?: PeerInfo[] | null;
}
function sortPeers (peers: PeerInfo[]) {
return peers
.map(({ bestHash, bestNumber, peerId, roles }) => ({
bestHash: bestHash.toHex(),
bestNumber,
peerId: peerId.toString(),
roles: stringPascalCase(roles)
}))
.sort((a, b) => a.peerId.localeCompare(b.peerId))
.sort((a, b) => a.roles.localeCompare(b.roles))
.sort((a, b) => b.bestNumber.cmp(a.bestNumber));
}
function Peers ({ className = '', peers }: Props): React.ReactElement<Props> {
const { t } = useTranslation();
const headerRef = useRef<([React.ReactNode?, string?, number?] | false)[]>([
[t('connected peers'), 'start', 2],
[t('best hash'), 'start'],
[t('best #'), 'number']
]);
const sorted = useMemo(
() => peers && sortPeers(peers),
[peers]
);
return (
<StyledTable
className={className}
empty={t('no peers connected')}
header={headerRef.current}
>
{sorted?.map(({ bestHash, bestNumber, peerId, roles }) => (
<tr key={peerId}>
<td className='roles'>{roles}</td>
<td className='hash overflow'>{peerId}</td>
<td className='hash overflow'>{bestHash}</td>
<td className='number bestNumber'>{formatNumber(bestNumber)}</td>
</tr>
))}
</StyledTable>
);
}
const StyledTable = styled(Table)`
overflow-x: auto;
td.roles {
max-width: 9ch;
width: 9ch;
}
td.bestNumber {
max-width: 11ch;
width: 11ch;
}
`;
export default React.memo(Peers);