Some checks are pending
NordaBiz Tests / Unit & Integration Tests (push) Waiting to run
NordaBiz Tests / E2E Tests (Playwright) (push) Blocked by required conditions
NordaBiz Tests / Smoke Tests (Production) (push) Blocked by required conditions
NordaBiz Tests / Send Failure Notification (push) Blocked by required conditions
Store previous analysis before regeneration and show comparison table with priority breakdown, new/removed actions diff. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
1217 lines
54 KiB
HTML
1217 lines
54 KiB
HTML
{% extends "base.html" %}
|
|
|
|
{% block title %}Audyt SEO - {{ company.name }} - Norda Biznes Partner{% endblock %}
|
|
|
|
{% block extra_css %}
|
|
<style>
|
|
.audit-header {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: flex-start;
|
|
margin-bottom: var(--spacing-xl);
|
|
flex-wrap: wrap;
|
|
gap: var(--spacing-md);
|
|
}
|
|
|
|
.audit-header-info h1 {
|
|
font-size: var(--font-size-2xl);
|
|
color: var(--text-primary);
|
|
margin-bottom: var(--spacing-xs);
|
|
}
|
|
|
|
.audit-header-info p {
|
|
color: var(--text-secondary);
|
|
font-size: var(--font-size-sm);
|
|
}
|
|
|
|
.data-source-info {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
gap: var(--spacing-xs);
|
|
margin-top: var(--spacing-sm);
|
|
padding: var(--spacing-xs) var(--spacing-sm);
|
|
background: var(--success-light, #dcfce7);
|
|
border-radius: var(--radius);
|
|
font-size: var(--font-size-sm);
|
|
color: var(--success, #16a34a);
|
|
}
|
|
|
|
.header-actions {
|
|
display: flex;
|
|
gap: var(--spacing-sm);
|
|
align-items: center;
|
|
}
|
|
|
|
/* Score Section */
|
|
.score-section {
|
|
display: grid;
|
|
grid-template-columns: auto 1fr;
|
|
gap: var(--spacing-xl);
|
|
margin-bottom: var(--spacing-xl);
|
|
background: var(--surface);
|
|
padding: var(--spacing-xl);
|
|
border-radius: var(--radius-lg);
|
|
box-shadow: var(--shadow);
|
|
}
|
|
|
|
@media (max-width: 768px) {
|
|
.score-section {
|
|
grid-template-columns: 1fr;
|
|
text-align: center;
|
|
}
|
|
}
|
|
|
|
.score-circle {
|
|
width: 180px;
|
|
height: 180px;
|
|
border-radius: 50%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
justify-content: center;
|
|
position: relative;
|
|
background: conic-gradient(
|
|
var(--score-color, var(--secondary)) calc(var(--score-percent, 0) * 3.6deg),
|
|
#e2e8f0 0deg
|
|
);
|
|
margin: 0 auto;
|
|
}
|
|
|
|
.score-circle::before {
|
|
content: '';
|
|
position: absolute;
|
|
width: 150px;
|
|
height: 150px;
|
|
border-radius: 50%;
|
|
background: var(--surface);
|
|
}
|
|
|
|
.score-value {
|
|
position: relative;
|
|
z-index: 1;
|
|
font-size: 3rem;
|
|
font-weight: 700;
|
|
line-height: 1;
|
|
}
|
|
|
|
/* Unified 5-level color scale: 0-29 red, 30-49 orange, 50-69 amber, 70-89 lime, 90-100 green */
|
|
.score-value.score-excellent { color: #10b981; }
|
|
.score-value.score-good { color: #84cc16; }
|
|
.score-value.score-average { color: #f59e0b; }
|
|
.score-value.score-needs-work { color: #f97316; }
|
|
.score-value.score-poor { color: #ef4444; }
|
|
|
|
.score-label {
|
|
position: relative;
|
|
z-index: 1;
|
|
font-size: var(--font-size-sm);
|
|
color: var(--text-secondary);
|
|
margin-top: var(--spacing-xs);
|
|
}
|
|
|
|
.score-details {
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: center;
|
|
}
|
|
|
|
.score-category {
|
|
font-size: var(--font-size-xl);
|
|
font-weight: 600;
|
|
margin-bottom: var(--spacing-sm);
|
|
}
|
|
|
|
/* Unified 5-level color scale */
|
|
.score-category.excellent { color: #10b981; }
|
|
.score-category.good { color: #84cc16; }
|
|
.score-category.average { color: #f59e0b; }
|
|
.score-category.needs-work { color: #f97316; }
|
|
.score-category.poor { color: #ef4444; }
|
|
|
|
.score-description {
|
|
color: var(--text-secondary);
|
|
line-height: 1.6;
|
|
margin-bottom: var(--spacing-md);
|
|
}
|
|
|
|
.audit-meta {
|
|
display: flex;
|
|
gap: var(--spacing-lg);
|
|
font-size: var(--font-size-sm);
|
|
color: var(--text-secondary);
|
|
flex-wrap: wrap;
|
|
}
|
|
|
|
.audit-meta-item {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: var(--spacing-xs);
|
|
}
|
|
|
|
/* Metrics Grid */
|
|
.metrics-grid {
|
|
display: grid;
|
|
grid-template-columns: repeat(auto-fill, minmax(220px, 1fr));
|
|
gap: var(--spacing-md);
|
|
margin-bottom: var(--spacing-xl);
|
|
}
|
|
|
|
.metric-card {
|
|
background: var(--surface);
|
|
border-radius: var(--radius-lg);
|
|
padding: var(--spacing-lg);
|
|
box-shadow: var(--shadow-sm);
|
|
text-align: center;
|
|
border-left: 4px solid var(--border);
|
|
}
|
|
|
|
.metric-card.good { border-left-color: var(--success); }
|
|
.metric-card.medium { border-left-color: var(--warning); }
|
|
.metric-card.poor { border-left-color: var(--error); }
|
|
.metric-card.none { border-left-color: var(--border); }
|
|
|
|
.metric-icon {
|
|
width: 48px;
|
|
height: 48px;
|
|
margin: 0 auto var(--spacing-sm);
|
|
border-radius: 12px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
}
|
|
|
|
.metric-icon.good { background: #dcfce7; color: #16a34a; }
|
|
.metric-icon.medium { background: #fef3c7; color: #d97706; }
|
|
.metric-icon.poor { background: #fee2e2; color: #dc2626; }
|
|
.metric-icon.none { background: var(--bg-tertiary); color: var(--text-tertiary); }
|
|
|
|
.metric-name {
|
|
font-size: var(--font-size-sm);
|
|
color: var(--text-secondary);
|
|
margin-bottom: var(--spacing-xs);
|
|
text-transform: uppercase;
|
|
letter-spacing: 0.5px;
|
|
}
|
|
|
|
.metric-value {
|
|
font-size: var(--font-size-2xl);
|
|
font-weight: 700;
|
|
}
|
|
|
|
.metric-value.good { color: var(--success); }
|
|
.metric-value.medium { color: var(--warning); }
|
|
.metric-value.poor { color: var(--error); }
|
|
.metric-value.none { color: var(--text-tertiary); }
|
|
|
|
/* No Audit State */
|
|
.no-audit-state {
|
|
text-align: center;
|
|
padding: var(--spacing-2xl);
|
|
background: var(--surface);
|
|
border-radius: var(--radius-lg);
|
|
box-shadow: var(--shadow);
|
|
}
|
|
|
|
.no-audit-state svg {
|
|
width: 80px;
|
|
height: 80px;
|
|
color: var(--text-secondary);
|
|
opacity: 0.5;
|
|
margin-bottom: var(--spacing-md);
|
|
}
|
|
|
|
.no-audit-state h2 {
|
|
font-size: var(--font-size-xl);
|
|
color: var(--text-primary);
|
|
margin-bottom: var(--spacing-sm);
|
|
}
|
|
|
|
.no-audit-state p {
|
|
color: var(--text-secondary);
|
|
margin-bottom: var(--spacing-lg);
|
|
}
|
|
|
|
/* Section Title */
|
|
.section-title {
|
|
font-size: var(--font-size-xl);
|
|
font-weight: 600;
|
|
color: var(--text-primary);
|
|
margin-bottom: var(--spacing-md);
|
|
display: flex;
|
|
align-items: center;
|
|
gap: var(--spacing-sm);
|
|
}
|
|
|
|
/* Breadcrumb */
|
|
.breadcrumb {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: var(--spacing-sm);
|
|
font-size: var(--font-size-sm);
|
|
color: var(--text-secondary);
|
|
margin-bottom: var(--spacing-md);
|
|
}
|
|
|
|
.breadcrumb a {
|
|
color: var(--primary);
|
|
text-decoration: none;
|
|
}
|
|
|
|
.breadcrumb a:hover {
|
|
text-decoration: underline;
|
|
}
|
|
|
|
.breadcrumb-separator {
|
|
color: var(--border);
|
|
}
|
|
|
|
/* Loading Overlay */
|
|
.loading-overlay {
|
|
display: none;
|
|
position: fixed;
|
|
top: 0;
|
|
left: 0;
|
|
right: 0;
|
|
bottom: 0;
|
|
background: rgba(255, 255, 255, 0.95);
|
|
z-index: 1000;
|
|
align-items: center;
|
|
justify-content: center;
|
|
flex-direction: column;
|
|
gap: var(--spacing-lg);
|
|
}
|
|
|
|
.loading-overlay.active {
|
|
display: flex;
|
|
}
|
|
|
|
.loading-content {
|
|
background: var(--surface);
|
|
padding: var(--spacing-xl);
|
|
border-radius: var(--radius-lg);
|
|
box-shadow: var(--shadow-lg);
|
|
max-width: 400px;
|
|
width: 90%;
|
|
text-align: center;
|
|
}
|
|
|
|
.loading-spinner {
|
|
width: 48px;
|
|
height: 48px;
|
|
border: 4px solid var(--border);
|
|
border-top-color: var(--primary);
|
|
border-radius: 50%;
|
|
animation: spin 1s linear infinite;
|
|
margin: 0 auto var(--spacing-md);
|
|
}
|
|
|
|
@keyframes spin {
|
|
to { transform: rotate(360deg); }
|
|
}
|
|
|
|
/* Modal */
|
|
.modal {
|
|
display: none;
|
|
position: fixed;
|
|
top: 0;
|
|
left: 0;
|
|
width: 100%;
|
|
height: 100%;
|
|
background: rgba(0, 0, 0, 0.5);
|
|
z-index: 1000;
|
|
align-items: center;
|
|
justify-content: center;
|
|
}
|
|
|
|
.modal.active {
|
|
display: flex;
|
|
}
|
|
|
|
.modal-content {
|
|
background: var(--surface);
|
|
padding: var(--spacing-xl);
|
|
border-radius: var(--radius-lg);
|
|
max-width: 480px;
|
|
width: 90%;
|
|
box-shadow: var(--shadow-lg);
|
|
}
|
|
|
|
.modal-header {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: var(--spacing-md);
|
|
margin-bottom: var(--spacing-md);
|
|
}
|
|
|
|
.modal-icon {
|
|
width: 48px;
|
|
height: 48px;
|
|
border-radius: 50%;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
}
|
|
|
|
.modal-icon.success { background: #dcfce7; color: #16a34a; }
|
|
.modal-icon.info { background: #dbeafe; color: #2563eb; }
|
|
|
|
.modal-title {
|
|
font-size: var(--font-size-xl);
|
|
font-weight: 600;
|
|
}
|
|
|
|
.modal-body {
|
|
color: var(--text-secondary);
|
|
margin-bottom: var(--spacing-lg);
|
|
}
|
|
|
|
.modal-footer {
|
|
display: flex;
|
|
justify-content: flex-end;
|
|
gap: var(--spacing-sm);
|
|
}
|
|
|
|
/* Website URL display */
|
|
.website-url {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
gap: var(--spacing-xs);
|
|
padding: var(--spacing-xs) var(--spacing-sm);
|
|
background: var(--bg-tertiary);
|
|
border-radius: var(--radius);
|
|
font-size: var(--font-size-sm);
|
|
color: var(--text-secondary);
|
|
margin-top: var(--spacing-sm);
|
|
}
|
|
|
|
.website-url a {
|
|
color: var(--primary);
|
|
text-decoration: none;
|
|
}
|
|
|
|
.website-url a:hover {
|
|
text-decoration: underline;
|
|
}
|
|
</style>
|
|
{% endblock %}
|
|
|
|
{% block content %}
|
|
<!-- Breadcrumb -->
|
|
<div class="breadcrumb">
|
|
<a href="{{ url_for('index') }}">Firmy</a>
|
|
<span class="breadcrumb-separator">/</span>
|
|
<a href="{{ url_for('company_detail', company_id=company.id) }}">{{ company.name }}</a>
|
|
<span class="breadcrumb-separator">/</span>
|
|
<span>Audyt SEO</span>
|
|
</div>
|
|
|
|
<div class="audit-header">
|
|
<div class="audit-header-info">
|
|
<h1>Audyt SEO Strony WWW</h1>
|
|
<p>{{ company.name }}</p>
|
|
<div class="data-source-info">
|
|
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"/>
|
|
</svg>
|
|
<span>Analiza SEO i wydajnosci strony WWW (Google PageSpeed Insights)</span>
|
|
</div>
|
|
{% if company.website %}
|
|
<div class="website-url">
|
|
<svg width="14" height="14" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 12a9 9 0 01-9 9m9-9a9 9 0 00-9-9m9 9H3m9 9a9 9 0 01-9-9m9 9c1.657 0 3-4.03 3-9s-1.343-9-3-9m0 18c-1.657 0-3-4.03-3-9s1.343-9 3-9m-9 9a9 9 0 019-9"/>
|
|
</svg>
|
|
<a href="{{ company.website }}" target="_blank" rel="noopener">{{ company.website }}</a>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
<div class="header-actions">
|
|
<a href="{{ url_for('company_detail', company_id=company.id) }}" class="btn btn-outline btn-sm">
|
|
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 19l-7-7m0 0l7-7m-7 7h18"/>
|
|
</svg>
|
|
Profil firmy
|
|
</a>
|
|
{% if can_audit and company.website %}
|
|
<button class="btn btn-primary btn-sm" onclick="runAudit()" id="runAuditBtn">
|
|
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"/>
|
|
</svg>
|
|
Uruchom audyt
|
|
</button>
|
|
{% endif %}
|
|
</div>
|
|
</div>
|
|
|
|
{% if seo_data %}
|
|
<!-- Score Section -->
|
|
{# Unified 5-level color scale: 0-29 red, 30-49 orange, 50-69 amber, 70-89 lime, 90-100 green #}
|
|
{% set score = seo_data.seo_score or 0 %}
|
|
<div class="score-section">
|
|
<div class="score-circle" style="--score-percent: {{ score }}; --score-color: {% if score >= 90 %}#10b981{% elif score >= 70 %}#84cc16{% elif score >= 50 %}#f59e0b{% elif score >= 30 %}#f97316{% else %}#ef4444{% endif %};">
|
|
<span class="score-value" style="color: {% if score >= 90 %}#10b981{% elif score >= 70 %}#84cc16{% elif score >= 50 %}#f59e0b{% elif score >= 30 %}#f97316{% else %}#ef4444{% endif %};">{{ score }}</span>
|
|
<span class="score-label">/ 100</span>
|
|
</div>
|
|
<div class="score-details">
|
|
<div class="score-category" style="color: {% if score >= 90 %}#10b981{% elif score >= 70 %}#84cc16{% elif score >= 50 %}#f59e0b{% elif score >= 30 %}#f97316{% else %}#ef4444{% endif %};">
|
|
{% if score >= 90 %}
|
|
Doskonaly wynik SEO (Google Lighthouse)
|
|
{% elif score >= 70 %}
|
|
Dobry wynik SEO (Google Lighthouse)
|
|
{% elif score >= 50 %}
|
|
Przecietny wynik SEO (Google Lighthouse)
|
|
{% elif score >= 30 %}
|
|
Wynik SEO wymaga poprawy (Google Lighthouse)
|
|
{% else %}
|
|
Slaby wynik SEO (Google Lighthouse)
|
|
{% endif %}
|
|
</div>
|
|
<p class="score-description">
|
|
{% if score >= 90 %}
|
|
Strona jest bardzo dobrze zoptymalizowana pod katem SEO. Utrzymuj wysoki standard i monitoruj zmiany.
|
|
{% elif score >= 70 %}
|
|
Strona ma dobra optymalizacje SEO, ale sa obszary do poprawy. Skup sie na wydajnosci i dostepnosci.
|
|
{% elif score >= 50 %}
|
|
Strona wymaga pracy nad optymalizacja SEO. Warto poprawic wydajnosc i dostepnosc.
|
|
{% else %}
|
|
Strona ma powazne problemy z SEO. Priorytetowo popraw wydajnosc i optymalizacje.
|
|
{% endif %}
|
|
</p>
|
|
<p style="font-size: var(--font-size-xs); color: var(--text-tertiary); margin-top: var(--spacing-xs);">
|
|
Wynik pochodzi z Google PageSpeed Insights i ocenia techniczne aspekty SEO (meta tagi, robots.txt, indeksowalnosc).
|
|
Pelna ocena SEO, wlaczajac lokalne SEO i widocznosc, jest dostepna w analizie AI ponizej.
|
|
</p>
|
|
<div class="audit-meta">
|
|
<div class="audit-meta-item">
|
|
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
|
|
</svg>
|
|
<span>Ostatni audyt: {{ seo_data.audited_at.strftime('%d.%m.%Y %H:%M') if seo_data.audited_at else 'Brak danych' }}</span>
|
|
</div>
|
|
{% if seo_data.url %}
|
|
<div class="audit-meta-item">
|
|
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13.828 10.172a4 4 0 00-5.656 0l-4 4a4 4 0 105.656 5.656l1.102-1.101m-.758-4.899a4 4 0 005.656 0l4-4a4 4 0 00-5.656-5.656l-1.1 1.1"/>
|
|
</svg>
|
|
<span>{{ seo_data.url|truncate(40) }}</span>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Metrics Grid -->
|
|
<h2 class="section-title">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 19v-6a2 2 0 00-2-2H5a2 2 0 00-2 2v6a2 2 0 002 2h2a2 2 0 002-2zm0 0V9a2 2 0 012-2h2a2 2 0 012 2v10m-6 0a2 2 0 002 2h2a2 2 0 002-2m0 0V5a2 2 0 012-2h2a2 2 0 012 2v14a2 2 0 01-2 2h-2a2 2 0 01-2-2z"/>
|
|
</svg>
|
|
Szczegolowe metryki
|
|
</h2>
|
|
|
|
<div class="metrics-grid">
|
|
<!-- SEO Score -->
|
|
{% set seo = seo_data.seo_score %}
|
|
{% set seo_class = 'good' if seo and seo >= 90 else ('medium' if seo and seo >= 50 else ('poor' if seo else 'none')) %}
|
|
<div class="metric-card {{ seo_class }}">
|
|
<div class="metric-icon {{ seo_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">Wynik SEO</div>
|
|
<div class="metric-value {{ seo_class }}">{{ seo if seo else '-' }}</div>
|
|
</div>
|
|
|
|
<!-- Performance -->
|
|
{% set perf = seo_data.performance_score %}
|
|
{% set perf_class = 'good' if perf and perf >= 90 else ('medium' if perf and perf >= 50 else ('poor' if perf else 'none')) %}
|
|
<div class="metric-card {{ perf_class }}">
|
|
<div class="metric-icon {{ perf_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 10V3L4 14h7v7l9-11h-7z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">Wydajnosc</div>
|
|
<div class="metric-value {{ perf_class }}">{{ perf if perf else '-' }}</div>
|
|
</div>
|
|
|
|
<!-- Accessibility -->
|
|
{% set acc = seo_data.accessibility_score %}
|
|
{% set acc_class = 'good' if acc and acc >= 90 else ('medium' if acc and acc >= 50 else ('poor' if acc else 'none')) %}
|
|
<div class="metric-card {{ acc_class }}">
|
|
<div class="metric-icon {{ acc_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"/>
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">Dostepnosc</div>
|
|
<div class="metric-value {{ acc_class }}">{{ acc if acc else '-' }}</div>
|
|
</div>
|
|
|
|
<!-- Best Practices -->
|
|
{% set bp = seo_data.best_practices_score %}
|
|
{% set bp_class = 'good' if bp and bp >= 90 else ('medium' if bp and bp >= 50 else ('poor' if bp else 'none')) %}
|
|
<div class="metric-card {{ bp_class }}">
|
|
<div class="metric-icon {{ bp_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">Best Practices</div>
|
|
<div class="metric-value {{ bp_class }}">{{ bp if bp else '-' }}</div>
|
|
</div>
|
|
</div>
|
|
|
|
{% if seo_data.lcp_ms is not none or seo_data.fid_ms is not none or seo_data.cls is not none %}
|
|
<!-- Core Web Vitals -->
|
|
<h2 class="section-title">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 10V3L4 14h7v7l9-11h-7z"/>
|
|
</svg>
|
|
Core Web Vitals
|
|
</h2>
|
|
|
|
<div class="metrics-grid">
|
|
{% if seo_data.lcp_ms is not none %}
|
|
{% set lcp = seo_data.lcp_ms %}
|
|
{% set lcp_class = 'good' if lcp < 2500 else ('medium' if lcp < 4000 else 'poor') %}
|
|
<div class="metric-card {{ lcp_class }}">
|
|
<div class="metric-icon {{ lcp_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">LCP</div>
|
|
<div class="metric-value {{ lcp_class }}">{{ '%.1f'|format(lcp / 1000) }}s</div>
|
|
<div style="font-size: var(--font-size-xs); color: var(--text-tertiary); margin-top: 4px;">Largest Contentful Paint</div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.fid_ms is not none %}
|
|
{% set fid = seo_data.fid_ms %}
|
|
{% set fid_class = 'good' if fid < 100 else ('medium' if fid < 300 else 'poor') %}
|
|
<div class="metric-card {{ fid_class }}">
|
|
<div class="metric-icon {{ fid_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 15l-2 5L9 9l11 4-5 2zm0 0l5 5M7.188 2.239l.777 2.897M5.136 7.965l-2.898-.777M13.95 4.05l-2.122 2.122m-5.657 5.656l-2.12 2.122"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">FID</div>
|
|
<div class="metric-value {{ fid_class }}">{{ fid }}ms</div>
|
|
<div style="font-size: var(--font-size-xs); color: var(--text-tertiary); margin-top: 4px;">First Input Delay</div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.cls is not none %}
|
|
{% set cls = seo_data.cls %}
|
|
{% set cls_class = 'good' if cls < 0.1 else ('medium' if cls < 0.25 else 'poor') %}
|
|
<div class="metric-card {{ cls_class }}">
|
|
<div class="metric-icon {{ cls_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 5a1 1 0 011-1h14a1 1 0 011 1v2a1 1 0 01-1 1H5a1 1 0 01-1-1V5zM4 13a1 1 0 011-1h6a1 1 0 011 1v6a1 1 0 01-1 1H5a1 1 0 01-1-1v-6zM16 13a1 1 0 011-1h2a1 1 0 011 1v6a1 1 0 01-1 1h-2a1 1 0 01-1-1v-6z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">CLS</div>
|
|
<div class="metric-value {{ cls_class }}">{{ '%.3f'|format(cls) }}</div>
|
|
<div style="font-size: var(--font-size-xs); color: var(--text-tertiary); margin-top: 4px;">Cumulative Layout Shift</div>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.local_seo_score is not none %}
|
|
<!-- Local SEO Section -->
|
|
<h2 class="section-title">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z"/>
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 11a3 3 0 11-6 0 3 3 0 016 0z"/>
|
|
</svg>
|
|
Local SEO
|
|
</h2>
|
|
|
|
{% set lscore = seo_data.local_seo_score %}
|
|
<div style="background: var(--surface); padding: var(--spacing-lg); border-radius: var(--radius-lg); box-shadow: var(--shadow); margin-bottom: var(--spacing-xl);">
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-lg); margin-bottom: var(--spacing-lg);">
|
|
<div style="width: 80px; height: 80px; border-radius: 50%; display: flex; align-items: center; justify-content: center; font-size: 1.5rem; font-weight: 700; color: {% if lscore >= 70 %}#10b981{% elif lscore >= 40 %}#f59e0b{% else %}#ef4444{% endif %}; background: conic-gradient({% if lscore >= 70 %}#10b981{% elif lscore >= 40 %}#f59e0b{% else %}#ef4444{% endif %} calc({{ lscore }} * 3.6deg), #e2e8f0 0deg); position: relative;">
|
|
<span style="position: absolute; width: 64px; height: 64px; border-radius: 50%; background: var(--surface);"></span>
|
|
<span style="position: relative; z-index: 1;">{{ lscore }}</span>
|
|
</div>
|
|
<div>
|
|
<div style="font-size: var(--font-size-lg); font-weight: 600; color: {% if lscore >= 70 %}#10b981{% elif lscore >= 40 %}#f59e0b{% else %}#ef4444{% endif %};">
|
|
{% if lscore >= 70 %}Dobry Local SEO{% elif lscore >= 40 %}Przecietny Local SEO{% else %}Slaby Local SEO{% endif %}
|
|
</div>
|
|
<p style="font-size: var(--font-size-sm); color: var(--text-secondary); margin: 0;">Ocena optymalizacji pod lokalne wyszukiwanie</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Local SEO Checklist -->
|
|
<div style="display: grid; grid-template-columns: repeat(auto-fill, minmax(280px, 1fr)); gap: var(--spacing-sm);">
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_local_business_schema else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_local_business_schema else '#ef4444' }};">{{ '✓' if seo_data.has_local_business_schema else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Schema.org LocalBusiness</span>
|
|
</div>
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_google_maps_embed else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_google_maps_embed else '#ef4444' }};">{{ '✓' if seo_data.has_google_maps_embed else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Mapa Google na stronie</span>
|
|
</div>
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_local_keywords else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_local_keywords else '#ef4444' }};">{{ '✓' if seo_data.has_local_keywords else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Lokalne slowa kluczowe</span>
|
|
</div>
|
|
{% if seo_data.nap_on_website %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: #dcfce7;">
|
|
<span style="color: #10b981;">✓</span>
|
|
<span style="font-size: var(--font-size-sm);">NAP na stronie (Nazwa, Adres, Telefon)</span>
|
|
</div>
|
|
{% else %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: #fee2e2;">
|
|
<span style="color: #ef4444;">✗</span>
|
|
<span style="font-size: var(--font-size-sm);">NAP na stronie (Nazwa, Adres, Telefon)</span>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
|
|
{% if seo_data.local_keywords_found %}
|
|
<div style="margin-top: var(--spacing-md);">
|
|
<span style="font-size: var(--font-size-sm); font-weight: 600; color: var(--text-primary);">Znalezione slowa kluczowe:</span>
|
|
<div style="margin-top: var(--spacing-xs); display: flex; flex-wrap: wrap; gap: var(--spacing-xs);">
|
|
{% for kw in seo_data.local_keywords_found[:10] %}
|
|
<span style="padding: 2px 8px; background: #dbeafe; color: #1e40af; border-radius: var(--radius-sm); font-size: var(--font-size-xs);">{{ kw }}</span>
|
|
{% endfor %}
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.content_freshness_score is not none %}
|
|
<!-- Content Freshness -->
|
|
<div class="metrics-grid" style="margin-bottom: var(--spacing-xl);">
|
|
{% set fresh = seo_data.content_freshness_score %}
|
|
{% set fresh_class = 'good' if fresh >= 70 else ('medium' if fresh >= 40 else 'poor') %}
|
|
<div class="metric-card {{ fresh_class }}">
|
|
<div class="metric-icon {{ fresh_class }}">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">Swiezosc tresci</div>
|
|
<div class="metric-value {{ fresh_class }}">{{ fresh }}</div>
|
|
</div>
|
|
{% if seo_data.last_modified_date %}
|
|
<div class="metric-card medium">
|
|
<div class="metric-icon medium">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"/>
|
|
</svg>
|
|
</div>
|
|
<div class="metric-name">Ostatnia modyfikacja</div>
|
|
<div class="metric-value medium" style="font-size: var(--font-size-lg);">{{ seo_data.last_modified_date.strftime('%d.%m.%Y') }}</div>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.citations and seo_data.citations|length > 0 %}
|
|
<!-- Local Citations Section -->
|
|
<h2 class="section-title">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 21V5a2 2 0 00-2-2H7a2 2 0 00-2 2v16m14 0h2m-2 0h-5m-9 0H3m2 0h5M9 7h1m-1 4h1m4-4h1m-1 4h1m-5 10v-5a1 1 0 011-1h2a1 1 0 011 1v5m-4 0h4"/>
|
|
</svg>
|
|
Cytacje lokalne ({{ seo_data.citations_count or seo_data.citations|length }})
|
|
</h2>
|
|
|
|
<div style="background: var(--surface); border-radius: var(--radius-lg); box-shadow: var(--shadow); overflow: hidden; margin-bottom: var(--spacing-xl);">
|
|
{% for citation in seo_data.citations %}
|
|
<div style="display: flex; align-items: center; justify-content: space-between; padding: var(--spacing-sm) var(--spacing-md); border-bottom: 1px solid var(--border); {% if loop.last %}border-bottom: none;{% endif %}">
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm);">
|
|
<div style="width: 8px; height: 8px; border-radius: 50%; background: {{ '#10b981' if citation.status == 'found' else ('#f59e0b' if citation.status == 'incorrect' else '#ef4444') }};"></div>
|
|
<span style="font-size: var(--font-size-sm); font-weight: 500;">{{ citation.directory_name }}</span>
|
|
</div>
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm);">
|
|
{% if citation.status == 'found' %}
|
|
<span style="font-size: var(--font-size-xs); color: #10b981; padding: 2px 6px; background: #dcfce7; border-radius: var(--radius-sm);">Znaleziono</span>
|
|
{% if citation.listing_url %}
|
|
<a href="{{ citation.listing_url }}" target="_blank" rel="noopener" style="font-size: var(--font-size-xs); color: var(--primary);">Link</a>
|
|
{% endif %}
|
|
{% elif citation.status == 'incorrect' %}
|
|
<span style="font-size: var(--font-size-xs); color: #f59e0b; padding: 2px 6px; background: #fef3c7; border-radius: var(--radius-sm);">Bledne dane</span>
|
|
{% else %}
|
|
<span style="font-size: var(--font-size-xs); color: #ef4444; padding: 2px 6px; background: #fee2e2; border-radius: var(--radius-sm);">Nie znaleziono</span>
|
|
{% endif %}
|
|
</div>
|
|
</div>
|
|
{% endfor %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.has_ssl is not none or seo_data.has_google_analytics is not none or seo_data.h1_count is not none or seo_data.total_images is not none %}
|
|
<!-- Technical SEO Checklist -->
|
|
<h2 class="section-title">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"/>
|
|
</svg>
|
|
Technical SEO
|
|
</h2>
|
|
|
|
<div style="background: var(--surface); padding: var(--spacing-lg); border-radius: var(--radius-lg); box-shadow: var(--shadow); margin-bottom: var(--spacing-xl);">
|
|
<div style="display: grid; grid-template-columns: repeat(auto-fill, minmax(280px, 1fr)); gap: var(--spacing-sm);">
|
|
{% if seo_data.has_ssl is not none %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_ssl else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_ssl else '#ef4444' }};">{{ '✓' if seo_data.has_ssl else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Certyfikat SSL{% if seo_data.has_ssl and seo_data.ssl_expires_at %} (wazny do {{ seo_data.ssl_expires_at.strftime('%d.%m.%Y') }}){% endif %}</span>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.has_google_analytics is not none %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_google_analytics else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_google_analytics else '#ef4444' }};">{{ '✓' if seo_data.has_google_analytics else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Google Analytics</span>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.has_google_tag_manager is not none %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_google_tag_manager else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_google_tag_manager else '#ef4444' }};">{{ '✓' if seo_data.has_google_tag_manager else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Google Tag Manager</span>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.has_og_tags is not none %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_og_tags else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_og_tags else '#ef4444' }};">{{ '✓' if seo_data.has_og_tags else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Open Graph Tags</span>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.has_twitter_cards is not none %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if seo_data.has_twitter_cards else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if seo_data.has_twitter_cards else '#ef4444' }};">{{ '✓' if seo_data.has_twitter_cards else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Twitter Cards</span>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.h1_count is not none %}
|
|
{% set h1_ok = seo_data.h1_count == 1 %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if h1_ok else '#fef3c7' }};">
|
|
<span style="color: {{ '#10b981' if h1_ok else '#f59e0b' }};">{{ '✓' if h1_ok else '⚠' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">H1: {{ seo_data.h1_count }}{% if not h1_ok %} (powinien byc 1){% endif %}{% if seo_data.h2_count is not none %}, H2: {{ seo_data.h2_count }}{% endif %}{% if seo_data.h3_count is not none %}, H3: {{ seo_data.h3_count }}{% endif %}</span>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.total_images is not none %}
|
|
{% set alt_ok = (seo_data.images_without_alt or 0) == 0 %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if alt_ok else '#fef3c7' }};">
|
|
<span style="color: {{ '#10b981' if alt_ok else '#f59e0b' }};">{{ '✓' if alt_ok else '⚠' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Obrazy: {{ seo_data.total_images }}{% if not alt_ok %} ({{ seo_data.images_without_alt }} bez alt){% else %} (wszystkie z alt){% endif %}</span>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data.internal_links_count is not none or seo_data.external_links_count is not none %}
|
|
{% set broken = seo_data.broken_links_count or 0 %}
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm); padding: var(--spacing-sm); border-radius: var(--radius); background: {{ '#dcfce7' if broken == 0 else '#fee2e2' }};">
|
|
<span style="color: {{ '#10b981' if broken == 0 else '#ef4444' }};">{{ '✓' if broken == 0 else '✗' }}</span>
|
|
<span style="font-size: var(--font-size-sm);">Linki: {{ seo_data.internal_links_count or 0 }} wew., {{ seo_data.external_links_count or 0 }} zew.{% if broken > 0 %}, {{ broken }} uszkodzonych{% endif %}</span>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
|
|
{% if seo_data.h1_text %}
|
|
<div style="margin-top: var(--spacing-md); padding: var(--spacing-sm); background: var(--bg-tertiary); border-radius: var(--radius); font-size: var(--font-size-sm);">
|
|
<span style="font-weight: 600; color: var(--text-primary);">Tytul H1:</span>
|
|
<span style="color: var(--text-secondary);">{{ seo_data.h1_text }}</span>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% else %}
|
|
<!-- No Audit State -->
|
|
<div class="no-audit-state">
|
|
<svg fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="1.5" d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"/>
|
|
</svg>
|
|
<h2>Brak danych audytu SEO</h2>
|
|
{% if company.website %}
|
|
<p>Nie przeprowadzono jeszcze audytu SEO dla strony tej firmy. Uruchom audyt, aby sprawdzic optymalizacje strony.</p>
|
|
{% if can_audit %}
|
|
<button class="btn btn-primary" onclick="runAudit()">
|
|
<svg width="20" height="20" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"/>
|
|
</svg>
|
|
Uruchom pierwszy audyt
|
|
</button>
|
|
{% endif %}
|
|
{% else %}
|
|
<p>Ta firma nie ma zdefiniowanej strony WWW. Dodaj adres strony w profilu firmy, aby moc przeprowadzic audyt SEO.</p>
|
|
<a href="{{ url_for('company_detail', company_id=company.id) }}" class="btn btn-outline">
|
|
Przejdz do profilu firmy
|
|
</a>
|
|
{% endif %}
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if seo_data %}
|
|
{% with audit_type='seo' %}
|
|
{% include 'partials/audit_ai_actions.html' %}
|
|
{% endwith %}
|
|
{% endif %}
|
|
|
|
<!-- Loading Overlay -->
|
|
<div class="loading-overlay" id="loadingOverlay">
|
|
<div class="loading-content">
|
|
<div class="loading-spinner"></div>
|
|
<h3>Analiza SEO w toku...</h3>
|
|
<p>Pobieranie danych z Google PageSpeed Insights. Moze to potrwac do 30 sekund.</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Info Modal -->
|
|
<div class="modal" id="infoModal">
|
|
<div class="modal-content">
|
|
<div class="modal-header">
|
|
<div class="modal-icon success" id="modalIcon">
|
|
<svg width="24" height="24" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/>
|
|
</svg>
|
|
</div>
|
|
<div class="modal-title" id="modalTitle">Informacja</div>
|
|
</div>
|
|
<div class="modal-body" id="modalBody">
|
|
Tresc informacji.
|
|
</div>
|
|
<div class="modal-footer">
|
|
<button class="btn btn-primary" onclick="closeInfoModal()">OK</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{% endblock %}
|
|
|
|
{% block extra_js %}
|
|
const csrfToken = '{{ csrf_token() }}';
|
|
const companySlug = '{{ company.slug }}';
|
|
|
|
function showLoading() {
|
|
document.getElementById('loadingOverlay').classList.add('active');
|
|
}
|
|
|
|
function hideLoading() {
|
|
document.getElementById('loadingOverlay').classList.remove('active');
|
|
}
|
|
|
|
function showInfoModal(title, body, isSuccess) {
|
|
document.getElementById('modalTitle').textContent = title;
|
|
document.getElementById('modalBody').textContent = body;
|
|
const icon = document.getElementById('modalIcon');
|
|
icon.className = 'modal-icon ' + (isSuccess ? 'success' : 'info');
|
|
document.getElementById('infoModal').classList.add('active');
|
|
}
|
|
|
|
function closeInfoModal() {
|
|
document.getElementById('infoModal').classList.remove('active');
|
|
}
|
|
|
|
document.getElementById('infoModal')?.addEventListener('click', (e) => {
|
|
if (e.target.id === 'infoModal') closeInfoModal();
|
|
});
|
|
|
|
async function runAudit() {
|
|
const btn = document.getElementById('runAuditBtn');
|
|
if (btn) {
|
|
btn.disabled = true;
|
|
}
|
|
showLoading();
|
|
|
|
try {
|
|
const response = await fetch('/api/seo/audit', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRFToken': csrfToken
|
|
},
|
|
body: JSON.stringify({ slug: companySlug })
|
|
});
|
|
|
|
const data = await response.json();
|
|
|
|
hideLoading();
|
|
|
|
if (response.ok && data.success) {
|
|
showInfoModal('Audyt zakonczony', 'Audyt SEO zostal zakonczony pomyslnie. Strona zostanie odswiezona.', true);
|
|
setTimeout(() => location.reload(), 1500);
|
|
} else {
|
|
showInfoModal('Blad', data.error || 'Wystapil nieznany blad podczas audytu.', false);
|
|
if (btn) btn.disabled = false;
|
|
}
|
|
} catch (error) {
|
|
hideLoading();
|
|
showInfoModal('Blad polaczenia', 'Nie udalo sie polaczyc z serwerem: ' + error.message, false);
|
|
if (btn) btn.disabled = false;
|
|
}
|
|
}
|
|
|
|
/* ============================================================
|
|
AI AUDIT ACTIONS
|
|
============================================================ */
|
|
const companyId = {{ company.id }};
|
|
const auditType = 'seo';
|
|
|
|
function simpleMarkdown(text) {
|
|
return text
|
|
.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>')
|
|
.replace(/\*\*(.+?)\*\*/g, '<strong>$1</strong>')
|
|
.replace(/\*(.+?)\*/g, '<em>$1</em>')
|
|
.replace(/^### (.+)$/gm, '<h4>$1</h4>')
|
|
.replace(/^## (.+)$/gm, '<h3>$1</h3>')
|
|
.replace(/^- (.+)$/gm, '<li>$1</li>')
|
|
.replace(/(<li>.*<\/li>)/gs, '<ul>$1</ul>')
|
|
.replace(/\n/g, '<br>');
|
|
}
|
|
|
|
async function runAIAnalysis(force) {
|
|
const prompt = document.getElementById('aiAnalyzePrompt');
|
|
const loading = document.getElementById('aiLoading');
|
|
const results = document.getElementById('aiResults');
|
|
const btn = document.getElementById('aiAnalyzeBtn');
|
|
|
|
if (btn) btn.disabled = true;
|
|
if (prompt) prompt.style.display = 'none';
|
|
if (results) results.style.display = 'none';
|
|
if (loading) loading.style.display = 'block';
|
|
|
|
// Start timer
|
|
let seconds = 0;
|
|
const timerEl = document.getElementById('aiTimer');
|
|
const timerInterval = setInterval(() => {
|
|
seconds++;
|
|
if (timerEl) timerEl.textContent = seconds + 's';
|
|
}, 1000);
|
|
|
|
try {
|
|
const response = await fetch('/api/audit/analyze', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRFToken': csrfToken
|
|
},
|
|
body: JSON.stringify({
|
|
company_id: companyId,
|
|
audit_type: auditType,
|
|
force: !!force
|
|
})
|
|
});
|
|
|
|
const data = await response.json();
|
|
clearInterval(timerInterval);
|
|
if (loading) loading.style.display = 'none';
|
|
|
|
if (data.success) {
|
|
renderAIResults(data);
|
|
} else {
|
|
if (prompt) prompt.style.display = 'none';
|
|
if (btn) btn.disabled = false;
|
|
const results = document.getElementById('aiResults');
|
|
results.innerHTML = `
|
|
<div style="background: #fef2f2; border: 1px solid #fecaca; padding: var(--spacing-lg); border-radius: var(--radius-lg); text-align: center;">
|
|
<p style="color: #dc2626; font-weight: 600; margin-bottom: var(--spacing-sm);">Blad analizy AI</p>
|
|
<p style="color: #991b1b; font-size: var(--font-size-sm);">${escapeHtml(data.error || 'Nieznany blad')}</p>
|
|
<button class="btn btn-outline btn-sm" onclick="runAIAnalysis()" style="margin-top: var(--spacing-md);">Sprobuj ponownie</button>
|
|
</div>`;
|
|
results.style.display = 'block';
|
|
}
|
|
} catch (error) {
|
|
clearInterval(timerInterval);
|
|
if (loading) loading.style.display = 'none';
|
|
if (prompt) prompt.style.display = 'none';
|
|
if (btn) btn.disabled = false;
|
|
const results = document.getElementById('aiResults');
|
|
results.innerHTML = `
|
|
<div style="background: #fef2f2; border: 1px solid #fecaca; padding: var(--spacing-lg); border-radius: var(--radius-lg); text-align: center;">
|
|
<p style="color: #dc2626; font-weight: 600; margin-bottom: var(--spacing-sm);">Blad polaczenia</p>
|
|
<p style="color: #991b1b; font-size: var(--font-size-sm);">${escapeHtml(error.message)}</p>
|
|
<button class="btn btn-outline btn-sm" onclick="runAIAnalysis()" style="margin-top: var(--spacing-md);">Sprobuj ponownie</button>
|
|
</div>`;
|
|
results.style.display = 'block';
|
|
}
|
|
}
|
|
|
|
function renderAIResults(data) {
|
|
const results = document.getElementById('aiResults');
|
|
const summaryEl = document.getElementById('aiSummaryText');
|
|
const cacheInfo = document.getElementById('aiCacheInfo');
|
|
const actionsList = document.getElementById('aiActionsList');
|
|
|
|
summaryEl.textContent = data.summary || '';
|
|
if (data.cached && data.generated_at) {
|
|
const d = new Date(data.generated_at);
|
|
document.getElementById('aiCacheDate').textContent =
|
|
d.toLocaleDateString('pl-PL') + ' ' + d.toLocaleTimeString('pl-PL', {hour:'2-digit', minute:'2-digit'});
|
|
cacheInfo.style.display = 'block';
|
|
} else {
|
|
cacheInfo.style.display = 'none';
|
|
}
|
|
|
|
actionsList.innerHTML = '';
|
|
const actions = data.actions || [];
|
|
const priorityLabels = {critical: 'KRYTYCZNE', high: 'WYSOKI', medium: 'SREDNI', low: 'NISKI'};
|
|
|
|
actions.forEach((action, idx) => {
|
|
const card = document.createElement('div');
|
|
card.className = 'ai-action-card priority-' + (action.priority || 'medium');
|
|
card.id = 'ai-action-' + idx;
|
|
|
|
const impact = action.impact_score || 5;
|
|
const effort = action.effort_score || 5;
|
|
|
|
card.innerHTML = `
|
|
<div style="display: flex; justify-content: space-between; align-items: flex-start; margin-bottom: var(--spacing-sm); flex-wrap: wrap; gap: var(--spacing-xs);">
|
|
<div style="display: flex; align-items: center; gap: var(--spacing-sm);">
|
|
<span class="ai-priority-badge ${action.priority || 'medium'}">${priorityLabels[action.priority] || 'SREDNI'}</span>
|
|
<span class="ai-action-title" style="font-weight: 600; color: var(--text-primary);">${escapeHtml(action.title || '')}</span>
|
|
</div>
|
|
</div>
|
|
<p style="color: var(--text-secondary); font-size: var(--font-size-sm); margin-bottom: var(--spacing-sm);">${escapeHtml(action.description || '')}</p>
|
|
<div style="display: grid; grid-template-columns: 1fr 1fr; gap: var(--spacing-md); margin-bottom: var(--spacing-sm);">
|
|
<div>
|
|
<div style="font-size: var(--font-size-xs); color: var(--text-tertiary); margin-bottom: 2px;">Wplyw: ${impact}/10</div>
|
|
<div class="ai-score-bar"><div class="ai-score-bar-fill impact" style="width: ${impact * 10}%;"></div></div>
|
|
</div>
|
|
<div>
|
|
<div style="font-size: var(--font-size-xs); color: var(--text-tertiary); margin-bottom: 2px;">Wysilek: ${effort}/10</div>
|
|
<div class="ai-score-bar"><div class="ai-score-bar-fill effort" style="width: ${effort * 10}%;"></div></div>
|
|
</div>
|
|
</div>
|
|
<div class="ai-action-buttons">
|
|
<button class="btn btn-outline btn-sm" onclick="generateContent('${action.action_type}', ${idx})">
|
|
<svg width="14" height="14" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9.663 17h4.673M12 3v1m6.364 1.636l-.707.707M21 12h-1M4 12H3m3.343-5.657l-.707-.707m2.828 9.9a5 5 0 117.072 0l-.548.547A3.374 3.374 0 0014 18.469V19a2 2 0 11-4 0v-.531c0-.895-.356-1.754-.988-2.386l-.548-.547z"/></svg>
|
|
Wygeneruj tresc
|
|
</button>
|
|
<button class="btn btn-outline btn-sm" onclick="markAction(${idx}, 'implemented')" style="color: #10b981; border-color: #10b981;">
|
|
<svg width="14" height="14" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/></svg>
|
|
Zrobione
|
|
</button>
|
|
<button class="btn btn-outline btn-sm" onclick="markAction(${idx}, 'dismissed')" style="color: var(--text-tertiary); border-color: var(--border);">
|
|
Odrzuc
|
|
</button>
|
|
</div>
|
|
<div id="ai-content-${idx}" style="display: none;"></div>
|
|
`;
|
|
actionsList.appendChild(card);
|
|
});
|
|
|
|
// Render comparison with previous analysis if available
|
|
if (typeof renderAIComparison === 'function') renderAIComparison(data);
|
|
|
|
results.style.display = 'block';
|
|
document.getElementById('aiActionsSection').scrollIntoView({behavior: 'smooth', block: 'start'});
|
|
|
|
// Store actions data for content generation
|
|
window._aiActions = actions;
|
|
}
|
|
|
|
async function generateContent(actionType, idx) {
|
|
const container = document.getElementById('ai-content-' + idx);
|
|
if (!container) return;
|
|
|
|
// If already has content, toggle visibility
|
|
if (container.dataset.loaded === 'true') {
|
|
container.style.display = container.style.display === 'none' ? 'block' : 'none';
|
|
return;
|
|
}
|
|
|
|
container.innerHTML = '<div style="padding: var(--spacing-md); color: var(--text-secondary); font-size: var(--font-size-sm);">Generowanie tresci...</div>';
|
|
container.style.display = 'block';
|
|
|
|
try {
|
|
const response = await fetch('/api/audit/generate-content', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRFToken': csrfToken
|
|
},
|
|
body: JSON.stringify({
|
|
company_id: companyId,
|
|
action_type: actionType,
|
|
context: {}
|
|
})
|
|
});
|
|
|
|
const data = await response.json();
|
|
|
|
if (data.success && data.content) {
|
|
const isCode = data.content.includes('{') && (data.content.includes('<script') || data.content.includes('"@type"') || data.content.trim().startsWith('{') || data.content.trim().startsWith('<'));
|
|
if (isCode) {
|
|
container.innerHTML = `
|
|
<div class="ai-content-output">
|
|
<button class="ai-copy-btn" onclick="copyContent(this)">Kopiuj</button>
|
|
<code>${escapeHtml(data.content)}</code>
|
|
</div>
|
|
`;
|
|
} else {
|
|
container.innerHTML = `
|
|
<div style="background: var(--surface); border: 1px solid var(--border); padding: var(--spacing-md); border-radius: var(--radius); margin-top: var(--spacing-md); position: relative; line-height: 1.6; font-size: var(--font-size-sm); color: var(--text-primary);">
|
|
<button class="ai-copy-btn" style="position: absolute; top: 8px; right: 8px; background: var(--bg-tertiary); border: 1px solid var(--border); color: var(--text-secondary); padding: 4px 10px; border-radius: var(--radius-sm); font-size: var(--font-size-xs); cursor: pointer;" onclick="copyContent(this)">Kopiuj</button>
|
|
<div class="ai-markdown-content">${simpleMarkdown(data.content)}</div>
|
|
</div>
|
|
`;
|
|
}
|
|
container.dataset.loaded = 'true';
|
|
container.scrollIntoView({behavior: 'smooth', block: 'nearest'});
|
|
} else {
|
|
container.innerHTML = `
|
|
<div style="padding: var(--spacing-sm); background: #fef2f2; border-radius: var(--radius-sm); margin-top: var(--spacing-sm);">
|
|
<span style="color: #dc2626;">${escapeHtml(data.error || 'Blad generowania')}</span>
|
|
<button class="btn btn-outline btn-sm" onclick="generateContent('${actionType}', ${idx})" style="margin-left: var(--spacing-sm);">Ponow</button>
|
|
</div>`;
|
|
}
|
|
} catch (error) {
|
|
container.innerHTML = `
|
|
<div style="padding: var(--spacing-sm); background: #fef2f2; border-radius: var(--radius-sm); margin-top: var(--spacing-sm);">
|
|
<span style="color: #dc2626;">${escapeHtml(error.message)}</span>
|
|
<button class="btn btn-outline btn-sm" onclick="generateContent('${actionType}', ${idx})" style="margin-left: var(--spacing-sm);">Ponow</button>
|
|
</div>`;
|
|
}
|
|
}
|
|
|
|
function copyContent(btn) {
|
|
const code = btn.parentElement.querySelector('code') || btn.parentElement.querySelector('.ai-markdown-content');
|
|
if (!code) return;
|
|
|
|
navigator.clipboard.writeText(code.textContent).then(() => {
|
|
const orig = btn.textContent;
|
|
btn.textContent = 'Skopiowano!';
|
|
setTimeout(() => { btn.textContent = orig; }, 2000);
|
|
});
|
|
}
|
|
|
|
function markAction(idx, status) {
|
|
const card = document.getElementById('ai-action-' + idx);
|
|
if (!card) return;
|
|
|
|
if (status === 'implemented') {
|
|
card.classList.add('implemented');
|
|
} else if (status === 'dismissed') {
|
|
card.classList.add('dismissed');
|
|
}
|
|
|
|
// Fire and forget status update to backend
|
|
const actions = window._aiActions || [];
|
|
if (actions[idx] && actions[idx].id) {
|
|
fetch('/api/audit/actions/' + actions[idx].id + '/status', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRFToken': csrfToken
|
|
},
|
|
body: JSON.stringify({ status: status })
|
|
}).catch(() => {});
|
|
}
|
|
}
|
|
|
|
function escapeHtml(text) {
|
|
const div = document.createElement('div');
|
|
div.textContent = text;
|
|
return div.innerHTML;
|
|
}
|
|
{% endblock %}
|