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 | 5x 8x 8x 5x 8x 8x 22x | 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 => {
Iif (!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;
|