diff --git a/cmd/ntech/main.go b/cmd/ntech/main.go index 4e4cb63..60095ec 100644 --- a/cmd/ntech/main.go +++ b/cmd/ntech/main.go @@ -266,6 +266,10 @@ func main() { 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.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/novi", h.NoviArtikal) r.With(doz("artikal.dodaj")).Post("/magacin/novi", h.SacuvajArtikal) diff --git a/internal/db/sqlite/dobavljac.go b/internal/db/sqlite/dobavljac.go index b3f96d4..48cebfa 100644 --- a/internal/db/sqlite/dobavljac.go +++ b/internal/db/sqlite/dobavljac.go @@ -21,7 +21,7 @@ func NoviDobavljacRepo(db *sql.DB) *DobavljacRepo { // Lista vraća listu dobavljača sa opcionom pretragom po nazivu func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dobavljac, error) { 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 WHERE 1=1` @@ -43,9 +43,9 @@ func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dob var rezultat []model.Dobavljac for redovi.Next() { var d model.Dobavljac - var kontaktOsoba, telefon, email, napomena sql.NullString + var kontaktOsoba, telefon, email, pib, mesto, napomena sql.NullString 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 { 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.Telefon = telefon.String d.Email = email.String + d.PIB = pib.String + d.Mesto = mesto.String d.Napomena = napomena.String 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 func (r *DobavljacRepo) DohvatiID(ctx context.Context, id int64) (*model.Dobavljac, error) { 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, ` - 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( - &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 { 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.Telefon = telefon.String d.Email = email.String + d.PIB = pib.String + d.Mesto = mesto.String d.Napomena = napomena.String 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 func (r *DobavljacRepo) Kreiraj(ctx context.Context, d *model.Dobavljac) (int64, error) { rezultat, err := r.db.ExecContext(ctx, ` - INSERT INTO dobavljaci (naziv, kontakt_osoba, telefon, email, napomena) - VALUES (?, ?, ?, ?, ?)`, + INSERT INTO dobavljaci (naziv, kontakt_osoba, telefon, email, pib, mesto, napomena) + VALUES (?, ?, ?, ?, ?, ?, ?)`, 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 { 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 { _, err := r.db.ExecContext(ctx, ` UPDATE dobavljaci SET - naziv = ?, kontakt_osoba = ?, telefon = ?, email = ?, napomena = ? + naziv = ?, kontakt_osoba = ?, telefon = ?, email = ?, pib = ?, mesto = ?, napomena = ? WHERE id = ?`, 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 { return fmt.Errorf("ntech: DobavljacRepo.Izmeni: %w", err) @@ -127,4 +131,3 @@ func (r *DobavljacRepo) Obrisi(ctx context.Context, id int64) error { return nil } - diff --git a/internal/handler/dobavljac.go b/internal/handler/dobavljac.go index 36ba23f..1e92408 100644 --- a/internal/handler/dobavljac.go +++ b/internal/handler/dobavljac.go @@ -214,6 +214,8 @@ func parseFormuDobavljaca(r *http.Request) (model.Dobavljac, string) { KontaktOsoba: strings.TrimSpace(r.FormValue("kontakt_osoba")), Telefon: strings.TrimSpace(r.FormValue("telefon")), Email: email, + PIB: strings.TrimSpace(r.FormValue("pib")), + Mesto: strings.TrimSpace(r.FormValue("mesto")), Napomena: strings.TrimSpace(r.FormValue("napomena")), }, "" } diff --git a/internal/handler/kes.go b/internal/handler/kes.go index 31b12cb..35ab4da 100644 --- a/internal/handler/kes.go +++ b/internal/handler/kes.go @@ -27,6 +27,7 @@ var saSidebar = []string{ "podesavanja", "podesavanja_opste", "podesavanja_izgled", "podesavanja_sistem", "pdv_stope", "pdv_kir", "pdv_kir_forma", + "pdv_kpr", "pdv_kpr_forma", "podsetnici", "podsetnik_forma", "profil_tema", "prodaja", "prodaja_detalji", "prodaja_forma", diff --git a/internal/handler/pdv_kpr.go b/internal/handler/pdv_kpr.go new file mode 100644 index 0000000..0a79452 --- /dev/null +++ b/internal/handler/pdv_kpr.go @@ -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) +} diff --git a/internal/model/dobavljac.go b/internal/model/dobavljac.go index f99c766..7c6c075 100644 --- a/internal/model/dobavljac.go +++ b/internal/model/dobavljac.go @@ -9,6 +9,8 @@ type Dobavljac struct { KontaktOsoba string Telefon string Email string + PIB string + Mesto string Napomena string DatumUnosa time.Time } diff --git a/internal/model/pdv_evidencija.go b/internal/model/pdv_evidencija.go index e909dd7..c9f8204 100644 --- a/internal/model/pdv_evidencija.go +++ b/internal/model/pdv_evidencija.go @@ -94,3 +94,43 @@ type PdvKpr struct { Napomena string 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 +} diff --git a/migrations/043_dobavljac_pib_mesto.sql b/migrations/043_dobavljac_pib_mesto.sql new file mode 100644 index 0000000..50218c3 --- /dev/null +++ b/migrations/043_dobavljac_pib_mesto.sql @@ -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; diff --git a/web/templates/komponente/sidebar.html b/web/templates/komponente/sidebar.html index 4b3df81..720ab75 100644 --- a/web/templates/komponente/sidebar.html +++ b/web/templates/komponente/sidebar.html @@ -100,6 +100,11 @@ KIR + + + KPR + + {{end}}
diff --git a/web/templates/stranice/dobavljac_forma.html b/web/templates/stranice/dobavljac_forma.html index f2d108d..f334738 100644 --- a/web/templates/stranice/dobavljac_forma.html +++ b/web/templates/stranice/dobavljac_forma.html @@ -59,6 +59,20 @@ style="width:100%;"> + +