All files / src/components/common ResourceCard.tsx

81.35% Statements 48/59
45.45% Branches 5/11
50% Functions 2/4
81.35% Lines 48/59

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                      1x 8x 8x 8x   1x 8x 8x 8x 8x 8x 8x               8x 8x 8x 8x 8x 8x 8x         8x   8x 8x 8x 8x 8x 8x 8x 8x   8x 8x 8x   8x 8x     8x 8x   8x 8x   8x 8x 21x 21x 21x   21x 21x 8x 8x 8x   8x   1x  
import React from "react";
import { SecurityResource } from "../../types/securityResources";
 
interface ResourceCardProps {
  resource: SecurityResource;
  onClick?: (resource: SecurityResource) => void;
  className?: string;
  testId?: string;
}
 
// Helper function to truncate text with ellipsis
const truncateText = (text: string | undefined, maxLength: number): string => {
  if (!text) return "";
  return text.length > maxLength ? `${text.substring(0, maxLength)}...` : text;
};
 
const ResourceCard: React.FC<ResourceCardProps> = ({
  resource,
  onClick,
  className = "",
  testId,
}) => {
  const handleClick = () => {
    if (onClick) {
      onClick(resource);
    } else {
      window.open(resource.url, "_blank", "noopener,noreferrer");
    }
  };
 
  return (
    <div
      className={`bg-white dark:bg-gray-800 rounded-lg shadow-md p-4 mb-4 hover:shadow-lg transition-shadow ${className}`}
      onClick={handleClick}
      role="button"
      tabIndex={0}
      onKeyDown={(e) => {
        if (e.key === "Enter" || e.key === " ") {
          handleClick();
        }
      }}
      data-testid={testId || "resource-item"}
    >
      <div className="flex justify-between items-start mb-2">
        <h3 className="text-lg font-semibold text-gray-800 dark:text-gray-100">
          {resource.title}
        </h3>
        <div className="text-xs bg-blue-100 dark:bg-blue-900 text-blue-800 dark:text-blue-100 px-2 py-1 rounded">
          {resource.type || "General"}
        </div>
      </div>
 
      <p className="text-sm text-gray-600 dark:text-gray-300 mb-3">
        {truncateText(resource.description || "", 100)}
      </p>
 
      <div className="mt-2 text-xs text-gray-500 dark:text-gray-400">
        {resource.component && (
          <span className="mr-2">Component: {resource.component}</span>
        )}
        {resource.source && (
          <span className="mr-2">Source: {resource.source}</span>
        )}
        {resource.level && <span>Level: {resource.level}</span>}
      </div>
 
      <div className="mt-2 flex flex-wrap">
        {resource.tags?.map((tag, index) => (
          <span
            key={index}
            className="text-xs bg-gray-100 dark:bg-gray-700 text-gray-800 dark:text-gray-300 px-2 py-1 rounded mr-1 mb-1"
          >
            {tag}
          </span>
        ))}
      </div>
    </div>
  );
};
 
export default ResourceCard;