Implementeaza PRD 3.6 (US-001..007), pe canalul de import + stratul web;
worker / masina stari / idempotenta / mapare raman neatinse.
- US-003/004: tab-ul "Trimiteri" eliminat; Trimiterile devin sectiune
permanenta sub upload pe Acasa ("Trimiterile tale"); upload comprimat la
bara slim (hero pastrat la first-run); ?tab=coada si /_fragments/coada
servesc Acasa (fara fragment orfan); poll gated pe visibilityState.
- US-001: coloana noua import_rows.override_json (nullable, Fernet, Approach B)
+ _migrate defensiv; ruta v1 + alias web .../rand/{i}/editeaza aplica patch
canonic ULTIMUL in _resolve_row_for_preview si commit_import (mutatie pura,
status rederivat, fara drift). Scoping JOIN -> 404, guard committed -> 409,
semantica empty=clear, decrypt fail -> no-op.
- US-002: buton "Editeaza" pe rand; swap pe <tr> + OOB contoare (nu pe sectiune);
form propriu (confirm dezactivat la editare); refoloseste grila responsiva +
error-map din _trimitere_detaliu.html; mutual-exclusion intre randuri.
- US-005/006: "De rezolvat", "Operatii salvate" si "Formate de coloane" ca
tabele (.tablewrap); H4: comutatorul reflecta auto_send STOCAT.
- US-007: bifa "auto-send" devine comutator etichetat pe COADA ("Pune automat
in coada" / "Tine pentru verificare"), scoped pe operatie; name="auto_send"
pastrat (semantica de prezenta -> bool corect cu ambele parsere, zero backend).
Fix-uri gasite la verificarea E2E in browser (htmx 1.9.12, JS — invizibile la
TestClient): useTemplateFragments=true (raspuns <tr>+OOB era parsat in context
de tabel -> swapError + contoare pierdute); re-activarea confirm-btn dupa salvare
deferita pe tick (evita editing=true tranzitoriu); n-hint actualizat de updateN.
Teste: 523 passed. E2E browser: Acasa unificata, upload slim, editare rand
(needs_data -> ok, swap pe rand, contoare OOB), Mapari tabelar + comutator.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
100 lines
6.0 KiB
HTML
100 lines
6.0 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="ro">
|
|
<head>
|
|
<meta charset="utf-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
<title>{% block title %}Gateway RAR AUTOPASS{% endblock %}</title>
|
|
<script src="/static/htmx.min.js"></script>
|
|
<script>
|
|
// US-002 (3.6): raspunsurile de editare-rand contin un <tr> (swap pe rand) PLUS
|
|
// elemente OOB non-rand (#preview-rezumat, #preview-ok-count). Fara fragmente-template,
|
|
// htmx parseaza raspunsul care incepe cu <tr> in context de tabel (<table><tbody>) si
|
|
// "foster-parent"-eaza div/span-urile OOB afara din fragment -> swapError + contoare pierdute.
|
|
// useTemplateFragments parseaza tot intr-un <template>, pastrand rand + OOB impreuna.
|
|
htmx.config.useTemplateFragments = true;
|
|
</script>
|
|
<style>
|
|
:root { --bg:#0f1115; --card:#181b22; --ink:#e6e9ef; --muted:#8b93a7; --line:#262b36;
|
|
--ok:#3ecf8e; --warn:#e6b34a; --err:#e5605e; --accent:#5b8def; }
|
|
* { box-sizing:border-box; }
|
|
body { margin:0; font:15px/1.5 ui-sans-serif,system-ui,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif;
|
|
background:var(--bg); color:var(--ink); -webkit-font-smoothing:antialiased; }
|
|
header { padding:16px 24px; border-bottom:1px solid var(--line); display:flex; align-items:center; gap:12px; }
|
|
header h1 { font-size:20px; margin:0; font-weight:700; letter-spacing:-.01em; }
|
|
header .env { font-size:12px; color:var(--muted); border:1px solid var(--line); padding:2px 8px; border-radius:99px; }
|
|
main { padding:24px; max-width:1100px; margin:0 auto; }
|
|
.card { background:var(--card); border:1px solid var(--line); border-radius:10px; padding:16px 20px; margin-bottom:16px; }
|
|
.banner { border-left:3px solid var(--err); background:#241a1a; }
|
|
.banner.hidden { display:none; }
|
|
/* Tabelele de date au multe coloane; pe ecrane inguste scroll IN card, nu
|
|
impinge layout-ul paginii (altfel toata pagina scrolleaza orizontal). */
|
|
.tablewrap { overflow-x:auto; -webkit-overflow-scrolling:touch; }
|
|
table { width:100%; border-collapse:collapse; font-size:14px; font-variant-numeric:tabular-nums; }
|
|
th,td { text-align:left; padding:8px 10px; border-bottom:1px solid var(--line); white-space:nowrap; }
|
|
th { color:var(--muted); font-weight:500; font-size:12px; text-transform:uppercase; letter-spacing:.04em; }
|
|
.empty { color:var(--muted); padding:24px; text-align:center; }
|
|
.pill { font-size:12px; padding:2px 8px; border-radius:99px; border:1px solid var(--line); }
|
|
.s-queued{color:var(--accent);} .s-sending{color:var(--warn);} .s-sent{color:var(--ok);}
|
|
.s-error,.s-needs_data,.s-needs_mapping{color:var(--err);}
|
|
.s-ok{color:var(--ok);}
|
|
.s-needs_review{color:var(--warn);}
|
|
.s-already_sent,.s-duplicate_in_file{color:var(--muted);}
|
|
.muted { color:var(--muted); }
|
|
a { color:var(--accent); }
|
|
/* Drop zone upload fisier */
|
|
.drop-zone { border:2px dashed var(--line); border-radius:8px; padding:32px 20px;
|
|
text-align:center; transition:border-color .15s,background .15s; }
|
|
.drop-zone.drag-over { border-color:var(--accent); background:rgba(91,141,239,.05); }
|
|
/* Banner varianta warn (nu eroare) */
|
|
.banner.warn { border-left-color:var(--warn); background:#201c0f; }
|
|
/* Bara confirmare sticky */
|
|
.sticky-bar { position:sticky; bottom:0; background:var(--card); border-top:1px solid var(--line);
|
|
padding:12px 16px; display:flex; align-items:flex-start; gap:16px;
|
|
flex-wrap:wrap; z-index:10; }
|
|
/* Indicator HTMX — ascuns pana la request */
|
|
.htmx-indicator { display:none; }
|
|
.htmx-indicator.htmx-request { display:inline; }
|
|
/* Link-uri de actiune in antetul cardurilor: zona de atins mai mare (>=36px) si
|
|
feedback la hover; pe ecrane inguste antetul se rupe curat sub titlu. */
|
|
.cardlink { font-size:13px; padding:7px 10px; border-radius:6px; display:inline-flex;
|
|
align-items:center; min-height:36px; white-space:nowrap; }
|
|
.cardlink:hover { background:var(--line); }
|
|
.flash { background:#16241c; border-left:3px solid var(--ok); padding:8px 12px; border-radius:6px;
|
|
margin:0 0 12px; font-size:13px; }
|
|
.maprow { display:flex; gap:16px; align-items:center; padding:12px 0; border-bottom:1px solid var(--line);
|
|
flex-wrap:wrap; }
|
|
.maprow:last-child { border-bottom:0; }
|
|
.mapcol.grow { flex:1 1 280px; min-width:240px; }
|
|
.sugg { color:var(--accent); }
|
|
select, button, input[type=text] { font:inherit; background:var(--bg); color:var(--ink);
|
|
border:1px solid var(--line); border-radius:6px; padding:6px 10px; }
|
|
select { max-width:340px; }
|
|
button { background:var(--accent); border-color:var(--accent); color:#fff; cursor:pointer; }
|
|
button:hover { filter:brightness(1.08); }
|
|
.chk { font-size:13px; color:var(--muted); display:flex; align-items:center; gap:6px; }
|
|
/* Tab-bar (US-003) */
|
|
.tab-bar { display:flex; gap:2px; overflow-x:auto; -webkit-overflow-scrolling:touch;
|
|
border-bottom:1px solid var(--line); margin-bottom:16px; padding-bottom:0;
|
|
scrollbar-width:none; }
|
|
.tab-bar::-webkit-scrollbar { display:none; }
|
|
.tab-link { display:inline-flex; align-items:center; padding:8px 16px; font-size:14px;
|
|
font-weight:500; color:var(--muted); text-decoration:none; border-radius:6px 6px 0 0;
|
|
border:1px solid transparent; border-bottom:none; white-space:nowrap;
|
|
transition:color .12s, background .12s; margin-bottom:-1px; }
|
|
.tab-link:hover { color:var(--ink); background:var(--line); }
|
|
.tab-link.tab-activ { color:var(--ink); background:var(--card);
|
|
border-color:var(--line); border-bottom-color:var(--card); }
|
|
.tab-panel { min-height:120px; }
|
|
.status-bar { margin-bottom:12px; }
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<header>
|
|
<h1>Gateway RAR AUTOPASS</h1>
|
|
<span class="env">{{ rar_env }}</span>
|
|
<span class="muted" style="margin-left:auto; font-size:13px;">v{{ version }}</span>
|
|
</header>
|
|
<main>{% block content %}{% endblock %}</main>
|
|
</body>
|
|
</html>
|