{% extends "base.html" %} {% block title %}Katalog firm - Norda Biznes Hub{% endblock %} {% block extra_css %} /* Event Banner - Ankieta "Kto weźmie udział?" */ .event-banner { background: linear-gradient(135deg, #f59e0b 0%, #d97706 100%); border-radius: var(--radius-lg); padding: var(--spacing-lg); margin-bottom: var(--spacing-xl); color: white; display: flex; align-items: center; gap: var(--spacing-lg); box-shadow: var(--shadow-md); position: relative; overflow: hidden; text-decoration: none; cursor: pointer; transition: var(--transition); } .event-banner:hover { transform: translateY(-2px); box-shadow: var(--shadow-lg); filter: brightness(1.05); } .event-banner::before { content: ''; position: absolute; top: -50%; right: -10%; width: 200px; height: 200px; background: rgba(255,255,255,0.1); border-radius: 50%; } .event-banner-icon { font-size: 2.5rem; flex-shrink: 0; } .event-banner-content { flex: 1; min-width: 0; } .event-banner-label { font-size: var(--font-size-xs); text-transform: uppercase; letter-spacing: 1px; opacity: 0.9; margin-bottom: var(--spacing-xs); } .event-banner-title { font-size: var(--font-size-lg); font-weight: 700; margin-bottom: var(--spacing-xs); } .event-banner-meta { font-size: var(--font-size-sm); opacity: 0.9; display: flex; flex-wrap: wrap; gap: var(--spacing-md); } .event-banner-attendees { display: flex; align-items: center; gap: var(--spacing-xs); background: rgba(255,255,255,0.2); padding: var(--spacing-xs) var(--spacing-sm); border-radius: var(--radius); font-weight: 600; margin-top: var(--spacing-sm); } .event-banner-action { flex-shrink: 0; } .event-banner .btn-light { background: white; color: #d97706; border: none; padding: var(--spacing-sm) var(--spacing-lg); font-weight: 600; font-size: var(--font-size-base); border-radius: var(--radius); text-decoration: none; display: inline-block; cursor: pointer; transition: var(--transition); } .event-banner .btn-light:disabled { opacity: 0.7; cursor: wait; } .event-banner .btn-light:hover { background: #fef3c7; transform: translateY(-1px); } .event-banner .btn-registered { background: #166534; color: white; } .event-banner .btn-registered:hover { background: #15803d; } @media (max-width: 640px) { .event-banner { flex-direction: column; text-align: center; } .event-banner-meta { justify-content: center; } .event-banner-attendees { justify-content: center; } } /* NordaGPT Chat Banner */ .chat-banner { background: linear-gradient(135deg, #7c3aed 0%, #5b21b6 100%); border-radius: var(--radius-lg); padding: var(--spacing-lg); margin-bottom: var(--spacing-xl); color: white; display: flex; align-items: center; gap: var(--spacing-lg); box-shadow: var(--shadow-md); position: relative; overflow: hidden; transition: var(--transition); cursor: pointer; } .chat-banner:hover { transform: translateY(-2px); box-shadow: var(--shadow-lg); filter: brightness(1.05); } /* Chat minimized state - banner pulsing to indicate active session */ .chat-banner.chat-active { animation: chatPulse 2s ease-in-out infinite; border: 2px solid rgba(255,255,255,0.5); } .chat-banner.chat-active .chat-banner-btn { background: #10b981; color: white; } @keyframes chatPulse { 0%, 100% { box-shadow: var(--shadow-md), 0 0 0 0 rgba(124, 58, 237, 0.4); } 50% { box-shadow: var(--shadow-lg), 0 0 0 8px rgba(124, 58, 237, 0); } } .chat-banner::before { content: ''; position: absolute; top: -50%; right: -10%; width: 200px; height: 200px; background: rgba(255,255,255,0.1); border-radius: 50%; } .chat-banner-icon { font-size: 2.5rem; flex-shrink: 0; } .chat-banner-content { flex: 1; min-width: 0; } .chat-banner-label { font-size: var(--font-size-xs); text-transform: uppercase; letter-spacing: 1px; opacity: 0.9; margin-bottom: var(--spacing-xs); } .chat-banner-title { font-size: var(--font-size-lg); font-weight: 700; margin-bottom: var(--spacing-sm); } .chat-banner-input-wrapper { display: flex; gap: var(--spacing-sm); align-items: center; } .chat-banner-input { flex: 1; padding: var(--spacing-sm) var(--spacing-md); border: none; border-radius: var(--radius); font-size: var(--font-size-sm); background: rgba(255,255,255,0.95); color: var(--text-primary); cursor: pointer; } .chat-banner-input:focus { outline: none; background: white; } .chat-banner-input::placeholder { color: var(--text-secondary); } .chat-banner-btn { background: white; color: #7c3aed; border: none; padding: var(--spacing-sm) var(--spacing-md); font-weight: 600; font-size: var(--font-size-sm); border-radius: var(--radius); cursor: pointer; transition: var(--transition); white-space: nowrap; } .chat-banner-btn:hover { background: #f3e8ff; transform: translateY(-1px); } /* NordaGPT Fullscreen Modal */ .nordagpt-modal { display: none; position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0,0.5); z-index: 1000; animation: fadeIn 0.2s ease; } .nordagpt-modal.active { display: flex; } .nordagpt-modal.minimized { display: none; } @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } } .nordagpt-container { position: absolute; top: 20px; left: 20px; right: 20px; bottom: 20px; background: white; border-radius: var(--radius-xl); box-shadow: var(--shadow-xl); display: flex; flex-direction: column; overflow: hidden; animation: slideUp 0.3s ease; } @keyframes slideUp { from { transform: translateY(20px); opacity: 0; } to { transform: translateY(0); opacity: 1; } } .nordagpt-header { display: flex; align-items: center; justify-content: space-between; padding: var(--spacing-md) var(--spacing-lg); background: linear-gradient(135deg, #7c3aed 0%, #5b21b6 100%); color: white; } .nordagpt-header-left { display: flex; align-items: center; gap: var(--spacing-sm); } .nordagpt-header h2 { font-size: var(--font-size-lg); font-weight: 600; margin: 0; } .nordagpt-header-badge { background: rgba(255,255,255,0.2); padding: 2px 8px; border-radius: var(--radius-sm); font-size: var(--font-size-xs); } .nordagpt-header-actions { display: flex; gap: var(--spacing-xs); } .nordagpt-header-btn { background: rgba(255,255,255,0.2); border: none; color: white; width: 32px; height: 32px; border-radius: var(--radius); cursor: pointer; display: flex; align-items: center; justify-content: center; transition: var(--transition); } .nordagpt-header-btn:hover { background: rgba(255,255,255,0.3); } .nordagpt-messages { flex: 1; overflow-y: auto; padding: var(--spacing-lg); display: flex; flex-direction: column; gap: var(--spacing-md); } .nordagpt-message { display: flex; gap: var(--spacing-sm); max-width: 85%; } .nordagpt-message.user { align-self: flex-end; flex-direction: row-reverse; } .nordagpt-message-avatar { width: 36px; height: 36px; border-radius: 50%; display: flex; align-items: center; justify-content: center; font-weight: 600; font-size: var(--font-size-sm); flex-shrink: 0; } .nordagpt-message.assistant .nordagpt-message-avatar { background: linear-gradient(135deg, #7c3aed 0%, #5b21b6 100%); color: white; } .nordagpt-message.user .nordagpt-message-avatar { background: var(--primary); color: white; } .nordagpt-message-content { padding: var(--spacing-sm) var(--spacing-md); border-radius: var(--radius-lg); line-height: 1.5; } .nordagpt-message.assistant .nordagpt-message-content { background: var(--background); color: var(--text-primary); border-bottom-left-radius: var(--radius-sm); } .nordagpt-message.user .nordagpt-message-content { background: var(--primary); color: white; border-bottom-right-radius: var(--radius-sm); } .nordagpt-message-content a { color: inherit; text-decoration: underline; } /* AI response list styles */ .nordagpt-message-content .ai-list { margin: var(--spacing-xs) 0; padding-left: var(--spacing-lg); } .nordagpt-message-content .ai-list li { margin-bottom: var(--spacing-xs); line-height: 1.4; } .nordagpt-message-content ol.ai-list { list-style-type: decimal; } .nordagpt-message-content ul.ai-list { list-style-type: disc; } .nordagpt-message-content strong { font-weight: 600; } .nordagpt-input-area { padding: var(--spacing-md) var(--spacing-lg); border-top: 1px solid var(--border); display: flex; gap: var(--spacing-sm); } .nordagpt-input { flex: 1; padding: var(--spacing-md); border: 1px solid var(--border); border-radius: var(--radius-lg); font-size: var(--font-size-base); resize: none; } .nordagpt-input:focus { outline: none; border-color: #7c3aed; box-shadow: 0 0 0 3px rgba(124, 58, 237, 0.1); } .nordagpt-send-btn { background: linear-gradient(135deg, #7c3aed 0%, #5b21b6 100%); color: white; border: none; padding: var(--spacing-md) var(--spacing-lg); border-radius: var(--radius-lg); font-weight: 600; cursor: pointer; transition: var(--transition); } .nordagpt-send-btn:hover { filter: brightness(1.1); } .nordagpt-send-btn:disabled { opacity: 0.6; cursor: not-allowed; } .nordagpt-typing { display: flex; gap: 4px; padding: var(--spacing-sm); } .nordagpt-typing span { width: 8px; height: 8px; background: #7c3aed; border-radius: 50%; animation: typing 1.4s infinite; } .nordagpt-typing span:nth-child(2) { animation-delay: 0.2s; } .nordagpt-typing span:nth-child(3) { animation-delay: 0.4s; } @keyframes typing { 0%, 60%, 100% { transform: translateY(0); opacity: 0.4; } 30% { transform: translateY(-4px); opacity: 1; } } @media (max-width: 640px) { .chat-banner { flex-direction: column; text-align: center; } .chat-banner-input-wrapper { width: 100%; flex-direction: column; } .chat-banner-input { width: 100%; } .nordagpt-container { top: 0; left: 0; right: 0; bottom: 0; border-radius: 0; } .nordagpt-message { max-width: 95%; } } /* Search Bar */ .search-section { margin-bottom: var(--spacing-2xl); } .search-bar { display: flex; gap: var(--spacing-md); max-width: 800px; margin: 0 auto; } .search-input { flex: 1; padding: var(--spacing-md); border: 1px solid var(--border); border-radius: var(--radius); font-size: var(--font-size-base); } .search-input:focus { outline: none; border-color: var(--primary); box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.1); } /* Category Filter */ .category-filter { display: flex; gap: var(--spacing-sm); flex-wrap: wrap; justify-content: center; margin-bottom: var(--spacing-xl); } .category-badge { padding: var(--spacing-sm) var(--spacing-md); background-color: var(--background); border: 1px solid var(--border); border-radius: var(--radius-lg); color: var(--text-secondary); text-decoration: none; font-size: var(--font-size-sm); font-weight: 500; transition: var(--transition); cursor: pointer; } .category-badge:hover, .category-badge.active { background-color: var(--primary); color: white; border-color: var(--primary); } /* Company Grid */ .companies-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(320px, 1fr)); gap: var(--spacing-lg); margin-bottom: var(--spacing-2xl); } .company-card { background-color: var(--surface); border-radius: var(--radius-lg); padding: var(--spacing-lg); box-shadow: var(--shadow); transition: var(--transition); display: flex; flex-direction: column; height: 100%; } .company-card:hover { box-shadow: var(--shadow-lg); transform: translateY(-2px); } .company-logo { width: 100%; height: 80px; display: flex; align-items: center; justify-content: center; margin-bottom: var(--spacing-md); background: var(--background); border-radius: var(--radius-md); overflow: hidden; } .company-logo img { max-width: 100%; max-height: 100%; object-fit: contain; } .company-header { margin-bottom: var(--spacing-md); } .company-category { display: inline-block; padding: var(--spacing-xs) var(--spacing-sm); background-color: var(--background); color: var(--text-secondary); font-size: var(--font-size-sm); border-radius: var(--radius); margin-bottom: var(--spacing-sm); } .company-name { font-size: var(--font-size-xl); font-weight: 600; color: var(--text-primary); margin-bottom: var(--spacing-sm); text-decoration: none; display: block; } .company-name:hover { color: var(--primary); } .company-description { color: var(--text-secondary); font-size: var(--font-size-sm); line-height: 1.6; margin-bottom: var(--spacing-md); flex: 1; } .company-contact { display: flex; flex-direction: column; gap: var(--spacing-xs); font-size: var(--font-size-sm); color: var(--text-secondary); padding-top: var(--spacing-md); border-top: 1px solid var(--border); } .company-contact-item { display: flex; align-items: center; gap: var(--spacing-sm); } .company-contact a { color: var(--primary); text-decoration: none; } .company-contact a:hover { text-decoration: underline; } .empty-state { text-align: center; padding: var(--spacing-2xl); color: var(--text-secondary); } .empty-state svg { opacity: 0.3; margin-bottom: var(--spacing-md); } /* Loading State */ .loading { text-align: center; padding: var(--spacing-2xl); color: var(--text-secondary); } @media (max-width: 768px) { .companies-grid { grid-template-columns: 1fr; } .search-bar { flex-direction: column; } } {% endblock %} {% block content %}

Katalog firm Norda Biznes

{{ total_companies }} firm członkowskich • {{ total_categories }} branż

{% if next_event %}
📅
Najbliższe wydarzenie – Kto weźmie udział?
{{ next_event.title }} →
📆 {{ next_event.event_date.strftime('%d.%m.%Y') }} ({{ ['Pon', 'Wt', 'Śr', 'Czw', 'Pt', 'Sob', 'Nd'][next_event.event_date.weekday()] }}) {% if next_event.time_start %} 🕕 {{ next_event.time_start.strftime('%H:%M') }} {% endif %} {% if next_event.location %} 📍 {{ next_event.location[:30] }}{% if next_event.location|length > 30 %}...{% endif %} {% endif %}
👥 Zapisanych: {{ next_event.attendee_count }} {% if next_event.attendee_count == 1 %}osoba{% elif next_event.attendee_count in [2,3,4] %}osoby{% else %}osób{% endif %}
{% if user_registered %} ✓ Jesteś zapisany/a {% else %} {% endif %}
{% endif %} {% if current_user.is_authenticated %}
🤖
NordaGPT - Asystent AI Norda Biznes
Zapytaj o firmy, usługi, wydarzenia...
Np. Kto oferuje usługi IT? Kiedy następne spotkanie? Rozpocznij chat →
{% endif %}
🤖

NordaGPT

Gemini 2.0
AI
Cześć! Jestem NordaGPT - asystentem AI Norda Biznes. Mogę pomóc Ci znaleźć firmy, usługi, sprawdzić kalendarz wydarzeń, rekomendacje i wiele więcej. O co chcesz zapytać?
{% if categories %}
{% for category in categories %} {% set count = companies|selectattr('category_id', 'equalto', category.id)|list|length %} {% if count > 0 %} {% endif %} {% endfor %}
{% endif %} {% if companies %}
{% for company in companies %}
{% if company.category %} {{ company.category.name }} {% endif %} {{ company.name }}
{{ company.description_short|truncate(150) if company.description_short else 'Brak opisu' }}
{% if company.website %} {% endif %} {% if company.phone %} {% endif %} {% if company.address_city %}
{{ company.address_city }}
{% endif %}
{% endfor %}
{% else %}

Brak firm w katalogu

Nie znaleziono żadnych firm w systemie.

{% endif %} {% endblock %} {% block extra_js %} // RSVP and redirect to event async function rsvpAndGo(e, eventId) { e.preventDefault(); e.stopPropagation(); const btn = e.target; btn.disabled = true; btn.textContent = 'Zapisuję...'; try { const response = await fetch('/kalendarz/' + eventId + '/rsvp', { method: 'POST', headers: { 'Content-Type': 'application/json', 'X-CSRFToken': '{{ csrf_token() }}' } }); const data = await response.json(); if (data.success) { // Update counter visually before redirect const counter = document.querySelector('.event-banner-attendees'); if (counter && data.action === 'added') { const newCount = data.attendee_count; let suffix = 'osób'; if (newCount === 1) suffix = 'osoba'; else if (newCount >= 2 && newCount <= 4) suffix = 'osoby'; counter.innerHTML = '👥 Zapisanych: ' + newCount + ' ' + suffix; } btn.textContent = '✓ Zapisano!'; // Redirect after short delay setTimeout(() => { window.location.href = '/kalendarz/' + eventId; }, 500); } else { btn.textContent = 'Błąd'; setTimeout(() => { btn.textContent = 'Zapisz się →'; btn.disabled = false; }, 2000); } } catch (error) { btn.textContent = 'Błąd sieci'; setTimeout(() => { btn.textContent = 'Zapisz się →'; btn.disabled = false; }, 2000); } } // Category filter function filterCategory(slug) { const cards = document.querySelectorAll('.company-card'); const badges = document.querySelectorAll('.category-badge'); // Update active badge badges.forEach(badge => { badge.classList.remove('active'); if (badge.textContent.toLowerCase().includes(slug) || (slug === 'all' && badge.textContent.includes('Wszystkie'))) { badge.classList.add('active'); } }); // Filter cards cards.forEach(card => { if (slug === 'all') { card.style.display = 'flex'; } else { const cardCategory = card.getAttribute('data-category'); card.style.display = cardCategory === slug ? 'flex' : 'none'; } }); } // Smooth scroll to companies on search const urlParams = new URLSearchParams(window.location.search); if (urlParams.get('q')) { document.getElementById('companiesGrid')?.scrollIntoView({ behavior: 'smooth' }); } // ======================================== // NordaGPT Chat Functions // ======================================== let nordaGPTConversationId = null; let nordaGPTIsMinimized = false; function openNordaGPT() { document.getElementById('nordagptModal').classList.add('active'); document.getElementById('nordagptModal').classList.remove('minimized'); document.getElementById('nordagptInput').focus(); document.body.style.overflow = 'hidden'; nordaGPTIsMinimized = false; // Remove active indicator from banner const banner = document.getElementById('chatBanner'); if (banner) { banner.classList.remove('chat-active'); } } function minimizeNordaGPT() { document.getElementById('nordagptModal').classList.remove('active'); document.getElementById('nordagptModal').classList.add('minimized'); document.body.style.overflow = ''; nordaGPTIsMinimized = true; // Show banner with active indicator const banner = document.getElementById('chatBanner'); const title = document.getElementById('chatBannerTitle'); const btn = banner?.querySelector('.chat-banner-btn'); if (banner) { banner.classList.add('chat-active'); } if (title) { title.textContent = '💬 Chat aktywny - kliknij aby kontynuować'; } if (btn) { btn.textContent = 'Wznów chat →'; } } function closeNordaGPT() { document.getElementById('nordagptModal').classList.remove('active'); document.getElementById('nordagptModal').classList.remove('minimized'); document.body.style.overflow = ''; nordaGPTIsMinimized = false; // Reset banner to initial state const banner = document.getElementById('chatBanner'); const title = document.getElementById('chatBannerTitle'); const btn = banner?.querySelector('.chat-banner-btn'); if (banner) { banner.classList.remove('chat-active'); } if (title) { title.textContent = 'Zapytaj o firmy, usługi, wydarzenia...'; } if (btn) { btn.textContent = 'Rozpocznij chat →'; } } // Convert URLs, emails, markdown to HTML (linkify + formatting) function linkifyNordaGPT(text) { let escaped = text .replace(/&/g, '&') .replace(//g, '>') .replace(/"/g, '"'); // Use placeholders to protect converted elements const placeholders = []; function addPlaceholder(html) { const placeholder = '__PH_' + placeholders.length + '__'; placeholders.push(html); return placeholder; } // 1. Markdown links first const markdownLinkRegex = /\[([^\]]+)\]\((https?:\/\/[^)]+)\)/gi; escaped = escaped.replace(markdownLinkRegex, function(match, linkText, url) { return addPlaceholder('' + linkText + ''); }); // 2. Plain URLs const urlRegex = /(https?:\/\/[^\s<]+|www\.[^\s<]+)/gi; escaped = escaped.replace(urlRegex, function(url) { let cleanUrl = url.replace(/[.,;:!?)\]]+$/, ''); const trailingPunct = url.slice(cleanUrl.length); const href = cleanUrl.startsWith('www.') ? 'https://' + cleanUrl : cleanUrl; return addPlaceholder('' + cleanUrl + '') + trailingPunct; }); // 3. Emails const emailRegex = /([a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,})/gi; escaped = escaped.replace(emailRegex, function(email) { let cleanEmail = email.replace(/[.,;:!?)\]]+$/, ''); const trailingPunct = email.slice(cleanEmail.length); return addPlaceholder('' + cleanEmail + '') + trailingPunct; }); // 4. Convert **bold** to escaped = escaped.replace(/\*\*([^*]+)\*\*/g, function(match, boldText) { return addPlaceholder('' + boldText + ''); }); // 5. Process lines for lists and newlines const lines = escaped.split('\n'); const processedLines = []; let inList = false; let listType = null; for (let i = 0; i < lines.length; i++) { let line = lines[i]; const trimmedLine = line.trim(); // Check for numbered list (1. 2. 3. etc.) const numberedMatch = trimmedLine.match(/^(\d+)\.\s+(.*)$/); // Check for bullet list (- or * at start) const bulletMatch = trimmedLine.match(/^[-*]\s+(.*)$/); if (numberedMatch) { if (!inList || listType !== 'ol') { if (inList) processedLines.push(listType === 'ol' ? '' : ''); processedLines.push('
    '); inList = true; listType = 'ol'; } processedLines.push('
  1. ' + numberedMatch[2] + '
  2. '); } else if (bulletMatch) { if (!inList || listType !== 'ul') { if (inList) processedLines.push(listType === 'ol' ? '
' : ''); processedLines.push(''); inList = false; listType = null; } if (trimmedLine === '') { if (!inList) processedLines.push('
'); } else { processedLines.push(line); if (i < lines.length - 1) processedLines.push('
'); } } } if (inList) { processedLines.push(listType === 'ol' ? '' : ''); } escaped = processedLines.join('\n'); // 6. Restore all placeholders placeholders.forEach(function(html, i) { escaped = escaped.replace('__PH_' + i + '__', html); }); // Clean up multiple consecutive
tags escaped = escaped.replace(/(
\s*){3,}/g, '

'); return escaped; } function addNordaGPTMessage(role, content) { const messagesDiv = document.getElementById('nordagptMessages'); const messageDiv = document.createElement('div'); messageDiv.className = 'nordagpt-message ' + role; const avatar = document.createElement('div'); avatar.className = 'nordagpt-message-avatar'; avatar.textContent = role === 'user' ? 'U' : 'AI'; const contentDiv = document.createElement('div'); contentDiv.className = 'nordagpt-message-content'; if (role === 'assistant') { contentDiv.innerHTML = linkifyNordaGPT(content); } else { contentDiv.textContent = content; } messageDiv.appendChild(avatar); messageDiv.appendChild(contentDiv); messagesDiv.appendChild(messageDiv); messagesDiv.scrollTop = messagesDiv.scrollHeight; } function showNordaGPTTyping() { const messagesDiv = document.getElementById('nordagptMessages'); const typingDiv = document.createElement('div'); typingDiv.className = 'nordagpt-message assistant'; typingDiv.id = 'nordagptTyping'; typingDiv.innerHTML = `
AI
`; messagesDiv.appendChild(typingDiv); messagesDiv.scrollTop = messagesDiv.scrollHeight; } function hideNordaGPTTyping() { const typing = document.getElementById('nordagptTyping'); if (typing) typing.remove(); } async function sendNordaGPTMessage() { const input = document.getElementById('nordagptInput'); const sendBtn = document.getElementById('nordagptSendBtn'); const message = input.value.trim(); if (!message) return; // Add user message addNordaGPTMessage('user', message); input.value = ''; sendBtn.disabled = true; // Show typing indicator showNordaGPTTyping(); try { // Step 1: Start conversation if we don't have one if (!nordaGPTConversationId) { const startResponse = await fetch('/api/chat/start', { method: 'POST', headers: { 'Content-Type': 'application/json', 'X-CSRFToken': '{{ csrf_token() }}' }, body: JSON.stringify({ title: 'NordaGPT - ' + new Date().toLocaleDateString('pl-PL') }) }); const startData = await startResponse.json(); if (startData.success) { nordaGPTConversationId = startData.conversation_id; } else { throw new Error(startData.error || 'Nie udało się rozpocząć rozmowy'); } } // Step 2: Send message to conversation const response = await fetch('/api/chat/' + nordaGPTConversationId + '/message', { method: 'POST', headers: { 'Content-Type': 'application/json', 'X-CSRFToken': '{{ csrf_token() }}' }, body: JSON.stringify({ message: message }) }); const data = await response.json(); hideNordaGPTTyping(); if (data.success && data.message) { addNordaGPTMessage('assistant', data.message); } else if (data.error) { addNordaGPTMessage('assistant', 'Przepraszam, wystąpił błąd: ' + data.error); } } catch (error) { hideNordaGPTTyping(); console.error('NordaGPT error:', error); addNordaGPTMessage('assistant', 'Przepraszam, nie mogę teraz odpowiedzieć. Spróbuj ponownie później.'); } sendBtn.disabled = false; input.focus(); } // Close modal on Escape key document.addEventListener('keydown', function(e) { if (e.key === 'Escape') { const modal = document.getElementById('nordagptModal'); if (modal.classList.contains('active')) { minimizeNordaGPT(); } } }); // Close modal when clicking outside document.getElementById('nordagptModal')?.addEventListener('click', function(e) { if (e.target === this) { minimizeNordaGPT(); } }); {% endblock %}