Compare commits
	
		
			126 Commits
		
	
	
		
			dd8a818aa9
			...
			master
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 4128d617a7 | ||
|   | a51e44847e | ||
|   | 45a6ab7249 | ||
|   | a363fb9ef8 | ||
|   | 2c246ac40a | ||
|   | 43b7b93ffa | ||
|   | cabc2c6a4a | ||
|   | 226b10b5a1 | ||
|   | b24748a7b6 | ||
|   | 11065cd007 | ||
|   | 05d364bcd4 | ||
|   | 57a553037b | ||
|   | 5ed356a61c | ||
|   | 5da660b4c3 | ||
|   | d439002241 | ||
|   | 4246cde484 | ||
|   | a902205960 | ||
|   | 355b73775f | ||
|   | 81744b5c5e | ||
|   | 735fc69562 | ||
|   | 17a5fd2086 | ||
|   | 9986716e9e | ||
|   | 759c78ce87 | ||
|   | 365791cd35 | ||
|   | 08b680f030 | ||
|   | 4d6be819e1 | ||
|   | d803f49713 | ||
|   | 01114b4ca9 | ||
|   | 873e81d95d | ||
|   | d809dcb361 | ||
|   | fa017ce290 | ||
|   | c2cf310f89 | ||
| e1350d722c | |||
| af1019f01c | |||
|   | 3433d85471 | ||
|   | a8b3a14044 | ||
|   | c944cadff3 | ||
|   | 0a5debe45a | ||
|   | dbead3d719 | ||
|   | 34065bc288 | ||
|   | 6236657d9a | ||
|   | 68a7e07c58 | ||
|   | eca635a175 | ||
|   | bcdbc49aa8 | ||
|   | 419d01f74d | ||
|   | 9b131824e8 | ||
|   | 0286ee351e | ||
|   | ee59c3e561 | ||
|   | b9c3204db0 | ||
|   | 3324564160 | ||
|   | 7821f25b61 | ||
|   | 8e38576dbc | ||
|   | e118ac533d | ||
|   | 939f55d9aa | ||
|   | c34aad68f1 | ||
|   | c2c7adf950 | ||
|   | a5bf017c30 | ||
|   | a9f21dd4b9 | ||
|   | 4663445fb8 | ||
|   | 2d85991db0 | ||
|   | 69ecc26236 | ||
|   | 44c3f8eb5b | ||
|   | da882a9a24 | ||
|   | 06618b1e27 | ||
|   | 5fe052648d | ||
|   | fe213d4acd | ||
|   | 3a99d1a936 | ||
|   | 0f45ae94af | ||
|   | 11f89307eb | ||
|   | c9d5ab22c8 | ||
|   | ce74879d15 | ||
|   | 0120feff33 | ||
|   | 7eb29b271a | ||
|   | 2015065af4 | ||
|   | e7f6389ca3 | ||
|   | 767730831e | ||
|   | 556b1fd4b9 | ||
|   | 577ac3f463 | ||
|   | f2e99821f7 | ||
|   | 065f67c45e | ||
|   | e2761584a3 | ||
|   | e4a33ad6aa | ||
|   | cee5e31646 | ||
|   | b386364cd6 | ||
|   | 92bc3e59ae | ||
|   | 174161b667 | ||
|   | 4ec1d4405f | ||
|   | f911fc2c10 | ||
|   | 866f9ca2fd | ||
|   | 1326d5b4ef | ||
|   | ad219cdf4b | ||
|   | d87a0aacfb | ||
|   | 3f9011aac1 | ||
|   | 74117ccf5b | ||
|   | e992717c45 | ||
|   | 070c89b582 | ||
|   | 07913bbf61 | ||
|   | 3fcd1881a5 | ||
|   | b43d89cf94 | ||
| 7da8c1ae2f | |||
|   | eb9187a965 | ||
|   | 45302341e2 | ||
|   | c93194ba3e | ||
|   | f2dafd6fe8 | ||
|   | 8e96702d8e | ||
|   | 2a67217008 | ||
|   | 9bff1a43b3 | ||
|   | 016f9896b7 | ||
|   | 74b44dd8e8 | ||
|   | b709c8252c | ||
|   | 736b34231a | ||
|   | ec200a3819 | ||
|   | 554340dd64 | ||
|   | e860202af8 | ||
|   | 50af5ce44d | ||
|   | 86b104f007 | ||
|   | 7496442276 | ||
|   | 4c0df73e74 | ||
|   | a69bf21fbb | ||
|   | 3ade00fe08 | ||
|   | 14c53aa856 | ||
|   | 0e4375b561 | ||
|   | 7bdd9239eb | ||
|   | ce430f0f22 | ||
|   | bf1c2e2a29 | ||
| 5674b4acbf | 
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -8,4 +8,6 @@ uploads/ | ||||
| db/mysql/* | ||||
| db/pgsql/* | ||||
| db/shopping.db | ||||
| *.swp | ||||
| *.swp | ||||
| version.txt | ||||
| deploy/varnish/default.vcl | ||||
							
								
								
									
										36
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,36 +0,0 @@ | ||||
| # Używamy lekkiego obrazu Pythona | ||||
| FROM python:3.13-slim | ||||
|  | ||||
| # Ustawiamy katalog roboczy | ||||
| WORKDIR /app | ||||
|  | ||||
| # Zależności systemowe do OCR, obrazów, tesseract i języka PL | ||||
| RUN apt-get update && apt-get install -y --no-install-recommends \ | ||||
|     tesseract-ocr \ | ||||
|     tesseract-ocr-pol \ | ||||
|     libglib2.0-0 \ | ||||
|     libsm6 \ | ||||
|     libxrender1 \ | ||||
|     libxext6 \ | ||||
|     poppler-utils \ | ||||
|     && apt-get clean \ | ||||
|     && rm -rf /var/lib/apt/lists/* | ||||
|  | ||||
| # Kopiujemy wymagania | ||||
| COPY requirements.txt requirements.txt | ||||
|  | ||||
| # Instalujemy zależności | ||||
| RUN pip install --no-cache-dir -r requirements.txt | ||||
|  | ||||
| # Kopiujemy resztę aplikacji | ||||
| COPY . . | ||||
|  | ||||
| # Kopiujemy entrypoint i ustawiamy uprawnienia | ||||
| COPY entrypoint.sh /entrypoint.sh | ||||
| RUN chmod +x /entrypoint.sh | ||||
|  | ||||
| # Otwieramy port | ||||
| EXPOSE 8000 | ||||
|  | ||||
| # Ustawiamy entrypoint | ||||
| ENTRYPOINT ["/entrypoint.sh"] | ||||
							
								
								
									
										1
									
								
								Dockerfile
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						
									
										1
									
								
								Dockerfile
									
									
									
									
									
										Symbolic link
									
								
							| @@ -0,0 +1 @@ | ||||
| deploy/app/Dockerfile | ||||
| @@ -1,66 +0,0 @@ | ||||
| # ========================= | ||||
| #  Stage 1 – Build | ||||
| # ========================= | ||||
| FROM python:3.13-alpine AS builder | ||||
|  | ||||
| WORKDIR /app | ||||
|  | ||||
| # Instalacja bibliotek do kompilacji + zależności runtime | ||||
| RUN apk add --no-cache \ | ||||
|     tesseract-ocr \ | ||||
|     tesseract-ocr-data-pol \ | ||||
|     poppler-utils \ | ||||
|     libjpeg-turbo \ | ||||
|     zlib \ | ||||
|     libpng \ | ||||
|     libwebp \ | ||||
|     libffi \ | ||||
|     libmagic \ | ||||
|     && apk add --no-cache --virtual .build-deps \ | ||||
|     build-base \ | ||||
|     jpeg-dev \ | ||||
|     zlib-dev \ | ||||
|     libpng-dev \ | ||||
|     libwebp-dev \ | ||||
|     libffi-dev | ||||
|  | ||||
| # Kopiujemy plik wymagań | ||||
| COPY requirements.txt . | ||||
|  | ||||
| # Instalujemy zależności Pythona do folderu tymczasowego | ||||
| RUN pip install --no-cache-dir --prefix=/install -r requirements.txt | ||||
|  | ||||
|  | ||||
| # ========================= | ||||
| #  Stage 2 – Final image | ||||
| # ========================= | ||||
| FROM python:3.13-alpine | ||||
|  | ||||
| WORKDIR /app | ||||
|  | ||||
| # Instalacja tylko bibliotek runtime (bez dev) | ||||
| RUN apk add --no-cache \ | ||||
|     tesseract-ocr \ | ||||
|     tesseract-ocr-data-pol \ | ||||
|     poppler-utils \ | ||||
|     libjpeg-turbo \ | ||||
|     zlib \ | ||||
|     libpng \ | ||||
|     libwebp \ | ||||
|     libffi \ | ||||
|     libmagic | ||||
|  | ||||
| # Kopiujemy zbudowane biblioteki z buildera | ||||
| COPY --from=builder /install /usr/local | ||||
|  | ||||
| # Kopiujemy kod aplikacji | ||||
| COPY . . | ||||
|  | ||||
| # Ustawiamy entrypoint | ||||
| COPY entrypoint.sh /entrypoint.sh | ||||
| RUN chmod +x /entrypoint.sh | ||||
|  | ||||
| # Otwieramy port aplikacji | ||||
| EXPOSE 8000 | ||||
|  | ||||
| ENTRYPOINT ["/entrypoint.sh"] | ||||
| @@ -10,6 +10,8 @@ class Config: | ||||
|      | ||||
|     SECRET_KEY = os.environ.get("SECRET_KEY", "D8pceNZ8q%YR7^7F&9wAC2") | ||||
|  | ||||
|     APP_PORT = int(os.environ.get("APP_PORT", "8000") or "8000") | ||||
|  | ||||
|     DB_ENGINE = os.environ.get("DB_ENGINE", "sqlite").lower() | ||||
|     if DB_ENGINE == "sqlite": | ||||
|         SQLALCHEMY_DATABASE_URI = ( | ||||
| @@ -75,9 +77,9 @@ class Config: | ||||
|     DEFAULT_CATEGORIES = [ | ||||
|         c.strip() for c in os.environ.get( | ||||
|             "DEFAULT_CATEGORIES", | ||||
|             "Spożywcze,Budowlane,Zabawki,Chemia,Inne,Elektronika,Odzież i obuwie," | ||||
|             "Spożywcze,Budowlane,Zabawki,Chemia,Inne,Elektronika,Odzież i obuwie,Jedzenie poza domem," | ||||
|             "Artykuły biurowe,Kosmetyki i higiena,Motoryzacja,Ogród i rośliny," | ||||
|             "Zwierzęta,Sprzęt sportowy,Książki i prasa,Narzędzia i majsterkowanie," | ||||
|             "RTV / AGD,Apteka i suplementy,Artykuły dekoracyjne,Gry i hobby,Usługi,Pieczywo,Różne,Chiny,Dom" | ||||
|             "RTV / AGD,Apteka i suplementy,Artykuły dekoracyjne,Gry i hobby,Usługi,Pieczywo,Różne,Chiny,Dom,Leki,Odzież,Samochód,Dzieci" | ||||
|         ).split(",") if c.strip() | ||||
|     ] | ||||
|     ] | ||||
|   | ||||
							
								
								
									
										35
									
								
								deploy/app/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								deploy/app/Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
|  | ||||
| FROM python:3.14-rc-trixie | ||||
| #FROM python:3.13-slim | ||||
| WORKDIR /app | ||||
|  | ||||
| # Zależności systemowe do OCR, obrazów, tesseract i języka PL | ||||
| RUN apt-get update && apt-get install -y --no-install-recommends \ | ||||
|     tesseract-ocr \ | ||||
|     tesseract-ocr-pol \ | ||||
|     libglib2.0-0 \ | ||||
|     libsm6 \ | ||||
|     libxrender1 \ | ||||
|     libxext6 \ | ||||
|     poppler-utils \ | ||||
|     && apt-get clean \ | ||||
|     && rm -rf /var/lib/apt/lists/* | ||||
|  | ||||
| # Kopiujemy wymagania | ||||
| COPY requirements.txt requirements.txt | ||||
|  | ||||
| # Instalujemy zależności | ||||
| RUN pip install --no-cache-dir -r requirements.txt | ||||
|  | ||||
| # Kopiujemy resztę aplikacji | ||||
| COPY . . | ||||
|  | ||||
| # Kopiujemy entrypoint i ustawiamy uprawnienia | ||||
| COPY entrypoint.sh /entrypoint.sh | ||||
| RUN chmod +x /entrypoint.sh | ||||
|  | ||||
| # Otwieramy port | ||||
| #EXPOSE 8000 | ||||
|  | ||||
| # Ustawiamy entrypoint | ||||
| ENTRYPOINT ["/entrypoint.sh"] | ||||
							
								
								
									
										264
									
								
								deploy/varnish/default.vcl.template
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										264
									
								
								deploy/varnish/default.vcl.template
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,264 @@ | ||||
| vcl 4.1; | ||||
|  | ||||
| import vsthrottle; | ||||
| import std; | ||||
|  | ||||
| # ===== Backend ===== | ||||
| backend app { | ||||
|   .host = "app"; | ||||
|   .port = "${APP_PORT}"; | ||||
| } | ||||
|  | ||||
| # ===== ACL ===== | ||||
| acl purge { | ||||
|   "127.0.0.1"; | ||||
|   "::1"; | ||||
| } | ||||
|  | ||||
| # ===== RECV ===== | ||||
| sub vcl_recv { | ||||
|   # RATE LIMIT: 200 żądań / 10s, blokada 60s | ||||
|   if (vsthrottle.is_denied(client.identity, 200, 10s, 60s)) { | ||||
|     return (synth(429, "Too Many Requests")); | ||||
|   } | ||||
|  | ||||
|   # PURGE tylko lokalnie | ||||
|   if (req.method == "PURGE") { | ||||
|     if (!client.ip ~ purge) { return (synth(405, "Not allowed")); } | ||||
|     return (purge); | ||||
|   } | ||||
|  | ||||
|   # omijamy cache dla healthchecków / wewnętrznych nagłówków | ||||
|   if (req.url == "/healthcheck" || req.http.X-Internal-Check) { return (pass); } | ||||
|  | ||||
|   # Specjalna obsługa WebSocket i socket.io | ||||
|   if (req.http.Upgrade ~ "(?i)websocket" || req.url ~ "^/socket.io/") { | ||||
|     return (pipe); | ||||
|   } | ||||
|  | ||||
|   # metody inne niż GET/HEAD bez cache | ||||
|   if (req.method != "GET" && req.method != "HEAD") { return (pass); } | ||||
|  | ||||
|   # Żądania z Authorization nie są buforowane | ||||
|   if (req.http.Authorization) { return (pass); } | ||||
|  | ||||
|   # ---- Normalizacja Accept-Encoding (kolejność: zstd > br > gzip) ---- | ||||
|   if (req.http.Accept-Encoding) { | ||||
|     if (req.http.Accept-Encoding ~ "zstd") { | ||||
|       set req.http.Accept-Encoding = "zstd"; | ||||
|     } else if (req.http.Accept-Encoding ~ "br") { | ||||
|       set req.http.Accept-Encoding = "br"; | ||||
|     } else if (req.http.Accept-Encoding ~ "gzip") { | ||||
|       set req.http.Accept-Encoding = "gzip"; | ||||
|     } else { | ||||
|       set req.http.Accept-Encoding = "identity"; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   # ---- (Opcjonalnie) Normalizacja Accept dla obrazów generowanych wariantowo ---- | ||||
|   # if (req.url ~ "\.(png|jpe?g|gif|bmp)$") { | ||||
|   #   if (req.http.Accept ~ "image/webp") { | ||||
|   #     set req.http.X-Accept-Image = "modern";   # webp | ||||
|   #   } else { | ||||
|   #     set req.http.X-Accept-Image = "legacy";   # jpg/png | ||||
|   #   } | ||||
|   # } | ||||
|  | ||||
|   # ---- STATYCZNE – agresywny cache + ignorujemy sesję ---- | ||||
|   if (req.url ~ "^/static/" || req.url ~ "\.(css|js|png|jpe?g|webp|svg|ico|woff2?)$") { | ||||
|     unset req.http.Cookie; | ||||
|     unset req.http.Authorization; | ||||
|     return (hash); | ||||
|   } | ||||
|  | ||||
|   if (!req.http.X-Forwarded-Proto) { | ||||
|     set req.http.X-Forwarded-Proto = "https"; | ||||
|   } | ||||
|  | ||||
|   if (req.url == "/healthcheck" || req.http.X-Internal-Check) { | ||||
|     set req.http.X-Pass-Reason = "internal"; | ||||
|     return (pass); | ||||
|   } | ||||
|  | ||||
|   if (req.method != "GET" && req.method != "HEAD") { | ||||
|     set req.http.X-Pass-Reason = "method"; | ||||
|     return (pass); | ||||
|   } | ||||
|  | ||||
|   if (req.http.Authorization) { | ||||
|     set req.http.X-Pass-Reason = "auth"; | ||||
|     return (pass); | ||||
|   } | ||||
|  | ||||
|   # jeśli chcesz PASS przy cookie: | ||||
|   # if (req.http.Cookie) { | ||||
|   #   set req.http.X-Pass-Reason = "cookie"; | ||||
|   #   return (pass); | ||||
|   # } | ||||
|  | ||||
|   return (hash); | ||||
| } | ||||
|  | ||||
| # ===== PIPE (WebSocket passthrough) ===== | ||||
| sub vcl_pipe { | ||||
|   if (req.http.Upgrade) { | ||||
|     set bereq.http.Upgrade = req.http.Upgrade; | ||||
|     set bereq.http.Connection = req.http.Connection; | ||||
|   } | ||||
| } | ||||
|  | ||||
| # ===== HASH ===== | ||||
| sub vcl_hash { | ||||
|   hash_data(req.url); | ||||
|   if (req.http.host) { hash_data(req.http.host); } else { hash_data(server.ip); } | ||||
|  | ||||
|   # Cookie: zostają dla dynamicznych (dla statyków wyczyszczone wcześniej) | ||||
|   if (req.http.Cookie) { hash_data(req.http.Cookie); } | ||||
|  | ||||
|   # Accept-Encoding: już znormalizowany do zstd/br/gzip/identity | ||||
|   if (req.http.Accept-Encoding) { hash_data(req.http.Accept-Encoding); } | ||||
|  | ||||
|   # (Opcjonalnie) sygnał obrazów z negocjacją po Accept | ||||
|   if (req.http.X-Accept-Image) { hash_data(req.http.X-Accept-Image); } | ||||
| } | ||||
|  | ||||
| # ===== BACKEND_RESPONSE ===== | ||||
| sub vcl_backend_response { | ||||
|   # Zakaz cache – respektujemy | ||||
|   if (beresp.http.Cache-Control ~ "(?i)no-store|private") { | ||||
|     set beresp.uncacheable = true; | ||||
|     set beresp.ttl = 0s; | ||||
|     set beresp.http.X-Pass-Reason = "no-store"; | ||||
|     return (deliver); | ||||
|   } | ||||
|  | ||||
|   # NIE cache'uj redirectów do loginu (HTML) z backendu | ||||
|   if (beresp.status >= 300 && beresp.status < 400) { | ||||
|     set beresp.uncacheable = true; | ||||
|     set beresp.ttl = 0s; | ||||
|     set beresp.http.X-Pass-Reason = "redirect"; | ||||
|     return (deliver); | ||||
|   } | ||||
|  | ||||
|   # Nie cache'uj statyków, jeśli status ≠ 200 | ||||
|   if (bereq.url ~ "^/static/" || | ||||
|       bereq.url ~ "\.(css|js|png|jpe?g|webp|svg|ico|woff2?)($|\?)") { | ||||
|     if (beresp.status != 200) { | ||||
|       set beresp.uncacheable = true; | ||||
|       set beresp.ttl = 0s; | ||||
|       return (deliver); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   # Jeśli pod .js przychodzi text/html — też nie cache'uj (to zwykle redirect/login) | ||||
|   if (bereq.url ~ "\.js(\?.*)?$" && beresp.http.Content-Type ~ "(?i)text/html") { | ||||
|     set beresp.uncacheable = true; | ||||
|     set beresp.ttl = 0s; | ||||
|     return (deliver); | ||||
|   } | ||||
|  | ||||
|   # Wymuś poprawny Content-Type dla .js/.css, gdy backend zwróci HTML | ||||
|   if (bereq.url ~ "\.js(\?.*)?$") { | ||||
|     if (!beresp.http.Content-Type || beresp.http.Content-Type ~ "(?i)text/html") { | ||||
|       set beresp.http.Content-Type = "application/javascript; charset=utf-8"; | ||||
|     } | ||||
|   } | ||||
|   if (bereq.url ~ "\.css(\?.*)?$") { | ||||
|     if (!beresp.http.Content-Type || beresp.http.Content-Type ~ "(?i)text/html") { | ||||
|       set beresp.http.Content-Type = "text/css; charset=utf-8"; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   # ---- STATYCZNE: zdejmij Set-Cookie i Vary: Cookie, zapewnij TTL ---- | ||||
|   if (bereq.url ~ "^/static/" || bereq.url ~ "\.(css|js|png|jpe?g|webp|svg|ico|woff2?)$") { | ||||
|     unset beresp.http.Set-Cookie; | ||||
|  | ||||
|     # Jeśli backend dodał Vary: Cookie, usuńmy ten element (nie wpływa na statyki) | ||||
|     if (beresp.http.Vary) { | ||||
|       set beresp.http.Vary = regsuball(beresp.http.Vary, "(?i)(^|,)[[:space:]]*Cookie[[:space:]]*(,|$)", "\1"); | ||||
|       set beresp.http.Vary = regsuball(beresp.http.Vary, ",[[:space:]]*,", ","); | ||||
|       set beresp.http.Vary = regsub(beresp.http.Vary, "^[[:space:]]*,[[:space:]]*", ""); | ||||
|       set beresp.http.Vary = regsub(beresp.http.Vary, "[[:space:]]*,[[:space:]]*$", ""); | ||||
|       if (beresp.http.Vary ~ "^[[:space:]]*$") { unset beresp.http.Vary; } | ||||
|     } | ||||
|  | ||||
|     # Jeśli brak kontroli czasu życia – ustawiamy twarde wartości | ||||
|     if (!(beresp.http.Cache-Control ~ "(?i)(s-maxage|max-age)")) { | ||||
|       set beresp.ttl = 24h; | ||||
|       set beresp.http.Cache-Control = "public, max-age=86400, immutable"; | ||||
|     } | ||||
|  | ||||
|     set beresp.grace = 1h; | ||||
|     set beresp.keep  = 24h; | ||||
|   } | ||||
|  | ||||
|   # ---- Ogólne TTL z nagłówków ---- | ||||
|   if (beresp.http.Cache-Control ~ "(?i)s-maxage=([0-9]+)") { | ||||
|     set beresp.ttl = std.duration(regsub(beresp.http.Cache-Control, "(?i).*s-maxage=([0-9]+).*", "\1") + "s", 0s); | ||||
|   } else if (beresp.http.Cache-Control ~ "(?i)max-age=([0-9]+)") { | ||||
|     set beresp.ttl = std.duration(regsub(beresp.http.Cache-Control, "(?i).*max-age=([0-9]+).*", "\1") + "s", 0s); | ||||
|   } else if (beresp.http.Expires) { | ||||
|     set beresp.ttl = std.time(beresp.http.Expires, now) - now; | ||||
|     if (beresp.ttl < 0s) { set beresp.ttl = 0s; } | ||||
|   } else { | ||||
|     if (beresp.ttl <= 0s) { set beresp.ttl = 60s; } | ||||
|   } | ||||
|  | ||||
|   # Immutable => dłuższe grace/keep | ||||
|   if (beresp.http.Cache-Control ~ "(?i)immutable") { | ||||
|     set beresp.grace = 1h; | ||||
|     set beresp.keep  = 24h; | ||||
|   } | ||||
|  | ||||
|   # Kompresja po stronie Varnisha wyłącznie dla klientów akceptujących gzip | ||||
|   # i tylko jeśli backend nie dostarczył już Content-Encoding. | ||||
|   if (!beresp.http.Content-Encoding && bereq.http.Accept-Encoding ~ "gzip") { | ||||
|     # Kompresujemy tylko „tekstowe” typy; wykluczamy WASM | ||||
|     if (beresp.http.Content-Type ~ "(?i)text/|application/(javascript|json|xml)") { | ||||
|       set beresp.do_gzip = true; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   # Duże odpowiedzi streamujemy | ||||
|   if (beresp.http.Content-Length && std.integer(beresp.http.Content-Length, 0) > 1048576) { | ||||
|     set beresp.do_stream = true; | ||||
|   } | ||||
| } | ||||
|  | ||||
| # (Opcjonalnie) Serwuj „stale” przy błędach backendu, jeśli jest obiekt w grace | ||||
| sub vcl_backend_error { | ||||
|   return (deliver); | ||||
| } | ||||
|  | ||||
| # ===== DELIVER ===== | ||||
| sub vcl_deliver { | ||||
|   if (obj.uncacheable) { | ||||
|     if (req.http.X-Pass-Reason) { | ||||
|       set resp.http.X-Cache = "PASS:" + req.http.X-Pass-Reason; | ||||
|     } else if (resp.http.X-Pass-Reason) {    # z backendu | ||||
|       set resp.http.X-Cache = "PASS:" + resp.http.X-Pass-Reason; | ||||
|     } else { | ||||
|       set resp.http.X-Cache = "PASS"; | ||||
|     } | ||||
|     unset resp.http.X-Pass-Reason; | ||||
|     unset resp.http.Age; | ||||
|   } else if (obj.hits > 0) { | ||||
|     set resp.http.X-Cache = "HIT"; | ||||
|   } else { | ||||
|     set resp.http.X-Cache = "MISS"; | ||||
|     unset resp.http.Age; | ||||
|   } | ||||
|  | ||||
|   unset resp.http.Via; | ||||
|   unset resp.http.X-Varnish; | ||||
|   unset resp.http.Server; | ||||
| } | ||||
|  | ||||
| sub vcl_synth { | ||||
|   set resp.http.X-Cache = "SYNTH"; | ||||
| } | ||||
|  | ||||
| # ===== PURGE HANDLER ===== | ||||
| sub vcl_purge { | ||||
|   return (synth(200, "Purged")); | ||||
| } | ||||
| @@ -1,14 +1,23 @@ | ||||
| #!/bin/bash | ||||
| set -e | ||||
|  | ||||
| # --- Wczytaj zmienne z .env --- | ||||
| if [[ -f .env ]]; then | ||||
|   set -a | ||||
|   source .env | ||||
|   set +a | ||||
| fi | ||||
|  | ||||
| APP_PORT="${APP_PORT:-8080}" | ||||
|  | ||||
| PROFILE=$1 | ||||
|  | ||||
| if [[ -z "$PROFILE" ]]; then | ||||
|   echo "Uzycie: $0 {pgsql|mysql|sqlite}" | ||||
|   echo "Użycie: $0 {pgsql|mysql|sqlite}" | ||||
|   exit 1 | ||||
| fi | ||||
|  | ||||
| echo "Zatrzymuje kontenery aplikacji i bazy..." | ||||
| echo "Zatrzymuję kontenery aplikacji i bazy..." | ||||
| if [[ "$PROFILE" == "sqlite" ]]; then | ||||
|   docker compose stop | ||||
| else | ||||
| @@ -18,11 +27,17 @@ fi | ||||
| echo "Pobieram najnowszy kod z repozytorium..." | ||||
| git pull | ||||
|  | ||||
| echo "Buduje i uruchamiam kontenery..." | ||||
| echo "Generowanie default.vcl z APP_PORT=$APP_PORT" | ||||
| envsubst < deploy/varnish/default.vcl.template > deploy/varnish/default.vcl | ||||
|  | ||||
| echo "Zapisuję hash commita do version.txt..." | ||||
| git rev-parse --short HEAD > version.txt | ||||
|  | ||||
| echo "Buduję i uruchamiam kontenery..." | ||||
| if [[ "$PROFILE" == "sqlite" ]]; then | ||||
|   docker compose up -d --build | ||||
| else | ||||
|   DB_ENGINE="$PROFILE" docker compose --profile "$PROFILE" up -d --build | ||||
| fi | ||||
|  | ||||
| echo "Gotowe!" | ||||
| echo "Gotowe! Wersja aplikacji: $(cat version.txt)" | ||||
|   | ||||
| @@ -1,11 +1,17 @@ | ||||
| services: | ||||
|   app: | ||||
|     build: . | ||||
|     container_name: live-lista-zakupow | ||||
|     ports: | ||||
|       - "${APP_PORT:-8000}:8000" | ||||
|     container_name: lista-zakupow-app | ||||
|     expose: | ||||
|       - "${APP_PORT:-8000}" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "python", "-c", "import urllib.request; import sys; req = urllib.request.Request('http://localhost:8000/healthcheck', headers={'X-Internal-Check': '${HEALTHCHECK_TOKEN}'}); sys.exit(0) if urllib.request.urlopen(req).read() == b'OK' else sys.exit(1)"] | ||||
|       test: | ||||
|         [ | ||||
|           "CMD", | ||||
|           "python", | ||||
|           "-c", | ||||
|           "import urllib.request; import sys; req = urllib.request.Request('http://localhost:${APP_PORT:-8000}/healthcheck', headers={'X-Internal-Check': '${HEALTHCHECK_TOKEN}'}); sys.exit(0) if urllib.request.urlopen(req).read() == b'OK' else sys.exit(1)", | ||||
|         ] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 3 | ||||
| @@ -16,23 +22,29 @@ services: | ||||
|       - .:/app | ||||
|       - ./uploads:/app/uploads | ||||
|       - ./instance:/app/instance | ||||
|     networks: | ||||
|       - lista-zakupow_network | ||||
|     restart: unless-stopped | ||||
|  | ||||
|   pgsql: | ||||
|     image: postgres:17 | ||||
|     container_name: pgsql-db | ||||
|     environment: | ||||
|       POSTGRES_DB: ${DB_NAME} | ||||
|       POSTGRES_USER: ${DB_USER} | ||||
|       POSTGRES_PASSWORD: ${DB_PASSWORD} | ||||
|   varnish: | ||||
|     image: varnish:latest | ||||
|     container_name: lista-zakupow-varnish | ||||
|     depends_on: | ||||
|       app: | ||||
|         condition: service_healthy | ||||
|     ports: | ||||
|       - "${APP_PORT:-8000}:80" | ||||
|     volumes: | ||||
|       - ./db/pgsql:/var/lib/postgresql/data | ||||
|       - ./deploy/varnish/default.vcl:/etc/varnish/default.vcl:ro | ||||
|     environment: | ||||
|       - VARNISH_SIZE=256m | ||||
|     networks: | ||||
|       - lista-zakupow_network | ||||
|     restart: unless-stopped | ||||
|     profiles: ["pgsql"] | ||||
|  | ||||
|   mysql: | ||||
|     image: mysql:8 | ||||
|     container_name: mysql-db | ||||
|     container_name: lista-zakupow-mysql-db | ||||
|     environment: | ||||
|       MYSQL_DATABASE: ${DB_NAME} | ||||
|       MYSQL_USER: ${DB_USER} | ||||
| @@ -41,4 +53,25 @@ services: | ||||
|     volumes: | ||||
|       - ./db/mysql:/var/lib/mysql | ||||
|     restart: unless-stopped | ||||
|     profiles: ["mysql"] | ||||
|     networks: | ||||
|       - lista-zakupow_network | ||||
|     profiles: ["mysql"] | ||||
|  | ||||
|   pgsql: | ||||
|     image: postgres:18 | ||||
|     container_name: lista-zakupow-pgsql | ||||
|     environment: | ||||
|       POSTGRES_DB: ${DB_NAME} | ||||
|       POSTGRES_USER: ${DB_USER} | ||||
|       POSTGRES_PASSWORD: ${DB_PASSWORD} | ||||
|       PGDATA: /var/lib/postgresql | ||||
|     volumes: | ||||
|       - ./db/pgsql/:/var/lib/postgresql | ||||
|     networks: | ||||
|       - lista-zakupow_network | ||||
|     restart: unless-stopped | ||||
|     profiles: ["pgsql"] | ||||
|  | ||||
| networks: | ||||
|   lista-zakupow_network: | ||||
|     driver: bridge | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										416
									
								
								static/css/style_old.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										416
									
								
								static/css/style_old.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,416 @@ | ||||
| /* --- Rozmiary i kursory --- */ | ||||
| .large-checkbox { | ||||
|     width: 1.5em; | ||||
|     height: 1.5em; | ||||
| } | ||||
|  | ||||
| .clickable-item { | ||||
|     cursor: pointer; | ||||
| } | ||||
|  | ||||
| /* --- Kolory tła (nadpisane klasy Bootstrapa) --- */ | ||||
| .bg-success { | ||||
|     background-color: #1e7e34 !important; | ||||
| } | ||||
|  | ||||
| .btn-outline-light:hover { | ||||
|     background-color: #ffc107 !important; | ||||
|     color: #000 !important; | ||||
|     border-color: #ffc107 !important; | ||||
| } | ||||
|  | ||||
| .progress-dark { | ||||
|     background-color: #212529 !important; | ||||
|     border-radius: 20px !important; | ||||
|     overflow: hidden; | ||||
| } | ||||
|  | ||||
| .progress-bar { | ||||
|     border-radius: 0 !important; | ||||
|     transition: width 0.4s ease, background-color 0.4s ease; | ||||
| } | ||||
|  | ||||
| .progress-bar:first-child { | ||||
|     border-top-left-radius: 20px !important; | ||||
|     border-bottom-left-radius: 20px !important; | ||||
| } | ||||
|  | ||||
| .progress-bar:last-child { | ||||
|     border-top-right-radius: 20px !important; | ||||
|     border-bottom-right-radius: 20px !important; | ||||
| } | ||||
|  | ||||
|  | ||||
| /* rodzic już ma position-relative */ | ||||
| .progress-label { | ||||
|     position: absolute; | ||||
|     top: 50%; | ||||
|     left: 50%; | ||||
|     transform: translate(-50%, -50%); | ||||
|     pointer-events: none; | ||||
|     /* klikalne przyciski obok paska nie ucierpią */ | ||||
|     white-space: nowrap; | ||||
| } | ||||
|  | ||||
| .progress-thin { | ||||
|     height: 12px; | ||||
| } | ||||
|  | ||||
| .item-not-checked { | ||||
|     background-color: #2c2f33 !important; | ||||
|     color: white !important; | ||||
| } | ||||
|  | ||||
| /* --- Styl przycisku wyboru pliku --- */ | ||||
| input[type="file"]::file-selector-button { | ||||
|     background-color: #225d36; | ||||
|     color: #fff; | ||||
|     border: none; | ||||
|     padding: 0.5em 1em; | ||||
|     border-radius: 4px; | ||||
|     font-weight: bold; | ||||
|     cursor: pointer; | ||||
|     transition: background 0.2s; | ||||
| } | ||||
|  | ||||
| /* --- Ciemniejsze alerty Bootstrapa --- */ | ||||
| .alert-success { | ||||
|     background-color: #225d36 !important; | ||||
|     color: #eaffea !important; | ||||
|     border-color: #174428 !important; | ||||
| } | ||||
|  | ||||
| .alert-danger { | ||||
|     background-color: #7a1f23 !important; | ||||
|     color: #ffeaea !important; | ||||
|     border-color: #531417 !important; | ||||
| } | ||||
|  | ||||
| .alert-info { | ||||
|     background-color: #1d3a4d !important; | ||||
|     color: #eaf6ff !important; | ||||
|     border-color: #152837 !important; | ||||
| } | ||||
|  | ||||
| .alert-warning { | ||||
|     background-color: #665c1e !important; | ||||
|     color: #fffbe5 !important; | ||||
|     border-color: #4d4415 !important; | ||||
| } | ||||
|  | ||||
| /* Badge - kolory pasujące do ciemnych alertów */ | ||||
| .badge.bg-success, | ||||
| .badge.text-bg-success { | ||||
|     background-color: #225d36 !important; | ||||
|     color: #eaffea !important; | ||||
| } | ||||
|  | ||||
| .badge.bg-danger, | ||||
| .badge.text-bg-danger { | ||||
|     background-color: #7a1f23 !important; | ||||
|     color: #ffeaea !important; | ||||
| } | ||||
|  | ||||
| .badge.bg-info, | ||||
| .badge.text-bg-info { | ||||
|     background-color: #1d3a4d !important; | ||||
|     color: #eaf6ff !important; | ||||
| } | ||||
|  | ||||
| .badge.bg-warning, | ||||
| .badge.text-bg-warning { | ||||
|     background-color: #665c1e !important; | ||||
|     color: #fffbe5 !important; | ||||
| } | ||||
|  | ||||
| .badge.bg-secondary, | ||||
| .badge.text-bg-secondary { | ||||
|     background-color: #343a40 !important; | ||||
|     color: #e2e3e5 !important; | ||||
| } | ||||
|  | ||||
| .badge.bg-primary, | ||||
| .badge.text-bg-primary { | ||||
|     background-color: #184076 !important; | ||||
|     color: #e6f0ff !important; | ||||
| } | ||||
|  | ||||
| .badge.bg-light, | ||||
| .badge.text-bg-light { | ||||
|     background-color: #444950 !important; | ||||
|     color: #f8f9fa !important; | ||||
| } | ||||
|  | ||||
| .badge.bg-dark, | ||||
| .badge.text-bg-dark { | ||||
|     background-color: #181a1b !important; | ||||
|     color: #f8f9fa !important; | ||||
| } | ||||
|  | ||||
| /* --- Styl dla własnych checkboxów --- */ | ||||
| input[type="checkbox"].large-checkbox { | ||||
|     appearance: none; | ||||
|     -webkit-appearance: none; | ||||
|     -moz-appearance: none; | ||||
|     width: 1.5em; | ||||
|     height: 1.5em; | ||||
|     margin: 0; | ||||
|     padding: 0; | ||||
|     outline: none; | ||||
|     background: none; | ||||
|     cursor: pointer; | ||||
|     position: relative; | ||||
|     vertical-align: middle; | ||||
| } | ||||
|  | ||||
| input[type="checkbox"].large-checkbox::before { | ||||
|     content: '✗'; | ||||
|     color: #dc3545; | ||||
|     font-size: 1.5em; | ||||
|     font-weight: bold; | ||||
|     position: absolute; | ||||
|     left: 0; | ||||
|     top: 50%; | ||||
|     transform: translateY(-50%); | ||||
|     line-height: 1; | ||||
|     transition: color 0.2s; | ||||
| } | ||||
|  | ||||
| input[type="checkbox"].large-checkbox:checked::before { | ||||
|     content: '✓'; | ||||
|     color: #ffffff; | ||||
| } | ||||
|  | ||||
| input[type="checkbox"].large-checkbox:disabled::before { | ||||
|     opacity: 0.5; | ||||
|     cursor: not-allowed; | ||||
| } | ||||
|  | ||||
| input[type="checkbox"].large-checkbox:disabled { | ||||
|     cursor: not-allowed; | ||||
| } | ||||
|  | ||||
| #tempToggle { | ||||
|     border-top-left-radius: 0; | ||||
|     border-bottom-left-radius: 0; | ||||
| } | ||||
|  | ||||
| input.form-control { | ||||
|     border-top-right-radius: 0; | ||||
|     border-bottom-right-radius: 0; | ||||
| } | ||||
|  | ||||
| .info-bar-fixed { | ||||
|     width: 100%; | ||||
|     color: #f8f9fa; | ||||
|     background-color: #212529; | ||||
|     border-radius: 12px 12px 0 0; | ||||
|     text-align: center; | ||||
|     padding: 10px 10px; | ||||
|     font-size: 0.95rem; | ||||
|     box-sizing: border-box; | ||||
|     margin-top: 2rem; | ||||
|     box-shadow: 0 -1px 4px rgba(0, 0, 0, 0.25); | ||||
| } | ||||
|  | ||||
| @media (max-width: 768px) { | ||||
|     .info-bar-fixed { | ||||
|         position: static; | ||||
|         font-size: 0.85rem; | ||||
|         padding: 8px 4px; | ||||
|         border-radius: 0; | ||||
|     } | ||||
| } | ||||
|  | ||||
| .table-responsive { | ||||
|     overflow-x: auto; | ||||
|     -webkit-overflow-scrolling: touch; | ||||
| } | ||||
|  | ||||
| .table-responsive table { | ||||
|     min-width: 1000px; | ||||
| } | ||||
|  | ||||
| .bg-dark .form-control::placeholder { | ||||
|     color: #ccc !important; | ||||
|     opacity: 1; | ||||
| } | ||||
|  | ||||
| .toast-body { | ||||
|     color: #ffffff !important; | ||||
|     font-weight: 500 !important; | ||||
| } | ||||
|  | ||||
| .toast { | ||||
|     animation: fadeInUp 0.5s ease; | ||||
| } | ||||
|  | ||||
| @keyframes fadeInUp { | ||||
|     from { | ||||
|         opacity: 0; | ||||
|         transform: translateY(20px); | ||||
|     } | ||||
|  | ||||
|     to { | ||||
|         opacity: 1; | ||||
|         transform: translateY(0); | ||||
|     } | ||||
| } | ||||
|  | ||||
| #mass-add-list li.active { | ||||
|     background: #198754 !important; | ||||
|     color: #fff !important; | ||||
|     border: 1px solid #000000 !important; | ||||
| } | ||||
|  | ||||
| #mass-add-list li { | ||||
|     transition: background 0.2s; | ||||
| } | ||||
|  | ||||
| .quantity-input { | ||||
|     width: 60px; | ||||
|     background: #343a40; | ||||
|     color: #fff; | ||||
|     border: 1px solid #495057; | ||||
|     border-radius: 4px; | ||||
|     text-align: center; | ||||
| } | ||||
|  | ||||
| .add-btn { | ||||
|     margin-left: 10px; | ||||
| } | ||||
|  | ||||
| .quantity-controls { | ||||
|     min-width: 120px; | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     justify-content: flex-end; | ||||
|     gap: 4px; | ||||
| } | ||||
|  | ||||
| .list-group-item { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
|     justify-content: space-between; | ||||
| } | ||||
|  | ||||
| #empty-placeholder { | ||||
|     font-style: italic; | ||||
|     pointer-events: none; | ||||
| } | ||||
|  | ||||
| #items li.hide-purchased { | ||||
|     display: none !important; | ||||
| } | ||||
|  | ||||
| .list-group-item:first-child, | ||||
| .list-group-item:last-child { | ||||
|     border-radius: 0 !important; | ||||
| } | ||||
|  | ||||
| .fade-out { | ||||
|     opacity: 0; | ||||
|     transition: opacity 0.5s ease; | ||||
| } | ||||
|  | ||||
| @media (pointer: fine) { | ||||
|     .only-mobile { | ||||
|         display: none !important; | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| .ts-dropdown .active { | ||||
|     background-color: #495057 !important; | ||||
| } | ||||
|  | ||||
| .pagination-dark .page-link { | ||||
|     color: #fff; | ||||
|     background-color: #212529; | ||||
|     border: 1px solid #495057; | ||||
| } | ||||
|  | ||||
| .pagination-dark .page-link:hover { | ||||
|     background-color: #343a40; | ||||
|     border-color: #6c757d; | ||||
|     color: #fff; | ||||
| } | ||||
|  | ||||
| .pagination-dark .page-item.active .page-link { | ||||
|     background-color: #0d6efd; | ||||
|     border-color: #0d6efd; | ||||
|     color: #fff; | ||||
| } | ||||
|  | ||||
| .pagination-dark .page-item.disabled .page-link { | ||||
|     background-color: #2b3035; | ||||
|     border-color: #495057; | ||||
|     color: #6c757d; | ||||
| } | ||||
|  | ||||
| .tom-dark .ts-control { | ||||
|     background-color: #212529 !important; | ||||
|     color: #fff !important; | ||||
|     border: 1px solid #495057 !important; | ||||
|     border-radius: 0.375rem; | ||||
|     min-height: 38px; | ||||
|     padding: 0.25rem 0.5rem; | ||||
|     box-sizing: border-box; | ||||
| } | ||||
|  | ||||
| .tom-dark .ts-control .item { | ||||
|     background-color: #343a40 !important; | ||||
|     color: #fff !important; | ||||
|     border-radius: 0.25rem; | ||||
|     padding: 2px 8px; | ||||
|     margin-right: 4px; | ||||
| } | ||||
|  | ||||
| .ts-dropdown { | ||||
|     background-color: #212529 !important; | ||||
|     color: #fff !important; | ||||
|     border: 1px solid #495057; | ||||
|     border-radius: 0.375rem; | ||||
|     z-index: 9999 !important; | ||||
|     max-height: 300px; | ||||
|     overflow-y: auto; | ||||
| } | ||||
|  | ||||
| .ts-dropdown .active { | ||||
|     background-color: #495057 !important; | ||||
|     color: #fff !important; | ||||
| } | ||||
|  | ||||
| td select.tom-dark { | ||||
|     width: 100%; | ||||
|     max-width: 100%; | ||||
|     box-sizing: border-box; | ||||
| } | ||||
|  | ||||
| .table-dark.table-striped tbody tr:nth-of-type(odd) { | ||||
|     background-color: rgba(255, 255, 255, 0.025); | ||||
| } | ||||
|  | ||||
| .table-dark tbody tr:hover { | ||||
|     background-color: rgba(255, 255, 255, 0.04); | ||||
| } | ||||
|  | ||||
| .table-dark thead th { | ||||
|     background-color: #1c1f22; | ||||
|     color: #e1e1e1; | ||||
|     font-weight: 500; | ||||
|     border-bottom: 1px solid #3a3f44; | ||||
| } | ||||
|  | ||||
| .table-dark td, | ||||
| .table-dark th { | ||||
|     padding: 0.6rem 0.75rem; | ||||
|     vertical-align: middle; | ||||
|     border-top: 1px solid #3a3f44; | ||||
| } | ||||
|  | ||||
| .card .table { | ||||
|     border-radius: 0 !important; | ||||
|     overflow: hidden; | ||||
|     margin-bottom: 0; | ||||
| } | ||||
							
								
								
									
										176
									
								
								static/js/access_users.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								static/js/access_users.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| (function () { | ||||
|     const $ = (s, root = document) => root.querySelector(s); | ||||
|     const $$ = (s, root = document) => Array.from(root.querySelectorAll(s)); | ||||
|     const toast = (m, t = 'info') => (window.showToast ? window.showToast(m, t) : console.log(`[${t}]`, m)); | ||||
|  | ||||
|     function appendToken(box, user) { | ||||
|         const tokensBox = $('.tokens', box); | ||||
|         if (!tokensBox || !user?.id || !user?.username) return; | ||||
|         const empty = $('.no-perms', box); | ||||
|         if (empty) empty.remove(); | ||||
|  | ||||
|         const btn = document.createElement('button'); | ||||
|         btn.type = 'button'; | ||||
|         btn.className = 'btn btn-sm btn-outline-secondary rounded-pill token'; | ||||
|         btn.dataset.userId = user.id; | ||||
|         btn.dataset.username = user.username; | ||||
|         btn.title = 'Kliknij, aby odebrać dostęp'; | ||||
|         btn.innerHTML = `@${user.username} <span aria-hidden="true">×</span>`; | ||||
|         tokensBox.appendChild(btn); | ||||
|     } | ||||
|  | ||||
|     function wantsJSON() { | ||||
|         return { | ||||
|             'Accept': 'application/json', | ||||
|             'X-Requested-With': 'fetch' | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     async function postAction(postUrl, nextPath, params) { | ||||
|         const form = new FormData(); | ||||
|         for (const [k, v] of Object.entries(params)) form.set(k, v); | ||||
|         form.set('next', nextPath); // dla trybu HTML fallback | ||||
|  | ||||
|         try { | ||||
|             const res = await fetch(postUrl, { | ||||
|                 method: 'POST', | ||||
|                 body: form, | ||||
|                 credentials: 'same-origin', | ||||
|                 headers: wantsJSON() | ||||
|             }); | ||||
|  | ||||
|             const ct = res.headers.get('content-type') || ''; | ||||
|             if (ct.includes('application/json')) { | ||||
|                 const data = await res.json().catch(() => ({})); | ||||
|                 return { ok: !!data?.ok, data, status: res.status }; | ||||
|             } | ||||
|             return { ok: res.ok, data: null, status: res.status }; | ||||
|         } catch (e) { | ||||
|             console.error('POST failed', e); | ||||
|             return { ok: false, data: null, status: 0 }; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function initEditor(box) { | ||||
|         if (!box || !box.classList?.contains('access-editor')) return; | ||||
|         if (box.dataset._accessEditorInit === '1') return; | ||||
|         box.dataset._accessEditorInit = '1'; | ||||
|  | ||||
|         const postUrl = box.dataset.postUrl || location.pathname; | ||||
|         const nextPath = box.dataset.next || location.pathname; | ||||
|         const suggestUrl = box.dataset.suggestUrl || ''; | ||||
|         const grantAction = box.dataset.grantAction || 'grant'; | ||||
|         const revokeField = box.dataset.revokeField || 'revoke_user_id'; | ||||
|  | ||||
|         const tokensBox = $('.tokens', box); | ||||
|         const input = $('.access-input', box); | ||||
|         const addBtn = $('.access-add', box); | ||||
|  | ||||
|         // współdzielony datalist do sugestii | ||||
|         let datalist = $('#userHintsGeneric'); | ||||
|         if (!datalist) { | ||||
|             datalist = document.createElement('datalist'); | ||||
|             datalist.id = 'userHintsGeneric'; | ||||
|             document.body.appendChild(datalist); | ||||
|         } | ||||
|         input?.setAttribute('list', datalist.id); | ||||
|  | ||||
|         const unique = (arr) => Array.from(new Set(arr)); | ||||
|         const parseUserText = (txt) => unique((txt || '').split(/[\s,;]+/g).map(s => s.trim().replace(/^@/, '').toLowerCase()).filter(Boolean)); | ||||
|         const debounce = (fn, ms = 200) => { let t; return (...a) => { clearTimeout(t); t = setTimeout(() => fn(...a), ms); }; }; | ||||
|  | ||||
|         // Sugestie (GET JSON) | ||||
|         const renderHints = (users = []) => { datalist.innerHTML = users.slice(0, 20).map(u => `<option value="${u}">@${u}</option>`).join(''); }; | ||||
|         let acCtrl = null; | ||||
|         const fetchHints = debounce(async (q) => { | ||||
|             if (!suggestUrl) return; | ||||
|             try { | ||||
|                 acCtrl?.abort(); | ||||
|                 acCtrl = new AbortController(); | ||||
|                 const res = await fetch(`${suggestUrl}?q=${encodeURIComponent(q || '')}`, { credentials: 'same-origin', signal: acCtrl.signal }); | ||||
|                 if (!res.ok) return renderHints([]); | ||||
|                 const data = await res.json().catch(() => ({ users: [] })); | ||||
|                 renderHints(data.users || []); | ||||
|             } catch { renderHints([]); } | ||||
|         }, 200); | ||||
|  | ||||
|         input?.addEventListener('focus', () => fetchHints(input.value)); | ||||
|         input?.addEventListener('input', () => fetchHints(input.value)); | ||||
|  | ||||
|         // Revoke (klik w token) | ||||
|         box.addEventListener('click', async (e) => { | ||||
|             const btn = e.target.closest('.token'); | ||||
|             if (!btn || !box.contains(btn)) return; | ||||
|  | ||||
|             const userId = btn.dataset.userId; | ||||
|             const username = btn.dataset.username; | ||||
|             if (!userId) return toast('Brak identyfikatora użytkownika.', 'danger'); | ||||
|  | ||||
|             btn.disabled = true; btn.classList.add('disabled'); | ||||
|             const res = await postAction(postUrl, nextPath, { [revokeField]: userId }); | ||||
|  | ||||
|             if (res.ok) { | ||||
|                 btn.remove(); | ||||
|                 if (!$$('.token', box).length && tokensBox) { | ||||
|                     const empty = document.createElement('span'); | ||||
|                     empty.className = 'no-perms text-warning small'; | ||||
|                     empty.textContent = 'Brak dodanych uprawnień.'; | ||||
|                     tokensBox.appendChild(empty); | ||||
|                 } | ||||
|                 toast(`Odebrano dostęp: @${username}`, 'success'); | ||||
|             } else { | ||||
|                 btn.disabled = false; btn.classList.remove('disabled'); | ||||
|                 toast(`Nie udało się odebrać dostępu @${username}`, 'danger'); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         // Grant (wiele loginów, bez przeładowania strony) | ||||
|         async function addUsers() { | ||||
|             const users = parseUserText(input?.value); | ||||
|             if (!users?.length) return toast('Podaj co najmniej jednego użytkownika', 'warning'); | ||||
|  | ||||
|             addBtn.disabled = true; | ||||
|             const prevText = addBtn.textContent; | ||||
|             addBtn.textContent = 'Dodaję…'; | ||||
|  | ||||
|             let okCount = 0, failCount = 0, appended = 0; | ||||
|  | ||||
|             for (const u of users) { | ||||
|                 const res = await postAction(postUrl, nextPath, { action: grantAction, grant_username: u }); | ||||
|                 if (res.ok) { | ||||
|                     okCount++; | ||||
|                     // jeśli backend odda JSON z userem – dolep token live | ||||
|                     if (res.data?.user) { | ||||
|                         appendToken(box, res.data.user); | ||||
|                         appended++; | ||||
|                     } | ||||
|                 } else { | ||||
|                     failCount++; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             addBtn.disabled = false; | ||||
|             addBtn.textContent = prevText; | ||||
|             if (input) input.value = ''; | ||||
|  | ||||
|             if (okCount) toast(`Dodano dostęp: ${okCount} użytkownika`, 'success'); | ||||
|             if (failCount) toast(`Błędy przy dodawaniu: ${failCount}`, 'danger'); | ||||
|  | ||||
|             // fallback: jeśli nic nie dolepiliśmy (brak JSON), odśwież, by zobaczyć nowe tokeny | ||||
|             if (okCount && appended === 0) { | ||||
|                 // opóźnij minimalnie, by toast mignął | ||||
|                 setTimeout(() => location.reload(), 400); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         addBtn?.addEventListener('click', addUsers); | ||||
|         input?.addEventListener('keydown', (e) => { if (e.key === 'Enter') { e.preventDefault(); addUsers(); } }); | ||||
|     } | ||||
|  | ||||
|     document.addEventListener('DOMContentLoaded', () => { | ||||
|         $$('.access-editor').forEach(initEditor); | ||||
|     }); | ||||
|     document.addEventListener('shown.bs.modal', (ev) => { | ||||
|         $$('.access-editor', ev.target).forEach(initEditor); | ||||
|     }); | ||||
| })(); | ||||
| @@ -1,39 +0,0 @@ | ||||
| (function () { | ||||
|     document.addEventListener("DOMContentLoaded", function () { | ||||
|         const cropModal = document.getElementById("adminCropModal"); | ||||
|         const cropImage = document.getElementById("adminCropImage"); | ||||
|         const spinner = document.getElementById("adminCropLoading"); | ||||
|         const saveButton = document.getElementById("adminSaveCrop"); | ||||
|  | ||||
|         if (!cropModal || !cropImage || !spinner || !saveButton) return; | ||||
|  | ||||
|         let cropper; | ||||
|         let currentReceiptId; | ||||
|         const currentEndpoint = "/admin/crop_receipt"; | ||||
|  | ||||
|         cropModal.addEventListener("shown.bs.modal", function (event) { | ||||
|             const button = event.relatedTarget; | ||||
|             const imgSrc = button.getAttribute("data-img-src"); | ||||
|             currentReceiptId = button.getAttribute("data-receipt-id"); | ||||
|             cropImage.src = imgSrc; | ||||
|  | ||||
|             document.querySelectorAll('.cropper-container').forEach(e => e.remove()); | ||||
|  | ||||
|             if (cropper) cropper.destroy(); | ||||
|             cropImage.onload = () => { | ||||
|                 cropper = cropUtils.initCropper(cropImage); | ||||
|             }; | ||||
|         }); | ||||
|  | ||||
|         cropModal.addEventListener("hidden.bs.modal", function () { | ||||
|             cropUtils.cleanUpCropper(cropImage, cropper); | ||||
|             cropper = null; | ||||
|         }); | ||||
|  | ||||
|         saveButton.addEventListener("click", function () { | ||||
|             if (!cropper) return; | ||||
|             spinner.classList.remove("d-none"); | ||||
|             cropUtils.handleCrop(currentEndpoint, currentReceiptId, cropper, spinner); | ||||
|         }); | ||||
|     }); | ||||
| })(); | ||||
							
								
								
									
										130
									
								
								static/js/admin_settings.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										130
									
								
								static/js/admin_settings.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,130 @@ | ||||
| (function () { | ||||
|   const form = document.getElementById("settings-form"); | ||||
|   const resetAllBtn = document.getElementById("reset-all"); | ||||
|  | ||||
|   function ensureHiddenClear(input) { | ||||
|     let hidden = input.parentElement.querySelector(`input[type="hidden"][name="${input.name}"]`); | ||||
|     if (!hidden) { | ||||
|       hidden = document.createElement("input"); | ||||
|       hidden.type = "hidden"; | ||||
|       hidden.name = input.name; | ||||
|       hidden.value = ""; | ||||
|       input.parentElement.appendChild(hidden); | ||||
|     } | ||||
|   } | ||||
|   function removeHiddenClear(input) { | ||||
|     const hidden = input.parentElement.querySelector(`input[type="hidden"][name="${input.name}"]`); | ||||
|     if (hidden) hidden.remove(); | ||||
|   } | ||||
|  | ||||
|   function updatePreview(input) { | ||||
|     const card = input.closest(".col-12, .col-md-6, .col-lg-4"); | ||||
|     const hexAutoEl = card.querySelector(".hex-auto"); | ||||
|     const hexEffEl  = card.querySelector(".hex-effective"); | ||||
|     const barAuto   = card.querySelector('.bar[data-kind="auto"]'); | ||||
|     const barEff    = card.querySelector('.bar[data-kind="effective"]'); | ||||
|  | ||||
|     const raw = (input.value || "").trim(); | ||||
|     const autoHex = hexAutoEl.textContent.trim(); | ||||
|     const effHex  = (raw || autoHex).toUpperCase(); | ||||
|  | ||||
|     if (barEff) barEff.style.backgroundColor = effHex; | ||||
|     if (hexEffEl) hexEffEl.textContent = effHex; | ||||
|  | ||||
|     if (!raw) { | ||||
|       ensureHiddenClear(input); | ||||
|       input.disabled = true; | ||||
|     } else { | ||||
|       removeHiddenClear(input); | ||||
|       input.disabled = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   form.querySelectorAll(".use-default").forEach(btn => { | ||||
|     btn.addEventListener("click", () => { | ||||
|       const name = btn.getAttribute("data-target"); | ||||
|       const input = form.querySelector(`input[name="${name}"]`); | ||||
|       if (!input) return; | ||||
|       input.value = ""; | ||||
|       updatePreview(input); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   form.querySelectorAll(".reset-one").forEach(btn => { | ||||
|     btn.addEventListener("click", () => { | ||||
|       const name = btn.getAttribute("data-target"); | ||||
|       const input = form.querySelector(`input[name="${name}"]`); | ||||
|       if (!input) return; | ||||
|       input.value = ""; | ||||
|       updatePreview(input); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   resetAllBtn?.addEventListener("click", () => { | ||||
|     form.querySelectorAll('input[type="color"].category-color').forEach(input => { | ||||
|       input.value = ""; | ||||
|       updatePreview(input); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   form.querySelectorAll('input[type="color"].category-color').forEach(input => { | ||||
|     updatePreview(input); | ||||
|     input.addEventListener("input", () => updatePreview(input)); | ||||
|     input.addEventListener("change", () => updatePreview(input)); | ||||
|   }); | ||||
|  | ||||
|   form.addEventListener("submit", () => { | ||||
|     form.querySelectorAll('input[type="color"].category-color').forEach(updatePreview); | ||||
|   }); | ||||
|  | ||||
|   form.querySelectorAll(".use-default").forEach(btn => { | ||||
|     btn.addEventListener("click", () => { | ||||
|       const name = btn.getAttribute("data-target"); | ||||
|       const input = form.querySelector(`input[name="${name}"]`); | ||||
|       if (!input) return; | ||||
|  | ||||
|       const card = input.closest(".col-12, .col-md-6, .col-lg-4") || input.closest(".col-12"); | ||||
|       let autoHex = (input.dataset.auto || "").trim(); | ||||
|       if (!autoHex && card) { | ||||
|         autoHex = (card.querySelector(".hex-auto")?.textContent || "").trim(); | ||||
|       } | ||||
|       if (autoHex && !autoHex.startsWith("#")) autoHex = `#${autoHex}`; | ||||
|  | ||||
|       if (autoHex) { | ||||
|         input.disabled = false; | ||||
|         removeHiddenClear(input); | ||||
|         input.value = autoHex; | ||||
|         updatePreview(input); | ||||
|       } | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   (function () { | ||||
|     const slider = document.getElementById("ocr_sensitivity"); | ||||
|     const badge  = document.getElementById("ocr_sens_badge"); | ||||
|     const value  = document.getElementById("ocr_sens_value"); | ||||
|     if (!slider || !badge || !value) return; | ||||
|  | ||||
|     function labelFor(v) { | ||||
|       v = Number(v); | ||||
|       if (v <= 3) return "Niski"; | ||||
|       if (v <= 7) return "Średni"; | ||||
|       return "Wysoki"; | ||||
|     } | ||||
|     function clsFor(v) { | ||||
|       v = Number(v); | ||||
|       if (v <= 3) return "sens-low"; | ||||
|       if (v <= 7) return "sens-mid"; | ||||
|       return "sens-high"; | ||||
|     } | ||||
|     function update() { | ||||
|       value.textContent = `(${slider.value})`; | ||||
|       badge.textContent = labelFor(slider.value); | ||||
|       badge.classList.remove("sens-low","sens-mid","sens-high"); | ||||
|       badge.classList.add(clsFor(slider.value)); | ||||
|     } | ||||
|     slider.addEventListener("input", update); | ||||
|     slider.addEventListener("change", update); | ||||
|     update(); | ||||
|   })(); | ||||
| })(); | ||||
							
								
								
									
										43
									
								
								static/js/categories_autosave.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								static/js/categories_autosave.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| (function () { | ||||
|     const $$ = (sel, ctx = document) => Array.from(ctx.querySelectorAll(sel)); | ||||
|     const $ = (sel, ctx = document) => ctx.querySelector(sel); | ||||
|  | ||||
|     const saveCategories = async (listId, ids, names, listTitle) => { | ||||
|         try { | ||||
|             const res = await fetch(`/admin/edit_categories/${listId}/save`, { | ||||
|                 method: 'POST', | ||||
|                 headers: { 'Content-Type': 'application/json' }, | ||||
|                 body: JSON.stringify({ category_ids: ids }) | ||||
|             }); | ||||
|             const data = await res.json().catch(() => ({})); | ||||
|             if (!res.ok || !data.ok) throw new Error(data.error || 'save_failed'); | ||||
|  | ||||
|             const cats = names.length ? names.join(', ') : 'brak'; | ||||
|             showToast(`Zapisano kategorie [${cats}] dla listy <b>${listTitle}</b>`, 'success'); | ||||
|         } catch (err) { | ||||
|             console.error('Autosave error:', err); | ||||
|             showToast(`Błąd zapisu kategorii dla listy <b>${listTitle}</b>`, 'danger'); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     const timers = new Map(); | ||||
|     const debounce = (key, fn, delay = 300) => { | ||||
|         clearTimeout(timers.get(key)); | ||||
|         timers.set(key, setTimeout(fn, delay)); | ||||
|     }; | ||||
|  | ||||
|     $$('.form-select[name^="categories_"]').forEach(select => { | ||||
|         const listId = select.getAttribute('data-list-id') || select.name.replace('categories_', ''); | ||||
|         const listTitle = select.closest('tr')?.querySelector('td a')?.textContent.trim() || `#${listId}`; | ||||
|  | ||||
|         select.addEventListener('change', () => { | ||||
|             const selectedOptions = Array.from(select.options).filter(o => o.selected); | ||||
|             const ids = selectedOptions.map(o => o.value);              // <-- ID | ||||
|             const names = selectedOptions.map(o => o.textContent.trim()); | ||||
|             debounce(listId, () => saveCategories(listId, ids, names, listTitle)); | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     const fallback = $('#fallback-save-btn'); | ||||
|     if (fallback) fallback.classList.add('d-none'); | ||||
| })(); | ||||
							
								
								
									
										18
									
								
								static/js/category_modal.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								static/js/category_modal.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| document.addEventListener('DOMContentLoaded', () => { | ||||
|     document.querySelectorAll('#categoriesModal .category-suggestion').forEach(btn => { | ||||
|         btn.addEventListener('click', () => { | ||||
|             const select = document.getElementById('category_id'); | ||||
|             if (!select) return; | ||||
|  | ||||
|             select.value = btn.dataset.catId || ''; | ||||
|             const form = btn.closest('form'); | ||||
|             if (form) { | ||||
|                 if (typeof form.requestSubmit === 'function') { | ||||
|                     form.requestSubmit(); | ||||
|                 } else { | ||||
|                     form.submit(); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
|     }); | ||||
| }); | ||||
							
								
								
									
										179
									
								
								static/js/chart_controls.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										179
									
								
								static/js/chart_controls.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,179 @@ | ||||
| // chart_controls.js | ||||
| // Logika UI: wybór zakresu, przełączanie dzienny/miesięczny, kategorie, show_all. | ||||
| // Współpracuje z window.loadExpenses (z expense_chart.js). | ||||
| document.addEventListener("DOMContentLoaded", function () { | ||||
|     const toggleMonthlySplit = document.getElementById("toggleMonthlySplit"); | ||||
|     const toggleDailySplit = document.getElementById("toggleDailySplit"); | ||||
|     const toggleCategory = document.getElementById("toggleCategorySplit"); | ||||
|     const startDateInput = document.getElementById("startDate"); | ||||
|     const endDateInput = document.getElementById("endDate"); | ||||
|     const customRangeBtn = document.getElementById("customRangeBtn"); | ||||
|     const showAllCheckbox = document.getElementById("showAllLists"); | ||||
|  | ||||
|     // pomocnicze | ||||
|     const iso = (d) => d.toISOString().split("T")[0]; | ||||
|     const today = () => new Date(); | ||||
|     const daysAgo = (n) => { const d = new Date(); d.setDate(d.getDate() - n); return d; }; | ||||
|  | ||||
|     function setActiveTimeSplit(active) { | ||||
|         const on = (btn) => { btn.classList.add("btn-primary"); btn.classList.remove("btn-outline-light"); btn.setAttribute("aria-pressed", "true"); }; | ||||
|         const off = (btn) => { btn.classList.remove("btn-primary"); btn.classList.add("btn-outline-light"); btn.setAttribute("aria-pressed", "false"); }; | ||||
|         if (active === "monthly") { on(toggleMonthlySplit); off(toggleDailySplit); } | ||||
|         else { on(toggleDailySplit); off(toggleMonthlySplit); } | ||||
|     } | ||||
|     function isDailyActive() { return toggleDailySplit?.classList.contains("btn-primary"); } | ||||
|  | ||||
|     // ——— KLUCZOWE: jedno miejsce, które przeładowuje wykres zgodnie z aktualnym trybem ——— | ||||
|     function reloadRespectingSplit(preferredRange = null) { | ||||
|         // preferredRange używamy dla przycisków typu monthly/quarterly/halfyearly/yearly | ||||
|         const sd = startDateInput?.value || null; | ||||
|         const ed = endDateInput?.value || null; | ||||
|  | ||||
|         if (isDailyActive()) { | ||||
|             // Dzienny ZAWSZE z datami (fallback: ostatnie 30 dni), bo inaczej backend spadnie na monthly | ||||
|             const _sd = sd && ed ? sd : iso(daysAgo(30)); | ||||
|             const _ed = sd && ed ? ed : iso(today()); | ||||
|             window.loadExpenses("daily", _sd, _ed); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Miesięczny | ||||
|         if (sd && ed) { | ||||
|             window.loadExpenses("monthly", sd, ed); | ||||
|         } else if (preferredRange) { | ||||
|             window.loadExpenses(preferredRange); | ||||
|         } else { | ||||
|             window.loadExpenses("monthly"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // ——— Przełączniki czasu ——— | ||||
|     toggleMonthlySplit?.addEventListener("click", () => { | ||||
|         setActiveTimeSplit("monthly"); | ||||
|         reloadRespectingSplit("monthly"); | ||||
|     }); | ||||
|  | ||||
|     toggleDailySplit?.addEventListener("click", () => { | ||||
|         setActiveTimeSplit("daily"); | ||||
|         reloadRespectingSplit(); | ||||
|     }); | ||||
|  | ||||
|     // ——— Podział na kategorie ——— | ||||
|     toggleCategory?.addEventListener("click", function () { | ||||
|         const active = this.classList.contains("btn-primary"); | ||||
|         if (active) { | ||||
|             this.classList.remove("btn-primary"); | ||||
|             this.classList.add("btn-outline-light"); | ||||
|             this.setAttribute("aria-pressed", "false"); | ||||
|             this.textContent = "Przełącz na kategorie"; | ||||
|             window.setCategorySplit(false); | ||||
|         } else { | ||||
|             this.classList.add("btn-primary"); | ||||
|             this.classList.remove("btn-outline-light"); | ||||
|             this.setAttribute("aria-pressed", "true"); | ||||
|             this.textContent = "Przełącz na sumy"; | ||||
|             window.setCategorySplit(true); | ||||
|         } | ||||
|  | ||||
|         // porzucenie zakresu | ||||
|         document.querySelectorAll("#chartTab .range-btn").forEach(b => b.classList.remove("active")); | ||||
|         reloadRespectingSplit(); | ||||
|     }); | ||||
|  | ||||
|  | ||||
|     // ——— Własny zakres ——— | ||||
|     customRangeBtn?.addEventListener("click", function () { | ||||
|         const sd = startDateInput?.value; | ||||
|         const ed = endDateInput?.value; | ||||
|         if (!(sd && ed)) return alert("Proszę wybrać obie daty!"); | ||||
|         reloadRespectingSplit(); | ||||
|     }); | ||||
|  | ||||
|     // ——— Predefiniowane zakresy pod wykresem ——— | ||||
|     document.querySelectorAll("#chartTab .range-btn").forEach((btn) => { | ||||
|         btn.addEventListener("click", function () { | ||||
|             document.querySelectorAll("#chartTab .range-btn").forEach((b) => b.classList.remove("active")); | ||||
|             this.classList.add("active"); | ||||
|             const r = this.getAttribute("data-range"); // last30days/currentmonth/monthly/quarterly/halfyearly/yearly | ||||
|  | ||||
|             // Zakresy kubełkowane – bez start/end, bez "daily" | ||||
|             if (["monthly", "quarterly", "halfyearly", "yearly"].includes(r)) { | ||||
|                 if (startDateInput) startDateInput.value = ""; | ||||
|                 if (endDateInput) endDateInput.value = ""; | ||||
|                 window.loadExpenses(r); // => /expenses_data?range=monthly|quarterly|halfyearly|yearly | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if (r === "currentmonth") { | ||||
|                 const t = today(); | ||||
|                 const first = new Date(t.getFullYear(), t.getMonth(), 1); | ||||
|                 if (isDailyActive()) { | ||||
|                     window.loadExpenses("daily", iso(first), iso(t)); | ||||
|                 } else { | ||||
|                     window.loadExpenses("monthly", iso(first), iso(t)); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|             if (r === "last30days") { | ||||
|                 if (isDailyActive()) { | ||||
|                     window.loadExpenses("daily", iso(daysAgo(30)), iso(today())); | ||||
|                 } else { | ||||
|                     window.loadExpenses("last30days"); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             // reset pickera | ||||
|             if (startDateInput) startDateInput.value = ""; | ||||
|             if (endDateInput) endDateInput.value = ""; | ||||
|  | ||||
|             reloadRespectingSplit(r); | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     // ——— KATEGORIE (🌐 Wszystkie + pojedyncze) ——— | ||||
|     document.querySelectorAll(".category-filter").forEach((btn) => { | ||||
|         btn.addEventListener("click", function () { | ||||
|             // UI: podmień podświetlenie | ||||
|             document.querySelectorAll(".category-filter").forEach(b => { | ||||
|                 b.classList.remove("btn-success"); | ||||
|                 b.classList.add("btn-outline-light"); | ||||
|             }); | ||||
|             this.classList.add("btn-success"); | ||||
|             this.classList.remove("btn-outline-light"); | ||||
|  | ||||
|             // Zapisz filtr kategorii do globalnej zmiennej, którą odczytuje expense_chart.js | ||||
|             const cid = this.getAttribute("data-category-id") || ""; | ||||
|             window.selectedCategoryId = cid; | ||||
|  | ||||
|             // I ważne: przeładuj zgodnie z aktualnym trybem (to naprawia Twój przypadek #1) | ||||
|             reloadRespectingSplit(); | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     // ——— SHOW ALL (Uwzględnij listy udostępnione/publiczne) ——— | ||||
|     showAllCheckbox?.addEventListener("change", () => { | ||||
|         reloadRespectingSplit(); | ||||
|     }); | ||||
|  | ||||
|  | ||||
|     // ——— Inicjalizacja ——— | ||||
|     // Podpowiedź dat do inputów | ||||
|  | ||||
|  | ||||
|     //if (startDateInput && endDateInput) { | ||||
|     //    startDateInput.value = iso(daysAgo(7)); | ||||
|     //    endDateInput.value = iso(today()); | ||||
|     //} | ||||
|  | ||||
|     if (startDateInput && endDateInput) { | ||||
|         const now = new Date(); | ||||
|         const startOfMonth = new Date(now.getFullYear(), now.getMonth(), 1); | ||||
|  | ||||
|         startDateInput.value = iso(startOfMonth); | ||||
|         endDateInput.value = iso(now); | ||||
|     } | ||||
|  | ||||
|     setActiveTimeSplit("daily"); | ||||
|     reloadRespectingSplit(); | ||||
| }); | ||||
							
								
								
									
										67
									
								
								static/js/download_chart.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								static/js/download_chart.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | ||||
| // download_chart.js — eksport PNG z ciemnym tłem (tymczasowo), bez wielokrotnego bindowania | ||||
|  | ||||
| document.addEventListener("DOMContentLoaded", () => { | ||||
|     const dlBtn = document.getElementById("downloadMainChartBtn"); | ||||
|     if (!dlBtn) return; | ||||
|  | ||||
|     // helper: bezpieczna nazwa pliku | ||||
|     const sanitize = (s) => | ||||
|         (s || "") | ||||
|             .normalize("NFD").replace(/[\u0300-\u036f]/g, "") | ||||
|             .replace(/[^a-zA-Z0-9-_]+/g, "_") | ||||
|             .replace(/_+/g, "_").replace(/^_+|_+$/g, ""); | ||||
|  | ||||
|     // helper: eksport z tymczasowym tłem | ||||
|     const exportChartPNG = (chart, bgColor = "#1e1e1e") => { | ||||
|         const canvas = chart.canvas; | ||||
|         const ctx = canvas.getContext("2d"); | ||||
|  | ||||
|         // 1) zapisz obraz | ||||
|         const snapshot = ctx.getImageData(0, 0, canvas.width, canvas.height); | ||||
|  | ||||
|         // 2) podłóż tło pod istniejący rysunek | ||||
|         ctx.save(); | ||||
|         ctx.globalCompositeOperation = "destination-over"; | ||||
|         ctx.fillStyle = bgColor; | ||||
|         ctx.fillRect(0, 0, canvas.width, canvas.height); | ||||
|         ctx.restore(); | ||||
|  | ||||
|         // 3) wygeneruj PNG | ||||
|         const dataUrl = chart.toBase64Image("image/png", 1.0); | ||||
|  | ||||
|         // 4) przywróć pierwotny obraz (transparentny) | ||||
|         ctx.putImageData(snapshot, 0, 0); | ||||
|  | ||||
|         return dataUrl; | ||||
|     }; | ||||
|  | ||||
|     // jednorazowe bindowanie click | ||||
|     if (!dlBtn.dataset.bound) { | ||||
|         dlBtn.addEventListener("click", () => { | ||||
|             const chart = window.expensesChart || Chart.getChart(document.getElementById("expensesChart")); | ||||
|             if (!chart) return; | ||||
|  | ||||
|             // nazwa: zakres + timestamp | ||||
|             const now = new Date(); | ||||
|             const pad = (n) => String(n).padStart(2, "0"); | ||||
|             const stamp = `${now.getFullYear()}-${pad(now.getMonth() + 1)}-${pad(now.getDate())}_${pad(now.getHours())}-${pad(now.getMinutes())}-${pad(now.getSeconds())}`; | ||||
|             const rangeLabel = document.getElementById("chartRangeLabel")?.textContent || ""; | ||||
|             const filename = `wydatki-${sanitize(rangeLabel)}-${stamp}.png`; | ||||
|  | ||||
|             // (opcjonalnie) upewnij się, że layout jest świeży | ||||
|             chart.resize(); | ||||
|             chart.update("none"); | ||||
|  | ||||
|             const a = document.createElement("a"); | ||||
|             a.href = exportChartPNG(chart, "#1e1e1e"); // tu ustawiasz kolor tła eksportu | ||||
|             a.download = filename; | ||||
|             a.click(); | ||||
|         }); | ||||
|         dlBtn.dataset.bound = "1"; | ||||
|     } | ||||
|  | ||||
|     // aktywuj przycisk, gdy wykres istnieje | ||||
|     const enableIfReady = () => { dlBtn.disabled = !window.expensesChart; }; | ||||
|     document.addEventListener("expensesChart:ready", enableIfReady); | ||||
|     enableIfReady(); | ||||
| }); | ||||
| @@ -1,174 +1,150 @@ | ||||
| // expense_chart.js | ||||
| // Czyste generowanie wykresu + publiczne API: window.loadExpenses, window.setCategorySplit | ||||
| // Współpracuje z backendem /expenses_data (range_type, start/end, by_category) – patrz app.py :contentReference[oaicite:3]{index=3} | ||||
|  | ||||
| document.addEventListener("DOMContentLoaded", function () { | ||||
|     let expensesChart = null; | ||||
|     let categorySplit = true; | ||||
|     const rangeLabel = document.getElementById("chartRangeLabel"); | ||||
|     let categorySplit = false; // domyślnie wykres całościowy; przycisk w HTML startuje z aria-pressed="false" | ||||
|  | ||||
|     const rangeLabel = document.getElementById("chartRangeLabel"); | ||||
|     const showAllCheckbox = document.getElementById("showAllLists"); | ||||
|     const ctx = document.getElementById("expensesChart")?.getContext("2d"); | ||||
|  | ||||
|     // Pomocnicze | ||||
|     const iso = (d) => d.toISOString().split("T")[0]; | ||||
|     const today = () => new Date(); | ||||
|     const daysAgo = (n) => { const d = new Date(); d.setDate(d.getDate() - n); return d; }; | ||||
|  | ||||
|     // Jeśli ktoś nie wstrzyknął globalnie selectedCategoryId (np. przez inny widok), | ||||
|     // zapewniamy istnienie zmiennej: | ||||
|     if (typeof window.selectedCategoryId === "undefined") { | ||||
|         window.selectedCategoryId = ""; | ||||
|     } | ||||
|  | ||||
|     function loadExpenses(range = "currentmonth", startDate = null, endDate = null) { | ||||
|         let url = '/expenses_data?range=' + range; | ||||
|     // Ustawia tryb podziału na kategorie, bez odświeżania (kontroler zadzwoni potem w loadExpenses) | ||||
|     function setCategorySplit(on) { | ||||
|         categorySplit = !!on; | ||||
|     } | ||||
|  | ||||
|         const showAllCheckbox = document.getElementById("showAllLists"); | ||||
|     // Budowa URL dla /expenses_data zgodnie z backendem (range/start/end/show_all/category_id/by_category) :contentReference[oaicite:4]{index=4} | ||||
|     function buildUrl(range, startDate, endDate) { | ||||
|         let url = `/expenses_data?range=${encodeURIComponent(range)}`; | ||||
|  | ||||
|         // show_all | ||||
|         if (showAllCheckbox) { | ||||
|             url += showAllCheckbox.checked ? '&show_all=true' : '&show_all=false'; | ||||
|             url += showAllCheckbox.checked ? "&show_all=true" : "&show_all=false"; | ||||
|         } else { | ||||
|             url += '&show_all=true'; | ||||
|             url += "&show_all=true"; | ||||
|         } | ||||
|  | ||||
|         // daty (dodaj tylko, gdy kompletne) | ||||
|         if (startDate && endDate) { | ||||
|             url += `&start_date=${startDate}&end_date=${endDate}`; | ||||
|             url += `&start_date=${encodeURIComponent(startDate)}&end_date=${encodeURIComponent(endDate)}`; | ||||
|         } | ||||
|  | ||||
|         // filtr kategorii list (z listy, nie "podziału na kategorie" na wykresie) | ||||
|         if (window.selectedCategoryId) { | ||||
|             url += `&category_id=${window.selectedCategoryId}`; | ||||
|             url += `&category_id=${encodeURIComponent(window.selectedCategoryId)}`; | ||||
|         } | ||||
|  | ||||
|         // podział na kategorie na wykresie | ||||
|         if (categorySplit) { | ||||
|             url += '&by_category=true'; | ||||
|             url += "&by_category=true"; | ||||
|         } | ||||
|  | ||||
|         return url; | ||||
|     } | ||||
|  | ||||
|     // Label dla UI | ||||
|     function applyRangeLabel(range, startDate, endDate) { | ||||
|         if (startDate && endDate) { | ||||
|             rangeLabel.textContent = `Widok: własny zakres (${startDate} → ${endDate})`; | ||||
|             return; | ||||
|         } | ||||
|         const map = { | ||||
|             last30days: "Widok: ostatnie 30 dni", | ||||
|             currentmonth: "Widok: bieżący miesiąc", | ||||
|             monthly: "Widok: miesięczne", | ||||
|             quarterly: "Widok: kwartalne", | ||||
|             halfyearly: "Widok: półroczne", | ||||
|             yearly: "Widok: roczne", | ||||
|             daily: "Widok: dzienne", | ||||
|         }; | ||||
|         rangeLabel.textContent = map[range] || "Widok: miesięczne"; | ||||
|     } | ||||
|  | ||||
|     // Publiczne API – kontroler zawsze woła nas z odpowiednim 'range' i (dla daily) z datami. | ||||
|     function loadExpenses(range = "monthly", startDate = null, endDate = null) { | ||||
|         // Naprawa: daily bez dat => ostatnie 30 dni | ||||
|         if (range === "daily" && !(startDate && endDate)) { | ||||
|             startDate = iso(daysAgo(30)); | ||||
|             endDate = iso(today()); | ||||
|         } | ||||
|  | ||||
|         const url = buildUrl(range, startDate, endDate); | ||||
|  | ||||
|         fetch(url, { cache: "no-store" }) | ||||
|             .then(response => response.json()) | ||||
|             .then(data => { | ||||
|                 const ctx = document.getElementById('expensesChart').getContext('2d'); | ||||
|             .then((r) => r.json()) | ||||
|             .then((data) => { | ||||
|                 if (!ctx) return; | ||||
|  | ||||
|                 if (expensesChart) { | ||||
|                     expensesChart.destroy(); | ||||
|                 } | ||||
|                 if (expensesChart) { expensesChart.destroy(); window.expensesChart = null; } | ||||
|  | ||||
|                 //if (expensesChart) expensesChart.destroy(); | ||||
|  | ||||
|                 const tooltipOptions = { | ||||
|                     mode: 'index', | ||||
|                     mode: "index", | ||||
|                     intersect: false, | ||||
|                     callbacks: { | ||||
|                         label: function (context) { | ||||
|                             if (context.parsed.y === 0) { | ||||
|                                 return ''; // pomija kategorie o wartości 0 | ||||
|                             } | ||||
|                             return context.dataset.label + ': ' + context.parsed.y; | ||||
|                         } | ||||
|                     } | ||||
|                             if (context.parsed.y === 0) return ""; | ||||
|                             return (context.dataset.label || "Suma") + ": " + context.parsed.y; | ||||
|                         }, | ||||
|                     }, | ||||
|                 }; | ||||
|  | ||||
|                 if (categorySplit) { | ||||
|                     // Stacked per-kategoria – backend zwraca datasets z labelami kategorii :contentReference[oaicite:6]{index=6} | ||||
|                     expensesChart = new Chart(ctx, { | ||||
|                         type: 'bar', | ||||
|                         data: { labels: data.labels, datasets: data.datasets }, | ||||
|  | ||||
|                         type: "bar", | ||||
|                         data: { labels: data.labels || [], datasets: data.datasets || [] }, | ||||
|                         options: { | ||||
|                             responsive: true, | ||||
|                             plugins: { | ||||
|                                 tooltip: tooltipOptions, | ||||
|                                 legend: { position: 'top' } | ||||
|                             }, | ||||
|                             scales: { | ||||
|                                 x: { stacked: true }, | ||||
|                                 y: { stacked: true, beginAtZero: true } | ||||
|                             } | ||||
|                         } | ||||
|                             plugins: { tooltip: tooltipOptions, legend: { position: "top" } }, | ||||
|                             scales: { x: { stacked: true }, y: { stacked: true, beginAtZero: true } }, | ||||
|                         }, | ||||
|                     }); | ||||
|                 } else { | ||||
|                     // Całościowo – backend zwraca labels + expenses (sumy) :contentReference[oaicite:7]{index=7} | ||||
|                     expensesChart = new Chart(ctx, { | ||||
|                         type: 'bar', | ||||
|  | ||||
|                         type: "bar", | ||||
|                         data: { | ||||
|                             labels: data.labels, | ||||
|                             labels: data.labels || [], | ||||
|                             datasets: [{ | ||||
|                                 label: 'Suma wydatków [PLN]', | ||||
|                                 data: data.expenses, | ||||
|                                 backgroundColor: '#0d6efd' | ||||
|                             }] | ||||
|                                 label: "Suma wydatków [PLN]", | ||||
|                                 data: data.expenses || [], | ||||
|                             }], | ||||
|                         }, | ||||
|                         options: { | ||||
|                             responsive: true, | ||||
|                             plugins: { | ||||
|                                 tooltip: tooltipOptions | ||||
|                             }, | ||||
|                             scales: { y: { beginAtZero: true } } | ||||
|                         } | ||||
|                             plugins: { tooltip: tooltipOptions }, | ||||
|                             scales: { y: { beginAtZero: true } }, | ||||
|                         }, | ||||
|                     }); | ||||
|                 } | ||||
|  | ||||
|                 if (startDate && endDate) { | ||||
|                     rangeLabel.textContent = `Widok: własny zakres (${startDate} → ${endDate})`; | ||||
|                 } else { | ||||
|                     let labelText = ""; | ||||
|                     if (range === "last30days") labelText = "Widok: ostatnie 30 dni"; | ||||
|                     else if (range === "currentmonth") labelText = "Widok: bieżący miesiąc"; | ||||
|                     else if (range === "monthly") labelText = "Widok: miesięczne"; | ||||
|                     else if (range === "quarterly") labelText = "Widok: kwartalne"; | ||||
|                     else if (range === "halfyearly") labelText = "Widok: półroczne"; | ||||
|                     else if (range === "yearly") labelText = "Widok: roczne"; | ||||
|                     rangeLabel.textContent = labelText; | ||||
|                 } | ||||
|                 // na potrzeby otwarciu w modalu | ||||
|                 window.expensesChart = expensesChart; | ||||
|                 document.dispatchEvent(new Event('expensesChart:ready')); | ||||
|  | ||||
|                 applyRangeLabel(range, startDate, endDate); | ||||
|             }) | ||||
|             .catch(error => console.error("Błąd pobierania danych:", error)); | ||||
|             .catch((e) => console.error("Błąd pobierania danych:", e)); | ||||
|     } | ||||
|  | ||||
|     // Udostępnienie globalne, żeby inne skrypty mogły wywołać reload | ||||
|     // Eksport publiczny dla kontrolerów | ||||
|     window.loadExpenses = loadExpenses; | ||||
|  | ||||
|     const toggleBtn = document.getElementById("toggleCategorySplit"); | ||||
|     toggleBtn.addEventListener("click", function () { | ||||
|         categorySplit = !categorySplit; | ||||
|         if (categorySplit) { | ||||
|             this.textContent = "🔵 Pokaż całościowo"; | ||||
|             this.classList.remove("btn-outline-warning"); | ||||
|             this.classList.add("btn-outline-info"); | ||||
|         } else { | ||||
|             this.textContent = "🎨 Pokaż podział na kategorie"; | ||||
|             this.classList.remove("btn-outline-info"); | ||||
|             this.classList.add("btn-outline-warning"); | ||||
|         } | ||||
|         loadExpenses(); | ||||
|     }); | ||||
|  | ||||
|     toggleBtn.textContent = "🔵 Pokaż całościowo"; | ||||
|     toggleBtn.classList.remove("btn-outline-warning"); | ||||
|     toggleBtn.classList.add("btn-outline-info"); | ||||
|  | ||||
|     const startDateInput = document.getElementById("startDate"); | ||||
|     const endDateInput = document.getElementById("endDate"); | ||||
|  | ||||
|     const today = new Date(); | ||||
|     const lastWeek = new Date(today); | ||||
|     lastWeek.setDate(today.getDate() - 7); | ||||
|     const formatDate = (d) => d.toISOString().split('T')[0]; | ||||
|     startDateInput.value = formatDate(lastWeek); | ||||
|     endDateInput.value = formatDate(today); | ||||
|  | ||||
|     document.getElementById('customRangeBtn').addEventListener('click', function () { | ||||
|         const startDate = startDateInput.value; | ||||
|         const endDate = endDateInput.value; | ||||
|         if (startDate && endDate) { | ||||
|             document.querySelectorAll('.range-btn').forEach(b => b.classList.remove('active')); | ||||
|             loadExpenses('custom', startDate, endDate); | ||||
|         } else { | ||||
|             alert("Proszę wybrać obie daty!"); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     document.querySelectorAll('.range-btn').forEach(btn => { | ||||
|         btn.addEventListener('click', function () { | ||||
|             document.querySelectorAll('.range-btn').forEach(b => b.classList.remove('active')); | ||||
|             this.classList.add('active'); | ||||
|             const range = this.getAttribute('data-range'); | ||||
|  | ||||
|             if (range === "currentmonth") { | ||||
|                 const today = new Date(); | ||||
|                 const firstDay = new Date(today.getFullYear(), today.getMonth(), 1); | ||||
|                 const formatDate = (d) => d.toISOString().split('T')[0]; | ||||
|                 loadExpenses('custom', formatDate(firstDay), formatDate(today)); | ||||
|             } else { | ||||
|                 loadExpenses(range); | ||||
|             } | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     // Automatyczne ładowanie danych po przełączeniu na zakładkę Wykres | ||||
|     document.getElementById('chart-tab').addEventListener('shown.bs.tab', function () { | ||||
|         loadExpenses(); | ||||
|     }); | ||||
|  | ||||
|     // Jeśli jesteśmy od razu na zakładce Wykres | ||||
|     if (document.getElementById('chart-tab').classList.contains('active')) { | ||||
|         loadExpenses("currentmonth"); | ||||
|     } | ||||
|     window.setCategorySplit = setCategorySplit; | ||||
| }); | ||||
|   | ||||
| @@ -224,17 +224,17 @@ function toggleVisibility(listId) { | ||||
|       const copyBtn = document.getElementById('copyBtn'); | ||||
|       const toggleBtn = document.getElementById('toggleVisibilityBtn'); | ||||
|  | ||||
|       // URL zawsze widoczny i aktywny | ||||
|       shareUrlSpan.style.display = 'inline'; | ||||
|       shareUrlSpan.textContent = data.share_url; | ||||
|       copyBtn.disabled = false; | ||||
|  | ||||
|       if (data.is_public) { | ||||
|         shareHeader.textContent = '🔗 Udostępnij link:'; | ||||
|         shareUrlSpan.style.display = 'inline'; | ||||
|         shareUrlSpan.textContent = data.share_url; | ||||
|         copyBtn.disabled = false; | ||||
|         shareHeader.textContent = '🔗 Udostępnij link (lista publiczna)'; | ||||
|         toggleBtn.innerHTML = '🙈 Ukryj listę'; | ||||
|       } else { | ||||
|         shareHeader.textContent = '🙈 Lista jest ukryta. Link udostępniania nie zadziała!'; | ||||
|         shareUrlSpan.style.display = 'none'; | ||||
|         copyBtn.disabled = true; | ||||
|         toggleBtn.innerHTML = '👁️ Udostępnij ponownie'; | ||||
|         shareHeader.textContent = '🔗 Udostępnij link (widoczna tylko przez link / uprawnienia)'; | ||||
|         toggleBtn.innerHTML = '🐵 Uczyń publiczną'; | ||||
|       } | ||||
|     }); | ||||
| } | ||||
|   | ||||
							
								
								
									
										254
									
								
								static/js/lists_access.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										254
									
								
								static/js/lists_access.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,254 @@ | ||||
| (function () { | ||||
|     const $ = (s, root = document) => root.querySelector(s); | ||||
|     const $$ = (s, root = document) => Array.from(root.querySelectorAll(s)); | ||||
|  | ||||
|     const filterInput = $('#listFilter'); | ||||
|     const filterCount = $('#filterCount'); | ||||
|     const selectAll = $('#selectAll'); | ||||
|     const bulkTokens = $('#bulkTokens'); | ||||
|     const bulkInput = $('#bulkUsersInput'); | ||||
|     const bulkBtn = $('#bulkAddBtn'); | ||||
|     const datalist = $('#userHints'); | ||||
|  | ||||
|     const unique = (arr) => Array.from(new Set(arr)); | ||||
|     const parseUserText = (txt) => unique((txt || '') | ||||
|         .split(/[\s,;]+/g) | ||||
|         .map(s => s.trim().replace(/^@/, '').toLowerCase()) | ||||
|         .filter(Boolean) | ||||
|     ); | ||||
|  | ||||
|     const selectedListIds = () => | ||||
|         $$('.row-check:checked').map(ch => ch.dataset.listId); | ||||
|  | ||||
|     const visibleRows = () => | ||||
|         $$('#listsTable tbody tr').filter(r => r.style.display !== 'none'); | ||||
|  | ||||
|     // ===== Podpowiedzi (datalist) z DOM-u ===== | ||||
|     (function buildHints() { | ||||
|         const names = new Set(); | ||||
|         $$('.owner-username').forEach(el => names.add(el.dataset.username)); | ||||
|         $$('.permitted-username').forEach(el => names.add(el.dataset.username)); | ||||
|         // również tokeny już wyrenderowane | ||||
|         $$('.token[data-username]').forEach(el => names.add(el.dataset.username)); | ||||
|         datalist.innerHTML = Array.from(names) | ||||
|             .sort((a, b) => a.localeCompare(b)) | ||||
|             .map(u => `<option value="${u}">@${u}</option>`) | ||||
|             .join(''); | ||||
|     })(); | ||||
|  | ||||
|     // ===== Live filter ===== | ||||
|     function applyFilter() { | ||||
|         const q = (filterInput?.value || '').trim().toLowerCase(); | ||||
|         let shown = 0; | ||||
|         $$('#listsTable tbody tr').forEach(tr => { | ||||
|             const hay = `${tr.dataset.id || ''} ${tr.dataset.title || ''} ${tr.dataset.owner || ''}`; | ||||
|             const ok = !q || hay.includes(q); | ||||
|             tr.style.display = ok ? '' : 'none'; | ||||
|             if (ok) shown++; | ||||
|         }); | ||||
|         if (filterCount) filterCount.textContent = shown ? `Widoczne: ${shown}` : 'Brak wyników'; | ||||
|     } | ||||
|     filterInput?.addEventListener('input', applyFilter); | ||||
|     applyFilter(); | ||||
|  | ||||
|     // ===== Select all ===== | ||||
|     selectAll?.addEventListener('change', () => { | ||||
|         visibleRows().forEach(tr => { | ||||
|             const cb = tr.querySelector('.row-check'); | ||||
|             if (cb) cb.checked = selectAll.checked; | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     // ===== Copy share URL ===== | ||||
|     $$('.copy-share').forEach(btn => { | ||||
|         btn.addEventListener('click', async () => { | ||||
|             const url = btn.dataset.url; | ||||
|             try { | ||||
|                 await navigator.clipboard.writeText(url); | ||||
|                 showToast('Skopiowano link udostępnienia', 'success'); | ||||
|             } catch { | ||||
|                 const ta = Object.assign(document.createElement('textarea'), { value: url }); | ||||
|                 document.body.appendChild(ta); ta.select(); document.execCommand('copy'); ta.remove(); | ||||
|                 showToast('Skopiowano link udostępnienia', 'success'); | ||||
|             } | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     // ===== Tokenized users field (global – belka) ===== | ||||
|     function addGlobalToken(username) { | ||||
|         if (!username) return; | ||||
|         const exists = $(`.user-token[data-user="${username}"]`, bulkTokens); | ||||
|         if (exists) return; | ||||
|         const token = document.createElement('span'); | ||||
|         token.className = 'badge rounded-pill text-bg-secondary user-token'; | ||||
|         token.dataset.user = username; | ||||
|         token.innerHTML = `@${username} <button type="button" class="btn btn-sm btn-link p-0 ms-1 text-white">✕</button>`; | ||||
|         token.querySelector('button').addEventListener('click', () => token.remove()); | ||||
|         bulkTokens.appendChild(token); | ||||
|     } | ||||
|     bulkInput?.addEventListener('keydown', (e) => { | ||||
|         if (e.key === 'Enter') { | ||||
|             e.preventDefault(); | ||||
|             parseUserText(bulkInput.value).forEach(addGlobalToken); | ||||
|             bulkInput.value = ''; | ||||
|         } | ||||
|     }); | ||||
|     bulkInput?.addEventListener('change', () => { | ||||
|         parseUserText(bulkInput.value).forEach(addGlobalToken); | ||||
|         bulkInput.value = ''; | ||||
|     }); | ||||
|  | ||||
|     // ===== Bulk grant (z belki) ===== | ||||
|     async function bulkGrant() { | ||||
|         const lists = selectedListIds(); | ||||
|         const users = $$('.user-token', bulkTokens).map(t => t.dataset.user); | ||||
|  | ||||
|         if (!lists.length) { showToast('Zaznacz przynajmniej jedną listę', 'warning'); return; } | ||||
|         if (!users.length) { showToast('Dodaj przynajmniej jednego użytkownika', 'warning'); return; } | ||||
|  | ||||
|         bulkBtn.disabled = true; | ||||
|         bulkBtn.textContent = 'Pracuję…'; | ||||
|  | ||||
|         const url = location.pathname + location.search; | ||||
|         let ok = 0, fail = 0; | ||||
|  | ||||
|         for (const lid of lists) { | ||||
|             for (const u of users) { | ||||
|                 const form = new FormData(); | ||||
|                 form.set('action', 'grant'); | ||||
|                 form.set('target_list_id', lid); | ||||
|                 form.set('grant_username', u); | ||||
|  | ||||
|                 try { | ||||
|                     const res = await fetch(url, { method: 'POST', body: form, credentials: 'same-origin' }); | ||||
|                     if (res.ok) ok++; else fail++; | ||||
|                 } catch { fail++; } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         bulkBtn.disabled = false; | ||||
|         bulkBtn.textContent = '➕ Nadaj dostęp'; | ||||
|  | ||||
|         showToast(`Gotowe. Sukcesy: ${ok}${fail ? `, błędy: ${fail}` : ''}`, fail ? 'danger' : 'success'); | ||||
|         location.reload(); | ||||
|     } | ||||
|     bulkBtn?.addEventListener('click', bulkGrant); | ||||
|  | ||||
|     // ===== Per-row "Access editor" (tokeny + dodawanie) ===== | ||||
|     async function postAction(params) { | ||||
|         const url = location.pathname + location.search; | ||||
|         const form = new FormData(); | ||||
|         for (const [k, v] of Object.entries(params)) form.set(k, v); | ||||
|         const res = await fetch(url, { method: 'POST', body: form, credentials: 'same-origin' }); | ||||
|         return res.ok; | ||||
|     } | ||||
|  | ||||
|     // Delegacja zdarzeń: kliknięcie tokenu = revoke | ||||
|     document.addEventListener('click', async (e) => { | ||||
|         const btn = e.target.closest('.access-editor .token'); | ||||
|         if (!btn) return; | ||||
|  | ||||
|         const wrapper = btn.closest('.access-editor'); | ||||
|         const listId = wrapper?.dataset.listId; | ||||
|         const userId = btn.dataset.userId; | ||||
|         const username = btn.dataset.username; | ||||
|  | ||||
|         if (!listId || !userId) return; | ||||
|  | ||||
|         btn.disabled = true; | ||||
|         btn.classList.add('disabled'); | ||||
|  | ||||
|         const ok = await postAction({ | ||||
|             action: 'revoke', | ||||
|             target_list_id: listId, | ||||
|             revoke_user_id: userId | ||||
|         }); | ||||
|  | ||||
|         if (ok) { | ||||
|             btn.remove(); | ||||
|             const tokens = $$('.token', wrapper); | ||||
|             if (!tokens.length) { | ||||
|                 // pokaż info „brak uprawnień” | ||||
|                 let empty = $('.no-perms', wrapper); | ||||
|                 if (!empty) { | ||||
|                     empty = document.createElement('span'); | ||||
|                     empty.className = 'text-warning small no-perms'; | ||||
|                     empty.textContent = 'Brak dodanych uprawnień.'; | ||||
|                     $('.tokens', wrapper).appendChild(empty); | ||||
|                 } | ||||
|             } | ||||
|             showToast(`Odebrano dostęp: @${username}`, 'success'); | ||||
|         } else { | ||||
|             btn.disabled = false; | ||||
|             btn.classList.remove('disabled'); | ||||
|             showToast(`Nie udało się odebrać dostępu @${username}`, 'danger'); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     // Dodawanie wielu użytkowników per-row | ||||
|     document.addEventListener('click', async (e) => { | ||||
|         const addBtn = e.target.closest('.access-editor .access-add'); | ||||
|         if (!addBtn) return; | ||||
|  | ||||
|         const wrapper = addBtn.closest('.access-editor'); | ||||
|         const listId = wrapper?.dataset.listId; | ||||
|         const input = $('.access-input', wrapper); | ||||
|         if (!listId || !input) return; | ||||
|  | ||||
|         const users = parseUserText(input.value); | ||||
|         if (!users.length) { showToast('Podaj co najmniej jednego użytkownika', 'warning'); return; } | ||||
|  | ||||
|         addBtn.disabled = true; | ||||
|         addBtn.textContent = 'Dodaję…'; | ||||
|  | ||||
|         let okCount = 0, failCount = 0; | ||||
|  | ||||
|         for (const u of users) { | ||||
|             const ok = await postAction({ | ||||
|                 action: 'grant', | ||||
|                 target_list_id: listId, | ||||
|                 grant_username: u | ||||
|             }); | ||||
|             if (ok) { | ||||
|                 okCount++; | ||||
|                 // usuń info „brak uprawnień” | ||||
|                 $('.no-perms', wrapper)?.remove(); | ||||
|                 // dodaj token jeśli nie ma | ||||
|                 const exists = $(`.token[data-username="${u}"]`, wrapper); | ||||
|                 if (!exists) { | ||||
|                     const token = document.createElement('button'); | ||||
|                     token.type = 'button'; | ||||
|                     token.className = 'btn btn-sm btn-outline-secondary rounded-pill token'; | ||||
|                     token.dataset.username = u; | ||||
|                     token.dataset.userId = ''; // nie znamy ID — token nadal klikany, ale bez revoke po ID | ||||
|                     token.title = '@' + u; | ||||
|                     token.innerHTML = `@${u} <span aria-hidden="true">×</span>`; | ||||
|                     $('.tokens', wrapper).appendChild(token); | ||||
|                 } | ||||
|             } else { | ||||
|                 failCount++; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         addBtn.disabled = false; | ||||
|         addBtn.textContent = '➕ Dodaj'; | ||||
|         input.value = ''; | ||||
|  | ||||
|         if (okCount) showToast(`Dodano dostęp: ${okCount} użytk.`, 'success'); | ||||
|         if (failCount) showToast(`Błędy przy dodawaniu: ${failCount}`, 'danger'); | ||||
|  | ||||
|         // Odśwież, by mieć poprawne user_id w tokenach (backend wie lepiej) | ||||
|         if (okCount) location.reload(); | ||||
|     }); | ||||
|  | ||||
|     // Enter w polu per-row = zadziałaj jak przycisk | ||||
|     document.addEventListener('keydown', (e) => { | ||||
|         const inp = e.target.closest('.access-editor .access-input'); | ||||
|         if (inp && e.key === 'Enter') { | ||||
|             e.preventDefault(); | ||||
|             const btn = inp.closest('.access-editor')?.querySelector('.access-add'); | ||||
|             btn?.click(); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| })(); | ||||
							
								
								
									
										118
									
								
								static/js/modal_chart.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								static/js/modal_chart.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| // modal_chart.js — final: kopiuje kolory z oryginałów, bez fallbacków i bez debugów | ||||
|  | ||||
| function openChartFullscreen(sourceChartIdOrKey, title) { | ||||
|     const modalEl = document.getElementById("chartFullscreenModal"); | ||||
|     const canvas = document.getElementById("chartFullscreenCanvas"); | ||||
|     const titleEl = document.getElementById("chartModalTitle"); | ||||
|     if (titleEl) titleEl.textContent = title || "Wykres"; | ||||
|  | ||||
|     // Znajdź wykres źródłowy (po elemencie, id Chart.js lub globalu) | ||||
|     const srcEl = document.getElementById(sourceChartIdOrKey); | ||||
|     const srcChart = | ||||
|         (srcEl && Chart.getChart(srcEl)) || | ||||
|         Chart.getChart(sourceChartIdOrKey) || | ||||
|         window[sourceChartIdOrKey] || | ||||
|         window.expensesChart || | ||||
|         null; | ||||
|  | ||||
|     if (!srcChart) { | ||||
|         bootstrap.Modal.getOrCreateInstance(modalEl).show(); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Skopiuj labels i datasets 1:1 (tylko bezpieczne klucze, żeby nie przenosić referencji Chart.js) | ||||
|     const safeDataset = (d) => { | ||||
|         const out = { | ||||
|             // dane i opis | ||||
|             label: d.label, | ||||
|             data: Array.isArray(d.data) ? d.data.slice() : [], | ||||
|             type: d.type, | ||||
|             // kolory / styl — dokładnie z oryginału, jeśli były | ||||
|             backgroundColor: d.backgroundColor, | ||||
|             borderColor: d.borderColor, | ||||
|             borderWidth: d.borderWidth, | ||||
|             borderSkipped: d.borderSkipped, | ||||
|             // stacking / kolejność | ||||
|             stack: d.stack, | ||||
|             order: d.order, | ||||
|             // wszystko co może być ważne dla Twoich barów/konfiguracji | ||||
|             parsing: d.parsing, | ||||
|             indexAxis: d.indexAxis, | ||||
|         }; | ||||
|         // usuń klucze undefined (Chart.js lubi czyste configi) | ||||
|         Object.keys(out).forEach((k) => out[k] === undefined && delete out[k]); | ||||
|         return out; | ||||
|     }; | ||||
|  | ||||
|     const freshData = { | ||||
|         labels: Array.isArray(srcChart.data?.labels) ? srcChart.data.labels.slice() : [], | ||||
|         datasets: (srcChart.data?.datasets || []).map(safeDataset), | ||||
|     }; | ||||
|  | ||||
|     // Typ wykresu z oryginału (np. "bar") | ||||
|     const chartType = (srcChart.config && srcChart.config.type) || "bar"; | ||||
|  | ||||
|     // Minimalne, bezpieczne opcje: responsywność + stacking + orientacja | ||||
|     const scx = srcChart.config?.options?.scales?.x || {}; | ||||
|     const scy = srcChart.config?.options?.scales?.y || {}; | ||||
|  | ||||
|     const freshOptions = { | ||||
|         responsive: true, | ||||
|         maintainAspectRatio: false, | ||||
|         // jeżeli oryginał miał pion/poziom, zachowaj | ||||
|         indexAxis: srcChart.config?.options?.indexAxis || "x", | ||||
|         // nie kopiujemy całych pluginów (unikamy referencji) — domyślne legend/tooltip są OK | ||||
|         plugins: {}, | ||||
|         scales: { | ||||
|             x: { stacked: !!scx.stacked }, | ||||
|             y: { stacked: !!scy.stacked, beginAtZero: scy.beginAtZero !== false }, | ||||
|         }, | ||||
|     }; | ||||
|  | ||||
|     // Helper: zniszcz wykres na canvasie modala, jeśli istnieje | ||||
|     const destroyOnCanvas = () => { | ||||
|         if (canvas._chartInstance) { | ||||
|             try { canvas._chartInstance.destroy(); } catch { } | ||||
|             canvas._chartInstance = null; | ||||
|         } | ||||
|         const existing = Chart.getChart(canvas); | ||||
|         if (existing) { | ||||
|             try { existing.destroy(); } catch { } | ||||
|         } | ||||
|     }; | ||||
|     destroyOnCanvas(); | ||||
|  | ||||
|     // Po pokazaniu modala twórz wykres (gdy ma już wymiary) | ||||
|     const onShown = () => { | ||||
|         destroyOnCanvas(); | ||||
|         const ctx = canvas.getContext("2d"); | ||||
|         canvas._chartInstance = new Chart(ctx, { | ||||
|             type: chartType, | ||||
|             data: freshData, | ||||
|             options: freshOptions, | ||||
|         }); | ||||
|         // lekki nudge layoutu | ||||
|         requestAnimationFrame(() => { | ||||
|             canvas._chartInstance.resize(); | ||||
|             canvas._chartInstance.update(); | ||||
|         }); | ||||
|     }; | ||||
|  | ||||
|     const onHidden = () => { destroyOnCanvas(); }; | ||||
|  | ||||
|     const modal = bootstrap.Modal.getOrCreateInstance(modalEl); | ||||
|     modalEl.addEventListener("shown.bs.modal", onShown, { once: true }); | ||||
|     modalEl.addEventListener("hidden.bs.modal", onHidden, { once: true }); | ||||
|  | ||||
|     modal.show(); | ||||
| } | ||||
|  | ||||
| // Odblokuj ⛶ gdy bazowy wykres gotowy | ||||
| document.addEventListener("expensesChart:ready", () => { | ||||
|     const b = document.getElementById("openFsBtn"); | ||||
|     if (b) b.disabled = false; | ||||
| }); | ||||
| document.addEventListener("DOMContentLoaded", () => { | ||||
|     const b = document.getElementById("openFsBtn"); | ||||
|     if (b && window.expensesChart) b.disabled = false; | ||||
| }); | ||||
							
								
								
									
										57
									
								
								static/js/receipt_crop.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								static/js/receipt_crop.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| (function () { | ||||
|     const configs = (window.CROP_CONFIGS && Array.isArray(window.CROP_CONFIGS)) | ||||
|         ? window.CROP_CONFIGS | ||||
|         : (window.CROP_CONFIG ? [window.CROP_CONFIG] : []); | ||||
|  | ||||
|     if (!configs.length) return; | ||||
|  | ||||
|     document.addEventListener("DOMContentLoaded", function () { | ||||
|         configs.forEach((cfg) => initCropperSet(cfg)); | ||||
|     }); | ||||
|  | ||||
|     function initCropperSet(cfg) { | ||||
|         const { | ||||
|             modalId, | ||||
|             imageId, | ||||
|             spinnerId, | ||||
|             saveBtnId, | ||||
|             endpoint | ||||
|         } = cfg || {}; | ||||
|  | ||||
|         const cropModal = document.getElementById(modalId); | ||||
|         const cropImage = document.getElementById(imageId); | ||||
|         const spinner = document.getElementById(spinnerId); | ||||
|         const saveButton = document.getElementById(saveBtnId); | ||||
|  | ||||
|         if (!cropModal || !cropImage || !spinner || !saveButton) return; | ||||
|  | ||||
|         let cropper; | ||||
|         let currentReceiptId; | ||||
|         const currentEndpoint = endpoint; | ||||
|  | ||||
|         cropModal.addEventListener("shown.bs.modal", function (event) { | ||||
|             const button = event.relatedTarget; | ||||
|             const baseSrc = button?.getAttribute("data-img-src") || ""; | ||||
|             const ver = button?.getAttribute("data-version") || Date.now(); | ||||
|             const sep = baseSrc.includes("?") ? "&" : "?"; | ||||
|             cropImage.src = baseSrc + sep + "cb=" + ver; | ||||
|  | ||||
|             currentReceiptId = button?.getAttribute("data-receipt-id"); | ||||
|  | ||||
|             document.querySelectorAll('.cropper-container').forEach(e => e.remove()); | ||||
|             if (cropper && cropper.destroy) cropper.destroy(); | ||||
|             cropImage.onload = () => { cropper = cropUtils.initCropper(cropImage); }; | ||||
|         }); | ||||
|  | ||||
|         cropModal.addEventListener("hidden.bs.modal", function () { | ||||
|             cropUtils.cleanUpCropper(cropImage, cropper); | ||||
|             cropper = null; | ||||
|         }); | ||||
|  | ||||
|         saveButton.addEventListener("click", function () { | ||||
|             if (!cropper) return; | ||||
|             spinner.classList.remove("d-none"); | ||||
|             cropUtils.handleCrop(currentEndpoint, currentReceiptId, cropper, spinner); | ||||
|         }); | ||||
|     } | ||||
| })(); | ||||
| @@ -1,39 +0,0 @@ | ||||
| (function () { | ||||
|     document.addEventListener("DOMContentLoaded", function () { | ||||
|         const cropModal = document.getElementById("userCropModal"); | ||||
|         const cropImage = document.getElementById("userCropImage"); | ||||
|         const spinner = document.getElementById("userCropLoading"); | ||||
|         const saveButton = document.getElementById("userSaveCrop"); | ||||
|  | ||||
|         if (!cropModal || !cropImage || !spinner || !saveButton) return; | ||||
|  | ||||
|         let cropper; | ||||
|         let currentReceiptId; | ||||
|         const currentEndpoint = "/user_crop_receipt"; | ||||
|  | ||||
|         cropModal.addEventListener("shown.bs.modal", function (event) { | ||||
|             const button = event.relatedTarget; | ||||
|             const imgSrc = button.getAttribute("data-img-src"); | ||||
|             currentReceiptId = button.getAttribute("data-receipt-id"); | ||||
|             cropImage.src = imgSrc; | ||||
|  | ||||
|             document.querySelectorAll('.cropper-container').forEach(e => e.remove()); | ||||
|  | ||||
|             if (cropper) cropper.destroy(); | ||||
|             cropImage.onload = () => { | ||||
|                 cropper = cropUtils.initCropper(cropImage); | ||||
|             }; | ||||
|         }); | ||||
|  | ||||
|         cropModal.addEventListener("hidden.bs.modal", function () { | ||||
|             cropUtils.cleanUpCropper(cropImage, cropper); | ||||
|             cropper = null; | ||||
|         }); | ||||
|  | ||||
|         saveButton.addEventListener("click", function () { | ||||
|             if (!cropper) return; | ||||
|             spinner.classList.remove("d-none"); | ||||
|             cropUtils.handleCrop(currentEndpoint, currentReceiptId, cropper, spinner); | ||||
|         }); | ||||
|     }); | ||||
| })(); | ||||
| @@ -11,9 +11,11 @@ | ||||
|   <div class="card-body p-2"> | ||||
|     <div class="d-flex flex-wrap gap-2"> | ||||
|       <a href="{{ url_for('list_users') }}" class="btn btn-outline-light btn-sm">👥 Użytkownicy</a> | ||||
|       <a href="{{ url_for('admin_receipts', id='all') }}" class="btn btn-outline-light btn-sm">📸 Paragony</a> | ||||
|       <a href="{{ url_for('admin_receipts') }}" class="btn btn-outline-light btn-sm">📸 Paragony</a> | ||||
|       <a href="{{ url_for('list_products') }}" class="btn btn-outline-light btn-sm">🛍️ Produkty</a> | ||||
|       <a href="{{ url_for('admin_mass_edit_categories') }}" class="btn btn-outline-light btn-sm">🗂 Kategorie</a> | ||||
|       <a href="{{ url_for('admin_edit_categories') }}" class="btn btn-outline-light btn-sm">🗂 Kategorie</a> | ||||
|       <a href="{{ url_for('admin_lists_access') }}" class="btn btn-outline-light btn-sm">🔐 Uprawnienia</a> | ||||
|       <a href="{{ url_for('admin_settings') }}" class="btn btn-outline-light btn-sm">⚙️ Ustawienia</a> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| @@ -217,7 +219,8 @@ | ||||
|         — <strong>{{ month_str|replace('-', ' / ') }}</strong> | ||||
|         {% endif %} | ||||
|       </h3> | ||||
|       <form method="post" action="{{ url_for('admin_delete_list') }}"> | ||||
|       <form method="post" action="{{ url_for('admin_delete_list') }}" | ||||
|         onsubmit="return confirm('Na pewno usunąć tę listę?')" class="d-inline"> | ||||
|         <div class="table-responsive"> | ||||
|           <table class="table table-dark align-middle sortable"> | ||||
|             <thead> | ||||
| @@ -299,11 +302,6 @@ | ||||
|                       title="Podgląd produktów"> | ||||
|                       👁️ | ||||
|                     </button> | ||||
|                     <form method="post" action="{{ url_for('admin_delete_list') }}" | ||||
|                       onsubmit="return confirm('Na pewno usunąć tę listę?')" class="d-inline"> | ||||
|                       <input type="hidden" name="single_list_id" value="{{ l.id }}"> | ||||
|                       <button type="submit" class="btn btn-sm btn-outline-light" title="Usuń">🗑️</button> | ||||
|                     </form> | ||||
|                   </div> | ||||
|                 </td> | ||||
|               </tr> | ||||
| @@ -354,7 +352,7 @@ | ||||
|       checkboxes.forEach(cb => cb.checked = this.checked); | ||||
|     }); | ||||
|   </script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='preview_list_modal.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='preview_list_modal.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   {% endblock %} | ||||
|  | ||||
|   {% endblock %} | ||||
| @@ -12,25 +12,24 @@ | ||||
| <div class="card bg-dark text-white mb-5"> | ||||
|     <div class="card-body"> | ||||
|         <div class="alert alert-warning border-warning text-dark" role="alert"> | ||||
|             ⚠️ <strong>Uwaga!</strong> Przypisanie więcej niż jednej kategorii do listy może zaburzyć | ||||
|             poprawne zliczanie wydatków, ponieważ wydatki tej listy będą jednocześnie | ||||
|             klasyfikowane do kilku kategorii. | ||||
|             ⚠️ <strong>Uwaga!</strong> Przypisanie więcej niż jednej kategorii do listy może zaburzyć poprawne zliczanie | ||||
|             wydatków. | ||||
|         </div> | ||||
| 
 | ||||
|         <form method="post"> | ||||
|             <div class="card bg-dark text-white mb-5"> | ||||
|         <form method="post" id="mass-edit-form"> | ||||
|             <div class="card bg-dark text-white mb-4"> | ||||
|                 <div class="card-body p-0"> | ||||
|                     <div class="table-responsive"> | ||||
|                         <table class="table table-dark align-middle sortable"> | ||||
|                             <thead> | ||||
|                         <table class="table table-dark align-middle sortable mb-0"> | ||||
|                             <thead class="position-sticky top-0 bg-dark"> | ||||
|                                 <tr> | ||||
|                                     <th scope="col">ID</th> | ||||
|                                     <th scope="col">Nazwa listy</th> | ||||
|                                     <th scope="col">Właściciel</th> | ||||
|                                     <th scope="col">Data utworzenia</th> | ||||
|                                     <th scope="col">Data</th> | ||||
|                                     <th scope="col">Status</th> | ||||
|                                     <th scope="col">Podgląd produktów</th> | ||||
|                                     <th scope="col">Kategorie</th> | ||||
|                                     <th scope="col">Podgląd</th> | ||||
|                                     <th scope="col" style="min-width: 260px;">Kategorie</th> | ||||
|                                 </tr> | ||||
|                             </thead> | ||||
|                             <tbody> | ||||
| @@ -44,22 +43,17 @@ | ||||
|                                     <td> | ||||
|                                         {% if l.owner %} | ||||
|                                         👤 {{ l.owner.username }} ({{ l.owner.id }}) | ||||
|                                         {% else %} | ||||
|                                         - | ||||
|                                         {% endif %} | ||||
|                                         {% else %}-{% endif %} | ||||
|                                     </td> | ||||
|                                     <td>{{ l.created_at.strftime('%Y-%m-%d %H:%M') if l.created_at else '-' }}</td> | ||||
|                                     <td> | ||||
|                                         {% if l.is_archived %}<span | ||||
|                                             class="badge rounded-pill bg-secondary">Archiwalna</span>{% | ||||
|                                         endif %} | ||||
|                                             class="badge rounded-pill bg-secondary me-1">Archiwalna</span>{% endif %} | ||||
|                                         {% if l.is_temporary %}<span | ||||
|                                             class="badge rounded-pill bg-warning text-dark">Tymczasowa</span>{% | ||||
|                                             class="badge rounded-pill bg-warning text-dark me-1">Tymczasowa</span>{% | ||||
|                                         endif %} | ||||
|                                         {% if l.is_public %}<span | ||||
|                                             class="badge rounded-pill bg-success">Publiczna</span>{% else | ||||
|                                         %} | ||||
|                                         <span class="badge rounded-pill bg-dark">Prywatna</span>{% endif %} | ||||
|                                         {% if l.is_public %}<span class="badge rounded-pill bg-success">Publiczna</span> | ||||
|                                         {% else %}<span class="badge rounded-pill bg-dark">Prywatna</span>{% endif %} | ||||
|                                     </td> | ||||
|                                     <td> | ||||
|                                         <button type="button" class="btn btn-sm btn-outline-light preview-btn" | ||||
| @@ -67,24 +61,25 @@ | ||||
|                                             🔍 Zobacz | ||||
|                                         </button> | ||||
|                                     </td> | ||||
|                                     <td style="min-width: 220px;"> | ||||
|                                         <select name="categories_{{ l.id }}" multiple | ||||
|                                             class="form-select tom-dark bg-dark text-white border-secondary rounded"> | ||||
|                                             {% for cat in categories %} | ||||
|                                             <option value="{{ cat.id }}" {% if cat in l.categories %}selected{% endif | ||||
|                                                 %}> | ||||
|                                                 {{ cat.name }} | ||||
|                                             </option> | ||||
|                                             {% endfor %} | ||||
|                                         </select> | ||||
|                                     <td> | ||||
|                                         <div class="d-flex align-items-center gap-2"> | ||||
|                                             <select name="categories_{{ l.id }}" multiple | ||||
|                                                 class="form-select tom-dark bg-dark text-white border-secondary rounded" | ||||
|                                                 data-list-id="{{ l.id }}" | ||||
|                                                 aria-label="Wybierz kategorie dla listy {{ l.id }}"> | ||||
|                                                 {% for cat in categories %} | ||||
|                                                 <option value="{{ cat.id }}" {% if cat in l.categories %}selected{% | ||||
|                                                     endif %}>{{ cat.name }}</option> | ||||
|                                                 {% endfor %} | ||||
|                                             </select> | ||||
|                                         </div> | ||||
|                                     </td> | ||||
|                                 </tr> | ||||
|                                 {% endfor %} | ||||
| 
 | ||||
|                                 {% if lists|length == 0 %} | ||||
|                                 <tr> | ||||
|                                     <td colspan="12" class="text-center py-4"> | ||||
|                                         Brak list zakupowych do wyświetlenia | ||||
|                                     </td> | ||||
|                                     <td colspan="12" class="text-center py-4">Brak list zakupowych do wyświetlenia</td> | ||||
|                                 </tr> | ||||
|                                 {% endif %} | ||||
|                             </tbody> | ||||
| @@ -92,9 +87,9 @@ | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|             <div> | ||||
|                 <button type="submit" class="btn btn-sm btn-outline-light">💾 Zapisz zmiany</button> | ||||
|             </div> | ||||
| 
 | ||||
|             {# Fallback – ukryty przez JS #} | ||||
|             <button type="submit" class="btn btn-sm btn-outline-light" id="fallback-save-btn">💾 Zapisz zmiany</button> | ||||
|         </form> | ||||
|     </div> | ||||
| </div> | ||||
| @@ -120,8 +115,7 @@ | ||||
|             </li> | ||||
|             {% for p in range(1, total_pages + 1) %} | ||||
|             <li class="page-item {% if p == page %}active{% endif %}"> | ||||
|                 <a class="page-link" href="?{{ query_string }}{% if query_string %}&{% endif %}page={{ p }}">{{ | ||||
|                     p }}</a> | ||||
|                 <a class="page-link" href="?{{ query_string }}{% if query_string %}&{% endif %}page={{ p }}">{{ p }}</a> | ||||
|             </li> | ||||
|             {% endfor %} | ||||
|             <li class="page-item {% if page >= total_pages %}disabled{% endif %}"> | ||||
| @@ -132,7 +126,6 @@ | ||||
|     </nav> | ||||
| </div> | ||||
| 
 | ||||
| 
 | ||||
| <!-- Modal podglądu produktów --> | ||||
| <div class="modal fade" id="productPreviewModal" tabindex="-1" aria-labelledby="previewModalLabel" aria-hidden="true"> | ||||
|     <div class="modal-dialog modal-lg modal-dialog-scrollable"> | ||||
| @@ -150,7 +143,9 @@ | ||||
| </div> | ||||
| 
 | ||||
| {% endblock %} | ||||
| 
 | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='preview_list_modal.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='categories_select_admin.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='preview_list_modal.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='categories_select_admin.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='categories_autosave.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
| @@ -117,6 +117,31 @@ | ||||
|           value="{{ request.url_root }}share/{{ list.share_token }}"> | ||||
|       </div> | ||||
|  | ||||
|  | ||||
|       <!-- Dostęp / uprawnienia --> | ||||
|       <div class="mb-4 border-top pt-3 mt-4"> | ||||
|         <h5 class="mb-3">🔐 Użytkownicy z dostępem</h5> | ||||
|  | ||||
|         <a class="btn btn-outline-warning btn-sm mb-3" href="{{ url_for('admin_lists_access', list_id=list.id) }}"> | ||||
|           ⚙️ Edytuj uprawnienia | ||||
|         </a> | ||||
|  | ||||
|         {% if permitted_users %} | ||||
|         <ul class="list-group list-group-flush mb-3"> | ||||
|           {% for u in permitted_users %} | ||||
|           <li | ||||
|             class="list-group-item bg-dark text-white d-flex justify-content-between align-items-center border-secondary"> | ||||
|             <div> | ||||
|               <span class="fw-semibold">@{{ u.username }}</span> | ||||
|             </div> | ||||
|           </li> | ||||
|           {% endfor %} | ||||
|         </ul> | ||||
|         {% else %} | ||||
|         <div class="text-warning small">Brak dodatkowych uprawnień.</div> | ||||
|         {% endif %} | ||||
|       </div> | ||||
|  | ||||
|       <button type="submit" class="btn btn-outline-light btn-sm me-2">💾 Zapisz zmiany</button> | ||||
|     </form> | ||||
|  | ||||
| @@ -276,5 +301,5 @@ | ||||
|  | ||||
| {% endblock %} | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
| @@ -168,8 +168,8 @@ | ||||
| </div> | ||||
|  | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='product_suggestion.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='table_search.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='product_suggestion.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='table_search.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
|  | ||||
| {% endblock %} | ||||
							
								
								
									
										211
									
								
								templates/admin/lists_access.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										211
									
								
								templates/admin/lists_access.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,211 @@ | ||||
| {% extends 'base.html' %} | ||||
| {% block title %}Zarządzanie dostępem do list{% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
| <div class="d-flex justify-content-between align-items-center flex-wrap mb-3"> | ||||
|   <h2 class="mb-2">🔐{% if list_id %} Zarządzanie dostępem listy #{{ list_id }}{% else %} Zarządzanie dostępem do list | ||||
|     {% endif %}</h2> | ||||
|   <div class="d-flex gap-2"> | ||||
|     {% if list_id %} | ||||
|     <a href="{{ url_for('admin_lists_access') }}" class="btn btn-outline-light">Powrót do wszystkich list</a> | ||||
|     {% endif %} | ||||
|     <a href="{{ url_for('admin_panel') }}" class="btn btn-outline-secondary">← Powrót do panelu</a> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
| <!-- STICKY ACTION BAR --> | ||||
| <div id="bulkBar" class="position-sticky top-0 z-3 mb-3" style="backdrop-filter: blur(6px);"> | ||||
|   <div class="card bg-dark border-secondary shadow-sm"> | ||||
|     <div class="card-body py-2 d-flex flex-wrap align-items-center gap-3"> | ||||
|       <div class="d-flex align-items-center gap-2"> | ||||
|         <input id="selectAll" class="form-check-input" type="checkbox" /> | ||||
|         <label for="selectAll" class="form-check-label">Zaznacz wszystko</label> | ||||
|       </div> | ||||
|  | ||||
|       <div class="vr text-secondary"></div> | ||||
|  | ||||
|       <div class="flex-grow-1 d-flex align-items-center gap-2"> | ||||
|         <input id="listFilter" class="form-control form-control-sm bg-dark text-white border-secondary" | ||||
|           placeholder="Szukaj po tytule/ID/właścicielu…" aria-label="Filtruj listy"> | ||||
|         <span class="text-secondary small ms-1" id="filterCount">—</span> | ||||
|       </div> | ||||
|  | ||||
|       <div class="vr text-secondary d-none d-md-block"></div> | ||||
|  | ||||
|       <!-- BULK GRANT --> | ||||
|       <div class="flex-grow-1"> | ||||
|         <div class="input-group input-group-sm"> | ||||
|           <input id="bulkUsersInput" class="form-control bg-dark text-white border-secondary" | ||||
|             placeholder="Podaj użytkowników (po przecinku lub enterach)" list="userHints"> | ||||
|           <button id="bulkAddBtn" class="btn btn-outline-light" type="button">➕ Nadaj dostęp</button> | ||||
|         </div> | ||||
|         <div id="bulkTokens" class="d-flex flex-wrap gap-2 mt-2"></div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
|  | ||||
| <!-- HINTS --> | ||||
| <datalist id="userHints"></datalist> | ||||
|  | ||||
| <div class="card bg-dark text-white mb-5"> | ||||
|   <div class="card-body"> | ||||
|     <form id="statusForm" method="post"> | ||||
|       <input type="hidden" name="action" value="save_changes"> | ||||
|  | ||||
|       <div class="table-responsive"> | ||||
|         <table class="table table-dark align-middle sortable" id="listsTable"> | ||||
|           <thead class="align-middle"> | ||||
|             <tr> | ||||
|               <th scope="col" style="width:36px;"></th> | ||||
|               <th scope="col">ID</th> | ||||
|               <th scope="col">Nazwa listy</th> | ||||
|               <th scope="col">Właściciel</th> | ||||
|               <th scope="col">Utworzono</th> | ||||
|               <th scope="col">Statusy</th> | ||||
|               <th scope="col">Udostępnianie</th> | ||||
|               <th scope="col" style="min-width: 340px;">Uprawnienia</th> | ||||
|             </tr> | ||||
|           </thead> | ||||
|           <tbody> | ||||
|             {% for l in lists %} | ||||
|             <tr data-id="{{ l.id }}" data-title="{{ l.title|lower }}" | ||||
|               data-owner="{{ (l.owner.username if l.owner else '-')|lower }}"> | ||||
|               <td> | ||||
|                 <input class="row-check form-check-input" type="checkbox" data-list-id="{{ l.id }}"> | ||||
|                 <input type="hidden" name="visible_ids" value="{{ l.id }}"> | ||||
|               </td> | ||||
|  | ||||
|               <td class="text-nowrap">{{ l.id }}</td> | ||||
|  | ||||
|               <td class="fw-bold align-middle"> | ||||
|                 <a href="{{ url_for('view_list', list_id=l.id) }}" class="text-white text-decoration-none">{{ l.title | ||||
|                   }}</a> | ||||
|               </td> | ||||
|  | ||||
|               <td> | ||||
|                 {% if l.owner %} | ||||
|                 👤 <span class="owner-username" data-username="{{ l.owner.username }}">@{{ l.owner.username }}</span> | ||||
|                 ({{ l.owner.id }}) | ||||
|                 {% else %}-{% endif %} | ||||
|               </td> | ||||
|  | ||||
|               <td class="text-nowrap">{{ l.created_at.strftime('%Y-%m-%d %H:%M') if l.created_at else '-' }}</td> | ||||
|  | ||||
|               <td style="min-width: 230px;"> | ||||
|                 <div class="form-check form-switch"> | ||||
|                   <input class="form-check-input" type="checkbox" id="pub_{{ l.id }}" name="is_public_{{ l.id }}" {% if | ||||
|                     l.is_public %}checked{% endif %}> | ||||
|                   <label class="form-check-label" for="pub_{{ l.id }}">🌐 Publiczna</label> | ||||
|                 </div> | ||||
|                 <div class="form-check form-switch"> | ||||
|                   <input class="form-check-input" type="checkbox" id="tmp_{{ l.id }}" name="is_temporary_{{ l.id }}" {% | ||||
|                     if l.is_temporary %}checked{% endif %}> | ||||
|                   <label class="form-check-label" for="tmp_{{ l.id }}">⏳ Tymczasowa</label> | ||||
|                 </div> | ||||
|                 <div class="form-check form-switch"> | ||||
|                   <input class="form-check-input" type="checkbox" id="arc_{{ l.id }}" name="is_archived_{{ l.id }}" {% | ||||
|                     if l.is_archived %}checked{% endif %}> | ||||
|                   <label class="form-check-label" for="arc_{{ l.id }}">📦 Archiwalna</label> | ||||
|                 </div> | ||||
|               </td> | ||||
|  | ||||
|               <td style="min-width: 260px;"> | ||||
|                 {% if l.share_token %} | ||||
|                 {% set share_url = url_for('shared_list', token=l.share_token, _external=True) %} | ||||
|                 <div class="d-flex align-items-center gap-2"> | ||||
|                   <div class="flex-grow-1 text-truncate mono small" title="{{ share_url }}">{{ share_url }}</div> | ||||
|                   <button class="btn btn-sm btn-outline-secondary copy-share" type="button" data-url="{{ share_url }}" | ||||
|                     aria-label="Kopiuj link">📋</button> | ||||
|                 </div> | ||||
|                 <div class="text-info small mt-1"> | ||||
|                   {% if l.is_public %}Lista widoczna publicznie{% else %}Dostęp przez link / uprawnienia{% endif %} | ||||
|                 </div> | ||||
|                 {% else %} | ||||
|                 <div class="text-warning small">Brak tokenu</div> | ||||
|                 {% endif %} | ||||
|               </td> | ||||
|  | ||||
|               <td> | ||||
|                 <div class="access-editor" data-list-id="{{ l.id }}"> | ||||
|                   <!-- Tokeny z uprawnieniami --> | ||||
|                   <div class="d-flex flex-wrap gap-2 mb-2 tokens"> | ||||
|                     {% for u in permitted_by_list.get(l.id, []) %} | ||||
|                     <button type="button" class="btn btn-sm btn-outline-secondary rounded-pill token" | ||||
|                       data-user-id="{{ u.id }}" data-username="{{ u.username }}" title="Kliknij, aby odebrać dostęp"> | ||||
|                       @{{ u.username }} <span aria-hidden="true">×</span> | ||||
|                     </button> | ||||
|                     {% endfor %} | ||||
|                     {% if permitted_by_list.get(l.id, [])|length == 0 %} | ||||
|                     <span class="text-warning small no-perms">Brak dodanych uprawnień.</span> | ||||
|                     {% endif %} | ||||
|                   </div> | ||||
|  | ||||
|                   <!-- Dodawanie (wiele na raz) --> | ||||
|                   <div class="input-group input-group-sm"> | ||||
|                     <input type="text" | ||||
|                       class="form-control form-control-sm bg-dark text-white border-secondary access-input" | ||||
|                       placeholder="Dodaj @użytkownika (wiele: przecinki/enter)" list="userHints" | ||||
|                       aria-label="Dodaj użytkowników"> | ||||
|                     <button type="button" class="btn btn-sm btn-outline-light access-add">➕ Dodaj</button> | ||||
|                   </div> | ||||
|                   <div class="text-secondary small mt-1">Kliknij token, aby odebrać dostęp.</div> | ||||
|                 </div> | ||||
|               </td> | ||||
|  | ||||
|             </tr> | ||||
|             {% endfor %} | ||||
|             {% if lists|length == 0 %} | ||||
|             <tr> | ||||
|               <td colspan="8" class="text-center py-4">Brak list do wyświetlenia</td> | ||||
|             </tr> | ||||
|             {% endif %} | ||||
|           </tbody> | ||||
|         </table> | ||||
|       </div> | ||||
|  | ||||
|       <div class="mt-3 d-flex justify-content-end"> | ||||
|         <button type="submit" class="btn btn-sm btn-outline-light">💾 Zapisz zmiany statusów</button> | ||||
|       </div> | ||||
|     </form> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
| {% if not list_id %} | ||||
| <hr> | ||||
| <div class="d-flex justify-content-between align-items-center mt-4"> | ||||
|   <form method="get" class="d-flex align-items-center"> | ||||
|     <label for="per_page" class="me-2">🔢 Pozycji na stronę:</label> | ||||
|     <select id="per_page" name="per_page" class="form-select form-select-sm me-2" | ||||
|       onchange="this.form.page.value = 1; this.form.submit();"> | ||||
|       <option value="25" {% if per_page==25 %}selected{% endif %}>25</option> | ||||
|       <option value="50" {% if per_page==50 %}selected{% endif %}>50</option> | ||||
|       <option value="100" {% if per_page==100 %}selected{% endif %}>100</option> | ||||
|     </select> | ||||
|     <input type="hidden" name="page" value="{{ page }}"> | ||||
|   </form> | ||||
|  | ||||
|   <nav aria-label="Nawigacja stron"> | ||||
|     <ul class="pagination pagination-dark mb-0"> | ||||
|       <li class="page-item {% if page <= 1 %}disabled{% endif %}"> | ||||
|         <a class="page-link" href="?{{ query_string }}{% if query_string %}&{% endif %}page={{ page - 1 }}">«</a> | ||||
|       </li> | ||||
|       {% for p in range(1, total_pages + 1) %} | ||||
|       <li class="page-item {% if p == page %}active{% endif %}"> | ||||
|         <a class="page-link" href="?{{ query_string }}{% if query_string %}&{% endif %}page={{ p }}">{{ p }}</a> | ||||
|       </li> | ||||
|       {% endfor %} | ||||
|       <li class="page-item {% if page >= total_pages %}disabled{% endif %}"> | ||||
|         <a class="page-link" href="?{{ query_string }}{% if query_string %}&{% endif %}page={{ page + 1 }}">»</a> | ||||
|       </li> | ||||
|     </ul> | ||||
|   </nav> | ||||
| </div> | ||||
| {% endif %} | ||||
|  | ||||
| {% endblock %} | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='lists_access.js') }}?v={{ APP_VERSION }}"></script> | ||||
|  | ||||
| {% endblock %} | ||||
| @@ -20,8 +20,8 @@ | ||||
|       {{ (page_filesize / 1024) | round(1) }} kB | ||||
|       {% endif %} | ||||
|     </strong> | ||||
|     | | ||||
|     Łącznie: | ||||
|     {% if not (id != 'all' and (id|string).isdigit()) %} | ||||
|     | Łącznie: | ||||
|     <strong> | ||||
|       {% if total_filesize >= 1024*1024 %} | ||||
|       {{ (total_filesize / 1024 / 1024) | round(2) }} MB | ||||
| @@ -29,12 +29,19 @@ | ||||
|       {{ (total_filesize / 1024) | round(1) }} kB | ||||
|       {% endif %} | ||||
|     </strong> | ||||
|     {% endif %} | ||||
|   </p> | ||||
|  | ||||
|   <div> | ||||
|     {% if id is string and id.isdigit() and id|int > 0 %} | ||||
|     <a href="{{ url_for('admin_receipts', id='all') }}" class="btn btn-outline-light me-2"> | ||||
|       Pokaż wszystkie paragony | ||||
|     </a> | ||||
|     {% else %} | ||||
|     <a href="{{ url_for('recalculate_filesizes_all') }}" class="btn btn-outline-light me-2"> | ||||
|       Przelicz rozmiary plików | ||||
|     </a> | ||||
|     {% endif %} | ||||
|     <a href="{{ url_for('admin_panel') }}" class="btn btn-outline-secondary">← Powrót do panelu</a> | ||||
|   </div> | ||||
| </div> | ||||
| @@ -78,8 +85,12 @@ | ||||
|                 </li> | ||||
|                 <li> | ||||
|                   <a class="dropdown-item" href="#" data-bs-toggle="modal" data-bs-target="#adminCropModal" | ||||
|                     data-img-src="{{ url_for('uploaded_file', filename=r.filename) }}" data-receipt-id="{{ r.id }}" | ||||
|                     data-crop-endpoint="{{ url_for('crop_receipt_admin') }}">✂️ Przytnij</a> | ||||
|                     data-img-src="{{ url_for('uploaded_file', filename=r.filename) }}?v={{ r.version_token or '0' }}" | ||||
|                     data-receipt-id="{{ r.id }}" data-crop-endpoint="{{ url_for('crop_receipt_admin') }}" | ||||
|                     data-version="{{ r.version_token or '0' }}"> | ||||
|                     ✂️ Przytnij | ||||
|                   </a> | ||||
|  | ||||
|                 </li> | ||||
|                 <li> | ||||
|                   <a class="dropdown-item" href="{{ url_for('rename_receipt', receipt_id=r.id) }}">✏️ Zmień nazwę</a> | ||||
| @@ -118,8 +129,8 @@ | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
| {% if id == 'all' %} | ||||
| <hr> | ||||
|  | ||||
| <div class="d-flex justify-content-between align-items-center mt-4"> | ||||
|   <form method="get" class="d-flex align-items-center"> | ||||
|     <label for="per_page" class="me-2">🔢 Pozycji na stronę:</label> | ||||
| @@ -149,7 +160,7 @@ | ||||
|     </ul> | ||||
|   </nav> | ||||
| </div> | ||||
|  | ||||
| {% endif %} | ||||
|  | ||||
| {% if orphan_files and request.path.endswith('/all') %} | ||||
| <hr class="my-4"> | ||||
| @@ -202,9 +213,17 @@ | ||||
| </div> | ||||
|  | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_crop.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='admin_receipt_crop.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_crop_logic.js') }}"></script> | ||||
| <script> | ||||
|   window.CROP_CONFIG = { | ||||
|     modalId: "adminCropModal", | ||||
|     imageId: "adminCropImage", | ||||
|     spinnerId: "adminCropLoading", | ||||
|     saveBtnId: "adminSaveCrop", | ||||
|     endpoint: "/admin/crop_receipt" | ||||
|   }; | ||||
| </script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_crop.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_crop_logic.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
|  | ||||
| {% endblock %} | ||||
							
								
								
									
										145
									
								
								templates/admin/settings.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								templates/admin/settings.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
| {% extends "base.html" %} | ||||
| {% block title %}Ustawienia{% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
| <div class="d-flex justify-content-between align-items-center flex-wrap mb-4"> | ||||
|   <h2 class="mb-2">⚙️ Ustawienia</h2> | ||||
|   <a href="{{ url_for('admin_panel') }}" class="btn btn-outline-secondary">← Powrót do panelu</a> | ||||
| </div> | ||||
|  | ||||
| <form method="post" id="settings-form"> | ||||
|   <div class="card bg-dark text-white mb-4"> | ||||
|     <div class="card-header border-0"> | ||||
|       <strong>🔎 OCR — słowa kluczowe i czułość</strong> | ||||
|     </div> | ||||
|     <div class="card-body"> | ||||
|       <p class="small text-info mb-2"> | ||||
|         Dodaj lokalne frazy (CSV lub JSON), np.: <code>summe, gesamtbetrag, importe total</code> | ||||
|       </p> | ||||
|       <textarea | ||||
|         class="form-control settings-ocr-textarea mb-3" | ||||
|         name="ocr_keywords" | ||||
|         rows="3" | ||||
|         placeholder="suma, razem do zapłaty, total" | ||||
|       >{{ current_ocr }}</textarea> | ||||
|  | ||||
|       <label for="ocr_sensitivity" class="form-label d-flex align-items-center gap-2"> | ||||
|         Poziom czułości OCR | ||||
|         <span id="ocr_sens_badge" class="badge rounded-pill sens-badge">Średni</span> | ||||
|         <span id="ocr_sens_value" class="small">({{ ocr_sensitivity }})</span> | ||||
|       </label> | ||||
|       <input | ||||
|         type="range" | ||||
|         class="form-range" | ||||
|         min="1" | ||||
|         max="10" | ||||
|         step="1" | ||||
|         name="ocr_sensitivity" | ||||
|         id="ocr_sensitivity" | ||||
|         value="{{ ocr_sensitivity }}" | ||||
|       > | ||||
|       <div class="small mt-1"> | ||||
|         <ul class="mb-2 ps-3"> | ||||
|           <li><strong>Zalecane:</strong> <code>5–7</code> (balans dokładności i stabilności).</li> | ||||
|           <li><strong>Niskie (1–3):</strong> szybsze, mniejsza wykrywalność trudnych skanów.</li> | ||||
|           <li><strong>Średnie (4–7):</strong> dobre na większość paragonów — <em>polecane</em>.</li> | ||||
|           <li><strong>Wysokie (8–10):</strong> agresywne binaryzowanie — lepsze dla bladych skanów, | ||||
|             ale większe ryzyko fałszywych trafień i wolniejsze działanie.</li> | ||||
|         </ul> | ||||
|         Tip: jeśli pojawiają się „dziwne” sumy — obniż o 1–2 poziomy. | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <div class="card bg-dark text-white mb-4"> | ||||
|     <div class="card-header border-0 d-flex align-items-center justify-content-between"> | ||||
|       <strong>🎨 Kolory kategorii</strong> | ||||
|       <button type="button" class="btn btn-outline-light btn-sm" id="reset-all">🔄 Wyczyść nadpisania</button> | ||||
|     </div> | ||||
|  | ||||
|     <div class="card-body"> | ||||
|       <div class="row g-3" id="categories-grid"> | ||||
|         {% for c in categories %} | ||||
|         {% set hex_override = overrides.get(c.id) %} | ||||
|         {% set hex_auto = auto_colors[c.id] %} | ||||
|         {% set hex_effective = effective_colors[c.id] %} | ||||
|         <div class="col-12 col-md-6 col-lg-4"> | ||||
|           <label class="form-label d-block mb-2">{{ c.name }}</label> | ||||
|  | ||||
|         <div class="input-group"> | ||||
|         <input | ||||
|         type="color" | ||||
|         class="form-control form-control-color category-color" | ||||
|         name="color_{{ c.id }}" | ||||
|         value="{{ hex_override or '' }}" | ||||
|         data-auto="{{ hex_auto }}" | ||||
|         {% if not hex_override %}data-empty="1"{% endif %} | ||||
|         aria-label="Kolor kategorii {{ c.name }}" | ||||
|         > | ||||
|  | ||||
|         <div class="btn-group" role="group" aria-label="Akcje koloru"> | ||||
|             <button type="button" | ||||
|                     class="btn btn-outline-light btn-sm reset-one" | ||||
|                     data-target="color_{{ c.id }}"> | ||||
|             🔄 Reset | ||||
|             </button> | ||||
|             <button type="button" | ||||
|                     class="btn btn-outline-light btn-sm use-default" | ||||
|                     data-target="color_{{ c.id }}"> | ||||
|             🎯 Przywróć domyślny | ||||
|             </button> | ||||
|         </div> | ||||
|         </div> | ||||
|  | ||||
|           <div class="color-indicators mt-2"> | ||||
|             <div class="indicator"> | ||||
|               <span class="badge text-bg-dark me-2">Efektywny</span> | ||||
|               <span class="bar" data-kind="effective" style="background-color: {{ hex_effective }};"></span> | ||||
|               <span class="hex hex-effective ms-2">{{ hex_effective|upper }}</span> | ||||
|             </div> | ||||
|             <div class="indicator mt-1"> | ||||
|               <span class="badge text-bg-light me-2">Domyślny</span> | ||||
|               <span class="bar" data-kind="auto" style="background-color: {{ hex_auto }};"></span> | ||||
|               <span class="hex hex-auto ms-2">{{ hex_auto|upper }}</span> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         {% endfor %} | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <div class="card bg-dark text-white mb-4"> | ||||
|     <div class="card-header border-0"> | ||||
|       <strong>🔐 Bezpieczeństwo</strong> | ||||
|     </div> | ||||
|     <div class="card-body"> | ||||
|       <label for="max_login_attempts" class="form-label">Limit błędnych logowań (hasło główne)</label> | ||||
|       <input | ||||
|         type="number" | ||||
|         class="form-control" | ||||
|         name="max_login_attempts" | ||||
|         id="max_login_attempts" | ||||
|         min="1" | ||||
|         max="20" | ||||
|         value="{{ max_login_attempts }}" | ||||
|       > | ||||
|       <div class="form-text text-muted"> | ||||
|         Po przekroczeniu limitu IP zostaje tymczasowo zablokowane. | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <div class="mt-4 d-flex"> | ||||
|     <div class="btn-group" role="group" aria-label="Akcje ustawień"> | ||||
|       <button type="submit" class="btn btn-outline-light">💾 Zapisz</button> | ||||
|       <a href="{{ url_for('admin_panel') }}" class="btn btn-outline-light">❌ Anuluj</a> | ||||
|     </div> | ||||
|   </div> | ||||
| </form> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block scripts %} | ||||
| <link rel="stylesheet" href="{{ url_for('static_bp.serve_css', filename='admin_settings.css') }}?v={{ APP_VERSION }}"> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='admin_settings.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
| @@ -115,7 +115,7 @@ | ||||
| </div> | ||||
|  | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='user_management.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='user_management.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -8,28 +8,33 @@ | ||||
|   <link rel="icon" type="image/svg+xml" href="{{ url_for('favicon') }}"> | ||||
|  | ||||
|   {# --- Bootstrap i główny css zawsze --- #} | ||||
|   <link href="{{ url_for('static_bp.serve_css', filename='style.css') }}" rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='bootstrap.min.css') }}" rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css', filename='style.css') }}?v={{ APP_VERSION }}" rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='bootstrap.min.css') }}?v={{ APP_VERSION }}" | ||||
|     rel="stylesheet"> | ||||
|  | ||||
|   {# --- Style CSS ładowane tylko dla niezablokowanych --- #} | ||||
|   {% set exclude_paths = ['/system-auth'] %} | ||||
|   {% if (exclude_paths | select("in", request.path) | list | length == 0) | ||||
|   and has_authorized_cookie | ||||
|   and not is_blocked %} | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='glightbox.min.css') }}" rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='sort_table.min.css') }}" rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='glightbox.min.css') }}?v={{ APP_VERSION }}" | ||||
|     rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='sort_table.min.css') }}?v={{ APP_VERSION }}" | ||||
|     rel="stylesheet"> | ||||
|   {% endif %} | ||||
|  | ||||
|   {# --- Cropper CSS tylko dla wybranych podstron --- #} | ||||
|   {% set substrings_cropper = ['/admin/receipts', '/edit_my_list'] %} | ||||
|   {% if substrings_cropper | select("in", request.path) | list | length > 0 %} | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='cropper.min.css') }}" rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='cropper.min.css') }}?v={{ APP_VERSION }}" | ||||
|     rel="stylesheet"> | ||||
|   {% endif %} | ||||
|  | ||||
|   {# --- Tom Select CSS tylko dla wybranych podstron --- #} | ||||
|   {% set substrings_tomselect = ['/edit_my_list', '/admin/edit_list', '/admin/mass_edit_categories'] %} | ||||
|   {% set substrings_tomselect = ['/edit_my_list', '/admin/edit_list', '/admin/edit_categories'] %} | ||||
|   {% if substrings_tomselect | select("in", request.path) | list | length > 0 %} | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='tom-select.bootstrap5.min.css') }}" rel="stylesheet"> | ||||
|   <link href="{{ url_for('static_bp.serve_css_lib', filename='tom-select.bootstrap5.min.css') }}?v={{ APP_VERSION }}" | ||||
|     rel="stylesheet"> | ||||
|   {% endif %} | ||||
| </head> | ||||
|  | ||||
| @@ -81,6 +86,7 @@ | ||||
|     <hr class="text-secondary"> | ||||
|     <p class="mb-0">© 2025 <strong>linuxiarz.pl</strong> · <a href="https://gitea.linuxiarz.pl/gru/lista_zakupowa_live" | ||||
|         target="_blank" class="link-success text-decoration-none"> source code</a> | ||||
|     <div class="small">v{{ APP_VERSION }}</div> | ||||
|   </footer> | ||||
|  | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='bootstrap.bundle.min.js') }}"></script> | ||||
| @@ -101,14 +107,14 @@ | ||||
|   </script> | ||||
|  | ||||
|   {% if request.endpoint != 'system_auth' %} | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='glightbox.min.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='socket.io.min.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='sort_table.min.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='functions.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='live.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='sockets.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='glightbox.min.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='socket.io.min.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='sort_table.min.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='functions.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='live.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='sockets.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   {% endif %} | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='toasts.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js', filename='toasts.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   <script> | ||||
|     let lightbox = GLightbox({ | ||||
|       selector: '.glightbox' | ||||
| @@ -117,12 +123,13 @@ | ||||
|  | ||||
|   {% set substrings = ['/admin/receipts', '/edit_my_list'] %} | ||||
|   {% if substrings | select("in", request.path) | list | length > 0 %} | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='cropper.min.js') }}"></script> | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='cropper.min.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   {% endif %} | ||||
|  | ||||
|   {% set substrings = ['/edit_my_list', '/admin/edit_list', '/admin/mass_edit_categories'] %} | ||||
|   {% set substrings = ['/edit_my_list', '/admin/edit_list', '/admin/edit_categories'] %} | ||||
|   {% if substrings | select("in", request.path) | list | length > 0 %} | ||||
|   <script src="{{ url_for('static_bp.serve_js_lib', filename='tom-select.complete.min.js') }}"></script> | ||||
|   <script | ||||
|     src="{{ url_for('static_bp.serve_js_lib', filename='tom-select.complete.min.js') }}?v={{ APP_VERSION }}"></script> | ||||
|   {% endif %} | ||||
|  | ||||
|   {% endif %} | ||||
|   | ||||
| @@ -24,13 +24,13 @@ | ||||
|           <div class="form-check form-switch"> | ||||
|             <input class="form-check-input" type="checkbox" id="public" name="is_public" {% if list.is_public | ||||
|               %}checked{% endif %}> | ||||
|             <label class="form-check-label" for="public">🌐 Publiczna</label> | ||||
|             <label class="form-check-label" for="public">🌐 Publiczna (czyli mogą zobaczyć goście)</label> | ||||
|           </div> | ||||
|  | ||||
|           <div class="form-check form-switch"> | ||||
|             <input class="form-check-input" type="checkbox" id="temporary" name="is_temporary" {% if list.is_temporary | ||||
|               %}checked{% endif %}> | ||||
|             <label class="form-check-label" for="temporary">⏳ Tymczasowa</label> | ||||
|             <label class="form-check-label" for="temporary">⏳ Tymczasowa (ustaw date wygasania)</label> | ||||
|           </div> | ||||
|  | ||||
|           <div class="form-check form-switch"> | ||||
| @@ -85,17 +85,46 @@ | ||||
|           {% endfor %} | ||||
|         </select> | ||||
|       </div> | ||||
|  | ||||
|       <!-- Przyciski --> | ||||
|       <div class="btn-group mt-4" role="group"> | ||||
|         <button type="submit" class="btn btn-sm btn-outline-light">💾 Zapisz</button> | ||||
|         <a href="{{ url_for('main_page') }}" class="btn btn-sm btn-outline-light">❌ Anuluj</a> | ||||
|       </div> | ||||
|  | ||||
|     </form> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
| <!-- DOSTĘP DO LISTY --> | ||||
| <div class="mb-3"> | ||||
|   <label class="form-label">👥 Użytkownicy z dostępem</label> | ||||
|  | ||||
|   <div class="access-editor border rounded p-2 bg-dark" data-post-url="{{ request.path }}" | ||||
|     data-suggest-url="{{ url_for('edit_my_list_suggestions', list_id=list.id) }}" data-next="{{ request.path }}" | ||||
|     data-list-id="{{ list.id }}"> | ||||
|  | ||||
|     <!-- Tokeny uprawnionych --> | ||||
|     <div class="tokens d-flex flex-wrap gap-2 mb-2"> | ||||
|       {% for u in permitted_users %} | ||||
|       <button type="button" class="btn btn-sm btn-outline-secondary rounded-pill token" data-user-id="{{ u.id }}" | ||||
|         data-username="{{ u.username }}" title="Kliknij, aby odebrać dostęp"> | ||||
|         @{{ u.username }} <span aria-hidden="true">×</span> | ||||
|       </button> | ||||
|       {% endfor %} | ||||
|       {% if not permitted_users or permitted_users|length == 0 %} | ||||
|       <span class="no-perms text-warning small">Brak dodanych uprawnień.</span> | ||||
|       {% endif %} | ||||
|     </div> | ||||
|  | ||||
|     <!-- Dodawanie (wiele: przecinki/enter) + prywatne podpowiedzi --> | ||||
|     <div class="input-group input-group-sm"> | ||||
|       <input type="text" class="access-input form-control form-control-sm bg-dark text-white border-secondary" | ||||
|         placeholder="Dodaj @użytkownika (wiele: przecinki/enter)" aria-label="Dodaj użytkowników"> | ||||
|       <button type="button" class="access-add btn btn-sm btn-outline-light">➕ Dodaj</button> | ||||
|     </div> | ||||
|     <div class="text-secondary small mt-1">Kliknij token, aby odebrać dostęp.</div> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
| {% if receipts %} | ||||
| <hr class="my-4"> | ||||
| <h5>Paragony przypisane do tej listy</h5> | ||||
| @@ -215,8 +244,18 @@ | ||||
| {% endblock %} | ||||
|  | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='confirm_delete.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='user_receipt_crop.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_crop_logic.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select.js') }}"></script> | ||||
| <script> | ||||
|   window.CROP_CONFIG = { | ||||
|     modalId: "userCropModal", | ||||
|     imageId: "userCropImage", | ||||
|     spinnerId: "userCropLoading", | ||||
|     saveBtnId: "userSaveCrop", | ||||
|     endpoint: "/user_crop_receipt" | ||||
|   }; | ||||
| </script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='confirm_delete.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_crop.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_crop_logic.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='access_users.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
| @@ -13,7 +13,7 @@ | ||||
|       <div class="form-check form-switch"> | ||||
|         <input class="form-check-input" type="checkbox" id="showAllLists" {% if show_all %}checked{% endif %}> | ||||
|         <label class="form-check-label ms-2 text-white" for="showAllLists"> | ||||
|           Pokaż wszystkie publiczne listy innych | ||||
|           Uwzględnij listy udostępnione dla mnie i publiczne | ||||
|         </label> | ||||
|       </div> | ||||
|     </div> | ||||
| @@ -132,10 +132,35 @@ | ||||
|       <div class="tab-pane fade" id="chartTab" role="tabpanel"> | ||||
|         <div class="card bg-dark text-white mb-4"> | ||||
|           <div class="card-body"> | ||||
|             <button class="btn btn-outline-light w-100 py-2 mb-2 d-flex align-items-center justify-content-center gap-2" | ||||
|               id="toggleCategorySplit"> | ||||
|               🎨 Pokaż podział na kategorie | ||||
|             </button> | ||||
|  | ||||
|             <div class="text-end mb-2"> | ||||
|               <div class="btn-group btn-group-sm"> | ||||
|                 <button class="btn btn-outline-light" id="openFsBtn" | ||||
|                   onclick="openChartFullscreen('expensesChart','Wydatki')" title="Pełny ekran" disabled>⛶</button> | ||||
|  | ||||
|                 <button type="button" class="btn btn-outline-light" id="downloadMainChartBtn" title="Pobierz jako PNG" | ||||
|                   disabled>⬇</button> | ||||
|               </div> | ||||
|             </div> | ||||
|  | ||||
|             <div class="d-flex gap-3 mb-3"> | ||||
|               <div> | ||||
|                 <h6 class="text-white">Podział według czasu</h6> | ||||
|                 <div class="btn-group" role="group" aria-label="Podział czasu"> | ||||
|                   <button type="button" class="btn btn-outline-light btn-sm" id="toggleMonthlySplit" | ||||
|                     aria-pressed="true">Miesięczny</button> | ||||
|                   <button type="button" class="btn btn-outline-light btn-sm" id="toggleDailySplit" | ||||
|                     aria-pressed="false">Dzienny</button> | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div> | ||||
|                 <h6 class="text-white">Kategorie/Sumy wydatków</h6> | ||||
|                 <button class="btn btn-outline-light btn-sm" id="toggleCategorySplit" aria-pressed="false">Przełącz na | ||||
|                   kategorie</button> | ||||
|               </div> | ||||
|             </div> | ||||
|  | ||||
|  | ||||
|             <p id="chartRangeLabel" class="fw-bold mb-3">Widok: miesięczne</p> | ||||
|             <canvas id="expensesChart" height="120"></canvas> | ||||
|           </div> | ||||
| @@ -145,7 +170,7 @@ | ||||
|           <div class="btn-group btn-group-sm" role="group"> | ||||
|             <button class="btn btn-outline-light range-btn" data-range="last30days">🗓️ Ostatnie 30 | ||||
|               dni</button> | ||||
|             <button class="btn btn-outline-light range-btn active" data-range="currentmonth">📅 Bieżący miesiąc</button> | ||||
|             <button class="btn btn-outline-light range-btn" data-range="currentmonth">📅 Bieżący miesiąc</button> | ||||
|             <button class="btn btn-outline-light range-btn" data-range="monthly">📆 Miesięczne</button> | ||||
|             <button class="btn btn-outline-light range-btn" data-range="quarterly">📊 Kwartalne</button> | ||||
|             <button class="btn btn-outline-light range-btn" data-range="halfyearly">🗓️ Półroczne</button> | ||||
| @@ -167,13 +192,31 @@ | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
| <div class="modal fade" id="chartFullscreenModal" tabindex="-1" aria-hidden="true"> | ||||
|   <div class="modal-dialog modal-xl modal-dialog-centered"> | ||||
|     <div class="modal-content bg-dark"> | ||||
|       <div class="modal-header"> | ||||
|         <h5 class="modal-title" id="chartModalTitle">Wykres</h5> | ||||
|         <button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Zamknij"></button> | ||||
|       </div> | ||||
|       <div class="modal-body p-0"> | ||||
|         <canvas id="chartFullscreenCanvas"></canvas> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
|  | ||||
| {% endblock %} | ||||
|  | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js_lib', filename='chart.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='show_all_expense.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='expense_chart.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='expense_table.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='expense_tab.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select_all_table.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js_lib', filename='chart.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='show_all_expense.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='expense_chart.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='expense_table.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='expense_tab.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select_all_table.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='chart_controls.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='modal_chart.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='download_chart.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
| @@ -12,52 +12,55 @@ | ||||
|     {% if list.category_badges %} | ||||
|     {% for cat in list.category_badges %} | ||||
|     <span class="badge rounded-pill rounded-pill text-dark ms-1" style="background-color: {{ cat.color }}; | ||||
|                    font-size: 0.75rem; | ||||
|                    opacity: 0.85;"> | ||||
|                      font-size: 0.75rem; | ||||
|                      opacity: 0.85;"> | ||||
|       {{ cat.name }} | ||||
|     </span> | ||||
|     {% endfor %} | ||||
|     <!-- PRZYCISK DO MODALA KATEGORII --> | ||||
|     <button class="btn btn-sm btn-outline-light ms-2" data-bs-toggle="modal" data-bs-target="#categoriesModal"> | ||||
|       ✏️ Zmień kategorie | ||||
|     </button> | ||||
|     {% else %} | ||||
|     <a href="{{ url_for('edit_my_list', list_id=list.id, next=url_for('view_list', list_id=list.id)) }}" | ||||
|       class="ms-2 text-light small fw-light" style="opacity: 0.9;"> | ||||
|     <!-- ZAMIAST LINKU: OTWARCIE MODALA KATEGORII --> | ||||
|     <button class="btn btn-sm btn-outline-light ms-2" data-bs-toggle="modal" data-bs-target="#categoriesModal"> | ||||
|       ➕ Dodaj kategorię | ||||
|     </a> | ||||
|     </button> | ||||
|     {% endif %} | ||||
|   </h2> | ||||
| </div> | ||||
|  | ||||
| <a href="{{ request.url_root }}share/{{ list.share_token }}" class="btn btn-primary btn-sm w-100 mb-3" {% if not | ||||
| <a href="{{ request.url_root }}share/{{ list.share_token }}" class="btn btn-outline-primary btn-sm w-100 mb-3" {% if not | ||||
|   list.is_public %}disabled{% endif %}> | ||||
|   ✅ Otwórz tryb zakupowy / odznaczania produktów | ||||
| </a> | ||||
|  | ||||
| <div id="share-card" class="card bg-secondary bg-opacity-10 text-white mb-4"> | ||||
|   <div class="card-body"> | ||||
|     <div class="mb-2"> | ||||
|       <strong id="share-header"> | ||||
|         {% if list.is_public %} | ||||
|         🔗 Udostępnij link: | ||||
|         {% else %} | ||||
|         🙈 Lista jest ukryta przed gośćmi | ||||
|         {% endif %} | ||||
|         {% if list.is_public %}🔗 Udostępnij link (lista publiczna){% else %}🔗 Udostępnij link (widoczna przez link / | ||||
|         uprawnienia){% endif %} | ||||
|       </strong> | ||||
|       <span id="share-url" class="badge rounded-pill bg-secondary text-wrap" | ||||
|         style="font-size: 0.7rem; {% if not list.is_public %}display: none;{% endif %}"> | ||||
|       <span id="share-url" class="badge rounded-pill bg-secondary text-wrap" style="font-size: 0.7rem;"> | ||||
|         {{ request.url_root }}share/{{ list.share_token }} | ||||
|       </span> | ||||
|     </div> | ||||
|     <div class="d-flex flex-column flex-md-row gap-2"> | ||||
|       <button id="copyBtn" class="btn btn-success btn-sm flex-fill" | ||||
|         onclick="copyLink('{{ request.url_root }}share/{{ list.share_token }}')" {% if not list.is_public %}disabled{% | ||||
|         endif %}> | ||||
|       <button id="copyBtn" class="btn btn-outline-success btn-sm flex-fill" | ||||
|         onclick="copyLink('{{ request.url_root }}share/{{ list.share_token }}')"> | ||||
|         📋 Skopiuj / Udostępnij | ||||
|       </button> | ||||
|  | ||||
|       <button id="toggleVisibilityBtn" class="btn btn-outline-light btn-sm flex-fill" | ||||
|         onclick="toggleVisibility({{ list.id }})"> | ||||
|         {% if list.is_public %} | ||||
|         🙈 Ukryj listę | ||||
|         {% else %} | ||||
|         👁️ Udostępnij ponownie | ||||
|         {% endif %} | ||||
|         {% if list.is_public %}🙈 Ustaw niepubliczną{% else %}🐵 Uczyń publiczną{% endif %} | ||||
|       </button> | ||||
|  | ||||
|       <!-- ZAMIAST LINKU: OTWARCIE MODALA NADAWANIA DOSTĘPU --> | ||||
|       <button class="btn btn-outline-primary btn-sm flex-fill" data-bs-toggle="modal" | ||||
|         data-bs-target="#grantAccessModal"> | ||||
|         ➕ Nadaj dostęp | ||||
|       </button> | ||||
|     </div> | ||||
|   </div> | ||||
| @@ -73,14 +76,11 @@ | ||||
|  | ||||
| <div class="progress progress-dark position-relative"> | ||||
|   <div id="progress-bar-purchased" class="progress-bar bg-success" role="progressbar" data-bs-toggle="tooltip" | ||||
|     title="Kupione produkty"> | ||||
|   </div> | ||||
|     title="Kupione produkty"></div> | ||||
|   <div id="progress-bar-not-purchased" class="progress-bar bg-warning" role="progressbar" data-bs-toggle="tooltip" | ||||
|     title="Oznaczone jako niekupione"> | ||||
|   </div> | ||||
|     title="Oznaczone jako niekupione"></div> | ||||
|   <div id="progress-bar-remaining" class="progress-bar bg-transparent" role="progressbar" data-bs-toggle="tooltip" | ||||
|     title="Pozostałe do kupienia"> | ||||
|   </div> | ||||
|     title="Pozostałe do kupienia"></div> | ||||
|   <span id="progress-label" class="progress-label small fw-bold"></span> | ||||
| </div> | ||||
|  | ||||
| @@ -95,9 +95,8 @@ | ||||
| {% endif %} | ||||
|  | ||||
| <div class="d-flex justify-content-between align-items-center mb-3 flex-wrap"> | ||||
|   <button id="sort-toggle-btn" class="btn btn-sm btn-outline-warning" onclick="toggleSortMode()"> | ||||
|     ✳️ Zmień kolejność | ||||
|   </button> | ||||
|   <button id="sort-toggle-btn" class="btn btn-sm btn-outline-warning" onclick="toggleSortMode()">✳️ Zmień | ||||
|     kolejność</button> | ||||
|   <div class="form-check form-switch"> | ||||
|     <input class="form-check-input" type="checkbox" id="hidePurchasedToggle"> | ||||
|     <label class="form-check-label ms-2" for="hidePurchasedToggle">Ukryj zaznaczone</label> | ||||
| @@ -108,14 +107,12 @@ | ||||
|   {% for item in items %} | ||||
|   <li data-name="{{ item.name|lower }}" id="item-{{ item.id }}" | ||||
|     class="list-group-item d-flex justify-content-between align-items-center flex-wrap clickable-item  | ||||
|          {% if item.purchased %}bg-success text-white{% elif item.not_purchased %}bg-warning text-dark{% else %}item-not-checked{% endif %}" | ||||
|            {% if item.purchased %}bg-success text-white{% elif item.not_purchased %}bg-warning text-dark{% else %}item-not-checked{% endif %}" | ||||
|     data-is-share="{{ 'true' if is_share else 'false' }}"> | ||||
|  | ||||
|     <div class="d-flex align-items-center gap-2 flex-grow-1"> | ||||
|  | ||||
|       <input id="checkbox-{{ item.id }}" class="large-checkbox" type="checkbox" {% if item.purchased %}checked{% endif | ||||
|         %} {% if list.is_archived or item.not_purchased %}disabled{% endif %}> | ||||
|  | ||||
|       <span id="name-{{ item.id }}" class="text-white"> | ||||
|         {{ item.name }} | ||||
|         {% if item.quantity and item.quantity > 1 %} | ||||
| @@ -125,18 +122,12 @@ | ||||
|  | ||||
|       <div class="info-line ms-4 small d-flex flex-wrap gap-2" id="info-{{ item.id }}"> | ||||
|         {% set info_parts = [] %} | ||||
|         {% if item.note %} | ||||
|         {% set _ = info_parts.append('<span class="text-danger">[ <b>' ~ item.note ~ '</b> ]</span>') %} | ||||
|         {% endif %} | ||||
|         {% if item.not_purchased_reason %} | ||||
|         {% set _ = info_parts.append('<span class="text-dark">[ <b>Powód: ' ~ item.not_purchased_reason ~ '</b> | ||||
|           ]</span>') %} | ||||
|         {% endif %} | ||||
|         {% if item.added_by_display %} | ||||
|         {% set _ = info_parts.append('<span class="text-info">[ Dodał/a: <b>' ~ item.added_by_display ~ '</b> ]</span>') | ||||
|         %} | ||||
|         {% endif %} | ||||
|  | ||||
|         {% if item.note %}{% set _ = info_parts.append('<span class="text-danger">[ <b>' ~ item.note ~ '</b> ]</span>') | ||||
|         %}{% endif %} | ||||
|         {% if item.not_purchased_reason %}{% set _ = info_parts.append('<span class="text-dark">[ <b>Powód: ' ~ | ||||
|             item.not_purchased_reason ~ '</b> ]</span>') %}{% endif %} | ||||
|         {% if item.added_by_display %}{% set _ = info_parts.append('<span class="text-info">[ Dodał/a: <b>' ~ | ||||
|             item.added_by_display ~ '</b> ]</span>') %}{% endif %} | ||||
|         {% if info_parts %} | ||||
|         <div class="info-line ms-4 small d-flex flex-wrap gap-2" id="info-{{ item.id }}"> | ||||
|           {{ info_parts | join(' ') | safe }} | ||||
| @@ -147,34 +138,24 @@ | ||||
|  | ||||
|     <div class="btn-group btn-group-sm" role="group"> | ||||
|       {% if not is_share %} | ||||
|       <button type="button" class="btn btn-outline-light" {% if list.is_archived %}disabled{% else %} | ||||
|         onclick="editItem({{ item.id }}, '{{ item.name }}', {{ item.quantity or 1 }})" {% endif %}> | ||||
|         ✏️ | ||||
|       </button> | ||||
|       <button type="button" class="btn btn-outline-light" {% if list.is_archived %}disabled{% else %} | ||||
|         onclick="deleteItem({{ item.id }})" {% endif %}> | ||||
|         🗑️ | ||||
|       </button> | ||||
|       <button type="button" class="btn btn-outline-light" {% if list.is_archived %}disabled{% else | ||||
|         %}onclick="editItem({{ item.id }}, '{{ item.name }}', {{ item.quantity or 1 }})" {% endif %}>✏️</button> | ||||
|       <button type="button" class="btn btn-outline-light" {% if list.is_archived %}disabled{% else | ||||
|         %}onclick="deleteItem({{ item.id }})" {% endif %}>🗑️</button> | ||||
|       {% endif %} | ||||
|  | ||||
|       {% if item.not_purchased %} | ||||
|       <button type="button" class="btn btn-outline-light me-auto" {% if list.is_archived %}disabled{% else %} | ||||
|         onclick="unmarkNotPurchased({{ item.id }})" {% endif %}> | ||||
|         ✅ Przywróć | ||||
|       </button> | ||||
|       <button type="button" class="btn btn-outline-light me-auto" {% if list.is_archived %}disabled{% else | ||||
|         %}onclick="unmarkNotPurchased({{ item.id }})" {% endif %}>✅ Przywróć</button> | ||||
|       {% elif not item.not_purchased %} | ||||
|       <button type="button" class="btn btn-outline-light" {% if list.is_archived %}disabled{% else %} | ||||
|         onclick="markNotPurchasedModal(event, {{ item.id }})" {% endif %}> | ||||
|         ⚠️ | ||||
|       </button> | ||||
|       <button type="button" class="btn btn-outline-light" {% if list.is_archived %}disabled{% else | ||||
|         %}onclick="markNotPurchasedModal(event, {{ item.id }})" {% endif %}>⚠️</button> | ||||
|       {% endif %} | ||||
|     </div> | ||||
|   </li> | ||||
|  | ||||
|   {% else %} | ||||
|   <li id="empty-placeholder" class="list-group-item bg-dark text-secondary text-center w-100"> | ||||
|     Brak produktów w tej liście. | ||||
|   </li> | ||||
|   <li id="empty-placeholder" class="list-group-item bg-dark text-secondary text-center w-100">Brak produktów w tej | ||||
|     liście.</li> | ||||
|   {% endfor %} | ||||
| </ul> | ||||
|  | ||||
| @@ -191,7 +172,8 @@ | ||||
|         placeholder="Dodaj produkt i ilość" required> | ||||
|       <input type="number" id="newQuantity" name="quantity" class="form-control bg-dark text-white border-secondary" | ||||
|         placeholder="Ilość" min="1" value="1" style="max-width: 90px;"> | ||||
|       <button type="button" class="btn btn-success rounded-end" onclick="addItem({{ list.id }})">➕ Dodaj</button> | ||||
|       <button type="button" class="btn btn-outline-success rounded-end" onclick="addItem({{ list.id }})">➕ | ||||
|         Dodaj</button> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| @@ -213,12 +195,114 @@ | ||||
|   </div> | ||||
|   {% endfor %} | ||||
|   {% else %} | ||||
|   <div class="alert alert-info text-center w-100" role="alert"> | ||||
|     ℹ️ Brak wgranych paragonów do tej listy | ||||
|   </div> | ||||
|   <div class="alert alert-info text-center w-100" role="alert">ℹ️ Brak wgranych paragonów do tej listy</div> | ||||
|   {% endif %} | ||||
| </div> | ||||
|  | ||||
| <!-- MODAL: KATEGORIA (pojedynczy wybór) --> | ||||
| <div class="modal fade" id="categoriesModal" tabindex="-1" aria-labelledby="categoriesModalLabel" aria-hidden="true"> | ||||
|   <div class="modal-dialog modal-lg modal-dialog-scrollable"> | ||||
|     <div class="modal-content bg-dark text-white"> | ||||
|       <div class="modal-header"> | ||||
|         <h5 class="modal-title" id="categoriesModalLabel">Ustaw kategorię</h5> | ||||
|         <button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Zamknij"></button> | ||||
|       </div> | ||||
|  | ||||
|       <form method="post" action="{{ url_for('list_settings', list_id=list.id) }}"> | ||||
|         <div class="modal-body"> | ||||
|  | ||||
|           {% if popular_categories %} | ||||
|           <div class="mb-3"> | ||||
|             <div class="small text-secondary mb-1">Najczęściej używane:</div> | ||||
|             <div class="d-flex flex-wrap gap-2"> | ||||
|               {% for cat in popular_categories %} | ||||
|               <button type="button" class="btn btn-sm btn-outline-light category-suggestion" data-cat-id="{{ cat.id }}"> | ||||
|                 {{ cat.name }} | ||||
|               </button> | ||||
|               {% endfor %} | ||||
|               <button type="button" class="btn btn-sm btn-outline-secondary category-suggestion" data-cat-id=""> | ||||
|                 – brak – | ||||
|               </button> | ||||
|             </div> | ||||
|           </div> | ||||
|           {% endif %} | ||||
|  | ||||
|           <div class="mb-4"> | ||||
|             <label for="category_id" class="form-label">🏷️ Kategoria listy</label> | ||||
|             <select id="category_id" name="category_id" | ||||
|               class="form-select tom-dark bg-dark text-white border-secondary rounded"> | ||||
|               <option value="">– brak –</option> | ||||
|               {% for cat in categories %} | ||||
|               <option value="{{ cat.id }}" {% if cat.id in selected_categories %}selected{% endif %}> | ||||
|                 {{ cat.name }} | ||||
|               </option> | ||||
|               {% endfor %} | ||||
|             </select> | ||||
|           </div> | ||||
|  | ||||
|           <input type="hidden" name="action" value="set_category"> | ||||
|           <input type="hidden" name="next" value="{{ url_for('view_list', list_id=list.id) }}"> | ||||
|         </div> | ||||
|  | ||||
|         <div class="modal-footer justify-content-end"> | ||||
|           <div class="btn-group" role="group"> | ||||
|             <button type="button" class="btn btn-sm btn-outline-light" data-bs-dismiss="modal">❌ Anuluj</button> | ||||
|             <button type="submit" class="btn btn-sm btn-outline-light">💾 Zapisz</button> | ||||
|           </div> | ||||
|         </div> | ||||
|       </form> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
|  | ||||
| <!-- MODAL: NADAWANIE DOSTĘPU --> | ||||
| <div class="modal fade" id="grantAccessModal" tabindex="-1" aria-labelledby="grantAccessModalLabel" aria-hidden="true"> | ||||
|   <div class="modal-dialog modal-lg modal-dialog-scrollable"> | ||||
|     <div class="modal-content bg-dark text-white"> | ||||
|       <div class="modal-header"> | ||||
|         <h5 class="modal-title" id="grantAccessModalLabel">Nadaj dostęp użytkownikom</h5> | ||||
|         <button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Zamknij"></button> | ||||
|       </div> | ||||
|  | ||||
|       <div class="modal-body"> | ||||
|         <div class="access-editor border rounded p-2 bg-dark" | ||||
|           data-post-url="{{ url_for('list_settings', list_id=list.id) }}" | ||||
|           data-suggest-url="{{ url_for('edit_my_list_suggestions', list_id=list.id) }}" | ||||
|           data-next="{{ url_for('view_list', list_id=list.id) }}" data-list-id="{{ list.id }}" | ||||
|           data-grant-action="grant_access" data-revoke-field="revoke_user_id"> | ||||
|  | ||||
|           <!-- Tokeny aktualnie uprawnionych --> | ||||
|           <div class="tokens d-flex flex-wrap gap-2 mb-2"> | ||||
|             {% for u in permitted_users %} | ||||
|             <button type="button" class="btn btn-sm btn-outline-secondary rounded-pill token" data-user-id="{{ u.id }}" | ||||
|               data-username="{{ u.username }}" title="Kliknij, aby odebrać dostęp"> | ||||
|               @{{ u.username }} <span aria-hidden="true">×</span> | ||||
|             </button> | ||||
|             {% endfor %} | ||||
|             {% if not permitted_users or permitted_users|length == 0 %} | ||||
|             <span class="no-perms text-warning small">Brak dodanych uprawnień.</span> | ||||
|             {% endif %} | ||||
|           </div> | ||||
|  | ||||
|           <!-- Dodawanie wielu na raz + podpowiedzi prywatne --> | ||||
|           <div class="input-group input-group-sm"> | ||||
|             <input type="text" class="access-input form-control form-control-sm bg-dark text-white border-secondary" | ||||
|               placeholder="Dodaj @użytkownika (wiele: przecinki/enter)" aria-label="Dodaj użytkowników"> | ||||
|             <button type="button" class="access-add btn btn-sm btn-outline-light">➕ Dodaj</button> | ||||
|           </div> | ||||
|           <div class="text-secondary small mt-1">Kliknij token, aby odebrać dostęp.</div> | ||||
|         </div> | ||||
|       </div> | ||||
|  | ||||
|       <div class="modal-footer justify-content-end"> | ||||
|         <button type="button" class="btn btn-sm btn-outline-light" data-bs-dismiss="modal">Zamknij</button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
|  | ||||
|  | ||||
| <div class="modal fade" id="massAddModal" tabindex="-1" aria-labelledby="massAddModalLabel" aria-hidden="true"> | ||||
|   <div class="modal-dialog modal-lg modal-dialog-scrollable"> | ||||
|     <div class="modal-content bg-dark text-white"> | ||||
| @@ -230,20 +314,12 @@ | ||||
|         <button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Zamknij"></button> | ||||
|       </div> | ||||
|       <div class="modal-body"> | ||||
|  | ||||
|         <!-- SORTOWANIE i LICZNIK --> | ||||
|         <div id="sort-bar" class="mb-2"></div> | ||||
|  | ||||
|         <div class="mb-2"> | ||||
|           <span id="product-count" class="badge rounded-pill bg-primary ms-2"></span> | ||||
|         </div> | ||||
|  | ||||
|         <!-- LISTA PRODUKTÓW --> | ||||
|         <div class="mb-2"><span id="product-count" class="badge rounded-pill bg-primary ms-2"></span></div> | ||||
|         <ul id="mass-add-list" class="list-group"></ul> | ||||
|  | ||||
|       </div> | ||||
|       <div class="modal-footer"> | ||||
|         <button type="button" class="btn btn-outline-light" data-bs-dismiss="modal">Zamknij</button> | ||||
|         <button type="button" class="btn btn-outline-light btn-sm w-100" data-bs-dismiss="modal">Zamknij</button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| @@ -257,9 +333,11 @@ | ||||
|   window.LIST_ID = {{ list.id }}; | ||||
|   window.IS_OWNER = {{ 'true' if is_owner else 'false' }}; | ||||
| </script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='mass_add.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_upload.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='sort_mode.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='mass_add.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_upload.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='sort_mode.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='access_users.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='category_modal.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script> | ||||
|   setupList({{ list.id }}, '{{ current_user.username if current_user.is_authenticated else 'Gość' }}'); | ||||
| </script> | ||||
|   | ||||
| @@ -19,7 +19,6 @@ | ||||
|   </span> | ||||
|   {% endif %} | ||||
|  | ||||
|   {# Kategorie - tylko wyświetlenie, bez linków #} | ||||
|   {% if list.category_badges %} | ||||
|   {% for cat in list.category_badges %} | ||||
|   <span class="badge rounded-pill rounded-pill text-dark ms-1" style="background-color: {{ cat.color }}; | ||||
| @@ -112,8 +111,8 @@ | ||||
|     not current_user.is_authenticated %}disabled{% endif %}> | ||||
|   <input id="newQuantity" type="number" class="form-control bg-dark text-white border-secondary" placeholder="Ilość" | ||||
|     min="1" value="1" style="max-width: 90px;" {% if not current_user.is_authenticated %}disabled{% endif %}> | ||||
|   <button onclick="addItem({{ list.id }})" class="btn btn-success rounded-end" {% if not current_user.is_authenticated | ||||
|     %}disabled{% endif %}>➕ Dodaj</button> | ||||
|   <button onclick="addItem({{ list.id }})" class="btn btn-outline-success rounded-end" {% if not | ||||
|     current_user.is_authenticated %}disabled{% endif %}>➕ Dodaj</button> | ||||
| </div> | ||||
| {% endif %} | ||||
|  | ||||
| @@ -123,9 +122,8 @@ | ||||
| <div class="input-group mb-2"> | ||||
|   <input id="expenseAmount" type="number" step="0.01" min="0" class="form-control bg-dark text-white border-secondary" | ||||
|     placeholder="Kwota (PLN)"> | ||||
|   <button onclick="submitExpense({{ list.id }})" class="btn btn-success rounded-end">💾 Zapisz</button> | ||||
| </div> | ||||
| {% endif %} | ||||
|   <button onclick="submitExpense({{ list.id }})" class="btn btn-outline-primary rounded-end">💾 Zapisz</button> | ||||
| </div>{% endif %} | ||||
| <p id="total-expense2"><b>💸 Łącznie wydano:</b> {{ '%.2f'|format(total_expense) }} PLN</p> | ||||
|  | ||||
| <button id="toggleReceiptBtn" class="btn btn-outline-light mb-3 w-100 w-md-auto d-block mx-auto" type="button" | ||||
| @@ -216,20 +214,23 @@ | ||||
|  | ||||
| <!-- Modal notatki --> | ||||
| <div class="modal fade" id="noteModal" tabindex="-1" aria-hidden="true"> | ||||
|   <div class="modal-dialog"> | ||||
|   <div class="modal-dialog modal-lg modal-dialog-scrollable"> | ||||
|     <div class="modal-content bg-dark text-white"> | ||||
|       <div class="modal-header"> | ||||
|         <h5 class="modal-title">Dodaj notatkę</h5> | ||||
|         <button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Zamknij"></button> | ||||
|       </div> | ||||
|  | ||||
|       <form id="noteForm" onsubmit="submitNote(event)"> | ||||
|         <div class="modal-body"> | ||||
|           <textarea id="noteText" class="form-control" rows="4" | ||||
|             placeholder="Np. 'Nie było, zamieniłem na inny'"></textarea> | ||||
|           <textarea id="noteText" class="form-control" rows="4" placeholder="Np. 'Promocja 2+2'"></textarea> | ||||
|         </div> | ||||
|         <div class="modal-footer"> | ||||
|           <button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Anuluj</button> | ||||
|           <button type="submit" class="btn btn-success">💾 Zapisz</button> | ||||
|           <div class="btn-group" role="group"> | ||||
|             <button type="button" class="btn btn-outline-light btn-sm" data-bs-dismiss="modal">❌ Anuluj</button> | ||||
|             <button type="submit" class="btn btn-outline-light btn-sm">💾 Zapisz</button> | ||||
|           </div> | ||||
|  | ||||
|         </div> | ||||
|       </form> | ||||
|     </div> | ||||
| @@ -245,12 +246,12 @@ | ||||
|     var isSorting = false; | ||||
|   } | ||||
| </script> | ||||
| <script src="{{ url_for('static_bp.serve_js_lib', filename='Sortable.min.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='notes.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='clickable_row.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_section.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_upload.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_analysis.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js_lib', filename='Sortable.min.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='notes.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='clickable_row.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_section.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_upload.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='receipt_analysis.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script> | ||||
|   setupList({{ list.id }}, '{{ current_user.username if current_user.is_authenticated else 'Gość' }}'); | ||||
| </script> | ||||
|   | ||||
| @@ -63,7 +63,7 @@ | ||||
|   Twoje listy | ||||
|   <button type="button" class="btn btn-sm btn-outline-light ms-2" data-bs-toggle="modal" | ||||
|     data-bs-target="#archivedModal"> | ||||
|     📁 Zarchiwizowane | ||||
|     🗄️ Zarchiwizowane | ||||
|   </button> | ||||
| </h3> | ||||
| {% if user_lists %} | ||||
| @@ -87,17 +87,17 @@ | ||||
|  | ||||
|       <div class="btn-group mt-2 mt-md-0" role="group"> | ||||
|         <a href="{{ url_for('view_list', list_id=l.id) }}" | ||||
|           class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap">📄 Otwórz</a> | ||||
|           class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap">📂 Otwórz</a> | ||||
|         <a href="{{ url_for('shared_list', token=l.share_token) }}" | ||||
|           class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap">✏️ Odznaczaj</a> | ||||
|         <a href="{{ url_for('copy_list', list_id=l.id) }}" | ||||
|           class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap">📋 Kopiuj</a> | ||||
|         <a href="{{ url_for('edit_my_list', list_id=l.id) }}" | ||||
|           class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap">⚙️ Ustawienia</a> | ||||
|         <a href="{{ url_for('toggle_visibility', list_id=l.id) }}" | ||||
|           class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap"> | ||||
|           {% if l.is_public %}🙈 Ukryj{% else %}👁️ Odkryj{% endif %} | ||||
|           {% if l.is_public %}🙈 Ukryj{% else %}🐵 Odkryj{% endif %} | ||||
|         </a> | ||||
|         <a href="{{ url_for('edit_my_list', list_id=l.id) }}" | ||||
|           class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap">⚙️ Ustawienia</a> | ||||
|       </div> | ||||
|     </div> | ||||
|  | ||||
| @@ -135,21 +135,25 @@ | ||||
| {% endif %} | ||||
| {% endif %} | ||||
|  | ||||
| <h3 class="mt-4">Publiczne listy innych użytkowników</h3> | ||||
| {% if public_lists %} | ||||
|  | ||||
| <h3 class="mt-4"> {% if current_user.is_authenticated %}Udostępnione i publiczne listy innych użytkowników {% else %} | ||||
|   Publiczne listy innych użytkowników {% endif %}</h3> | ||||
|  | ||||
|  | ||||
| {% set lists_to_show = accessible_lists %} | ||||
| {% if lists_to_show %} | ||||
| <ul class="list-group"> | ||||
|   {% for l in public_lists %} | ||||
|   {% for l in lists_to_show %} | ||||
|   {% set purchased_count = l.purchased_count %} | ||||
|   {% set total_count = l.total_count %} | ||||
|   {% set percent = (purchased_count / total_count * 100) if total_count > 0 else 0 %} | ||||
|   <li class="list-group-item bg-dark text-white"> | ||||
|     <div class="d-flex justify-content-between align-items-center flex-wrap w-100"> | ||||
|       <span class="fw-bold"> | ||||
|         {{ l.title }} (Autor: {{ l.owner.username }}) | ||||
|         {{ l.title }} (Autor: {{ l.owner.username if l.owner else '—' }}) | ||||
|         {% for cat in l.category_badges %} | ||||
|         <span class="badge rounded-pill text-dark ms-1" style="background-color: {{ cat.color }}; | ||||
|                  font-size: 0.56rem; | ||||
|                  opacity: 0.85;"> | ||||
|                  font-size: 0.56rem; opacity: 0.85;"> | ||||
|           {{ cat.name }} | ||||
|         </span> | ||||
|         {% endfor %} | ||||
| @@ -158,37 +162,31 @@ | ||||
|       <a href="{{ url_for('shared_list', list_id=l.id) }}" | ||||
|         class="btn btn-sm btn-outline-light d-flex align-items-center text-nowrap">✏️ Odznaczaj</a> | ||||
|     </div> | ||||
|  | ||||
|     <div class="progress progress-dark progress-thin mt-2 position-relative"> | ||||
|       {# Kupione #} | ||||
|       <div class="progress-bar bg-success" role="progressbar" | ||||
|         style="width: {{ (purchased_count / total_count * 100) if total_count > 0 else 0 }}%" aria-valuemin="0" | ||||
|         aria-valuemax="100"></div> | ||||
|  | ||||
|       {# Niekupione #} | ||||
|       {% set not_purchased_count = l.not_purchased_count if l.total_count else 0 %} | ||||
|       <div class="progress-bar bg-warning" role="progressbar" | ||||
|         style="width: {{ (not_purchased_count / total_count * 100) if total_count > 0 else 0 }}%" aria-valuemin="0" | ||||
|         aria-valuemax="100"></div> | ||||
|  | ||||
|       {# Pozostałe #} | ||||
|       <div class="progress-bar bg-transparent" role="progressbar" | ||||
|         style="width: {{ 100 - ((purchased_count + not_purchased_count) / total_count * 100) if total_count > 0 else 100 }}%" | ||||
|         aria-valuemin="0" aria-valuemax="100"></div> | ||||
|  | ||||
|       <span class="progress-label small fw-bold  | ||||
|         {% if percent < 51 %}text-white{% else %}text-dark{% endif %}"> | ||||
|       <span class="progress-label small fw-bold {% if percent < 51 %}text-white{% else %}text-dark{% endif %}"> | ||||
|         Produkty: {{ purchased_count }}/{{ total_count }} ({{ percent|round(0) }}%) | ||||
|         {% if l.total_expense > 0 %} | ||||
|         — 💸 {{ '%.2f'|format(l.total_expense) }} PLN | ||||
|         {% endif %} | ||||
|         {% if l.total_expense > 0 %} — 💸 {{ '%.2f'|format(l.total_expense) }} PLN{% endif %} | ||||
|       </span> | ||||
|     </div> | ||||
|  | ||||
|   </li> | ||||
|   {% endfor %} | ||||
| </ul> | ||||
| {% else %} | ||||
| <p><span class="badge rounded-pill bg-secondary opacity-75">Brak dostępnych list publicznych do wyświetlenia</span></p> | ||||
| <p><span class="badge rounded-pill bg-secondary opacity-75">Brak list do wyświetlenia</span></p> | ||||
| {% endif %} | ||||
|  | ||||
| <div class="modal fade" id="archivedModal" tabindex="-1" aria-labelledby="archivedModalLabel" aria-hidden="true"> | ||||
| @@ -253,8 +251,8 @@ | ||||
| </div> | ||||
|  | ||||
| {% block scripts %} | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='toggle_button.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select_month.js') }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='toggle_button.js') }}?v={{ APP_VERSION }}"></script> | ||||
| <script src="{{ url_for('static_bp.serve_js', filename='select_month.js') }}?v={{ APP_VERSION }}"></script> | ||||
| {% endblock %} | ||||
|  | ||||
| {% endblock %} | ||||
		Reference in New Issue
	
	Block a user