From 6bd6284f49fddaae6860da182c0029567f10d09f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mateusz=20Gruszczy=C5=84ski?= Date: Fri, 7 Mar 2025 22:35:43 +0100 Subject: [PATCH] first commit --- .gitignore | 4 + Dockerfile | 13 ++ app.py | 268 ++++++++++++++++++++++++++++++ docker-compose.yml | 12 ++ requirements.txt | 6 + run_waitress.py | 7 + static/css/custom.css | 31 ++++ templates/admin/add_wplata.html | 16 ++ templates/admin/add_zbiorka.html | 43 +++++ templates/admin/dashboard.html | 50 ++++++ templates/admin/edit_zbiorka.html | 43 +++++ templates/admin/edytuj_stan.html | 13 ++ templates/base.html | 39 +++++ templates/index.html | 14 ++ templates/login.html | 17 ++ templates/register.html | 16 ++ templates/zbiorka.html | 60 +++++++ 17 files changed, 652 insertions(+) create mode 100644 .gitignore create mode 100644 Dockerfile create mode 100644 app.py create mode 100644 docker-compose.yml create mode 100644 requirements.txt create mode 100644 run_waitress.py create mode 100644 static/css/custom.css create mode 100644 templates/admin/add_wplata.html create mode 100644 templates/admin/add_zbiorka.html create mode 100644 templates/admin/dashboard.html create mode 100644 templates/admin/edit_zbiorka.html create mode 100644 templates/admin/edytuj_stan.html create mode 100644 templates/base.html create mode 100644 templates/index.html create mode 100644 templates/login.html create mode 100644 templates/register.html create mode 100644 templates/zbiorka.html diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..1931769 --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +__pycache__ +data/ +instance/ +venv/ diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..a659800 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,13 @@ +FROM python:3.13-slim +WORKDIR /app + +COPY requirements.txt requirements.txt +RUN apt-get update && apt-get install -y build-essential && \ + pip install --upgrade pip && pip install -r requirements.txt + +COPY . . +RUN mkdir -p /app/instance + +EXPOSE 8080 + +CMD ["python", "run_waitress.py"] diff --git a/app.py b/app.py new file mode 100644 index 0000000..9deeadc --- /dev/null +++ b/app.py @@ -0,0 +1,268 @@ +from flask import Flask, render_template, request, redirect, url_for, flash +from flask_sqlalchemy import SQLAlchemy +from flask_login import LoginManager, login_user, login_required, logout_user, current_user, UserMixin +from werkzeug.security import generate_password_hash, check_password_hash +from datetime import datetime +from markupsafe import Markup +import markdown as md +from flask import abort + +app = Flask(__name__) +app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///baza.db' + +app.config['SECRET_KEY'] = 'tajny_klucz' + +# Konfiguracja rejestracji i admina +app.config['ALLOW_REGISTRATION'] = False +app.config['MAIN_ADMIN_USERNAME'] = 'admin' +app.config['MAIN_ADMIN_PASSWORD'] = 'admin' + +db = SQLAlchemy(app) +login_manager = LoginManager(app) +login_manager.login_view = 'login' + +# MODELE + +class User(UserMixin, db.Model): + id = db.Column(db.Integer, primary_key=True) + username = db.Column(db.String(80), unique=True, nullable=False) + password_hash = db.Column(db.String(128), nullable=False) + is_admin = db.Column(db.Boolean, default=False) # Flaga głównego administratora + + def set_password(self, password): + self.password_hash = generate_password_hash(password) + + def check_password(self, password): + return check_password_hash(self.password_hash, password) + +class Zbiorka(db.Model): + id = db.Column(db.Integer, primary_key=True) + nazwa = db.Column(db.String(100), nullable=False) + opis = db.Column(db.Text, nullable=False) + numer_konta = db.Column(db.String(50), nullable=False) + numer_telefonu_blik = db.Column(db.String(50), nullable=False) + cel = db.Column(db.Float, nullable=False, default=0.0) + stan = db.Column(db.Float, default=0.0) + ukryta = db.Column(db.Boolean, default=False) + ukryj_kwote = db.Column(db.Boolean, default=False) + wplaty = db.relationship('Wplata', backref='zbiorka', lazy=True) + +class Wplata(db.Model): + id = db.Column(db.Integer, primary_key=True) + zbiorka_id = db.Column(db.Integer, db.ForeignKey('zbiorka.id'), nullable=False) + kwota = db.Column(db.Float, nullable=False) + data = db.Column(db.DateTime, default=datetime.utcnow) + opis = db.Column(db.Text, nullable=True) # Opis wpłaty + +@login_manager.user_loader +def load_user(user_id): + return User.query.get(int(user_id)) + +# Dodaj filtr Markdown – pozwala na zagnieżdżanie linków i obrazków w opisie +@app.template_filter('markdown') +def markdown_filter(text): + return Markup(md.markdown(text)) + +# TRASY PUBLICZNE + +@app.route('/') +def index(): + zbiorki = Zbiorka.query.filter_by(ukryta=False).all() + return render_template('index.html', zbiorki=zbiorki) + +@app.errorhandler(404) +def page_not_found(e): + return redirect(url_for('index')) + +@app.route('/zbiorka/') +def zbiorka(zbiorka_id): + zb = Zbiorka.query.get_or_404(zbiorka_id) + # Jeżeli zbiórka jest ukryta i użytkownik nie jest administratorem, zwróć 404 + if zb.ukryta and (not current_user.is_authenticated or not current_user.is_admin): + abort(404) + return render_template('zbiorka.html', zbiorka=zb) + +# TRASY LOGOWANIA I REJESTRACJI + +@app.route('/login', methods=['GET', 'POST']) +def login(): + if request.method == 'POST': + username = request.form['username'] + password = request.form['password'] + user = User.query.filter_by(username=username).first() + if user and user.check_password(password): + login_user(user) + flash('Zalogowano pomyślnie', 'success') + next_page = request.args.get('next') + return redirect(next_page) if next_page else redirect(url_for('admin_dashboard')) + else: + flash('Nieprawidłowe dane logowania', 'danger') + return render_template('login.html') + +@app.route('/logout') +@login_required +def logout(): + logout_user() + flash('Wylogowano', 'success') + return redirect(url_for('login')) + +@app.route('/register', methods=['GET', 'POST']) +def register(): + if not app.config.get('ALLOW_REGISTRATION', False): + flash('Rejestracja została wyłączona przez administratora', 'danger') + return redirect(url_for('login')) + if request.method == 'POST': + username = request.form['username'] + password = request.form['password'] + if User.query.filter_by(username=username).first(): + flash('Użytkownik już istnieje', 'danger') + return redirect(url_for('register')) + new_user = User(username=username) + new_user.set_password(password) + db.session.add(new_user) + db.session.commit() + flash('Konto utworzone, możesz się zalogować', 'success') + return redirect(url_for('login')) + return render_template('register.html') + +# PANEL ADMINISTRACYJNY + +@app.route('/admin') +@login_required +def admin_dashboard(): + # Tylko użytkownik z flagą is_admin ma dostęp do pełnych funkcji panelu + if not current_user.is_admin: + flash('Brak uprawnień do panelu administracyjnego', 'danger') + return redirect(url_for('index')) + zbiorki = Zbiorka.query.all() + return render_template('admin/dashboard.html', zbiorki=zbiorki) + +@app.route('/admin/zbiorka/dodaj', methods=['GET', 'POST']) +@login_required +def dodaj_zbiorka(): + if not current_user.is_admin: + flash('Brak uprawnień', 'danger') + return redirect(url_for('index')) + if request.method == 'POST': + nazwa = request.form['nazwa'] + opis = request.form['opis'] + numer_konta = request.form['numer_konta'] + numer_telefonu_blik = request.form['numer_telefonu_blik'] + cel = float(request.form['cel']) + # Jeśli checkbox jest zaznaczony, wartość będzie obecna w formularzu + ukryj_kwote = 'ukryj_kwote' in request.form + nowa_zbiorka = Zbiorka( + nazwa=nazwa, + opis=opis, + numer_konta=numer_konta, + numer_telefonu_blik=numer_telefonu_blik, + cel=cel, + ukryj_kwote=ukryj_kwote + ) + db.session.add(nowa_zbiorka) + db.session.commit() + flash('Zbiórka została dodana', 'success') + return redirect(url_for('admin_dashboard')) + + # Zwracamy szablon dla żądania GET + return render_template('admin/add_zbiorka.html') + +@app.route('/admin/zbiorka/edytuj/', methods=['GET', 'POST']) +@login_required +def edytuj_zbiorka(zbiorka_id): + if not current_user.is_admin: + flash('Brak uprawnień', 'danger') + return redirect(url_for('index')) + zb = Zbiorka.query.get_or_404(zbiorka_id) + if request.method == 'POST': + zb.nazwa = request.form['nazwa'] + zb.opis = request.form['opis'] + zb.numer_konta = request.form['numer_konta'] + zb.numer_telefonu_blik = request.form['numer_telefonu_blik'] + try: + zb.cel = float(request.form['cel']) + except ValueError: + flash('Podano nieprawidłową wartość dla celu zbiórki', 'danger') + return render_template('admin/edit_zbiorka.html', zbiorka=zb) + # Ustawienie opcji ukrywania kwot, jeśli checkbox jest zaznaczony + zb.ukryj_kwote = 'ukryj_kwote' in request.form + db.session.commit() + flash('Zbiórka została zaktualizowana', 'success') + return redirect(url_for('admin_dashboard')) + # Dla żądania GET zwracamy formularz edycji + return render_template('admin/edit_zbiorka.html', zbiorka=zb) + +# TRASA DODAWANIA WPŁATY Z OPISEM +# TRASA DODAWANIA WPŁATY W PANELU ADMINA +@app.route('/admin/zbiorka//wplata/dodaj', methods=['GET', 'POST']) +@login_required +def admin_dodaj_wplate(zbiorka_id): + if not current_user.is_admin: + flash('Brak uprawnień', 'danger') + return redirect(url_for('index')) + zb = Zbiorka.query.get_or_404(zbiorka_id) + if request.method == 'POST': + kwota = float(request.form['kwota']) + opis = request.form.get('opis', '') + nowa_wplata = Wplata(zbiorka_id=zb.id, kwota=kwota, opis=opis) + zb.stan += kwota # Aktualizacja stanu zbiórki + db.session.add(nowa_wplata) + db.session.commit() + flash('Wpłata została dodana', 'success') + return redirect(url_for('admin_dashboard')) + return render_template('admin/add_wplata.html', zbiorka=zb) + +@app.route('/admin/zbiorka/usun/', methods=['POST']) +@login_required +def usun_zbiorka(zbiorka_id): + if not current_user.is_admin: + flash('Brak uprawnień', 'danger') + return redirect(url_for('index')) + zb = Zbiorka.query.get_or_404(zbiorka_id) + db.session.delete(zb) + db.session.commit() + flash('Zbiórka została usunięta', 'success') + return redirect(url_for('admin_dashboard')) + +@app.route('/admin/zbiorka/edytuj_stan/', methods=['GET', 'POST']) +@login_required +def edytuj_stan(zbiorka_id): + if not current_user.is_admin: + flash('Brak uprawnień', 'danger') + return redirect(url_for('index')) + zb = Zbiorka.query.get_or_404(zbiorka_id) + if request.method == 'POST': + try: + nowy_stan = float(request.form['stan']) + except ValueError: + flash('Nieprawidłowa wartość kwoty', 'danger') + return redirect(url_for('edytuj_stan', zbiorka_id=zbiorka_id)) + zb.stan = nowy_stan + db.session.commit() + flash('Stan zbiórki został zaktualizowany', 'success') + return redirect(url_for('admin_dashboard')) + return render_template('admin/edytuj_stan.html', zbiorka=zb) + +@app.route('/admin/zbiorka/toggle_visibility/', methods=['POST']) +@login_required +def toggle_visibility(zbiorka_id): + if not current_user.is_admin: + flash('Brak uprawnień', 'danger') + return redirect(url_for('index')) + zb = Zbiorka.query.get_or_404(zbiorka_id) + zb.ukryta = not zb.ukryta + db.session.commit() + flash('Zbiórka została ' + ('ukryta' if zb.ukryta else 'przywrócona'), 'success') + return redirect(url_for('admin_dashboard')) + + +if __name__ == '__main__': + with app.app_context(): + db.create_all() + # Tworzenie konta głównego admina, jeśli nie istnieje + if not User.query.filter_by(is_admin=True).first(): + main_admin = User(username=app.config['MAIN_ADMIN_USERNAME'], is_admin=True) + main_admin.set_password(app.config['MAIN_ADMIN_PASSWORD']) + db.session.add(main_admin) + db.session.commit() + app.run(debug=True) diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 0000000..dc81b1b --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,12 @@ +version: '3.8' + +services: + app: + build: + context: . + dockerfile: Dockerfile + ports: + - "8080:8080" + volumes: + - ./instance:/app/instance + restart: unless-stopped diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..6e7db2e --- /dev/null +++ b/requirements.txt @@ -0,0 +1,6 @@ +Flask +Flask-SQLAlchemy +Flask-Login +Werkzeug +waitress +markdown \ No newline at end of file diff --git a/run_waitress.py b/run_waitress.py new file mode 100644 index 0000000..0eaf9dc --- /dev/null +++ b/run_waitress.py @@ -0,0 +1,7 @@ +from app import app, db +from waitress import serve + +if __name__ == '__main__': + with app.app_context(): + db.create_all() + serve(app, host='0.0.0.0', port=8080) diff --git a/static/css/custom.css b/static/css/custom.css new file mode 100644 index 0000000..9d0f04c --- /dev/null +++ b/static/css/custom.css @@ -0,0 +1,31 @@ +/* custom.css */ + +/* Dodatkowy odstęp od góry strony */ +body { + padding-top: 60px; +} + +/* Zwiększona wysokość progress baru */ +.progress { + height: 35px; + font-size: 1.2rem; +} + +/* Ujednolicenie wyglądu kart */ +.card { + margin-bottom: 20px; +} + +/* Drobne poprawki przycisków */ +.btn { + text-transform: uppercase; + font-weight: bold; +} + +/* Ewentualne zmiany przy linkach */ +a { + color: #ffc107; +} +a:hover { + color: #ffeb3b; +} diff --git a/templates/admin/add_wplata.html b/templates/admin/add_wplata.html new file mode 100644 index 0000000..e294e7e --- /dev/null +++ b/templates/admin/add_wplata.html @@ -0,0 +1,16 @@ +{% extends 'base.html' %} +{% block title %}Dodaj wpłatę{% endblock %} +{% block content %} +

Dodaj wpłatę do zbiórki: {{ zbiorka.nazwa }}

+
+
+ + +
+
+ + +
+ +
+{% endblock %} diff --git a/templates/admin/add_zbiorka.html b/templates/admin/add_zbiorka.html new file mode 100644 index 0000000..fd98692 --- /dev/null +++ b/templates/admin/add_zbiorka.html @@ -0,0 +1,43 @@ +{% extends 'base.html' %} +{% block title %}Dodaj zbiórkę{% endblock %} +{% block content %} +

Dodaj nową zbiórkę

+
+ +
+ + +
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+ +
+ + + + + +{% endblock %} diff --git a/templates/admin/dashboard.html b/templates/admin/dashboard.html new file mode 100644 index 0000000..255394c --- /dev/null +++ b/templates/admin/dashboard.html @@ -0,0 +1,50 @@ +{% extends 'base.html' %} +{% block title %}Panel Admina{% endblock %} +{% block content %} +

Panel Admina

+ + + + + + + + + + + + {% for z in zbiorki %} + + + + + + + {% else %} + + + + {% endfor %} + +
IDNazwaWidocznośćOpcje
{{ z.id }}{{ z.nazwa }} + {% if z.ukryta %} + Ukryta + {% else %} + Widoczna + {% endif %} + + Edytuj + Dodaj wpłatę + Edytuj stan +
+ +
+
+ +
+
Brak zbiórek
+{% endblock %} diff --git a/templates/admin/edit_zbiorka.html b/templates/admin/edit_zbiorka.html new file mode 100644 index 0000000..84c20aa --- /dev/null +++ b/templates/admin/edit_zbiorka.html @@ -0,0 +1,43 @@ +{% extends 'base.html' %} +{% block title %}Edytuj zbiórkę{% endblock %} +{% block content %} +

Edytuj zbiórkę

+
+ +
+ + +
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+
+ + +
+ +
+ + + + + +{% endblock %} diff --git a/templates/admin/edytuj_stan.html b/templates/admin/edytuj_stan.html new file mode 100644 index 0000000..d0165ef --- /dev/null +++ b/templates/admin/edytuj_stan.html @@ -0,0 +1,13 @@ +{% extends 'base.html' %} +{% block title %}Edytuj stan zbiórki{% endblock %} +{% block content %} +

Edytuj stan zbiórki: {{ zbiorka.nazwa }}

+
+
+ + +
+ + Powrót +
+{% endblock %} diff --git a/templates/base.html b/templates/base.html new file mode 100644 index 0000000..5d7b702 --- /dev/null +++ b/templates/base.html @@ -0,0 +1,39 @@ + + + + + {% block title %}Aplikacja Zbiórek{% endblock %} + + + + + + + +
+ {% with messages = get_flashed_messages(with_categories=true) %} + {% if messages %} + {% for category, message in messages %} +
{{ message }}
+ {% endfor %} + {% endif %} + {% endwith %} + {% block content %}{% endblock %} +
+ + + diff --git a/templates/index.html b/templates/index.html new file mode 100644 index 0000000..79d0b71 --- /dev/null +++ b/templates/index.html @@ -0,0 +1,14 @@ +{% extends 'base.html' %} +{% block title %}Lista Zbiórek{% endblock %} +{% block content %} +

Lista Zbiórek

+
+ {% for z in zbiorki %} + + {{ z.nazwa }} + + {% else %} +

Brak zbiórek

+ {% endfor %} +
+{% endblock %} diff --git a/templates/login.html b/templates/login.html new file mode 100644 index 0000000..3e9df19 --- /dev/null +++ b/templates/login.html @@ -0,0 +1,17 @@ +{% extends 'base.html' %} +{% block title %}Logowanie{% endblock %} +{% block content %} +

Logowanie

+
+
+ + +
+
+ + +
+ +
+

Nie masz konta? Zarejestruj się

+{% endblock %} diff --git a/templates/register.html b/templates/register.html new file mode 100644 index 0000000..474c89f --- /dev/null +++ b/templates/register.html @@ -0,0 +1,16 @@ +{% extends 'base.html' %} +{% block title %}Rejestracja{% endblock %} +{% block content %} +

Rejestracja

+
+
+ + +
+
+ + +
+ +
+{% endblock %} diff --git a/templates/zbiorka.html b/templates/zbiorka.html new file mode 100644 index 0000000..21f068c --- /dev/null +++ b/templates/zbiorka.html @@ -0,0 +1,60 @@ +{% extends 'base.html' %} +{% block title %}{{ zbiorka.nazwa }}{% endblock %} +{% block content %} +
+
+
+

{{ zbiorka.nazwa }}

+
+
+ +
+ {{ zbiorka.opis | markdown }} +
+
    +
  • Numer konta: {{ zbiorka.numer_konta }}
  • +
  • Numer telefonu BLIK: {{ zbiorka.numer_telefonu_blik }}
  • + {% if not zbiorka.ukryj_kwote %} +
  • Cel zbiórki: {{ zbiorka.cel }} PLN
  • +
  • Stan zbiórki: {{ zbiorka.stan }} PLN
  • + {% endif %} +
+ {% set progress = (zbiorka.stan / zbiorka.cel * 100) if zbiorka.cel > 0 else 0 %} +
Postęp zbiórki
+
+
+ {{ progress|round(2) }}% +
+
+
+ {% if current_user.is_authenticated and current_user.is_admin %} + Dodaj wpłatę + {% endif %} + Powrót do listy zbiórek +
+
+
+ +
+
+

Wpłaty

+
+
+ {% if zbiorka.wplaty|length > 0 %} +
    + {% for w in zbiorka.wplaty %} +
  • + {{ w.data.strftime('%Y-%m-%d %H:%M:%S') }}: {{ w.kwota }} PLN + {% if w.opis %} – {{ w.opis }}{% endif %} +
  • + {% endfor %} +
+ {% else %} +

Brak wpłat

+ {% endif %} +
+
+
+{% endblock %}