feat(5.13): carduri compacte mobil/tableta + fix editare preview (OOB tr) + toast
Dogfood pe import + Trimiteri (mobil/tableta <1024px), pur CSS + markup, backend
trimitere neatins:
- Card compact real pentru .tabel-trimiteri (preview + Trimiteri): vehicul=titlu,
stare=pill dreapta-sus, operatie+cod, meta data/km muted, nota mica. Inlocuieste
stiva generica eticheta+valoare (carduri de ~450px -> ~135px). Anuleaza regula
desktop tr.trimitere-row > td{padding:11px} in blocul compact.
- FIX editare preview: OOB swap pe <tr> esua tacit in htmx 1.9 (un <tr> brut se
pierde la parsarea unui fragment fara context de tabel) -> randul ramanea cu
starea veche dupa salvare. Inlocuit cu reload complet al preview-ului prin
HX-Trigger:reincarcaPreview + detalii randSalvat. /editeaza si /confirma-review
folosesc helper-ul _raspuns_rand_salvat.
- Feedback post-salvare: toast global "Randul N actualizat · <stare>" + scroll +
flash pe randul actualizat (base.html window.arataToast + listener randSalvat).
- Modal editare: Salveaza + Anuleaza pe acelasi rand (sistem .act): desktop text,
mobil doua iconite Lucide 44px alaturate (save/x). Macro icon('x') + .act-primary.
- Randuri deja-trimise/duplicate colapsate implicit in preview + toggle "Arata N".
- Select "Operatii de mapat" full-width pe mobil (nu mai iese din viewport).
- Bara de filtre Trimiteri adaptata mobil: pills pe banda cu scroll orizontal,
cautare vehicul proeminenta (nu 8 butoane full-width stivuite).
- Nota preview = culoarea camp-fix (accent) ca sa atraga atentia; hint-urile
camp-fix per-camp scoase (campul Note e self-explanatory).
- Confirmare trimitere: scos campul email (Declarant); text mai clar
("Confirma numarul din N gata de trimis"). Backend confirmed_by ramane optional.
Teste: contractul OOB (rupt in browser) inlocuit cu noul contract
(reincarcaPreview + randSalvat) in test_web_preview_edit / test_preview_edit_ui /
test_import_review. Suita: 992 passed (exclus live).
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -241,17 +241,23 @@ def test_editeaza_preview_serveste_fragment_modal(client):
|
||||
"Fragmentul modal nu trebuie sa contina confirm-form"
|
||||
|
||||
|
||||
def test_salvare_preview_inchide_modal_si_oob_rand(client):
|
||||
"""POST /_import/{id}/rand/0/editeaza cu date valide → HX-Trigger-After-Settle: inchideModal
|
||||
+ OOB pe rand (#preview-row-0) si contoare (#preview-rezumat).
|
||||
def test_salvare_preview_inchide_modal_si_reincarca(client):
|
||||
"""POST /_import/{id}/rand/0/editeaza cu date valide → inchide modalul + reincarca preview-ul.
|
||||
|
||||
Contractul a fost schimbat (dogfood 5.13): OOB swap pe <tr> esua tacit in htmx 1.9
|
||||
(un <tr> brut se pierde la parsarea fragmentului) -> randul ramanea cu starea veche.
|
||||
Acum raspunsul NU mai contine OOB pe rand; emite HX-Trigger:
|
||||
- reincarcaPreview -> #import-section isi reincarca preview-ul complet;
|
||||
- randSalvat (cu nr + stare) -> toast + evidentiere in base.html;
|
||||
iar HX-Trigger-After-Settle: inchideModal inchide modalul.
|
||||
|
||||
Verifica:
|
||||
- Status 200
|
||||
- Header HX-Trigger-After-Settle contine 'inchideModal'
|
||||
- Raspuns contine OOB pentru randul actualizat (hx-swap-oob prezent)
|
||||
- Raspuns contine OOB pentru rezumat (#preview-rezumat)
|
||||
- NU re-randeaza intreaga sectiune (#import-section absent)
|
||||
- HX-Trigger-After-Settle contine 'inchideModal'
|
||||
- HX-Trigger contine 'reincarcaPreview' si 'randSalvat' (cu numarul randului)
|
||||
- Raspunsul NU re-randeaza inline randul/sectiunea (reload via GET)
|
||||
"""
|
||||
import json as _json
|
||||
_seed_op1()
|
||||
iid = _upload_and_preview(client)
|
||||
|
||||
@@ -259,26 +265,21 @@ def test_salvare_preview_inchide_modal_si_oob_rand(client):
|
||||
"data_prestatie": "2026-06-15",
|
||||
})
|
||||
assert r.status_code == 200, r.text
|
||||
html = r.text
|
||||
|
||||
# Header de inchidere modal
|
||||
trigger = r.headers.get("HX-Trigger-After-Settle", "")
|
||||
assert "inchideModal" in trigger, \
|
||||
f"Header HX-Trigger-After-Settle trebuie sa contina 'inchideModal', gasit: '{trigger}'"
|
||||
trigger_settle = r.headers.get("HX-Trigger-After-Settle", "")
|
||||
assert "inchideModal" in trigger_settle, \
|
||||
f"HX-Trigger-After-Settle trebuie sa contina 'inchideModal', gasit: '{trigger_settle}'"
|
||||
|
||||
# OOB pe randul actualizat
|
||||
assert 'id="preview-row-0"' in html, \
|
||||
"Raspunsul trebuie sa contina randul actualizat (#preview-row-0)"
|
||||
assert "hx-swap-oob" in html, \
|
||||
"Raspunsul trebuie sa contina OOB swap"
|
||||
trigger = _json.loads(r.headers.get("HX-Trigger", "{}"))
|
||||
assert trigger.get("reincarcaPreview") is True, \
|
||||
"HX-Trigger trebuie sa ceara reincarcarea preview-ului (reincarcaPreview)"
|
||||
assert "randSalvat" in trigger, "HX-Trigger trebuie sa contina detaliile randului salvat"
|
||||
assert trigger["randSalvat"]["nr"] == 1, "randSalvat.nr = numarul (1-based) al randului editat"
|
||||
assert "stare" in trigger["randSalvat"], "randSalvat trebuie sa contina noua stare (pentru toast)"
|
||||
|
||||
# OOB pe rezumatul stari
|
||||
assert 'id="preview-rezumat"' in html, \
|
||||
"Raspunsul trebuie sa contina OOB pe #preview-rezumat"
|
||||
|
||||
# NU re-randeaza intreaga sectiune de import
|
||||
assert 'id="import-section"' not in html, \
|
||||
"Editarea randului NU trebuie sa re-randeze intreaga sectiune #import-section"
|
||||
# Raspunsul e doar un stub invizibil — randul real vine din reload, nu din OOB.
|
||||
assert 'id="preview-row-0"' not in r.text, \
|
||||
"Noul contract NU mai face OOB swap pe rand (reload complet via reincarcaPreview)"
|
||||
|
||||
|
||||
def test_anuleaza_nu_lasa_rand_orfan(client):
|
||||
|
||||
Reference in New Issue
Block a user