0f1f65c7f7
Tabela nivelacije (migr 045) beleži svaku promenu prodajne cene: artikal, stara→nova cena, razlog, izvor, korisnik, datum. Dva okidača: posebna akcija „Promeni cenu" (modal, izvor 'rucno') i auto-trag pri izmeni artikla (izvor 'izmena'). PromeniCenu je transakciono (update cene + upis zapisa). Pregled /nivelacije sa filterom perioda i razlikom (+/− i %). Modal otvara svoj nextElementSibling — radi i na mobilnom uprkos dupliranim id-jevima iz dva rasporeda.
90 lines
2.6 KiB
Go
90 lines
2.6 KiB
Go
package handler
|
|
|
|
import (
|
|
"errors"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"ntech/internal/db/sqlite"
|
|
"ntech/internal/middleware"
|
|
"ntech/internal/model"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
)
|
|
|
|
// PodaciNivelacije su podaci za pregled istorije promene cena
|
|
type PodaciNivelacije struct {
|
|
model.PodaciStranice
|
|
Zapisi []model.Nivelacija
|
|
Od string
|
|
Do string
|
|
}
|
|
|
|
// Nivelacije renderuje istoriju promena prodajnih cena za izabrani period
|
|
func (h *Handler) Nivelacije(w http.ResponseWriter, r *http.Request) {
|
|
if _, ok := h.zahtevajDozvolu(w, r, "artikal.izmeni"); !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.NivelacijaRepo.Lista(r.Context(), parsiraDatumOpcionalno(odStr), parsiraDatumOpcionalno(doStr))
|
|
if err != nil {
|
|
http.Error(w, "Greška pri učitavanju nivelacija", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
ps := h.popuniPodaciStranice(r, podesavanja)
|
|
ps.Stranica = "nivelacije"
|
|
ps.NaslovStranice = "Nivelacije — promene cena"
|
|
h.renderujTemplate(w, "nivelacije", PodaciNivelacije{
|
|
PodaciStranice: ps,
|
|
Zapisi: zapisi,
|
|
Od: odStr,
|
|
Do: doStr,
|
|
})
|
|
}
|
|
|
|
// PromeniCenuArtikla menja prodajnu cenu artikla i upisuje nivelacioni zapis (izvor "rucno").
|
|
func (h *Handler) PromeniCenuArtikla(w http.ResponseWriter, r *http.Request) {
|
|
k, ok := h.zahtevajDozvolu(w, r, "artikal.izmeni")
|
|
if !ok {
|
|
return
|
|
}
|
|
id, err := strconv.ParseInt(chi.URLParam(r, "id"), 10, 64)
|
|
if err != nil {
|
|
http.Error(w, "Neispravan ID artikla", http.StatusBadRequest)
|
|
return
|
|
}
|
|
if err := r.ParseForm(); err != nil {
|
|
http.Error(w, "Greška pri čitanju forme", http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
novaCena := parsiraIznos(r.FormValue("nova_cena"))
|
|
razlog := strings.TrimSpace(r.FormValue("razlog"))
|
|
if novaCena <= 0 {
|
|
middleware.SetFlash(w, r, h.DB, "greska", "Nova cena mora biti veća od nule.")
|
|
http.Redirect(w, r, "/magacin", http.StatusSeeOther)
|
|
return
|
|
}
|
|
|
|
korisnikID := &k.ID
|
|
_, err = h.NivelacijaRepo.PromeniCenu(r.Context(), id, novaCena, razlog, korisnikID)
|
|
switch {
|
|
case errors.Is(err, sqlite.ErrArtikalNePostoji):
|
|
middleware.SetFlash(w, r, h.DB, "greska", "Artikal nije pronađen.")
|
|
case err != nil:
|
|
middleware.SetFlash(w, r, h.DB, "greska", "Greška pri promeni cene.")
|
|
default:
|
|
middleware.SetFlash(w, r, h.DB, "uspeh", "Prodajna cena je izmenjena.")
|
|
}
|
|
http.Redirect(w, r, "/magacin", http.StatusSeeOther)
|
|
}
|