nordabiz/templates/partials/audit_ai_actions.html
Maciej Pienczyn 7197af3933
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
feat(audit): Add previous vs current AI analysis comparison
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>
2026-02-07 18:05:17 +01:00

346 lines
14 KiB
HTML

{#
Partial: AI Audit Actions Section
Variables required:
company - Company object (with .id, .slug)
audit_type - 'seo', 'gbp', or 'social'
Include this at the bottom of audit templates:
{% include 'partials/audit_ai_actions.html' %}
#}
<!-- AI Analysis & Actions Section -->
<div id="aiActionsSection" style="margin-top: var(--spacing-2xl);">
<h2 class="section-title" style="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);">
<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.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>
Analiza AI i Rekomendacje
</h2>
<!-- Generate Analysis Button -->
<div id="aiAnalyzePrompt" style="background: var(--surface); padding: var(--spacing-xl); border-radius: var(--radius-lg); box-shadow: var(--shadow); text-align: center;">
<p style="color: var(--text-secondary); margin-bottom: var(--spacing-md);">
AI przeanalizuje wyniki audytu i zaproponuje priorytetowane akcje do podjecia.
</p>
<button class="btn btn-primary" onclick="runAIAnalysis()" id="aiAnalyzeBtn">
<svg width="18" height="18" 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 analize AI
</button>
</div>
<!-- AI Loading Spinner -->
<div id="aiLoading" style="display: none; background: var(--surface); padding: var(--spacing-xl); border-radius: var(--radius-lg); box-shadow: var(--shadow); text-align: center;">
<div style="width: 40px; height: 40px; border: 3px solid var(--border); border-top-color: var(--primary); border-radius: 50%; animation: spin 1s linear infinite; margin: 0 auto var(--spacing-md);"></div>
<p style="color: var(--text-secondary);">
Analiza AI w toku... (moze potrwac 15-30 sekund)
<span id="aiTimer" style="font-weight: 600;"></span>
</p>
</div>
<!-- AI Results Container -->
<div id="aiResults" style="display: none;">
<!-- Summary -->
<div id="aiSummary" style="background: linear-gradient(135deg, #eff6ff 0%, #f0fdf4 100%); padding: var(--spacing-lg); border-radius: var(--radius-lg); margin-bottom: var(--spacing-lg); border: 1px solid #bfdbfe;">
<div style="display: flex; align-items: flex-start; gap: var(--spacing-sm);">
<svg width="20" height="20" fill="none" stroke="#2563eb" viewBox="0 0 24 24" style="flex-shrink: 0; margin-top: 2px;">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
<p id="aiSummaryText" style="color: var(--text-primary); line-height: 1.6; margin: 0;"></p>
</div>
<div id="aiCacheInfo" style="display: none; margin-top: var(--spacing-sm); font-size: var(--font-size-xs); color: var(--text-tertiary);">
Analiza z <span id="aiCacheDate"></span> &mdash; <a href="#" onclick="runAIAnalysis(true); return false;" style="color: var(--primary);">Wygeneruj ponownie</a>
</div>
</div>
<!-- Comparison with previous analysis -->
<div id="aiComparison" style="display: none; margin-bottom: var(--spacing-lg);">
<div style="display: flex; align-items: center; gap: var(--spacing-sm); margin-bottom: var(--spacing-sm); cursor: pointer;" onclick="toggleComparison()">
<svg id="aiComparisonArrow" width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24" style="transition: transform 0.2s;">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"/>
</svg>
<span style="font-size: var(--font-size-sm); font-weight: 600; color: var(--text-secondary);">Porownanie z poprzednia analiza</span>
<span id="aiComparisonDate" style="font-size: var(--font-size-xs); color: var(--text-tertiary);"></span>
</div>
<div id="aiComparisonBody" style="display: none;">
<table class="ai-comparison-table">
<thead>
<tr>
<th>Aspekt</th>
<th>Poprzednia</th>
<th>Obecna</th>
</tr>
</thead>
<tbody id="aiComparisonRows"></tbody>
</table>
<div id="aiComparisonDiff" style="margin-top: var(--spacing-sm);"></div>
</div>
</div>
<!-- Actions List -->
<div style="font-size: var(--font-size-lg); font-weight: 600; color: var(--text-primary); margin-bottom: var(--spacing-md);">
Priorytetowe akcje
</div>
<div id="aiActionsList"></div>
</div>
</div>
<style>
.ai-action-card {
background: var(--surface);
border-radius: var(--radius-lg);
box-shadow: var(--shadow-sm);
padding: var(--spacing-lg);
margin-bottom: var(--spacing-md);
border-left: 4px solid var(--border);
transition: box-shadow 0.2s;
}
.ai-action-card:hover {
box-shadow: var(--shadow);
}
.ai-action-card.priority-critical { border-left-color: #ef4444; }
.ai-action-card.priority-high { border-left-color: #f97316; }
.ai-action-card.priority-medium { border-left-color: #f59e0b; }
.ai-action-card.priority-low { border-left-color: #84cc16; }
.ai-priority-badge {
display: inline-flex;
align-items: center;
gap: 4px;
padding: 2px 8px;
border-radius: var(--radius-sm);
font-size: var(--font-size-xs);
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.5px;
}
.ai-priority-badge.critical { background: #fee2e2; color: #dc2626; }
.ai-priority-badge.high { background: #ffedd5; color: #ea580c; }
.ai-priority-badge.medium { background: #fef3c7; color: #d97706; }
.ai-priority-badge.low { background: #ecfccb; color: #65a30d; }
.ai-score-bar {
height: 6px;
border-radius: 3px;
background: #e2e8f0;
overflow: hidden;
}
.ai-score-bar-fill {
height: 100%;
border-radius: 3px;
transition: width 0.3s;
}
.ai-score-bar-fill.impact { background: #3b82f6; }
.ai-score-bar-fill.effort { background: #f59e0b; }
.ai-content-output {
background: #1e293b;
color: #e2e8f0;
padding: var(--spacing-md);
border-radius: var(--radius);
margin-top: var(--spacing-md);
font-family: 'Menlo', 'Monaco', 'Consolas', monospace;
font-size: var(--font-size-sm);
white-space: pre-wrap;
word-break: break-all;
position: relative;
max-height: 400px;
overflow-y: auto;
}
.ai-copy-btn {
position: absolute;
top: 8px;
right: 8px;
background: rgba(255,255,255,0.1);
border: 1px solid rgba(255,255,255,0.2);
color: #e2e8f0;
padding: 4px 10px;
border-radius: var(--radius-sm);
font-size: var(--font-size-xs);
cursor: pointer;
transition: background 0.2s;
}
.ai-copy-btn:hover {
background: rgba(255,255,255,0.2);
}
.ai-action-buttons {
display: flex;
gap: var(--spacing-sm);
margin-top: var(--spacing-md);
flex-wrap: wrap;
}
.ai-action-card.implemented {
opacity: 0.6;
border-left-color: #10b981;
}
.ai-action-card.implemented .ai-action-title {
text-decoration: line-through;
}
.ai-action-card.dismissed {
display: none;
}
.ai-comparison-table {
width: 100%;
border-collapse: collapse;
font-size: var(--font-size-sm);
}
.ai-comparison-table th {
background: var(--bg-tertiary);
padding: 8px 12px;
text-align: left;
font-weight: 600;
color: var(--text-secondary);
border-bottom: 2px solid var(--border);
}
.ai-comparison-table td {
padding: 8px 12px;
border-bottom: 1px solid var(--border);
color: var(--text-primary);
vertical-align: top;
}
.ai-comparison-table tr:last-child td {
border-bottom: none;
}
.ai-diff-added {
display: inline-block;
background: #dcfce7;
color: #166534;
padding: 2px 8px;
border-radius: var(--radius-sm);
font-size: var(--font-size-xs);
margin: 2px;
}
.ai-diff-removed {
display: inline-block;
background: #fee2e2;
color: #991b1b;
padding: 2px 8px;
border-radius: var(--radius-sm);
font-size: var(--font-size-xs);
margin: 2px;
text-decoration: line-through;
}
</style>
<script>
document.addEventListener('DOMContentLoaded', function() {
if (typeof companyId === 'undefined' || typeof auditType === 'undefined') return;
fetch('/api/audit/analyze', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-CSRFToken': csrfToken
},
body: JSON.stringify({
company_id: companyId,
audit_type: auditType,
force: false
})
})
.then(function(r) { return r.json(); })
.then(function(data) {
if (data.success && data.cached) {
var prompt = document.getElementById('aiAnalyzePrompt');
if (prompt) prompt.style.display = 'none';
renderAIResults(data);
}
})
.catch(function() {});
});
function toggleComparison() {
var body = document.getElementById('aiComparisonBody');
var arrow = document.getElementById('aiComparisonArrow');
if (body.style.display === 'none') {
body.style.display = 'block';
arrow.style.transform = 'rotate(90deg)';
} else {
body.style.display = 'none';
arrow.style.transform = '';
}
}
function renderAIComparison(data) {
var section = document.getElementById('aiComparison');
if (!data.previous) {
section.style.display = 'none';
return;
}
var prev = data.previous;
var prevActions = prev.actions || [];
var currActions = data.actions || [];
var priorityOrder = ['critical', 'high', 'medium', 'low'];
function countPriority(actions, p) {
return actions.filter(function(a) { return a.priority === p; }).length;
}
var rows = document.getElementById('aiComparisonRows');
var html = '';
// Summary row
var prevSummaryShort = (prev.summary || '').substring(0, 120) + ((prev.summary || '').length > 120 ? '...' : '');
var currSummaryShort = (data.summary || '').substring(0, 120) + ((data.summary || '').length > 120 ? '...' : '');
html += '<tr><td><strong>Podsumowanie</strong></td><td style="color: var(--text-secondary);">' + escapeHtml(prevSummaryShort) + '</td><td>' + escapeHtml(currSummaryShort) + '</td></tr>';
// Count row
html += '<tr><td><strong>Liczba akcji</strong></td><td>' + prevActions.length + '</td><td>' + currActions.length + '</td></tr>';
// Priority breakdown
priorityOrder.forEach(function(p) {
var labels = {critical: 'Krytyczne', high: 'Wysokie', medium: 'Srednie', low: 'Niskie'};
var pc = countPriority(prevActions, p);
var cc = countPriority(currActions, p);
if (pc > 0 || cc > 0) {
var diff = cc - pc;
var diffStr = diff > 0 ? ' <span style="color:#16a34a;">(+' + diff + ')</span>' : diff < 0 ? ' <span style="color:#dc2626;">(' + diff + ')</span>' : '';
html += '<tr><td>' + labels[p] + '</td><td>' + pc + '</td><td>' + cc + diffStr + '</td></tr>';
}
});
// Date row
if (prev.generated_at) {
var pd = new Date(prev.generated_at);
html += '<tr><td><strong>Data analizy</strong></td><td>' + pd.toLocaleDateString('pl-PL') + ' ' + pd.toLocaleTimeString('pl-PL', {hour:'2-digit', minute:'2-digit'}) + '</td><td>teraz</td></tr>';
}
rows.innerHTML = html;
// Diff: new/removed actions
var prevTitles = prevActions.map(function(a) { return a.title; });
var currTitles = currActions.map(function(a) { return a.title; });
var added = currTitles.filter(function(t) { return prevTitles.indexOf(t) === -1; });
var removed = prevTitles.filter(function(t) { return currTitles.indexOf(t) === -1; });
var diffEl = document.getElementById('aiComparisonDiff');
var diffHtml = '';
if (added.length > 0) {
diffHtml += '<div style="margin-bottom: var(--spacing-xs);"><strong style="font-size: var(--font-size-xs); color: var(--text-secondary);">Nowe akcje:</strong> ';
added.forEach(function(t) { diffHtml += '<span class="ai-diff-added">' + escapeHtml(t) + '</span>'; });
diffHtml += '</div>';
}
if (removed.length > 0) {
diffHtml += '<div><strong style="font-size: var(--font-size-xs); color: var(--text-secondary);">Usuniete akcje:</strong> ';
removed.forEach(function(t) { diffHtml += '<span class="ai-diff-removed">' + escapeHtml(t) + '</span>'; });
diffHtml += '</div>';
}
diffEl.innerHTML = diffHtml;
// Show date in header
if (prev.generated_at) {
var pd2 = new Date(prev.generated_at);
document.getElementById('aiComparisonDate').textContent = '(z ' + pd2.toLocaleDateString('pl-PL') + ')';
}
section.style.display = 'block';
}
</script>