Merge feature/kw-kalkulacija-nivelacija: kalkulacija i nivelacija (§5.4 — Faza A i B)
This commit is contained in:
@@ -278,6 +278,8 @@ func main() {
|
||||
r.With(doz("artikal.izmeni")).Post("/magacin/izmeni/{id}", h.SacuvajIzmenuArtikla)
|
||||
r.With(doz("artikal.obrisi")).Get("/magacin/obrisi/{id}", h.ObrisiArtikal)
|
||||
r.With(doz("artikal.premesti")).Post("/magacin/premesti/{id}", h.PremestiArtikal)
|
||||
r.With(doz("artikal.izmeni")).Post("/magacin/promeni-cenu/{id}", h.PromeniCenuArtikla)
|
||||
r.With(doz("artikal.izmeni")).Get("/nivelacije", h.Nivelacije)
|
||||
r.Get("/magacin/kategorije", h.Kategorije)
|
||||
r.With(doz("kategorija.dodaj")).Post("/magacin/kategorije/dodaj", h.DodajKategoriju)
|
||||
r.With(doz("kategorija.obrisi")).Get("/magacin/kategorije/obrisi/{id}", h.ObrisiKategoriju)
|
||||
|
||||
@@ -13,6 +13,8 @@ type ArtikalRepository interface {
|
||||
DohvatiID(ctx context.Context, id int64) (*model.Artikal, error)
|
||||
Kreiraj(ctx context.Context, a *model.Artikal) (int64, error)
|
||||
Izmeni(ctx context.Context, a *model.Artikal) error
|
||||
// AzurirajCene menja samo nabavnu i prodajnu cenu (kalkulacija pri nabavci)
|
||||
AzurirajCene(ctx context.Context, id int64, nabavna, prodajna float64) error
|
||||
PremestiKategoriju(ctx context.Context, id int64, kategorijaID *int64) error
|
||||
Obrisi(ctx context.Context, id int64) error
|
||||
}
|
||||
@@ -52,6 +54,19 @@ type PdvKprRepository interface {
|
||||
ObrisiPoIzvoru(ctx context.Context, izvor string, izvorID int64) error
|
||||
}
|
||||
|
||||
// NivelacijaRepository definiše operacije nad evidencijom promene prodajnih cena
|
||||
type NivelacijaRepository interface {
|
||||
// PromeniCenu transakciono menja prodajnu cenu artikla i upisuje nivelacioni zapis;
|
||||
// vraća kreirani zapis (sa starom i novom cenom). Izvor je "rucno".
|
||||
PromeniCenu(ctx context.Context, artikalID int64, novaCena float64, razlog string, korisnikID *int64) (*model.Nivelacija, error)
|
||||
// Kreiraj upisuje gotov nivelacioni zapis (npr. auto-trag pri izmeni artikla)
|
||||
Kreiraj(ctx context.Context, n *model.Nivelacija) (int64, error)
|
||||
// Lista vraća nivelacije u periodu (po datumu); nulti datum znači bez granice
|
||||
Lista(ctx context.Context, od, do time.Time) ([]model.Nivelacija, error)
|
||||
// ListaZaArtikal vraća sve nivelacije jednog artikla (najnovije prvo)
|
||||
ListaZaArtikal(ctx context.Context, artikalID int64) ([]model.Nivelacija, error)
|
||||
}
|
||||
|
||||
// ArtikalFilter definiše parametre za filtriranje liste artikala
|
||||
type ArtikalFilter struct {
|
||||
Pretraga string
|
||||
|
||||
@@ -147,6 +147,16 @@ func (r *ArtikalRepo) Izmeni(ctx context.Context, a *model.Artikal) error {
|
||||
return nil
|
||||
}
|
||||
|
||||
// AzurirajCene menja samo nabavnu i prodajnu cenu artikla (kalkulacija pri prijemu robe).
|
||||
func (r *ArtikalRepo) AzurirajCene(ctx context.Context, id int64, nabavna, prodajna float64) error {
|
||||
_, err := r.db.ExecContext(ctx,
|
||||
"UPDATE artikli SET nabavna_cena = ?, prodajna_cena = ? WHERE id = ?", nabavna, prodajna, id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("ntech: ArtikalRepo.AzurirajCene: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// PremestiKategoriju menja samo kategoriju artikla (premeštanje u drugu kategoriju).
|
||||
// kategorijaID može biti nil — tada artikal ostaje bez kategorije.
|
||||
func (r *ArtikalRepo) PremestiKategoriju(ctx context.Context, id int64, kategorijaID *int64) error {
|
||||
|
||||
@@ -0,0 +1,150 @@
|
||||
package sqlite
|
||||
|
||||
import (
|
||||
"context"
|
||||
"database/sql"
|
||||
"errors"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"ntech/internal/model"
|
||||
)
|
||||
|
||||
// NivelacijaRepo je SQLite implementacija NivelacijaRepository interfejsa
|
||||
type NivelacijaRepo struct {
|
||||
db *sql.DB
|
||||
}
|
||||
|
||||
// NoviNivelacijaRepo kreira novi NivelacijaRepo
|
||||
func NoviNivelacijaRepo(db *sql.DB) *NivelacijaRepo {
|
||||
return &NivelacijaRepo{db: db}
|
||||
}
|
||||
|
||||
// ErrArtikalNePostoji se vraća kada se menja cena nepostojećeg artikla
|
||||
var ErrArtikalNePostoji = errors.New("artikal ne postoji")
|
||||
|
||||
// PromeniCenu transakciono menja prodajnu cenu artikla i upisuje nivelacioni zapis.
|
||||
// Stara cena se čita iz baze unutar transakcije; izvor je "rucno".
|
||||
func (r *NivelacijaRepo) PromeniCenu(ctx context.Context, artikalID int64, novaCena float64, razlog string, korisnikID *int64) (*model.Nivelacija, error) {
|
||||
tx, err := r.db.BeginTx(ctx, nil)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo.PromeniCenu: begin: %w", err)
|
||||
}
|
||||
defer tx.Rollback()
|
||||
|
||||
var stara float64
|
||||
err = tx.QueryRowContext(ctx, "SELECT prodajna_cena FROM artikli WHERE id = ?", artikalID).Scan(&stara)
|
||||
if errors.Is(err, sql.ErrNoRows) {
|
||||
return nil, ErrArtikalNePostoji
|
||||
}
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo.PromeniCenu: čitanje cene: %w", err)
|
||||
}
|
||||
|
||||
if _, err = tx.ExecContext(ctx, "UPDATE artikli SET prodajna_cena = ? WHERE id = ?", novaCena, artikalID); err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo.PromeniCenu: update cene: %w", err)
|
||||
}
|
||||
|
||||
sada := time.Now()
|
||||
rez, err := tx.ExecContext(ctx, `
|
||||
INSERT INTO nivelacije (artikal_id, stara_cena, nova_cena, razlog, izvor, korisnik_id, datum)
|
||||
VALUES (?, ?, ?, ?, 'rucno', ?, ?)`,
|
||||
artikalID, stara, novaCena, razlog, izvorIDArg(korisnikID), sada)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo.PromeniCenu: upis nivelacije: %w", err)
|
||||
}
|
||||
id, _ := rez.LastInsertId()
|
||||
|
||||
if err = tx.Commit(); err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo.PromeniCenu: commit: %w", err)
|
||||
}
|
||||
|
||||
return &model.Nivelacija{
|
||||
ID: id, ArtikalID: artikalID, StaraCena: stara, NovaCena: novaCena,
|
||||
Razlog: razlog, Izvor: "rucno", KorisnikID: korisnikID, Datum: sada,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// Kreiraj upisuje gotov nivelacioni zapis (npr. auto-trag pri izmeni artikla).
|
||||
func (r *NivelacijaRepo) Kreiraj(ctx context.Context, n *model.Nivelacija) (int64, error) {
|
||||
izvor := n.Izvor
|
||||
if izvor == "" {
|
||||
izvor = "rucno"
|
||||
}
|
||||
datum := n.Datum
|
||||
if datum.IsZero() {
|
||||
datum = time.Now()
|
||||
}
|
||||
rez, err := r.db.ExecContext(ctx, `
|
||||
INSERT INTO nivelacije (artikal_id, stara_cena, nova_cena, razlog, izvor, korisnik_id, datum)
|
||||
VALUES (?, ?, ?, ?, ?, ?, ?)`,
|
||||
n.ArtikalID, n.StaraCena, n.NovaCena, n.Razlog, izvor, izvorIDArg(n.KorisnikID), datum)
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: NivelacijaRepo.Kreiraj: %w", err)
|
||||
}
|
||||
id, err := rez.LastInsertId()
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: NivelacijaRepo.Kreiraj: id: %w", err)
|
||||
}
|
||||
return id, nil
|
||||
}
|
||||
|
||||
// Lista vraća nivelacije u periodu (po datumu); nulti datum znači bez granice.
|
||||
func (r *NivelacijaRepo) Lista(ctx context.Context, od, do time.Time) ([]model.Nivelacija, error) {
|
||||
upit := nivelacijaSelect + " WHERE 1=1"
|
||||
args := []any{}
|
||||
if !od.IsZero() {
|
||||
upit += " AND n.datum >= ?"
|
||||
args = append(args, od)
|
||||
}
|
||||
if !do.IsZero() {
|
||||
upit += " AND n.datum <= ?"
|
||||
args = append(args, do)
|
||||
}
|
||||
upit += " ORDER BY n.datum DESC, n.id DESC"
|
||||
return r.upitVisestruko(ctx, upit, args...)
|
||||
}
|
||||
|
||||
// ListaZaArtikal vraća sve nivelacije jednog artikla (najnovije prvo).
|
||||
func (r *NivelacijaRepo) ListaZaArtikal(ctx context.Context, artikalID int64) ([]model.Nivelacija, error) {
|
||||
upit := nivelacijaSelect + " WHERE n.artikal_id = ? ORDER BY n.datum DESC, n.id DESC"
|
||||
return r.upitVisestruko(ctx, upit, artikalID)
|
||||
}
|
||||
|
||||
// nivelacijaSelect je zajednički SELECT sa JOIN-ovima na artikle i korisnike (za prikaz).
|
||||
const nivelacijaSelect = `
|
||||
SELECT n.id, n.artikal_id, COALESCE(a.naziv, ''), n.stara_cena, n.nova_cena,
|
||||
COALESCE(n.razlog, ''), n.izvor, n.korisnik_id, COALESCE(k.korisnicko_ime, ''),
|
||||
n.datum, n.datum_unosa
|
||||
FROM nivelacije n
|
||||
LEFT JOIN artikli a ON a.id = n.artikal_id
|
||||
LEFT JOIN korisnici k ON k.id = n.korisnik_id`
|
||||
|
||||
// upitVisestruko izvršava SELECT i skenira sve redove u listu nivelacija.
|
||||
func (r *NivelacijaRepo) upitVisestruko(ctx context.Context, upit string, args ...any) ([]model.Nivelacija, error) {
|
||||
redovi, err := r.db.QueryContext(ctx, upit, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo: %w", err)
|
||||
}
|
||||
defer redovi.Close()
|
||||
|
||||
var rezultat []model.Nivelacija
|
||||
for redovi.Next() {
|
||||
var n model.Nivelacija
|
||||
var korisnikID sql.NullInt64
|
||||
if err := redovi.Scan(
|
||||
&n.ID, &n.ArtikalID, &n.ArtikalNaziv, &n.StaraCena, &n.NovaCena,
|
||||
&n.Razlog, &n.Izvor, &korisnikID, &n.KorisnikIme, &n.Datum, &n.DatumUnosa,
|
||||
); err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo: scan: %w", err)
|
||||
}
|
||||
if korisnikID.Valid {
|
||||
n.KorisnikID = &korisnikID.Int64
|
||||
}
|
||||
rezultat = append(rezultat, n)
|
||||
}
|
||||
if err := redovi.Err(); err != nil {
|
||||
return nil, fmt.Errorf("ntech: NivelacijaRepo: %w", err)
|
||||
}
|
||||
return rezultat, nil
|
||||
}
|
||||
@@ -39,6 +39,7 @@ type Handler struct {
|
||||
PdvStopeRepo db.PdvStopaRepository
|
||||
PdvKirRepo db.PdvKirRepository
|
||||
PdvKprRepo db.PdvKprRepository
|
||||
NivelacijaRepo db.NivelacijaRepository
|
||||
Verzija string
|
||||
AssetV string // verzija statičkih fajlova za cache-busting (postavlja se pri pokretanju)
|
||||
Templates map[string]*template.Template
|
||||
@@ -101,6 +102,7 @@ func Novi(baza *sql.DB, totpKljuc []byte) *Handler {
|
||||
PdvStopeRepo: sqlite.NoviPdvStopaRepo(baza),
|
||||
PdvKirRepo: sqlite.NoviPdvKirRepo(baza),
|
||||
PdvKprRepo: sqlite.NoviPdvKprRepo(baza),
|
||||
NivelacijaRepo: sqlite.NoviNivelacijaRepo(baza),
|
||||
}
|
||||
}
|
||||
|
||||
@@ -129,6 +131,7 @@ func (h *Handler) reinicijalizujRepozitorijume(novaDB *sql.DB) {
|
||||
h.PdvStopeRepo = sqlite.NoviPdvStopaRepo(novaDB)
|
||||
h.PdvKirRepo = sqlite.NoviPdvKirRepo(novaDB)
|
||||
h.PdvKprRepo = sqlite.NoviPdvKprRepo(novaDB)
|
||||
h.NivelacijaRepo = sqlite.NoviNivelacijaRepo(novaDB)
|
||||
}
|
||||
|
||||
// modulUkljucen vraća da li je zakonski modul (npr. „pdv") uključen za firmu prema profilu.
|
||||
|
||||
@@ -29,6 +29,7 @@ var saSidebar = []string{
|
||||
"pdv_kir", "pdv_kir_forma",
|
||||
"pdv_kpr", "pdv_kpr_forma",
|
||||
"pdv_obracun",
|
||||
"nivelacije",
|
||||
"podsetnici", "podsetnik_forma",
|
||||
"profil_tema",
|
||||
"prodaja", "prodaja_detalji", "prodaja_forma",
|
||||
|
||||
@@ -2,6 +2,7 @@ package handler
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
"strconv"
|
||||
|
||||
@@ -182,12 +183,32 @@ func (h *Handler) SacuvajIzmenuArtikla(w http.ResponseWriter, r *http.Request) {
|
||||
return
|
||||
}
|
||||
|
||||
// stara prodajna cena — za nivelacioni trag ako se promeni kroz izmenu
|
||||
var staraCena float64
|
||||
if stari, e := h.Artikli.DohvatiID(r.Context(), id); e == nil {
|
||||
staraCena = stari.ProdajnaCena
|
||||
}
|
||||
|
||||
artikal.ID = id
|
||||
if err := h.Artikli.Izmeni(r.Context(), &artikal); err != nil {
|
||||
http.Error(w, "Greška pri čuvanju izmene", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
// ako se prodajna cena promenila, automatski upiši nivelacioni zapis (izvor "izmena")
|
||||
if razlika := artikal.ProdajnaCena - staraCena; razlika > 0.005 || razlika < -0.005 {
|
||||
korisnikID := &k.ID
|
||||
if _, e := h.NivelacijaRepo.Kreiraj(r.Context(), &model.Nivelacija{
|
||||
ArtikalID: id,
|
||||
StaraCena: staraCena,
|
||||
NovaCena: artikal.ProdajnaCena,
|
||||
Izvor: "izmena",
|
||||
KorisnikID: korisnikID,
|
||||
}); e != nil {
|
||||
slog.Error("auto-nivelacija pri izmeni artikla nije upisana", "artikal_id", id, "error", e)
|
||||
}
|
||||
}
|
||||
|
||||
http.Redirect(w, r, "/magacin?sacuvano=1", http.StatusSeeOther)
|
||||
}
|
||||
|
||||
|
||||
@@ -30,6 +30,7 @@ type PodaciFormeNabavke struct {
|
||||
ArtikliJSON template.JS // JSON niz artikala za Alpine.js — bezbedan za umetanje u <script>
|
||||
Dobavljaci []model.Dobavljac
|
||||
Kategorije []model.Kategorija // za dropdown u modalu novog artikla
|
||||
Marza string // podrazumevana marža (%) za kalkulaciju
|
||||
Greska string
|
||||
}
|
||||
|
||||
@@ -44,12 +45,13 @@ type PodaciDetaljiNabavke struct {
|
||||
// artikalUJSON pretvara listu artikala u template.JS vrednost bezbednu za umetanje u <script> tag
|
||||
func artikalUJSON(artikli []model.ArtikalSaKategorijom) template.JS {
|
||||
type stavka struct {
|
||||
ID int64 `json:"id"`
|
||||
Naziv string `json:"naziv"`
|
||||
ID int64 `json:"id"`
|
||||
Naziv string `json:"naziv"`
|
||||
PdvStopa float64 `json:"pdv_stopa"`
|
||||
}
|
||||
lista := make([]stavka, 0, len(artikli))
|
||||
for _, a := range artikli {
|
||||
lista = append(lista, stavka{ID: a.ID, Naziv: a.Naziv})
|
||||
lista = append(lista, stavka{ID: a.ID, Naziv: a.Naziv, PdvStopa: a.PdvStopa})
|
||||
}
|
||||
b, _ := json.Marshal(lista)
|
||||
return template.JS(b)
|
||||
@@ -117,12 +119,14 @@ func (h *Handler) NovaNabavka(w http.ResponseWriter, r *http.Request) {
|
||||
ArtikliJSON: artikalUJSON(artikli),
|
||||
Dobavljaci: dobavljaci,
|
||||
Kategorije: kategorije,
|
||||
Marza: vrednostIliDefault(podesavanja, "kalkulacija_marza", "20"),
|
||||
})
|
||||
}
|
||||
|
||||
// SacuvajNabavku prima POST formu, parsira stavke i upisuje nabavku u bazu
|
||||
func (h *Handler) SacuvajNabavku(w http.ResponseWriter, r *http.Request) {
|
||||
if _, ok := h.zahtevajDozvolu(w, r, "nabavka.dodaj"); !ok {
|
||||
k, ok := h.zahtevajDozvolu(w, r, "nabavka.dodaj")
|
||||
if !ok {
|
||||
return
|
||||
}
|
||||
if err := r.ParseForm(); err != nil {
|
||||
@@ -145,6 +149,7 @@ func (h *Handler) SacuvajNabavku(w http.ResponseWriter, r *http.Request) {
|
||||
ArtikliJSON: artikalUJSON(artikli),
|
||||
Dobavljaci: dobavljaci,
|
||||
Kategorije: kategorije,
|
||||
Marza: vrednostIliDefault(podesavanja, "kalkulacija_marza", "20"),
|
||||
Greska: greska,
|
||||
})
|
||||
return
|
||||
@@ -182,6 +187,43 @@ func (h *Handler) SacuvajNabavku(w http.ResponseWriter, r *http.Request) {
|
||||
}
|
||||
}
|
||||
|
||||
// kalkulacija: ažuriraj nabavnu i prodajnu cenu artikla iz forme + nivelacioni trag.
|
||||
// prodajna[] je paralelni niz uz stavke (isti redosled kao artikal_id[]).
|
||||
prodajne := r.Form["prodajna[]"]
|
||||
var korisnikID *int64
|
||||
if k != nil {
|
||||
korisnikID = &k.ID
|
||||
}
|
||||
for i, s := range stavke {
|
||||
if i >= len(prodajne) {
|
||||
break
|
||||
}
|
||||
prodajna, e := strconv.ParseFloat(strings.TrimSpace(prodajne[i]), 64)
|
||||
if e != nil || prodajna <= 0 {
|
||||
continue // prazno/nula ne sme da pregazi postojeću cenu
|
||||
}
|
||||
// stara prodajna cena — za nivelacioni zapis
|
||||
var staraProdajna float64
|
||||
if a, e := h.Artikli.DohvatiID(r.Context(), s.ArtikalID); e == nil {
|
||||
staraProdajna = a.ProdajnaCena
|
||||
}
|
||||
if e := h.Artikli.AzurirajCene(r.Context(), s.ArtikalID, s.CenaPoKomadu, prodajna); e != nil {
|
||||
slog.Error("kalkulacija: ažuriranje cena nije uspelo", "artikal_id", s.ArtikalID, "error", e)
|
||||
continue
|
||||
}
|
||||
if razlika := prodajna - staraProdajna; razlika > 0.005 || razlika < -0.005 {
|
||||
if _, e := h.NivelacijaRepo.Kreiraj(r.Context(), &model.Nivelacija{
|
||||
ArtikalID: s.ArtikalID,
|
||||
StaraCena: staraProdajna,
|
||||
NovaCena: prodajna,
|
||||
Izvor: "kalkulacija",
|
||||
KorisnikID: korisnikID,
|
||||
}); e != nil {
|
||||
slog.Error("kalkulacija: nivelacija nije upisana", "artikal_id", s.ArtikalID, "error", e)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
http.Redirect(w, r, "/nabavke/"+strconv.FormatInt(id, 10)+"?sacuvano=1", http.StatusSeeOther)
|
||||
}
|
||||
|
||||
|
||||
@@ -0,0 +1,89 @@
|
||||
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)
|
||||
}
|
||||
@@ -42,6 +42,7 @@ type PodaciPodesavanja struct {
|
||||
Backupi []BackupInfo
|
||||
BackupIntervalSati string
|
||||
BackupBrojKopija string
|
||||
KalkulacijaMarza string
|
||||
LoginPozadina string
|
||||
LoginPozadinaOpacity string
|
||||
LoginPozadinaBlurPozadine string
|
||||
@@ -288,6 +289,20 @@ func (h *Handler) SacuvajPodesavanja(w http.ResponseWriter, r *http.Request) {
|
||||
}
|
||||
}
|
||||
|
||||
// podrazumevana marža za kalkulaciju (procenat, 0–1000)
|
||||
if v := strings.TrimSpace(r.FormValue("kalkulacija_marza")); v != "" {
|
||||
marza, err := strconv.ParseFloat(strings.Replace(v, ",", ".", 1), 64)
|
||||
if err != nil || marza < 0 || marza > 1000 {
|
||||
middleware.SetFlash(w, r, h.DB, "greska", "Marža mora biti broj između 0 i 1000.")
|
||||
http.Redirect(w, r, sledeci, http.StatusSeeOther)
|
||||
return
|
||||
}
|
||||
if err := ntechsqlite.SacuvajPodesavanje(r.Context(), h.DB, "kalkulacija_marza", strconv.FormatFloat(marza, 'f', -1, 64)); err != nil {
|
||||
http.Error(w, "Greška pri čuvanju podešavanja", http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
http.Redirect(w, r, sledeci+"?sacuvano=1", http.StatusSeeOther)
|
||||
}
|
||||
|
||||
@@ -627,6 +642,7 @@ func (h *Handler) napuniPodaciPodesavanja(r *http.Request, naslov string) (Podac
|
||||
LoginPozadinaZatamnjenjeKartice: vrednostIliDefault(podesavanja, "login_pozadina_zatamnjenje_kartice", "0"),
|
||||
BackupIntervalSati: vrednostIliDefault(podesavanja, "backup_interval_sati", "24"),
|
||||
BackupBrojKopija: vrednostIliDefault(podesavanja, "backup_broj_kopija", "7"),
|
||||
KalkulacijaMarza: vrednostIliDefault(podesavanja, "kalkulacija_marza", "20"),
|
||||
}, nil
|
||||
}
|
||||
|
||||
|
||||
@@ -0,0 +1,37 @@
|
||||
package model
|
||||
|
||||
import "time"
|
||||
|
||||
// Nivelacija je zapis o promeni prodajne cene artikla (revizioni trag).
|
||||
// Razlika i procenat se izvode iz stare i nove cene (ne čuvaju se u bazi).
|
||||
type Nivelacija struct {
|
||||
ID int64
|
||||
ArtikalID int64
|
||||
ArtikalNaziv string // iz JOIN-a, radi prikaza; nije kolona u nivelacije
|
||||
StaraCena float64
|
||||
NovaCena float64
|
||||
Razlog string
|
||||
Izvor string // "rucno" | "izmena" | "kalkulacija"
|
||||
KorisnikID *int64
|
||||
KorisnikIme string // iz JOIN-a, radi prikaza
|
||||
Datum time.Time
|
||||
DatumUnosa time.Time
|
||||
}
|
||||
|
||||
// Razlika vraća apsolutnu promenu cene (nova − stara); negativna znači pojeftinjenje.
|
||||
func (n Nivelacija) Razlika() float64 {
|
||||
return n.NovaCena - n.StaraCena
|
||||
}
|
||||
|
||||
// Procenat vraća procentualnu promenu u odnosu na staru cenu (0 ako je stara cena 0).
|
||||
func (n Nivelacija) Procenat() float64 {
|
||||
if n.StaraCena == 0 {
|
||||
return 0
|
||||
}
|
||||
return (n.NovaCena - n.StaraCena) / n.StaraCena * 100
|
||||
}
|
||||
|
||||
// Poskupljenje vraća true kada je nova cena veća od stare.
|
||||
func (n Nivelacija) Poskupljenje() bool {
|
||||
return n.NovaCena > n.StaraCena
|
||||
}
|
||||
@@ -0,0 +1,23 @@
|
||||
package model
|
||||
|
||||
import "testing"
|
||||
|
||||
func TestNivelacija(t *testing.T) {
|
||||
// poskupljenje: 100 → 120 = +20, +20%
|
||||
n := Nivelacija{StaraCena: 100, NovaCena: 120}
|
||||
if !blizu(n.Razlika(), 20) || !blizu(n.Procenat(), 20) || !n.Poskupljenje() {
|
||||
t.Errorf("razlika=%v procenat=%v poskupljenje=%v, očekivano 20/20/true", n.Razlika(), n.Procenat(), n.Poskupljenje())
|
||||
}
|
||||
|
||||
// pojeftinjenje: 200 → 150 = -50, -25%
|
||||
n2 := Nivelacija{StaraCena: 200, NovaCena: 150}
|
||||
if !blizu(n2.Razlika(), -50) || !blizu(n2.Procenat(), -25) || n2.Poskupljenje() {
|
||||
t.Errorf("razlika=%v procenat=%v poskupljenje=%v, očekivano -50/-25/false", n2.Razlika(), n2.Procenat(), n2.Poskupljenje())
|
||||
}
|
||||
|
||||
// stara cena 0 → procenat 0 (bez deljenja nulom)
|
||||
n3 := Nivelacija{StaraCena: 0, NovaCena: 80}
|
||||
if !blizu(n3.Procenat(), 0) {
|
||||
t.Errorf("procenat=%v, očekivano 0 kada je stara cena 0", n3.Procenat())
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,17 @@
|
||||
-- Nivelacije (Faza A kalkulacije/nivelacije): trag svake promene prodajne cene artikla.
|
||||
-- Razlika (nova − stara) se izvodi u programu, ne čuva se. Izvor razlikuje način promene:
|
||||
-- 'rucno' (posebna akcija sa razlogom), 'izmena' (auto pri izmeni artikla), 'kalkulacija' (kasnije).
|
||||
CREATE TABLE IF NOT EXISTS nivelacije (
|
||||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
artikal_id INTEGER NOT NULL REFERENCES artikli(id) ON DELETE CASCADE,
|
||||
stara_cena REAL NOT NULL,
|
||||
nova_cena REAL NOT NULL,
|
||||
razlog TEXT,
|
||||
izvor TEXT NOT NULL DEFAULT 'rucno',
|
||||
korisnik_id INTEGER REFERENCES korisnici(id) ON DELETE SET NULL,
|
||||
datum DATE NOT NULL, -- poslovni datum promene
|
||||
datum_unosa DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
|
||||
CREATE INDEX IF NOT EXISTS idx_nivelacije_artikal ON nivelacije(artikal_id);
|
||||
CREATE INDEX IF NOT EXISTS idx_nivelacije_datum ON nivelacije(datum);
|
||||
@@ -327,7 +327,7 @@ body {
|
||||
}
|
||||
|
||||
.kartica:hover {
|
||||
transform: translateY(-4px);
|
||||
transform: scale(1.02);
|
||||
box-shadow: var(--senka);
|
||||
}
|
||||
|
||||
|
||||
+17
-2
@@ -186,8 +186,9 @@ document.addEventListener('alpine:init', () => {
|
||||
|
||||
// forma za nabavku
|
||||
Alpine.data('nabavkaForma', () => ({
|
||||
stavke: [{artikal_id: '', kolicina: 1, cena: 0}],
|
||||
stavke: [{artikal_id: '', kolicina: 1, cena: 0, marza: 0, prodajna: 0}],
|
||||
artikliOpcije: [],
|
||||
marzaDefault: 0,
|
||||
isMobile: false,
|
||||
modal: false,
|
||||
modalUcitavanje: false,
|
||||
@@ -212,13 +213,27 @@ document.addEventListener('alpine:init', () => {
|
||||
modalDobNapomena: '',
|
||||
init() {
|
||||
this.artikliOpcije = window._ntechArtikli || []
|
||||
this.marzaDefault = parseFloat(window._ntechMarza) || 0
|
||||
this.stavke.forEach(s => { s.marza = this.marzaDefault })
|
||||
this.isMobile = window.matchMedia('(max-width: 768px)').matches
|
||||
window.matchMedia('(max-width: 768px)').addEventListener('change', e => {
|
||||
this.isMobile = e.matches
|
||||
})
|
||||
},
|
||||
dodajStavku() {
|
||||
this.stavke.push({artikal_id: '', kolicina: 1, cena: 0})
|
||||
this.stavke.push({artikal_id: '', kolicina: 1, cena: 0, marza: this.marzaDefault, prodajna: 0})
|
||||
},
|
||||
// PDV stopa izabranog artikla (iz JSON liste) — za obračun prodajne cene
|
||||
pdvStopa(artikalId) {
|
||||
const a = this.artikliOpcije.find(x => String(x.id) === String(artikalId))
|
||||
return a ? (parseFloat(a.pdv_stopa) || 0) : 0
|
||||
},
|
||||
// prodajna (sa PDV) = nabavna × (1 + marža/100) × (1 + pdvStopa/100), zaokruženo na 2 decimale
|
||||
izracunajProdajnu(s) {
|
||||
const cena = parseFloat(s.cena) || 0
|
||||
const marza = parseFloat(s.marza) || 0
|
||||
const pdv = this.pdvStopa(s.artikal_id)
|
||||
s.prodajna = Math.round(cena * (1 + marza / 100) * (1 + pdv / 100) * 100) / 100
|
||||
},
|
||||
ukloniStavku(i) {
|
||||
if (this.stavke.length > 1) this.stavke.splice(i, 1)
|
||||
|
||||
@@ -93,23 +93,52 @@
|
||||
</a>
|
||||
{{end}}
|
||||
|
||||
{{/* PDV evidencija — vidljivo samo kada je modul „pdv" uključen za firmu (profil firme) i korisnik ima dozvolu */}}
|
||||
{{if and (index .Moduli "pdv") (index .Dozvole "pdv.pregled")}}
|
||||
<a href="/pdv/kir" class="nav-stavka {{if eq .Stranica "pdv-kir"}}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="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/></svg>
|
||||
<span>KIR</span>
|
||||
<span class="nav-tooltip">KIR — knjiga izdatih računa</span>
|
||||
</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>
|
||||
<a href="/pdv/obracun" class="nav-stavka {{if eq .Stranica "pdv-obracun"}}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"><rect x="4" y="2" width="16" height="20" rx="2"/><line x1="8" y1="6" x2="16" y2="6"/><line x1="8" y1="10" x2="10" y2="10"/><line x1="14" y1="10" x2="16" y2="10"/><line x1="8" y1="14" x2="10" y2="14"/><line x1="14" y1="14" x2="16" y2="14"/><line x1="8" y1="18" x2="16" y2="18"/></svg>
|
||||
<span>Obračun</span>
|
||||
<span class="nav-tooltip">PDV obračun za period</span>
|
||||
</a>
|
||||
{{/* Knjigovodstvo — grupa: KIR/KPR/Obračun (modul „pdv") + Nivelacije (dozvola izmene cena) */}}
|
||||
{{$pdvVid := and (index .Moduli "pdv") (index .Dozvole "pdv.pregled")}}
|
||||
{{$knjVid := or $pdvVid (index .Dozvole "artikal.izmeni")}}
|
||||
{{if $knjVid}}
|
||||
{{$knjAktivan := or (eq .Stranica "nivelacije") (eq .Stranica "pdv-kir") (eq .Stranica "pdv-kpr") (eq .Stranica "pdv-obracun")}}
|
||||
<div>
|
||||
<button type="button" data-podmeni-dugme
|
||||
class="nav-stavka {{if $knjAktivan}}aktivan{{end}}"
|
||||
style="width:100%;background:none;border:none;cursor:pointer;">
|
||||
<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="M4 19.5A2.5 2.5 0 0 1 6.5 17H20"/><path d="M6.5 2H20v20H6.5A2.5 2.5 0 0 1 4 19.5v-15A2.5 2.5 0 0 1 6.5 2z"/></svg>
|
||||
<span>Knjigovodstvo</span>
|
||||
<span class="nav-strelica">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"
|
||||
style="transition:transform 0.2s;transform:{{if $knjAktivan}}rotate(180deg){{else}}rotate(0deg){{end}}">
|
||||
<polyline points="6 9 12 15 18 9"/>
|
||||
</svg>
|
||||
</span>
|
||||
<span class="nav-tooltip">Knjigovodstvo</span>
|
||||
</button>
|
||||
<div class="nav-podmeni {{if $knjAktivan}}otvoren{{end}}">
|
||||
{{if index .Dozvole "artikal.izmeni"}}
|
||||
<a href="/nivelacije" class="nav-stavka nav-podstavka {{if eq .Stranica "nivelacije"}}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"><line x1="12" y1="20" x2="12" y2="10"/><line x1="18" y1="20" x2="18" y2="4"/><line x1="6" y1="20" x2="6" y2="16"/></svg>
|
||||
<span>Nivelacije</span>
|
||||
<span class="nav-tooltip">Nivelacije — promene cena</span>
|
||||
</a>
|
||||
{{end}}
|
||||
{{if $pdvVid}}
|
||||
<a href="/pdv/kir" class="nav-stavka nav-podstavka {{if eq .Stranica "pdv-kir"}}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="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/></svg>
|
||||
<span>KIR</span>
|
||||
<span class="nav-tooltip">KIR — knjiga izdatih računa</span>
|
||||
</a>
|
||||
<a href="/pdv/kpr" class="nav-stavka nav-podstavka {{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>
|
||||
<a href="/pdv/obracun" class="nav-stavka nav-podstavka {{if eq .Stranica "pdv-obracun"}}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"><rect x="4" y="2" width="16" height="20" rx="2"/><line x1="8" y1="6" x2="16" y2="6"/><line x1="8" y1="10" x2="10" y2="10"/><line x1="14" y1="10" x2="16" y2="10"/><line x1="8" y1="14" x2="10" y2="14"/><line x1="14" y1="14" x2="16" y2="14"/><line x1="8" y1="18" x2="16" y2="18"/></svg>
|
||||
<span>Obračun</span>
|
||||
<span class="nav-tooltip">PDV obračun za period</span>
|
||||
</a>
|
||||
{{end}}
|
||||
</div>
|
||||
</div>
|
||||
{{end}}
|
||||
|
||||
<div class="nav-separator"></div>
|
||||
|
||||
@@ -83,6 +83,7 @@
|
||||
<a href="/magacin/izmeni/{{.ID}}" class="btn-primarno-malo">
|
||||
Izmeni
|
||||
</a>
|
||||
{{template "promeniCenuMeni" (dict "ID" .ID "Cena" .ProdajnaCena)}}
|
||||
{{end}}
|
||||
{{if index $.Dozvole "artikal.premesti"}}{{if $.Kategorije}}
|
||||
{{template "premestiMeni" (dict "ID" .ID "Kategorije" $.Kategorije)}}
|
||||
@@ -112,7 +113,7 @@
|
||||
<div class="magacin-kartice">
|
||||
{{range .Artikli}}
|
||||
<div class="kartica magacin-kartica animiraj">
|
||||
<div class="red-izmedju" style="align-items:flex-start;gap:12px;margin-bottom:10px;">
|
||||
<div class="red-izmedju" style="align-items:flex-start;gap:12px;margin-bottom:10px;flex-wrap:wrap;">
|
||||
<div>
|
||||
<div style="font-size:15px;font-weight:500;color:var(--tekst-glavni);">{{.Naziv}}</div>
|
||||
{{if .KategorijaNaziv}}
|
||||
@@ -122,6 +123,7 @@
|
||||
<div style="display:flex;gap:8px;flex-shrink:0;flex-wrap:wrap;justify-content:flex-end;">
|
||||
{{if index $.Dozvole "artikal.izmeni"}}
|
||||
<a href="/magacin/izmeni/{{.ID}}" class="btn-primarno-malo">Izmeni</a>
|
||||
{{template "promeniCenuMeni" (dict "ID" .ID "Cena" .ProdajnaCena)}}
|
||||
{{end}}
|
||||
{{if index $.Dozvole "artikal.premesti"}}{{if $.Kategorije}}
|
||||
{{template "premestiMeni" (dict "ID" .ID "Kategorije" $.Kategorije)}}
|
||||
@@ -162,7 +164,7 @@
|
||||
{{/* padajući meni za premeštanje artikla — prima dict {ID, Kategorije}; koristi se i u tabeli i u mobilnoj kartici */}}
|
||||
{{define "premestiMeni"}}
|
||||
<button type="button" class="btn-primarno-malo" style="align-self:center;"
|
||||
onclick="document.getElementById('premesti-{{.ID}}').showModal()">Premesti</button>
|
||||
onclick="this.nextElementSibling.showModal()">Premesti</button>
|
||||
{{/* nativni modal — showModal() ga stavlja u „top layer", pa je uvek iznad svega bez obzira na z-index/overflow */}}
|
||||
<dialog id="premesti-{{.ID}}" class="premesti-modal" onclick="if(event.target===this)this.close()">
|
||||
{{/* zaglavlje sa dugmetom za zatvaranje; method="dialog" zatvara modal bez slanja */}}
|
||||
@@ -178,3 +180,29 @@
|
||||
</form>
|
||||
</dialog>
|
||||
{{end}}
|
||||
|
||||
{{define "promeniCenuMeni"}}
|
||||
<button type="button" class="btn-primarno-malo" style="align-self:center;"
|
||||
onclick="this.nextElementSibling.showModal()">Promeni cenu</button>
|
||||
{{/* nativni modal — isti obrazac kao premesti (top layer, centriran) */}}
|
||||
<dialog id="cena-{{.ID}}" class="premesti-modal" onclick="if(event.target===this)this.close()">
|
||||
<form method="dialog" class="premesti-zaglavlje">
|
||||
<h3>Promeni prodajnu cenu</h3>
|
||||
<button type="submit" class="premesti-zatvori" aria-label="Zatvori">×</button>
|
||||
</form>
|
||||
<form method="POST" action="/magacin/promeni-cenu/{{.ID}}" style="display:flex;flex-direction:column;gap:12px;padding:16px;">
|
||||
<div style="font-size:13px;color:var(--tekst-sporedni);">Trenutna cena: <strong>{{printf "%.0f" .Cena}} din</strong></div>
|
||||
<div>
|
||||
<label class="polje-labela">Nova cena (din)</label>
|
||||
<input type="number" name="nova_cena" min="0" step="0.01" value="{{printf "%.2f" .Cena}}" 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">Razlog (opciono)</label>
|
||||
<input type="text" name="razlog" placeholder="npr. promena nabavne cene"
|
||||
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>
|
||||
<button type="submit" class="btn-primarno" style="align-self:flex-end;">Sačuvaj cenu</button>
|
||||
</form>
|
||||
</dialog>
|
||||
{{end}}
|
||||
|
||||
@@ -16,7 +16,7 @@
|
||||
{{define "sadrzaj"}}
|
||||
|
||||
<!-- lista artikala kao JSON — bezbedno serijalizovana na serveru -->
|
||||
<script>var _ntechArtikli = {{.ArtikliJSON}};</script>
|
||||
<script>var _ntechArtikli = {{.ArtikliJSON}}; var _ntechMarza = {{.Marza}};</script>
|
||||
|
||||
<div style="width:100%;" x-data="nabavkaForma">
|
||||
|
||||
@@ -87,6 +87,8 @@
|
||||
<th style="padding:8px 10px;text-align:left;font-size:12px;font-weight:500;color:var(--tekst-sporedni);">Artikal</th>
|
||||
<th style="padding:8px 10px;text-align:center;font-size:12px;font-weight:500;color:var(--tekst-sporedni);width:90px;">Količina</th>
|
||||
<th style="padding:8px 10px;text-align:right;font-size:12px;font-weight:500;color:var(--tekst-sporedni);width:130px;">Cena/kom (din)</th>
|
||||
<th style="padding:8px 10px;text-align:right;font-size:12px;font-weight:500;color:var(--tekst-sporedni);width:90px;">Marža %</th>
|
||||
<th style="padding:8px 10px;text-align:right;font-size:12px;font-weight:500;color:var(--tekst-sporedni);width:130px;">Prodajna/kom (din)</th>
|
||||
<th style="padding:8px 10px;text-align:right;font-size:12px;font-weight:500;color:var(--tekst-sporedni);width:110px;">Ukupno</th>
|
||||
<th style="width:40px;"></th>
|
||||
</tr>
|
||||
@@ -96,7 +98,7 @@
|
||||
<tr style="border-bottom:0.5px solid var(--ivica);">
|
||||
<td style="padding:8px 10px;">
|
||||
<select :name="'artikal_id[]'" x-model="stavka.artikal_id"
|
||||
:disabled="isMobile" style="width:100%;">
|
||||
@change="izracunajProdajnu(stavka)" :disabled="isMobile" style="width:100%;">
|
||||
<option value="">— odaberi artikal —</option>
|
||||
<template x-for="a in artikliOpcije" :key="a.id">
|
||||
<option :value="a.id" x-text="a.naziv"></option>
|
||||
@@ -109,6 +111,16 @@
|
||||
</td>
|
||||
<td style="padding:8px 10px;">
|
||||
<input type="number" :name="'cena_po_komadu[]'" x-model="stavka.cena"
|
||||
@input="izracunajProdajnu(stavka)"
|
||||
min="0" step="0.01" :disabled="isMobile" style="width:100%;text-align:right;">
|
||||
</td>
|
||||
<td style="padding:8px 10px;">
|
||||
<input type="number" :name="'marza[]'" x-model="stavka.marza"
|
||||
@input="izracunajProdajnu(stavka)"
|
||||
min="0" step="0.01" :disabled="isMobile" style="width:100%;text-align:right;">
|
||||
</td>
|
||||
<td style="padding:8px 10px;">
|
||||
<input type="number" :name="'prodajna[]'" x-model="stavka.prodajna"
|
||||
min="0" step="0.01" :disabled="isMobile" style="width:100%;text-align:right;">
|
||||
</td>
|
||||
<td style="padding:8px 10px;text-align:right;font-size:14px;font-weight:500;color:var(--tekst-glavni);">
|
||||
@@ -127,7 +139,7 @@
|
||||
</tbody>
|
||||
<tfoot>
|
||||
<tr style="border-top:0.5px solid var(--ivica);">
|
||||
<td colspan="3" style="padding:10px 10px;text-align:right;font-size:13px;color:var(--tekst-sporedni);font-weight:500;">Ukupno:</td>
|
||||
<td colspan="5" style="padding:10px 10px;text-align:right;font-size:13px;color:var(--tekst-sporedni);font-weight:500;">Ukupno:</td>
|
||||
<td style="padding:10px 10px;text-align:right;font-size:15px;font-weight:600;color:var(--tekst-glavni);">
|
||||
<span x-text="ukupnoSvega() + ' din'"></span>
|
||||
</td>
|
||||
@@ -154,7 +166,7 @@
|
||||
<div>
|
||||
<label style="font-size:12px;color:var(--tekst-sporedni);display:block;margin-bottom:4px;">Artikal</label>
|
||||
<select :name="'artikal_id[]'" x-model="stavka.artikal_id"
|
||||
:disabled="!isMobile" style="width:100%;">
|
||||
@change="izracunajProdajnu(stavka)" :disabled="!isMobile" style="width:100%;">
|
||||
<option value="">— odaberi artikal —</option>
|
||||
<template x-for="a in artikliOpcije" :key="a.id">
|
||||
<option :value="a.id" x-text="a.naziv"></option>
|
||||
@@ -168,7 +180,17 @@
|
||||
</div>
|
||||
<div>
|
||||
<label style="font-size:12px;color:var(--tekst-sporedni);display:block;margin-bottom:4px;">Cena/kom (din)</label>
|
||||
<input type="number" :name="'cena_po_komadu[]'" x-model="stavka.cena" min="0" step="0.01" :disabled="!isMobile" style="width:100%;">
|
||||
<input type="number" :name="'cena_po_komadu[]'" x-model="stavka.cena" @input="izracunajProdajnu(stavka)" min="0" step="0.01" :disabled="!isMobile" style="width:100%;">
|
||||
</div>
|
||||
</div>
|
||||
<div style="display:grid;grid-template-columns:1fr 1fr;gap:10px;">
|
||||
<div>
|
||||
<label style="font-size:12px;color:var(--tekst-sporedni);display:block;margin-bottom:4px;">Marža %</label>
|
||||
<input type="number" :name="'marza[]'" x-model="stavka.marza" @input="izracunajProdajnu(stavka)" min="0" step="0.01" :disabled="!isMobile" style="width:100%;">
|
||||
</div>
|
||||
<div>
|
||||
<label style="font-size:12px;color:var(--tekst-sporedni);display:block;margin-bottom:4px;">Prodajna/kom (din)</label>
|
||||
<input type="number" :name="'prodajna[]'" x-model="stavka.prodajna" min="0" step="0.01" :disabled="!isMobile" style="width:100%;">
|
||||
</div>
|
||||
</div>
|
||||
<div style="text-align:right;font-size:14px;font-weight:500;color:var(--tekst-glavni);">
|
||||
|
||||
@@ -0,0 +1,69 @@
|
||||
{{template "base" .}}
|
||||
|
||||
{{define "naslov"}}Nivelacije — promene cena — NTech{{end}}
|
||||
|
||||
{{define "sadrzaj"}}
|
||||
<div class="stranica-stack" style="width:100%;max-width:100%;">
|
||||
|
||||
<!-- filter perioda -->
|
||||
<div class="kartica animiraj" style="margin-bottom:16px;">
|
||||
<form method="GET" action="/nivelacije" 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="/nivelacije" class="nazad-link" style="margin-bottom:0;">Poništi filter</a>{{end}}
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<!-- istorija nivelacija -->
|
||||
<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:760px;">
|
||||
<thead>
|
||||
<tr style="text-align:left;color:var(--tekst-sporedni);border-bottom:0.5px solid var(--ivica);">
|
||||
<th style="padding:10px 12px;">Datum</th>
|
||||
<th style="padding:10px 12px;">Artikal</th>
|
||||
<th style="padding:10px 12px;text-align:right;">Stara cena</th>
|
||||
<th style="padding:10px 12px;text-align:right;">Nova cena</th>
|
||||
<th style="padding:10px 12px;text-align:right;">Razlika</th>
|
||||
<th style="padding:10px 12px;">Izvor</th>
|
||||
<th style="padding:10px 12px;">Razlog</th>
|
||||
<th style="padding:10px 12px;">Korisnik</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{{range .Zapisi}}
|
||||
<tr style="border-bottom:0.5px solid var(--ivica);">
|
||||
<td style="padding:10px 12px;white-space:nowrap;">{{.Datum.Format "02.01.2006."}}</td>
|
||||
<td style="padding:10px 12px;">{{.ArtikalNaziv}}</td>
|
||||
<td style="padding:10px 12px;text-align:right;color:var(--tekst-sporedni);">{{printf "%.2f" .StaraCena}}</td>
|
||||
<td style="padding:10px 12px;text-align:right;font-weight:500;">{{printf "%.2f" .NovaCena}}</td>
|
||||
<td style="padding:10px 12px;text-align:right;white-space:nowrap;color:{{if .Poskupljenje}}var(--greska){{else}}var(--uspeh){{end}};">
|
||||
{{if .Poskupljenje}}+{{end}}{{printf "%.2f" .Razlika}}
|
||||
<div style="font-size:11px;">{{if .Poskupljenje}}+{{end}}{{printf "%.1f" .Procenat}}%</div>
|
||||
</td>
|
||||
<td style="padding:10px 12px;">
|
||||
{{if eq .Izvor "rucno"}}<span title="Posebna akcija „Promeni cenu“">ručno</span>
|
||||
{{else if eq .Izvor "izmena"}}<span title="Promenjeno kroz izmenu artikla">izmena</span>
|
||||
{{else if eq .Izvor "kalkulacija"}}<span title="Iz kalkulacije pri prijemu robe">kalkulacija</span>
|
||||
{{else}}{{.Izvor}}{{end}}
|
||||
</td>
|
||||
<td style="padding:10px 12px;color:var(--tekst-sporedni);">{{if .Razlog}}{{.Razlog}}{{else}}—{{end}}</td>
|
||||
<td style="padding:10px 12px;color:var(--tekst-sporedni);">{{if .KorisnikIme}}{{.KorisnikIme}}{{else}}—{{end}}</td>
|
||||
</tr>
|
||||
{{else}}
|
||||
<tr><td colspan="8" style="padding:28px;text-align:center;color:var(--tekst-sporedni);">Nema zabeleženih promena cena u izabranom periodu.</td></tr>
|
||||
{{end}}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
{{end}}
|
||||
@@ -56,6 +56,27 @@
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<div style="margin-top:16px;border-top:0.5px solid var(--ivica);padding-top:16px;">
|
||||
<div style="font-size:13px;font-weight:500;color:var(--tekst-glavni);margin-bottom:10px;">Kalkulacija</div>
|
||||
<form method="POST" action="/podesavanja/sacuvaj">
|
||||
<input type="hidden" name="_next" value="/admin/podesavanja/sistem">
|
||||
<div style="display:flex;gap:16px;flex-wrap:wrap;align-items:flex-end;">
|
||||
<div>
|
||||
<label for="kalkulacija_marza" class="polje-labela">Podrazumevana marža (%)</label>
|
||||
<input type="number" id="kalkulacija_marza" name="kalkulacija_marza" min="0" max="1000" step="0.01" value="{{.KalkulacijaMarza}}"
|
||||
style="width:140px;padding:8px 12px;background:var(--pozadina);border:0.5px solid var(--ivica);border-radius:8px;color:var(--tekst-glavni);font-size:14px;">
|
||||
</div>
|
||||
<button type="submit"
|
||||
style="padding:9px 18px;background:var(--sb-akcent);border:none;border-radius:8px;color:#fff;font-size:13px;font-weight:500;cursor:pointer;">
|
||||
Sačuvaj
|
||||
</button>
|
||||
</div>
|
||||
<div style="font-size:12px;color:var(--tekst-sporedni);margin-top:8px;">
|
||||
Početna marža za formiranje prodajne cene pri nabavci. Po stavci je možeš promeniti.
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<!-- panel sa listom backupa -->
|
||||
<div id="backup-panel" style="display:none;margin-top:16px;border-top:0.5px solid var(--ivica);padding-top:16px;">
|
||||
<div style="font-size:13px;font-weight:500;color:var(--tekst-glavni);margin-bottom:10px;">Dostupne rezervne kopije</div>
|
||||
|
||||
Reference in New Issue
Block a user