All files / src/components/securitylevel Selection.tsx

90.52% Statements 86/95
85.71% Branches 12/14
100% Functions 3/3
90.52% Lines 86/95

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 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 1471x                                                             1x 14x 14x 14x 14x 14x 14x 14x 14x 14x 14x 14x 14x 14x 14x 14x   14x     14x 1x 1x     14x 2x 2x   14x 14x 14x 14x                 14x 14x 14x 14x 13x 13x 13x 13x 13x 13x   13x 13x   14x   14x           14x 14x 14x 14x 14x 14x 14x 14x 14x   14x 65x 65x 65x 14x 14x 14x 14x 14x 14x 14x 14x 14x   14x 14x 14x 14x 14x 14x 14x 14x   14x 13x 13x 13x     14x 1x 1x 1x 1x 1x 1x   14x   14x   1x  
import React, { useState } from "react";
 
/**
 * Props for the Selection component
 */
interface SelectionProps {
  id: string;
  label: string;
  icon?: string;
  description?: string;
  options: Array<{ value: string; label: string }>;
  value: string;
  onChange: (value: string) => void;
  iconClassName?: string;
  labelClassName?: string;
  infoContent?: string;
  contextInfo?: string;
  disabled?: boolean;
  "aria-label"?: string;
  "data-testid"?: string;
}
 
/**
 * Component for selecting security levels with consistent styling
 *
 * ## Business Perspective
 *
 * This component provides a standardized interface for selecting security levels,
 * with additional context information to help users understand the implications
 * of their choices. 🔒
 */
export const Selection: React.FC<SelectionProps> = ({
  id,
  label,
  icon,
  description,
  options,
  value,
  onChange,
  iconClassName = "",
  labelClassName = "",
  infoContent,
  contextInfo,
  disabled = false,
  "aria-label": ariaLabel,
  "data-testid": testId,
}) => {
  // State for showing/hiding info content
  const [showInfo, setShowInfo] = useState(false);
 
  // Handle selection change
  const handleChange = (e: React.ChangeEvent<HTMLSelectElement>) => {
    onChange(e.target.value);
  };
 
  // Toggle info panel
  const toggleInfo = () => {
    setShowInfo(!showInfo);
  };
 
  return (
    <div className="selection-component" data-testid={testId}>
      <div className="flex items-center mb-2">
        {icon && (
          <span
            className={`text-xl mr-2 ${iconClassName}`}
            role="img"
            aria-hidden="true"
          >
            {icon}
          </span>
        )}
        <label htmlFor={id} className={`font-medium ${labelClassName}`}>
          {label}
        </label>
        {infoContent && (
          <button
            type="button"
            onClick={toggleInfo}
            className="ml-2 text-sm text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-300 focus:outline-none"
            aria-expanded={showInfo}
            aria-label={`${showInfo ? "Hide" : "Show"} ${label} information`}
          >
            {showInfo ? "â–²" : "â–¼"}
          </button>
        )}
      </div>
 
      {description && (
        <p className="text-sm text-gray-600 dark:text-gray-400 mb-2">
          {description}
        </p>
      )}
 
      <div className="relative">
        <select
          id={id}
          className="block w-full px-4 py-2 pr-8 border border-gray-300 dark:border-gray-600 bg-white dark:bg-gray-800 rounded-md shadow-sm focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-blue-500 transition-colors"
          value={value}
          onChange={handleChange}
          disabled={disabled}
          aria-label={ariaLabel}
          data-testid={`${testId}-select`}
        >
          {options.map((option) => (
            <option key={option.value} value={option.value}>
              {option.label}
            </option>
          ))}
        </select>
        <div className="absolute inset-y-0 right-0 flex items-center px-2 pointer-events-none">
          <svg
            className="h-5 w-5 text-gray-400"
            xmlns="http://www.w3.org/2000/svg"
            viewBox="0 0 20 20"
            fill="currentColor"
            aria-hidden="true"
          >
            <path
              fillRule="evenodd"
              d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z"
              clipRule="evenodd"
            />
          </svg>
        </div>
      </div>
 
      {contextInfo && (
        <div className="mt-2 text-sm text-gray-600 dark:text-gray-400">
          {contextInfo}
        </div>
      )}
 
      {showInfo && infoContent && (
        <div className="mt-3 p-3 bg-gray-50 dark:bg-gray-700 border border-gray-200 dark:border-gray-600 rounded-md">
          <h4 className="text-sm font-medium mb-1">Implementation Details</h4>
          <p className="text-sm text-gray-600 dark:text-gray-300">
            {infoContent}
          </p>
        </div>
      )}
    </div>
  );
};
 
export default Selection;