feat(magacin): nivelacija — promena cene uz trag (Faza A)

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.
This commit is contained in:
2026-06-14 09:37:49 +02:00
parent c9d4704c3d
commit 0f1f65c7f7
13 changed files with 463 additions and 2 deletions
+2
View File
@@ -278,6 +278,8 @@ func main() {
r.With(doz("artikal.izmeni")).Post("/magacin/izmeni/{id}", h.SacuvajIzmenuArtikla) 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.obrisi")).Get("/magacin/obrisi/{id}", h.ObrisiArtikal)
r.With(doz("artikal.premesti")).Post("/magacin/premesti/{id}", h.PremestiArtikal) 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.Get("/magacin/kategorije", h.Kategorije)
r.With(doz("kategorija.dodaj")).Post("/magacin/kategorije/dodaj", h.DodajKategoriju) r.With(doz("kategorija.dodaj")).Post("/magacin/kategorije/dodaj", h.DodajKategoriju)
r.With(doz("kategorija.obrisi")).Get("/magacin/kategorije/obrisi/{id}", h.ObrisiKategoriju) r.With(doz("kategorija.obrisi")).Get("/magacin/kategorije/obrisi/{id}", h.ObrisiKategoriju)
+13
View File
@@ -52,6 +52,19 @@ type PdvKprRepository interface {
ObrisiPoIzvoru(ctx context.Context, izvor string, izvorID int64) error 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 // ArtikalFilter definiše parametre za filtriranje liste artikala
type ArtikalFilter struct { type ArtikalFilter struct {
Pretraga string Pretraga string
+150
View File
@@ -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
}
+3
View File
@@ -39,6 +39,7 @@ type Handler struct {
PdvStopeRepo db.PdvStopaRepository PdvStopeRepo db.PdvStopaRepository
PdvKirRepo db.PdvKirRepository PdvKirRepo db.PdvKirRepository
PdvKprRepo db.PdvKprRepository PdvKprRepo db.PdvKprRepository
NivelacijaRepo db.NivelacijaRepository
Verzija string Verzija string
AssetV string // verzija statičkih fajlova za cache-busting (postavlja se pri pokretanju) AssetV string // verzija statičkih fajlova za cache-busting (postavlja se pri pokretanju)
Templates map[string]*template.Template Templates map[string]*template.Template
@@ -101,6 +102,7 @@ func Novi(baza *sql.DB, totpKljuc []byte) *Handler {
PdvStopeRepo: sqlite.NoviPdvStopaRepo(baza), PdvStopeRepo: sqlite.NoviPdvStopaRepo(baza),
PdvKirRepo: sqlite.NoviPdvKirRepo(baza), PdvKirRepo: sqlite.NoviPdvKirRepo(baza),
PdvKprRepo: sqlite.NoviPdvKprRepo(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.PdvStopeRepo = sqlite.NoviPdvStopaRepo(novaDB)
h.PdvKirRepo = sqlite.NoviPdvKirRepo(novaDB) h.PdvKirRepo = sqlite.NoviPdvKirRepo(novaDB)
h.PdvKprRepo = sqlite.NoviPdvKprRepo(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. // modulUkljucen vraća da li je zakonski modul (npr. „pdv") uključen za firmu prema profilu.
+1
View File
@@ -29,6 +29,7 @@ var saSidebar = []string{
"pdv_kir", "pdv_kir_forma", "pdv_kir", "pdv_kir_forma",
"pdv_kpr", "pdv_kpr_forma", "pdv_kpr", "pdv_kpr_forma",
"pdv_obracun", "pdv_obracun",
"nivelacije",
"podsetnici", "podsetnik_forma", "podsetnici", "podsetnik_forma",
"profil_tema", "profil_tema",
"prodaja", "prodaja_detalji", "prodaja_forma", "prodaja", "prodaja_detalji", "prodaja_forma",
+21
View File
@@ -2,6 +2,7 @@ package handler
import ( import (
"fmt" "fmt"
"log/slog"
"net/http" "net/http"
"strconv" "strconv"
@@ -182,12 +183,32 @@ func (h *Handler) SacuvajIzmenuArtikla(w http.ResponseWriter, r *http.Request) {
return 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 artikal.ID = id
if err := h.Artikli.Izmeni(r.Context(), &artikal); err != nil { if err := h.Artikli.Izmeni(r.Context(), &artikal); err != nil {
http.Error(w, "Greška pri čuvanju izmene", http.StatusInternalServerError) http.Error(w, "Greška pri čuvanju izmene", http.StatusInternalServerError)
return 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) http.Redirect(w, r, "/magacin?sacuvano=1", http.StatusSeeOther)
} }
+89
View File
@@ -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)
}
+37
View File
@@ -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
}
+23
View File
@@ -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())
}
}
+17
View File
@@ -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);
+8
View File
@@ -39,6 +39,14 @@
<span class="nav-tooltip">Magacin</span> <span class="nav-tooltip">Magacin</span>
</a> </a>
{{if index .Dozvole "artikal.izmeni"}}
<a href="/nivelacije" class="nav-stavka {{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 index .Dozvole "nabavka.pregled"}} {{if index .Dozvole "nabavka.pregled"}}
<a href="/nabavke" class="nav-stavka {{if eq .Stranica "nabavke"}}aktivan{{end}}"> <a href="/nabavke" class="nav-stavka {{if eq .Stranica "nabavke"}}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"/><polyline points="10 9 9 9 8 9"/></svg> <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"/><polyline points="10 9 9 9 8 9"/></svg>
+30 -2
View File
@@ -83,6 +83,7 @@
<a href="/magacin/izmeni/{{.ID}}" class="btn-primarno-malo"> <a href="/magacin/izmeni/{{.ID}}" class="btn-primarno-malo">
Izmeni Izmeni
</a> </a>
{{template "promeniCenuMeni" (dict "ID" .ID "Cena" .ProdajnaCena)}}
{{end}} {{end}}
{{if index $.Dozvole "artikal.premesti"}}{{if $.Kategorije}} {{if index $.Dozvole "artikal.premesti"}}{{if $.Kategorije}}
{{template "premestiMeni" (dict "ID" .ID "Kategorije" $.Kategorije)}} {{template "premestiMeni" (dict "ID" .ID "Kategorije" $.Kategorije)}}
@@ -112,7 +113,7 @@
<div class="magacin-kartice"> <div class="magacin-kartice">
{{range .Artikli}} {{range .Artikli}}
<div class="kartica magacin-kartica animiraj"> <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>
<div style="font-size:15px;font-weight:500;color:var(--tekst-glavni);">{{.Naziv}}</div> <div style="font-size:15px;font-weight:500;color:var(--tekst-glavni);">{{.Naziv}}</div>
{{if .KategorijaNaziv}} {{if .KategorijaNaziv}}
@@ -122,6 +123,7 @@
<div style="display:flex;gap:8px;flex-shrink:0;flex-wrap:wrap;justify-content:flex-end;"> <div style="display:flex;gap:8px;flex-shrink:0;flex-wrap:wrap;justify-content:flex-end;">
{{if index $.Dozvole "artikal.izmeni"}} {{if index $.Dozvole "artikal.izmeni"}}
<a href="/magacin/izmeni/{{.ID}}" class="btn-primarno-malo">Izmeni</a> <a href="/magacin/izmeni/{{.ID}}" class="btn-primarno-malo">Izmeni</a>
{{template "promeniCenuMeni" (dict "ID" .ID "Cena" .ProdajnaCena)}}
{{end}} {{end}}
{{if index $.Dozvole "artikal.premesti"}}{{if $.Kategorije}} {{if index $.Dozvole "artikal.premesti"}}{{if $.Kategorije}}
{{template "premestiMeni" (dict "ID" .ID "Kategorije" $.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 */}} {{/* padajući meni za premeštanje artikla — prima dict {ID, Kategorije}; koristi se i u tabeli i u mobilnoj kartici */}}
{{define "premestiMeni"}} {{define "premestiMeni"}}
<button type="button" class="btn-primarno-malo" style="align-self:center;" <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 */}} {{/* 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()"> <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 */}} {{/* zaglavlje sa dugmetom za zatvaranje; method="dialog" zatvara modal bez slanja */}}
@@ -178,3 +180,29 @@
</form> </form>
</dialog> </dialog>
{{end}} {{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">&times;</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}}
+69
View File
@@ -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}}