mirror of
https://github.com/RoboSats/robosats.git
synced 2025-07-18 08:43:14 +00:00
345 lines
10 KiB
TypeScript
345 lines
10 KiB
TypeScript
import React, { useCallback, useContext, useEffect, useState } from 'react';
|
|
import { Link, LinkOff } from '@mui/icons-material';
|
|
import { Box, Checkbox, CircularProgress, Grid, Rating, Typography, useTheme } from '@mui/material';
|
|
import { DataGrid, type GridColDef, type GridValidRowModel } from '@mui/x-data-grid';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { AppContext, type UseAppStoreType } from '../../contexts/AppContext';
|
|
import { FederationContext, type UseFederationStoreType } from '../../contexts/FederationContext';
|
|
import { type Coordinator } from '../../models';
|
|
import headerStyleFix from '../DataGrid/HeaderFix';
|
|
import RobotAvatar from '../RobotAvatar';
|
|
import { verifyCoordinatorToken } from '../../utils/nostr';
|
|
|
|
interface FederationTableProps {
|
|
maxWidth?: number;
|
|
maxHeight?: number;
|
|
fillContainer?: boolean;
|
|
}
|
|
|
|
const FederationTable = ({
|
|
maxWidth = 90,
|
|
maxHeight = 50,
|
|
fillContainer = false,
|
|
}: FederationTableProps): React.JSX.Element => {
|
|
const { t } = useTranslation();
|
|
const { federation, federationUpdatedAt } = useContext<UseFederationStoreType>(FederationContext);
|
|
const { setOpen, windowSize, settings } = useContext<UseAppStoreType>(AppContext);
|
|
const theme = useTheme();
|
|
const [pageSize, setPageSize] = useState<number>(0);
|
|
const [loading, setLoading] = useState<boolean>(true);
|
|
const [ratings, setRatings] = useState<Record<string, Record<string, number>>>(
|
|
federation.getCoordinators().reduce((acc, coord) => {
|
|
if (coord.nostrHexPubkey) acc[coord.nostrHexPubkey] = {};
|
|
return acc;
|
|
}, {}),
|
|
);
|
|
const [useDefaultPageSize, setUseDefaultPageSize] = useState(true);
|
|
|
|
// all sizes in 'em'
|
|
const fontSize = theme.typography.fontSize;
|
|
const verticalHeightFrame = 3.3;
|
|
const verticalHeightRow = 3.27;
|
|
const defaultPageSize = Math.max(
|
|
Math.floor((maxHeight - verticalHeightFrame) / verticalHeightRow),
|
|
1,
|
|
);
|
|
const height = defaultPageSize * verticalHeightRow + verticalHeightFrame;
|
|
const mobile = windowSize.width < 44;
|
|
|
|
useEffect(() => {
|
|
loadRatings();
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
if (useDefaultPageSize) {
|
|
setPageSize(defaultPageSize);
|
|
}
|
|
}, [federationUpdatedAt]);
|
|
|
|
const loadRatings: () => void = () => {
|
|
if (settings.connection !== 'nostr') return;
|
|
setLoading(true);
|
|
federation.roboPool.subscribeRatings({
|
|
onevent: (event) => {
|
|
const verfied = verifyCoordinatorToken(event);
|
|
const coordinatorPubKey = event.tags.find((t) => t[0] === 'p')?.[1];
|
|
if (verfied && coordinatorPubKey) {
|
|
const rating = event.tags.find((t) => t[0] === 'rating')?.[1];
|
|
if (rating) {
|
|
setRatings((prev) => {
|
|
prev[coordinatorPubKey][event.pubkey] = parseFloat(rating);
|
|
return prev;
|
|
});
|
|
}
|
|
}
|
|
},
|
|
oneose: () => setLoading(false),
|
|
});
|
|
};
|
|
|
|
const localeText = {
|
|
noResultsOverlayLabel: t('No coordinators found.'),
|
|
};
|
|
|
|
const onClickCoordinator = function (shortAlias: string): void {
|
|
setOpen((open) => {
|
|
return { ...open, coordinator: shortAlias };
|
|
});
|
|
};
|
|
|
|
const aliasObj = useCallback(() => {
|
|
return {
|
|
field: 'longAlias',
|
|
headerName: mobile ? '' : t('Rating'),
|
|
width: mobile ? 60 : 190,
|
|
renderCell: (params: { row: Coordinator }) => {
|
|
const coordinator = federation.getCoordinator(params.row.shortAlias);
|
|
return (
|
|
<Grid
|
|
container
|
|
direction='row'
|
|
sx={{
|
|
cursor: 'pointer',
|
|
position: 'relative',
|
|
left: '-0.3em',
|
|
width: '50em',
|
|
marginTop: '2px',
|
|
}}
|
|
wrap='nowrap'
|
|
onClick={() => {
|
|
onClickCoordinator(params.row.shortAlias);
|
|
}}
|
|
alignItems='center'
|
|
spacing={1}
|
|
>
|
|
<Grid item>
|
|
<RobotAvatar
|
|
shortAlias={coordinator.federated ? params.row.shortAlias : undefined}
|
|
hashId={coordinator.federated ? undefined : coordinator.mainnet.onion}
|
|
style={{ width: '3.215em', height: '3.215em' }}
|
|
smooth={true}
|
|
small={true}
|
|
/>
|
|
</Grid>
|
|
{!mobile ? (
|
|
<Grid item>
|
|
<Typography>{params.row.longAlias}</Typography>
|
|
</Grid>
|
|
) : (
|
|
<></>
|
|
)}
|
|
</Grid>
|
|
);
|
|
},
|
|
};
|
|
}, []);
|
|
|
|
const ratingObj = useCallback(() => {
|
|
return {
|
|
field: 'rating',
|
|
headerName: t('Rating'),
|
|
width: mobile ? 60 : 180,
|
|
renderCell: (params: { row: Coordinator }) => {
|
|
const coordinator = federation.getCoordinator(params.row.shortAlias);
|
|
const coordinatorRating = ratings[coordinator.nostrHexPubkey];
|
|
|
|
if (!coordinatorRating) return <></>;
|
|
|
|
const totalRatings = Object.values(coordinatorRating);
|
|
const total = totalRatings.length;
|
|
const sum: number = Object.values(totalRatings).reduce((accumulator, currentValue) => {
|
|
return accumulator + currentValue;
|
|
}, 0);
|
|
const average = total < 1 ? 0 : sum / total;
|
|
|
|
return (
|
|
<>
|
|
{mobile ? (
|
|
<Grid container direction='column' alignItems='center' style={{ paddingTop: 10 }}>
|
|
<Typography>{`${parseFloat((average * 10).toFixed(1))}`}</Typography>
|
|
</Grid>
|
|
) : (
|
|
<>
|
|
<Rating
|
|
readOnly
|
|
precision={0.5}
|
|
name='size-large'
|
|
value={average * 5}
|
|
defaultValue={0}
|
|
disabled={settings.connection !== 'nostr'}
|
|
onClick={() => {
|
|
onClickCoordinator(params.row.shortAlias);
|
|
}}
|
|
/>
|
|
<Typography variant='caption' color='text.secondary'>
|
|
{`(${total})`}
|
|
</Typography>
|
|
</>
|
|
)}
|
|
</>
|
|
);
|
|
},
|
|
};
|
|
}, [federationUpdatedAt]);
|
|
|
|
const enabledObj = useCallback(
|
|
(width: number) => {
|
|
return {
|
|
field: 'enabled',
|
|
headerName: t('Enabled'),
|
|
width: width * fontSize,
|
|
renderCell: (params: { row: Coordinator }) => {
|
|
return (
|
|
<Checkbox
|
|
checked={params.row.enabled}
|
|
onClick={() => {
|
|
onEnableChange(params.row.shortAlias);
|
|
}}
|
|
/>
|
|
);
|
|
},
|
|
};
|
|
},
|
|
[federationUpdatedAt],
|
|
);
|
|
|
|
const upObj = useCallback(
|
|
(width: number) => {
|
|
return {
|
|
field: 'up',
|
|
headerName: t('Up'),
|
|
width: width * fontSize,
|
|
renderCell: (params: { row: Coordinator }) => {
|
|
return (
|
|
<div
|
|
style={{ cursor: 'pointer' }}
|
|
onClick={() => {
|
|
onClickCoordinator(params.row.shortAlias);
|
|
}}
|
|
>
|
|
{Boolean(params.row.loadingLimits) && Boolean(params.row.enabled) ? (
|
|
<CircularProgress thickness={0.35 * fontSize} size={1.5 * fontSize} />
|
|
) : params.row.limits !== undefined ? (
|
|
<Link color='success' />
|
|
) : (
|
|
<LinkOff color='error' />
|
|
)}
|
|
</div>
|
|
);
|
|
},
|
|
};
|
|
},
|
|
[federationUpdatedAt],
|
|
);
|
|
|
|
const columnSpecs = {
|
|
alias: {
|
|
priority: 2,
|
|
order: 1,
|
|
normal: {
|
|
width: 12.1,
|
|
object: aliasObj,
|
|
},
|
|
},
|
|
rating: {
|
|
priority: 2,
|
|
order: 2,
|
|
normal: {
|
|
width: 12.1,
|
|
object: ratingObj,
|
|
},
|
|
},
|
|
up: {
|
|
priority: 3,
|
|
order: 3,
|
|
normal: {
|
|
width: 3.5,
|
|
object: upObj,
|
|
},
|
|
},
|
|
enabled: {
|
|
priority: 1,
|
|
order: 4,
|
|
normal: {
|
|
width: 5,
|
|
object: enabledObj,
|
|
},
|
|
},
|
|
};
|
|
|
|
const filteredColumns = function (): {
|
|
columns: Array<GridColDef<GridValidRowModel>>;
|
|
width: number;
|
|
} {
|
|
const useSmall = maxWidth < 30;
|
|
const selectedColumns: object[] = [];
|
|
let width: number = 0;
|
|
|
|
for (const value of Object.values(columnSpecs)) {
|
|
const colWidth = Number(
|
|
useSmall && Boolean(value.small) ? value.small.width : value.normal.width,
|
|
);
|
|
const colObject = useSmall && Boolean(value.small) ? value.small.object : value.normal.object;
|
|
|
|
if (width + colWidth < maxWidth || selectedColumns.length < 2) {
|
|
width = width + colWidth;
|
|
selectedColumns.push([colObject(colWidth, false), value.order]);
|
|
} else {
|
|
selectedColumns.push([colObject(colWidth, true), value.order]);
|
|
}
|
|
}
|
|
|
|
// sort columns by column.order value
|
|
selectedColumns.sort(function (first, second) {
|
|
return first[1] - second[1];
|
|
});
|
|
|
|
const columns: Array<GridColDef<GridValidRowModel>> = selectedColumns.map(function (item) {
|
|
return item[0];
|
|
});
|
|
|
|
return { columns, width: width * 0.9 };
|
|
};
|
|
|
|
const { columns, width } = filteredColumns();
|
|
|
|
const onEnableChange = function (shortAlias: string): void {
|
|
if (federation.getCoordinator(shortAlias).enabled === true) {
|
|
federation.disableCoordinator(shortAlias);
|
|
} else {
|
|
federation.enableCoordinator(shortAlias);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<Box
|
|
sx={
|
|
fillContainer
|
|
? { width: '100%', height: '100%' }
|
|
: { width: `${width}em`, height: `${height}em`, overflow: 'auto' }
|
|
}
|
|
>
|
|
<DataGrid
|
|
loading={loading}
|
|
sx={headerStyleFix}
|
|
localeText={localeText}
|
|
rowHeight={3.714 * theme.typography.fontSize}
|
|
headerHeight={3.25 * theme.typography.fontSize}
|
|
rows={federation.getCoordinators()}
|
|
getRowId={(params: Coordinator) => params.shortAlias}
|
|
columns={columns}
|
|
checkboxSelection={false}
|
|
pageSize={pageSize}
|
|
rowsPerPageOptions={width < 22 ? [] : [0, pageSize, defaultPageSize * 2, 50, 100]}
|
|
onPageSizeChange={(newPageSize) => {
|
|
setPageSize(newPageSize);
|
|
setUseDefaultPageSize(false);
|
|
}}
|
|
hideFooter={true}
|
|
/>
|
|
</Box>
|
|
);
|
|
};
|
|
|
|
export default FederationTable;
|