Skip to content

Commit

Permalink
refactor into MedianRow & PercentTargetMetRow
Browse files Browse the repository at this point in the history
  • Loading branch information
fdelemarre committed Jul 2, 2024
1 parent 101d20c commit 411b40a
Show file tree
Hide file tree
Showing 3 changed files with 135 additions and 56 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
import React from "react";
import _ from "../../../../domain/entities/generic/Collection";
import { TableCell, TableRow } from "@material-ui/core";
import styled from "styled-components";
import { PerformanceOverviewTableProps } from "./PerformanceOverviewTable";

export type TableColumn = {
value: string;
label: string;
dark?: boolean;
};

type MedianRowProps = {
columns: TableColumn[];
rows: {
[key: TableColumn["value"]]: string;
}[];
calculateColumns: TableColumn["value"][];
};

export const MedianRow: React.FC<MedianRowProps> = React.memo(
({ rows, columns, calculateColumns }) => {
const calculateMedian = (
rows: PerformanceOverviewTableProps["rows"],
column: TableColumn["value"]
) => {
const values = rows.map(row => Number(row[column])).filter(value => !isNaN(value));
values.sort((a, b) => a - b);
const mid = Math.floor(values.length / 2);
return values.length % 2 !== 0
? values[mid]
: ((values[mid - 1] || 0) + (values[mid] || 0)) / 2;
};

return (
<TableRow>
{columns.map((column, columnIndex) => (
<FooterTableCell
key={`median-${column.value}`}
$boldUnderline={columnIndex === 0}
>
{columnIndex === 0 && "Median"}
{calculateColumns.includes(column.value)
? calculateMedian(rows, column.value)
: ""}
</FooterTableCell>
))}
</TableRow>
);
}
);

const FooterTableCell = styled(TableCell)<{ $boldUnderline: boolean }>`
background-color: ${props => props.theme.palette.common.greyLight};
text-decoration: ${props => props.$boldUnderline && "underline"};
font-weight: ${props => (props.$boldUnderline || !!props.color) && "600"};
`;
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
import React from "react";
import _ from "../../../../domain/entities/generic/Collection";
import { TableCell, TableRow } from "@material-ui/core";
import styled from "styled-components";
import { PerformanceOverviewTableProps } from "./PerformanceOverviewTable";

export type TableColumn = {
value: string;
label: string;
dark?: boolean;
};

type PercentTargetMetRowProps = {
columns: TableColumn[];
rows: {
[key: TableColumn["value"]]: string;
}[];
columnRules: {
[key: TableColumn["value"]]: number;
};
calculateColumns: TableColumn["value"][];
};

export const PercentTargetMetRow: React.FC<PercentTargetMetRowProps> = React.memo(
({ rows, columns, columnRules, calculateColumns }) => {
const calculatePercentTargetMet = (
rows: PerformanceOverviewTableProps["rows"],
column: TableColumn["value"],
target: number
) => {
const count = rows.filter(row => Number(row[column]) <= target).length;
const percentage = (count / rows.length) * 100 || 0;
return `${percentage.toFixed(0) || 0}%`;
};

return (
<TableRow>
{columns.map((column, columnIndex) => {
const rule = columnRules[column.value] || 7;

return (
<FooterTableCell
key={`percent-${column.value}`}
$boldUnderline={columnIndex === 0}
>
{columnIndex === 0 && "% Target Met"}

{calculateColumns.includes(column.value)
? calculatePercentTargetMet(rows, column.value, rule)
: ""}
</FooterTableCell>
);
})}
</TableRow>
);
}
);

const FooterTableCell = styled(TableCell)<{ $boldUnderline: boolean }>`
background-color: ${props => props.theme.palette.common.greyLight};
text-decoration: ${props => props.$boldUnderline && "underline"};
font-weight: ${props => (props.$boldUnderline || !!props.color) && "600"};
`;
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import React, { useEffect, useMemo, useState } from "react";
import React, { useMemo, useState } from "react";
import _ from "../../../../domain/entities/generic/Collection";
import {
Table,
Expand All @@ -13,14 +13,16 @@ import { SearchInput } from "../../search-input/SearchInput";
import { Selector } from "../../selector/Selector";
import { Maybe } from "../../../../utils/ts-utils";
import i18n from "../../../../utils/i18n";
import { MedianRow } from "./MedianRow";
import { PercentTargetMetRow } from "./PercentTargetMetRow";

export type TableColumn = {
value: string;
label: string;
dark?: boolean;
};

type PerformanceOverviewTableProps = {
export type PerformanceOverviewTableProps = {
columns: TableColumn[];
columnRules: {
[key: TableColumn["value"]]: number;
Expand Down Expand Up @@ -66,58 +68,6 @@ export const PerformanceOverviewTable: React.FC<PerformanceOverviewTableProps> =
return value <= rule ? "green" : "red";
};

const calculateMedian = (
rows: PerformanceOverviewTableProps["rows"],
column: TableColumn["value"]
) => {
const values = rows.map(row => Number(row[column])).filter(value => !isNaN(value));
values.sort((a, b) => a - b);
const mid = Math.floor(values.length / 2);
return values.length % 2 !== 0
? values[mid]
: ((values[mid - 1] || 0) + (values[mid] || 0)) / 2;
};

const calculatePercentTargetMet = (
rows: PerformanceOverviewTableProps["rows"],
column: TableColumn["value"],
target: number
) => {
const count = rows.filter(row => Number(row[column]) <= target).length;
const percentage = (count / rows.length) * 100 || 0;
return `${percentage.toFixed(0) || 0}%`;
};

const buildMedianRow = useMemo(() => {
return columns.map((column, columnIndex) => (
<FooterTableCell key={`median-${column.value}`} $boldUnderline={columnIndex === 0}>
{columnIndex === 0 && "Median"}
{calculateColumns.includes(column.value)
? calculateMedian(filteredRows, column.value)
: ""}
</FooterTableCell>
));
}, [filteredRows]);

const buildPercentTargetMetRow = useMemo(() => {
return columns.map((column, columnIndex) => {
const rule = columnRules[column.value] || 7;

return (
<FooterTableCell
key={`percent-${column.value}`}
$boldUnderline={columnIndex === 0}
>
{columnIndex === 0 && "% Target Met"}

{calculateColumns.includes(column.value)
? calculatePercentTargetMet(filteredRows, column.value, rule)
: ""}
</FooterTableCell>
);
});
}, [filteredRows]);

return (
<React.Fragment>
<Container>
Expand Down Expand Up @@ -162,8 +112,17 @@ export const PerformanceOverviewTable: React.FC<PerformanceOverviewTableProps> =
))}
</TableRow>
))}
<TableRow>{buildMedianRow}</TableRow>
<TableRow>{buildPercentTargetMetRow}</TableRow>
<MedianRow
columns={columns}
rows={filteredRows}
calculateColumns={calculateColumns}
/>
<PercentTargetMetRow
columns={columns}
rows={filteredRows}
calculateColumns={calculateColumns}
columnRules={columnRules}
/>
</TableBody>
</Table>
</StyledTableContainer>
Expand Down

0 comments on commit 411b40a

Please sign in to comment.