Press n or j to go to the next uncovered block, b, p or k for the previous block.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 | import React, { useMemo } from "react"; interface SecurityRiskScoreProps { score: number; maxScore?: number; label: string; className?: string; testId?: string; } /** * Displays a visual security score indicator * * ## Business Perspective * * This component provides a quantitative measure of security posture in an * easy-to-understand format. The numerical score and color-coding help * business stakeholders quickly gauge security maturity and track improvements * over time, supporting data-driven security investment decisions. 📈 * * @param props Component props * @returns React Element */ function SecurityRiskScore({ score, maxScore = 100, label, className = "", testId, }: SecurityRiskScoreProps): React.ReactElement { // Normalize score as a percentage const normalizedScore = useMemo(() => { if (score <= 0) return 0; if (score >= maxScore) return 100; return (score / maxScore) * 100; }, [score, maxScore]); // Determine color based on score range const scoreColor = useMemo(() => { if (normalizedScore >= 75) return "text-green-500 dark:text-green-400"; if (normalizedScore >= 50) return "text-yellow-500 dark:text-yellow-400"; return "text-red-500 dark:text-red-400"; }, [normalizedScore]); return ( <div className={`flex flex-col items-center ${className}`} data-testid={testId} > <div className="relative"> <svg width="64" height="64" viewBox="0 0 64 64"> {/* Background circle */} <circle cx="32" cy="32" r="28" fill="none" stroke="#e5e7eb" strokeWidth="8" className="dark:stroke-gray-700" /> {/* Score indicator */} <circle cx="32" cy="32" r="28" fill="none" strokeLinecap="round" stroke="currentColor" strokeWidth="8" strokeDasharray={`${normalizedScore * 1.76} 176`} transform="rotate(-90 32 32)" className={scoreColor} /> </svg> <div className="absolute inset-0 flex items-center justify-center"> <span className={`text-lg font-bold ${scoreColor}`}> {Math.round(normalizedScore)} </span> </div> </div> <span className="text-xs text-gray-600 dark:text-gray-400 mt-1"> {label} </span> </div> ); } export { SecurityRiskScore }; export default SecurityRiskScore; |