No puede seleccionar más de 25 temas Los temas deben comenzar con una letra o número, pueden incluir guiones ('-') y pueden tener hasta 35 caracteres de largo.
 
 
 

88 líneas
4.2 KiB

  1. 'use client';
  2. import { useState, useEffect } from 'react';
  3. import { Card } from '@/types';
  4. export default function PublicGrid({ cards, maxCols = 5 }: { cards: Card[], maxCols?: number }) {
  5. const [activeCard, setActiveCard] = useState<Card | null>(null);
  6. // Prevent background scrolling when modal is open
  7. useEffect(() => {
  8. if (activeCard) document.body.style.overflow = 'hidden';
  9. else document.body.style.overflow = 'unset';
  10. return () => { document.body.style.overflow = 'unset'; }
  11. }, [activeCard]);
  12. // Tailwind classes mapping based on the admin's chosen max columns
  13. const gridClasses: Record<number, string> = {
  14. 3: 'xl:grid-cols-3 lg:grid-cols-3 md:grid-cols-2 grid-cols-1', // ADDED THIS LINE
  15. 4: 'xl:grid-cols-4 lg:grid-cols-3 md:grid-cols-2 grid-cols-1',
  16. 5: 'xl:grid-cols-5 lg:grid-cols-4 md:grid-cols-3 grid-cols-1',
  17. 6: 'xl:grid-cols-6 lg:grid-cols-4 md:grid-cols-3 grid-cols-2',
  18. 7: 'xl:grid-cols-7 lg:grid-cols-5 md:grid-cols-4 grid-cols-2',
  19. 8: 'xl:grid-cols-8 lg:grid-cols-6 md:grid-cols-4 grid-cols-2',
  20. };
  21. const activeGridClass = gridClasses[maxCols] || gridClasses[5];
  22. return (
  23. <>
  24. <div className={`grid gap-4 ${activeGridClass}`}>
  25. {cards.map((card) => (
  26. <div
  27. key={card.id}
  28. onClick={() => setActiveCard(card)}
  29. className="group relative cursor-pointer overflow-hidden rounded-xl shadow-md aspect-square bg-gray-200 transition-all duration-300 hover:shadow-xl hover:-translate-y-1"
  30. >
  31. {card.imageUrl ? (
  32. <img src={card.imageUrl} alt={card.title} className="absolute inset-0 w-full h-full object-cover" />
  33. ) : (
  34. <div className="absolute inset-0 flex items-center justify-center bg-gradient-to-br from-blue-100 to-gray-200 text-gray-400">No Image</div>
  35. )}
  36. <div className="absolute inset-0 bg-gradient-to-t from-black/90 via-black/30 to-transparent flex flex-col justify-end p-5 text-white">
  37. <h3 className="text-xl font-bold drop-shadow-md">{card.title}</h3>
  38. <p className="text-sm opacity-0 group-hover:opacity-100 transition-opacity duration-300 line-clamp-2 mt-1 text-gray-200 drop-shadow">
  39. {card.shortDescription}
  40. </p>
  41. </div>
  42. </div>
  43. ))}
  44. </div>
  45. {/* Improved Modal Pop-up */}
  46. {activeCard && (
  47. <div
  48. className="fixed inset-0 z-50 flex items-center justify-center bg-black/70 backdrop-blur-md p-4 transition-opacity"
  49. onClick={() => setActiveCard(null)} // Click outside to close
  50. >
  51. <div
  52. className="bg-white rounded-2xl w-full max-w-2xl max-h-[90vh] overflow-y-auto shadow-2xl animate-in fade-in zoom-in-95 duration-200"
  53. onClick={(e) => e.stopPropagation()} // Prevent clicks inside modal from closing it
  54. >
  55. <div className="relative h-72 w-full bg-gray-100">
  56. {activeCard.imageUrl && (
  57. <img src={activeCard.imageUrl} className="w-full h-full object-cover rounded-t-2xl" alt="" />
  58. )}
  59. {/* Improved Close Button */}
  60. <button
  61. onClick={() => setActiveCard(null)}
  62. className="absolute top-4 right-4 bg-black/60 text-white w-10 h-10 flex items-center justify-center rounded-full hover:bg-black hover:scale-110 transition-all shadow-lg"
  63. title="Close"
  64. >
  65. </button>
  66. </div>
  67. <div className="p-8">
  68. <div className="text-xs text-blue-600 font-bold tracking-wider uppercase mb-2">{activeCard.cardType.replace('_', ' ')}</div>
  69. <h2 className="text-3xl font-bold mb-4 text-gray-900">{activeCard.title}</h2>
  70. {activeCard.fullContent ? (
  71. <div className="prose max-w-none text-gray-700" dangerouslySetInnerHTML={{ __html: activeCard.fullContent }} />
  72. ) : (
  73. <p className="text-gray-500 italic text-lg">{activeCard.shortDescription}</p>
  74. )}
  75. </div>
  76. </div>
  77. </div>
  78. )}
  79. </>
  80. );
  81. }