GrantForge Bot
Deploy to Hugging Face
afd56bc
import React, { useState } from 'react';
import { motion, AnimatePresence } from 'framer-motion';
import { Download, FileText, File as FileIcon, X } from 'lucide-react';
import { exportProjectDocument } from '../../api/client';
import toast from 'react-hot-toast';
interface ExportModalProps {
isOpen: boolean;
onClose: () => void;
projectId: string;
}
export const ExportModal: React.FC<ExportModalProps> = ({ isOpen, onClose, projectId }) => {
const [format, setFormat] = useState<'pdf' | 'docx'>('pdf');
const [template, setTemplate] = useState<string>('official');
const [isExporting, setIsExporting] = useState(false);
if (!isOpen) return null;
const handleExport = async () => {
try {
setIsExporting(true);
toast.loading('Generowanie dokumentu...', { id: 'export' });
const response = await exportProjectDocument(projectId, format, template);
// Create a blob URL and trigger download
const blob = new Blob([response.data], { type: response.headers['content-type'] });
const url = window.URL.createObjectURL(blob);
// Generate filename based on headers or default
let fileName = `Wniosek_${projectId}.${format}`;
const contentDisposition = response.headers['content-disposition'];
if (contentDisposition && contentDisposition.indexOf('filename=') !== -1) {
const matches = contentDisposition.match(/filename="?([^"]+)"?/);
if (matches && matches[1]) {
fileName = matches[1];
}
}
const link = document.createElement('a');
link.href = url;
link.setAttribute('download', fileName);
document.body.appendChild(link);
link.click();
link.remove();
window.URL.revokeObjectURL(url);
toast.success('Pomyślnie wyeksportowano dokument!', { id: 'export' });
onClose();
} catch (error) {
console.error(error);
toast.error('Wystąpił błąd podczas generowania dokumentu.', { id: 'export' });
} finally {
setIsExporting(false);
}
};
return (
<AnimatePresence>
<div style={{
position: 'fixed',
top: 0, left: 0, right: 0, bottom: 0,
backgroundColor: 'rgba(0,0,0,0.7)',
zIndex: 2000,
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
backdropFilter: 'blur(3px)'
}}>
<motion.div
initial={{ opacity: 0, scale: 0.95, y: 10 }}
animate={{ opacity: 1, scale: 1, y: 0 }}
exit={{ opacity: 0, scale: 0.95, y: 10 }}
style={{
background: '#1a1b1e',
border: '1px solid rgba(255,255,255,0.1)',
borderRadius: '16px',
width: '90%',
maxWidth: '500px',
overflow: 'hidden',
boxShadow: '0 20px 40px rgba(0,0,0,0.5)'
}}
>
<div style={{ padding: '1.5rem', borderBottom: '1px solid rgba(255,255,255,0.1)', display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
<h2 style={{ margin: 0, display: 'flex', alignItems: 'center', gap: '0.5rem', fontSize: '1.25rem' }}>
<Download size={20} color="#6366f1" />
Eksportuj Wniosek
</h2>
<button
onClick={onClose}
style={{ background: 'transparent', border: 'none', color: 'var(--text-muted)', cursor: 'pointer' }}
>
<X size={20} />
</button>
</div>
<div style={{ padding: '1.5rem', display: 'flex', flexDirection: 'column', gap: '1.5rem' }}>
{/* FORMAT WIDGET */}
<div>
<label style={{ display: 'block', marginBottom: '0.75rem', fontWeight: 600, color: 'var(--text-primary)' }}>Wybierz Format</label>
<div style={{ display: 'grid', gridTemplateColumns: '1fr 1fr', gap: '1rem' }}>
<button
onClick={() => setFormat('pdf')}
style={{
padding: '1rem',
background: format === 'pdf' ? 'rgba(99, 102, 241, 0.1)' : 'rgba(255,255,255,0.03)',
border: `1px solid ${format === 'pdf' ? '#6366f1' : 'rgba(255,255,255,0.1)'}`,
borderRadius: '12px',
display: 'flex',
flexDirection: 'column',
alignItems: 'center',
gap: '0.5rem',
cursor: 'pointer',
color: format === 'pdf' ? '#fff' : 'var(--text-muted)'
}}
>
<FileIcon size={32} color={format === 'pdf' ? '#6366f1' : 'currentColor'} />
<span style={{ fontWeight: 600 }}>Dokument PDF</span>
</button>
<button
onClick={() => setFormat('docx')}
style={{
padding: '1rem',
background: format === 'docx' ? 'rgba(99, 102, 241, 0.1)' : 'rgba(255,255,255,0.03)',
border: `1px solid ${format === 'docx' ? '#6366f1' : 'rgba(255,255,255,0.1)'}`,
borderRadius: '12px',
display: 'flex',
flexDirection: 'column',
alignItems: 'center',
gap: '0.5rem',
cursor: 'pointer',
color: format === 'docx' ? '#fff' : 'var(--text-muted)'
}}
>
<FileText size={32} color={format === 'docx' ? '#6366f1' : 'currentColor'} />
<span style={{ fontWeight: 600 }}>Microsoft Word (DOCX)</span>
</button>
</div>
</div>
{/* TEMPLATE WIDGET */}
<div>
<label style={{ display: 'block', marginBottom: '0.75rem', fontWeight: 600, color: 'var(--text-primary)' }}>Wybierz Szablon</label>
<div style={{ display: 'flex', flexDirection: 'column', gap: '0.5rem' }}>
{[
{ id: 'standard', name: 'Standardowy', desc: 'Dobry na start (Arial 11, klasyczny układ)' },
{ id: 'official', name: 'Urzędowy', desc: 'Całkowicie ustrukturyzowany, formalny układ zalecany do urzędów' },
{ id: 'modern', name: 'Nowoczesny', desc: 'Więcej przestrzeni i oddechu z subtelnymi akcentami kolorystycznymi' }
].map(tpl => (
<button
key={tpl.id}
onClick={() => setTemplate(tpl.id)}
style={{
padding: '1rem',
background: template === tpl.id ? 'rgba(99, 102, 241, 0.1)' : 'rgba(255,255,255,0.03)',
border: `1px solid ${template === tpl.id ? '#6366f1' : 'rgba(255,255,255,0.1)'}`,
borderRadius: '8px',
textAlign: 'left',
cursor: 'pointer',
display: 'flex',
flexDirection: 'column'
}}
>
<span style={{ fontWeight: 600, color: template === tpl.id ? '#6366f1' : '#fff' }}>{tpl.name}</span>
<span style={{ fontSize: '0.8rem', color: 'var(--text-muted)', marginTop: '0.2rem' }}>{tpl.desc}</span>
</button>
))}
</div>
{/* TIP FOR DOCX */}
{format === 'docx' && (
<div style={{ marginTop: '1rem', padding: '0.8rem', borderRadius: '8px', background: 'rgba(99, 102, 241, 0.05)', border: '1px dashed rgba(99, 102, 241, 0.3)', fontSize: '0.85rem', color: '#a5b4fc', display: 'flex', gap: '0.5rem', alignItems: 'center' }}>
<span style={{ fontSize: '1.2rem' }}>ℹ️</span>
<span>W wygenerowanym pliku DOCX spis treści jest początkowo pusty. Po otwarciu dokumentu w Microsoft Word naciśnij <b>F9</b>, aby zaktualizować stronę.</span>
</div>
)}
</div>
</div>
<div style={{ padding: '1rem 1.5rem', borderTop: '1px solid rgba(255,255,255,0.1)', display: 'flex', justifyContent: 'flex-end', gap: '1rem', background: 'rgba(0,0,0,0.2)' }}>
<button
onClick={onClose}
style={{ padding: '0.6rem 1.2rem', borderRadius: '6px', border: '1px solid rgba(255,255,255,0.2)', background: 'transparent', color: '#fff', cursor: 'pointer' }}
>
Anuluj
</button>
<button
onClick={handleExport}
disabled={isExporting}
style={{
padding: '0.6rem 1.2rem',
borderRadius: '6px',
border: 'none',
background: isExporting ? '#4f46e5' : '#6366f1',
color: '#fff',
cursor: isExporting ? 'not-allowed' : 'pointer',
fontWeight: 600,
display: 'flex',
alignItems: 'center',
gap: '0.5rem'
}}
>
{isExporting ? 'Eksportowanie...' : 'Generuj Dokument'}
</button>
</div>
</motion.div>
</div>
</AnimatePresence>
);
};