Files
orion/app/modules/prospecting/templates/prospecting/admin/leads.html
Samir Boulahtit 2287f4597d
All checks were successful
CI / ruff (push) Successful in 10s
CI / pytest (push) Successful in 48m48s
CI / validate (push) Successful in 24s
CI / dependency-scanning (push) Successful in 29s
CI / docs (push) Successful in 38s
CI / deploy (push) Successful in 51s
feat(hosting,prospecting): add hosting unit tests and fix template bugs
- Add 55 unit tests for hosting module (hosted site service, client
  service service, stats service) with full fixture setup
- Fix table_empty_state macro: add x_message param for dynamic Alpine.js
  expressions rendered via x-text instead of server-side Jinja
- Fix hosting templates (sites, clients) using message= with Alpine
  expressions that rendered as literal text
- Fix prospecting templates (leads, scan-jobs, prospects) using
  nonexistent subtitle= param, migrated to x_message=
- Align hosting and prospecting admin templates with shared design system

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-07 06:18:26 +01:00

144 lines
8.2 KiB
HTML

{% extends "admin/base.html" %}
{% from 'shared/macros/headers.html' import page_header %}
{% from 'shared/macros/alerts.html' import loading_state, error_state %}
{% from 'shared/macros/tables.html' import table_wrapper, table_header, table_empty_state %}
{% from 'shared/macros/pagination.html' import pagination %}
{% block title %}Leads{% endblock %}
{% block alpine_data %}leadsList(){% endblock %}
{% block content %}
{{ page_header('Leads', action_label='Export CSV', action_onclick='exportCSV()', action_icon='download') }}
{{ loading_state('Loading leads...') }}
{{ error_state('Error loading leads') }}
<!-- Filters -->
<div x-show="!loading && !error" class="mb-6 p-4 bg-white rounded-lg shadow-xs dark:bg-gray-800">
<div class="flex flex-col md:flex-row md:items-center md:justify-between gap-4">
<!-- Min Score -->
<div class="flex items-center gap-3">
<label class="text-sm font-medium text-gray-700 dark:text-gray-400 whitespace-nowrap">Min Score</label>
<input type="number" x-model.number="minScore" @change="pagination.page = 1; loadLeads()" min="0" max="100" {# noqa: FE008 - score filter, not a quantity stepper #}
class="w-24 px-3 py-2 text-sm border border-gray-300 dark:border-gray-600 rounded-lg focus:border-purple-400 focus:outline-none focus:shadow-outline-purple dark:focus:shadow-outline-gray dark:bg-gray-700 dark:text-gray-300">
</div>
<!-- Filters -->
<div class="flex flex-wrap gap-3">
<select x-model="filterTier" @change="pagination.page = 1; loadLeads()"
class="px-4 py-2 text-sm text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-700 border border-gray-300 dark:border-gray-600 rounded-lg focus:border-purple-400 focus:outline-none">
<option value="">All Tiers</option>
<option value="top_priority">Top Priority</option>
<option value="quick_win">Quick Win</option>
<option value="strategic">Strategic</option>
</select>
<select x-model="filterChannel" @change="pagination.page = 1; loadLeads()"
class="px-4 py-2 text-sm text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-700 border border-gray-300 dark:border-gray-600 rounded-lg focus:border-purple-400 focus:outline-none">
<option value="">All Channels</option>
<option value="digital">Digital</option>
<option value="offline">Offline</option>
</select>
<select x-model="filterIssue" @change="pagination.page = 1; loadLeads()"
class="px-4 py-2 text-sm text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-700 border border-gray-300 dark:border-gray-600 rounded-lg focus:border-purple-400 focus:outline-none">
<option value="">All Issues</option>
<option value="no_ssl">No SSL</option>
<option value="very_slow">Very Slow</option>
<option value="not_mobile_friendly">Not Mobile Friendly</option>
<option value="outdated_cms">Outdated CMS</option>
<option value="no_website">No Website</option>
<option value="uses_gmail">Uses Gmail</option>
</select>
<select x-model="filterHasEmail" @change="pagination.page = 1; loadLeads()"
class="px-4 py-2 text-sm text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-700 border border-gray-300 dark:border-gray-600 rounded-lg focus:border-purple-400 focus:outline-none">
<option value="">Has Email</option>
<option value="true">Yes</option>
<option value="false">No</option>
</select>
</div>
</div>
</div>
<!-- Leads Table -->
<div x-show="!loading && !error">
{% call table_wrapper() %}
{{ table_header(['Business / Domain', 'Score', 'Tier', 'Issues', 'Contact', 'Actions']) }}
<tbody class="bg-white divide-y dark:divide-gray-700 dark:bg-gray-800">
{{ table_empty_state(6, title='No leads found', x_message="filterTier || filterChannel || filterIssue || filterHasEmail || minScore > 0 ? 'Try adjusting your filters' : 'No scored leads yet'", show_condition='leads.length === 0', icon='user-group') }}
<template x-for="lead in leads" :key="lead.id">
<tr class="text-gray-700 dark:text-gray-400 hover:bg-gray-50 dark:hover:bg-gray-700 transition-colors">
<!-- Business / Domain with Avatar -->
<td class="px-4 py-3">
<div class="flex items-center text-sm">
<div class="relative hidden w-8 h-8 mr-3 rounded-full md:block">
<div class="absolute inset-0 rounded-full bg-purple-100 dark:bg-purple-600 flex items-center justify-center">
<span class="text-xs font-semibold text-purple-600 dark:text-purple-100"
x-text="(lead.business_name || lead.domain_name)?.charAt(0).toUpperCase() || '?'"></span>
</div>
</div>
<div>
<p class="font-semibold" x-text="lead.business_name || lead.domain_name"></p>
<p class="text-xs text-gray-600 dark:text-gray-400" x-show="lead.domain_name && lead.business_name" x-text="lead.domain_name"></p>
</div>
</div>
</td>
<!-- Score -->
<td class="px-4 py-3">
<span class="text-lg font-bold" :class="scoreColor(lead.score)"
x-text="lead.score ?? '—'"></span>
</td>
<!-- Tier Badge -->
<td class="px-4 py-3">
<span class="px-2.5 py-0.5 text-xs font-medium rounded-full"
:class="tierBadgeClass(lead.lead_tier)"
x-text="lead.lead_tier?.replace('_', ' ')"></span>
</td>
<!-- Issues -->
<td class="px-4 py-3">
<div class="flex flex-wrap gap-1">
<template x-for="flag in (lead.reason_flags || []).slice(0, 3)" :key="flag">
<span class="px-1.5 py-0.5 text-xs bg-red-100 text-red-700 rounded dark:bg-red-900 dark:text-red-300"
x-text="flag.replace('_', ' ')"></span>
</template>
</div>
</td>
<!-- Contact -->
<td class="px-4 py-3 text-sm">
<span x-text="lead.primary_email || lead.primary_phone || '—'" class="text-xs"></span>
</td>
<!-- Actions -->
<td class="px-4 py-3">
<div class="flex items-center space-x-2 text-sm">
<a :href="'/admin/prospecting/prospects/' + lead.id"
class="flex items-center justify-center p-2 text-purple-600 rounded-lg hover:bg-purple-50 dark:text-purple-400 dark:hover:bg-gray-700 focus:outline-none transition-colors"
title="View details">
<span x-html="$icon('eye', 'w-5 h-5')"></span>
</a>
<button @click="sendCampaign(lead)"
class="flex items-center justify-center p-2 text-green-600 rounded-lg hover:bg-green-50 dark:text-green-400 dark:hover:bg-gray-700 focus:outline-none transition-colors"
title="Send campaign">
<span x-html="$icon('mail', 'w-5 h-5')"></span>
</button>
</div>
</td>
</tr>
</template>
</tbody>
{% endcall %}
{{ pagination() }}
</div>
{% endblock %}
{% block extra_scripts %}
<script defer src="{{ url_for('prospecting_static', path='admin/js/leads.js') }}"></script>
{% endblock %}