feat(pdv): nabavka → KPR automatski (Faza 2b-2)
PDV se izvodi iz stope artikla po stavci (aproksimacija: nabavna cena = osnovica bez PDV). Grupisanje po stopi (20→opšta, 10→posebna, ostalo→oslobođena nabavka), broj dokumenta NAB-<id>, veza izvor/izvor_id. Auto-zapisi se ne mogu ručno brisati u KPR; brisanje nabavke uklanja vezani KPR zapis.
This commit is contained in:
@@ -3,6 +3,7 @@ package handler
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"html/template"
|
"html/template"
|
||||||
|
"log/slog"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
@@ -155,6 +156,32 @@ func (h *Handler) SacuvajNabavku(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// automatski zavedi u KPR ako je firma PDV obveznik; PDV se izvodi iz stope artikla
|
||||||
|
if h.modulUkljucen(r.Context(), "pdv") {
|
||||||
|
var stavkePdv []model.NabavkaStavkaPdv
|
||||||
|
for _, s := range stavke {
|
||||||
|
var stopa float64
|
||||||
|
if a, e := h.Artikli.DohvatiID(r.Context(), s.ArtikalID); e == nil {
|
||||||
|
stopa = a.PdvStopa
|
||||||
|
}
|
||||||
|
stavkePdv = append(stavkePdv, model.NabavkaStavkaPdv{
|
||||||
|
Osnovica: float64(s.Kolicina) * s.CenaPoKomadu,
|
||||||
|
PdvStopa: stopa,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
naziv, pib, mesto := "Nepoznat dobavljač", "", ""
|
||||||
|
if nabavka.DobavljacID != nil {
|
||||||
|
if d, e := h.DobavljaciRepo.DohvatiID(r.Context(), *nabavka.DobavljacID); e == nil {
|
||||||
|
naziv, pib, mesto = d.Naziv, d.PIB, d.Mesto
|
||||||
|
}
|
||||||
|
}
|
||||||
|
nabavka.ID = id
|
||||||
|
kpr := model.KprIzNabavke(nabavka, naziv, pib, mesto, stavkePdv)
|
||||||
|
if _, e := h.PdvKprRepo.Kreiraj(r.Context(), &kpr); e != nil {
|
||||||
|
slog.Error("auto-upis u KPR nije uspeo", "nabavka_id", id, "error", e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
http.Redirect(w, r, "/nabavke/"+strconv.FormatInt(id, 10)+"?sacuvano=1", http.StatusSeeOther)
|
http.Redirect(w, r, "/nabavke/"+strconv.FormatInt(id, 10)+"?sacuvano=1", http.StatusSeeOther)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -221,6 +248,10 @@ func (h *Handler) ObrisiNabavku(w http.ResponseWriter, r *http.Request) {
|
|||||||
http.Error(w, "Greška pri brisanju nabavke", http.StatusInternalServerError)
|
http.Error(w, "Greška pri brisanju nabavke", http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
// ukloni vezani auto-KPR zapis (ako ga je ova nabavka kreirala)
|
||||||
|
if err := h.PdvKprRepo.ObrisiPoIzvoru(r.Context(), "nabavka", id); err != nil {
|
||||||
|
slog.Error("brisanje vezanog KPR zapisa nije uspelo", "nabavka_id", id, "error", err)
|
||||||
|
}
|
||||||
|
|
||||||
http.Redirect(w, r, "/nabavke?obrisan=1", http.StatusSeeOther)
|
http.Redirect(w, r, "/nabavke?obrisan=1", http.StatusSeeOther)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,9 @@
|
|||||||
package model
|
package model
|
||||||
|
|
||||||
import "time"
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
// PdvKir je jedan zapis u knjizi izdatih računa (izlazni PDV).
|
// PdvKir je jedan zapis u knjizi izdatih računa (izlazni PDV).
|
||||||
// Iznosi se vode po vrsti stope (opšta/posebna) — vidi migraciju 041.
|
// Iznosi se vode po vrsti stope (opšta/posebna) — vidi migraciju 041.
|
||||||
@@ -176,3 +179,43 @@ func SumirajKpr(zapisi []PdvKpr) PdvKprSume {
|
|||||||
}
|
}
|
||||||
return s
|
return s
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NabavkaStavkaPdv je jedna stavka nabavke sa pripadajućom PDV stopom (iz artikla).
|
||||||
|
type NabavkaStavkaPdv struct {
|
||||||
|
Osnovica float64 // nabavna vrednost stavke (cena × količina) — tretira se kao osnovica bez PDV
|
||||||
|
PdvStopa float64
|
||||||
|
}
|
||||||
|
|
||||||
|
// KprIzNabavke gradi KPR zapis iz nabavke. PDV se izvodi iz stope artikla po stavci
|
||||||
|
// (⚠ aproksimacija: nabavna cena = osnovica bez PDV; stvaran PDV sa računa dobavljača može
|
||||||
|
// se razlikovati). Grupiše po stopi (20→opšta, 10→posebna, ostalo→oslobođena nabavka).
|
||||||
|
// Broj dokumenta je sintetički ("NAB-<id>") jer nabavka ne čuva broj računa dobavljača.
|
||||||
|
func KprIzNabavke(nabavka Nabavka, dobavljacNaziv, dobavljacPib, dobavljacMesto string, stavke []NabavkaStavkaPdv) PdvKpr {
|
||||||
|
id := nabavka.ID
|
||||||
|
k := PdvKpr{
|
||||||
|
DatumPrometa: nabavka.Datum,
|
||||||
|
DatumKnjizenja: nabavka.Datum,
|
||||||
|
BrojDokumenta: fmt.Sprintf("NAB-%d", id),
|
||||||
|
DobavljacNaziv: dobavljacNaziv,
|
||||||
|
DobavljacPib: dobavljacPib,
|
||||||
|
DobavljacMesto: dobavljacMesto,
|
||||||
|
Napomena: nabavka.Napomena,
|
||||||
|
Izvor: "nabavka",
|
||||||
|
IzvorID: &id,
|
||||||
|
}
|
||||||
|
for _, s := range stavke {
|
||||||
|
pdv := s.Osnovica * s.PdvStopa / 100
|
||||||
|
switch s.PdvStopa {
|
||||||
|
case 20:
|
||||||
|
k.OsnovicaOpsta += s.Osnovica
|
||||||
|
k.PdvOpsta += pdv
|
||||||
|
case 10:
|
||||||
|
k.OsnovicaPosebna += s.Osnovica
|
||||||
|
k.PdvPosebna += pdv
|
||||||
|
default:
|
||||||
|
k.OslobodenNabavka += s.Osnovica
|
||||||
|
}
|
||||||
|
k.Ukupno += s.Osnovica + pdv
|
||||||
|
}
|
||||||
|
return k
|
||||||
|
}
|
||||||
|
|||||||
@@ -40,3 +40,33 @@ func TestKirIzProdaje(t *testing.T) {
|
|||||||
t.Errorf("ukupno=%v, očekivano 400 (240+110+50)", k.Ukupno)
|
t.Errorf("ukupno=%v, očekivano 400 (240+110+50)", k.Ukupno)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestKprIzNabavke(t *testing.T) {
|
||||||
|
nabavka := Nabavka{ID: 3, Napomena: "test", Datum: time.Date(2026, 6, 2, 0, 0, 0, 0, time.UTC)}
|
||||||
|
stavke := []NabavkaStavkaPdv{
|
||||||
|
{Osnovica: 200, PdvStopa: 20}, // PDV 40
|
||||||
|
{Osnovica: 100, PdvStopa: 10}, // PDV 10
|
||||||
|
{Osnovica: 50, PdvStopa: 0}, // oslobođena nabavka
|
||||||
|
}
|
||||||
|
|
||||||
|
k := KprIzNabavke(nabavka, "Dobavljač doo", "987654321", "Beograd", stavke)
|
||||||
|
|
||||||
|
if k.Izvor != "nabavka" || k.IzvorID == nil || *k.IzvorID != 3 {
|
||||||
|
t.Errorf("izvor=%q izvor_id=%v, očekivano nabavka/3", k.Izvor, k.IzvorID)
|
||||||
|
}
|
||||||
|
if k.BrojDokumenta != "NAB-3" || k.DobavljacNaziv != "Dobavljač doo" || k.DobavljacPib != "987654321" {
|
||||||
|
t.Errorf("zaglavlje ne odgovara: %+v", k)
|
||||||
|
}
|
||||||
|
if !blizu(k.OsnovicaOpsta, 200) || !blizu(k.PdvOpsta, 40) {
|
||||||
|
t.Errorf("opšta: osnovica=%v pdv=%v, očekivano 200/40", k.OsnovicaOpsta, k.PdvOpsta)
|
||||||
|
}
|
||||||
|
if !blizu(k.OsnovicaPosebna, 100) || !blizu(k.PdvPosebna, 10) {
|
||||||
|
t.Errorf("posebna: osnovica=%v pdv=%v, očekivano 100/10", k.OsnovicaPosebna, k.PdvPosebna)
|
||||||
|
}
|
||||||
|
if !blizu(k.OslobodenNabavka, 50) {
|
||||||
|
t.Errorf("oslobođena nabavka=%v, očekivano 50", k.OslobodenNabavka)
|
||||||
|
}
|
||||||
|
if !blizu(k.Ukupno, 400) {
|
||||||
|
t.Errorf("ukupno=%v, očekivano 400 (240+110+50)", k.Ukupno)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -57,9 +57,13 @@
|
|||||||
<td style="padding:10px 12px;text-align:right;">{{printf "%.2f" .OslobodenNabavka}}</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;font-weight:500;">{{printf "%.2f" .Ukupno}}</td>
|
||||||
<td style="padding:10px 12px;text-align:right;white-space:nowrap;">
|
<td style="padding:10px 12px;text-align:right;white-space:nowrap;">
|
||||||
|
{{if eq .Izvor "rucno"}}
|
||||||
<form method="POST" action="/pdv/kpr/obrisi/{{.ID}}" style="display:inline;">
|
<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>
|
<button type="submit" class="btn-obrisi-malo" data-potvrda="Obrisati ovaj zapis iz KPR?">Obriši</button>
|
||||||
</form>
|
</form>
|
||||||
|
{{else}}
|
||||||
|
<span style="font-size:11px;color:var(--tekst-sporedni);" title="Zavedeno automatski — upravlja se preko nabavke (obriši nabavku)">iz nabavke</span>
|
||||||
|
{{end}}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{{else}}
|
{{else}}
|
||||||
|
|||||||
Reference in New Issue
Block a user