Compare commits
	
		
			58 Commits
		
	
	
		
			eec3985c5a
			...
			master
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 3be89fe9d8 | ||
|   | f6d2c9955c | ||
|   | 15dd9d1fe9 | ||
|   | e1367f8bf2 | ||
|   | 35a8d5dd8e | ||
|   | f9406a46cf | ||
|   | 1c69295b9b | ||
|   | a4887f9c73 | ||
|   | 22b4c3d99a | ||
|   | 16ec180822 | ||
|   | 3bad07ce69 | ||
|   | 086784b4ef | ||
|   | a4fa47e1a2 | ||
|   | 6709681dfd | ||
|   | 1e7f1e3990 | ||
|   | d844d354fa | ||
|   | 4de192b865 | ||
|   | 90b9c8d462 | ||
| ee2f498744 | |||
| 4a357ffd57 | |||
| f78fe5369c | |||
| 69edf7844a | |||
| 1bedab4825 | |||
| 609a411824 | |||
|   | a4fabcf4bd | ||
|   | d44681ed0c | ||
|   | 1d8a87216d | ||
|   | 2f961db2ac | ||
|   | e79eebed17 | ||
|   | c0daa93fe9 | ||
|   | 0d09abc4eb | ||
|   | 413c80baa2 | ||
|   | 0390a59008 | ||
|   | ee05bf74b5 | ||
|   | 052439b340 | ||
|   | a673fe99f8 | ||
|   | fd46242cf5 | ||
|   | 95b01665b9 | ||
|   | f7a99df93d | ||
|   | 64ad6b4bbf | ||
|   | d7abf96080 | ||
|   | 8887ed2c62 | ||
|   | dfa3028dad | ||
|   | b679700c6d | ||
|   | eb9f11b2d6 | ||
|   | 6a1734024a | ||
|   | fdcfaff80e | ||
|   | abb8ee0ae7 | ||
|   | 37061442d2 | ||
|   | 113f9e385e | ||
|   | 34d3f3ae0c | ||
|   | 72b82ae40c | ||
|   | 9a7660fdf7 | ||
|   | 56c02e072d | ||
|   | b45a259455 | ||
|   | d0df1cdee9 | ||
|   | 71a2b33a7a | ||
|   | 16beaac932 | 
							
								
								
									
										30
									
								
								.env.example
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								.env.example
									
									
									
									
									
								
							| @@ -36,4 +36,32 @@ USE_ETAGS=True | |||||||
| PRAGMA_HEADER= | PRAGMA_HEADER= | ||||||
|  |  | ||||||
| # Wartość nagłówka X-Robots-Tag, gdy BLOCK_BOTS=True | # Wartość nagłówka X-Robots-Tag, gdy BLOCK_BOTS=True | ||||||
| ROBOTS_TAG=noindex, nofollow, nosnippet, noarchive | ROBOTS_TAG="noindex, nofollow, nosnippet, noarchive" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Rodzaj bazy: sqlite, pgsql, mysql | ||||||
|  | # Mozliwe wartosci: sqlite / pgsql / mysql | ||||||
|  | DB_ENGINE=sqlite | ||||||
|  |  | ||||||
|  | # --- Konfiguracja dla sqlite --- | ||||||
|  | # Plik bazy bedzie utworzony automatycznie w katalogu ./instance | ||||||
|  | # Pozostale zmienne sa ignorowane przy DB_ENGINE=sqlite | ||||||
|  |  | ||||||
|  | # --- Konfiguracja dla pgsql --- | ||||||
|  | # Ustaw DB_ENGINE=pgsql | ||||||
|  | # Domyslny port PostgreSQL to 5432 | ||||||
|  | # Wymaga dzialajacego serwera PostgreSQL (np. kontener `postgres`) | ||||||
|  |  | ||||||
|  | # --- Konfiguracja dla mysql --- | ||||||
|  | # Ustaw DB_ENGINE=mysql | ||||||
|  | # Domyslny port MySQL to 3306 | ||||||
|  | # Wymaga kontenera z MySQL i uzytkownika z dostepem do bazy | ||||||
|  |  | ||||||
|  | # Wspolne zmienne (dla pgsql, mysql) | ||||||
|  | # DB_HOST = pgsql lub mysql zgodnie z deployem (profil w docker-compose.yml) | ||||||
|  |  | ||||||
|  | DB_HOST=pgsql  | ||||||
|  | DB_PORT=5432 | ||||||
|  | DB_NAME=myapp | ||||||
|  | DB_USER=user | ||||||
|  | DB_PASSWORD=pass | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -3,4 +3,7 @@ data/ | |||||||
| instance/ | instance/ | ||||||
| venv/ | venv/ | ||||||
| .env | .env | ||||||
| version.txt | version.txt | ||||||
|  | deploy/varnish/default.vcl | ||||||
|  | *.tar.gz | ||||||
|  | db/* | ||||||
							
								
								
									
										19
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,19 +0,0 @@ | |||||||
| FROM python:3.13-slim |  | ||||||
| WORKDIR /app |  | ||||||
|  |  | ||||||
| RUN apt-get update && apt-get install -y --no-install-recommends \ |  | ||||||
|     && apt-get install -y build-essential \ |  | ||||||
|     && apt-get clean \ |  | ||||||
|     && rm -rf /var/lib/apt/lists/* |  | ||||||
|  |  | ||||||
| COPY requirements.txt requirements.txt |  | ||||||
|  |  | ||||||
| RUN pip install --upgrade pip |  | ||||||
| RUN pip install --no-cache-dir -r requirements.txt |  | ||||||
|  |  | ||||||
| COPY . . |  | ||||||
| RUN mkdir -p /app/instance |  | ||||||
|  |  | ||||||
| EXPOSE 8080 |  | ||||||
|  |  | ||||||
| CMD ["python", "run_waitress.py"] |  | ||||||
							
								
								
									
										1
									
								
								Dockerfile
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						
									
										1
									
								
								Dockerfile
									
									
									
									
									
										Symbolic link
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | deploy/app/Dockerfile | ||||||
							
								
								
									
										38
									
								
								_tools/db/migrate.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								_tools/db/migrate.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | |||||||
|  | python3 -m venv venv_migrate | ||||||
|  | source venv_migrate/bin/activate   | ||||||
|  | pip install sqlalchemy psycopg2-binary dotenv  | ||||||
|  | docker compose --profile pgsql up -d --build | ||||||
|  | PYTHONPATH=. python3 _tools/db/migrate_sqlite_to_pgsql.py  | ||||||
|  | rm -rf venv_migrate | ||||||
|  |  | ||||||
|  | # reset wszystkich sekwencji w pgsql | ||||||
|  | docker exec -it zbiorka-pgsql-db psql -U zbiorki -d zbiorki | ||||||
|  |  | ||||||
|  |  | ||||||
|  | DO $$ | ||||||
|  | DECLARE | ||||||
|  |     r RECORD; | ||||||
|  | BEGIN | ||||||
|  |     FOR r IN | ||||||
|  |         SELECT  | ||||||
|  |             c.relname AS seq_name, | ||||||
|  |             t.relname AS table_name, | ||||||
|  |             a.attname AS column_name | ||||||
|  |         FROM  | ||||||
|  |             pg_class c | ||||||
|  |         JOIN  | ||||||
|  |             pg_depend d ON d.objid = c.oid | ||||||
|  |         JOIN  | ||||||
|  |             pg_class t ON d.refobjid = t.oid | ||||||
|  |         JOIN  | ||||||
|  |             pg_attribute a ON a.attrelid = t.oid AND a.attnum = d.refobjsubid | ||||||
|  |         WHERE  | ||||||
|  |             c.relkind = 'S' | ||||||
|  |             AND d.deptype = 'a' | ||||||
|  |     LOOP | ||||||
|  |         EXECUTE format( | ||||||
|  |             'SELECT setval(%L, COALESCE((SELECT MAX(%I) FROM %I), 1), true)', | ||||||
|  |             r.seq_name, r.column_name, r.table_name | ||||||
|  |         ); | ||||||
|  |     END LOOP; | ||||||
|  | END$$; | ||||||
							
								
								
									
										68
									
								
								_tools/db/migrate_sqlite_to_pgsql.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								_tools/db/migrate_sqlite_to_pgsql.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | |||||||
|  | import sys | ||||||
|  | import os | ||||||
|  | from dotenv import load_dotenv | ||||||
|  |  | ||||||
|  | sys.path.insert(0, os.path.abspath(os.path.join(os.path.dirname(__file__), "../../"))) | ||||||
|  |  | ||||||
|  | load_dotenv() | ||||||
|  |  | ||||||
|  | from sqlalchemy import create_engine, MetaData | ||||||
|  | from sqlalchemy.orm import sessionmaker | ||||||
|  | from config import Config | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Źródło: SQLite | ||||||
|  | sqlite_engine = create_engine("sqlite:///instance/baza.db") | ||||||
|  | sqlite_meta = MetaData() | ||||||
|  | sqlite_meta.reflect(bind=sqlite_engine) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Cel: PostgreSQL | ||||||
|  | pg_engine = create_engine(Config.SQLALCHEMY_DATABASE_URI) | ||||||
|  | pg_meta = MetaData() | ||||||
|  | pg_meta.reflect(bind=pg_engine) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Sesje | ||||||
|  | SQLiteSession = sessionmaker(bind=sqlite_engine) | ||||||
|  | PGSession = sessionmaker(bind=pg_engine) | ||||||
|  |  | ||||||
|  | sqlite_session = SQLiteSession() | ||||||
|  | pg_session = PGSession() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def migrate_table(table_name): | ||||||
|  |     print("Używana baza docelowa:", Config.SQLALCHEMY_DATABASE_URI) | ||||||
|  |     print(f"Migruję tabelę: {table_name}") | ||||||
|  |     source_table = sqlite_meta.tables.get(table_name) | ||||||
|  |     target_table = pg_meta.tables.get(table_name) | ||||||
|  |  | ||||||
|  |     if source_table is None or target_table is None: | ||||||
|  |         print(f"Pominięto: {table_name} (brak w jednej z baz)") | ||||||
|  |         return | ||||||
|  |  | ||||||
|  |     rows = sqlite_session.execute(source_table.select()).fetchall() | ||||||
|  |     if not rows: | ||||||
|  |         print("Brak danych do migracji.") | ||||||
|  |         return | ||||||
|  |  | ||||||
|  |     insert_data = [dict(row._mapping) for row in rows] | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         with pg_engine.begin() as conn: | ||||||
|  |             conn.execute(target_table.delete()) | ||||||
|  |             conn.execute(target_table.insert(), insert_data) | ||||||
|  |         print(f"Przeniesiono: {len(rows)} rekordów") | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"Błąd przy migracji {table_name}: {e}") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def main(): | ||||||
|  |     tables = ["zbiorka", "przedmiot", "uzytkownik", "wydatek", "ustawienia_globalne", "wplata"] | ||||||
|  |     for table in tables: | ||||||
|  |         migrate_table(table) | ||||||
|  |     print("\nMigracja zakończona pomyślnie.") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == "__main__": | ||||||
|  |     main() | ||||||
							
								
								
									
										75
									
								
								alters.txt
									
									
									
									
									
								
							
							
						
						
									
										75
									
								
								alters.txt
									
									
									
									
									
								
							| @@ -1,68 +1,17 @@ | |||||||
| -- WŁĄCZ/wyłącz FK zależnie od etapu migracji | ALTER TABLE zbiorka ALTER COLUMN numer_konta DROP NOT NULL; | ||||||
| PRAGMA foreign_keys = OFF; | ALTER TABLE zbiorka ALTER COLUMN numer_telefonu_blik DROP NOT NULL; | ||||||
|  |  | ||||||
| BEGIN TRANSACTION; | _______________________________ | ||||||
|  |  | ||||||
| -- 1) Nowa tabela z właściwym FK (ON DELETE CASCADE) | PGSQL | ||||||
| CREATE TABLE wplata_new ( | ALTER TABLE wplata  ADD COLUMN ukryta boolean NOT NULL DEFAULT false; | ||||||
|     id          INTEGER PRIMARY KEY, | ALTER TABLE wydatek ADD COLUMN ukryta boolean NOT NULL DEFAULT false; | ||||||
|     zbiorka_id  INTEGER NOT NULL, |  | ||||||
|     kwota       REAL    NOT NULL, |  | ||||||
|     data        DATETIME, |  | ||||||
|     opis        TEXT, |  | ||||||
|     FOREIGN KEY(zbiorka_id) REFERENCES zbiorka(id) ON DELETE CASCADE |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| -- 2) (opcjonalnie) upewnij się, że nie ma „sierotek” | -- po migracji można zdjąć DEFAULT (opcjonalnie) | ||||||
| -- SELECT w.* FROM wplata w LEFT JOIN zbiorka z ON z.id = w.zbiorka_id WHERE z.id IS NULL; | ALTER TABLE wplata  ALTER COLUMN ukryta DROP DEFAULT; | ||||||
|  | ALTER TABLE wydatek ALTER COLUMN ukryta DROP DEFAULT; | ||||||
| -- 3) Kopiowanie danych |  | ||||||
| INSERT INTO wplata_new (id, zbiorka_id, kwota, data, opis) |  | ||||||
| SELECT id, zbiorka_id, kwota, data, opis |  | ||||||
| FROM wplata; |  | ||||||
|  |  | ||||||
| -- 4) Usunięcie starej tabeli |  | ||||||
| DROP TABLE wplata; |  | ||||||
|  |  | ||||||
| -- 5) Zmiana nazwy nowej tabeli na właściwą |  | ||||||
| ALTER TABLE wplata_new RENAME TO wplata; |  | ||||||
|  |  | ||||||
| -- 6) Odtwórz indeksy/trigger-y jeśli jakieś były (przykład indeksu po FK) |  | ||||||
| -- CREATE INDEX idx_wplata_zbiorka_id ON wplata(zbiorka_id); |  | ||||||
|  |  | ||||||
| COMMIT; |  | ||||||
|  |  | ||||||
| PRAGMA foreign_keys = ON; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ | SQLite | ||||||
|  | ALTER TABLE wplata  ADD COLUMN ukryta INTEGER NOT NULL DEFAULT 0; | ||||||
| PRAGMA foreign_keys=OFF; | ALTER TABLE wydatek ADD COLUMN ukryta INTEGER NOT NULL DEFAULT 0; | ||||||
| BEGIN TRANSACTION; |  | ||||||
|  |  | ||||||
| ALTER TABLE global_settings ADD COLUMN logo_url TEXT DEFAULT ''; |  | ||||||
| ALTER TABLE global_settings ADD COLUMN site_title TEXT DEFAULT ''; |  | ||||||
| ALTER TABLE global_settings ADD COLUMN show_logo_in_navbar BOOLEAN DEFAULT 0; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| COMMIT; |  | ||||||
| PRAGMA foreign_keys=ON; |  | ||||||
|  |  | ||||||
| ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ |  | ||||||
|  |  | ||||||
|  |  | ||||||
| -- 1) Dodajemy nowe kolumny (SQLite pozwala tylko na ADD COLUMN) |  | ||||||
| ALTER TABLE global_settings ADD COLUMN navbar_brand_mode TEXT DEFAULT 'text'; |  | ||||||
| ALTER TABLE global_settings ADD COLUMN footer_brand_mode TEXT DEFAULT 'text'; |  | ||||||
| ALTER TABLE global_settings ADD COLUMN footer_text TEXT; |  | ||||||
|  |  | ||||||
| -- 2) Backfill: zgodność wsteczna z show_logo_in_navbar |  | ||||||
| UPDATE global_settings |  | ||||||
| SET navbar_brand_mode = 'logo' |  | ||||||
| WHERE COALESCE(show_logo_in_navbar, 0) = 1; |  | ||||||
|  |  | ||||||
| -- 3) Upewnij się, że wartości są ustawione (na wypadek NULL-i) |  | ||||||
| UPDATE global_settings |  | ||||||
| SET navbar_brand_mode = COALESCE(navbar_brand_mode, 'text'), |  | ||||||
|     footer_brand_mode = COALESCE(footer_brand_mode, 'text'); |  | ||||||
							
								
								
									
										440
									
								
								app.py
									
									
									
									
									
								
							
							
						
						
									
										440
									
								
								app.py
									
									
									
									
									
								
							| @@ -65,30 +65,35 @@ APP_VERSION = f"{deploy_date}+{commit}" | |||||||
| app.config["APP_VERSION"] = APP_VERSION | app.config["APP_VERSION"] = APP_VERSION | ||||||
|  |  | ||||||
| # MODELE | # MODELE | ||||||
| class User(UserMixin, db.Model): | class Uzytkownik(UserMixin, db.Model): | ||||||
|  |     __tablename__ = "uzytkownik" | ||||||
|  |  | ||||||
|     id = db.Column(db.Integer, primary_key=True) |     id = db.Column(db.Integer, primary_key=True) | ||||||
|     username = db.Column(db.String(80), unique=True, nullable=False) |     uzytkownik = db.Column(db.String(80), unique=True, nullable=False) | ||||||
|     password_hash = db.Column(db.String(128), nullable=False) |     haslo_hash = db.Column(db.String(128), nullable=False) | ||||||
|     is_admin = db.Column(db.Boolean, default=False)  # Flaga głównego administratora |     czy_admin = db.Column(db.Boolean, default=False) | ||||||
|  |  | ||||||
|     def set_password(self, password): |     def set_password(self, password): | ||||||
|         self.password_hash = generate_password_hash(password) |         self.haslo_hash = generate_password_hash(password) | ||||||
|  |  | ||||||
|     def check_password(self, password): |     def check_password(self, password): | ||||||
|         return check_password_hash(self.password_hash, password) |         return check_password_hash(self.haslo_hash, password) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Zbiorka(db.Model): | class Zbiorka(db.Model): | ||||||
|     id = db.Column(db.Integer, primary_key=True) |     id = db.Column(db.Integer, primary_key=True) | ||||||
|     nazwa = db.Column(db.String(100), nullable=False) |     nazwa = db.Column(db.String(100), nullable=False) | ||||||
|     opis = db.Column(db.Text, nullable=False) |     opis = db.Column(db.Text, nullable=False) | ||||||
|     numer_konta = db.Column(db.String(50), nullable=False) |     numer_konta = db.Column(db.String(50), nullable=True)  | ||||||
|     numer_telefonu_blik = db.Column(db.String(50), nullable=False) |     numer_telefonu_blik = db.Column(db.String(50), nullable=True) | ||||||
|     cel = db.Column(Numeric(12, 2), nullable=False, default=0) |     cel = db.Column(Numeric(12, 2), nullable=False, default=0) | ||||||
|     stan = db.Column(Numeric(12, 2), default=0) |     stan = db.Column(Numeric(12, 2), default=0) | ||||||
|     ukryta = db.Column(db.Boolean, default=False) |     ukryta = db.Column(db.Boolean, default=False) | ||||||
|     ukryj_kwote = db.Column(db.Boolean, default=False) |     ukryj_kwote = db.Column(db.Boolean, default=False) | ||||||
|     zrealizowana = db.Column(db.Boolean, default=False) |     zrealizowana = db.Column(db.Boolean, default=False) | ||||||
|  |     pokaz_postep_finanse = db.Column(db.Boolean, default=True, nullable=False) | ||||||
|  |     pokaz_postep_pozycje = db.Column(db.Boolean, default=True, nullable=False) | ||||||
|  |     pokaz_postep_kwotowo = db.Column(db.Boolean, default=True, nullable=False) | ||||||
|  |     uzyj_konta = db.Column(db.Boolean, default=True, nullable=False) | ||||||
|  |     uzyj_blik = db.Column(db.Boolean, default=True, nullable=False) | ||||||
|  |  | ||||||
|     wplaty = db.relationship( |     wplaty = db.relationship( | ||||||
|         "Wplata", |         "Wplata", | ||||||
| @@ -139,9 +144,8 @@ class Wplata(db.Model): | |||||||
|     kwota = db.Column(Numeric(12, 2), nullable=False) |     kwota = db.Column(Numeric(12, 2), nullable=False) | ||||||
|     data = db.Column(db.DateTime, default=datetime.utcnow) |     data = db.Column(db.DateTime, default=datetime.utcnow) | ||||||
|     opis = db.Column(db.Text, nullable=True) |     opis = db.Column(db.Text, nullable=True) | ||||||
|  |  | ||||||
|     zbiorka = db.relationship("Zbiorka", back_populates="wplaty") |     zbiorka = db.relationship("Zbiorka", back_populates="wplaty") | ||||||
|  |     ukryta = db.Column(db.Boolean, nullable=False, default=False) | ||||||
|  |  | ||||||
| class Wydatek(db.Model): | class Wydatek(db.Model): | ||||||
|     id = db.Column(db.Integer, primary_key=True) |     id = db.Column(db.Integer, primary_key=True) | ||||||
| @@ -153,35 +157,37 @@ class Wydatek(db.Model): | |||||||
|     kwota = db.Column(Numeric(12, 2), nullable=False) |     kwota = db.Column(Numeric(12, 2), nullable=False) | ||||||
|     data = db.Column(db.DateTime, default=datetime.utcnow) |     data = db.Column(db.DateTime, default=datetime.utcnow) | ||||||
|     opis = db.Column(db.Text, nullable=True) |     opis = db.Column(db.Text, nullable=True) | ||||||
|  |     ukryta = db.Column(db.Boolean, nullable=False, default=False) | ||||||
|  |  | ||||||
|  | class UstawieniaGlobalne(db.Model): | ||||||
|  |     __tablename__ = "ustawienia_globalne" | ||||||
|  |  | ||||||
| class GlobalSettings(db.Model): |  | ||||||
|     id = db.Column(db.Integer, primary_key=True) |     id = db.Column(db.Integer, primary_key=True) | ||||||
|     numer_konta = db.Column(db.String(50), nullable=False) |     numer_konta = db.Column(db.String(50), nullable=False) | ||||||
|     numer_telefonu_blik = db.Column(db.String(50), nullable=False) |     numer_telefonu_blik = db.Column(db.String(50), nullable=False) | ||||||
|     allowed_login_hosts = db.Column(db.Text, nullable=True) |     dozwolone_hosty_logowania = db.Column(db.Text, nullable=True) | ||||||
|     logo_url = db.Column(db.String(255), nullable=True) |     logo_url = db.Column(db.String(255), nullable=True) | ||||||
|     site_title = db.Column(db.String(120), nullable=True) |     tytul_strony = db.Column(db.String(120), nullable=True) | ||||||
|     show_logo_in_navbar = db.Column(db.Boolean, default=False) |     pokaz_logo_w_navbar = db.Column(db.Boolean, default=False) | ||||||
|     navbar_brand_mode = db.Column(db.String(10), default="text") |     typ_navbar = db.Column(db.String(10), default="text") | ||||||
|     footer_brand_mode = db.Column(db.String(10), default="text") |     typ_stopka = db.Column(db.String(10), default="text") | ||||||
|     footer_text = db.Column(db.String(200), nullable=True) |     stopka_text = db.Column(db.String(200), nullable=True) | ||||||
|  |  | ||||||
|  |  | ||||||
| @login_manager.user_loader | @login_manager.user_loader | ||||||
| def load_user(user_id): | def load_user(user_id): | ||||||
|     return db.session.get(User, int(user_id)) |     return db.session.get(Uzytkownik, int(user_id)) | ||||||
|  |  | ||||||
|  |  | ||||||
| @event.listens_for(Engine, "connect") | @event.listens_for(Engine, "connect") | ||||||
| def set_sqlite_pragma(dbapi_connection, connection_record): | def set_sqlite_pragma(dbapi_connection, connection_record): | ||||||
|     try: |     if dbapi_connection.__class__.__module__.startswith('sqlite3'): | ||||||
|         cursor = dbapi_connection.cursor() |         try: | ||||||
|         cursor.execute("PRAGMA foreign_keys=ON") |             cursor = dbapi_connection.cursor() | ||||||
|         cursor.close() |             cursor.execute("PRAGMA foreign_keys=ON") | ||||||
|     except Exception: |             cursor.close() | ||||||
|         pass |         except Exception: | ||||||
|  |             pass | ||||||
|  |  | ||||||
| def get_real_ip(): | def get_real_ip(): | ||||||
|     headers = request.headers |     headers = request.headers | ||||||
| @@ -249,11 +255,9 @@ def markdown_filter(text): | |||||||
|  |  | ||||||
| @app.context_processor | @app.context_processor | ||||||
| def inject_globals(): | def inject_globals(): | ||||||
|     settings = GlobalSettings.query.first() |     settings = UstawieniaGlobalne.query.first() | ||||||
|     allowed_hosts_str = ( |     allowed_hosts_str = ( | ||||||
|         settings.allowed_login_hosts |         settings.dozwolone_hosty_logowania if settings and settings.dozwolone_hosty_logowania else "" | ||||||
|         if settings and settings.allowed_login_hosts |  | ||||||
|         else "" |  | ||||||
|     ) |     ) | ||||||
|     client_ip = get_real_ip() |     client_ip = get_real_ip() | ||||||
|     return { |     return { | ||||||
| @@ -290,20 +294,28 @@ def zbiorka(zbiorka_id): | |||||||
|     zb = db.session.get(Zbiorka, zbiorka_id) |     zb = db.session.get(Zbiorka, zbiorka_id) | ||||||
|     if zb is None: |     if zb is None: | ||||||
|         abort(404) |         abort(404) | ||||||
|     if zb.ukryta and (not current_user.is_authenticated or not current_user.is_admin): |     if zb.ukryta and (not current_user.is_authenticated or not current_user.czy_admin): | ||||||
|         abort(404) |         abort(404) | ||||||
|  |  | ||||||
|     # scalona oś czasu: wpłaty + wydatki |     is_admin = current_user.is_authenticated and current_user.czy_admin | ||||||
|     aktywnosci = [ |     show_hidden = is_admin and (request.args.get("show_hidden") in ("1", "true", "yes")) | ||||||
|         {"typ": "wpłata", "kwota": w.kwota, "opis": w.opis, "data": w.data} |  | ||||||
|  |     # wpłaty / wydatki z filtrem ukrycia | ||||||
|  |     wplaty = [ | ||||||
|  |         {"typ": "wpłata", "kwota": w.kwota, "opis": w.opis, "data": w.data, "ukryta": getattr(w, "ukryta", False)} | ||||||
|         for w in zb.wplaty |         for w in zb.wplaty | ||||||
|     ] + [ |         if show_hidden or not getattr(w, "ukryta", False) | ||||||
|         {"typ": "wydatek", "kwota": x.kwota, "opis": x.opis, "data": x.data} |  | ||||||
|         for x in zb.wydatki |  | ||||||
|     ] |     ] | ||||||
|  |     wydatki = [ | ||||||
|  |         {"typ": "wydatek", "kwota": x.kwota, "opis": x.opis, "data": x.data, "ukryta": getattr(x, "ukryta", False)} | ||||||
|  |         for x in zb.wydatki | ||||||
|  |         if show_hidden or not getattr(x, "ukryta", False) | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     aktywnosci = wplaty + wydatki | ||||||
|     aktywnosci.sort(key=lambda a: a["data"], reverse=True) |     aktywnosci.sort(key=lambda a: a["data"], reverse=True) | ||||||
|  |  | ||||||
|     return render_template("zbiorka.html", zbiorka=zb, aktywnosci=aktywnosci) |     return render_template("zbiorka.html", zbiorka=zb, aktywnosci=aktywnosci, show_hidden=show_hidden) | ||||||
|  |  | ||||||
|  |  | ||||||
| # TRASY LOGOWANIA I REJESTRACJI | # TRASY LOGOWANIA I REJESTRACJI | ||||||
| @@ -311,34 +323,22 @@ def zbiorka(zbiorka_id): | |||||||
|  |  | ||||||
| @app.route("/zaloguj", methods=["GET", "POST"]) | @app.route("/zaloguj", methods=["GET", "POST"]) | ||||||
| def zaloguj(): | def zaloguj(): | ||||||
|     # Pobierz ustawienia globalne, w tym dozwolone hosty |     settings = UstawieniaGlobalne.query.first() | ||||||
|     settings = GlobalSettings.query.first() |     allowed_hosts_str = settings.dozwolone_hosty_logowania or "" if settings else "" | ||||||
|     allowed_hosts_str = "" |  | ||||||
|     if settings and settings.allowed_login_hosts: |  | ||||||
|         allowed_hosts_str = settings.allowed_login_hosts |  | ||||||
|  |  | ||||||
|     # Sprawdzenie, czy adres IP klienta jest dozwolony |  | ||||||
|     client_ip = get_real_ip() |     client_ip = get_real_ip() | ||||||
|     if not is_allowed_ip(client_ip, allowed_hosts_str): |     if not is_allowed_ip(client_ip, allowed_hosts_str): | ||||||
|         flash( |         flash("Dostęp do tego systemu jest zablokowany dla Twojego adresu IP", "danger") | ||||||
|             "Dostęp do tego systemu jest zablokowany dla Twojego adresu IP", |  | ||||||
|             "danger", |  | ||||||
|         ) |  | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|  |  | ||||||
|     if request.method == "POST": |     if request.method == "POST": | ||||||
|         username = request.form["username"] |         login = request.form["uzytkownik"] | ||||||
|         password = request.form["password"] |         password = request.form["haslo"] | ||||||
|         user = User.query.filter_by(username=username).first() |         user = Uzytkownik.query.filter_by(uzytkownik=login).first() | ||||||
|         if user and user.check_password(password): |         if user and user.check_password(password): | ||||||
|             login_user(user) |             login_user(user) | ||||||
|             flash("Zalogowano pomyślnie", "success") |             flash("Zalogowano pomyślnie", "success") | ||||||
|             next_page = request.args.get("next") |             next_page = request.args.get("next") | ||||||
|             return ( |             return redirect(next_page) if next_page else redirect(url_for("admin_dashboard")) | ||||||
|                 redirect(next_page) |  | ||||||
|                 if next_page |  | ||||||
|                 else redirect(url_for("admin_dashboard")) |  | ||||||
|             ) |  | ||||||
|         else: |         else: | ||||||
|             flash("Nieprawidłowe dane logowania", "danger") |             flash("Nieprawidłowe dane logowania", "danger") | ||||||
|     return render_template("login.html") |     return render_template("login.html") | ||||||
| @@ -358,12 +358,12 @@ def zarejestruj(): | |||||||
|         flash("Rejestracja została wyłączona przez administratora", "danger") |         flash("Rejestracja została wyłączona przez administratora", "danger") | ||||||
|         return redirect(url_for("zaloguj")) |         return redirect(url_for("zaloguj")) | ||||||
|     if request.method == "POST": |     if request.method == "POST": | ||||||
|         username = request.form["username"] |         login = request.form["uzytkownik"] | ||||||
|         password = request.form["password"] |         password = request.form["haslo"] | ||||||
|         if User.query.filter_by(username=username).first(): |         if Uzytkownik.query.filter_by(uzytkownik=login).first(): | ||||||
|             flash("Użytkownik już istnieje", "danger") |             flash("Użytkownik już istnieje", "danger") | ||||||
|             return redirect(url_for("register")) |             return redirect(url_for("register")) | ||||||
|         new_user = User(username=username) |         new_user = Uzytkownik(uzytkownik=login) | ||||||
|         new_user.set_password(password) |         new_user.set_password(password) | ||||||
|         db.session.add(new_user) |         db.session.add(new_user) | ||||||
|         db.session.commit() |         db.session.commit() | ||||||
| @@ -376,7 +376,7 @@ def zarejestruj(): | |||||||
| @app.route("/admin") | @app.route("/admin") | ||||||
| @login_required | @login_required | ||||||
| def admin_dashboard(): | def admin_dashboard(): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień do panelu administracyjnego", "danger") |         flash("Brak uprawnień do panelu administracyjnego", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|     active_zbiorki = Zbiorka.query.filter_by(zrealizowana=False).all() |     active_zbiorki = Zbiorka.query.filter_by(zrealizowana=False).all() | ||||||
| @@ -392,81 +392,118 @@ def admin_dashboard(): | |||||||
| @app.route("/admin/zbiorka/edytuj/<int:zbiorka_id>", methods=["GET", "POST"]) | @app.route("/admin/zbiorka/edytuj/<int:zbiorka_id>", methods=["GET", "POST"]) | ||||||
| @login_required | @login_required | ||||||
| def formularz_zbiorek(zbiorka_id=None): | def formularz_zbiorek(zbiorka_id=None): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger") |         flash("Brak uprawnień", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|  |  | ||||||
|     # Tryb |  | ||||||
|     is_edit = zbiorka_id is not None |     is_edit = zbiorka_id is not None | ||||||
|     zb = db.session.get(Zbiorka, zbiorka_id) |     zb = db.session.get(Zbiorka, zbiorka_id) if is_edit else None | ||||||
|     if zb is None: |     if is_edit and zb is None: | ||||||
|         abort(404) |         abort(404) | ||||||
|     global_settings = GlobalSettings.query.first() |  | ||||||
|  |     global_settings = UstawieniaGlobalne.query.first() | ||||||
|  |  | ||||||
|  |     def _temp_obj(): | ||||||
|  |         t = zb or Zbiorka() | ||||||
|  |         t.nazwa = (request.form.get("nazwa", "") or "").strip() | ||||||
|  |         t.opis = (request.form.get("opis", "") or "").strip() | ||||||
|  |         t.numer_konta = (request.form.get("numer_konta", "") or "").strip() | ||||||
|  |         t.numer_telefonu_blik = (request.form.get("numer_telefonu_blik", "") or "").strip() | ||||||
|  |         t.ukryj_kwote = "ukryj_kwote" in request.form | ||||||
|  |         t.pokaz_postep_finanse = "pokaz_postep_finanse" in request.form | ||||||
|  |         t.pokaz_postep_pozycje = "pokaz_postep_pozycje" in request.form | ||||||
|  |         t.pokaz_postep_kwotowo = "pokaz_postep_kwotowo" in request.form | ||||||
|  |         t.uzyj_konta = "uzyj_konta" in request.form | ||||||
|  |         t.uzyj_blik = "uzyj_blik" in request.form | ||||||
|  |         return t | ||||||
|  |  | ||||||
|     if request.method == "POST": |     if request.method == "POST": | ||||||
|         # Pola wspólne |         # Pola | ||||||
|         nazwa = request.form.get("nazwa", "").strip() |         nazwa = (request.form.get("nazwa", "") or "").strip() | ||||||
|         opis = request.form.get("opis", "").strip() |         opis = (request.form.get("opis", "") or "").strip() | ||||||
|  |         numer_konta = (request.form.get("numer_konta", "") or "").strip() | ||||||
|  |         numer_telefonu_blik = (request.form.get("numer_telefonu_blik", "") or "").strip() | ||||||
|  |  | ||||||
|         # IBAN/telefon — oczyść z nadmiarowych znaków odstępu (zostaw spacje w prezentacji frontu) |         # Przełączniki płatności | ||||||
|         numer_konta = request.form.get("numer_konta", "").strip() |         uzyj_konta = "uzyj_konta" in request.form | ||||||
|         numer_telefonu_blik = request.form.get("numer_telefonu_blik", "").strip() |         uzyj_blik = "uzyj_blik" in request.form | ||||||
|  |  | ||||||
|         # Cel — walidacja liczby (Decimal, nie float) |         # Widoczność/metryki | ||||||
|  |         ukryj_kwote = "ukryj_kwote" in request.form | ||||||
|  |         pokaz_postep_finanse = "pokaz_postep_finanse" in request.form | ||||||
|  |         pokaz_postep_pozycje = "pokaz_postep_pozycje" in request.form | ||||||
|  |         pokaz_postep_kwotowo = "pokaz_postep_kwotowo" in request.form | ||||||
|  |  | ||||||
|  |         # Walidacje | ||||||
|  |         if not nazwa: | ||||||
|  |             flash("Nazwa jest wymagana", "danger") | ||||||
|  |             return render_template("admin/formularz_zbiorek.html", zbiorka=_temp_obj(), global_settings=global_settings) | ||||||
|  |  | ||||||
|  |         if not opis: | ||||||
|  |             flash("Opis jest wymagany", "danger") | ||||||
|  |             return render_template("admin/formularz_zbiorek.html", zbiorka=_temp_obj(), global_settings=global_settings) | ||||||
|  |  | ||||||
|  |         # Co najmniej jeden kanał | ||||||
|  |         if not (uzyj_konta or uzyj_blik): | ||||||
|  |             flash("Włącz co najmniej jeden kanał wpłat (konto lub BLIK).", "danger") | ||||||
|  |             return render_template("admin/formularz_zbiorek.html", zbiorka=_temp_obj(), global_settings=global_settings) | ||||||
|  |  | ||||||
|  |         # Warunkowe wartości | ||||||
|  |         if uzyj_konta and not numer_konta: | ||||||
|  |             flash("Numer konta jest wymagany (kanał przelewu włączony).", "danger") | ||||||
|  |             return render_template("admin/formularz_zbiorek.html", zbiorka=_temp_obj(), global_settings=global_settings) | ||||||
|  |  | ||||||
|  |         if uzyj_blik and not numer_telefonu_blik: | ||||||
|  |             flash("Numer telefonu BLIK jest wymagany (kanał BLIK włączony).", "danger") | ||||||
|  |             return render_template("admin/formularz_zbiorek.html", zbiorka=_temp_obj(), global_settings=global_settings) | ||||||
|  |  | ||||||
|  |         # Cel > 0 | ||||||
|  |         cel_raw = (request.form.get("cel", "") or "") | ||||||
|  |         cel_norm = cel_raw.replace(" ", "").replace("\u00A0", "").replace(",", ".").strip() | ||||||
|         try: |         try: | ||||||
|             cel_str = request.form.get("cel", "").replace(",", ".").strip() |             if not cel_norm: | ||||||
|             cel = Decimal(cel_str) |                 raise InvalidOperation | ||||||
|  |             cel = Decimal(cel_norm) | ||||||
|             if cel <= Decimal("0"): |             if cel <= Decimal("0"): | ||||||
|                 raise InvalidOperation |                 raise InvalidOperation | ||||||
|         except (InvalidOperation, ValueError): |         except (InvalidOperation, ValueError): | ||||||
|             flash("Podano nieprawidłową wartość dla celu zbiórki", "danger") |             flash("Podano nieprawidłową wartość dla celu zbiórki", "danger") | ||||||
|             # render z dotychczasowo wpisanymi danymi (w trybie dodawania tworzymy tymczasowy obiekt) |             return render_template("admin/formularz_zbiorek.html", zbiorka=_temp_obj(), global_settings=global_settings) | ||||||
|             temp_zb = zb or Zbiorka( |  | ||||||
|                 nazwa=nazwa, |  | ||||||
|                 opis=opis, |  | ||||||
|                 numer_konta=numer_konta, |  | ||||||
|                 numer_telefonu_blik=numer_telefonu_blik, |  | ||||||
|                 cel=None, |  | ||||||
|                 ukryj_kwote=("ukryj_kwote" in request.form), |  | ||||||
|             ) |  | ||||||
|             return render_template( |  | ||||||
|                 "admin/formularz_zbiorek.html", |  | ||||||
|                 zbiorka=temp_zb if is_edit else None if zb is None else temp_zb, |  | ||||||
|                 global_settings=global_settings, |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|         ukryj_kwote = "ukryj_kwote" in request.form |  | ||||||
|  |  | ||||||
|  |         # Produkty | ||||||
|         names = request.form.getlist("item_nazwa[]") |         names = request.form.getlist("item_nazwa[]") | ||||||
|         links = request.form.getlist("item_link[]") |         links = request.form.getlist("item_link[]") | ||||||
|         prices = request.form.getlist("item_cena[]") |         prices = request.form.getlist("item_cena[]") | ||||||
|  |  | ||||||
|         def _read_price(val): |         def _read_price(val: str): | ||||||
|             """Zwraca Decimal(>=0) albo None; akceptuje przecinek jako separator dziesiętny.""" |  | ||||||
|             if not val or not val.strip(): |             if not val or not val.strip(): | ||||||
|                 return None |                 return None | ||||||
|             try: |             try: | ||||||
|                 d = Decimal(val.replace(",", ".")) |                 d = Decimal(val.replace(",", ".")) | ||||||
|                 if d < 0: |                 return d if d >= 0 else None | ||||||
|                     return None |  | ||||||
|                 return d |  | ||||||
|             except Exception: |             except Exception: | ||||||
|                 return None |                 return None | ||||||
|  |  | ||||||
|         # --- ZAPIS ZBIÓRKI + PRODUKTÓW --- |         # Zapis | ||||||
|         if is_edit: |         if is_edit: | ||||||
|             # Aktualizacja istniejącej zbiórki |  | ||||||
|             zb.nazwa = nazwa |             zb.nazwa = nazwa | ||||||
|             zb.opis = opis |             zb.opis = opis | ||||||
|             zb.numer_konta = numer_konta |  | ||||||
|             zb.numer_telefonu_blik = numer_telefonu_blik |             # NOT NULL-safe: puste stringi gdy wyłączone | ||||||
|             zb.cel = cel                    # ❗ bez float(cel) — zostaje Decimal |             zb.uzyj_konta = uzyj_konta | ||||||
|  |             zb.uzyj_blik = uzyj_blik | ||||||
|  |             zb.numer_konta = numer_konta if uzyj_konta else "" | ||||||
|  |             zb.numer_telefonu_blik = numer_telefonu_blik if uzyj_blik else "" | ||||||
|  |  | ||||||
|  |             zb.cel = cel | ||||||
|             zb.ukryj_kwote = ukryj_kwote |             zb.ukryj_kwote = ukryj_kwote | ||||||
|             db.session.commit()  # najpierw zapisz bazowe pola |             zb.pokaz_postep_finanse = pokaz_postep_finanse | ||||||
|  |             zb.pokaz_postep_pozycje = pokaz_postep_pozycje | ||||||
|  |             zb.pokaz_postep_kwotowo = pokaz_postep_kwotowo | ||||||
|  |             db.session.commit() | ||||||
|  |  | ||||||
|             # Nadpisz listę produktów (czyść i dodaj od nowa dla prostoty) |             # Nadpisz pozycje | ||||||
|             zb.przedmioty.clear() |             zb.przedmioty.clear() | ||||||
|  |  | ||||||
|             for i, raw_name in enumerate(names): |             for i, raw_name in enumerate(names): | ||||||
|                 name = (raw_name or "").strip() |                 name = (raw_name or "").strip() | ||||||
|                 if not name: |                 if not name: | ||||||
| @@ -474,32 +511,33 @@ def formularz_zbiorek(zbiorka_id=None): | |||||||
|                 link = (links[i] if i < len(links) else "").strip() or None |                 link = (links[i] if i < len(links) else "").strip() or None | ||||||
|                 cena_val = _read_price(prices[i] if i < len(prices) else "") |                 cena_val = _read_price(prices[i] if i < len(prices) else "") | ||||||
|                 kupione_val = request.form.get(f"item_kupione_val_{i}") == "1" |                 kupione_val = request.form.get(f"item_kupione_val_{i}") == "1" | ||||||
|  |  | ||||||
|                 db.session.add(Przedmiot( |                 db.session.add(Przedmiot( | ||||||
|                     zbiorka_id=zb.id, |                     zbiorka_id=zb.id, | ||||||
|                     nazwa=name, |                     nazwa=name, | ||||||
|                     link=link, |                     link=link, | ||||||
|                     cena=cena_val,           # Decimal albo None |                     cena=cena_val, | ||||||
|                     kupione=kupione_val |                     kupione=kupione_val | ||||||
|                 )) |                 )) | ||||||
|  |  | ||||||
|             db.session.commit() |             db.session.commit() | ||||||
|             flash("Zbiórka została zaktualizowana", "success") |             flash("Zbiórka została zaktualizowana", "success") | ||||||
|  |  | ||||||
|         else: |         else: | ||||||
|             # Utworzenie nowej zbiórki |  | ||||||
|             nowa = Zbiorka( |             nowa = Zbiorka( | ||||||
|                 nazwa=nazwa, |                 nazwa=nazwa, | ||||||
|                 opis=opis, |                 opis=opis, | ||||||
|                 numer_konta=numer_konta, |                 uzyj_konta=uzyj_konta, | ||||||
|                 numer_telefonu_blik=numer_telefonu_blik, |                 uzyj_blik=uzyj_blik, | ||||||
|                 cel=cel,                    # ❗ Decimal |                 numer_konta=(numer_konta if uzyj_konta else ""),              | ||||||
|  |                 numer_telefonu_blik=(numer_telefonu_blik if uzyj_blik else ""), | ||||||
|  |                 cel=cel, | ||||||
|                 ukryj_kwote=ukryj_kwote, |                 ukryj_kwote=ukryj_kwote, | ||||||
|  |                 pokaz_postep_finanse=pokaz_postep_finanse, | ||||||
|  |                 pokaz_postep_pozycje=pokaz_postep_pozycje, | ||||||
|  |                 pokaz_postep_kwotowo=pokaz_postep_kwotowo, | ||||||
|             ) |             ) | ||||||
|             db.session.add(nowa) |             db.session.add(nowa) | ||||||
|             db.session.commit()  # potrzebujemy ID nowej zbiórki |             db.session.commit()  # potrzebne ID | ||||||
|  |  | ||||||
|             # Dodaj produkty do nowej zbiórki |  | ||||||
|             for i, raw_name in enumerate(names): |             for i, raw_name in enumerate(names): | ||||||
|                 name = (raw_name or "").strip() |                 name = (raw_name or "").strip() | ||||||
|                 if not name: |                 if not name: | ||||||
| @@ -507,15 +545,13 @@ def formularz_zbiorek(zbiorka_id=None): | |||||||
|                 link = (links[i] if i < len(links) else "").strip() or None |                 link = (links[i] if i < len(links) else "").strip() or None | ||||||
|                 cena_val = _read_price(prices[i] if i < len(prices) else "") |                 cena_val = _read_price(prices[i] if i < len(prices) else "") | ||||||
|                 kupione_val = request.form.get(f"item_kupione_val_{i}") == "1" |                 kupione_val = request.form.get(f"item_kupione_val_{i}") == "1" | ||||||
|  |  | ||||||
|                 db.session.add(Przedmiot( |                 db.session.add(Przedmiot( | ||||||
|                     zbiorka_id=nowa.id, |                     zbiorka_id=nowa.id, | ||||||
|                     nazwa=name, |                     nazwa=name, | ||||||
|                     link=link, |                     link=link, | ||||||
|                     cena=cena_val,           # Decimal albo None |                     cena=cena_val, | ||||||
|                     kupione=kupione_val |                     kupione=kupione_val | ||||||
|                 )) |                 )) | ||||||
|  |  | ||||||
|             db.session.commit() |             db.session.commit() | ||||||
|             flash("Zbiórka została dodana", "success") |             flash("Zbiórka została dodana", "success") | ||||||
|  |  | ||||||
| @@ -523,14 +559,16 @@ def formularz_zbiorek(zbiorka_id=None): | |||||||
|  |  | ||||||
|     # GET |     # GET | ||||||
|     return render_template( |     return render_template( | ||||||
|         "admin/formularz_zbiorek.html", zbiorka=zb, global_settings=global_settings |         "admin/formularz_zbiorek.html", | ||||||
|  |         zbiorka=zb, | ||||||
|  |         global_settings=global_settings | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| @app.route("/admin/zbiorka/<int:zbiorka_id>/wplata/dodaj", methods=["GET", "POST"]) | @app.route("/admin/zbiorka/<int:zbiorka_id>/wplata/dodaj", methods=["GET", "POST"]) | ||||||
| @login_required | @login_required | ||||||
| def dodaj_wplate(zbiorka_id): | def dodaj_wplate(zbiorka_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger") |         flash("Brak uprawnień", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|  |  | ||||||
| @@ -562,7 +600,7 @@ def dodaj_wplate(zbiorka_id): | |||||||
| @app.route("/admin/zbiorka/usun/<int:zbiorka_id>", methods=["POST"]) | @app.route("/admin/zbiorka/usun/<int:zbiorka_id>", methods=["POST"]) | ||||||
| @login_required | @login_required | ||||||
| def usun_zbiorka(zbiorka_id): | def usun_zbiorka(zbiorka_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger") |         flash("Brak uprawnień", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|     zb = db.session.get(Zbiorka, zbiorka_id) |     zb = db.session.get(Zbiorka, zbiorka_id) | ||||||
| @@ -577,7 +615,7 @@ def usun_zbiorka(zbiorka_id): | |||||||
| @app.route("/admin/zbiorka/edytuj_stan/<int:zbiorka_id>", methods=["GET", "POST"]) | @app.route("/admin/zbiorka/edytuj_stan/<int:zbiorka_id>", methods=["GET", "POST"]) | ||||||
| @login_required | @login_required | ||||||
| def edytuj_stan(zbiorka_id): | def edytuj_stan(zbiorka_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger") |         flash("Brak uprawnień", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|     zb = db.session.get(Zbiorka, zbiorka_id) |     zb = db.session.get(Zbiorka, zbiorka_id) | ||||||
| @@ -599,7 +637,7 @@ def edytuj_stan(zbiorka_id): | |||||||
| @app.route("/admin/zbiorka/zmien_widzialnosc/<int:zbiorka_id>", methods=["POST"]) | @app.route("/admin/zbiorka/zmien_widzialnosc/<int:zbiorka_id>", methods=["POST"]) | ||||||
| @login_required | @login_required | ||||||
| def zmien_widzialnosc(zbiorka_id): | def zmien_widzialnosc(zbiorka_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger") |         flash("Brak uprawnień", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|     zb = db.session.get(Zbiorka, zbiorka_id) |     zb = db.session.get(Zbiorka, zbiorka_id) | ||||||
| @@ -612,15 +650,17 @@ def zmien_widzialnosc(zbiorka_id): | |||||||
|  |  | ||||||
|  |  | ||||||
| def create_admin_account(): | def create_admin_account(): | ||||||
|     admin = User.query.filter_by(is_admin=True).first() |     admin = Uzytkownik.query.filter_by(czy_admin=True).first() | ||||||
|     if not admin: |     if not admin: | ||||||
|         main_admin = User(username=app.config["MAIN_ADMIN_USERNAME"], is_admin=True) |         main_admin = Uzytkownik( | ||||||
|  |             uzytkownik=app.config["MAIN_ADMIN_USERNAME"], | ||||||
|  |             czy_admin=True | ||||||
|  |         ) | ||||||
|         main_admin.set_password(app.config["MAIN_ADMIN_PASSWORD"]) |         main_admin.set_password(app.config["MAIN_ADMIN_PASSWORD"]) | ||||||
|         db.session.add(main_admin) |         db.session.add(main_admin) | ||||||
|         db.session.commit() |         db.session.commit() | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @app.after_request | @app.after_request | ||||||
| def apply_headers(response): | def apply_headers(response): | ||||||
|     if request.path.startswith("/static/"): |     if request.path.startswith("/static/"): | ||||||
| @@ -635,9 +675,9 @@ def apply_headers(response): | |||||||
|         return response |         return response | ||||||
|  |  | ||||||
|     path_norm = request.path.lstrip("/") |     path_norm = request.path.lstrip("/") | ||||||
|     is_admin = path_norm.startswith("admin/") or path_norm == "admin" |     czy_admin = path_norm.startswith("admin/") or path_norm == "admin" | ||||||
|  |  | ||||||
|     if is_admin: |     if czy_admin: | ||||||
|         if (response.mimetype or "").startswith("text/html"): |         if (response.mimetype or "").startswith("text/html"): | ||||||
|             response.headers["Cache-Control"] = "no-store, no-cache" |             response.headers["Cache-Control"] = "no-store, no-cache" | ||||||
|             response.headers.pop("ETag", None) |             response.headers.pop("ETag", None) | ||||||
| @@ -664,7 +704,7 @@ def apply_headers(response): | |||||||
|  |  | ||||||
|     if ( |     if ( | ||||||
|         app.config.get("BLOCK_BOTS", False) |         app.config.get("BLOCK_BOTS", False) | ||||||
|         and not is_admin |         and not czy_admin | ||||||
|         and not request.path.startswith("/static/") |         and not request.path.startswith("/static/") | ||||||
|     ): |     ): | ||||||
|         cc_override = app.config.get("CACHE_CONTROL_HEADER") |         cc_override = app.config.get("CACHE_CONTROL_HEADER") | ||||||
| @@ -680,60 +720,58 @@ def apply_headers(response): | |||||||
| @app.route("/admin/ustawienia", methods=["GET", "POST"]) | @app.route("/admin/ustawienia", methods=["GET", "POST"]) | ||||||
| @login_required | @login_required | ||||||
| def admin_ustawienia(): | def admin_ustawienia(): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień do panelu administracyjnego", "danger") |         flash("Brak uprawnień do panelu administracyjnego", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|  |  | ||||||
|     client_ip = get_real_ip() |     client_ip = get_real_ip() | ||||||
|     settings = GlobalSettings.query.first() |     settings = UstawieniaGlobalne.query.first() | ||||||
|     if request.method == "POST": |     if request.method == "POST": | ||||||
|         numer_konta = request.form.get("numer_konta") |         numer_konta = request.form.get("numer_konta") | ||||||
|         numer_telefonu_blik = request.form.get("numer_telefonu_blik") |         numer_telefonu_blik = request.form.get("numer_telefonu_blik") | ||||||
|         allowed_login_hosts = request.form.get("allowed_login_hosts") |         dozwolone_hosty_logowania = request.form.get("dozwolone_hosty_logowania") | ||||||
|         logo_url = request.form.get("logo_url") |         logo_url = request.form.get("logo_url") | ||||||
|         site_title = request.form.get("site_title") |         tytul_strony = request.form.get("tytul_strony") | ||||||
|         navbar_brand_mode = request.form.get("navbar_brand_mode", "text") |         typ_navbar = request.form.get("typ_navbar", "text") | ||||||
|         footer_brand_mode = request.form.get("footer_brand_mode", "text") |         typ_stopka = request.form.get("typ_stopka", "text") | ||||||
|         footer_text = request.form.get("footer_text") or None |         stopka_text = request.form.get("stopka_text") or None | ||||||
|         show_logo_in_navbar = navbar_brand_mode == "logo" |         pokaz_logo_w_navbar = (typ_navbar == "logo") | ||||||
|  |  | ||||||
|         if settings is None: |         if settings is None: | ||||||
|             settings = GlobalSettings( |             settings = UstawieniaGlobalne( | ||||||
|                 numer_konta=numer_konta, |                 numer_konta=numer_konta, | ||||||
|                 numer_telefonu_blik=numer_telefonu_blik, |                 numer_telefonu_blik=numer_telefonu_blik, | ||||||
|                 allowed_login_hosts=allowed_login_hosts, |                 dozwolone_hosty_logowania=dozwolone_hosty_logowania, | ||||||
|                 logo_url=logo_url, |                 logo_url=logo_url, | ||||||
|                 site_title=site_title, |                 tytul_strony=tytul_strony, | ||||||
|                 show_logo_in_navbar=show_logo_in_navbar, |                 pokaz_logo_w_navbar=pokaz_logo_w_navbar, | ||||||
|                 navbar_brand_mode=navbar_brand_mode, |                 typ_navbar=typ_navbar, | ||||||
|                 footer_brand_mode=footer_brand_mode, |                 typ_stopka=typ_stopka, | ||||||
|                 footer_text=footer_text, |                 stopka_text=stopka_text, | ||||||
|             ) |             ) | ||||||
|             db.session.add(settings) |             db.session.add(settings) | ||||||
|         else: |         else: | ||||||
|             settings.numer_konta = numer_konta |             settings.numer_konta = numer_konta | ||||||
|             settings.numer_telefonu_blik = numer_telefonu_blik |             settings.numer_telefonu_blik = numer_telefonu_blik | ||||||
|             settings.allowed_login_hosts = allowed_login_hosts |             settings.dozwolone_hosty_logowania = dozwolone_hosty_logowania | ||||||
|             settings.logo_url = logo_url |             settings.logo_url = logo_url | ||||||
|             settings.site_title = site_title |             settings.tytul_strony = tytul_strony | ||||||
|             settings.show_logo_in_navbar = show_logo_in_navbar |             settings.pokaz_logo_w_navbar = pokaz_logo_w_navbar | ||||||
|             settings.navbar_brand_mode = navbar_brand_mode |             settings.typ_navbar = typ_navbar | ||||||
|             settings.footer_brand_mode = footer_brand_mode |             settings.typ_stopka = typ_stopka | ||||||
|             settings.footer_text = footer_text |             settings.stopka_text = stopka_text | ||||||
|  |  | ||||||
|         db.session.commit() |         db.session.commit() | ||||||
|         flash("Ustawienia globalne zostały zaktualizowane", "success") |         flash("Ustawienia globalne zostały zaktualizowane", "success") | ||||||
|         return redirect(url_for("admin_dashboard")) |         return redirect(url_for("admin_dashboard")) | ||||||
|  |  | ||||||
|     return render_template( |     return render_template("admin/ustawienia.html", settings=settings, client_ip=client_ip) | ||||||
|         "admin/ustawienia.html", settings=settings, client_ip=client_ip |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @app.route("/admin/zbiorka/<int:zbiorka_id>/wydatek/dodaj", methods=["GET", "POST"]) | @app.route("/admin/zbiorka/<int:zbiorka_id>/wydatek/dodaj", methods=["GET", "POST"]) | ||||||
| @login_required | @login_required | ||||||
| def dodaj_wydatek(zbiorka_id): | def dodaj_wydatek(zbiorka_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger") |         flash("Brak uprawnień", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|  |  | ||||||
| @@ -775,7 +813,7 @@ def dodaj_wydatek(zbiorka_id): | |||||||
| ) | ) | ||||||
| @login_required | @login_required | ||||||
| def oznacz_zbiorka(zbiorka_id): | def oznacz_zbiorka(zbiorka_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień do wykonania tej operacji", "danger") |         flash("Brak uprawnień do wykonania tej operacji", "danger") | ||||||
|         return redirect(url_for("index")) |         return redirect(url_for("index")) | ||||||
|  |  | ||||||
| @@ -807,23 +845,47 @@ def robots(): | |||||||
| @app.route("/admin/zbiorka/<int:zbiorka_id>/transakcje") | @app.route("/admin/zbiorka/<int:zbiorka_id>/transakcje") | ||||||
| @login_required | @login_required | ||||||
| def transakcje_zbiorki(zbiorka_id): | def transakcje_zbiorki(zbiorka_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) |         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) | ||||||
|  |  | ||||||
|     zb = db.session.get(Zbiorka, zbiorka_id) |     zb = db.session.get(Zbiorka, zbiorka_id) | ||||||
|     if zb is None: |     if zb is None: | ||||||
|         abort(404) |         abort(404) | ||||||
|  |  | ||||||
|     aktywnosci = ( |     aktywnosci = ( | ||||||
|         [{"typ": "wpłata", "id": w.id, "kwota": w.kwota, "opis": w.opis, "data": w.data} for w in zb.wplaty] + |         [ | ||||||
|         [{"typ": "wydatek","id": x.id, "kwota": x.kwota,"opis": x.opis,"data": x.data} for x in zb.wydatki] |             { | ||||||
|  |                 "typ": "wpłata", | ||||||
|  |                 "id": w.id, | ||||||
|  |                 "kwota": w.kwota, | ||||||
|  |                 "opis": w.opis, | ||||||
|  |                 "data": w.data, | ||||||
|  |                 "ukryta": bool(w.ukryta), | ||||||
|  |             } | ||||||
|  |             for w in zb.wplaty | ||||||
|  |         ] | ||||||
|  |         + | ||||||
|  |         [ | ||||||
|  |             { | ||||||
|  |                 "typ": "wydatek", | ||||||
|  |                 "id": x.id, | ||||||
|  |                 "kwota": x.kwota, | ||||||
|  |                 "opis": x.opis, | ||||||
|  |                 "data": x.data, | ||||||
|  |                 "ukryta": bool(x.ukryta), | ||||||
|  |             } | ||||||
|  |             for x in zb.wydatki | ||||||
|  |         ] | ||||||
|     ) |     ) | ||||||
|     aktywnosci.sort(key=lambda a: a["data"], reverse=True) |     aktywnosci.sort(key=lambda a: a["data"], reverse=True) | ||||||
|     return render_template("admin/transakcje.html", zbiorka=zb, aktywnosci=aktywnosci) |     return render_template("admin/transakcje.html", zbiorka=zb, aktywnosci=aktywnosci) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @app.route("/admin/wplata/<int:wplata_id>/zapisz", methods=["POST"]) | @app.route("/admin/wplata/<int:wplata_id>/zapisz", methods=["POST"]) | ||||||
| @login_required | @login_required | ||||||
| def zapisz_wplate(wplata_id): | def zapisz_wplate(wplata_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) |         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) | ||||||
|     w = db.session.get(Wplata, wplata_id) |     w = db.session.get(Wplata, wplata_id) | ||||||
|     if w is None: |     if w is None: | ||||||
| @@ -845,11 +907,55 @@ def zapisz_wplate(wplata_id): | |||||||
|     flash("Wpłata zaktualizowana", "success") |     flash("Wpłata zaktualizowana", "success") | ||||||
|     return redirect(url_for("transakcje_zbiorki", zbiorka_id=zb.id)) |     return redirect(url_for("transakcje_zbiorki", zbiorka_id=zb.id)) | ||||||
|  |  | ||||||
|  | @app.post("/wplata/<int:wplata_id>/ukryj") | ||||||
|  | @login_required | ||||||
|  | def ukryj_wplate(wplata_id): | ||||||
|  |     if not current_user.czy_admin: abort(403) | ||||||
|  |     w = db.session.get(Wplata, wplata_id) | ||||||
|  |     if not w: abort(404) | ||||||
|  |     w.ukryta = True | ||||||
|  |     db.session.commit() | ||||||
|  |     flash("Wpłata ukryta.", "success") | ||||||
|  |     return redirect(request.referrer or url_for("admin_dashboard")) | ||||||
|  |  | ||||||
|  | @app.post("/wplata/<int:wplata_id>/odkryj") | ||||||
|  | @login_required | ||||||
|  | def odkryj_wplate(wplata_id): | ||||||
|  |     if not current_user.czy_admin: abort(403) | ||||||
|  |     w = db.session.get(Wplata, wplata_id) | ||||||
|  |     if not w: abort(404) | ||||||
|  |     w.ukryta = False | ||||||
|  |     db.session.commit() | ||||||
|  |     flash("Wpłata odkryta.", "success") | ||||||
|  |     return redirect(request.referrer or url_for("admin_dashboard")) | ||||||
|  |  | ||||||
|  | @app.post("/wydatek/<int:wydatek_id>/ukryj") | ||||||
|  | @login_required | ||||||
|  | def ukryj_wydatek(wydatek_id): | ||||||
|  |     if not current_user.czy_admin: abort(403) | ||||||
|  |     w = db.session.get(Wydatek, wydatek_id) | ||||||
|  |     if not w: abort(404) | ||||||
|  |     w.ukryta = True | ||||||
|  |     db.session.commit() | ||||||
|  |     flash("Wydatek ukryty.", "success") | ||||||
|  |     return redirect(request.referrer or url_for("admin_dashboard")) | ||||||
|  |  | ||||||
|  | @app.post("/wydatek/<int:wydatek_id>/odkryj") | ||||||
|  | @login_required | ||||||
|  | def odkryj_wydatek(wydatek_id): | ||||||
|  |     if not current_user.czy_admin: abort(403) | ||||||
|  |     w = db.session.get(Wydatek, wydatek_id) | ||||||
|  |     if not w: abort(404) | ||||||
|  |     w.ukryta = False | ||||||
|  |     db.session.commit() | ||||||
|  |     flash("Wydatek odkryty.", "success") | ||||||
|  |     return redirect(request.referrer or url_for("admin_dashboard")) | ||||||
|  |  | ||||||
|  |  | ||||||
| @app.route("/admin/wplata/<int:wplata_id>/usun", methods=["POST"]) | @app.route("/admin/wplata/<int:wplata_id>/usun", methods=["POST"]) | ||||||
| @login_required | @login_required | ||||||
| def usun_wplate(wplata_id): | def usun_wplate(wplata_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) |         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) | ||||||
|     w = db.session.get(Wplata, wplata_id) |     w = db.session.get(Wplata, wplata_id) | ||||||
|     if w is None: |     if w is None: | ||||||
| @@ -865,7 +971,7 @@ def usun_wplate(wplata_id): | |||||||
| @app.route("/admin/wydatek/<int:wydatek_id>/zapisz", methods=["POST"]) | @app.route("/admin/wydatek/<int:wydatek_id>/zapisz", methods=["POST"]) | ||||||
| @login_required | @login_required | ||||||
| def zapisz_wydatek(wydatek_id): | def zapisz_wydatek(wydatek_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) |         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) | ||||||
|     x = db.session.get(Wydatek, wydatek_id) |     x = db.session.get(Wydatek, wydatek_id) | ||||||
|     if x is None: |     if x is None: | ||||||
| @@ -892,7 +998,7 @@ def zapisz_wydatek(wydatek_id): | |||||||
| @app.route("/admin/wydatek/<int:wydatek_id>/usun", methods=["POST"]) | @app.route("/admin/wydatek/<int:wydatek_id>/usun", methods=["POST"]) | ||||||
| @login_required | @login_required | ||||||
| def usun_wydatek(wydatek_id): | def usun_wydatek(wydatek_id): | ||||||
|     if not current_user.is_admin: |     if not current_user.czy_admin: | ||||||
|         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) |         flash("Brak uprawnień", "danger"); return redirect(url_for("index")) | ||||||
|     x = db.session.get(Wydatek, wydatek_id) |     x = db.session.get(Wydatek, wydatek_id) | ||||||
|     if x is None: |     if x is None: | ||||||
| @@ -923,14 +1029,12 @@ def healthcheck(): | |||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|     with app.app_context(): |     with app.app_context(): | ||||||
|         db.create_all() |         db.create_all() | ||||||
|         # Tworzenie konta głównego admina, jeśli nie istnieje |         stmt = select(Uzytkownik).filter_by(czy_admin=True) | ||||||
|         stmt = select(User).filter_by(is_admin=True) |  | ||||||
|         admin = db.session.execute(stmt).scalars().first() |         admin = db.session.execute(stmt).scalars().first() | ||||||
|  |  | ||||||
|         if not admin: |         if not admin: | ||||||
|             main_admin = User( |             main_admin = Uzytkownik( | ||||||
|                 username=app.config["MAIN_ADMIN_USERNAME"], |                 uzytkownik=app.config["MAIN_ADMIN_USERNAME"], | ||||||
|                 is_admin=True |                 czy_admin=True | ||||||
|             ) |             ) | ||||||
|             main_admin.set_password(app.config["MAIN_ADMIN_PASSWORD"]) |             main_admin.set_password(app.config["MAIN_ADMIN_PASSWORD"]) | ||||||
|             db.session.add(main_admin) |             db.session.add(main_admin) | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								config.py
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								config.py
									
									
									
									
									
								
							| @@ -1,5 +1,7 @@ | |||||||
| import os | import os | ||||||
|  |  | ||||||
|  | basedir = os.path.abspath(os.path.dirname(__file__)) | ||||||
|  |  | ||||||
| def _get_bool(name: str, default: bool) -> bool: | def _get_bool(name: str, default: bool) -> bool: | ||||||
|     val = os.environ.get(name) |     val = os.environ.get(name) | ||||||
|     if val is None: |     if val is None: | ||||||
| @@ -24,8 +26,8 @@ class Config: | |||||||
|       - ROBOTS_TAG |       - ROBOTS_TAG | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     # Baza danych |  | ||||||
|     SQLALCHEMY_DATABASE_URI = _get_str("DATABASE_URL", "sqlite:///baza.db") |     #SQLALCHEMY_DATABASE_URI = _get_str("DATABASE_URL", "sqlite:///baza.db") | ||||||
|  |  | ||||||
|     # Flask |     # Flask | ||||||
|     SECRET_KEY = _get_str("SECRET_KEY", "tajny_klucz") |     SECRET_KEY = _get_str("SECRET_KEY", "tajny_klucz") | ||||||
| @@ -48,4 +50,22 @@ class Config: | |||||||
|     # (opcjonalnie) wyłącz warningi track_modifications |     # (opcjonalnie) wyłącz warningi track_modifications | ||||||
|     SQLALCHEMY_TRACK_MODIFICATIONS = False |     SQLALCHEMY_TRACK_MODIFICATIONS = False | ||||||
|  |  | ||||||
|     HEALTHCHECK_TOKEN = _get_str("HEALTHCHECK_TOKEN", "healthcheck") |     HEALTHCHECK_TOKEN = _get_str("HEALTHCHECK_TOKEN", "healthcheck") | ||||||
|  |  | ||||||
|  |     # Baza danych | ||||||
|  |     DB_ENGINE = os.environ.get("DB_ENGINE", "sqlite").lower() | ||||||
|  |  | ||||||
|  |     if DB_ENGINE == "sqlite": | ||||||
|  |         SQLALCHEMY_DATABASE_URI = f"sqlite:///{os.path.join(basedir, 'db', 'database.db')}" | ||||||
|  |     elif DB_ENGINE == "pgsql": | ||||||
|  |         SQLALCHEMY_DATABASE_URI = ( | ||||||
|  |             f"postgresql://{os.environ['DB_USER']}:{os.environ['DB_PASSWORD']}@" | ||||||
|  |             f"{os.environ['DB_HOST']}:{os.environ.get('DB_PORT', 5432)}/{os.environ['DB_NAME']}" | ||||||
|  |         ) | ||||||
|  |     elif DB_ENGINE == "mysql": | ||||||
|  |         SQLALCHEMY_DATABASE_URI = ( | ||||||
|  |             f"mysql+pymysql://{os.environ['DB_USER']}:{os.environ['DB_PASSWORD']}@" | ||||||
|  |             f"{os.environ['DB_HOST']}:{os.environ.get('DB_PORT', 3306)}/{os.environ['DB_NAME']}" | ||||||
|  |         ) | ||||||
|  |     else: | ||||||
|  |         raise ValueError("Nieobsługiwany typ bazy danych.") | ||||||
							
								
								
									
										59
									
								
								deploy.sh
									
									
									
									
									
								
							
							
						
						
									
										59
									
								
								deploy.sh
									
									
									
									
									
								
							| @@ -1,59 +0,0 @@ | |||||||
| #!/usr/bin/env bash |  | ||||||
| set -euo pipefail |  | ||||||
|  |  | ||||||
| # === Konfiguracja (możesz nadpisać zmiennymi środowiskowymi) === |  | ||||||
| REPO_DIR="${REPO_DIR:-$(pwd)}" |  | ||||||
| COMPOSE_FILE="${COMPOSE_FILE:-docker-compose.yml}"   # albo compose.yaml |  | ||||||
| GIT_REMOTE="${GIT_REMOTE:-origin}" |  | ||||||
| GIT_BRANCH="${GIT_BRANCH:-$(git -C "$REPO_DIR" rev-parse --abbrev-ref HEAD 2>/dev/null || echo main)}" |  | ||||||
|  |  | ||||||
| # Jeśli chcesz uruchomić tylko wybrane serwisy, podaj je po komendzie, |  | ||||||
| # np.: ./deploy.sh web api |  | ||||||
| SERVICES=("$@") |  | ||||||
|  |  | ||||||
| log() { printf "\n==> %s\n" "$*"; } |  | ||||||
|  |  | ||||||
| # --- Kontrole wstępne --- |  | ||||||
| command -v git >/dev/null || { echo "Brak 'git' w PATH"; exit 1; } |  | ||||||
| command -v docker >/dev/null || { echo "Brak 'docker' w PATH"; exit 1; } |  | ||||||
|  |  | ||||||
| if ! docker compose version >/dev/null 2>&1; then |  | ||||||
|   echo "Wymagany jest 'docker compose' (plugin), nie stary 'docker-compose'." |  | ||||||
|   exit 1 |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| if [[ ! -f "$REPO_DIR/$COMPOSE_FILE" ]]; then |  | ||||||
|   if [[ -f "$REPO_DIR/compose.yaml" ]]; then |  | ||||||
|     COMPOSE_FILE="compose.yaml" |  | ||||||
|   else |  | ||||||
|     echo "Nie znaleziono pliku Compose w: $REPO_DIR/$COMPOSE_FILE" |  | ||||||
|     exit 1 |  | ||||||
|   fi |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # --- Praca w katalogu repo --- |  | ||||||
| cd "$REPO_DIR" |  | ||||||
|  |  | ||||||
| # --- Aktualizacja kodu --- |  | ||||||
| log "Aktualizacja repo: git pull --ff-only ($GIT_REMOTE/$GIT_BRANCH)" |  | ||||||
| git fetch --prune "$GIT_REMOTE" |  | ||||||
| git checkout "$GIT_BRANCH" >/dev/null 2>&1 || true |  | ||||||
| git pull --ff-only "$GIT_REMOTE" "$GIT_BRANCH" |  | ||||||
|  |  | ||||||
| # --- Zapisanie wersji do pliku --- |  | ||||||
| log "Zapisywanie hasha commita do version.txt" |  | ||||||
| git rev-parse --short HEAD > version.txt |  | ||||||
|  |  | ||||||
| # --- Zatrzymanie i usunięcie bieżącego stacka --- |  | ||||||
| log "Docker Compose DOWN (usuwanie kontenerów i osieroconych usług)" |  | ||||||
| docker compose -f "$COMPOSE_FILE" down --remove-orphans |  | ||||||
|  |  | ||||||
| # --- Budowanie i uruchamianie bez restartu zależności --- |  | ||||||
| log "Docker Compose UP (build bez deps) dla: ${SERVICES[*]:-(wszystkie)}" |  | ||||||
| if [[ ${#SERVICES[@]} -gt 0 ]]; then |  | ||||||
|   docker compose -f "$COMPOSE_FILE" up -d --no-deps --build "${SERVICES[@]}" |  | ||||||
| else |  | ||||||
|   docker compose -f "$COMPOSE_FILE" up -d --no-deps --build |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| log "Gotowe ✅  (wersja: $(cat version.txt))" |  | ||||||
							
								
								
									
										18
									
								
								deploy/app/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								deploy/app/Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | #FROM python:3.13-slim | ||||||
|  | FROM python:3.14-rc-trixie | ||||||
|  | WORKDIR /app | ||||||
|  |  | ||||||
|  | RUN apt-get update && apt-get install -y --no-install-recommends \ | ||||||
|  |     && apt-get install -y build-essential \ | ||||||
|  |     && apt-get clean \ | ||||||
|  |     && rm -rf /var/lib/apt/lists/* | ||||||
|  |  | ||||||
|  | COPY requirements.txt requirements.txt | ||||||
|  |  | ||||||
|  | RUN pip install --upgrade pip | ||||||
|  | RUN pip install --no-cache-dir -r requirements.txt | ||||||
|  |  | ||||||
|  | COPY . . | ||||||
|  | RUN mkdir -p /app/instance | ||||||
|  |  | ||||||
|  | CMD ["python", "run_waitress.py"] | ||||||
| @@ -1,58 +0,0 @@ | |||||||
| vcl 4.1; |  | ||||||
|  |  | ||||||
| import vsthrottle; |  | ||||||
|  |  | ||||||
| backend app { |  | ||||||
|   .host = "app"; |  | ||||||
|   .port = "8080"; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| acl purge { "localhost"; "127.0.0.1"; } |  | ||||||
|  |  | ||||||
| sub vcl_recv { |  | ||||||
|   # RATE LIMIT |  | ||||||
|   if (!vsthrottle.is_allowed(client.ip, 10, 10s)) { |  | ||||||
|     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. nagłówka |  | ||||||
|   if (req.url == "/healthcheck" || req.http.X-Internal-Check) { return (pass); } |  | ||||||
|  |  | ||||||
|   # metody inne niż GET/HEAD bez cache |  | ||||||
|   if (req.method != "GET" && req.method != "HEAD") { return (pass); } |  | ||||||
|  |  | ||||||
|   # static – agresywnie cache’ujemy |  | ||||||
|   if (req.url ~ "^/static/" || req.url ~ "\.(css|js|png|jpg|svg|ico|woff2?)$") { return (hash); } |  | ||||||
|  |  | ||||||
|   return (hash); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| sub vcl_backend_response { |  | ||||||
|   if (bereq.url ~ "^/static/" || bereq.url ~ "\.(css|js|png|jpg|svg|ico|woff2?)$") { |  | ||||||
|     set beresp.ttl = 24h; |  | ||||||
|   } else { |  | ||||||
|     if (beresp.http.Cache-Control ~ "no-cache|no-store|private") { |  | ||||||
|       set beresp.uncacheable = true; |  | ||||||
|       set beresp.ttl = 0s; |  | ||||||
|     } else { |  | ||||||
|       set beresp.ttl = 60s;  # domyślny TTL dla HTML/API |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| sub vcl_deliver { |  | ||||||
|   if (obj.hits > 0) { |  | ||||||
|     set resp.http.X-Cache = "HIT"; |  | ||||||
|   } else { |  | ||||||
|     set resp.http.X-Cache = "MISS"; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   set resp.http.X-RateLimit-Limit = "10"; |  | ||||||
|   set resp.http.X-RateLimit-Window = "10s"; |  | ||||||
| } |  | ||||||
							
								
								
									
										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")); | ||||||
|  | } | ||||||
							
								
								
									
										86
									
								
								deploy_docker.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										86
									
								
								deploy_docker.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,86 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  | set -euo pipefail | ||||||
|  |  | ||||||
|  | # --- Wczytaj zmienne z .env --- | ||||||
|  | if [[ -f .env ]]; then | ||||||
|  |   set -a | ||||||
|  |   source .env | ||||||
|  |   set +a | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | APP_PORT="${APP_PORT:-8080}" | ||||||
|  |  | ||||||
|  | REPO_DIR="${REPO_DIR:-$(pwd)}" | ||||||
|  | COMPOSE_FILE="${COMPOSE_FILE:-docker-compose.yml}" | ||||||
|  | GIT_REMOTE="${GIT_REMOTE:-origin}" | ||||||
|  | GIT_BRANCH="${GIT_BRANCH:-$(git -C "$REPO_DIR" rev-parse --abbrev-ref HEAD 2>/dev/null || echo main)}" | ||||||
|  |  | ||||||
|  | # Domyślny profil | ||||||
|  | PROFILE="${1:-sqlite}" | ||||||
|  |  | ||||||
|  | if [[ "$PROFILE" != "pgsql" && "$PROFILE" != "mysql" && "$PROFILE" != "sqlite" ]]; then | ||||||
|  |   echo "Użycie: $0 {pgsql|mysql|sqlite}" | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | SERVICES=() | ||||||
|  | if [[ $# -gt 1 ]]; then | ||||||
|  |   SERVICES=("${@:2}") | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | log() { printf "\n==> %s\n" "$*"; } | ||||||
|  |  | ||||||
|  | command -v git >/dev/null || { echo "Brak 'git' w PATH"; exit 1; } | ||||||
|  | command -v docker >/dev/null || { echo "Brak 'docker' w PATH"; exit 1; } | ||||||
|  | if ! docker compose version >/dev/null 2>&1; then | ||||||
|  |   echo "Wymagany jest 'docker compose' (plugin), nie stary 'docker-compose'." | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | if [[ ! -f "$REPO_DIR/$COMPOSE_FILE" ]]; then | ||||||
|  |   if [[ -f "$REPO_DIR/compose.yaml" ]]; then | ||||||
|  |     COMPOSE_FILE="compose.yaml" | ||||||
|  |   else | ||||||
|  |     echo "Nie znaleziono pliku Compose w: $REPO_DIR/$COMPOSE_FILE" | ||||||
|  |     exit 1 | ||||||
|  |   fi | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | cd "$REPO_DIR" | ||||||
|  |  | ||||||
|  | log "Aktualizacja repo: git pull --ff-only ($GIT_REMOTE/$GIT_BRANCH)" | ||||||
|  | git fetch --prune "$GIT_REMOTE" | ||||||
|  | git checkout "$GIT_BRANCH" >/dev/null 2>&1 || true | ||||||
|  | git pull --ff-only "$GIT_REMOTE" "$GIT_BRANCH" | ||||||
|  |  | ||||||
|  | log "Zapisywanie hasha commita do version.txt" | ||||||
|  | git rev-parse --short HEAD > version.txt | ||||||
|  |  | ||||||
|  | log "Docker Compose DOWN" | ||||||
|  | docker compose --profile "$PROFILE" stop | ||||||
|  |  | ||||||
|  | log "Generowanie default.vcl z APP_PORT=$APP_PORT" | ||||||
|  | envsubst < deploy/varnish/default.vcl.template > deploy/varnish/default.vcl | ||||||
|  |  | ||||||
|  | # Tworzenie katalogów danych dla baz jeśli brak | ||||||
|  | if [[ "$PROFILE" == "pgsql" ]]; then | ||||||
|  |   if [[ ! -d "./db/pgsql" ]]; then | ||||||
|  |     log "Tworzę katalog ./db/pgsql dla danych PostgreSQL" | ||||||
|  |     mkdir -p ./db/pgsql | ||||||
|  |   fi | ||||||
|  | elif [[ "$PROFILE" == "mysql" ]]; then | ||||||
|  |   if [[ ! -d "./db/mysql" ]]; then | ||||||
|  |     log "Tworzę katalog ./db/mysql dla danych MySQL" | ||||||
|  |     mkdir -p ./db/mysql | ||||||
|  |   fi | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | log "Docker Compose UP (build bez deps) dla profilu: $PROFILE i serwisów: ${SERVICES[*]:-(wszystkie)}" | ||||||
|  |  | ||||||
|  | if [[ ${#SERVICES[@]} -gt 0 ]]; then | ||||||
|  |   DB_ENGINE="$PROFILE" docker compose -f "$COMPOSE_FILE" --profile "$PROFILE" up -d --no-deps --build "${SERVICES[@]}" | ||||||
|  | else | ||||||
|  |   DB_ENGINE="$PROFILE" docker compose -f "$COMPOSE_FILE" --profile "$PROFILE" up -d --no-deps --build | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | log "Gotowe ✅  (wersja: $(cat version.txt))" | ||||||
| @@ -16,10 +16,13 @@ services: | |||||||
|       - .env |       - .env | ||||||
|     volumes: |     volumes: | ||||||
|       - ./instance:/app/instance |       - ./instance:/app/instance | ||||||
|  |     networks: | ||||||
|  |       - zbiorki_app_network | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
|   varnish: |   varnish: | ||||||
|     build: ./deploy/varnish |     #build: ./deploy/varnish | ||||||
|  |     image: varnish:latest | ||||||
|     container_name: zbiorka-varnish |     container_name: zbiorka-varnish | ||||||
|     depends_on: |     depends_on: | ||||||
|       app: |       app: | ||||||
| @@ -30,11 +33,42 @@ services: | |||||||
|       - ./deploy/varnish/default.vcl:/etc/varnish/default.vcl:ro |       - ./deploy/varnish/default.vcl:/etc/varnish/default.vcl:ro | ||||||
|     environment: |     environment: | ||||||
|       - VARNISH_SIZE=256m |       - VARNISH_SIZE=256m | ||||||
|     healthcheck: |  | ||||||
|       test: [ "CMD-SHELL", "curl -fsS -H 'X-Internal-Check=${HEALTHCHECK_TOKEN}' http://localhost/healthcheck | grep -q OK" ] |  | ||||||
|       interval: 30s |  | ||||||
|       timeout: 5s |  | ||||||
|       retries: 3 |  | ||||||
|     env_file: |     env_file: | ||||||
|       - .env |       - .env | ||||||
|  |     networks: | ||||||
|  |       - zbiorki_app_network | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
|  |   mysql: | ||||||
|  |     image: mysql:8 | ||||||
|  |     container_name: zbiorka-mysql-db | ||||||
|  |     environment: | ||||||
|  |       MYSQL_DATABASE: ${DB_NAME} | ||||||
|  |       MYSQL_USER: ${DB_USER} | ||||||
|  |       MYSQL_PASSWORD: ${DB_PASSWORD} | ||||||
|  |       MYSQL_ROOT_PASSWORD: 89o38kUX5T4C | ||||||
|  |     volumes: | ||||||
|  |       - ./db/mysql:/var/lib/mysql | ||||||
|  |     restart: unless-stopped | ||||||
|  |     networks: | ||||||
|  |       - zbiorki_app_network | ||||||
|  |     profiles: ["mysql"] | ||||||
|  |  | ||||||
|  |   pgsql: | ||||||
|  |     image: postgres:18 | ||||||
|  |     container_name: zbiorka-pgsql-db | ||||||
|  |     environment: | ||||||
|  |       POSTGRES_DB: ${DB_NAME} | ||||||
|  |       POSTGRES_USER: ${DB_USER} | ||||||
|  |       POSTGRES_PASSWORD: ${DB_PASSWORD} | ||||||
|  |       PGDATA: /var/lib/postgresql/ | ||||||
|  |     volumes: | ||||||
|  |       - ./db/pgsql:/var/lib/postgresql | ||||||
|  |     networks: | ||||||
|  |       - zbiorki_app_network | ||||||
|  |     restart: unless-stopped | ||||||
|  |     profiles: ["pgsql"] | ||||||
|  |  | ||||||
|  | networks: | ||||||
|  |   zbiorki_app_network: | ||||||
|  |     driver: bridge | ||||||
| @@ -3,4 +3,7 @@ Flask-SQLAlchemy | |||||||
| Flask-Login | Flask-Login | ||||||
| Werkzeug | Werkzeug | ||||||
| waitress | waitress | ||||||
| markdown | markdown | ||||||
|  | psycopg2-binary # pgsql | ||||||
|  | pymysql         # mysql | ||||||
|  | cryptography    # mysql8 | ||||||
| @@ -387,4 +387,54 @@ select.form-select:focus { | |||||||
|     border: 0; |     border: 0; | ||||||
|     border-top: 1px dashed rgba(255, 255, 255, 0.2); |     border-top: 1px dashed rgba(255, 255, 255, 0.2); | ||||||
|     margin: 10px 0; |     margin: 10px 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Tylko ten przycisk */ | ||||||
|  | .btn.btn-outline-light.btn-opis { | ||||||
|  |     color: #fff; | ||||||
|  |     background-color: transparent; | ||||||
|  |     border: 1px solid var(--border); | ||||||
|  |     transition: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn.btn-outline-light.btn-opis:hover, | ||||||
|  | .btn.btn-outline-light.btn-opis:focus { | ||||||
|  |     color: #fff; | ||||||
|  |     background-color: #161616; | ||||||
|  |     border-color: color-mix(in srgb, var(--accent) 20%, var(--border)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn.btn-outline-light { | ||||||
|  |     color: #fff; | ||||||
|  |     background-color: transparent; | ||||||
|  |     border: 1px solid rgba(255, 255, 255, 0.9); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn.btn-outline-light:hover, | ||||||
|  | .btn.btn-outline-light:focus { | ||||||
|  |     color: #fff; | ||||||
|  |     background-color: #161616; | ||||||
|  |     border-color: color-mix(in srgb, var(--accent) 20%, #ffffff); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn.btn-outline-light:active { | ||||||
|  |     color: #fff; | ||||||
|  |     background-color: #141414; | ||||||
|  |     border-color: color-mix(in srgb, var(--accent) 24%, #ffffff); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #kanalyWarning, | ||||||
|  | #postepyWarning { | ||||||
|  |     border: 1px solid #ffc107; | ||||||
|  |     background-color: #2c2c2c; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | input:disabled, | ||||||
|  | textarea:disabled, | ||||||
|  | select:disabled { | ||||||
|  |     background-color: #2b2b2b !important; | ||||||
|  |     color: #bbb !important; | ||||||
|  |     opacity: 1 !important; | ||||||
|  |     cursor: not-allowed; | ||||||
| } | } | ||||||
							
								
								
									
										74
									
								
								static/js/przelaczniki_zabezpieczenie.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								static/js/przelaczniki_zabezpieczenie.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | |||||||
|  | // static/js/przelaczniki_zabezpieczenie.js | ||||||
|  | (function () { | ||||||
|  |   'use strict'; | ||||||
|  |  | ||||||
|  |   function onReady(cb) { | ||||||
|  |     if (document.readyState === 'loading') { | ||||||
|  |       document.addEventListener('DOMContentLoaded', cb); | ||||||
|  |     } else { | ||||||
|  |       cb(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   onReady(function () { | ||||||
|  |     var boxes = Array.prototype.slice.call( | ||||||
|  |       document.querySelectorAll('input.form-check-input[type="checkbox"][data-group="postepy"]') | ||||||
|  |     ); | ||||||
|  |     var warning = document.getElementById('postepyWarning'); | ||||||
|  |  | ||||||
|  |     if (!boxes.length || !warning) { | ||||||
|  |       // Nic do zrobienia, brak elementów | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function atLeastOneChecked() { | ||||||
|  |       return boxes.some(function (b) { return !!b.checked; }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function showWarning(show) { | ||||||
|  |       warning.classList.toggle('d-none', !show); | ||||||
|  |       if (show) { | ||||||
|  |         // dyskretny highlight | ||||||
|  |         warning.classList.add('border', 'border-warning'); | ||||||
|  |         warning.style.transition = 'box-shadow 0.2s ease'; | ||||||
|  |         warning.style.boxShadow = '0 0 0.25rem rgba(255,193,7,.6)'; | ||||||
|  |         setTimeout(function () { | ||||||
|  |           warning.style.boxShadow = ''; | ||||||
|  |         }, 300); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function enforceAtLeastOne(e) { | ||||||
|  |       // Jeżeli po zmianie byłaby 0/3, przywróć zaznaczenie klikniętego i pokaż ostrzeżenie | ||||||
|  |       if (!atLeastOneChecked()) { | ||||||
|  |         e.target.checked = true; | ||||||
|  |         showWarning(true); | ||||||
|  |         e.target.classList.add('is-invalid'); | ||||||
|  |         setTimeout(function () { e.target.classList.remove('is-invalid'); }, 400); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       // Jeśli >=1, ostrzeżenie ukryj | ||||||
|  |       showWarning(false); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Podpinka zdarzeń | ||||||
|  |     boxes.forEach(function (box) { | ||||||
|  |       box.addEventListener('change', enforceAtLeastOne); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     // Walidacja przy submit (na wszelki wypadek) | ||||||
|  |     var form = boxes[0].closest('form'); | ||||||
|  |     if (form) { | ||||||
|  |       form.addEventListener('submit', function (e) { | ||||||
|  |         if (!atLeastOneChecked()) { | ||||||
|  |           e.preventDefault(); | ||||||
|  |           showWarning(true); | ||||||
|  |           boxes[0].focus(); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Inicjalny stan (np. po rerenderze z błędem) | ||||||
|  |     showWarning(!atLeastOneChecked()); | ||||||
|  |   }); | ||||||
|  | })(); | ||||||
							
								
								
									
										88
									
								
								static/js/sposoby_wplat.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								static/js/sposoby_wplat.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | |||||||
|  | (function () { | ||||||
|  |     const form = document.getElementById('form-edit-zbiorka') || document.getElementById('form-add-zbiorka') || document.querySelector('form'); | ||||||
|  |  | ||||||
|  |     const map = [ | ||||||
|  |         ['uzyj_konta', 'numer_konta'], | ||||||
|  |         ['uzyj_blik', 'numer_telefonu_blik'] | ||||||
|  |     ]; | ||||||
|  |  | ||||||
|  |     const warnBox = document.getElementById('kanalyWarning'); | ||||||
|  |  | ||||||
|  |     function showWarn(show) { | ||||||
|  |         if (!warnBox) return; | ||||||
|  |         warnBox.classList.toggle('d-none', !show); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function getEl(id) { return document.getElementById(id); } | ||||||
|  |  | ||||||
|  |     function toggleField(chkId, inputId) { | ||||||
|  |         const chk = getEl(chkId); | ||||||
|  |         const inp = getEl(inputId); | ||||||
|  |         if (!inp || !chk) return; | ||||||
|  |         const on = chk.checked; | ||||||
|  |         inp.disabled = !on; | ||||||
|  |         if (on) inp.setAttribute('required', ''); | ||||||
|  |         else inp.removeAttribute('required'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function atLeastOneOn() { | ||||||
|  |         return map.some(([c]) => getEl(c)?.checked); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function blinkInvalid(el) { | ||||||
|  |         if (!el) return; | ||||||
|  |         el.classList.add('is-invalid'); | ||||||
|  |         setTimeout(() => el.classList.remove('is-invalid'), 400); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function preventUncheckLast(e) { | ||||||
|  |         const target = e.target; | ||||||
|  |         if (target.checked) return; | ||||||
|  |         const after = map.map(([c]) => c === target.id ? false : !!getEl(c)?.checked); | ||||||
|  |         if (!after.some(Boolean)) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             target.checked = true; | ||||||
|  |             showWarn(true); | ||||||
|  |             blinkInvalid(target); | ||||||
|  |         } else { | ||||||
|  |             showWarn(false); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function onToggle(chkId, inputId) { | ||||||
|  |         toggleField(chkId, inputId); | ||||||
|  |         showWarn(!atLeastOneOn()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     map.forEach(([chkId, inputId]) => { | ||||||
|  |         const chk = getEl(chkId); | ||||||
|  |         if (!chk) return; | ||||||
|  |         chk.addEventListener('click', preventUncheckLast); | ||||||
|  |         chk.addEventListener('change', () => onToggle(chkId, inputId)); | ||||||
|  |         toggleField(chkId, inputId); | ||||||
|  |     }); | ||||||
|  |     showWarn(!atLeastOneOn()); | ||||||
|  |  | ||||||
|  |     if (form) { | ||||||
|  |         form.addEventListener('submit', function (e) { | ||||||
|  |             if (!atLeastOneOn()) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 showWarn(true); | ||||||
|  |                 blinkInvalid(getEl('uzyj_konta') || getEl('uzyj_blik')); | ||||||
|  |                 (getEl('uzyj_konta') || getEl('uzyj_blik'))?.focus(); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             for (const [chkId, inputId] of map) { | ||||||
|  |                 const chk = getEl(chkId), inp = getEl(inputId); | ||||||
|  |                 if (chk?.checked && inp && !inp.value.trim()) { | ||||||
|  |                     e.preventDefault(); | ||||||
|  |                     showWarn(true); | ||||||
|  |                     blinkInvalid(inp); | ||||||
|  |                     inp.focus(); | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | })(); | ||||||
| @@ -23,4 +23,5 @@ document.addEventListener('DOMContentLoaded', function () { | |||||||
|             modalX.show(); |             modalX.show(); | ||||||
|         }); |         }); | ||||||
|     }); |     }); | ||||||
| }); |  | ||||||
|  | }); | ||||||
|   | |||||||
| @@ -1,19 +1,20 @@ | |||||||
| (function () { | // static/js/ustawienia.js | ||||||
|     // IBAN: tylko cyfry, auto-grupowanie co 4 (po prefiksie PL) | document.addEventListener('DOMContentLoaded', () => { | ||||||
|  |     // Formatowanie IBAN (PL) | ||||||
|     const iban = document.getElementById('numer_konta'); |     const iban = document.getElementById('numer_konta'); | ||||||
|     if (iban) { |     if (iban) { | ||||||
|         iban.addEventListener('input', () => { |         iban.addEventListener('input', () => { | ||||||
|             const digits = iban.value.replace(/\\D/g, '').slice(0, 26); // 26 cyfr po "PL" |             const digits = iban.value.replace(/\D/g, '').slice(0, 26); | ||||||
|             const chunked = digits.replace(/(.{4})/g, '$1 ').trim(); |             const chunked = digits.replace(/(.{4})/g, '$1 ').trim(); | ||||||
|             iban.value = chunked; |             iban.value = chunked; | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Telefon BLIK: tylko cyfry, format 3-3-3 |     // Telefon BLIK 3-3-3 | ||||||
|     const tel = document.getElementById('numer_telefonu_blik'); |     const tel = document.getElementById('numer_telefonu_blik'); | ||||||
|     if (tel) { |     if (tel) { | ||||||
|         tel.addEventListener('input', () => { |         tel.addEventListener('input', () => { | ||||||
|             const digits = tel.value.replace(/\\D/g, '').slice(0, 9); |             const digits = tel.value.replace(/\D/g, '').slice(0, 9); | ||||||
|             const parts = []; |             const parts = []; | ||||||
|             if (digits.length > 0) parts.push(digits.substring(0, 3)); |             if (digits.length > 0) parts.push(digits.substring(0, 3)); | ||||||
|             if (digits.length > 3) parts.push(digits.substring(3, 6)); |             if (digits.length > 3) parts.push(digits.substring(3, 6)); | ||||||
| @@ -22,24 +23,23 @@ | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Biała lista IP/hostów — helpery |     // Biała lista IP/hostów | ||||||
|     const ta = document.getElementById('allowed_login_hosts'); |     const ta = document.getElementById('dozwolone_hosty_logowania'); | ||||||
|     const count = document.getElementById('hostsCount'); |     const count = document.getElementById('hostsCount'); | ||||||
|     const addBtn = document.getElementById('btn-add-host'); |     const addBtn = document.getElementById('btn-add-host'); | ||||||
|     const addMyBtn = document.getElementById('btn-add-my-ip'); |     const addMyBtn = document.getElementById('btn-add-my-ip'); | ||||||
|     const input = document.getElementById('host_input'); |     const input = document.getElementById('host_input'); | ||||||
|  |     const dedupeBtn = document.getElementById('btn-dedupe'); | ||||||
|  |  | ||||||
|     function parseList(text) { |     const parseList = (text) => | ||||||
|         // akceptuj przecinki, średniki i nowe linie; trimuj; usuń puste |         text | ||||||
|         return text |             .split(/[\r\n,;]+/)   // \r?\n, przecinek, średnik | ||||||
|             .split(/[\\n,;]+/) |  | ||||||
|             .map(s => s.trim()) |             .map(s => s.trim()) | ||||||
|             .filter(Boolean); |             .filter(Boolean); | ||||||
|     } |  | ||||||
|     function formatList(arr) { |     const formatList = (arr) => arr.join('\n'); | ||||||
|         return arr.join('\\n'); |  | ||||||
|     } |     const dedupe = (arr) => { | ||||||
|     function dedupe(arr) { |  | ||||||
|         const seen = new Set(); |         const seen = new Set(); | ||||||
|         const out = []; |         const out = []; | ||||||
|         for (const v of arr) { |         for (const v of arr) { | ||||||
| @@ -47,22 +47,23 @@ | |||||||
|             if (!seen.has(k)) { seen.add(k); out.push(v); } |             if (!seen.has(k)) { seen.add(k); out.push(v); } | ||||||
|         } |         } | ||||||
|         return out; |         return out; | ||||||
|     } |     }; | ||||||
|     function updateCount() { |  | ||||||
|  |     const updateCount = () => { | ||||||
|         if (!ta || !count) return; |         if (!ta || !count) return; | ||||||
|         count.textContent = parseList(ta.value).length.toString(); |         count.textContent = String(parseList(ta.value).length); | ||||||
|     } |     }; | ||||||
|     function addEntry(val) { |  | ||||||
|  |     const addEntry = (val) => { | ||||||
|         if (!ta || !val) return; |         if (!ta || !val) return; | ||||||
|         const list = dedupe([...parseList(ta.value), val]); |         const list = dedupe([...parseList(ta.value), val]); | ||||||
|         ta.value = formatList(list); |         ta.value = formatList(list); | ||||||
|         updateCount(); |         updateCount(); | ||||||
|     } |     }; | ||||||
|  |  | ||||||
|     if (ta) { |     if (ta) { | ||||||
|         ta.addEventListener('input', updateCount); |         ta.addEventListener('input', updateCount); | ||||||
|         // inicjalny przelicznik |         updateCount(); // inicjalne przeliczenie | ||||||
|         updateCount(); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (addBtn && input) { |     if (addBtn && input) { | ||||||
| @@ -82,11 +83,10 @@ | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const dedupeBtn = document.getElementById('btn-dedupe'); |  | ||||||
|     if (dedupeBtn && ta) { |     if (dedupeBtn && ta) { | ||||||
|         dedupeBtn.addEventListener('click', () => { |         dedupeBtn.addEventListener('click', () => { | ||||||
|             ta.value = formatList(dedupe(parseList(ta.value))); |             ta.value = formatList(dedupe(parseList(ta.value))); | ||||||
|             updateCount(); |             updateCount(); | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
| })(); | }); | ||||||
|   | |||||||
| @@ -9,11 +9,11 @@ | |||||||
|     }, false); |     }, false); | ||||||
| })(); | })(); | ||||||
|  |  | ||||||
| const pw = document.getElementById('password'); | const pw = document.getElementById("haslo"); | ||||||
| const toggle = document.getElementById('togglePw'); | const toggle = document.getElementById('togglePw'); | ||||||
| toggle.addEventListener('click', () => { | toggle.addEventListener('click', () => { | ||||||
|     const isText = pw.type === 'text'; |     const isText = pw.type === 'text'; | ||||||
|     pw.type = isText ? 'password' : 'text'; |     pw.type = isText ? "haslo" : 'text'; | ||||||
|     toggle.textContent = isText ? 'Pokaż' : 'Ukryj'; |     toggle.textContent = isText ? 'Pokaż' : 'Ukryj'; | ||||||
|     toggle.setAttribute('aria-pressed', (!isText).toString()); |     toggle.setAttribute('aria-pressed', (!isText).toString()); | ||||||
|     pw.focus(); |     pw.focus(); | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
|             e.preventDefault(); |             e.preventDefault(); | ||||||
|             e.stopPropagation(); |             e.stopPropagation(); | ||||||
|         } |         } | ||||||
|         const pw1 = document.getElementById('password'); |         const pw1 = document.getElementById("haslo"); | ||||||
|         const pw2 = document.getElementById('password2'); |         const pw2 = document.getElementById('password2'); | ||||||
|         if (pw1.value !== pw2.value) { |         if (pw1.value !== pw2.value) { | ||||||
|             e.preventDefault(); |             e.preventDefault(); | ||||||
| @@ -19,11 +19,11 @@ | |||||||
|     }, false); |     }, false); | ||||||
| })(); | })(); | ||||||
|  |  | ||||||
| const pw = document.getElementById('password'); | const pw = document.getElementById("haslo"); | ||||||
| const toggle = document.getElementById('togglePw'); | const toggle = document.getElementById('togglePw'); | ||||||
| toggle.addEventListener('click', () => { | toggle.addEventListener('click', () => { | ||||||
|     const isText = pw.type === 'text'; |     const isText = pw.type === 'text'; | ||||||
|     pw.type = isText ? 'password' : 'text'; |     pw.type = isText ? "haslo" : 'text'; | ||||||
|     toggle.textContent = isText ? 'Pokaż' : 'Ukryj'; |     toggle.textContent = isText ? 'Pokaż' : 'Ukryj'; | ||||||
|     pw.focus(); |     pw.focus(); | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -11,8 +11,8 @@ | |||||||
|             <a href="{{ url_for('formularz_zbiorek') }}" class="btn btn-primary"> |             <a href="{{ url_for('formularz_zbiorek') }}" class="btn btn-primary"> | ||||||
|                 ➕ Dodaj zbiórkę |                 ➕ Dodaj zbiórkę | ||||||
|             </a> |             </a> | ||||||
|             <a href="{{ url_for('admin_ustawienia') }}" class="btn btn-outline-light border"> |             <a href="{{ url_for('admin_ustawienia') }}" class="btn btn-outline-light"> | ||||||
|                 ⚙️ Ustawienia |                 Ustawienia główne | ||||||
|             </a> |             </a> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
| @@ -71,8 +71,7 @@ | |||||||
|                                 <span class="badge bg-secondary border" |                                 <span class="badge bg-secondary border" | ||||||
|                                     style="border-color: var(--border);">Ukryta</span> |                                     style="border-color: var(--border);">Ukryta</span> | ||||||
|                                 {% else %} |                                 {% else %} | ||||||
|                                 <span class="badge rounded-pill" |                                 <span class="badge bg-success">Widoczna</span> | ||||||
|                                     style="background: var(--accent); color:#111;">Widoczna</span> |  | ||||||
|                                 {% endif %} |                                 {% endif %} | ||||||
|                             </td> |                             </td> | ||||||
|                             <td class="text-end"> |                             <td class="text-end"> | ||||||
| @@ -261,7 +260,7 @@ | |||||||
|                 <div class="card-body text-center py-5"> |                 <div class="card-body text-center py-5"> | ||||||
|                     <h5 class="mb-2">Brak zbiórek zrealizowanych</h5> |                     <h5 class="mb-2">Brak zbiórek zrealizowanych</h5> | ||||||
|                     <p class="text-muted mb-3">Gdy jakaś zbiórka osiągnie 100%, pojawi się tutaj.</p> |                     <p class="text-muted mb-3">Gdy jakaś zbiórka osiągnie 100%, pojawi się tutaj.</p> | ||||||
|                     <a href="{{ url_for('formularz_zbiorek') }}" class="btn btn-outline-light border">Utwórz nową |                     <a href="{{ url_for('formularz_zbiorek') }}" class="btn btn-outline-light">Utwórz nową | ||||||
|                         zbiórkę</a> |                         zbiórkę</a> | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
| <div class="container my-4"> | <div class="container my-4"> | ||||||
|  |  | ||||||
|   <div class="d-flex align-items-center gap-2 mb-3"> |   <div class="d-flex align-items-center gap-2 mb-3"> | ||||||
|     <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light border">← Powrót do |     <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light">← Powrót do | ||||||
|       zbiórki</a> |       zbiórki</a> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
| @@ -60,14 +60,14 @@ | |||||||
|         <div id="kwotaHelp" class="form-text">Podaj kwotę w złotówkach (min. 0,01).</div> |         <div id="kwotaHelp" class="form-text">Podaj kwotę w złotówkach (min. 0,01).</div> | ||||||
|  |  | ||||||
|         <div class="d-flex flex-wrap gap-2 mt-2"> |         <div class="d-flex flex-wrap gap-2 mt-2"> | ||||||
|           {% for preset in [10,25,50,100,200] %} |           {% for preset in [5,10,20,25,30,50,60,100,150,200] %} | ||||||
|           <button type="button" class="btn btn-sm btn-outline-light border btn-kwota" data-amount="{{ preset }}"> |           <button type="button" class="btn btn-sm btn-outline-light btn-kwota" data-amount="{{ preset }}"> | ||||||
|             {{ preset }} PLN |             {{ preset }} PLN | ||||||
|           </button> |           </button> | ||||||
|           {% endfor %} |           {% endfor %} | ||||||
|           {% if zbiorka.cel and zbiorka.cel > 0 %} |           {% if zbiorka.cel and zbiorka.cel > 0 %} | ||||||
|           {% set brakujace = (zbiorka.cel - zbiorka.stan) if (zbiorka.cel - zbiorka.stan) > 0 else 0 %} |           {% set brakujace = (zbiorka.cel - zbiorka.stan) if (zbiorka.cel - zbiorka.stan) > 0 else 0 %} | ||||||
|           <button type="button" class="btn btn-sm btn-outline-light border btn-kwota" |           <button type="button" class="btn btn-sm btn-outline-light btn-kwota" | ||||||
|             data-amount="{{ brakujace|round(2) }}"> |             data-amount="{{ brakujace|round(2) }}"> | ||||||
|             Do celu: {{ brakujace|round(2) }} PLN |             Do celu: {{ brakujace|round(2) }} PLN | ||||||
|           </button> |           </button> | ||||||
| @@ -87,7 +87,7 @@ | |||||||
|  |  | ||||||
|       <div class="d-flex flex-wrap gap-2"> |       <div class="d-flex flex-wrap gap-2"> | ||||||
|         <button type="submit" class="btn btn-success">Dodaj wpłatę</button> |         <button type="submit" class="btn btn-success">Dodaj wpłatę</button> | ||||||
|         <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-outline-light border">Anuluj</a> |         <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-outline-light">Anuluj</a> | ||||||
|       </div> |       </div> | ||||||
|     </form> |     </form> | ||||||
|   </div> |   </div> | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
| <div class="container my-4"> | <div class="container my-4"> | ||||||
|  |  | ||||||
|     <div class="d-flex align-items-center gap-2 mb-3"> |     <div class="d-flex align-items-center gap-2 mb-3"> | ||||||
|         <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light border">← Powrót |         <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light">← Powrót | ||||||
|             do zbiórki</a> |             do zbiórki</a> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
| @@ -63,9 +63,9 @@ | |||||||
|                 </div> |                 </div> | ||||||
|  |  | ||||||
|                 <div class="d-flex flex-wrap gap-2"> |                 <div class="d-flex flex-wrap gap-2"> | ||||||
|                     <button type="submit" class="btn btn-danger">Dodaj wydatek</button> |                     <button type="submit" class="btn btn-success">Dodaj wydatek</button> | ||||||
|                     <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" |                     <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" | ||||||
|                         class="btn btn-outline-light border">Anuluj</a> |                         class="btn btn-outline-light">Anuluj</a> | ||||||
|                 </div> |                 </div> | ||||||
|             </form> |             </form> | ||||||
|         </div> |         </div> | ||||||
|   | |||||||
| @@ -6,9 +6,9 @@ | |||||||
|  |  | ||||||
|   <!-- Nawigacja --> |   <!-- Nawigacja --> | ||||||
|   <div class="d-flex align-items-center gap-2 mb-3"> |   <div class="d-flex align-items-center gap-2 mb-3"> | ||||||
|     <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light border">← Szczegóły |     <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light">← Szczegóły | ||||||
|       zbiórki</a> |       zbiórki</a> | ||||||
|     <a href="{{ url_for('admin_dashboard') }}" class="btn btn-sm btn-outline-light border">← Panel Admina</a> |     <a href="{{ url_for('admin_dashboard') }}" class="btn btn-sm btn-outline-light">← Panel Admina</a> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|   {# Obliczenia wstępne (do inicjalnego podglądu) #} |   {# Obliczenia wstępne (do inicjalnego podglądu) #} | ||||||
| @@ -72,18 +72,18 @@ | |||||||
|           <!-- Szybkie korekty --> |           <!-- Szybkie korekty --> | ||||||
|           <div class="d-flex flex-wrap gap-2 mt-2"> |           <div class="d-flex flex-wrap gap-2 mt-2"> | ||||||
|             {% for delta in [10,50,100,200] %} |             {% for delta in [10,50,100,200] %} | ||||||
|             <button type="button" class="btn btn-sm btn-outline-light border btn-delta" data-delta="{{ delta }}">+{{ |             <button type="button" class="btn btn-sm btn-outline-light btn-delta" data-delta="{{ delta }}">+{{ | ||||||
|               delta }} PLN</button> |               delta }} PLN</button> | ||||||
|             <button type="button" class="btn btn-sm btn-outline-light border btn-delta" data-delta="-{{ delta }}">-{{ |             <button type="button" class="btn btn-sm btn-outline-light btn-delta" data-delta="-{{ delta }}">-{{ | ||||||
|               delta }} PLN</button> |               delta }} PLN</button> | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|             {% if has_cel %} |             {% if has_cel %} | ||||||
|             <button type="button" class="btn btn-sm btn-outline-light border btn-set" |             <button type="button" class="btn btn-sm btn-outline-light btn-set" | ||||||
|               data-value="{{ zbiorka.cel|round(2) }}">Ustaw: do celu</button> |               data-value="{{ zbiorka.cel|round(2) }}">Ustaw: do celu</button> | ||||||
|             {% set brakujace = (zbiorka.cel - zbiorka.stan) if (zbiorka.cel - zbiorka.stan) > 0 else 0 %} |             {% set brakujace = (zbiorka.cel - zbiorka.stan) if (zbiorka.cel - zbiorka.stan) > 0 else 0 %} | ||||||
|  |  | ||||||
|             {% endif %} |             {% endif %} | ||||||
|             <button type="button" class="btn btn-sm btn-outline-light border btn-set" data-value="0">Ustaw: 0</button> |             <button type="button" class="btn btn-sm btn-outline-light btn-set" data-value="0">Ustaw: 0</button> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
| @@ -125,7 +125,7 @@ | |||||||
|         <!-- CTA --> |         <!-- CTA --> | ||||||
|         <div class="d-flex flex-wrap gap-2"> |         <div class="d-flex flex-wrap gap-2"> | ||||||
|           <button type="submit" class="btn btn-success">Aktualizuj stan</button> |           <button type="submit" class="btn btn-success">Aktualizuj stan</button> | ||||||
|           <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-outline-light border">Anuluj</a> |           <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-outline-light">Anuluj</a> | ||||||
|         </div> |         </div> | ||||||
|       </form> |       </form> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| {# templates/zbiorka_form.html #} | {# templates/formularz_zbiorek.html #} | ||||||
| {% extends 'base.html' %} | {% extends 'base.html' %} | ||||||
|  |  | ||||||
| {% set is_edit = zbiorka is not none %} | {% set has_obj = zbiorka is not none %} | ||||||
|  | {% set is_edit = has_obj and zbiorka.id is not none %} | ||||||
|  |  | ||||||
| {% block title %}{{ 'Edytuj zbiórkę' if is_edit else 'Dodaj zbiórkę' }}{% endblock %} | {% block title %}{{ 'Edytuj zbiórkę' if is_edit else 'Dodaj zbiórkę' }}{% endblock %} | ||||||
|  |  | ||||||
| @@ -15,11 +16,11 @@ | |||||||
|  |  | ||||||
|     <!-- Nawigacja / powrót --> |     <!-- Nawigacja / powrót --> | ||||||
|     <div class="d-flex align-items-center gap-2 mb-3"> |     <div class="d-flex align-items-center gap-2 mb-3"> | ||||||
|         {% if is_edit %} |         {% if is_edit and zbiorka and zbiorka.id %} | ||||||
|         <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light border">← |         <a href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light">← Szczegóły | ||||||
|             Szczegóły zbiórki</a> |             zbiórki</a> | ||||||
|         {% else %} |         {% else %} | ||||||
|         <a href="{{ url_for('admin_dashboard') }}" class="btn btn-sm btn-outline-light border">← Panel Admina</a> |         <a href="{{ url_for('admin_dashboard') }}" class="btn btn-sm btn-outline-light">← Panel Admina</a> | ||||||
|         {% endif %} |         {% endif %} | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
| @@ -40,11 +41,11 @@ | |||||||
|  |  | ||||||
|                 {% if not zbiorka.ukryj_kwote %} |                 {% if not zbiorka.ukryj_kwote %} | ||||||
|                 <span class="badge bg-dark border" style="border-color: var(--border);"> |                 <span class="badge bg-dark border" style="border-color: var(--border);"> | ||||||
|                     Stan: {{ zbiorka.stan|round(2) }} PLN |                     Stan: {{ (zbiorka.stan or 0)|round(2) }} PLN | ||||||
|                 </span> |                 </span> | ||||||
|  |  | ||||||
|                 {% if zbiorka.cel %} |                 {% if zbiorka.cel %} | ||||||
|                 {% set delta = zbiorka.cel - zbiorka.stan %} |                 {% set delta = (zbiorka.cel or 0) - (zbiorka.stan or 0) %} | ||||||
|                 {% if delta > 0 %} |                 {% if delta > 0 %} | ||||||
|                 <span class="badge bg-dark border" style="border-color: var(--border);"> |                 <span class="badge bg-dark border" style="border-color: var(--border);"> | ||||||
|                     Brakuje: {{ delta|round(2) }} PLN |                     Brakuje: {{ delta|round(2) }} PLN | ||||||
| @@ -65,7 +66,6 @@ | |||||||
|             {% else %} |             {% else %} | ||||||
|             <small class="opacity-75">Uzupełnij podstawowe dane i dane płatności</small> |             <small class="opacity-75">Uzupełnij podstawowe dane i dane płatności</small> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|  |  | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <div class="card-body"> |         <div class="card-body"> | ||||||
| @@ -80,17 +80,18 @@ | |||||||
|                             <label for="nazwa" class="form-label">Nazwa zbiórki</label> |                             <label for="nazwa" class="form-label">Nazwa zbiórki</label> | ||||||
|                             <input type="text" class="form-control" id="nazwa" name="nazwa" maxlength="120" |                             <input type="text" class="form-control" id="nazwa" name="nazwa" maxlength="120" | ||||||
|                                 placeholder="{{ 'Krótki, zrozumiały tytuł' if is_edit else 'Np. Wsparcie dla schroniska Azor' }}" |                                 placeholder="{{ 'Krótki, zrozumiały tytuł' if is_edit else 'Np. Wsparcie dla schroniska Azor' }}" | ||||||
|                                 value="{{ zbiorka.nazwa if is_edit else '' }}" required aria-describedby="nazwaHelp"> |                                 value="{{ (zbiorka.nazwa if zbiorka else request.form.get('nazwa','')) }}" required | ||||||
|  |                                 aria-describedby="nazwaHelp"> | ||||||
|                             <div id="nazwaHelp" class="form-text">Krótki, zrozumiały tytuł. Max 120 znaków.</div> |                             <div id="nazwaHelp" class="form-text">Krótki, zrozumiały tytuł. Max 120 znaków.</div> | ||||||
|                         </div> |                         </div> | ||||||
|  |  | ||||||
|                         <div class="col-12"> |                         <div class="col-12"> | ||||||
|                             <label for="opis" class="form-label">Opis (Markdown)</label> |                             <label for="opis" class="form-label">Opis (Markdown)</label> | ||||||
|                             <textarea class="form-control" id="opis" name="opis" rows="8" required |                             <textarea class="form-control" id="opis" name="opis" rows="8" required | ||||||
|                                 aria-describedby="opisHelp">{{ zbiorka.opis if is_edit else '' }}</textarea> |                                 aria-describedby="opisHelp">{{ (zbiorka.opis if zbiorka else request.form.get('opis','')) }}</textarea> | ||||||
|                             <div class="d-flex justify-content-between"> |                             <div class="d-flex justify-content-between"> | ||||||
|                                 <small id="opisHelp" class="form-text text-muted"> |                                 <small id="opisHelp" class="form-text text-muted"> | ||||||
|                                     Możesz używać **Markdown** (nagłówki, listy, linki). W edytorze włącz podgląd 👁️. |                                     Możesz używać Markdown (nagłówki, listy, linki). W edytorze włącz podgląd 👁️. | ||||||
|                                 </small> |                                 </small> | ||||||
|                                 <small class="text-muted"><span id="opisCount">0</span> znaków</small> |                                 <small class="text-muted"><span id="opisCount">0</span> znaków</small> | ||||||
|                             </div> |                             </div> | ||||||
| @@ -120,8 +121,7 @@ | |||||||
|                                 </tr> |                                 </tr> | ||||||
|                             </thead> |                             </thead> | ||||||
|                             <tbody id="produkty-body"> |                             <tbody id="produkty-body"> | ||||||
|                                 {% set items = zbiorka.przedmioty if is_edit and zbiorka and zbiorka.przedmioty else [] |                                 {% set items = (zbiorka.przedmioty if zbiorka and zbiorka.przedmioty else []) %} | ||||||
|                                 %} |  | ||||||
|                                 {% if items %} |                                 {% if items %} | ||||||
|                                 {% for it in items %} |                                 {% for it in items %} | ||||||
|                                 {% set i = loop.index0 %} |                                 {% set i = loop.index0 %} | ||||||
| @@ -151,7 +151,7 @@ | |||||||
|                                         </div> |                                         </div> | ||||||
|                                     </td> |                                     </td> | ||||||
|                                     <td class="text-end"> |                                     <td class="text-end"> | ||||||
|                                         <button type="button" class="btn btn-sm btn-outline-light border remove-row" |                                         <button type="button" class="btn btn-sm btn-outline-light remove-row" | ||||||
|                                             title="Usuń wiersz">✕</button> |                                             title="Usuń wiersz">✕</button> | ||||||
|                                     </td> |                                     </td> | ||||||
|                                 </tr> |                                 </tr> | ||||||
| @@ -173,7 +173,7 @@ | |||||||
|                                         </div> |                                         </div> | ||||||
|                                     </td> |                                     </td> | ||||||
|                                     <td class="text-end"> |                                     <td class="text-end"> | ||||||
|                                         <button type="button" class="btn btn-sm btn-outline-light border remove-row" |                                         <button type="button" class="btn btn-sm btn-outline-light remove-row" | ||||||
|                                             title="Usuń wiersz">✕</button> |                                             title="Usuń wiersz">✕</button> | ||||||
|                                     </td> |                                     </td> | ||||||
|                                 </tr> |                                 </tr> | ||||||
| @@ -183,9 +183,8 @@ | |||||||
|                     </div> |                     </div> | ||||||
|  |  | ||||||
|                     <div class="d-flex gap-2"> |                     <div class="d-flex gap-2"> | ||||||
|                         <button type="button" class="btn btn-sm btn-outline-light border" id="add-row">+ Dodaj |                         <button type="button" class="btn btn-sm btn-outline-light" id="add-row">+ Dodaj pozycję</button> | ||||||
|                             pozycję</button> |                         <button type="button" class="btn btn-sm btn-outline-light" id="clear-empty">Usuń puste | ||||||
|                         <button type="button" class="btn btn-sm btn-outline-light border" id="clear-empty">Usuń puste |  | ||||||
|                             wiersze</button> |                             wiersze</button> | ||||||
|                     </div> |                     </div> | ||||||
|                 </div> |                 </div> | ||||||
| @@ -195,36 +194,65 @@ | |||||||
|                 <!-- SEKCJA: Dane płatności --> |                 <!-- SEKCJA: Dane płatności --> | ||||||
|                 <div class="mb-4"> |                 <div class="mb-4"> | ||||||
|                     <h6 class="text-muted mb-2">Dane płatności</h6> |                     <h6 class="text-muted mb-2">Dane płatności</h6> | ||||||
|  |  | ||||||
|                     <div class="row g-3"> |                     <div class="row g-3"> | ||||||
|                         <div class="col-12"> |                         <!-- Przełączniki kanałów --> | ||||||
|  |                         <div class="col-12 col-md-6"> | ||||||
|  |                             <div class="form-check form-switch"> | ||||||
|  |                                 <input class="form-check-input" type="checkbox" id="uzyj_konta" name="uzyj_konta" {% if | ||||||
|  |                                     zbiorka is none or zbiorka.uzyj_konta %}checked{% endif %}> | ||||||
|  |                                 <label class="form-check-label" for="uzyj_konta">Przelew na konto</label> | ||||||
|  |                             </div> | ||||||
|  |                         </div> | ||||||
|  |  | ||||||
|  |                         <div class="col-12 col-md-6"> | ||||||
|  |                             <div class="form-check form-switch"> | ||||||
|  |                                 <input class="form-check-input" type="checkbox" id="uzyj_blik" name="uzyj_blik" {% if | ||||||
|  |                                     zbiorka is none or zbiorka.uzyj_blik %}checked{% endif %}> | ||||||
|  |                                 <label class="form-check-label" for="uzyj_blik">BLIK</label> | ||||||
|  |                             </div> | ||||||
|  |                         </div> | ||||||
|  |                         <br> | ||||||
|  |                         <div id="kanalyWarning" class="alert alert-warning d-none mt-2" role="alert"> | ||||||
|  |                             Musi być włączony co najmniej jeden kanał wpłat (konto lub BLIK). | ||||||
|  |                         </div> | ||||||
|  |  | ||||||
|  |                         <!-- IBAN --> | ||||||
|  |                         <div class="col-12" id="pole_konto"> | ||||||
|                             <label for="numer_konta" class="form-label">Numer konta (IBAN)</label> |                             <label for="numer_konta" class="form-label">Numer konta (IBAN)</label> | ||||||
|                             <div class="input-group"> |                             <div class="input-group"> | ||||||
|                                 <span class="input-group-text">PL</span> |                                 <span class="input-group-text">PL</span> | ||||||
|                                 <input type="text" class="form-control" id="numer_konta" name="numer_konta" |                                 <input type="text" class="form-control" id="numer_konta" name="numer_konta" | ||||||
|                                     inputmode="numeric" autocomplete="off" |                                     inputmode="numeric" autocomplete="off" | ||||||
|                                     placeholder="12 3456 7890 1234 5678 9012 3456" required aria-describedby="ibanHelp" |                                     placeholder="12 3456 7890 1234 5678 9012 3456" {% if zbiorka and not | ||||||
|                                     value="{% if is_edit and zbiorka.numer_konta %}{{ zbiorka.numer_konta }}{% elif global_settings %}{{ global_settings.numer_konta }}{% else %}{% endif %}"> |                                     zbiorka.uzyj_konta %}disabled{% endif %} {% if zbiorka is none or zbiorka.uzyj_konta | ||||||
|  |                                     %}required{% endif %} aria-describedby="ibanHelp" | ||||||
|  |                                     value="{% if zbiorka and zbiorka.numer_konta %}{{ zbiorka.numer_konta }}{% elif global_settings %}{{ global_settings.numer_konta }}{% else %}{{ request.form.get('numer_konta','') }}{% endif %}"> | ||||||
|                             </div> |                             </div> | ||||||
|                             <div id="ibanHelp" class="form-text">Wpisz ciąg cyfr; spacje dodadzą się automatycznie dla |                             <div id="ibanHelp" class="form-text">Wpisz ciąg cyfr; spacje dodadzą się automatycznie dla | ||||||
|                                 czytelności.</div> |                                 czytelności.</div> | ||||||
|                         </div> |                         </div> | ||||||
|  |  | ||||||
|                         <div class="col-12 col-md-6"> |                         <!-- BLIK --> | ||||||
|  |                         <div class="col-12 col-md-6" id="pole_blik"> | ||||||
|                             <label for="numer_telefonu_blik" class="form-label">Numer telefonu BLIK</label> |                             <label for="numer_telefonu_blik" class="form-label">Numer telefonu BLIK</label> | ||||||
|                             <div class="input-group"> |                             <div class="input-group"> | ||||||
|                                 <span class="input-group-text">+48</span> |                                 <span class="input-group-text">+48</span> | ||||||
|                                 <input type="tel" class="form-control" id="numer_telefonu_blik" |                                 <input type="tel" class="form-control" id="numer_telefonu_blik" | ||||||
|                                     name="numer_telefonu_blik" inputmode="tel" pattern="[0-9 ]{9,13}" |                                     name="numer_telefonu_blik" inputmode="tel" pattern="[0-9 ]{9,13}" | ||||||
|                                     placeholder="123 456 789" required aria-describedby="blikHelp" |                                     placeholder="123 456 789" {% if zbiorka and not zbiorka.uzyj_blik %}disabled{% endif | ||||||
|                                     value="{% if is_edit and zbiorka.numer_telefonu_blik %}{{ zbiorka.numer_telefonu_blik }}{% elif global_settings %}{{ global_settings.numer_telefonu_blik }}{% else %}{% endif %}"> |                                     %} {% if zbiorka is none or zbiorka.uzyj_blik %}required{% endif %} | ||||||
|  |                                     aria-describedby="blikHelp" | ||||||
|  |                                     value="{% if zbiorka and zbiorka.numer_telefonu_blik %}{{ zbiorka.numer_telefonu_blik }}{% elif global_settings %}{{ global_settings.numer_telefonu_blik }}{% else %}{{ request.form.get('numer_telefonu_blik','') }}{% endif %}"> | ||||||
|                             </div> |                             </div> | ||||||
|                             <div id="blikHelp" class="form-text">Dziewięć cyfr telefonu powiązanego z BLIK. Spacje |                             <div id="blikHelp" class="form-text">Dziewięć cyfr telefonu powiązanego z BLIK. Spacje | ||||||
|                                 opcjonalne.</div> |                                 opcjonalne.</div> | ||||||
|  |  | ||||||
|                         </div> |                         </div> | ||||||
|  |  | ||||||
|                         {% if is_edit %} |                         {% if is_edit %} | ||||||
|                         <div class="col-12 col-md-12 d-flex align-items-end"> |                         <div class="col-12 col-md-12 d-flex align-items-end"> | ||||||
|                             <button type="button" class="btn btn-sm btn-outline-light border" id="ustaw-globalne" |                             <button type="button" class="btn btn-sm btn-outline-light" id="ustaw-globalne" | ||||||
|                                 title="Wstaw wartości z ustawień globalnych" {% if global_settings %} |                                 title="Wstaw wartości z ustawień globalnych" {% if global_settings %} | ||||||
|                                 data-iban="{{ global_settings.numer_konta }}" |                                 data-iban="{{ global_settings.numer_konta }}" | ||||||
|                                 data-blik="{{ global_settings.numer_telefonu_blik }}" {% endif %}> |                                 data-blik="{{ global_settings.numer_telefonu_blik }}" {% endif %}> | ||||||
| @@ -245,7 +273,7 @@ | |||||||
|                         <div class="d-flex flex-wrap align-items-center justify-content-between gap-2"> |                         <div class="d-flex flex-wrap align-items-center justify-content-between gap-2"> | ||||||
|                             <div id="celSyncMsg" class="small"></div> |                             <div id="celSyncMsg" class="small"></div> | ||||||
|                             <button type="button" id="btnApplyCelFromSum" |                             <button type="button" id="btnApplyCelFromSum" | ||||||
|                                 class="btn btn-sm btn-outline-light border d-none"></button> |                                 class="btn btn-sm btn-outline-light d-none"></button> | ||||||
|                         </div> |                         </div> | ||||||
|                     </div> |                     </div> | ||||||
|  |  | ||||||
| @@ -254,21 +282,55 @@ | |||||||
|                             <label for="cel" class="form-label">Cel zbiórki</label> |                             <label for="cel" class="form-label">Cel zbiórki</label> | ||||||
|                             <div class="input-group"> |                             <div class="input-group"> | ||||||
|                                 <span class="input-group-text">PLN</span> |                                 <span class="input-group-text">PLN</span> | ||||||
|                                 <input type="number" class="form-control" id="cel" name="cel" step="0.01" min="0.01" |                                 <input type="text" inputmode="decimal" class="form-control" id="cel" name="cel" | ||||||
|                                     placeholder="0,00" required aria-describedby="celHelp" |                                     placeholder="0,00" required aria-describedby="celHelp" | ||||||
|                                     value="{{ zbiorka.cel if is_edit else '' }}"> |                                     value="{% if zbiorka and zbiorka.cel is not none %}{{ zbiorka.cel }}{% else %}{{ request.form.get('cel','') }}{% endif %}"> | ||||||
|                             </div> |                             </div> | ||||||
|                             <div id="celHelp" class="form-text">Minimalnie 0,01 PLN. Możesz to później edytować.</div> |                             <div id="celHelp" class="form-text">Minimalnie 0,01 PLN. Można później edytować.</div> | ||||||
|                         </div> |                         </div> | ||||||
|  |  | ||||||
|                         <div class="col-12 col-md-12 d-flex align-items-end"> |                         <div class="col-12 col-md-12 d-flex align-items-end"> | ||||||
|                             <div class="form-check form-switch"> |                             <div class="form-check form-switch"> | ||||||
|                                 <input class="form-check-input" type="checkbox" id="ukryj_kwote" name="ukryj_kwote" {% |                                 <input class="form-check-input" type="checkbox" id="ukryj_kwote" name="ukryj_kwote" {% | ||||||
|                                     if is_edit and zbiorka.ukryj_kwote %}checked{% endif %}> |                                     if zbiorka %}{% if zbiorka.ukryj_kwote %}checked{% endif %}{% endif %}> | ||||||
|                                 <label class="form-check-label" for="ukryj_kwote">Ukryj kwoty (cel i stan)</label> |                                 <label class="form-check-label" for="ukryj_kwote">Ukryj kwoty (cel i stan)</label> | ||||||
|                             </div> |                             </div> | ||||||
|                         </div> |                         </div> | ||||||
|                     </div> |                     </div> | ||||||
|  |  | ||||||
|  |                     <div class="row g-3 mt-2"> | ||||||
|  |                         <div class="col-12 col-md-4"> | ||||||
|  |                             <div class="form-check form-switch"> | ||||||
|  |                                 <input class="form-check-input" type="checkbox" id="pokaz_postep_finanse" | ||||||
|  |                                     name="pokaz_postep_finanse" data-group="postepy" {% if zbiorka %}{% if | ||||||
|  |                                     zbiorka.pokaz_postep_finanse %}checked{% endif %}{% else %}checked{% endif %}> | ||||||
|  |                                 <label class="form-check-label" for="pokaz_postep_finanse">Pokaż postęp: Finanse</label> | ||||||
|  |                             </div> | ||||||
|  |                         </div> | ||||||
|  |  | ||||||
|  |                         <div class="col-12 col-md-4"> | ||||||
|  |                             <div class="form-check form-switch"> | ||||||
|  |                                 <input class="form-check-input" type="checkbox" id="pokaz_postep_pozycje" | ||||||
|  |                                     name="pokaz_postep_pozycje" data-group="postepy" {% if zbiorka %}{% if | ||||||
|  |                                     zbiorka.pokaz_postep_pozycje %}checked{% endif %}{% else %}checked{% endif %}> | ||||||
|  |                                 <label class="form-check-label" for="pokaz_postep_pozycje">Pokaż postęp: Zakupy | ||||||
|  |                                     (liczba)</label> | ||||||
|  |                             </div> | ||||||
|  |                         </div> | ||||||
|  |  | ||||||
|  |                         <div class="col-12 col-md-4"> | ||||||
|  |                             <div class="form-check form-switch"> | ||||||
|  |                                 <input class="form-check-input" type="checkbox" id="pokaz_postep_kwotowo" | ||||||
|  |                                     name="pokaz_postep_kwotowo" data-group="postepy" {% if zbiorka %}{% if | ||||||
|  |                                     zbiorka.pokaz_postep_kwotowo %}checked{% endif %}{% else %}checked{% endif %}> | ||||||
|  |                                 <label class="form-check-label" for="pokaz_postep_kwotowo">Pokaż postęp: Zakupy | ||||||
|  |                                     (kwotowo)</label> | ||||||
|  |                             </div> | ||||||
|  |                         </div> | ||||||
|  |                     </div><br> | ||||||
|  |                     <div id="postepyWarning" class="alert alert-warning d-none mt-2" role="alert"> | ||||||
|  |                         Nie można wyłączyć wszystkich wskaźników postępu. Pozostaw przynajmniej jeden włączony. | ||||||
|  |                     </div> | ||||||
|                 </div> |                 </div> | ||||||
|  |  | ||||||
|                 <!-- CTA --> |                 <!-- CTA --> | ||||||
| @@ -276,7 +338,7 @@ | |||||||
|                     <button type="submit" class="btn btn-success"> |                     <button type="submit" class="btn btn-success"> | ||||||
|                         {{ ' Zaktualizuj zbiórkę' if is_edit else 'Dodaj zbiórkę' }} |                         {{ ' Zaktualizuj zbiórkę' if is_edit else 'Dodaj zbiórkę' }} | ||||||
|                     </button> |                     </button> | ||||||
|                     <a href="{{ url_for('admin_dashboard') }}" class="btn btn-outline-light border">Anuluj</a> |                     <a href="{{ url_for('admin_dashboard') }}" class="btn btn-outline-light">Anuluj</a> | ||||||
|                 </div> |                 </div> | ||||||
|             </form> |             </form> | ||||||
|         </div> |         </div> | ||||||
| @@ -291,4 +353,6 @@ | |||||||
| <script src="{{ url_for('static', filename='js/formularz_zbiorek.js') }}?v={{ APP_VERSION }}"></script> | <script src="{{ url_for('static', filename='js/formularz_zbiorek.js') }}?v={{ APP_VERSION }}"></script> | ||||||
| <script src="{{ url_for('static', filename='js/produkty_formularz.js') }}?v={{ APP_VERSION }}"></script> | <script src="{{ url_for('static', filename='js/produkty_formularz.js') }}?v={{ APP_VERSION }}"></script> | ||||||
| <script src="{{ url_for('static', filename='js/kwoty_formularz.js') }}?v={{ APP_VERSION }}"></script> | <script src="{{ url_for('static', filename='js/kwoty_formularz.js') }}?v={{ APP_VERSION }}"></script> | ||||||
|  | <script src="{{ url_for('static', filename='js/przelaczniki_zabezpieczenie.js') }}?v={{ APP_VERSION }}"></script> | ||||||
|  | <script src="{{ url_for('static', filename='js/sposoby_wplat.js') }}?v={{ APP_VERSION }}"></script> | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -6,13 +6,20 @@ | |||||||
|  |  | ||||||
|     <div class="d-flex justify-content-between align-items-center mb-3"> |     <div class="d-flex justify-content-between align-items-center mb-3"> | ||||||
|         <h3 class="mb-0">Transakcje: {{ zbiorka.nazwa }}</h3> |         <h3 class="mb-0">Transakcje: {{ zbiorka.nazwa }}</h3> | ||||||
|         <div class="btn-group"> |  | ||||||
|             <a class="btn btn-sm btn-outline-light border" href="{{ url_for('dodaj_wplate', zbiorka_id=zbiorka.id) }}">+ |         <div class="btn-group" role="group" aria-label="Akcje zbiórki"> | ||||||
|                 Wpłata</a> |             <a class="btn btn-sm btn-outline-light" href="{{ url_for('dodaj_wplate', zbiorka_id=zbiorka.id) }}"> | ||||||
|             <a class="btn btn-sm btn-outline-light border" |                 <i class="fas fa-plus-circle"></i> Dodaj wpłatę | ||||||
|                 href="{{ url_for('dodaj_wydatek', zbiorka_id=zbiorka.id) }}">+ Wydatek</a> |             </a> | ||||||
|             <a class="btn btn-sm btn-outline-light border" |             <a class="btn btn-sm btn-outline-light" href="{{ url_for('dodaj_wydatek', zbiorka_id=zbiorka.id) }}"> | ||||||
|                 href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}">Szczegóły zbiórki</a> |                 Dodaj wydatek | ||||||
|  |             </a> | ||||||
|  |             <a class="btn btn-sm btn-outline-light" href="{{ url_for('edytuj_stan', zbiorka_id=zbiorka.id) }}"> | ||||||
|  |                 Edytuj stan | ||||||
|  |             </a> | ||||||
|  |             <a class="btn btn-sm btn-outline-light" href="{{ url_for('zbiorka', zbiorka_id=zbiorka.id) }}"> | ||||||
|  |                 Otwórz ↗ | ||||||
|  |             </a> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
| @@ -24,6 +31,7 @@ | |||||||
|                         <tr> |                         <tr> | ||||||
|                             <th>Data</th> |                             <th>Data</th> | ||||||
|                             <th>Typ</th> |                             <th>Typ</th> | ||||||
|  |                             <th>Widoczność</th> | ||||||
|                             <th class="text-end">Kwota</th> |                             <th class="text-end">Kwota</th> | ||||||
|                             <th>Opis</th> |                             <th>Opis</th> | ||||||
|                             <th class="text-end"></th> |                             <th class="text-end"></th> | ||||||
| @@ -31,39 +39,77 @@ | |||||||
|                     </thead> |                     </thead> | ||||||
|                     <tbody> |                     <tbody> | ||||||
|                         {% for a in aktywnosci %} |                         {% for a in aktywnosci %} | ||||||
|                         <tr> |                         <tr data-tx-id="{{ a.id }}" data-tx-typ="{{ a.typ }}"> | ||||||
|                             <td>{{ a.data|dt("%d.%m.%Y %H:%M") }}</td> |                             <td>{{ a.data|dt("%d.%m.%Y %H:%M") }}</td> | ||||||
|                             <td> |                             <td> | ||||||
|                                 <span class="badge {{ 'bg-success' if a.typ=='wpłata' else 'bg-danger' }}">{{ a.typ |                                 <span class="badge {{ 'bg-success' if a.typ=='wpłata' else 'bg-danger' }}">{{ a.typ | ||||||
|                                     }}</span> |                                     }}</span> | ||||||
|  |  | ||||||
|  |                             </td> | ||||||
|  |  | ||||||
|  |                             <td> | ||||||
|  |                                 {% if a.ukryta %} | ||||||
|  |                                 <span class="badge bg-warning ms-1">ukryta</span> | ||||||
|  |                                 {% else %} | ||||||
|  |                                 <span class="badge bg-success ms-1">widoczna</span> | ||||||
|  |                                 {% endif %} | ||||||
|  |  | ||||||
|                             </td> |                             </td> | ||||||
|                             <td class="text-end">{{ '%.2f'|format(a.kwota) }} PLN</td> |                             <td class="text-end">{{ '%.2f'|format(a.kwota) }} PLN</td> | ||||||
|                             <td class="text-muted">{{ a.opis or '—' }}</td> |                             <td class="text-muted">{{ a.opis or '—' }}</td> | ||||||
|                             <td class="text-end"> |                             <td class="text-end"> | ||||||
|                                 {% if a.typ == 'wpłata' %} |  | ||||||
|                                 <button class="btn btn-sm btn-outline-light border btn-edit-wplata" data-id="{{ a.id }}" |                                 <div class="d-inline-flex flex-nowrap align-items-center gap-2"> | ||||||
|                                     data-kwota="{{ '%.2f'|format(a.kwota) }}" data-opis="{{ a.opis|e if a.opis }}" |                                     {% if a.typ == 'wpłata' %} | ||||||
|                                     data-action="{{ url_for('zapisz_wplate', wplata_id=a.id) }}"> |                                     <button class="btn btn-sm btn-outline-light btn-edit-wplata" data-id="{{ a.id }}" | ||||||
|                                     Edytuj |                                         data-kwota="{{ '%.2f'|format(a.kwota) }}" data-opis="{{ a.opis|e if a.opis }}" | ||||||
|                                 </button> |                                         data-action="{{ url_for('zapisz_wplate', wplata_id=a.id) }}"> | ||||||
|                                 <form class="d-inline" method="post" |                                         Edytuj | ||||||
|                                     action="{{ url_for('usun_wplate', wplata_id=a.id) }}" |                                     </button> | ||||||
|                                     onsubmit="return confirm('Usunąć wpłatę? Cofnie to wpływ na stan.');"> |                                     <form class="d-inline" method="post" | ||||||
|                                     <button class="btn btn-sm btn-outline-danger">Usuń</button> |                                         action="{{ url_for('usun_wplate', wplata_id=a.id) }}" | ||||||
|                                 </form> |                                         onsubmit="return confirm('Usunąć wpłatę? Cofnie to wpływ na stan.');"> | ||||||
|                                 {% else %} |                                         <button class="btn btn-sm btn-outline-danger">Usuń</button> | ||||||
|                                 <button class="btn btn-sm btn-outline-light border btn-edit-wydatek" |                                     </form> | ||||||
|                                     data-id="{{ a.id }}" data-kwota="{{ '%.2f'|format(a.kwota) }}" |  | ||||||
|                                     data-opis="{{ a.opis|e if a.opis }}" |                                     {% if a.ukryta %} | ||||||
|                                     data-action="{{ url_for('zapisz_wydatek', wydatek_id=a.id) }}"> |                                     <form class="d-inline" method="post" | ||||||
|                                     Edytuj |                                         action="{{ url_for('odkryj_wplate', wplata_id=a.id) }}"> | ||||||
|                                 </button> |                                         <button class="btn btn-sm btn-outline-secondary">Odkryj</button> | ||||||
|                                 <form class="d-inline" method="post" |                                     </form> | ||||||
|                                     action="{{ url_for('usun_wydatek', wydatek_id=a.id) }}" |                                     {% else %} | ||||||
|                                     onsubmit="return confirm('Usunąć wydatek? Cofnie to wpływ na stan.');"> |                                     <form class="d-inline" method="post" | ||||||
|                                     <button class="btn btn-sm btn-outline-danger">Usuń</button> |                                         action="{{ url_for('ukryj_wplate', wplata_id=a.id) }}"> | ||||||
|                                 </form> |                                         <button class="btn btn-sm btn-outline-secondary">Ukryj</button> | ||||||
|                                 {% endif %} |                                     </form> | ||||||
|  |                                     {% endif %} | ||||||
|  |  | ||||||
|  |                                     {% else %} | ||||||
|  |                                     <button class="btn btn-sm btn-outline-light btn-edit-wydatek" data-id="{{ a.id }}" | ||||||
|  |                                         data-kwota="{{ '%.2f'|format(a.kwota) }}" data-opis="{{ a.opis|e if a.opis }}" | ||||||
|  |                                         data-action="{{ url_for('zapisz_wydatek', wydatek_id=a.id) }}"> | ||||||
|  |                                         Edytuj | ||||||
|  |                                     </button> | ||||||
|  |                                     <form class="d-inline" method="post" | ||||||
|  |                                         action="{{ url_for('usun_wydatek', wydatek_id=a.id) }}" | ||||||
|  |                                         onsubmit="return confirm('Usunąć wydatek? Cofnie to wpływ na stan.');"> | ||||||
|  |                                         <button class="btn btn-sm btn-outline-danger">Usuń</button> | ||||||
|  |                                     </form> | ||||||
|  |  | ||||||
|  |                                     {% if a.ukryta %} | ||||||
|  |                                     <form class="d-inline" method="post" | ||||||
|  |                                         action="{{ url_for('odkryj_wydatek', wydatek_id=a.id) }}"> | ||||||
|  |                                         <button class="btn btn-sm btn-outline-secondary">Odkryj</button> | ||||||
|  |                                     </form> | ||||||
|  |                                     {% else %} | ||||||
|  |                                     <form class="d-inline" method="post" | ||||||
|  |                                         action="{{ url_for('ukryj_wydatek', wydatek_id=a.id) }}"> | ||||||
|  |                                         <button class="btn btn-sm btn-outline-secondary">Ukryj</button> | ||||||
|  |                                     </form> | ||||||
|  |                                     {% endif %} | ||||||
|  |  | ||||||
|  |                                     {% endif %} | ||||||
|  |                                 </div> | ||||||
|                             </td> |                             </td> | ||||||
|                         </tr> |                         </tr> | ||||||
|                         {% else %} |                         {% else %} | ||||||
| @@ -103,7 +149,7 @@ | |||||||
|             </div> |             </div> | ||||||
|             <div class="modal-footer"> |             <div class="modal-footer"> | ||||||
|                 <button class="btn btn-success">Zapisz</button> |                 <button class="btn btn-success">Zapisz</button> | ||||||
|                 <button type="button" class="btn btn-outline-light border" data-bs-dismiss="modal">Anuluj</button> |                 <button type="button" class="btn btn-outline-light" data-bs-dismiss="modal">Anuluj</button> | ||||||
|             </div> |             </div> | ||||||
|         </form> |         </form> | ||||||
|     </div> |     </div> | ||||||
| @@ -130,7 +176,7 @@ | |||||||
|             </div> |             </div> | ||||||
|             <div class="modal-footer"> |             <div class="modal-footer"> | ||||||
|                 <button class="btn btn-success">Zapisz</button> |                 <button class="btn btn-success">Zapisz</button> | ||||||
|                 <button type="button" class="btn btn-outline-light border" data-bs-dismiss="modal">Anuluj</button> |                 <button type="button" class="btn btn-outline-light" data-bs-dismiss="modal">Anuluj</button> | ||||||
|             </div> |             </div> | ||||||
|         </form> |         </form> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -44,38 +44,55 @@ | |||||||
|     <div class="card shadow-sm mb-4"> |     <div class="card shadow-sm mb-4"> | ||||||
|       <div class="card-header bg-secondary text-white d-flex align-items-center justify-content-between gap-2"> |       <div class="card-header bg-secondary text-white d-flex align-items-center justify-content-between gap-2"> | ||||||
|         <h3 class="card-title mb-0">Dostęp — dozwolone adresy IP / hosty</h3> |         <h3 class="card-title mb-0">Dostęp — dozwolone adresy IP / hosty</h3> | ||||||
|         <small class="opacity-75">Zależnie od konfiguracji, logowanie może wymagać dopasowania do białej listy</small> |         <small class="opacity-75">Zależnie od konfiguracji logowanie może wymagać dopasowania do białej listy</small> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
|       <div class="card-body"> |       <div class="card-body"> | ||||||
|  |         <!-- Wiersz z inputem i przyciskiem dodawania --> | ||||||
|         <div class="row g-3 align-items-end"> |         <div class="row g-3 align-items-end"> | ||||||
|           <div class="col-12 col-md-6"> |           <div class="col-12 col-lg-8"> | ||||||
|             <label for="host_input" class="form-label">Dodaj pojedynczy IP/host</label> |             <label for="host_input" class="form-label">Dodaj IP/host</label> | ||||||
|             <input type="text" class="form-control" id="host_input" placeholder="np. 203.0.113.42 lub corp.example.com" |             <div class="input-group"> | ||||||
|               aria-describedby="hostAddHelp"> |               <input type="text" class="form-control" id="host_input" | ||||||
|             <div id="hostAddHelp" class="form-text">Po wpisaniu kliknij „Dodaj do listy”. Duplikaty są pomijane.</div> |                 placeholder="np. 203.0.113.42 lub corp.example.com" aria-describedby="hostAddHelp"> | ||||||
|  |               <button type="button" class="btn btn-outline-light" id="btn-add-host"> | ||||||
|  |                 ➕ Dodaj | ||||||
|  |               </button> | ||||||
|  |             </div> | ||||||
|  |             <div id="hostAddHelp" class="form-text">Po wpisaniu kliknij „Dodaj”. Duplikaty są pomijane.</div> | ||||||
|           </div> |           </div> | ||||||
|           <div class="col-12 col-md-6 d-flex gap-2"> |  | ||||||
|             <button type="button" class="btn btn-outline-light border" id="btn-add-host">Dodaj do listy</button> |           <div class="col-12 col-lg-4"> | ||||||
|             <button type="button" class="btn btn-light text-dark" id="btn-add-my-ip" data-my-ip="{{ client_ip }}">Dodaj |             <div class="d-flex flex-wrap gap-2 justify-content-lg-end"> | ||||||
|               moje IP ({{ client_ip }})</button> |               <button type="button" class="btn btn-light text-dark" id="btn-add-my-ip" data-my-ip="{{ client_ip }}"> | ||||||
|             <button type="button" class="btn btn-outline-light border" id="btn-dedupe">Usuń duplikaty</button> |                 <i class="fas fa-location-arrow"></i> ➕ Dodaj moje IP ({{ client_ip }}) | ||||||
|  |               </button> | ||||||
|  |               <button type="button" class="btn btn-outline-light" id="btn-dedupe"> | ||||||
|  |                Usuń duplikaty | ||||||
|  |               </button> | ||||||
|  |             </div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|         <div class="mt-3"> |         <div class="mt-3"> | ||||||
|           <label for="allowed_login_hosts" class="form-label">Dozwolone hosty logowania (jeden na linię lub rozdzielone |           <div class="d-flex justify-content-between align-items-center mb-1"> | ||||||
|             przecinkami)</label> |             <label for="dozwolone_hosty_logowania" class="form-label mb-0"> | ||||||
|           <textarea class="form-control" id="allowed_login_hosts" name="allowed_login_hosts" rows="6" |               Dozwolone hosty logowania (jeden na linię lub rozdzielone przecinkami) | ||||||
|             placeholder="Adresy IP lub nazwy domen — każdy w osobnej linii lub rozdzielony przecinkiem">{{ settings.allowed_login_hosts if settings and settings.allowed_login_hosts else '' }}</textarea> |             </label> | ||||||
|           <div class="d-flex justify-content-between mt-1"> |             <span class="badge text-bg-secondary">Pozycji: <span id="hostsCount">0</span></span> | ||||||
|             <small class="text-muted">Akceptowane separatory: przecinek (`,`), średnik (`;`) i nowa linia.</small> |  | ||||||
|             <small class="text-muted">Pozycji na liście: <span id="hostsCount">0</span></small> |  | ||||||
|           </div> |           </div> | ||||||
|  |  | ||||||
|  |           <textarea class="form-control" id="dozwolone_hosty_logowania" name="dozwolone_hosty_logowania" rows="6" | ||||||
|  |             placeholder="Adresy IP lub nazwy domen — każdy w osobnej linii lub rozdzielony przecinkiem">{{ settings.dozwolone_hosty_logowania if settings and settings.dozwolone_hosty_logowania else '' }}</textarea> | ||||||
|  |  | ||||||
|  |           <small class="text-muted d-block mt-1"> | ||||||
|  |             Akceptowane separatory: przecinek (`,`), średnik (`;`) i nowa linia. | ||||||
|  |           </small> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|  |  | ||||||
|     <!-- SEKCJA: Branding --> |     <!-- SEKCJA: Branding --> | ||||||
|     <div class="card shadow-sm mb-4"> |     <div class="card shadow-sm mb-4"> | ||||||
|       <div class="card-header bg-secondary text-white d-flex align-items-center justify-content-between gap-2"> |       <div class="card-header bg-secondary text-white d-flex align-items-center justify-content-between gap-2"> | ||||||
| @@ -99,9 +116,9 @@ | |||||||
|           </div> |           </div> | ||||||
|  |  | ||||||
|           <div class="col-md-6"> |           <div class="col-md-6"> | ||||||
|             <label for="site_title" class="form-label">Tytuł serwisu</label> |             <label for="tytul_strony" class="form-label">Tytuł serwisu</label> | ||||||
|             <input type="text" class="form-control" id="site_title" name="site_title" |             <input type="text" class="form-control" id="tytul_strony" name="tytul_strony" | ||||||
|               value="{{ settings.site_title if settings else '' }}" placeholder="Np. Zbiórki unitraklub.pl"> |               value="{{ settings.tytul_strony if settings else '' }}" placeholder="Np. Zbiórki unitraklub.pl"> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
| @@ -112,15 +129,15 @@ | |||||||
|           <div class="col-md-6"> |           <div class="col-md-6"> | ||||||
|             <h6 class="mb-2">Menu (navbar)</h6> |             <h6 class="mb-2">Menu (navbar)</h6> | ||||||
|             <div class="form-check"> |             <div class="form-check"> | ||||||
|               <input class="form-check-input" type="radio" name="navbar_brand_mode" id="navbar_mode_logo" value="logo" |               <input class="form-check-input" type="radio" name="typ_navbar" id="navbar_mode_logo" value="logo" {% if | ||||||
|                 {% if settings and settings.navbar_brand_mode=='logo' or (settings and settings.show_logo_in_navbar) |                 settings and settings.typ_navbar=='logo' or (settings and settings.pokaz_logo_w_navbar) %}checked{% | ||||||
|                 %}checked{% endif %}> |                 endif %}> | ||||||
|               <label class="form-check-label" for="navbar_mode_logo">Pokaż logo</label> |               <label class="form-check-label" for="navbar_mode_logo">Pokaż logo</label> | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-check"> |             <div class="form-check"> | ||||||
|               <input class="form-check-input" type="radio" name="navbar_brand_mode" id="navbar_mode_text" value="text" |               <input class="form-check-input" type="radio" name="typ_navbar" id="navbar_mode_text" value="text" {% if | ||||||
|                 {% if not settings or (settings and settings.navbar_brand_mode !='logo' and not |                 not settings or (settings and settings.typ_navbar !='logo' and not settings.pokaz_logo_w_navbar) | ||||||
|                 settings.show_logo_in_navbar) %}checked{% endif %}> |                 %}checked{% endif %}> | ||||||
|               <label class="form-check-label" for="navbar_mode_text">Pokaż tekst</label> |               <label class="form-check-label" for="navbar_mode_text">Pokaż tekst</label> | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-text mt-1">Jeśli wybierzesz logo, użyjemy adresu z pola "Tytuł serwisu".</div> |             <div class="form-text mt-1">Jeśli wybierzesz logo, użyjemy adresu z pola "Tytuł serwisu".</div> | ||||||
| @@ -130,19 +147,19 @@ | |||||||
|           <div class="col-md-6"> |           <div class="col-md-6"> | ||||||
|             <h6 class="mb-2">Stopka</h6> |             <h6 class="mb-2">Stopka</h6> | ||||||
|             <div class="form-check"> |             <div class="form-check"> | ||||||
|               <input class="form-check-input" type="radio" name="footer_brand_mode" id="footer_mode_logo" value="logo" |               <input class="form-check-input" type="radio" name="typ_stopka" id="footer_mode_logo" value="logo" {% if | ||||||
|                 {% if settings and settings.footer_brand_mode=='logo' %}checked{% endif %}> |                 settings and settings.typ_stopka=='logo' %}checked{% endif %}> | ||||||
|               <label class="form-check-label" for="footer_mode_logo">Logo</label> |               <label class="form-check-label" for="footer_mode_logo">Logo</label> | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-check"> |             <div class="form-check"> | ||||||
|               <input class="form-check-input" type="radio" name="footer_brand_mode" id="footer_mode_text" value="text" |               <input class="form-check-input" type="radio" name="typ_stopka" id="footer_mode_text" value="text" {% if | ||||||
|                 {% if not settings or (settings and settings.footer_brand_mode !='logo' ) %}checked{% endif %}> |                 not settings or (settings and settings.typ_stopka !='logo' ) %}checked{% endif %}> | ||||||
|               <label class="form-check-label" for="footer_mode_text">Tekst</label> |               <label class="form-check-label" for="footer_mode_text">Tekst</label> | ||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|             <label for="footer_text" class="form-label mt-2">Tekst w stopce (gdy wybrano „Tekst”)</label> |             <label for="stopka_text" class="form-label mt-2">Tekst w stopce (gdy wybrano „Tekst”)</label> | ||||||
|             <input type="text" class="form-control" id="footer_text" name="footer_text" |             <input type="text" class="form-control" id="stopka_text" name="stopka_text" | ||||||
|               value="{{ settings.footer_text if settings and settings.footer_text else '' }}" |               value="{{ settings.stopka_text if settings and settings.stopka_text else '' }}" | ||||||
|               placeholder="Np. © {{ now().year if now else '2025' }} Zbiórki"> |               placeholder="Np. © {{ now().year if now else '2025' }} Zbiórki"> | ||||||
|             <div class="form-text">Pozostaw pusty, by użyć domyślnego.</div> |             <div class="form-text">Pozostaw pusty, by użyć domyślnego.</div> | ||||||
|           </div> |           </div> | ||||||
| @@ -153,8 +170,8 @@ | |||||||
|  |  | ||||||
|     <!-- CTA --> |     <!-- CTA --> | ||||||
|     <div class="d-flex justify-content-between"> |     <div class="d-flex justify-content-between"> | ||||||
|       <a href="{{ url_for('admin_dashboard') }}" class="btn btn-outline-light border">Powrót</a> |       <a href="{{ url_for('admin_dashboard') }}" class="btn btn-outline-light">Powrót</a> | ||||||
|       <button type="submit" class="btn btn-primary">Zapisz ustawienia</button> |       <button type="submit" class="btn btn-success">Zapisz ustawienia</button> | ||||||
|     </div> |     </div> | ||||||
|   </form> |   </form> | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -14,13 +14,13 @@ | |||||||
|     <nav class="navbar navbar-expand-lg"> |     <nav class="navbar navbar-expand-lg"> | ||||||
|         <div class="container"> |         <div class="container"> | ||||||
|             <a class="navbar-brand d-flex align-items-center gap-2" href="{{ url_for('index') }}"> |             <a class="navbar-brand d-flex align-items-center gap-2" href="{{ url_for('index') }}"> | ||||||
|                 {% set nav_mode = (global_settings.navbar_brand_mode if global_settings and |                 {% set nav_mode = (global_settings.typ_navbar if global_settings and | ||||||
|                 global_settings.navbar_brand_mode else ('logo' if global_settings and |                 global_settings.typ_navbar else ('logo' if global_settings and | ||||||
|                 global_settings.show_logo_in_navbar else 'text')) %} |                 global_settings.pokaz_logo_w_navbar else 'text')) %} | ||||||
|                 {% if nav_mode == 'logo' and global_settings and global_settings.logo_url %} |                 {% if nav_mode == 'logo' and global_settings and global_settings.logo_url %} | ||||||
|                 <img src="{{ global_settings.logo_url }}" alt="Logo" style="max-height:40px; vertical-align:middle;"> |                 <img src="{{ global_settings.logo_url }}" alt="Logo" style="max-height:40px; vertical-align:middle;"> | ||||||
|                 {% else %} |                 {% else %} | ||||||
|                 <span>{{ global_settings.site_title if global_settings and global_settings.site_title else "Zbiórki" |                 <span>{{ global_settings.tytul_strony if global_settings and global_settings.tytul_strony else "Zbiórki" | ||||||
|                     }}</span> |                     }}</span> | ||||||
|                 {% endif %} |                 {% endif %} | ||||||
|             </a> |             </a> | ||||||
| @@ -72,12 +72,12 @@ | |||||||
|  |  | ||||||
|     <!-- stopka --> |     <!-- stopka --> | ||||||
|     <footer class="mt-auto text-center py-3 border-top" style="background: var(--surface-0);"> |     <footer class="mt-auto text-center py-3 border-top" style="background: var(--surface-0);"> | ||||||
|         {% set footer_mode = global_settings.footer_brand_mode if global_settings and global_settings.footer_brand_mode |         {% set footer_mode = global_settings.typ_stopka if global_settings and global_settings.typ_stopka | ||||||
|         else 'text' %} |         else 'text' %} | ||||||
|         {% if footer_mode == 'logo' and global_settings and global_settings.logo_url %} |         {% if footer_mode == 'logo' and global_settings and global_settings.logo_url %} | ||||||
|         <img src="{{ global_settings.logo_url }}" alt="Logo" style="max-height:28px;"> |         <img src="{{ global_settings.logo_url }}" alt="Logo" style="max-height:28px;"> | ||||||
|         {% else %} |         {% else %} | ||||||
|         {{ global_settings.footer_text if global_settings and global_settings.footer_text else "© " ~ (now().year if now |         {{ global_settings.stopka_text if global_settings and global_settings.stopka_text else "© " ~ (now().year if now | ||||||
|         else '2025') ~ " linuxiarz.pl" }} |         else '2025') ~ " linuxiarz.pl" }} | ||||||
|         {% endif %} |         {% endif %} | ||||||
|         <div class="small text-muted">v{{ APP_VERSION }}</div> |         <div class="small text-muted">v{{ APP_VERSION }}</div> | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ zbiórki{% endif %}{% endblock %} | |||||||
| </div> | </div> | ||||||
|  |  | ||||||
| {% if zbiorki and zbiorki|length > 0 %} | {% if zbiorki and zbiorki|length > 0 %} | ||||||
| <div class="row g-4"> | <div class="row g-4 pb-5"> | ||||||
|     {% for z in zbiorki %} |     {% for z in zbiorki %} | ||||||
|     {% set progress = (z.stan / z.cel * 100) if z.cel > 0 else 0 %} |     {% set progress = (z.stan / z.cel * 100) if z.cel > 0 else 0 %} | ||||||
|     {% set progress_clamped = 100 if progress > 100 else (0 if progress < 0 else progress) %} <div |     {% set progress_clamped = 100 if progress > 100 else (0 if progress < 0 else progress) %} <div | ||||||
| @@ -54,7 +54,7 @@ zbiórki{% endif %}{% endblock %} | |||||||
|                     {% if z.cel > 0 %} |                     {% if z.cel > 0 %} | ||||||
|                     {% set delta = z.cel - z.stan %} |                     {% set delta = z.cel - z.stan %} | ||||||
|                     {% if delta > 0 %} |                     {% if delta > 0 %} | ||||||
|                     {# CHANGE: mocniejszy badge „Brakuje” #} |  | ||||||
|                     <span class="badge bg-dark border border-warning"> |                     <span class="badge bg-dark border border-warning"> | ||||||
|                         Brakuje: {{ delta|round(2) }} PLN |                         Brakuje: {{ delta|round(2) }} PLN | ||||||
|                     </span> |                     </span> | ||||||
| @@ -83,10 +83,14 @@ zbiórki{% endif %}{% endblock %} | |||||||
|                 </div> |                 </div> | ||||||
|  |  | ||||||
|                 <div class="mt-auto pt-2"> |                 <div class="mt-auto pt-2"> | ||||||
|                     <a href="{{ url_for('zbiorka', zbiorka_id=z.id) }}" class="stretched-link"></a> |  | ||||||
|  |                     {# TO POWODUJE ZE BLOK JEST KLIKALNY #} | ||||||
|  |                     <!-- <a href="{{ url_for('zbiorka', zbiorka_id=z.id) }}" class="stretched-link"></a> --> | ||||||
|  |  | ||||||
|                     <div class="d-grid"> |                     <div class="d-grid"> | ||||||
|                         <a href="{{ url_for('zbiorka', zbiorka_id=z.id) }}" class="btn btn-outline-light btn-sm w-100"> |                         <a href="{{ url_for('zbiorka', zbiorka_id=z.id) }}" | ||||||
|                             Szczegóły |                             class="btn btn-outline-light btn-sm w-100 btn-opis"> | ||||||
|  |                             Otwórz opis | ||||||
|                         </a> |                         </a> | ||||||
|                     </div> |                     </div> | ||||||
|                 </div> |                 </div> | ||||||
| @@ -105,7 +109,7 @@ zbiórki{% endif %}{% endblock %} | |||||||
|         {% else %} |         {% else %} | ||||||
|         <h5 class="mb-2">Brak aktywnych zbiórek</h5> |         <h5 class="mb-2">Brak aktywnych zbiórek</h5> | ||||||
|         <p class="text-muted mb-4">Wygląda na to, że teraz nic nie zbieramy.</p> |         <p class="text-muted mb-4">Wygląda na to, że teraz nic nie zbieramy.</p> | ||||||
|         {% if current_user.is_authenticated and current_user.is_admin %} |         {% if current_user.is_authenticated and current_user.czy_admin %} | ||||||
|         <a href="{{ url_for('admin_dashboard') }}" class="btn btn-primary">Utwórz nową zbiórkę</a> |         <a href="{{ url_for('admin_dashboard') }}" class="btn btn-primary">Utwórz nową zbiórkę</a> | ||||||
|         {% else %} |         {% else %} | ||||||
|         <a href="{{ url_for('zbiorki_zrealizowane') }}" class="btn btn-primary">Zobacz zrealizowane</a> |         <a href="{{ url_for('zbiorki_zrealizowane') }}" class="btn btn-primary">Zobacz zrealizowane</a> | ||||||
|   | |||||||
| @@ -18,21 +18,21 @@ | |||||||
|                         {% endif %} |                         {% endif %} | ||||||
|  |  | ||||||
|                         <div class="mb-3"> |                         <div class="mb-3"> | ||||||
|                             <label for="username" class="form-label">Nazwa użytkownika</label> |                             <label for="uzytkownik" class="form-label">Nazwa użytkownika</label> | ||||||
|                             <input type="text" class="form-control" id="username" name="username" |                             <input type="text" class="form-control" id="uzytkownik" name="uzytkownik" | ||||||
|                                 autocomplete="username" autocapitalize="none" spellcheck="false" required autofocus> |                                 autocomplete="username" autocapitalize="none" spellcheck="false" required autofocus> | ||||||
|                             <div class="invalid-feedback">Podaj nazwę użytkownika.</div> |                             <div class="invalid-feedback">Podaj nazwę użytkownika.</div> | ||||||
|                         </div> |                         </div> | ||||||
|  |  | ||||||
|                         <div class="mb-2"> |                         <div class="mb-2"> | ||||||
|                             <label for="password" class="form-label d-flex justify-content-between align-items-center"> |                             <label for="haslo" class="form-label d-flex justify-content-between align-items-center"> | ||||||
|                                 <span>Hasło</span> |                                 <span>Hasło</span> | ||||||
|                                 <small id="capsWarning" class="text-muted" style="display:none;">CAPS LOCK |                                 <small id="capsWarning" class="text-muted" style="display:none;">CAPS LOCK | ||||||
|                                     włączony</small> |                                     włączony</small> | ||||||
|                             </label> |                             </label> | ||||||
|  |  | ||||||
|                             <div class="input-group"> |                             <div class="input-group"> | ||||||
|                                 <input type="password" class="form-control" id="password" name="password" |                                 <input type="password" class="form-control" id="haslo" name="haslo" | ||||||
|                                     autocomplete="current-password" required minlength="5"> |                                     autocomplete="current-password" required minlength="5"> | ||||||
|                                 <button type="button" class="btn btn-secondary rounded-end" id="togglePw" |                                 <button type="button" class="btn btn-secondary rounded-end" id="togglePw" | ||||||
|                                     aria-label="Pokaż/ukryj hasło">Pokaż</button> |                                     aria-label="Pokaż/ukryj hasło">Pokaż</button> | ||||||
|   | |||||||
| @@ -14,20 +14,20 @@ | |||||||
|                     <form method="post" class="needs-validation" novalidate> |                     <form method="post" class="needs-validation" novalidate> | ||||||
|  |  | ||||||
|                         <div class="mb-3"> |                         <div class="mb-3"> | ||||||
|                             <label for="username" class="form-label">Nazwa użytkownika</label> |                             <label for="uzytkownik" class="form-label">Nazwa użytkownika</label> | ||||||
|                             <input type="text" class="form-control" id="username" name="username" |                             <input type="text" class="form-control" id="uzytkownik" name="uzytkownik" | ||||||
|                                 autocomplete="username" autocapitalize="none" spellcheck="false" required autofocus> |                                 autocomplete="username" autocapitalize="none" spellcheck="false" required autofocus> | ||||||
|                             <div class="invalid-feedback">Podaj nazwę użytkownika.</div> |                             <div class="invalid-feedback">Podaj nazwę użytkownika.</div> | ||||||
|                         </div> |                         </div> | ||||||
|  |  | ||||||
|                         <div class="mb-3"> |                         <div class="mb-3"> | ||||||
|                             <label for="password" class="form-label d-flex justify-content-between align-items-center"> |                             <label for="haslo" class="form-label d-flex justify-content-between align-items-center"> | ||||||
|                                 <span>Hasło</span> |                                 <span>Hasło</span> | ||||||
|                                 <small id="capsWarning" class="text-muted" style="display:none;">CAPS LOCK |                                 <small id="capsWarning" class="text-muted" style="display:none;">CAPS LOCK | ||||||
|                                     włączony</small> |                                     włączony</small> | ||||||
|                             </label> |                             </label> | ||||||
|                             <div class="input-group"> |                             <div class="input-group"> | ||||||
|                                 <input type="password" class="form-control" id="password" name="password" |                                 <input type="password" class="form-control" id="haslo" name="haslo" | ||||||
|                                     autocomplete="new-password" required minlength="6"> |                                     autocomplete="new-password" required minlength="6"> | ||||||
|                                 <button type="button" class="btn btn-secondary" id="togglePw" |                                 <button type="button" class="btn btn-secondary" id="togglePw" | ||||||
|                                     aria-label="Pokaż/ukryj hasło">Pokaż</button> |                                     aria-label="Pokaż/ukryj hasło">Pokaż</button> | ||||||
|   | |||||||
| @@ -62,8 +62,8 @@ | |||||||
|                   {% endif %} |                   {% endif %} | ||||||
|                   <span class="fw-semibold">{{ it.nazwa }}</span> |                   <span class="fw-semibold">{{ it.nazwa }}</span> | ||||||
|                   {% if it.link %} |                   {% if it.link %} | ||||||
|                   <a href="{{ it.link }}" target="_blank" rel="noopener" |                   <a href="{{ it.link }}" target="_blank" rel="noopener" class="btn btn-sm btn-outline-light ms-2">Sklep | ||||||
|                     class="btn btn-sm btn-outline-light border ms-2">Sklep ↗</a> |                     ↗</a> | ||||||
|                   {% endif %} |                   {% endif %} | ||||||
|                 </div> |                 </div> | ||||||
|                 <div> |                 <div> | ||||||
| @@ -103,23 +103,28 @@ | |||||||
|               <h5 class="mb-0">Postęp</h5> |               <h5 class="mb-0">Postęp</h5> | ||||||
|  |  | ||||||
|               <div class="d-flex flex-wrap align-items-center gap-2"> |               <div class="d-flex flex-wrap align-items-center gap-2"> | ||||||
|                 {% if has_cel and not zbiorka.ukryj_kwote %} |                 {% if has_cel and not zbiorka.ukryj_kwote and zbiorka.pokaz_postep_finanse %} | ||||||
|                 <span class="badge bg-dark border" style="border-color: var(--border);"> |                 <span class="badge bg-dark border" style="border-color: var(--border);"> | ||||||
|                   Finanse: {{ zbiorka.stan|round(2) }} / {{ zbiorka.cel|round(2) }} PLN |                   Finanse: {{ zbiorka.stan|round(2) }} / {{ zbiorka.cel|round(2) }} PLN | ||||||
|                 </span> |                 </span> | ||||||
|                 {% endif %} |                 {% endif %} | ||||||
|                 {% if has_items %} |  | ||||||
|  |                 {% if has_items and zbiorka.pokaz_postep_pozycje %} | ||||||
|                 <span class="badge bg-secondary">Pozycje: {{ kupione_cnt }}/{{ total_cnt }}</span> |                 <span class="badge bg-secondary">Pozycje: {{ kupione_cnt }}/{{ total_cnt }}</span> | ||||||
|                 {% if not zbiorka.ukryj_kwote and (suma_all or 0) > 0 %} |  | ||||||
|                 <span class="badge bg-secondary">Zakupy (kwotowo): |  | ||||||
|                   {{ (suma_kupione or 0)|round(2) }} / {{ (suma_all or 0)|round(2) }} PLN |  | ||||||
|                 </span> |  | ||||||
|                 {% endif %} |                 {% endif %} | ||||||
|  |  | ||||||
|  |                 {% if has_items and not zbiorka.ukryj_kwote and (suma_all or 0) > 0 and zbiorka.pokaz_postep_kwotowo %} | ||||||
|  |                 <span class="badge bg-secondary"> | ||||||
|  |                   Zakupy (kwotowo): {{ (suma_kupione or 0)|round(2) }} / {{ (suma_all or 0)|round(2) }} PLN | ||||||
|  |                 </span> | ||||||
|                 {% endif %} |                 {% endif %} | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|             <hr class="hr-bw"> |             <hr class="hr-bw"> | ||||||
|             <!-- Pasek: Finanse (zawsze) --> |  | ||||||
|  |             {# Pasek: Finanse #} | ||||||
|  |             {% if zbiorka.pokaz_postep_finanse %} | ||||||
|             <div class="mb-3"> |             <div class="mb-3"> | ||||||
|               <small class="text-muted">Finanse</small> |               <small class="text-muted">Finanse</small> | ||||||
|               <div class="progress" role="progressbar" aria-valuenow="{{ progress_clamped|round(2) }}" aria-valuemin="0" |               <div class="progress" role="progressbar" aria-valuenow="{{ progress_clamped|round(2) }}" aria-valuemin="0" | ||||||
| @@ -130,9 +135,10 @@ | |||||||
|                 {% if zbiorka.ukryj_kwote %}—{% else %}{{ progress|round(1) }}%{% endif %} |                 {% if zbiorka.ukryj_kwote %}—{% else %}{{ progress|round(1) }}%{% endif %} | ||||||
|               </small> |               </small> | ||||||
|             </div> |             </div> | ||||||
|  |             {% endif %} | ||||||
|  |  | ||||||
|             {% if has_items %} |             {# Pasek: Zakupy sztukami #} | ||||||
|             <!-- Pasek: Zakupy sztukami --> |             {% if has_items and zbiorka.pokaz_postep_pozycje %} | ||||||
|             <div class="mb-3"> |             <div class="mb-3"> | ||||||
|               <small class="text-muted">Zakupy (liczba pozycji)</small> |               <small class="text-muted">Zakupy (liczba pozycji)</small> | ||||||
|               <div class="progress" role="progressbar" aria-valuenow="{{ items_pct|round(2) }}" aria-valuemin="0" |               <div class="progress" role="progressbar" aria-valuenow="{{ items_pct|round(2) }}" aria-valuemin="0" | ||||||
| @@ -141,9 +147,10 @@ | |||||||
|               </div> |               </div> | ||||||
|               <small class="text-muted">{{ items_pct|round(1) }}%</small> |               <small class="text-muted">{{ items_pct|round(1) }}%</small> | ||||||
|             </div> |             </div> | ||||||
|  |             {% endif %} | ||||||
|  |  | ||||||
|             {% if not zbiorka.ukryj_kwote and (suma_all or 0) > 0 %} |             {# Pasek: Zakupy kwotowo #} | ||||||
|             <!-- Pasek: Zakupy kwotowo --> |             {% if has_items and not zbiorka.ukryj_kwote and (suma_all or 0) > 0 and zbiorka.pokaz_postep_kwotowo %} | ||||||
|             <div> |             <div> | ||||||
|               <small class="text-muted">Zakupy (kwotowo)</small> |               <small class="text-muted">Zakupy (kwotowo)</small> | ||||||
|               <div class="progress" role="progressbar" aria-valuenow="{{ suma_pct|round(2) }}" aria-valuemin="0" |               <div class="progress" role="progressbar" aria-valuenow="{{ suma_pct|round(2) }}" aria-valuemin="0" | ||||||
| @@ -153,12 +160,14 @@ | |||||||
|               <small class="text-muted">{{ suma_pct|round(1) }}%</small> |               <small class="text-muted">{{ suma_pct|round(1) }}%</small> | ||||||
|             </div> |             </div> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|             {% endif %} |  | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
|  |       {% set show_iban = zbiorka.uzyj_konta and zbiorka.numer_konta %} | ||||||
|  |       {% set show_blik = zbiorka.uzyj_blik and zbiorka.numer_telefonu_blik %} | ||||||
|  |  | ||||||
|       <!-- Kolumna prawa: płatności (sticky) --> |       <!-- Kolumna prawa: płatności (sticky) --> | ||||||
|       <div class="col-md-4"> |       <div class="col-md-4"> | ||||||
|         <div class="card shadow-sm wspomoz-card sticky-md" style="top: var(--sticky-offset, 1rem);"> |         <div class="card shadow-sm wspomoz-card sticky-md" style="top: var(--sticky-offset, 1rem);"> | ||||||
| @@ -169,17 +178,15 @@ | |||||||
|               {% if has_cel and not zbiorka.ukryj_kwote %} |               {% if has_cel and not zbiorka.ukryj_kwote %} | ||||||
|               {% set brak = (zbiorka.cel - zbiorka.stan) %} |               {% set brak = (zbiorka.cel - zbiorka.stan) %} | ||||||
|               {% if brak > 0 %} |               {% if brak > 0 %} | ||||||
|               <span class="badge bg-warning text-dark border border-warning"> |               <span class="badge bg-warning text-dark border border-warning">Brakuje: {{ brak|round(2) }} PLN</span> | ||||||
|                 Brakuje: {{ brak|round(2) }} PLN |  | ||||||
|               </span> |  | ||||||
|               {% else %} |               {% else %} | ||||||
|               <span class="badge rounded-pill" style="background: var(--accent); color:#111;"> |               <span class="badge rounded-pill" style="background: var(--accent); color:#111;">Zrealizowana</span> | ||||||
|                 Zrealizowana |  | ||||||
|               </span> |  | ||||||
|               {% endif %} |               {% endif %} | ||||||
|               {% endif %} |               {% endif %} | ||||||
|             </div> |             </div> | ||||||
|  |  | ||||||
|  |             {% if show_iban or show_blik %} | ||||||
|  |             {% if show_iban %} | ||||||
|             <!-- Numer konta --> |             <!-- Numer konta --> | ||||||
|             <div> |             <div> | ||||||
|               <label for="ibanInput" class="form-label fw-semibold mb-1">Numer konta</label> |               <label for="ibanInput" class="form-label fw-semibold mb-1">Numer konta</label> | ||||||
| @@ -188,11 +195,13 @@ | |||||||
|                   class="form-control form-control-sm bg-transparent text-light border monospace-input text-truncate" |                   class="form-control form-control-sm bg-transparent text-light border monospace-input text-truncate" | ||||||
|                   value="{{ zbiorka.numer_konta }}" readonly autocomplete="off" autocorrect="off" autocapitalize="off" |                   value="{{ zbiorka.numer_konta }}" readonly autocomplete="off" autocorrect="off" autocapitalize="off" | ||||||
|                   spellcheck="false" inputmode="text" aria-label="Numer konta do wpłaty"> |                   spellcheck="false" inputmode="text" aria-label="Numer konta do wpłaty"> | ||||||
|                 <button class="btn btn-sm btn-outline-light border copy-btn" type="button" data-copy-input="#ibanInput" |                 <button class="btn btn-sm btn-outline-light copy-btn" type="button" data-copy-input="#ibanInput" | ||||||
|                   aria-label="Kopiuj numer konta">Kopiuj</button> |                   aria-label="Kopiuj numer konta">Kopiuj</button> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|  |             {% endif %} | ||||||
|  |  | ||||||
|  |             {% if show_blik %} | ||||||
|             <!-- Telefon BLIK --> |             <!-- Telefon BLIK --> | ||||||
|             <div> |             <div> | ||||||
|               <label for="blikInput" class="form-label fw-semibold mb-1">Telefon BLIK</label> |               <label for="blikInput" class="form-label fw-semibold mb-1">Telefon BLIK</label> | ||||||
| @@ -201,10 +210,16 @@ | |||||||
|                   class="form-control form-control-sm bg-transparent text-light border monospace-input text-truncate" |                   class="form-control form-control-sm bg-transparent text-light border monospace-input text-truncate" | ||||||
|                   value="{{ zbiorka.numer_telefonu_blik }}" readonly autocomplete="off" autocorrect="off" |                   value="{{ zbiorka.numer_telefonu_blik }}" readonly autocomplete="off" autocorrect="off" | ||||||
|                   autocapitalize="off" spellcheck="false" inputmode="numeric" aria-label="Telefon BLIK"> |                   autocapitalize="off" spellcheck="false" inputmode="numeric" aria-label="Telefon BLIK"> | ||||||
|                 <button class="btn btn-sm btn-outline-light border copy-btn" type="button" data-copy-input="#blikInput" |                 <button class="btn btn-sm btn-outline-light copy-btn" type="button" data-copy-input="#blikInput" | ||||||
|                   aria-label="Kopiuj numer BLIK">Kopiuj</button> |                   aria-label="Kopiuj numer BLIK">Kopiuj</button> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|  |             {% endif %} | ||||||
|  |             {% else %} | ||||||
|  |             <div class="alert alert-secondary mb-0"> | ||||||
|  |               Kanały płatności są wyłączone dla tej zbiórki. | ||||||
|  |             </div> | ||||||
|  |             {% endif %} | ||||||
|  |  | ||||||
|             {% if not zbiorka.ukryj_kwote %} |             {% if not zbiorka.ukryj_kwote %} | ||||||
|             <ul class="list-group list-group-flush small"> |             <ul class="list-group list-group-flush small"> | ||||||
| @@ -237,8 +252,7 @@ | |||||||
|             </ul> |             </ul> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|  |  | ||||||
|  |             {% if current_user.is_authenticated and current_user.czy_admin %} | ||||||
|             {% if current_user.is_authenticated and current_user.is_admin %} |  | ||||||
|             <hr> |             <hr> | ||||||
|             <div class="d-grid gap-2 mt-2"> |             <div class="d-grid gap-2 mt-2"> | ||||||
|               <a href="{{ url_for('dodaj_wplate', zbiorka_id=zbiorka.id) }}" class="btn btn-outline-light btn-sm">Dodaj |               <a href="{{ url_for('dodaj_wplate', zbiorka_id=zbiorka.id) }}" class="btn btn-outline-light btn-sm">Dodaj | ||||||
| @@ -264,9 +278,8 @@ | |||||||
|             {% if aktywnosci and aktywnosci|length > 0 %} |             {% if aktywnosci and aktywnosci|length > 0 %} | ||||||
|             <small class="text-muted">Łącznie pozycji: {{ aktywnosci|length }}</small> |             <small class="text-muted">Łącznie pozycji: {{ aktywnosci|length }}</small> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|             {% if current_user.is_authenticated and current_user.is_admin %} |             {% if current_user.is_authenticated and current_user.czy_admin %} | ||||||
|             <a href="{{ url_for('transakcje_zbiorki', zbiorka_id=zbiorka.id) }}" |             <a href="{{ url_for('transakcje_zbiorki', zbiorka_id=zbiorka.id) }}" class="btn btn-sm btn-outline-light"> | ||||||
|               class="btn btn-sm btn-outline-light border"> |  | ||||||
|               Zarządzaj |               Zarządzaj | ||||||
|             </a> |             </a> | ||||||
|             {% endif %} |             {% endif %} | ||||||
| @@ -308,7 +321,7 @@ | |||||||
|       <!-- Akcje dolne --> |       <!-- Akcje dolne --> | ||||||
|       <div class="d-flex gap-2 justify-content-between mt-3"> |       <div class="d-flex gap-2 justify-content-between mt-3"> | ||||||
|         <div></div> |         <div></div> | ||||||
|         <a href="{{ url_for('index') }}" class="btn btn-outline-light border">Powrót do listy</a> |         <a href="{{ url_for('index') }}" class="btn btn-outline-light">Powrót do listy</a> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
|     </div> |     </div> | ||||||
| @@ -318,5 +331,4 @@ | |||||||
|     {{ super() }} |     {{ super() }} | ||||||
|     <script src="{{ url_for('static', filename='js/zbiorka.js') }}?v={{ APP_VERSION }}"></script> |     <script src="{{ url_for('static', filename='js/zbiorka.js') }}?v={{ APP_VERSION }}"></script> | ||||||
|     <script src="{{ url_for('static', filename='js/progress.js') }}?v={{ APP_VERSION }}"></script> |     <script src="{{ url_for('static', filename='js/progress.js') }}?v={{ APP_VERSION }}"></script> | ||||||
|     <script src="{{ url_for('static', filename='js/progress.js') }}?v={{ APP_VERSION }}"></script> |  | ||||||
|     {% endblock %} |     {% endblock %} | ||||||
		Reference in New Issue
	
	Block a user