Replace entire Barclays colour palette (navy #1A2142, lime #C3FB5A, violet #7A0FF9) with Oliver brand tokens: black #1A1A1A, gold #FFCB05, orange #FF5C00, azure #0487B6, sky #5DF5EA, grey #EFEFEF, green #09821F. - Switch font from Inter/Barclays Effra to Arial (system font) - Add new Oliver logo asset (BAR-ModComms-logo-v4.png) - Sidebar: black background, new logo, azure active state - Hero: orange "Intelligent Review" text, hide AI-Powered tagline - Hide ChecksOverview on Home page per Oliver design - Toast notification: orange background with black text - All tables: sky headers, alternating white/grey rows - Campaign badges: gold "In Progress", green "Completed" - Analytics: grey KPI cards, sky accent on Key Insight, oliver trend colours - All buttons: azure fill, pill-shaped (rounded-full) - All tabs/toggles/dropdowns: azure accent colour - Update HTML title to "Mod Comms - Intelligent Review" - Default border radius set to 10px Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
124 lines
No EOL
5.6 KiB
TypeScript
Executable file
124 lines
No EOL
5.6 KiB
TypeScript
Executable file
import React from 'react';
|
|
import type { ReviewStatus, AgentName, AgentStatus } from '../types';
|
|
import { AGENT_NAMES } from '../constants';
|
|
import { LegalIcon } from './icons/LegalIcon';
|
|
import { BrandIcon } from './icons/BrandIcon';
|
|
import { CheckCircleIcon, ExclamationTriangleIcon, InformationCircleIcon } from './icons/StatusIcons';
|
|
import { ChannelIcon } from './icons/ChannelIcon';
|
|
import { SpinnerIcon } from './icons/SpinnerIcon';
|
|
|
|
interface StatusDashboardProps {
|
|
status: ReviewStatus;
|
|
variant?: 'default' | 'hero';
|
|
}
|
|
|
|
interface StatusInfoProps {
|
|
status: string;
|
|
isHeroVariant?: boolean;
|
|
}
|
|
|
|
// FIX: Storing components instead of instantiated elements to avoid React.cloneElement type issues.
|
|
const agentIcons: Record<AgentName, React.FC<React.SVGProps<SVGSVGElement>>> = {
|
|
'Legal Agent': LegalIcon,
|
|
'Brand Agent': BrandIcon,
|
|
'Channel Best Practices Agent': ChannelIcon,
|
|
'Channel Tech Specs Agent': ChannelIcon,
|
|
};
|
|
|
|
const StatusInfo: React.FC<StatusInfoProps> = ({ status, isHeroVariant = false }) => {
|
|
const colors = isHeroVariant ? {
|
|
complete: 'text-success-light',
|
|
issues: 'text-warning-light',
|
|
error: 'text-error-light',
|
|
pending: 'text-grey-300'
|
|
} : {
|
|
complete: 'text-success',
|
|
issues: 'text-warning',
|
|
error: 'text-error',
|
|
pending: 'text-oliver-black/60'
|
|
};
|
|
|
|
switch (status) {
|
|
case 'complete':
|
|
return <div className={`flex items-center ${colors.complete}`}><CheckCircleIcon className="h-5 w-5 mr-2" /> Pass</div>;
|
|
case 'issues-found':
|
|
return <div className={`flex items-center ${colors.issues}`}><ExclamationTriangleIcon className="h-5 w-5 mr-2" /> Issues Found</div>;
|
|
case 'error':
|
|
return <div className={`flex items-center ${colors.error}`}><InformationCircleIcon className="h-5 w-5 mr-2" /> Error</div>;
|
|
default:
|
|
return <div className={`flex items-center ${colors.pending}`}><InformationCircleIcon className="h-5 w-5 mr-2" /> Pending</div>;
|
|
}
|
|
};
|
|
|
|
export const StatusDashboard: React.FC<StatusDashboardProps> = ({ status, variant = 'default' }) => {
|
|
if (Object.keys(status).length === 0) return null;
|
|
|
|
const isHeroVariant = variant === 'hero';
|
|
|
|
const getTileClassName = (agentStatus: AgentStatus, isHero: boolean): string => {
|
|
const baseClasses = 'rounded-[10px] flex flex-col text-center transition-colors duration-500 ease-in-out';
|
|
|
|
if (isHero) {
|
|
const heroBase = `${baseClasses} p-4`;
|
|
switch (agentStatus) {
|
|
case 'complete':
|
|
return `${heroBase} bg-success/20`;
|
|
case 'issues-found':
|
|
return `${heroBase} bg-warning/20`;
|
|
case 'error':
|
|
return `${heroBase} bg-error/20`;
|
|
default: // 'pending' or 'in-progress'
|
|
return `${heroBase} bg-white/10`;
|
|
}
|
|
} else {
|
|
const defaultBase = `${baseClasses} p-6 shadow-lg border`;
|
|
switch (agentStatus) {
|
|
case 'complete':
|
|
return `${defaultBase} bg-success-light border-success`;
|
|
case 'issues-found':
|
|
return `${defaultBase} bg-warning-light border-warning`;
|
|
case 'error':
|
|
return `${defaultBase} bg-error-light border-error`;
|
|
default: // 'pending' or 'in-progress'
|
|
return `${defaultBase} bg-oliver-grey border-grey-300`;
|
|
}
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className={isHeroVariant ? '' : "mb-12"}>
|
|
<h3 className={`text-2xl font-bold text-center mb-8 ${isHeroVariant ? 'text-white' : 'text-oliver-black'}`}>Review Progress</h3>
|
|
<div className={`grid grid-cols-2 sm:grid-cols-3 md:grid-cols-5 ${isHeroVariant ? 'gap-4' : 'gap-6'}`}>
|
|
{AGENT_NAMES.map((agentName) => {
|
|
const agentStatus = status[agentName] || 'pending';
|
|
const iconColor = isHeroVariant ? 'text-oliver-azure' : 'text-oliver-azure';
|
|
const Icon = agentIcons[agentName];
|
|
|
|
return (
|
|
<div key={agentName} className={getTileClassName(agentStatus, isHeroVariant)}>
|
|
<div className="flex-grow flex flex-col items-center justify-center">
|
|
<div className="mb-3 h-8 w-8 flex items-center justify-center">
|
|
<Icon className={`h-8 w-8 ${iconColor}`} />
|
|
</div>
|
|
<h4 className={`text-base font-bold ${isHeroVariant ? 'text-white' : 'text-oliver-black'}`}>{agentName}</h4>
|
|
</div>
|
|
<div className="font-semibold text-xs mt-2 h-6 flex items-center justify-center">
|
|
{agentStatus === 'in-progress' ? (
|
|
<div className={`flex items-center ${isHeroVariant ? 'text-white/80' : 'text-oliver-black/60'}`}>
|
|
<SpinnerIcon className="h-4 w-4 mr-2 custom-spinner" />
|
|
<span>Analysing...</span>
|
|
</div>
|
|
) : (
|
|
<StatusInfo
|
|
status={agentStatus}
|
|
isHeroVariant={isHeroVariant}
|
|
/>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
);
|
|
}; |