Files
Elecciones-2025/Elecciones-Web/frontend/src/components/SenateLayout.tsx

140 lines
8.5 KiB
TypeScript
Raw Normal View History

// src/components/SenateLayout.tsx
import React from 'react';
// Interfaces
interface SeatFillData {
color: string;
// Añadimos la propiedad opcional 'ocupante'
ocupante?: {
nombreOcupante: string;
fotoUrl: string | null;
} | null;
}
interface SenateLayoutProps {
seatData: SeatFillData[];
size?: number;
presidenteBancada: { color: string | null } | null;
}
const PRESIDENTE_SEAT_INDEX = 45; // El último asiento (índice 45 de 46)
export const SenateLayout: React.FC<SenateLayoutProps> = ({
seatData,
size = 400,
presidenteBancada,
}) => {
const uniqueColors = [...new Set(seatData.map(d => d.color).filter(Boolean))];
// --- NUEVO ARRAY DE ELEMENTOS ORDENADO PARA EL SENADO ---
const seatElements = [
<circle key="senate-seat-0" id="senate-seat-0" r="9" cy="354.956" cx="-121.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-1" id="senate-seat-1" r="9" cy="354.956" cx="-164.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-2" id="senate-seat-2" r="9" cy="331.318" cx="-121.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-3" id="senate-seat-3" r="9" cy="333.318" cx="-164.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-4" id="senate-seat-4" r="9" cy="307.598" cx="-121.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-5" id="senate-seat-5" r="9" cy="312.598" cx="-164.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-6" id="senate-seat-6" r="9" cy="285.358" cx="-121.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-7" id="senate-seat-7" r="9" cy="292.358" cx="-164.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-8" id="senate-seat-8" r="9" cy="292.358" cx="-206.672" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-9" id="senate-seat-9" r="9" cy="261.241" cx="-121.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-10" id="senate-seat-10" r="9" cy="269.241" cx="-165.933" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-11" id="senate-seat-11" r="9" cy="268.057" cx="-208.672" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-12" id="senate-seat-12" r="9" cy="184.514" cx="327.2" transform="matrix(-0.986454, -0.16404, -0.16404, 0.986454, 488.237, 84.441)" />,
<circle key="senate-seat-13" id="senate-seat-13" r="9" cy="246.521" cx="-170.309" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-14" id="senate-seat-14" r="9" cy="184.514" cx="327.2" transform="matrix(-0.986454, -0.16404, -0.16404, 0.986454, 507.83, 65.575)" />,
<circle key="senate-seat-15" id="senate-seat-15" r="9" cy="229.407" cx="-183.766" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-16" id="senate-seat-16" r="9" cy="184.514" cx="327.2" transform="matrix(-0.986454, -0.16404, -0.16404, 0.986454, 528.776, 52.17)" />,
<circle key="senate-seat-17" id="senate-seat-17" r="9" cy="217.294" cx="-201.175" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-18" id="senate-seat-18" r="9" cy="252.557" cx="-227.172" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-19" id="senate-seat-19" r="9" cy="170.401" cx="311.791" transform="matrix(-0.986454, -0.16404, -0.16404, 0.986454, 535.427, 55.561)" />,
<circle key="senate-seat-20" id="senate-seat-20" r="9" cy="211.882" cx="-223.172" transform="matrix(-1, 0, 0, 1, 0, 0)" />,
<circle key="senate-seat-21" id="senate-seat-21" r="9" cy="159.989" cx="295.143" transform="matrix(-0.986454, -0.16404, -0.16404, 0.986454, 540.544, 61.56)" />,
<circle key="senate-seat-22" id="senate-seat-22" r="9" cy="159.989" cx="295.143" transform="matrix(0.986454, -0.16404, 0.16404, 0.986454, -54.66, 61.555)" />,
<circle key="senate-seat-23" id="senate-seat-23" r="9" cy="211.882" cx="262.728" />,
<circle key="senate-seat-24" id="senate-seat-24" r="9" cy="252.557" cx="258.729" />,
<circle key="senate-seat-25" id="senate-seat-25" r="9" cy="170.401" cx="311.791" transform="matrix(0.986454, -0.16404, 0.16404, 0.986454, -49.543, 55.556)" />,
<circle key="senate-seat-26" id="senate-seat-26" r="9" cy="217.294" cx="284.726" />,
<circle key="senate-seat-27" id="senate-seat-27" r="9" cy="184.514" cx="327.2" transform="matrix(0.986454, -0.16404, 0.16404, 0.986454, -42.892, 52.164)" />,
<circle key="senate-seat-28" id="senate-seat-28" r="9" cy="184.514" cx="327.2" transform="matrix(0.986454, -0.16404, 0.16404, 0.986454, -21.946, 65.569)" />,
<circle key="senate-seat-29" id="senate-seat-29" r="9" cy="229.407" cx="302.134" />,
<circle key="senate-seat-30" id="senate-seat-30" r="9" cy="268.057" cx="277.228" />,
<circle key="senate-seat-31" id="senate-seat-31" r="9" cy="184.514" cx="327.2" transform="matrix(0.986454, -0.16404, 0.16404, 0.986454, -2.353, 84.436)" />,
<circle key="senate-seat-32" id="senate-seat-32" r="9" cy="246.521" cx="315.591" />,
<circle key="senate-seat-33" id="senate-seat-33" r="9" cy="184.514" cx="327.2" transform="matrix(0.986454, -0.16404, 0.16404, 0.986454, 8.933, 107.941)" />,
<circle key="senate-seat-34" id="senate-seat-34" r="9" cy="269.241" cx="319.968" />,
<circle key="senate-seat-35" id="senate-seat-35" r="9" cy="292.358" cx="279.228" />,
<circle key="senate-seat-36" id="senate-seat-36" r="9" cy="261.241" cx="363.968" />,
<circle key="senate-seat-37" id="senate-seat-37" r="9" cy="292.358" cx="320.968" />,
<circle key="senate-seat-38" id="senate-seat-38" r="9" cy="285.358" cx="363.968" />,
<circle key="senate-seat-39" id="senate-seat-39" r="9" cy="312.598" cx="320.968" />,
<circle key="senate-seat-40" id="senate-seat-40" r="9" cy="307.598" cx="363.968" />,
<circle key="senate-seat-41" id="senate-seat-41" r="9" cy="333.318" cx="320.968" />,
<circle key="senate-seat-42" id="senate-seat-42" r="9" cy="331.318" cx="363.968" />,
<circle key="senate-seat-43" id="senate-seat-43" r="9" cy="354.956" cx="320.968" />,
<circle key="senate-seat-44" id="senate-seat-44" r="9" cy="354.956" cx="363.968" />,
<circle key="senate-seat-45" id="senate-seat-45" r="9" cy="347.619" cx="243.841" />,
];
// Creamos una copia mutable de los datos de asientos para poder modificarla.
const finalSeatData = [...seatData];
// Si hay un presidente y su partido tiene bancas, le quitamos una para asignarla.
if (presidenteBancada && presidenteBancada.color) {
// Encontramos el índice del último asiento que pertenece al partido del presidente.
const lastSeatIndex = finalSeatData.map(s => s.color).lastIndexOf(presidenteBancada.color);
if (lastSeatIndex !== -1) {
// Eliminamos ese asiento de la lista de asientos electorales.
finalSeatData.splice(lastSeatIndex, 1);
}
}
const renderedElements = seatElements.map((child, index) => {
// El asiento presidencial sigue siendo un caso especial
if (index === PRESIDENTE_SEAT_INDEX) {
return React.cloneElement(child, {
fill: presidenteBancada?.color || '#A9A9A9',
stroke: '#000000',
strokeWidth: 2,
});
}
// La lógica ahora es simple: el asiento en el índice X del SVG
// corresponde al asiento en el índice X de los datos.
const seat = seatData[index];
if (!seat) {
return React.cloneElement(child, { fill: '#E0E0E0', stroke: '#ffffff', strokeWidth: 1.5 });
}
return React.cloneElement(child, {
fill: seat.color,
stroke: '#ffffff',
strokeWidth: 1.5,
'data-tooltip-id': seat.ocupante ? 'seat-tooltip' : undefined,
'data-tooltip-html': seat.ocupante
? `<div class="seat-tooltip"><img src="${seat.ocupante.fotoUrl || '/default-avatar.png'}" alt="${seat.ocupante.nombreOcupante}" /><p>${seat.ocupante.nombreOcupante}</p></div>`
: undefined,
});
});
return (
<svg viewBox="73.512 141.174 338.86 280.338" width={size} height={size * (280.338 / 338.86)} style={{ display: 'block', margin: 'auto' }}>
<defs>
{uniqueColors.map(color => {
const patternId = `stripes-${color.replace('#', '')}`;
return (
<pattern key={patternId} id={patternId} patternUnits="userSpaceOnUse" width="4" height="4" patternTransform="rotate(45)">
<rect width="4" height="4" fill={color}></rect>
<path d="M-1,1 l2,-2 M0,4 l4,-4 M3,5 l2,-2" stroke="rgba(255,255,255,0.7)" strokeWidth="3"></path>
</pattern>
);
})}
</defs>
<g>
{renderedElements}
</g>
</svg>
);
};