feat(pdv): KPR — knjiga primljenih računa + PIB/mesto dobavljača (Faza 2a)

KPR (handler, rute pod RequireModul("pdv"), UI sa sumama po stopama,
izbor dobavljača, datum plaćanja, PDV bez odbitka / oslobođena nabavka)
+ stavka u meniju. Dobavljači dobili PIB i mesto (migracija 043) jer KPR
traži PIB dobavljača za POPDV. Time je Faza 2a kompletna (KIR + KPR).
This commit is contained in:
2026-06-14 02:05:33 +02:00
parent e8fdd2dc51
commit 966d1f6c98
12 changed files with 466 additions and 12 deletions
+4
View File
@@ -266,6 +266,10 @@ func main() {
r.With(modul("pdv")).Get("/pdv/kir/nova", h.NoviPdvKir) r.With(modul("pdv")).Get("/pdv/kir/nova", h.NoviPdvKir)
r.With(modul("pdv"), doz("pdv.dodaj")).Post("/pdv/kir/nova", h.SacuvajPdvKir) r.With(modul("pdv"), doz("pdv.dodaj")).Post("/pdv/kir/nova", h.SacuvajPdvKir)
r.With(modul("pdv"), doz("pdv.obrisi")).Post("/pdv/kir/obrisi/{id}", h.ObrisiPdvKir) r.With(modul("pdv"), doz("pdv.obrisi")).Post("/pdv/kir/obrisi/{id}", h.ObrisiPdvKir)
r.With(modul("pdv")).Get("/pdv/kpr", h.PdvKpr)
r.With(modul("pdv")).Get("/pdv/kpr/nova", h.NoviPdvKpr)
r.With(modul("pdv"), doz("pdv.dodaj")).Post("/pdv/kpr/nova", h.SacuvajPdvKpr)
r.With(modul("pdv"), doz("pdv.obrisi")).Post("/pdv/kpr/obrisi/{id}", h.ObrisiPdvKpr)
r.Get("/magacin", h.Magacin) r.Get("/magacin", h.Magacin)
r.Get("/magacin/novi", h.NoviArtikal) r.Get("/magacin/novi", h.NoviArtikal)
r.With(doz("artikal.dodaj")).Post("/magacin/novi", h.SacuvajArtikal) r.With(doz("artikal.dodaj")).Post("/magacin/novi", h.SacuvajArtikal)
+15 -12
View File
@@ -21,7 +21,7 @@ func NoviDobavljacRepo(db *sql.DB) *DobavljacRepo {
// Lista vraća listu dobavljača sa opcionom pretragom po nazivu // Lista vraća listu dobavljača sa opcionom pretragom po nazivu
func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dobavljac, error) { func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dobavljac, error) {
upit := ` upit := `
SELECT id, naziv, kontakt_osoba, telefon, email, napomena, datum_unosa SELECT id, naziv, kontakt_osoba, telefon, email, pib, mesto, napomena, datum_unosa
FROM dobavljaci FROM dobavljaci
WHERE 1=1` WHERE 1=1`
@@ -43,9 +43,9 @@ func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dob
var rezultat []model.Dobavljac var rezultat []model.Dobavljac
for redovi.Next() { for redovi.Next() {
var d model.Dobavljac var d model.Dobavljac
var kontaktOsoba, telefon, email, napomena sql.NullString var kontaktOsoba, telefon, email, pib, mesto, napomena sql.NullString
err := redovi.Scan( err := redovi.Scan(
&d.ID, &d.Naziv, &kontaktOsoba, &telefon, &email, &napomena, &d.DatumUnosa, &d.ID, &d.Naziv, &kontaktOsoba, &telefon, &email, &pib, &mesto, &napomena, &d.DatumUnosa,
) )
if err != nil { if err != nil {
return nil, fmt.Errorf("ntech: DobavljacRepo.Lista: scan: %w", err) return nil, fmt.Errorf("ntech: DobavljacRepo.Lista: scan: %w", err)
@@ -53,6 +53,8 @@ func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dob
d.KontaktOsoba = kontaktOsoba.String d.KontaktOsoba = kontaktOsoba.String
d.Telefon = telefon.String d.Telefon = telefon.String
d.Email = email.String d.Email = email.String
d.PIB = pib.String
d.Mesto = mesto.String
d.Napomena = napomena.String d.Napomena = napomena.String
rezultat = append(rezultat, d) rezultat = append(rezultat, d)
} }
@@ -63,12 +65,12 @@ func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dob
// DohvatiID vraća jednog dobavljača po ID-u // DohvatiID vraća jednog dobavljača po ID-u
func (r *DobavljacRepo) DohvatiID(ctx context.Context, id int64) (*model.Dobavljac, error) { func (r *DobavljacRepo) DohvatiID(ctx context.Context, id int64) (*model.Dobavljac, error) {
var d model.Dobavljac var d model.Dobavljac
var kontaktOsoba, telefon, email, napomena sql.NullString var kontaktOsoba, telefon, email, pib, mesto, napomena sql.NullString
err := r.db.QueryRowContext(ctx, ` err := r.db.QueryRowContext(ctx, `
SELECT id, naziv, kontakt_osoba, telefon, email, napomena, datum_unosa SELECT id, naziv, kontakt_osoba, telefon, email, pib, mesto, napomena, datum_unosa
FROM dobavljaci WHERE id = ?`, id).Scan( FROM dobavljaci WHERE id = ?`, id).Scan(
&d.ID, &d.Naziv, &kontaktOsoba, &telefon, &email, &napomena, &d.DatumUnosa, &d.ID, &d.Naziv, &kontaktOsoba, &telefon, &email, &pib, &mesto, &napomena, &d.DatumUnosa,
) )
if err != nil { if err != nil {
return nil, fmt.Errorf("ntech: DobavljacRepo.DohvatiID: %w", err) return nil, fmt.Errorf("ntech: DobavljacRepo.DohvatiID: %w", err)
@@ -77,6 +79,8 @@ func (r *DobavljacRepo) DohvatiID(ctx context.Context, id int64) (*model.Dobavlj
d.KontaktOsoba = kontaktOsoba.String d.KontaktOsoba = kontaktOsoba.String
d.Telefon = telefon.String d.Telefon = telefon.String
d.Email = email.String d.Email = email.String
d.PIB = pib.String
d.Mesto = mesto.String
d.Napomena = napomena.String d.Napomena = napomena.String
return &d, nil return &d, nil
@@ -85,10 +89,10 @@ func (r *DobavljacRepo) DohvatiID(ctx context.Context, id int64) (*model.Dobavlj
// Kreiraj dodaje novog dobavljača u bazu // Kreiraj dodaje novog dobavljača u bazu
func (r *DobavljacRepo) Kreiraj(ctx context.Context, d *model.Dobavljac) (int64, error) { func (r *DobavljacRepo) Kreiraj(ctx context.Context, d *model.Dobavljac) (int64, error) {
rezultat, err := r.db.ExecContext(ctx, ` rezultat, err := r.db.ExecContext(ctx, `
INSERT INTO dobavljaci (naziv, kontakt_osoba, telefon, email, napomena) INSERT INTO dobavljaci (naziv, kontakt_osoba, telefon, email, pib, mesto, napomena)
VALUES (?, ?, ?, ?, ?)`, VALUES (?, ?, ?, ?, ?, ?, ?)`,
d.Naziv, nullString(d.KontaktOsoba), nullString(d.Telefon), d.Naziv, nullString(d.KontaktOsoba), nullString(d.Telefon),
nullString(d.Email), nullString(d.Napomena), nullString(d.Email), nullString(d.PIB), nullString(d.Mesto), nullString(d.Napomena),
) )
if err != nil { if err != nil {
return 0, fmt.Errorf("ntech: DobavljacRepo.Kreiraj: %w", err) return 0, fmt.Errorf("ntech: DobavljacRepo.Kreiraj: %w", err)
@@ -106,10 +110,10 @@ func (r *DobavljacRepo) Kreiraj(ctx context.Context, d *model.Dobavljac) (int64,
func (r *DobavljacRepo) Izmeni(ctx context.Context, d *model.Dobavljac) error { func (r *DobavljacRepo) Izmeni(ctx context.Context, d *model.Dobavljac) error {
_, err := r.db.ExecContext(ctx, ` _, err := r.db.ExecContext(ctx, `
UPDATE dobavljaci SET UPDATE dobavljaci SET
naziv = ?, kontakt_osoba = ?, telefon = ?, email = ?, napomena = ? naziv = ?, kontakt_osoba = ?, telefon = ?, email = ?, pib = ?, mesto = ?, napomena = ?
WHERE id = ?`, WHERE id = ?`,
d.Naziv, nullString(d.KontaktOsoba), nullString(d.Telefon), d.Naziv, nullString(d.KontaktOsoba), nullString(d.Telefon),
nullString(d.Email), nullString(d.Napomena), d.ID, nullString(d.Email), nullString(d.PIB), nullString(d.Mesto), nullString(d.Napomena), d.ID,
) )
if err != nil { if err != nil {
return fmt.Errorf("ntech: DobavljacRepo.Izmeni: %w", err) return fmt.Errorf("ntech: DobavljacRepo.Izmeni: %w", err)
@@ -127,4 +131,3 @@ func (r *DobavljacRepo) Obrisi(ctx context.Context, id int64) error {
return nil return nil
} }
+2
View File
@@ -214,6 +214,8 @@ func parseFormuDobavljaca(r *http.Request) (model.Dobavljac, string) {
KontaktOsoba: strings.TrimSpace(r.FormValue("kontakt_osoba")), KontaktOsoba: strings.TrimSpace(r.FormValue("kontakt_osoba")),
Telefon: strings.TrimSpace(r.FormValue("telefon")), Telefon: strings.TrimSpace(r.FormValue("telefon")),
Email: email, Email: email,
PIB: strings.TrimSpace(r.FormValue("pib")),
Mesto: strings.TrimSpace(r.FormValue("mesto")),
Napomena: strings.TrimSpace(r.FormValue("napomena")), Napomena: strings.TrimSpace(r.FormValue("napomena")),
}, "" }, ""
} }
+1
View File
@@ -27,6 +27,7 @@ var saSidebar = []string{
"podesavanja", "podesavanja_opste", "podesavanja_izgled", "podesavanja_sistem", "podesavanja", "podesavanja_opste", "podesavanja_izgled", "podesavanja_sistem",
"pdv_stope", "pdv_stope",
"pdv_kir", "pdv_kir_forma", "pdv_kir", "pdv_kir_forma",
"pdv_kpr", "pdv_kpr_forma",
"podsetnici", "podsetnik_forma", "podsetnici", "podsetnik_forma",
"profil_tema", "profil_tema",
"prodaja", "prodaja_detalji", "prodaja_forma", "prodaja", "prodaja_detalji", "prodaja_forma",
+164
View File
@@ -0,0 +1,164 @@
package handler
import (
"net/http"
"strings"
"time"
"ntech/internal/db/sqlite"
"ntech/internal/middleware"
"ntech/internal/model"
"github.com/go-chi/chi/v5"
)
// PodaciPdvKpr su podaci za pregled knjige primljenih računa
type PodaciPdvKpr struct {
model.PodaciStranice
Zapisi []model.PdvKpr
Sume model.PdvKprSume
Od string
Do string
}
// PodaciPdvKprForma su podaci za formu unosa zapisa KPR
type PodaciPdvKprForma struct {
model.PodaciStranice
Greska string
Danas string
Dobavljaci []model.Dobavljac // za izbor dobavljača iz postojećih
}
// PdvKpr renderuje pregled knjige primljenih računa sa sumama po stopama
func (h *Handler) PdvKpr(w http.ResponseWriter, r *http.Request) {
if _, ok := h.zahtevajDozvolu(w, r, "pdv.pregled"); !ok {
return
}
podesavanja, err := sqlite.DohvatiSvaPodesavanja(r.Context(), h.DB)
if err != nil {
http.Error(w, "Greška pri učitavanju podešavanja", http.StatusInternalServerError)
return
}
odStr := r.URL.Query().Get("od")
doStr := r.URL.Query().Get("do")
zapisi, err := h.PdvKprRepo.Lista(r.Context(), parsiraDatumOpcionalno(odStr), parsiraDatumOpcionalno(doStr))
if err != nil {
http.Error(w, "Greška pri učitavanju knjige primljenih računa", http.StatusInternalServerError)
return
}
ps := h.popuniPodaciStranice(r, podesavanja)
ps.Stranica = "pdv-kpr"
ps.NaslovStranice = "KPR — knjiga primljenih računa"
h.renderujTemplate(w, "pdv_kpr", PodaciPdvKpr{
PodaciStranice: ps,
Zapisi: zapisi,
Sume: model.SumirajKpr(zapisi),
Od: odStr,
Do: doStr,
})
}
// NoviPdvKpr prikazuje praznu formu za unos zapisa u KPR
func (h *Handler) NoviPdvKpr(w http.ResponseWriter, r *http.Request) {
if _, ok := h.zahtevajDozvolu(w, r, "pdv.dodaj"); !ok {
return
}
podesavanja, err := sqlite.DohvatiSvaPodesavanja(r.Context(), h.DB)
if err != nil {
http.Error(w, "Greška pri učitavanju podešavanja", http.StatusInternalServerError)
return
}
dobavljaci, _ := h.DobavljaciRepo.Lista(r.Context(), "")
ps := h.popuniPodaciStranice(r, podesavanja)
ps.Stranica = "pdv-kpr"
ps.NaslovStranice = "Novi ulazni račun (KPR)"
h.renderujTemplate(w, "pdv_kpr_forma", PodaciPdvKprForma{
PodaciStranice: ps,
Danas: time.Now().Format("2006-01-02"),
Dobavljaci: dobavljaci,
})
}
// SacuvajPdvKpr prima POST i upisuje novi zapis u KPR
func (h *Handler) SacuvajPdvKpr(w http.ResponseWriter, r *http.Request) {
if _, ok := h.zahtevajDozvolu(w, r, "pdv.dodaj"); !ok {
return
}
if err := r.ParseForm(); err != nil {
http.Error(w, "Greška pri čitanju forme", http.StatusBadRequest)
return
}
datumPrometa, e1 := time.Parse("2006-01-02", strings.TrimSpace(r.FormValue("datum_prometa")))
datumKnjizenja, e2 := time.Parse("2006-01-02", strings.TrimSpace(r.FormValue("datum_knjizenja")))
brojDokumenta := strings.TrimSpace(r.FormValue("broj_dokumenta"))
dobavljacNaziv := strings.TrimSpace(r.FormValue("dobavljac_naziv"))
greska := ""
switch {
case e1 != nil:
greska = "Datum prometa je obavezan i mora biti ispravan."
case e2 != nil:
greska = "Datum knjiženja je obavezan i mora biti ispravan."
case brojDokumenta == "":
greska = "Broj dokumenta je obavezan."
case dobavljacNaziv == "":
greska = "Naziv dobavljača je obavezan."
}
if greska != "" {
middleware.SetFlash(w, r, h.DB, "greska", greska)
http.Redirect(w, r, "/pdv/kpr/nova", http.StatusSeeOther)
return
}
z := model.PdvKpr{
DatumPrometa: datumPrometa,
DatumKnjizenja: datumKnjizenja,
BrojDokumenta: brojDokumenta,
DobavljacNaziv: dobavljacNaziv,
DobavljacPib: strings.TrimSpace(r.FormValue("dobavljac_pib")),
DobavljacMesto: strings.TrimSpace(r.FormValue("dobavljac_mesto")),
OsnovicaOpsta: parsiraIznos(r.FormValue("osnovica_opsta")),
PdvOpsta: parsiraIznos(r.FormValue("pdv_opsta")),
OsnovicaPosebna: parsiraIznos(r.FormValue("osnovica_posebna")),
PdvPosebna: parsiraIznos(r.FormValue("pdv_posebna")),
PdvBezOdbitka: parsiraIznos(r.FormValue("pdv_bez_odbitka")),
OslobodenNabavka: parsiraIznos(r.FormValue("osloboden_nabavka")),
Napomena: strings.TrimSpace(r.FormValue("napomena")),
}
// datum plaćanja je opcionalan
if dp := parsiraDatumOpcionalno(r.FormValue("datum_placanja")); !dp.IsZero() {
z.DatumPlacanja = &dp
}
// ukupna vrednost računa — zbir osnovica, PDV-a i oslobođene nabavke (računa server)
z.Ukupno = z.OsnovicaOpsta + z.PdvOpsta + z.OsnovicaPosebna + z.PdvPosebna +
z.PdvBezOdbitka + z.OslobodenNabavka
if _, err := h.PdvKprRepo.Kreiraj(r.Context(), &z); err != nil {
http.Error(w, "Greška pri čuvanju zapisa", http.StatusInternalServerError)
return
}
middleware.SetFlash(w, r, h.DB, "uspeh", "Ulazni račun je dodat u KPR.")
http.Redirect(w, r, "/pdv/kpr", http.StatusSeeOther)
}
// ObrisiPdvKpr briše zapis iz KPR
func (h *Handler) ObrisiPdvKpr(w http.ResponseWriter, r *http.Request) {
if _, ok := h.zahtevajDozvolu(w, r, "pdv.obrisi"); !ok {
return
}
id, err := parseID(chi.URLParam(r, "id"))
if err != nil {
http.Error(w, "Neispravan ID zapisa", http.StatusBadRequest)
return
}
if err := h.PdvKprRepo.Obrisi(r.Context(), id); err != nil {
http.Error(w, "Greška pri brisanju zapisa", http.StatusInternalServerError)
return
}
middleware.SetFlash(w, r, h.DB, "uspeh", "Zapis je obrisan iz KPR.")
http.Redirect(w, r, "/pdv/kpr", http.StatusSeeOther)
}
+2
View File
@@ -9,6 +9,8 @@ type Dobavljac struct {
KontaktOsoba string KontaktOsoba string
Telefon string Telefon string
Email string Email string
PIB string
Mesto string
Napomena string Napomena string
DatumUnosa time.Time DatumUnosa time.Time
} }
+40
View File
@@ -94,3 +94,43 @@ type PdvKpr struct {
Napomena string Napomena string
DatumUnosa time.Time DatumUnosa time.Time
} }
// OznakaPoreskogBroja vraća „JMBG" za 13-cifreni broj, inače „PIB" (dobavljači su obično firme).
func (k PdvKpr) OznakaPoreskogBroja() string {
cifre := 0
for _, r := range k.DobavljacPib {
if r >= '0' && r <= '9' {
cifre++
}
}
if cifre == 13 {
return "JMBG"
}
return "PIB"
}
// PdvKprSume su zbirovi kolona KPR-a (za red „ukupno" u pregledu knjige).
type PdvKprSume struct {
OsnovicaOpsta float64
PdvOpsta float64
OsnovicaPosebna float64
PdvPosebna float64
PdvBezOdbitka float64
OslobodenNabavka float64
Ukupno float64
}
// SumirajKpr sabira sve kolone iz liste KPR zapisa.
func SumirajKpr(zapisi []PdvKpr) PdvKprSume {
var s PdvKprSume
for _, z := range zapisi {
s.OsnovicaOpsta += z.OsnovicaOpsta
s.PdvOpsta += z.PdvOpsta
s.OsnovicaPosebna += z.OsnovicaPosebna
s.PdvPosebna += z.PdvPosebna
s.PdvBezOdbitka += z.PdvBezOdbitka
s.OslobodenNabavka += z.OslobodenNabavka
s.Ukupno += z.Ukupno
}
return s
}
+4
View File
@@ -0,0 +1,4 @@
-- Dodaje PIB i mesto/grad u dobavljače. PIB dobavljača je obavezan podatak za
-- knjigu primljenih računa (KPR) i POPDV (evidencija prethodnog/odbitnog PDV-a).
ALTER TABLE dobavljaci ADD COLUMN pib TEXT;
ALTER TABLE dobavljaci ADD COLUMN mesto TEXT;
+5
View File
@@ -100,6 +100,11 @@
<span>KIR</span> <span>KIR</span>
<span class="nav-tooltip">KIR — knjiga izdatih računa</span> <span class="nav-tooltip">KIR — knjiga izdatih računa</span>
</a> </a>
<a href="/pdv/kpr" class="nav-stavka {{if eq .Stranica "pdv-kpr"}}aktivan{{end}}">
<svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/><line x1="8" y1="13" x2="16" y2="13"/><line x1="8" y1="17" x2="16" y2="17"/><polyline points="10 9 9 9 8 9"/></svg>
<span>KPR</span>
<span class="nav-tooltip">KPR — knjiga primljenih računa</span>
</a>
{{end}} {{end}}
<div class="nav-separator"></div> <div class="nav-separator"></div>
@@ -59,6 +59,20 @@
style="width:100%;"> style="width:100%;">
</div> </div>
<!-- PIB i mesto (za KPR / POPDV) -->
<div class="forma-grid-2">
<div>
<label class="polje-labela">PIB</label>
<input type="text" name="pib" value="{{.Dobavljac.PIB}}"
placeholder="npr. 123456789" style="width:100%;">
</div>
<div>
<label class="polje-labela">Mesto / grad</label>
<input type="text" name="mesto" value="{{.Dobavljac.Mesto}}"
placeholder="npr. Beograd" style="width:100%;">
</div>
</div>
<!-- napomena --> <!-- napomena -->
<div> <div>
<label class="polje-labela">Napomena</label> <label class="polje-labela">Napomena</label>
+89
View File
@@ -0,0 +1,89 @@
{{template "base" .}}
{{define "naslov"}}KPR — knjiga primljenih računa — NTech{{end}}
{{define "sadrzaj"}}
<div class="stranica-stack" style="width:100%;max-width:100%;">
<!-- zaglavlje: filter perioda + dugme za novi zapis -->
<div class="kartica animiraj" style="margin-bottom:16px;">
<div style="display:flex;justify-content:space-between;align-items:flex-end;gap:12px;flex-wrap:wrap;">
<form method="GET" action="/pdv/kpr" style="display:flex;gap:10px;align-items:flex-end;flex-wrap:wrap;">
<div>
<label class="polje-labela">Od datuma</label>
<input type="date" name="od" value="{{.Od}}" style="padding:8px 10px;border:0.5px solid var(--ivica);border-radius:8px;font-size:13px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">Do datuma</label>
<input type="date" name="do" value="{{.Do}}" style="padding:8px 10px;border:0.5px solid var(--ivica);border-radius:8px;font-size:13px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<button type="submit" style="padding:8px 16px;background:var(--sb-aktivan);color:var(--tekst-jak);border:0.5px solid var(--ivica);border-radius:8px;font-size:13px;cursor:pointer;">Prikaži</button>
{{if or .Od .Do}}<a href="/pdv/kpr" class="nazad-link" style="margin-bottom:0;">Poništi filter</a>{{end}}
</form>
<a href="/pdv/kpr/nova" class="btn-primarno" style="font-size:13px;padding:8px 16px;white-space:nowrap;">+ Nov ulazni račun</a>
</div>
</div>
<!-- knjiga -->
<div class="kartica animiraj" style="padding:0;overflow:hidden;">
<div style="overflow-x:auto;">
<table style="width:100%;border-collapse:collapse;font-size:13px;min-width:980px;">
<thead>
<tr style="text-align:left;color:var(--tekst-sporedni);border-bottom:0.5px solid var(--ivica);">
<th style="padding:10px 12px;">Datum prometa</th>
<th style="padding:10px 12px;">Broj dok.</th>
<th style="padding:10px 12px;">Dobavljač</th>
<th style="padding:10px 12px;text-align:right;">Osn. 20%</th>
<th style="padding:10px 12px;text-align:right;">PDV 20%</th>
<th style="padding:10px 12px;text-align:right;">Osn. 10%</th>
<th style="padding:10px 12px;text-align:right;">PDV 10%</th>
<th style="padding:10px 12px;text-align:right;">PDV bez odb.</th>
<th style="padding:10px 12px;text-align:right;">Oslob. nab.</th>
<th style="padding:10px 12px;text-align:right;">Ukupno</th>
<th style="padding:10px 12px;"></th>
</tr>
</thead>
<tbody>
{{range .Zapisi}}
<tr style="border-bottom:0.5px solid var(--ivica);">
<td style="padding:10px 12px;white-space:nowrap;">{{.DatumPrometa.Format "02.01.2006."}}</td>
<td style="padding:10px 12px;">{{.BrojDokumenta}}</td>
<td style="padding:10px 12px;">{{.DobavljacNaziv}}{{if .DobavljacPib}}<div style="font-size:11px;color:var(--tekst-sporedni);">{{.OznakaPoreskogBroja}}: {{.DobavljacPib}}</div>{{end}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .OsnovicaOpsta}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .PdvOpsta}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .OsnovicaPosebna}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .PdvPosebna}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .PdvBezOdbitka}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .OslobodenNabavka}}</td>
<td style="padding:10px 12px;text-align:right;font-weight:500;">{{printf "%.2f" .Ukupno}}</td>
<td style="padding:10px 12px;text-align:right;white-space:nowrap;">
<form method="POST" action="/pdv/kpr/obrisi/{{.ID}}" style="display:inline;">
<button type="submit" class="btn-obrisi-malo" data-potvrda="Obrisati ovaj zapis iz KPR?">Obriši</button>
</form>
</td>
</tr>
{{else}}
<tr><td colspan="11" style="padding:28px;text-align:center;color:var(--tekst-sporedni);">Nema zapisa u izabranom periodu. <a href="/pdv/kpr/nova" style="color:var(--sb-akcent);">Dodaj prvi.</a></td></tr>
{{end}}
</tbody>
{{if .Zapisi}}
<tfoot>
<tr style="border-top:0.5px solid var(--ivica);font-weight:500;background:var(--pozadina);">
<td style="padding:10px 12px;" colspan="3">UKUPNO ({{len .Zapisi}})</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .Sume.OsnovicaOpsta}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .Sume.PdvOpsta}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .Sume.OsnovicaPosebna}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .Sume.PdvPosebna}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .Sume.PdvBezOdbitka}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .Sume.OslobodenNabavka}}</td>
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .Sume.Ukupno}}</td>
<td></td>
</tr>
</tfoot>
{{end}}
</table>
</div>
</div>
</div>
{{end}}
+126
View File
@@ -0,0 +1,126 @@
{{template "base" .}}
{{define "naslov"}}Novi ulazni račun (KPR) — NTech{{end}}
{{define "sadrzaj"}}
<div class="stranica-stack" style="width:100%;max-width:760px;">
<a href="/pdv/kpr" class="nazad-link">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="15 18 9 12 15 6"/></svg>
Nazad na KPR
</a>
<form method="POST" action="/pdv/kpr/nova">
<!-- dokument -->
<div class="kartica animiraj" style="margin-bottom:16px;">
<div style="font-size:15px;font-weight:500;color:var(--tekst-glavni);margin-bottom:16px;padding-bottom:12px;border-bottom:0.5px solid var(--ivica);">Dokument</div>
<div style="display:grid;grid-template-columns:1fr 1fr;gap:14px;">
<div>
<label class="polje-labela">Datum prometa</label>
<input type="date" name="datum_prometa" value="{{.Danas}}" required style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">Datum knjiženja</label>
<input type="date" name="datum_knjizenja" value="{{.Danas}}" required style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">Datum plaćanja (opciono)</label>
<input type="date" name="datum_placanja" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">Broj dokumenta</label>
<input type="text" name="broj_dokumenta" required style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;" placeholder="npr. 2026-1234">
</div>
</div>
</div>
<!-- dobavljač -->
<div class="kartica animiraj" style="margin-bottom:16px;">
<div style="font-size:15px;font-weight:500;color:var(--tekst-glavni);margin-bottom:16px;padding-bottom:12px;border-bottom:0.5px solid var(--ivica);">Dobavljač</div>
<div class="kolona" style="gap:14px;">
{{if .Dobavljaci}}
<div>
<label class="polje-labela">Izaberi iz dobavljača (opciono)</label>
<select onchange="popuniDobavljaca(this)" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
<option value="">— ručni unos —</option>
{{range .Dobavljaci}}
<option data-naziv="{{.Naziv}}" data-pib="{{.PIB}}" data-mesto="{{.Mesto}}">{{.Naziv}}{{if .Mesto}} — {{.Mesto}}{{end}}</option>
{{end}}
</select>
<div class="pomocni-tekst" style="font-size:12px;margin-top:4px;">Izbor popunjava naziv, PIB i mesto; možeš ih i ručno izmeniti.</div>
</div>
{{end}}
<div>
<label class="polje-labela">Naziv dobavljača</label>
<input type="text" name="dobavljac_naziv" required style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div style="display:grid;grid-template-columns:1fr 1fr;gap:14px;">
<div>
<label class="polje-labela">PIB</label>
<input type="text" name="dobavljac_pib" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">Mesto</label>
<input type="text" name="dobavljac_mesto" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
</div>
</div>
</div>
<!-- iznosi po stopama -->
<div class="kartica animiraj" style="margin-bottom:16px;">
<div style="font-size:15px;font-weight:500;color:var(--tekst-glavni);margin-bottom:6px;padding-bottom:12px;border-bottom:0.5px solid var(--ivica);">Iznosi po stopama</div>
<div class="pomocni-tekst" style="font-size:12px;margin-bottom:14px;">PDV (opšta/posebna) je iznos koji se može odbiti. „Ukupno" se računa automatski.</div>
<div style="display:grid;grid-template-columns:1fr 1fr;gap:14px;">
<div>
<label class="polje-labela">Osnovica opšta (20%)</label>
<input type="number" step="0.01" min="0" name="osnovica_opsta" value="0" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">PDV opšta (20%, odbitni)</label>
<input type="number" step="0.01" min="0" name="pdv_opsta" value="0" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">Osnovica posebna (10%)</label>
<input type="number" step="0.01" min="0" name="osnovica_posebna" value="0" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">PDV posebna (10%, odbitni)</label>
<input type="number" step="0.01" min="0" name="pdv_posebna" value="0" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">PDV bez prava na odbitak</label>
<input type="number" step="0.01" min="0" name="pdv_bez_odbitka" value="0" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div>
<label class="polje-labela">Oslobođena nabavka / bez PDV</label>
<input type="number" step="0.01" min="0" name="osloboden_nabavka" value="0" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
</div>
<div style="margin-top:14px;">
<label class="polje-labela">Napomena</label>
<input type="text" name="napomena" style="width:100%;padding:8px 12px;border:0.5px solid var(--ivica);border-radius:8px;font-size:14px;background:var(--pozadina);color:var(--tekst-glavni);outline:none;">
</div>
<div style="display:flex;justify-content:flex-end;gap:10px;margin-top:20px;">
<a href="/pdv/kpr" class="btn-sekundarno" style="font-size:14px;padding:10px 20px;">Odustani</a>
<button type="submit" style="background:var(--sb-akcent);color:#fff;border:none;padding:10px 24px;border-radius:8px;font-size:14px;font-weight:500;cursor:pointer;">Sačuvaj</button>
</div>
</div>
</form>
</div>
<script>
// Izbor dobavljača iz padajuće liste popunjava polja (naziv, PIB, mesto).
function popuniDobavljaca(sel) {
var o = sel.options[sel.selectedIndex];
var naziv = o.getAttribute('data-naziv');
if (!naziv) return; // „— ručni unos —"
var f = sel.closest('form');
f.querySelector('input[name="dobavljac_naziv"]').value = naziv;
f.querySelector('input[name="dobavljac_pib"]').value = o.getAttribute('data-pib') || '';
f.querySelector('input[name="dobavljac_mesto"]').value = o.getAttribute('data-mesto') || '';
}
</script>
{{end}}