{% extends "base.html" %} {% block title %}Katalog firm - Norda Biznes Partner{% endblock %} {% block extra_css %} /* Events Row - vertical stack in left column */ .events-row { display: flex; flex-direction: column; gap: var(--spacing-md); } .events-row-label { font-size: var(--font-size-xs); text-transform: uppercase; letter-spacing: 1px; color: rgba(255,255,255,0.9); margin-bottom: var(--spacing-xs); } /* Event Banner - Ankieta "Kto weźmie udział?" (niebieski primary) */ .event-banner { background: linear-gradient(135deg, #1e3050 0%, #2E4872 100%); border-radius: var(--radius-lg); padding: var(--spacing-md); color: white; display: flex; flex-direction: column; gap: var(--spacing-sm); box-shadow: var(--shadow-md); position: relative; overflow: hidden; text-decoration: none; cursor: pointer; transition: var(--transition); flex: 1; min-width: 0; } .event-banner:hover { transform: translateY(-2px); box-shadow: 0 10px 30px rgba(46, 72, 114, 0.25); 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-top { display: flex; align-items: flex-start; gap: var(--spacing-sm); } .event-banner-icon { font-size: 2rem; flex-shrink: 0; } .event-banner-content { flex: 1; min-width: 0; } .event-banner-title { font-size: var(--font-size-base); font-weight: 700; margin-bottom: var(--spacing-xs); } .event-banner-meta { font-size: var(--font-size-xs); opacity: 0.9; display: flex; flex-wrap: wrap; gap: var(--spacing-sm); } .event-banner-bottom { display: flex; align-items: center; justify-content: space-between; gap: var(--spacing-sm); } .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; font-size: var(--font-size-xs); } .event-banner-action { flex-shrink: 0; } .event-banner .btn-light { background: white; color: #2E4872; border: none; padding: var(--spacing-xs) var(--spacing-md); font-weight: 600; font-size: var(--font-size-sm); border-radius: var(--radius-btn); 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: #EDF0F5; transform: translateY(-1px); } .event-banner .btn-registered { background: #166534; color: white; } .event-banner .btn-registered:hover { background: #15803d; } /* Events filter buttons */ .events-filter { display: flex; align-items: center; gap: var(--spacing-sm); margin-bottom: var(--spacing-md); } .events-filter-label { font-size: var(--font-size-sm); color: var(--text-muted); font-weight: 500; } .events-filter-btn { padding: var(--spacing-xs) var(--spacing-md); border-radius: var(--radius-btn); border: 1.5px solid #cbd5e1; background: white; color: var(--text-secondary); font-size: var(--font-size-sm); font-weight: 500; cursor: pointer; transition: var(--transition); } .events-filter-btn:hover { border-color: var(--primary); color: var(--primary); } .events-filter-btn.active { background: var(--primary); color: white; border-color: var(--primary); } .event-banner[data-event-type="external"] { background: linear-gradient(135deg, #1a3a2a 0%, #2d5a3e 100%); } .event-banner[data-event-type="external"]:hover { box-shadow: 0 10px 30px rgba(45, 90, 62, 0.25); } .event-banner[data-event-type="external"] .btn-light { color: #2d5a3e; } .event-banner.hidden-by-filter { display: none; } /* Homepage 2-column grid */ .homepage-grid { display: grid; grid-template-columns: 1fr 1fr; gap: var(--spacing-md); margin-bottom: var(--spacing-lg); } @media (max-width: 768px) { .homepage-grid { grid-template-columns: 1fr; } .events-row { grid-template-columns: 1fr; } .event-banner { flex-direction: column; text-align: center; } .event-banner-top { flex-direction: column; align-items: center; } .event-banner-meta { justify-content: center; } .event-banner-bottom { flex-direction: column; } .event-banner-attendees { justify-content: center; } } /* NordaGPT Chat Banner (niebieski primary) */ .chat-banner { background: linear-gradient(135deg, #1e3050 0%, #2E4872 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: 0 10px 30px rgba(46, 72, 114, 0.25); 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(46, 72, 114, 0.4); } 50% { box-shadow: var(--shadow-lg), 0 0 0 8px rgba(46, 72, 114, 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: #2E4872; border: none; padding: var(--spacing-sm) var(--spacing-md); font-weight: 600; font-size: var(--font-size-sm); border-radius: var(--radius-btn); cursor: pointer; transition: var(--transition); white-space: nowrap; } .chat-banner-btn:hover { background: #EDF0F5; 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, #1e3050 0%, #2E4872 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, #1e3050 0%, #2E4872 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: #2E4872; box-shadow: 0 0 0 3px rgba(46, 72, 114, 0.1); } .nordagpt-send-btn { background: linear-gradient(135deg, #1e3050 0%, #2E4872 100%); color: white; border: none; padding: var(--spacing-md) var(--spacing-lg); border-radius: var(--radius-btn); 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: #2E4872; 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); } /* Category hierarchy - two rows */ .category-filter-wrapper { display: flex; flex-direction: column; gap: var(--spacing-md); margin-bottom: var(--spacing-xl); } .category-filter-main { display: flex; gap: var(--spacing-sm); flex-wrap: wrap; justify-content: center; } .category-filter-sub { display: none; gap: var(--spacing-sm); flex-wrap: wrap; justify-content: center; padding: var(--spacing-md); background: var(--bg-secondary); border-radius: var(--radius-lg); } .category-filter-sub.visible { display: flex; } .category-badge.category-main { font-weight: 600; background-color: var(--surface); color: var(--text-primary); border-color: var(--primary); } .category-badge.category-main:hover { background: linear-gradient(135deg, var(--primary) 0%, var(--primary-dark, #1d4ed8) 100%); color: white; border-color: var(--primary); } .category-badge.category-main.active { background: linear-gradient(135deg, var(--primary) 0%, var(--primary-dark, #1d4ed8) 100%); color: white; border-color: var(--primary); box-shadow: 0 2px 8px rgba(37, 99, 235, 0.3); } .category-badge.category-sub { font-size: var(--font-size-sm); padding: var(--spacing-xs) var(--spacing-md); background-color: var(--surface); border-color: var(--border); } .category-badge.category-sub:hover, .category-badge.category-sub.active { background-color: var(--primary); color: white; border-color: var(--primary); } /* Kategoria "Do uzupełnienia" - żółty styl, z prawej strony */ .category-badge.category-todo { margin-left: auto; background: linear-gradient(135deg, #fbbf24 0%, #f59e0b 100%); color: #78350f; border-color: #f59e0b; font-weight: 600; } .category-badge.category-todo:hover { filter: brightness(1.1); box-shadow: 0 2px 8px rgba(245, 158, 11, 0.4); } .category-badge.category-todo.active { background: linear-gradient(135deg, #f59e0b 0%, #d97706 100%); box-shadow: 0 2px 8px rgba(245, 158, 11, 0.5); } /* 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: 0; padding: var(--spacing-lg); border: 1px solid #E4E4E4; transition: var(--transition); display: flex; flex-direction: column; height: 100%; } .company-card:hover { border-color: var(--primary); box-shadow: 0 10px 30px rgba(46, 72, 114, 0.15); } .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.dark-bg { background: #1a1a2e; } .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: #EDF0F5; color: #464646; font-size: 12px; font-weight: 500; text-transform: uppercase; letter-spacing: 0.5px; border-radius: 4px; 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-relation { display: inline-block; font-size: 11px; color: var(--text-secondary); background: var(--background); padding: 2px 8px; border-radius: var(--radius); margin-bottom: var(--spacing-xs); } .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; } } /* What's New Widget */ .whats-new-widget { display: flex; align-items: flex-start; gap: var(--spacing-md); background: var(--surface); border: 1px solid var(--border-color, #e5e7eb); border-left: 4px solid var(--primary); border-radius: var(--radius-lg); padding: var(--spacing-md) var(--spacing-lg); margin-bottom: var(--spacing-lg); } .whats-new-icon { font-size: 1.5rem; flex-shrink: 0; margin-top: 2px; } .whats-new-items { font-size: var(--font-size-sm); color: var(--text-secondary); margin-top: 4px; } .whats-new-item { padding: 3px 0; } .whats-new-link { color: var(--primary); font-weight: 500; font-size: var(--font-size-sm); white-space: nowrap; flex-shrink: 0; text-decoration: none; margin-top: 2px; } .whats-new-link:hover { text-decoration: underline; } @media (max-width: 768px) { .whats-new-widget { flex-wrap: wrap; gap: var(--spacing-xs); padding: var(--spacing-sm) var(--spacing-md); } .whats-new-icon { font-size: 1.2rem; } .whats-new-desc { display: none; } .whats-new-extra { display: none; } .whats-new-link { width: 100%; text-align: right; margin-top: 0; } } /* New on portal section */ .new-on-portal { margin-bottom: var(--spacing-lg); } .new-on-portal h2 { font-size: var(--font-size-lg); color: var(--text-primary); margin-bottom: var(--spacing-md); display: flex; align-items: center; gap: var(--spacing-sm); } .new-on-portal-grid { display: grid; grid-template-columns: repeat(4, 1fr); gap: var(--spacing-md); } .new-card { text-decoration: none; display: block; background: var(--surface); border: 1px solid var(--border); border-radius: var(--radius); padding: var(--spacing-md); transition: all 0.2s; } .new-card:hover { border-color: var(--primary); box-shadow: 0 2px 8px rgba(0,0,0,0.08); } .new-card-badge { display: inline-block; font-size: 10px; font-weight: 600; text-transform: uppercase; letter-spacing: 0.5px; padding: 2px 8px; border-radius: 4px; margin-bottom: var(--spacing-sm); } .new-card-badge.forum { background: #dbeafe; color: #1e40af; } .new-card-badge.b2b { background: #dcfce7; color: #166534; } .new-card-title { font-weight: 600; color: var(--text-primary); font-size: var(--font-size-sm); line-height: 1.4; margin-bottom: var(--spacing-sm); display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical; overflow: hidden; } .new-card-meta { font-size: var(--font-size-xs); color: var(--text-secondary); } @media (max-width: 768px) { .new-on-portal-grid { grid-template-columns: repeat(2, 1fr); } } {% endblock %} {% block content %} {% if current_user.is_authenticated and not current_user.is_norda_member and not current_user.company_id %} {% if pending_application %}
📋

Deklaracja członkowska w toku

{% if pending_application.status == 'submitted' or pending_application.status == 'under_review' %} Twoja deklaracja dla firmy "{{ pending_application.company_name }}" oczekuje na rozpatrzenie {% elif pending_application.status == 'pending_user_approval' %} Administrator zaproponował zmiany - sprawdź i zaakceptuj {% elif pending_application.status == 'changes_requested' %} Wymagane są poprawki w Twojej deklaracji {% else %} Kontynuuj wypełnianie deklaracji dla firmy "{{ pending_application.company_name or 'Twoja firma' }}" {% endif %}

{% if pending_application.status == 'draft' %}Kontynuuj →{% else %}Sprawdź status →{% endif %}
Przeglądasz listę firm Izby NORDA. Pełny dostęp do szczegółów firm otrzymasz po zatwierdzeniu deklaracji.
{% else %}
🤝

Dołącz do Izby Przedsiębiorców NORDA

Złóż deklarację członkowską i zyskaj pełny dostęp do katalog {{ total_companies }} firm, wydarzeń i funkcji portalu

Złóż deklarację →
🔒 Przeglądasz listę firm Izby NORDA. Aby zobaczyć szczegóły każdej firmy, złóż deklarację członkowską.
{% endif %} {% else %}

Katalog firm Norda Biznes

{{ COMPANY_COUNT }} podmiotów gospodarczych • 4 kategorie • 17 podkategorii

{% endif %} {% if latest_release %}
Co nowego na platformie? {{ latest_release.version }} · {{ latest_release.date }}
{% set all_items = (latest_release.new or []) + (latest_release.improve or []) %} {% set starred = [] %} {% for item in all_items %}{% if item.startswith('★') %}{% if starred.append(item) %}{% endif %}{% endif %}{% endfor %} {% set starred_links = latest_release.get('links', {}) if latest_release.get is defined else {} %} {% if starred %} {% for item in starred %} {% set item_title = item|striptags|replace('★ ', '') %} {% set item_title_short = item_title.split(' - ')[0]|trim %} {% set item_desc = item_title.split(' - ')[1]|trim if ' - ' in item_title else '' %} {% set link = starred_links.get(item_title_short, '') if starred_links.get is defined else '' %}
{{ item_title_short }}{{ ' - ' + item_desc if item_desc else '' }} {% if link %} Wypróbuj → {% endif %}
{% endfor %} {% else %} {% set highlight = (all_items[0] if all_items else '')|striptags %}
{{ highlight }}
{% endif %} {% set total_changes = (latest_release.new|length if latest_release.new else 0) + (latest_release.improve|length if latest_release.improve else 0) + (latest_release.fix|length if latest_release.fix else 0) %} {% set starred_count = starred|length %} {% set hidden_starred = starred|length - 2 if starred|length > 2 else 0 %} {% set extra_count = total_changes - starred_count + hidden_starred %} {% if extra_count > 0 %}
+ {{ extra_count }} {{ 'inna zmiana' if extra_count == 1 else ('inne zmiany' if extra_count < 5 else 'innych zmian') }} →
{% endif %}
Zobacz wszystko →
{% endif %} {% if upcoming_events %}
Pokaż:
{% endif %} {% if upcoming_events or newest_users or latest_admitted %}
{% if upcoming_events %} {% for ue in upcoming_events[:2] %} {% set ev = ue.event %}
📅
{% if loop.first %}
Najbliższe wydarzenia – Kto weźmie udział?
{% endif %}
{{ ev.title }} → {% if ev.is_external and ev.external_source %} 🌐 {{ ev.external_source }} {% endif %}
📆 {{ ev.event_date.strftime('%d.%m.%Y') }} ({{ ['Pon', 'Wt', 'Śr', 'Czw', 'Pt', 'Sob', 'Nd'][ev.event_date.weekday()] }}) {% if ev.time_start %} 🕕 {{ ev.time_start.strftime('%H:%M') }} {% endif %} {% if ev.location %} 📍 {{ ev.location[:30] }}{% if ev.location|length > 30 %}...{% endif %} {% endif %}
👥 Zapisanych: {{ ev.total_attendee_count }} {% if ev.total_attendee_count == 1 %}osoba{% elif ev.total_attendee_count in [2,3,4] %}osoby{% else %}osób{% endif %}
{% if ue.user_registered %} ✓ Jesteś zapisany/a {% elif ue.user_can_attend %} {% endif %}
{% endfor %} {% endif %}
{% if newest_users %}
👤 Nowi użytkownicy portalu
{% for user in newest_users %}
{% if user.avatar_path %} {% else %} {{ (user.name or user.email)[0].upper() }} {% endif %}
{{ user.name or user.email.split('@')[0] }}
{{ user.created_at|local_time('%d.%m.%Y') }}{% if user.company %} · {{ user.company.name }}{% endif %}
{% endfor %}
{% endif %}
🏢 Nowi członkowie Izby {% if last_meeting %} · Rada {{ last_meeting.meeting_date.strftime('%d.%m.%Y') }} {% endif %}
{% if latest_admitted %}
{% for company in latest_admitted[:4] %}
{{ company.name }} {% if company.status != 'active' %} · profil w trakcie uzupełniania {% endif %}
{% endfor %} {% if latest_admitted|length > 4 %}
+ {{ latest_admitted|length - 4 }} więcej
{% endif %}
{% else %}
Wkrótce informacje o nowych członkach przyjętych na posiedzeniu Rady.
{% endif %}
Zobacz wszystkich członków →
{% endif %} {% if latest_forum_topics or latest_classifieds %}

📋 Nowe na portalu

{% for item in latest_forum_topics %} 💬 Forum
{{ item.topic.title }}
{{ item.last_author.name if item.last_author else 'Anonim' }} · {{ item.last_activity|local_time('%d.%m.%Y') }}
{% endfor %} {% for cl in latest_classifieds %} 🏢 B2B {% if cl.listing_type == 'szukam' %}Szukam{% else %}Oferuję{% endif %}
{{ cl.title }}
{{ cl.author.name if cl.author else 'Anonim' }} · {{ (cl.updated_at or cl.created_at)|local_time('%d.%m.%Y') }}
{% endfor %}
{% endif %} {% if current_user.is_authenticated %}
NordaGPT
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

NordaGPT

Gemini 3
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 current_user.is_authenticated and zopk_facts %}
💡

Czy wiesz, że... Zielony Okręg Przemysłowy Kaszubia

{% for fact in zopk_facts %} {% if fact.fact_type == 'investment' %}inwestycja{% elif fact.fact_type == 'decision' %}decyzja{% elif fact.fact_type == 'event' %}wydarzenie{% elif fact.fact_type == 'milestone' %}kamień milowy{% elif fact.fact_type == 'statistic' %}dane{% elif fact.fact_type == 'partnership' %}współpraca{% elif fact.fact_type == 'project' %}projekt{% else %}fakt{% endif %}

{{ fact.full_text[:200] }}{% if fact.full_text|length > 200 %}...{% endif %}

{% if fact.source_news %}
{{ fact.source_news.source_name or fact.source_news.source_domain }} • {{ fact.source_news.published_at|local_time('%d.%m.%Y') if fact.source_news.published_at else '' }} Czytaj →
{% endif %}
{% endfor %}
{% endif %}
{% if main_categories %}
{# Collect categories with counts and sort by count descending #} {% set cat_counts = [] %} {% for main_cat in main_categories %} {% set main_count = companies|selectattr('category_id', 'equalto', main_cat.id)|list|length %} {% set sub_count = namespace(val=0) %} {% for sub in main_cat.subcategories %} {% set sub_count.val = sub_count.val + companies|selectattr('category_id', 'equalto', sub.id)|list|length %} {% endfor %} {% set total_count = main_count + sub_count.val %} {% if total_count > 0 %} {% set _ = cat_counts.append({'cat': main_cat, 'count': total_count}) %} {% endif %} {% endfor %} {# Renderuj normalne kategorie (bez "do-uzupelnienia") #} {% for item in cat_counts|sort(attribute='count', reverse=true) %} {% if item.cat.slug != 'do-uzupelnienia' %} {% endif %} {% endfor %} {# Kategoria "Do uzupełnienia" osobno, z prawej strony (żółta) #} {% for item in cat_counts if item.cat.slug == 'do-uzupelnienia' %} {% endfor %}
{% for main_cat in main_categories %}
{# Zbierz podkategorie z licznikami i posortuj malejąco #} {% set sub_counts = [] %} {% for sub in main_cat.subcategories %} {% set count = companies|selectattr('category_id', 'equalto', sub.id)|list|length %} {% if count > 0 %} {% set _ = sub_counts.append({'sub': sub, 'count': count}) %} {% endif %} {% endfor %} {% for item in sub_counts|sort(attribute='count', reverse=true) %} {% endfor %}
{% endfor %}
{% elif 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 }} {% if company_parent and company.id in company_parent %} marka {{ company_parent[company.id] }} {% elif company_children and company.id in company_children %} {{ company_children[company.id]|join(', ') }} {% endif %}
{{ 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 %} // Filter events by type function filterEvents(type, btn) { document.querySelectorAll('.events-filter-btn').forEach(b => b.classList.remove('active')); btn.classList.add('active'); fetch('/api/upcoming-events?filter=' + type + '&limit=2') .then(r => r.json()) .then(data => { const row = document.querySelector('.events-row'); if (!data.events || data.events.length === 0) { row.innerHTML = '
Brak wydarzeń w tej kategorii
'; return; } row.innerHTML = data.events.slice(0, 2).map((ev, i) => { let typeCls = ev.is_external ? 'external' : 'norda'; let sourceBadge = ev.is_external && ev.external_source ? '🌐 ' + ev.external_source + '' : ''; let accessBadge = ''; if (ev.access_level === 'admin_only') accessBadge = 'UKRYTE'; else if (ev.access_level === 'rada_only') accessBadge = 'IZBA'; let timePart = ev.time ? '🕕 ' + ev.time + '' : ''; let locPart = ev.location ? '📍 ' + ev.location + '' : ''; let cnt = ev.attendee_count; let cntWord = cnt === 1 ? 'osoba' : (cnt >= 2 && cnt <= 4 ? 'osoby' : 'osób'); let actionHtml = ''; if (ev.user_registered) { actionHtml = '✓ Jesteś zapisany/a'; } else if (ev.user_can_attend) { actionHtml = ''; } else if (ev.access_level === 'rada_only') { actionHtml = '🔒 Rada Izby'; } let label = i === 0 ? '
Najbliższe wydarzenia – Kto weźmie udział?
' : ''; return '' + '
📅
' + label + '
' + ev.title + ' →' + sourceBadge + accessBadge + '
' + '
📆 ' + ev.date + ' (' + ev.day + ')' + timePart + locPart + '
' + '
' + '
' + '
👥 Zapisanych: ' + cnt + ' ' + cntWord + '
' + '
' + actionHtml + '
' + '
'; }).join(''); }); } // 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 - show all function filterCategory(slug) { const cards = document.querySelectorAll('.company-card'); // Hide all subcategory rows document.querySelectorAll('.category-filter-sub').forEach(row => { row.classList.remove('visible'); }); // Remove active from all badges document.querySelectorAll('.category-badge').forEach(badge => { badge.classList.remove('active'); }); // Activate "Wszystkie" button const allBtn = document.querySelector('.category-badge[onclick*="filterCategory(\'all\')"]'); if (allBtn) allBtn.classList.add('active'); // Show all cards cards.forEach(card => { card.style.display = 'flex'; }); } // Select main category - show subcategories and filter function selectMainCategory(mainSlug) { const cards = document.querySelectorAll('.company-card'); // Hide all subcategory rows, show only selected document.querySelectorAll('.category-filter-sub').forEach(row => { row.classList.remove('visible'); }); const subRow = document.getElementById('subcats-' + mainSlug); if (subRow) subRow.classList.add('visible'); // Update active badges document.querySelectorAll('.category-badge').forEach(badge => { badge.classList.remove('active'); }); const mainBtn = document.querySelector('.category-badge.category-main[data-main-slug="' + mainSlug + '"]'); if (mainBtn) mainBtn.classList.add('active'); // Get all valid slugs (main + subcategories) const validSlugs = [mainSlug]; document.querySelectorAll('.category-badge.category-sub[data-parent="' + mainSlug + '"]').forEach(badge => { const onclick = badge.getAttribute('onclick'); if (onclick) { const match = onclick.match(/filterSubCategory\('([^']+)'/); if (match) validSlugs.push(match[1]); } }); // Filter cards cards.forEach(card => { const cardCategory = card.getAttribute('data-category'); card.style.display = validSlugs.includes(cardCategory) ? 'flex' : 'none'; }); } // Filter by subcategory function filterSubCategory(subSlug, parentSlug) { const cards = document.querySelectorAll('.company-card'); // Keep subcategory row visible document.querySelectorAll('.category-filter-sub').forEach(row => { row.classList.remove('visible'); }); const subRow = document.getElementById('subcats-' + parentSlug); if (subRow) subRow.classList.add('visible'); // Update active badges - main stays highlighted, sub is active document.querySelectorAll('.category-badge').forEach(badge => { badge.classList.remove('active'); }); const mainBtn = document.querySelector('.category-badge.category-main[data-main-slug="' + parentSlug + '"]'); if (mainBtn) mainBtn.classList.add('active'); // Find and activate the sub badge document.querySelectorAll('.category-badge.category-sub[data-parent="' + parentSlug + '"]').forEach(badge => { const onclick = badge.getAttribute('onclick'); if (onclick && onclick.includes("'" + subSlug + "'")) { badge.classList.add('active'); } }); // Filter cards - only show matching subcategory cards.forEach(card => { const cardCategory = card.getAttribute('data-category'); card.style.display = cardCategory === subSlug ? '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(); } }); // ZOPK "Czy wiesz, że..." — load more facts let zopkOffset = 3; async function loadMoreFacts() { const btn = document.getElementById('zopkMoreBtn'); btn.textContent = 'Ładowanie...'; btn.disabled = true; try { const resp = await fetch('/api/zopk-facts?offset=' + zopkOffset); const data = await resp.json(); if (data.facts && data.facts.length > 0) { const grid = btn.closest('div[data-animate]').querySelector('[style*="display: grid"]'); const typeColors = {investment:'rgba(16,185,129,0.3)',event:'rgba(59,130,246,0.3)',decision:'rgba(245,158,11,0.3)',milestone:'rgba(139,92,246,0.3)'}; data.facts.forEach(f => { const card = document.createElement('a'); card.href = f.source_url || '/zopk'; card.target = '_blank'; card.rel = 'noopener'; card.style.cssText = 'background:rgba(255,255,255,0.12);border-radius:var(--radius);padding:var(--spacing-md);text-decoration:none;color:white;display:block;transition:background 0.2s,transform 0.2s;cursor:pointer;opacity:0;'; card.onmouseover = function(){this.style.background='rgba(255,255,255,0.22)';this.style.transform='translateY(-2px)';}; card.onmouseout = function(){this.style.background='rgba(255,255,255,0.12)';this.style.transform='none';}; const color = typeColors[f.type] || 'rgba(255,255,255,0.2)'; card.innerHTML = ''+f.type_label+'' + '

'+f.text+(f.text.length>=200?'...':'')+'

' + '
'+f.source_name+' • '+f.source_date+'Czytaj →
'; grid.appendChild(card); requestAnimationFrame(() => { card.style.transition = 'opacity 0.4s'; card.style.opacity = '1'; }); }); zopkOffset += data.facts.length; if (!data.has_more) { btn.textContent = 'To wszystko'; btn.disabled = true; btn.style.opacity = '0.5'; } else { btn.textContent = 'Pokaż więcej'; btn.disabled = false; } } else { btn.textContent = 'Brak więcej faktów'; btn.disabled = true; btn.style.opacity = '0.5'; } } catch(e) { btn.textContent = 'Pokaż więcej'; btn.disabled = false; } } {% endblock %}