Skip to content
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
97d852d
feat: add hint object to guide users in quality management
Lasserich Aug 20, 2025
e5d5796
feat: populate crossplane and flux with data
Lasserich Aug 20, 2025
319d67d
fix: make linitii happii
Lasserich Aug 20, 2025
8263c6f
feat: use new types
Lasserich Aug 21, 2025
81a95ec
fix: I love AI sometimes
Lasserich Aug 21, 2025
5f5620d
feat: expand radar chart
Lasserich Aug 21, 2025
0a15232
Merge branch 'main' into pr/251
lucasgoral Aug 21, 2025
ba9de76
Update listManagedResources.ts
lucasgoral Aug 21, 2025
4ad9302
Update Hints.tsx
lucasgoral Aug 21, 2025
1cc1189
fix: smaller stuff
Lasserich Aug 21, 2025
e0cb0ef
Merge branch 'ft/hints' of https://github.com/Lasserich/ui-frontend i…
Lasserich Aug 21, 2025
26001d9
feat: fix darkmode
Lasserich Aug 21, 2025
4939d67
Merge branch 'main' into ft/hints
Lasserich Aug 21, 2025
82a4341
feat: nice bars!!
Lasserich Aug 22, 2025
79b952c
feat: generify
Lasserich Aug 22, 2025
7fbd8cd
fix: cleanup
Lasserich Aug 22, 2025
ccf9035
feat: cleanup
Lasserich Aug 22, 2025
ccc3686
test: added some
Lasserich Aug 22, 2025
fee5f26
fix: darkmode
Lasserich Aug 22, 2025
fadf3ac
fix: linting
Lasserich Aug 25, 2025
5dace2d
fix: on activate
Lasserich Aug 25, 2025
291c54e
feat: re-usable hover
Lasserich Aug 25, 2025
b2f9f93
feat: crossplane and flux component ready for review
Lasserich Aug 25, 2025
8b89e31
fix: disable vault for now since it does not have stats yet
Lasserich Aug 25, 2025
2b45192
Merge branch 'main' into ft/hints
lucasgoral Aug 25, 2025
bd550d9
feat: incorporate feedback
Lasserich Aug 27, 2025
874a80b
Merge branch 'ft/hints' of https://github.com/Lasserich/ui-frontend i…
Lasserich Aug 27, 2025
aa745ad
fix: npm run lint:fix :)(:
Lasserich Aug 27, 2025
530a862
fix: zIndex
Lasserich Aug 27, 2025
9bdb8b2
fix: remove info, coming soon, downsize vault
Lasserich Aug 27, 2025
4978b76
fix: build
Lasserich Aug 27, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
297 changes: 296 additions & 1 deletion package-lock.json

Large diffs are not rendered by default.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
"@ui5/webcomponents-fiori": "^2.7.2",
"@ui5/webcomponents-icons": "^2.7.2",
"@ui5/webcomponents-react": "^2.7.2",
"@ui5/webcomponents-react-charts": "^2.13.1",
"@xyflow/react": "^12.8.2",
"clsx": "^2.1.1",
"dagre": "^0.8.5",
Expand Down
56 changes: 55 additions & 1 deletion public/locales/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -172,6 +172,7 @@
"defaultNamespaceInfo": "Leave empty to use <span>default</span> namespace",
"serviceAccoutsGuide": "You can also use our <link1>Service Account Guide</link1> for more information."
},

"ProjectsPage": {
"header": "Your instances of <span>ManagedControlPlane</span>",
"projectHeader": "Project:"
Expand All @@ -185,6 +186,7 @@
"McpPage": {
"accessError": "Managed Control Plane does not have access information yet",
"componentsTitle": "Components",
"overviewTitle": "Overview",
"crossplaneTitle": "Crossplane",
"gitOpsTitle": "GitOps",
"landscapersTitle": "Landscapers",
Expand Down Expand Up @@ -334,7 +336,12 @@
"synced": "Synced",
"healthy": "Healthy",
"installed": "Installed",
"none": "None"
"none": "None",
"creating": "Creating",
"unhealthy": "Unhealthy",
"progress": "Managed",
"remaining": "Remaining",
"active": "Active"
},
"errors": {
"installError": "Install error",
Expand Down Expand Up @@ -366,5 +373,52 @@
"selectedComponents": "Selected Components",
"pleaseSelectComponents": "Choose the components you want to add to your Managed Control Plane.",
"cannotLoad": "Cannot load components list"
},
"Hints": {
"CrossplaneHint": {
"title": "Crossplane",
"subtitle": "Managed Resources Readiness",
"activeStatus": "Active v",
"progressAvailable": "% Available",
"noResources": "No Resources",
"inactive": "Inactive",
"activate": "Activate",
"healthy": "Healthy",
"hoverContent": {
"totalResources": "Total Resources",
"healthy": "Healthy",
"creating": "Creating",
"failing": "Failing"
}
},
"GitOpsHint": {
"title": "Flux",
"subtitle": "GitOps Progress",
"activeStatus": "Active v",
"progressAvailable": "% Available",
"noResources": "No Resources",
"inactive": "Inactive",
"activate": "Activate",
"managed": "Managed",
"hoverContent": {
"totalResources": "Total Resources",
"managed": "Managed",
"unmanaged": "Unmanaged"
}
},
"VaultHint": {
"title": "Vault",
"subtitle": "Rotating Secrets Progress",
"activeStatus": "Active v",
"progressAvailable": "% Available",
"noResources": "No Resources",
"inactive": "Inactive",
"activate": "Activate"
},
"common": {
"loading": "Loading...",
"errorLoadingResources": "Error loading resources",
"activate": "Activate"
}
}
}
Binary file added public/vault.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
/* Card Hover Content Styles */
.hoverContent {
width: 100%;
display: flex;
flex-direction: column;
align-items: center;
margin: 1rem 0;
overflow: visible;
}

.chartContainer {
width: 100%;
height: 300px;
display: flex;
justify-content: center;
align-items: center;
}
129 changes: 129 additions & 0 deletions src/components/HintsCardsRow/CardHoverContent/CardHoverContent.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,129 @@
import React, { useMemo } from 'react';
import { RadarChart } from '@ui5/webcomponents-react-charts';
import { LegendSection } from '../LegendSection/LegendSection';
import { styles } from '../HintsCardsRow';
import styles2 from './CardHoverContent.module.css';
import cx from 'clsx';

export interface LegendItem {
label: string;
count: number;
color: string;
}

export interface RadarDataPoint {
[key: string]: string | number;
}

export interface RadarMeasure {
accessor: string;
color: string;
hideDataLabel?: boolean;
label: string;
}

export interface RadarDimension {
accessor: string;
}

export interface HoverContentProps {
enabled: boolean;
totalCount: number;
totalLabel: string;
legendItems: LegendItem[];
radarDataset: RadarDataPoint[];
radarDimensions: RadarDimension[];
radarMeasures: RadarMeasure[];
isLoading?: boolean;
}

// Helper function to truncate labels to max 13 characters
const truncateLabel = (label: string, maxLength: number = 13): string => {
if (label.length <= maxLength) {
return label;
}
return label.substring(0, maxLength) + '...';
};

export const HoverContent: React.FC<HoverContentProps> = ({
enabled,
totalCount,
totalLabel,
legendItems,
radarDataset,
radarDimensions,
radarMeasures,
isLoading = false,
}) => {
// Process the dataset to truncate labels
const processedDataset = useMemo(() => {
return radarDataset.map((dataPoint) => {
const processedDataPoint = { ...dataPoint };

// Truncate labels for each dimension accessor
radarDimensions.forEach((dimension) => {
const value = dataPoint[dimension.accessor];
if (typeof value === 'string') {
processedDataPoint[dimension.accessor] = truncateLabel(value);
}
});

return processedDataPoint;
});
}, [radarDataset, radarDimensions]);

if (!enabled) {
return null;
}

return (
<div className={cx(styles.hoverContent, styles2.hoverContent)}>
<LegendSection title={`${totalCount} ${totalLabel}`} items={legendItems} />
<div className={styles2.chartContainer}>
{isLoading || radarDataset.length === 0 ? (
<div className={cx(styles.hoverContentLoading)}>
<RadarChart
dataset={[]}
dimensions={[
{
accessor: 'name',
formatter: (value: string | number) => String(value || ''),
},
]}
measures={[
{
accessor: 'users',
formatter: (value: string | number) => String(value || ''),
label: 'Users',
},
{
accessor: 'sessions',
formatter: (value: string | number) => String(value || ''),
hideDataLabel: true,
label: 'Active Sessions',
},
{
accessor: 'volume',
label: 'Vol.',
},
]}
style={{ width: '100%', height: '100%', minWidth: 280, minHeight: 280 }}
noLegend={true}
onClick={() => {}}
onDataPointClick={() => {}}
onLegendClick={() => {}}
/>
</div>
) : (
<RadarChart
dataset={processedDataset}
dimensions={radarDimensions}
measures={radarMeasures}
style={{ width: '100%', height: '100%', minWidth: 280, minHeight: 280 }}
noLegend={true}
/>
)}
</div>
</div>
);
};
85 changes: 85 additions & 0 deletions src/components/HintsCardsRow/CardHoverContent/hoverCalculations.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
import { ManagedResourceItem, Condition } from '../../../lib/shared/types';

export interface ResourceTypeStats {
type: string;
total: number;
healthy: number;
creating: number;
unhealthy: number;
healthyPercentage: number;
creatingPercentage: number;
unhealthyPercentage: number;
}

export interface OverallStats {
total: number;
healthy: number;
creating: number;
unhealthy: number;
}

export interface CrossplaneHoverData {
resourceTypeStats: ResourceTypeStats[];
overallStats: OverallStats;
}

/**
* Calculate comprehensive statistics for Crossplane hover content
*/
export const calculateCrossplaneHoverData = (allItems: ManagedResourceItem[]): CrossplaneHoverData => {
const typeStats: Record<string, { total: number; healthy: number; creating: number; unhealthy: number }> = {};
let totalHealthy = 0;
let totalCreating = 0;
let totalUnhealthy = 0;

allItems.forEach((item: ManagedResourceItem) => {
const type = item.kind || 'Unknown';

if (!typeStats[type]) {
typeStats[type] = { total: 0, healthy: 0, creating: 0, unhealthy: 0 };
}

typeStats[type].total++;

const conditions = item.status?.conditions || [];
const ready = conditions.find((c: Condition) => c.type === 'Ready' && c.status === 'True');
const synced = conditions.find((c: Condition) => c.type === 'Synced' && c.status === 'True');

if (ready && synced) {
typeStats[type].healthy++;
totalHealthy++;
} else if (synced && !ready) {
// Resource is synced but not ready - it's creating
typeStats[type].creating++;
totalCreating++;
} else {
// Resource has issues or is not synced
typeStats[type].unhealthy++;
totalUnhealthy++;
}
});

const resourceTypeStats: ResourceTypeStats[] = Object.keys(typeStats).map((type) => {
const stats = typeStats[type];
return {
type,
total: stats.total,
healthy: stats.healthy,
creating: stats.creating,
unhealthy: stats.unhealthy,
healthyPercentage: Math.round((stats.healthy / stats.total) * 100),
creatingPercentage: Math.round((stats.creating / stats.total) * 100),
unhealthyPercentage: Math.round((stats.unhealthy / stats.total) * 100),
};
});

return {
resourceTypeStats,
overallStats: {
total: allItems.length,
healthy: totalHealthy,
creating: totalCreating,
unhealthy: totalUnhealthy,
},
};
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
.container {
position: relative;
width: 100%;
}

.avatar {
width: 50px;
height: 50px;
border-radius: 50%;
background: transparent;
object-fit: cover;
}

.contentContainer {
display: flex;
flex-direction: column;
align-items: center;
padding: 0.5rem 0;
}

.progressBarContainer {
display: flex;
gap: 8px;
width: 100%;
max-width: 500px;
padding: 0 0.5rem;
}

.progressBar {
width: 100%;
}

.activateButton {
position: absolute;
top: 16px;
right: 16px;
z-index: 2;
pointer-events: auto;
}

.activateButtonClickable {
cursor: pointer;
}

.activateButtonDefault {
cursor: default;
}
Loading
Loading