From 93a34c4b8fa1cc4a9d9ba9c814056df126af787f Mon Sep 17 00:00:00 2001 From: Ehsan Date: Fri, 23 Aug 2024 18:45:29 +0000 Subject: [PATCH 1/2] Accordion style presentation of composite resources. --- .../src/components/CompositeResourcesList.tsx | 86 +++++++++++++++++-- 1 file changed, 80 insertions(+), 6 deletions(-) diff --git a/pkg/frontend/src/components/CompositeResourcesList.tsx b/pkg/frontend/src/components/CompositeResourcesList.tsx index 92f0cc9..aba70f3 100644 --- a/pkg/frontend/src/components/CompositeResourcesList.tsx +++ b/pkg/frontend/src/components/CompositeResourcesList.tsx @@ -1,4 +1,5 @@ -import {Card, CardContent, Grid} from '@mui/material'; +import {Box, Alert, Stack, List, Card, CardContent, Grid, Accordion, AccordionSummary, AccordionDetails} from '@mui/material'; +import {ExpandMore as ExpandMoreIcon} from '@mui/icons-material'; import {CompositeResource, CompositeResourceExtended, ItemList, K8sReference, K8sResource} from "../types.ts"; import Typography from "@mui/material/Typography"; import ReadySynced from "./ReadySynced.tsx"; @@ -98,13 +99,86 @@ export default function CompositeResourcesList({items}: ItemListProps) { ) } + // Define Grouped Items + const groupedItems: { [itemIndex: string]: CompositeResource[] } = {}; + items.items.forEach((item) => { + const itemIndex = item.kind + if (!groupedItems[itemIndex]) { + groupedItems[itemIndex] = []; + } + groupedItems[itemIndex].push(item); + }); + + // Accordion State + const [expandedItems, setExpandedItems] = useState<{[itemIndex: string]: boolean}>({}); + const handleAccordionChange = (itemIndex: string) => { + setExpandedItems((prevState) => ({ + ...prevState, + [itemIndex]: !prevState[itemIndex], + })); + }; + return ( <> - - {items?.items?.map((item: CompositeResource) => ( - - ))} - + {Object.entries(groupedItems).map(([itemIndex, items]) => ( + + handleAccordionChange(itemIndex)}> + }> + + {itemIndex} + + *': { + py: '4px !important', + },}} + severity="success"> + Ready: {items.filter((item) => item.status?.conditions?.find((condition) => + condition.status === "True" && condition.type === "Ready")).length} + + + { + items.filter((item) => !item.status?.conditions?.find((condition) => + condition.status === "True" && condition.type === "Ready")).length > 0 ? ( + + *': { + py: '4px !important', + },}} + severity="error" color="warning"> + Not Ready: {items.filter((item) => !item.status?.conditions?.find((condition) => + condition.status === "True" && condition.type === "Ready")).length} + + + ) : null + } + { + items.filter((item) => !item.status?.conditions?.find((condition) => + condition.status === "True" && condition.type === "Synced")).length > 0 ? ( + + *': { + py: '4px !important', + },}} + severity="info" color="info"> + Not Synced: {items.filter((item) => !item.status?.conditions?.find((condition) => + condition.status === "True" && condition.type === "Sync")).length} + + + ) : null + } + + + + + {items.map((item: CompositeResource) => ( + + ))} + + + + + ))} From 72b5032a0c18b74c30ee42be5333b331faec4508 Mon Sep 17 00:00:00 2001 From: Ehsan Date: Sun, 25 Aug 2024 23:14:42 +0000 Subject: [PATCH 2/2] Resolve linter issue: call useState hook conditionally --- pkg/frontend/src/components/CompositeResourcesList.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/frontend/src/components/CompositeResourcesList.tsx b/pkg/frontend/src/components/CompositeResourcesList.tsx index aba70f3..74bdfc6 100644 --- a/pkg/frontend/src/components/CompositeResourcesList.tsx +++ b/pkg/frontend/src/components/CompositeResourcesList.tsx @@ -45,6 +45,7 @@ export default function CompositeResourcesList({items}: ItemListProps) { const [isDrawerOpen, setDrawerOpen] = useState(focusedName != undefined); const nullFocused = {metadata: {name: ""}, kind: "", apiVersion: ""}; const [focused, setFocused] = useState(nullFocused); + const [expandedItems, setExpandedItems] = useState<{[itemIndex: string]: boolean}>({}); const navigate = useNavigate(); const onClose = () => { @@ -110,7 +111,6 @@ export default function CompositeResourcesList({items}: ItemListProps) { }); // Accordion State - const [expandedItems, setExpandedItems] = useState<{[itemIndex: string]: boolean}>({}); const handleAccordionChange = (itemIndex: string) => { setExpandedItems((prevState) => ({ ...prevState,