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:
@@ -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
|
||||
}
|
||||
|
||||
|
||||
@@ -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")),
|
||||
}, ""
|
||||
}
|
||||
|
||||
@@ -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",
|
||||
|
||||
@@ -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)
|
||||
}
|
||||
@@ -9,6 +9,8 @@ type Dobavljac struct {
|
||||
KontaktOsoba string
|
||||
Telefon string
|
||||
Email string
|
||||
PIB string
|
||||
Mesto string
|
||||
Napomena string
|
||||
DatumUnosa time.Time
|
||||
}
|
||||
|
||||
@@ -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
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user