feat(5.9): US-002 - tabel trimiteri: eticheta umana sub stare, cod RAR simplu, rand->modal

- Sub pill-ul de Stare apare eticheta umana scurta (`eticheta_problema` din US-001),
  text mic `s-error`, doar cand e ne-goala — stare transmisa prin text, nu doar culoare.
- Coloana Operatie linia 2: codul RAR ca chip discret FARA prefixul "cod RAR:";
  cand nemapat ramane "nemapat" muted (comportament 5.8 pastrat).
- R8: regula touch 44px (min-height + padding) pe `tr.trimitere-row` + afordanta hover/focus;
  chevron inexistent in cod (randul declanseaza deja modalul din US-003, fara aria-expanded).
- Teste: 7 teste noi US-002 + actualizate test_operatie_contine_cod_rar / test_tabel_nu_are_coloana_motiv;
  suita completa 826 passed.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
Claude Agent
2026-06-25 08:52:33 +00:00
parent fd4a05436d
commit 878e319ac5
5 changed files with 229 additions and 22 deletions

View File

@@ -51,6 +51,12 @@
<td class="col-id muted" data-eticheta="#">{{ r.id }}</td>
<td class="col-stare" data-eticheta="Stare">
<span class="pill {{ r.stare_css }}" title="{{ r.stare_text }}">{{ r.stare_scurt }}</span>
{# PRD 5.9 US-002 (R1): eticheta umana scurta sub pill — text mic, `s-error`
pe error/needs_* (singurele stari pe care `eticheta_problema` e ne-goala).
Stare transmisa prin TEXT, nu doar culoare. Codul brut ramane in modal. #}
{% if r.eticheta_problema %}
<div class="eticheta-problema s-error">{{ r.eticheta_problema }}</div>
{% endif %}
</td>
<td class="col-vehicul" data-eticheta="Vehicul">
{{ r.prez.vehicul_nr }}
@@ -60,8 +66,10 @@
</td>
<td class="col-operatie" data-eticheta="Operatie">
<div>{{ r.prez.operatie }}</div>
{# PRD 5.9 US-002: doar codul RAR (ex. OE-2), FARA prefixul "cod RAR:" — chip
muted discret; cand nemapat afiseaza "nemapat" muted (comportament 5.8). #}
{% if r.prez.cod_rar and r.prez.cod_rar != '—' %}
<div class="muted cod-rar-sub">cod RAR: {{ r.prez.cod_rar }}</div>
<div class="cod-rar-sub"><span class="cod-rar-cod">{{ r.prez.cod_rar }}</span></div>
{% else %}
<div class="muted cod-rar-sub">nemapat</div>
{% endif %}

View File

@@ -176,6 +176,21 @@
.tabel-trimiteri .col-operatie > div { line-height:1.35; }
/* secundarul muted („cod RAR" / „nemapat") — >=12px, contrast pe var(--muted) >=4.5:1 */
.tabel-trimiteri .cod-rar-sub { font-size:12px; margin-top:2px; }
/* PRD 5.9 US-002: codul RAR pe linia 2 — chip discret, fara prefixul „cod RAR:". */
.tabel-trimiteri .cod-rar-cod { display:inline-block; font-family:ui-monospace,SFMono-Regular,Menlo,monospace;
font-size:12px; padding:1px 7px; border:1px solid var(--line);
border-radius:99px; color:var(--muted); }
/* PRD 5.9 US-002 (R1): eticheta umana scurta sub pill — text mic; clasa `s-error`
o coloreaza (apare doar pe error/needs_*). Stare prin text, nu doar culoare. */
.tabel-trimiteri .eticheta-problema { font-size:12px; line-height:1.3; margin-top:3px; }
/* PRD 5.9 US-002 (R8): randul e clickabil (deschide modalul) -> tinta de atins >=44px
(touch) + afordanta hover/focus. Inlocuieste vechea regula `@media pointer:coarse
.chevron` (chevron eliminat); este SINGURA regula 44px pe rand. */
.tabel-trimiteri tr.trimitere-row { min-height:44px; }
.tabel-trimiteri tr.trimitere-row > td { padding-top:11px; padding-bottom:11px; }
.tabel-trimiteri tr.trimitere-row:hover { background:color-mix(in srgb, var(--accent) 6%, transparent); }
.tabel-trimiteri tr.trimitere-row:focus,
.tabel-trimiteri tr.trimitere-row:focus-visible { outline:2px solid var(--accent); outline-offset:-2px; }
/* 768-1024px: ascunde Actualizat (e in detaliu) -> 7 coloane, fara scroll */
@media (max-width:1024px) {
.tabel-trimiteri .col-actualizat { display:none; }

View File

@@ -29,7 +29,9 @@
"Teste in `tests/test_web_submissions.py`: `test_eticheta_umana_sub_pill` (R1 rename), eticheta prezenta pe error/needs_mapping, goala pe rand ok.",
"`python3 -m pytest tests/test_web_submissions.py -q` trece."
],
"tags": ["backend"],
"tags": [
"backend"
],
"dependsOn": [],
"requiresBrowserCheck": false,
"requiresDesignReview": false,
@@ -57,7 +59,9 @@
"Teste in `tests/test_web_modal.py`: `test_modal_container_in_afara_submissions_wrap`, `test_fragment_detaliu_tinteste_modalul`.",
"`python3 -m pytest tests/test_web_modal.py -q` trece."
],
"tags": ["ui"],
"tags": [
"ui"
],
"dependsOn": [],
"requiresBrowserCheck": true,
"requiresDesignReview": true,
@@ -81,8 +85,12 @@
"Teste in `tests/test_web_responsive.py`: `test_viewport_meta_prezent`, `test_modal_fullscreen_clasa_mobil`, `test_nav_colapsabil_sub_breakpoint`.",
"`python3 -m pytest tests/test_web_responsive.py -q` trece."
],
"tags": ["ui"],
"dependsOn": ["US-003"],
"tags": [
"ui"
],
"dependsOn": [
"US-003"
],
"requiresBrowserCheck": true,
"requiresDesignReview": true,
"passes": false,
@@ -107,16 +115,21 @@
"`python3 -m pytest tests/test_web_submissions.py -q` trece.",
"E2E (requiresBrowserCheck): gstack browser pe `/` — rand `error` arata eticheta umana sub pill; fara chevron; cod RAR fara `cod RAR:`; click pe rand deschide modalul (nu rand-sibling)."
],
"tags": ["ui"],
"dependsOn": ["US-001", "US-003"],
"tags": [
"ui"
],
"dependsOn": [
"US-001",
"US-003"
],
"requiresBrowserCheck": true,
"requiresDesignReview": false,
"passes": false,
"passes": true,
"failed": false,
"blocked": false,
"retries": 0,
"failureReason": "",
"notes": ""
"notes": "Mare parte deja livrat de US-001 (camp eticheta_problema) si US-003 (rand declanseaza modalul). Atins: app/web/templates/_submissions.html (eticheta_problema sub pill cu clasa s-error doar cand ne-gol; col-operatie cod RAR ca chip cod-rar-cod fara prefixul 'cod RAR:'; nemapat muted), app/web/templates/base.html (CSS .eticheta-problema/.cod-rar-cod + R8 regula 44px touch-target pe tr.trimitere-row + hover/focus), tests/test_web_submissions.py (7 teste noi US-002 + actualizat test_operatie_contine_cod_rar si test_tabel_nu_are_coloana_motiv). Chevron inexistent in cod -> R8 doar adaugare regula 44px. Gates: pytest PASS (fisier 20; suita 826 passed, 1 deselected). DEFERAT la VERIFY: requiresBrowserCheck (gstack E2E pe /) + verificarea JS keyboard (Enter/Space deschide modal, Esc readuce focus) — netestabile in TestClient, acoperite prin assert pe markup/atribute (role=button, tabindex=0, aria-haspopup=dialog, fara aria-expanded) + prezenta handler-elor in base.html."
},
{
"id": "US-004",
@@ -138,8 +151,12 @@
"`python3 -m pytest tests/test_web_corectie.py -q` trece.",
"E2E (requiresBrowserCheck): gstack browser — `needs_data` arata fiecare camp o data (nr. pe un rand, VIN dedesubt), corectez data, `Salveaza si retrimite` -> rand `queued`; `error` arata `Re-pune in coada`; `Sterge` clar separat si rosu."
],
"tags": ["ui"],
"dependsOn": ["US-003"],
"tags": [
"ui"
],
"dependsOn": [
"US-003"
],
"requiresBrowserCheck": true,
"requiresDesignReview": true,
"passes": false,
@@ -165,8 +182,12 @@
"`python3 -m pytest tests/test_web_responsive.py -q` trece.",
"E2E (requiresBrowserCheck): gstack browser la 375px pe `/?tab=mapari`, `?tab=jurnal`, Cont/Nomenclator/Integrare si `/admin` — fiecare fara scroll orizontal, tabele lizibile, formulare pe o coloana."
],
"tags": ["ui"],
"dependsOn": ["US-006"],
"tags": [
"ui"
],
"dependsOn": [
"US-006"
],
"requiresBrowserCheck": true,
"requiresDesignReview": false,
"passes": false,
@@ -191,8 +212,12 @@
"`python3 -m pytest tests/test_web_responsive.py -q` trece.",
"E2E (requiresBrowserCheck): gstack browser la 375px pe `/`, `/login`, `/signup` — fara scroll orizontal; upload + filtre pe o coloana; carduri de trimiteri intacte."
],
"tags": ["ui"],
"dependsOn": ["US-006"],
"tags": [
"ui"
],
"dependsOn": [
"US-006"
],
"requiresBrowserCheck": true,
"requiresDesignReview": false,
"passes": false,
@@ -217,8 +242,13 @@
"`python3 -m pytest tests/test_web_modal.py -q` trece.",
"E2E (requiresBrowserCheck): gstack browser — bifez 2 trimiteri, astept >15s: bifele raman; deschid modalul, astept >15s: modalul ramane deschis cu datele intacte."
],
"tags": ["ui"],
"dependsOn": ["US-002", "US-003"],
"tags": [
"ui"
],
"dependsOn": [
"US-002",
"US-003"
],
"requiresBrowserCheck": true,
"requiresDesignReview": false,
"passes": false,

View File

@@ -34,3 +34,37 @@ Note: PRD APROBAT 2026-06-24 cu revizii obligatorii R1-R12 (raport AUTOPLAN). R1
- US-002 depinde de US-001 (acum done) + US-003.
---
## Rate limit la iter 10 — sleep 1800
## Iteratie: 2026-06-25
### Story implementat: US-002 - Tabel trimiteri: eticheta umana sub stare, fara chevron, cod RAR simplu, rand declanseaza modalul (tags: ui)
### Status: Complete
### Gates rulate:
- Typecheck: SKIP (techStack.commands.typecheck gol)
- Lint: SKIP (techStack.commands.lint gol)
- Tests: PASS (tests/test_web_submissions.py 20; suita completa 826 passed, 1 deselected)
- Browser (gstack/E2E requiresBrowserCheck) si /review: DEFERATE la VERIFY (loop fara browser) — vezi AC E2E.
### Ce s-a schimbat:
- app/web/templates/_submissions.html: sub pill (col-stare) randeaza `r.eticheta_problema`
(text mic `eticheta-problema s-error`, DOAR cand e ne-gol); col-operatie linia 2 arata
codul RAR ca chip `cod-rar-cod` FARA prefixul "cod RAR:" (nemapat -> "nemapat" muted).
- app/web/templates/base.html: CSS nou `.eticheta-problema`, `.cod-rar-cod`, si regula R8
44px touch-target pe `tr.trimitere-row` (min-height + padding 11px) + afordanta hover/focus.
- tests/test_web_submissions.py: 7 teste noi US-002 (eticheta sub pill + absenta pe ok, fara
chevron, cod RAR fara prefix, nemapat, rand deschide modal, a11y+keyboard markup); actualizat
`test_operatie_contine_cod_rar` (fara prefix) si `test_tabel_nu_are_coloana_motiv` (eticheta sub pill).
### Learnings:
- Mare parte din scope-ul US-002 era deja livrat de US-001 (campul `eticheta_problema`) si US-003
(randul declanseaza modalul: role=button/tabindex/aria-haspopup, fara aria-expanded; htmx:beforeRequest
fara toggle). Chevron-ul NU exista in cod (grep zero) -> R8 a insemnat doar adaugarea regulii 44px, nu eliminare.
- `motiv_uman` echo-ul mesajului RAR => eticheta_problema poate contine textul brut al erorii. Asta a
intrat in conflict cu vechiul `test_tabel_nu_are_coloana_motiv` (5.8) care interzicea continutul Motiv
in tabel; US-002 R1 il surfaceaza intentionat sub pill, deci testul a fost actualizat (pastreaza doar
invarianta "fara coloana/celula Motiv").
### Next:
- US-004 (detaliu editabil in-place + butoane in modal), US-005 (poll nu inchide modalul), US-006/007/008 (responsive).
- VERIFY: ruleaza gstack E2E pe `/` pentru AC requiresBrowserCheck (eticheta sub pill, fara chevron, cod fara prefix, click deschide modal).
---

View File

@@ -125,7 +125,11 @@ def test_motiv_needs_data_afisat(client):
def test_tabel_nu_are_coloana_motiv(client):
"""PRD 5.8 US-007: coloana Motiv eliminata din thead/tbody (e in detaliu)."""
"""PRD 5.8 US-007: Motiv nu e o coloana separata in thead/tbody.
Nota (PRD 5.9 US-002, R1): eticheta umana scurta a problemei apare acum sub pill-ul
de Stare ca text mic (NU intr-o coloana proprie) — vezi test_eticheta_umana_apare_sub_pill.
"""
acct = _create_account_user("nomotiv@test.com")
_insert_submission(
acct, "needs_data",
@@ -136,12 +140,12 @@ def test_tabel_nu_are_coloana_motiv(client):
assert resp.status_code == 200
html = resp.text
assert "<th>Motiv</th>" not in html
# continutul Motiv nu mai apare in tabel (a fost mutat in detaliu)
assert "lipsa odometru xyz" not in html
# Motiv nu are coloana/celula dedicata (label-ul scurt sta sub pill in col-stare)
assert 'data-eticheta="Motiv"' not in html
def test_operatie_contine_cod_rar(client):
"""PRD 5.8 US-007: coloana Operatie arata 'cod RAR: XXX' cand mapat, 'nemapat' cand nu."""
"""PRD 5.9 US-002: coloana Operatie arata codul RAR simplu (FARA prefix) cand mapat, 'nemapat' cand nu."""
acct = _create_account_user("codrar@test.com")
# mapat: are cod_prestatie -> cod RAR vizibil
_insert_submission(acct, "sent", payload={
@@ -163,7 +167,9 @@ def test_operatie_contine_cod_rar(client):
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
html = resp.text
assert "cod RAR: OE-2" in html
# US-002: codul ramane, dar prefixul textual "cod RAR:" a fost eliminat (R8)
assert "OE-2" in html
assert "cod RAR:" not in html
assert "nemapat" in html
@@ -195,6 +201,120 @@ def test_detaliu_trimitere(client):
assert "99001" in html # nr prezentare RAR
# ---------------------------------------------------------------------------
# US-002 (PRD 5.9): tabel trimiteri — eticheta umana sub pill, fara chevron,
# cod RAR simplu (fara prefix), randul declanseaza modalul. Teste de randare.
# ---------------------------------------------------------------------------
def test_eticheta_umana_apare_sub_pill(client):
"""Sub pill-ul de Stare apare eticheta umana scurta (text), nu codul brut."""
acct = _create_account_user("ets@test.com")
_insert_submission(
acct, "needs_data",
rar_error=json.dumps([{"field": "odometru_final", "message": "lipsa odometru"}]),
)
_login(client, "ets@test.com")
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
html = resp.text
assert "eticheta-problema" in html # containerul textului uman sub pill
assert "s-error" in html # colorat ca problema (error/needs_*)
# NU randeaza cod brut de catalog pe rand
assert "COD_" not in html
assert "RAR_EROARE" not in html
def test_eticheta_umana_absenta_pe_rand_ok(client):
"""Pe randuri fara problema (sent) nu apare eticheta-problema (string gol -> nimic)."""
acct = _create_account_user("etok@test.com")
_insert_submission(acct, "sent", id_prezentare=70123)
_login(client, "etok@test.com")
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
assert "eticheta-problema" not in resp.text
def test_fara_chevron_in_rand(client):
"""R8: niciun chevron in coloana # / pe rand (eliminat impreuna cu CSS/JS asociat)."""
acct = _create_account_user("chev@test.com")
_insert_submission(acct, "sent")
_login(client, "chev@test.com")
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
assert "chevron" not in resp.text
def test_cod_rar_fara_prefix_text(client):
"""Coloana Operatie linia 2: doar codul RAR (ex. OE-2), FARA prefixul 'cod RAR:'."""
acct = _create_account_user("crp@test.com")
_insert_submission(acct, "sent", payload={
"vin": "WVWZZZ1JZXW000111",
"nr_inmatriculare": "B111AAA",
"data_prestatie": "2026-06-18",
"odometru_final": "10000",
"prestatii": [{"cod_prestatie": "OE-2", "denumire": "Verificare X"}],
})
_login(client, "crp@test.com")
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
html = resp.text
assert "OE-2" in html # codul ramane vizibil
assert "cod RAR:" not in html # prefixul textual a disparut
def test_cod_rar_nemapat_muted(client):
"""Cand nemapat, linia 2 arata 'nemapat' muted (comportament 5.8 pastrat)."""
acct = _create_account_user("crn@test.com")
_insert_submission(acct, "needs_mapping", payload={
"vin": "WVWZZZ1JZXW000222",
"nr_inmatriculare": "B222BBB",
"data_prestatie": "2026-06-18",
"odometru_final": "20000",
"prestatii": [{"cod_op_service": "INTERN9", "denumire": "Spalare auto"}],
})
_login(client, "crn@test.com")
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
assert "nemapat" in resp.text
def test_rand_deschide_modal(client):
"""Randul tinteste corpul modalului (#detaliu-modal-body), NU un rand-sibling (US-003)."""
acct = _create_account_user("mod@test.com")
sid = _insert_submission(acct, "sent")
_login(client, "mod@test.com")
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
html = resp.text
assert f'hx-get="/_fragments/trimitere/{sid}"' in html
assert 'hx-target="#detaliu-modal-body"' in html
# randul-sibling de detaliu din 5.8 a fost eliminat
assert "detaliu-rand" not in html
def test_rand_a11y_si_keyboard_markup(client):
"""R8 a11y: rand role=button, tabindex=0, aria-haspopup=dialog, FARA aria-expanded.
Limitare: deschiderea efectiva pe Enter/Space si readucerea focusului pe Esc sunt
gestionate de JS in base.html (keydown delegat), netestabile in TestClient (fara DOM).
Verificam markup-ul/atributele care le activeaza + prezenta handler-elor de tastatura.
"""
acct = _create_account_user("kbd@test.com")
_insert_submission(acct, "sent")
_login(client, "kbd@test.com")
resp = client.get("/_fragments/submissions")
assert resp.status_code == 200
html = resp.text
assert 'role="button"' in html
assert 'tabindex="0"' in html
assert 'aria-haspopup="dialog"' in html
assert "aria-expanded" not in html # R8: nu mai e expand/collapse pe rand
# Handler-ele de tastatura traiesc in base.html (pagina completa).
base = client.get("/?tab=coada").text
assert "Enter" in base and "Spacebar" in base # Enter/Space deschid modalul
assert "Escape" in base # Esc inchide + readuce focusul
# ---------------------------------------------------------------------------
# US-001 (R1): eticheta umana scurta a problemei pe randul de tabel.
# Teste pure pe _submission_row_view — randarea sub pill e US-002.