Paginacija, interaktivna pretraga i optimizacija prikaza
- Dodata server-side paginacija za magacin (127 artikala) i klijente (1040) — Limit/Offset u ArtikalFilter i KlijentFilter, 100 po stranici — PrebrojiPoFilteru za izračunavanje ukupnog broja stranica - Interaktivna pretraga (search-as-you-type) sa HTMX: — hx-trigger="keyup changed delay:300ms" na polju pretrage — HTMX menja samo #magacin-rezultati / #klijenti-rezultati — Polje pretrage ostaje u fokusu tokom osvežavanja - Popravljena pretraga klijenata po imenu i prezimenu: — Dodato (ime || ' ' || prezime) LIKE u sva tri upita — "Ivana Lazić" sada pronalazi klijenta - CSS optimizacije za velike liste: — content-visibility: auto na redovima tabela i karticama — contain-intrinsic-size za stabilan scroll — animation-delay produžen do 20. reda / 10. kartice
This commit is contained in:
@@ -1109,6 +1109,29 @@ select {
|
||||
.tabela tbody tr:nth-child(8) { animation-delay: 0.32s; }
|
||||
.tabela tbody tr:nth-child(9) { animation-delay: 0.36s; }
|
||||
.tabela tbody tr:nth-child(10) { animation-delay: 0.40s; }
|
||||
.tabela tbody tr:nth-child(11) { animation-delay: 0.44s; }
|
||||
.tabela tbody tr:nth-child(12) { animation-delay: 0.48s; }
|
||||
.tabela tbody tr:nth-child(13) { animation-delay: 0.52s; }
|
||||
.tabela tbody tr:nth-child(14) { animation-delay: 0.56s; }
|
||||
.tabela tbody tr:nth-child(15) { animation-delay: 0.60s; }
|
||||
.tabela tbody tr:nth-child(16) { animation-delay: 0.64s; }
|
||||
.tabela tbody tr:nth-child(17) { animation-delay: 0.68s; }
|
||||
.tabela tbody tr:nth-child(18) { animation-delay: 0.72s; }
|
||||
.tabela tbody tr:nth-child(19) { animation-delay: 0.76s; }
|
||||
.tabela tbody tr:nth-child(20) { animation-delay: 0.80s; }
|
||||
|
||||
/* content-visibility: auto – browser preskače render elemenata van viewport-a.
|
||||
Ovo rešava problem sa 1000+ redova gde skrolovanje postaje prazno
|
||||
dok browser ne stigne da iscrta sve. */
|
||||
.tabela tbody tr {
|
||||
content-visibility: auto;
|
||||
contain-intrinsic-size: 48px;
|
||||
}
|
||||
[class*="-kartice"] > .animiraj,
|
||||
.klijenti-kartice > .animiraj {
|
||||
content-visibility: auto;
|
||||
contain-intrinsic-size: 120px;
|
||||
}
|
||||
|
||||
/* Stagger mobilnih lista-kartica (parnjak gornjeg za tabele). Pogađa kartice
|
||||
unutar bilo kog .X-kartice kontejnera; ostali tipovi kartica (detalji/forma/
|
||||
@@ -1118,6 +1141,11 @@ select {
|
||||
[class*="-kartice"] > .animiraj:nth-child(3) { animation-delay: 0.16s; }
|
||||
[class*="-kartice"] > .animiraj:nth-child(4) { animation-delay: 0.22s; }
|
||||
[class*="-kartice"] > .animiraj:nth-child(5) { animation-delay: 0.28s; }
|
||||
[class*="-kartice"] > .animiraj:nth-child(6) { animation-delay: 0.34s; }
|
||||
[class*="-kartice"] > .animiraj:nth-child(7) { animation-delay: 0.40s; }
|
||||
[class*="-kartice"] > .animiraj:nth-child(8) { animation-delay: 0.46s; }
|
||||
[class*="-kartice"] > .animiraj:nth-child(9) { animation-delay: 0.52s; }
|
||||
[class*="-kartice"] > .animiraj:nth-child(10) { animation-delay: 0.58s; }
|
||||
|
||||
/* Stagger naslaganih .kartica.animiraj na stranicama podešavanja/profila — JEDNO mesto.
|
||||
Descendant selektor (razmak, ne >): nth-child se računa po neposrednom roditelju kao i
|
||||
|
||||
@@ -12,19 +12,24 @@
|
||||
<div class="poruka-uspeh poruka-animacija">Klijent je uspešno obrisan.</div>
|
||||
{{end}}
|
||||
|
||||
<!-- gornja traka: dugme + pretraga -->
|
||||
<!-- gornja traka: dugme + interaktivna pretraga -->
|
||||
<div style="display:flex;gap:10px;flex-wrap:wrap;align-items:center;">
|
||||
<a href="/klijenti/novi" class="btn-primarno">+ Novi klijent</a>
|
||||
<form method="GET" action="/klijenti" style="display:flex;gap:8px;flex:1;min-width:200px;">
|
||||
<form method="GET" action="/klijenti" style="display:flex;gap:8px;flex:1;min-width:200px;"
|
||||
hx-get="/klijenti" hx-target="#klijenti-rezultati" hx-select="#klijenti-rezultati" hx-swap="innerHTML" hx-push-url="true">
|
||||
<input type="text" name="pretraga" value="{{.Pretraga}}"
|
||||
placeholder="Pretraži po imenu ili nazivu firme..."
|
||||
style="flex:1;">
|
||||
style="flex:1;"
|
||||
hx-trigger="keyup changed delay:300ms, search"
|
||||
hx-get="/klijenti" hx-target="#klijenti-rezultati" hx-select="#klijenti-rezultati" hx-swap="innerHTML" hx-push-url="true">
|
||||
<button type="submit" class="btn-primarno">
|
||||
Traži
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<div id="klijenti-rezultati">
|
||||
|
||||
<!-- desktop tabela -->
|
||||
<div class="kartica klijenti-tabela kartica-tabela animiraj">
|
||||
<div class="tabela-skrol">
|
||||
@@ -140,5 +145,22 @@
|
||||
{{end}}
|
||||
</div>
|
||||
|
||||
<!-- paginacija -->
|
||||
{{if gt .UkupnoStranica 1}}
|
||||
<div class="paginacija" style="display:flex;align-items:center;justify-content:center;gap:12px;padding:12px 0;flex-wrap:wrap;">
|
||||
<span style="font-size:13px;color:var(--tekst-sporedni);">{{.UkupnoKlijenata}} klijenata — {{.StranicaBr}}/{{.UkupnoStranica}}</span>
|
||||
<div style="display:flex;gap:4px;">
|
||||
{{if gt .StranicaBr 1}}
|
||||
<a href="?stranica={{.StranicaPrev}}{{.StranicaQueryUrl}}" class="btn-sekundarno-malo" hx-target="#klijenti-rezultati" hx-select="#klijenti-rezultati" hx-swap="innerHTML" hx-push-url="true">← Prethodna</a>
|
||||
{{end}}
|
||||
{{if lt .StranicaBr .UkupnoStranica}}
|
||||
<a href="?stranica={{.StranicaNext}}{{.StranicaQueryUrl}}" class="btn-primarno-malo" hx-target="#klijenti-rezultati" hx-select="#klijenti-rezultati" hx-swap="innerHTML" hx-push-url="true">Sledeća →</a>
|
||||
{{end}}
|
||||
</div>
|
||||
</div>
|
||||
{{end}}
|
||||
|
||||
</div><!-- kraj #klijenti-rezultati -->
|
||||
|
||||
</div>
|
||||
{{end}}
|
||||
|
||||
@@ -25,20 +25,27 @@
|
||||
{{end}}
|
||||
</div>
|
||||
|
||||
<!-- pretraga i filteri -->
|
||||
<form method="GET" action="/magacin" class="kolona" style="gap:10px;">
|
||||
<!-- pretraga i filteri — interaktivna pretraga (hx-trigger) -->
|
||||
<form method="GET" action="/magacin" class="kolona" style="gap:10px;"
|
||||
hx-get="/magacin" hx-target="#magacin-rezultati" hx-select="#magacin-rezultati" hx-swap="innerHTML" hx-push-url="true">
|
||||
<input type="text" name="pretraga" value="{{.Filter.Pretraga}}"
|
||||
placeholder="Pretraži artikle..."
|
||||
style="width:100%;">
|
||||
style="width:100%;"
|
||||
hx-trigger="keyup changed delay:300ms, search"
|
||||
hx-get="/magacin" hx-target="#magacin-rezultati" hx-select="#magacin-rezultati" hx-swap="innerHTML" hx-push-url="true">
|
||||
<div style="display:flex;gap:10px;flex-wrap:wrap;align-items:center;">
|
||||
<select name="kategorija" style="flex:1;min-width:140px;">
|
||||
<select name="kategorija" style="flex:1;min-width:140px;"
|
||||
hx-trigger="change"
|
||||
hx-get="/magacin" hx-target="#magacin-rezultati" hx-select="#magacin-rezultati" hx-swap="innerHTML" hx-push-url="true">
|
||||
<option value="">Sve kategorije</option>
|
||||
{{range .Kategorije}}
|
||||
<option value="{{.ID}}" {{if eq (printf "%d" .ID) $.KategorijaIDStr}}selected{{end}}>{{.Naziv}}</option>
|
||||
{{end}}
|
||||
</select>
|
||||
<label style="display:inline-flex;align-items:center;gap:6px;font-size:13px;color:var(--tekst-sporedni);cursor:pointer;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;background:var(--kartica);white-space:nowrap;">
|
||||
<input type="checkbox" name="kriticni" value="1" {{if .Filter.SamoKriticni}}checked{{end}}>
|
||||
<input type="checkbox" name="kriticni" value="1" {{if .Filter.SamoKriticni}}checked{{end}}
|
||||
hx-trigger="change"
|
||||
hx-get="/magacin" hx-target="#magacin-rezultati" hx-select="#magacin-rezultati" hx-swap="innerHTML" hx-push-url="true">
|
||||
Samo kritični
|
||||
</label>
|
||||
<button type="submit" class="btn-primarno">
|
||||
@@ -47,6 +54,9 @@
|
||||
</div>
|
||||
</form>
|
||||
|
||||
<!-- rezultati pretrage — HTMX zamenjuje samo ovo, polje za pretragu ostaje u fokusu -->
|
||||
<div id="magacin-rezultati">
|
||||
|
||||
<!-- tabela artikala -->
|
||||
<div class="kartica animiraj" style="padding:0;overflow:hidden;">
|
||||
<div style="overflow-x:auto;">
|
||||
@@ -162,6 +172,23 @@
|
||||
{{end}}
|
||||
</div>
|
||||
|
||||
<!-- paginacija -->
|
||||
{{if gt .UkupnoStranica 1}}
|
||||
<div class="paginacija" style="display:flex;align-items:center;justify-content:center;gap:12px;padding:12px 0;flex-wrap:wrap;">
|
||||
<span style="font-size:13px;color:var(--tekst-sporedni);">{{.UkupnoArtikala}} artikala — {{.StranicaBr}}/{{.UkupnoStranica}}</span>
|
||||
<div style="display:flex;gap:4px;">
|
||||
{{if gt .StranicaBr 1}}
|
||||
<a href="?stranica={{.StranicaPrev}}{{.StranicaQueryUrl}}" class="btn-sekundarno-malo" hx-target="#magacin-rezultati" hx-select="#magacin-rezultati" hx-swap="innerHTML" hx-push-url="true">← Prethodna</a>
|
||||
{{end}}
|
||||
{{if lt .StranicaBr .UkupnoStranica}}
|
||||
<a href="?stranica={{.StranicaNext}}{{.StranicaQueryUrl}}" class="btn-primarno-malo" hx-target="#magacin-rezultati" hx-select="#magacin-rezultati" hx-swap="innerHTML" hx-push-url="true">Sledeća →</a>
|
||||
{{end}}
|
||||
</div>
|
||||
</div>
|
||||
{{end}}
|
||||
|
||||
</div><!-- kraj #magacin-rezultati -->
|
||||
|
||||
</div>
|
||||
{{end}}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user