Signup: - /signup aliniat ca format la formularul din landing (campuri, etichete, placeholder-uri, select plan, checkbox GDPR, buton). Eticheta `name` = "Companie" (corecta: backendul salveaza nume de firma), uniform si in landing. - Consimtamant GDPR validat server-side (functional, nu doar client-side) + salvat cu marca temporala (accounts.consent_at). - Plan ales la signup salvat in accounts.requested_plan (intentie, NU drept): tier ramane sursa de adevar pentru gate-ul API; coloana pregateste integrarea platilor. - landing: valorile `plan` = coduri tier (free/standard/pro/premium), data-plan sincronizat pe butoanele de pret; checkbox consimtamant primeste name. Schema/DB: - accounts: coloane noi requested_plan + consent_at (cu migrare aditiva in db.py). Panou admin: - Coloane noi: Plan curent (plan EFECTIV acum + zile trial ramase) si Plan cerut. - Buton "Aplica" (POST /admin/set-tier): aloca plan real si INCHEIE trial-ul (efect imediat; altfel trial-ul Pro universal de 30z masca alegerea). - Control "Trial Pro N zile" (POST /admin/set-trial via accounts.set_trial): acorda/prelungeste trial fara a schimba tier-ul de baza. Teste: signup (consent obligatoriu, requested_plan persistat, tier ramane free), panou admin (set-tier incheie trial, free opreste Pro imediat, set-trial, validari + CSRF). Call-site-urile existente POST /signup actualizate cu consent. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
189 lines
9.3 KiB
HTML
189 lines
9.3 KiB
HTML
{% extends "base.html" %}
|
|
{% block title %}Conturi clienti — Gateway RAR AUTOPASS{% endblock %}
|
|
{% block content %}
|
|
|
|
{# Metadate verbe de ciclu de viata (eticheta, ruta, clasa). #}
|
|
{% set VERBS = {
|
|
'activate': ('Activeaza', '/admin/activate', ''),
|
|
'block': ('Blocheaza', '/admin/block', ''),
|
|
'archive': ('Arhiveaza', '/admin/archive', ''),
|
|
'delete': ('Sterge', '/admin/delete', 'danger')
|
|
} %}
|
|
|
|
{# Tier-uri selectabile in panou (cod, eticheta). Aliniat cu app/plans.py#PLANS. #}
|
|
{% set TIERS = [('free', 'Gratuit'), ('standard', 'Standard'), ('pro', 'Pro'), ('premium', 'Premium')] %}
|
|
|
|
{% macro lifecycle_block(title, rows, block_id, bulk_verbs, row_verbs) %}
|
|
<div class="card">
|
|
<h3 style="margin-top:0;">{{ title }} ({{ rows|length }})</h3>
|
|
{% if rows %}
|
|
{# Bara bulk: form propriu (id=bulk-<block>); checkbox-urile randurilor se leaga prin atributul
|
|
HTML5 form= (fara form-uri imbricate). Ascunsa pana exista o selectie (JS). #}
|
|
<form id="bulk-{{ block_id }}" method="post" class="bulk-form" data-block="{{ block_id }}">
|
|
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
|
<div class="bulk-bar" hidden>
|
|
<span class="bulk-count muted" style="font-size:13px;">0 selectate</span>
|
|
{% for v in bulk_verbs %}
|
|
{% set label, action, cls = VERBS[v] %}
|
|
<button type="submit" formaction="{{ action }}"
|
|
{% if v == 'delete' %}onclick="return confirm('Stergi conturile selectate? (stergere soft, datele se purjeaza)');"{% endif %}
|
|
style="{% if cls == 'danger' %}background:var(--card); color:var(--err); border-color:var(--err);{% endif %}">{{ label }}</button>
|
|
{% endfor %}
|
|
</div>
|
|
</form>
|
|
|
|
<div class="tablewrap">
|
|
<table>
|
|
<thead><tr>
|
|
<th style="width:28px;"><input type="checkbox" class="master-check" data-block="{{ block_id }}"
|
|
aria-label="Selecteaza tot"></th>
|
|
<th>ID</th><th>Companie</th><th>CUI</th><th>Email</th><th>Plan curent</th><th>Plan cerut</th><th>Stare</th><th>Inregistrat</th><th>Actiuni</th>
|
|
</tr></thead>
|
|
<tbody>
|
|
{% for acct in rows %}
|
|
<tr>
|
|
<td><input type="checkbox" name="account_id" value="{{ acct.id }}" form="bulk-{{ block_id }}"
|
|
class="row-check" data-block="{{ block_id }}"
|
|
aria-label="Selecteaza contul {{ acct.name }}"></td>
|
|
<td class="muted">{{ acct.id }}</td>
|
|
<td>{{ acct.name }}</td>
|
|
<td class="muted">{{ acct.cui or "—" }}</td>
|
|
<td>{{ acct.email or "—" }}</td>
|
|
<td style="white-space:nowrap;">
|
|
{# Plan EFECTIV acum (prominent): trial Pro activ ridica free->pro. #}
|
|
<div style="margin-bottom:5px;">
|
|
<span class="pill" style="font-weight:600;">{{ acct.tier_efectiv_label }}</span>
|
|
{% if acct.trial_activ %}
|
|
<span class="muted" style="font-size:11px;">
|
|
trial{% if acct.trial_zile %} · {{ acct.trial_zile }} {{ 'zi' if acct.trial_zile == 1 else 'zile' }} ramase{% endif %}
|
|
→ apoi {{ acct.tier_label }}
|
|
</span>
|
|
{% endif %}
|
|
</div>
|
|
{# Schimbare plan inline: select tier de baza + Aplica. Form propriu (nu imbricat in bulk-form).
|
|
Aplica INCHEIE trial-ul si seteaza planul ales ca real, cu efect imediat. #}
|
|
<form method="post" action="/admin/set-tier" class="tier-form"
|
|
style="display:flex;align-items:center;gap:6px;">
|
|
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
|
<input type="hidden" name="account_id" value="{{ acct.id }}">
|
|
<select name="tier" aria-label="Plan pentru {{ acct.name }}"
|
|
style="padding:4px 8px;min-height:32px;max-width:130px;">
|
|
{% for code, label in TIERS %}
|
|
<option value="{{ code }}"{% if acct.tier == code %} selected{% endif %}>{{ label }}</option>
|
|
{% endfor %}
|
|
</select>
|
|
<button type="submit" class="btn-sm"
|
|
title="Aplica planul ales ca plan real (incheie trial-ul daca e activ)">Aplica</button>
|
|
</form>
|
|
{# Acorda/prelungeste trial Pro de N zile, fara a schimba tier-ul de baza. #}
|
|
<form method="post" action="/admin/set-trial" class="trial-form"
|
|
style="display:flex;align-items:center;gap:6px;margin-top:5px;">
|
|
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
|
<input type="hidden" name="account_id" value="{{ acct.id }}">
|
|
<input type="number" name="trial_days" value="30" min="1" max="3650"
|
|
aria-label="Zile trial Pro pentru {{ acct.name }}"
|
|
style="padding:4px 8px;min-height:32px;width:64px;">
|
|
<button type="submit" class="btn-sm"
|
|
title="Acorda/prelungeste trial Pro de la acum (nu schimba tier-ul de baza)">Trial Pro</button>
|
|
</form>
|
|
</td>
|
|
<td class="muted">{{ acct.requested_plan_label }}</td>
|
|
<td><span class="pill">{{ acct.status }}</span></td>
|
|
<td class="muted">{{ acct.created_at or "—" }}</td>
|
|
<td style="white-space:nowrap;">
|
|
<details class="kebab">
|
|
<summary aria-label="Actiuni pentru {{ acct.name }}">⋯</summary>
|
|
<div class="kebab-menu">
|
|
{% for v in row_verbs %}
|
|
{% set label, action, cls = VERBS[v] %}
|
|
{# Confirm fara nume interpolat: un apostrof in numele firmei (free-form) ar rupe
|
|
string-ul JS din atributul inline (entitatea ' e decodata inainte de parse). #}
|
|
<form method="post" action="{{ action }}"
|
|
{% if v == 'delete' %}onsubmit="return confirm('Stergi acest cont? (stergere soft)');"{% endif %}>
|
|
<input type="hidden" name="csrf_token" value="{{ csrf_token }}">
|
|
<input type="hidden" name="account_id" value="{{ acct.id }}">
|
|
{% if v == 'activate' and not acct.is_complete %}
|
|
<button type="submit"{% if cls == 'danger' %} class="danger"{% endif %}
|
|
disabled
|
|
title="Completeaza datele firmei (companie + email + CUI) inainte de activare">{{ label }}</button>
|
|
{% else %}
|
|
<button type="submit"{% if cls == 'danger' %} class="danger"{% endif %}>{{ label }}</button>
|
|
{% endif %}
|
|
</form>
|
|
{% endfor %}
|
|
</div>
|
|
</details>
|
|
</td>
|
|
</tr>
|
|
{% endfor %}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
{% else %}
|
|
<p class="empty">Niciun cont.</p>
|
|
{% endif %}
|
|
</div>
|
|
{% endmacro %}
|
|
|
|
<style>
|
|
/* Bara de actiuni bulk — ascunsa pana exista selectie. `[hidden]` trebuie sa invinga
|
|
display-ul, deci stilul sta in CSS (NU inline cu display:flex, care ar invinge [hidden]). */
|
|
.bulk-bar { display:flex; align-items:center; gap:8px; flex-wrap:wrap; margin-bottom:10px;
|
|
padding:8px 10px; border:1px solid var(--line); border-radius:8px;
|
|
background:color-mix(in srgb, var(--accent) 8%, var(--card)); }
|
|
.bulk-bar[hidden] { display:none; }
|
|
/* Kebab per-rand: stiluri partajate in base.html (position:fixed, anti-clipping tablewrap). */
|
|
</style>
|
|
|
|
<div style="display:flex;align-items:center;gap:16px;margin-bottom:20px;">
|
|
<h2 style="margin:0;">Conturi clienti</h2>
|
|
<a href="/" class="cardlink muted">Inapoi la dashboard</a>
|
|
</div>
|
|
|
|
{% if error %}
|
|
<div class="banner" style="margin-bottom:16px;padding:10px 14px;">{{ error }}</div>
|
|
{% endif %}
|
|
|
|
{{ lifecycle_block("Conturi in asteptare", pending, "pending",
|
|
['activate', 'block', 'archive', 'delete'],
|
|
['activate', 'block', 'archive', 'delete']) }}
|
|
|
|
{{ lifecycle_block("Conturi active", active, "active",
|
|
['block', 'archive', 'delete'],
|
|
['block', 'archive', 'delete']) }}
|
|
|
|
{# Conturi suspendate (blocate/arhivate): reactivare sau stergere. Stare reala in pill. #}
|
|
{{ lifecycle_block("Conturi blocate / arhivate", suspended, "suspended",
|
|
['activate', 'delete'],
|
|
['activate', 'delete']) }}
|
|
|
|
<script>
|
|
(function() {
|
|
// Selectie + bara bulk, scoped pe fiecare bloc (pending/active) prin data-block.
|
|
document.querySelectorAll('.master-check').forEach(function(master) {
|
|
var block = master.getAttribute('data-block');
|
|
var rows = Array.prototype.slice.call(
|
|
document.querySelectorAll('.row-check[data-block="' + block + '"]'));
|
|
var form = document.getElementById('bulk-' + block);
|
|
var bar = form ? form.querySelector('.bulk-bar') : null;
|
|
var count = form ? form.querySelector('.bulk-count') : null;
|
|
|
|
function refresh() {
|
|
var n = rows.filter(function(r) { return r.checked; }).length;
|
|
if (bar) bar.hidden = (n === 0);
|
|
if (count) count.textContent = n + ' selectate';
|
|
master.checked = (n > 0 && n === rows.length);
|
|
master.indeterminate = (n > 0 && n < rows.length);
|
|
}
|
|
master.addEventListener('change', function() {
|
|
rows.forEach(function(r) { r.checked = master.checked; });
|
|
refresh();
|
|
});
|
|
rows.forEach(function(r) { r.addEventListener('change', refresh); });
|
|
refresh();
|
|
});
|
|
})();
|
|
</script>
|
|
|
|
{% endblock %}
|