feat: Add Team-Members management page
- New /team route to view all team members - Add/activate/deactivate team members via UI - Display name, role, email, responsibilities, telegram_id - Integrated into main navigation - Self-learning system info displayed
This commit is contained in:
parent
320a1d4d87
commit
0e02611c87
3 changed files with 226 additions and 0 deletions
63
app.py
63
app.py
|
|
@ -2774,6 +2774,69 @@ def journal_clear():
|
||||||
return redirect(url_for('settings'))
|
return redirect(url_for('settings'))
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/team')
|
||||||
|
def team():
|
||||||
|
"""Zeigt alle Team-Members an."""
|
||||||
|
team_members = get_team_members(active_only=False)
|
||||||
|
return render_template('team.html', team_members=team_members, agents=AGENTS)
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/team/add', methods=['POST'])
|
||||||
|
def team_add():
|
||||||
|
"""Fügt ein neues Team-Mitglied hinzu."""
|
||||||
|
name = request.form.get('name', '').strip()
|
||||||
|
email = request.form.get('email', '').strip()
|
||||||
|
role = request.form.get('role', '').strip()
|
||||||
|
responsibilities = request.form.get('responsibilities', '').strip()
|
||||||
|
telegram_chat_id = request.form.get('telegram_chat_id', '').strip()
|
||||||
|
|
||||||
|
if not name or not email or not role or not responsibilities:
|
||||||
|
flash('Alle Felder außer Telegram Chat ID sind Pflichtfelder!', 'danger')
|
||||||
|
return redirect(url_for('team'))
|
||||||
|
|
||||||
|
# Optional: Telegram Chat ID als Integer
|
||||||
|
chat_id = None
|
||||||
|
if telegram_chat_id:
|
||||||
|
try:
|
||||||
|
chat_id = int(telegram_chat_id)
|
||||||
|
except ValueError:
|
||||||
|
flash('Telegram Chat ID muss eine Zahl sein!', 'warning')
|
||||||
|
|
||||||
|
success = add_team_member(name, role, responsibilities, email, telegram_id=chat_id)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
flash(f'✅ Team-Member "{name}" erfolgreich hinzugefügt!', 'success')
|
||||||
|
logger.info(f"[Team] Neues Mitglied hinzugefügt: {name} ({email})")
|
||||||
|
else:
|
||||||
|
flash(f'❌ Team-Member konnte nicht hinzugefügt werden (Email evtl. bereits vorhanden).', 'danger')
|
||||||
|
|
||||||
|
return redirect(url_for('team'))
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/team/<int:member_id>/activate', methods=['POST'])
|
||||||
|
def team_activate(member_id):
|
||||||
|
"""Aktiviert ein Team-Mitglied."""
|
||||||
|
con = sqlite3.connect(EMAIL_JOURNAL_DB)
|
||||||
|
con.execute("UPDATE team_members SET active = 1 WHERE id = ?", (member_id,))
|
||||||
|
con.commit()
|
||||||
|
con.close()
|
||||||
|
|
||||||
|
logger.info(f"[Team] Member #{member_id} aktiviert")
|
||||||
|
return jsonify({'success': True})
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/team/<int:member_id>/deactivate', methods=['POST'])
|
||||||
|
def team_deactivate(member_id):
|
||||||
|
"""Deaktiviert ein Team-Mitglied."""
|
||||||
|
con = sqlite3.connect(EMAIL_JOURNAL_DB)
|
||||||
|
con.execute("UPDATE team_members SET active = 0 WHERE id = ?", (member_id,))
|
||||||
|
con.commit()
|
||||||
|
con.close()
|
||||||
|
|
||||||
|
logger.info(f"[Team] Member #{member_id} deaktiviert")
|
||||||
|
return jsonify({'success': True})
|
||||||
|
|
||||||
|
|
||||||
@app.route('/api/telegram-qr')
|
@app.route('/api/telegram-qr')
|
||||||
def telegram_qr():
|
def telegram_qr():
|
||||||
"""Generiert QR-Code für Telegram Bot."""
|
"""Generiert QR-Code für Telegram Bot."""
|
||||||
|
|
|
||||||
|
|
@ -64,6 +64,11 @@
|
||||||
<span>📋</span> Log
|
<span>📋</span> Log
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
|
<li class="nav-item">
|
||||||
|
<a class="nav-link {% if request.path == '/team' %}active{% endif %}" href="/team">
|
||||||
|
<span>👥</span> Team
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
<li class="nav-item">
|
<li class="nav-item">
|
||||||
<a class="nav-link {% if request.path == '/settings' %}active{% endif %}" href="/settings">
|
<a class="nav-link {% if request.path == '/settings' %}active{% endif %}" href="/settings">
|
||||||
<span>⚙</span> Settings
|
<span>⚙</span> Settings
|
||||||
|
|
|
||||||
158
templates/team.html
Normal file
158
templates/team.html
Normal file
|
|
@ -0,0 +1,158 @@
|
||||||
|
{% extends "base.html" %}
|
||||||
|
|
||||||
|
{% block title %}Team-Members{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div class="row mb-4">
|
||||||
|
<div class="col">
|
||||||
|
<h1 class="mb-3">
|
||||||
|
<span style="font-size: 2rem;">👥</span> Team-Members
|
||||||
|
</h1>
|
||||||
|
<p class="text-muted">
|
||||||
|
Verwalte die registrierten Team-Mitglieder des Diversity-Ball Wien 2026.
|
||||||
|
Das Self-Learning System lernt automatisch neue Absender kennen.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% if team_members %}
|
||||||
|
<div class="row">
|
||||||
|
{% for member in team_members %}
|
||||||
|
<div class="col-md-6 col-lg-4 mb-4">
|
||||||
|
<div class="card h-100">
|
||||||
|
<div class="card-body">
|
||||||
|
<div class="d-flex justify-content-between align-items-start mb-3">
|
||||||
|
<h5 class="card-title mb-0">{{ member.name }}</h5>
|
||||||
|
{% if member.active %}
|
||||||
|
<span class="badge bg-success">Aktiv</span>
|
||||||
|
{% else %}
|
||||||
|
<span class="badge bg-secondary">Inaktiv</span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h6 class="text-muted mb-3">{{ member.role }}</h6>
|
||||||
|
|
||||||
|
<div class="mb-3">
|
||||||
|
<small class="text-muted d-block mb-1">Email:</small>
|
||||||
|
<a href="mailto:{{ member.email }}" class="text-decoration-none">
|
||||||
|
{{ member.email }}
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-3">
|
||||||
|
<small class="text-muted d-block mb-1">Aufgaben:</small>
|
||||||
|
<p class="small mb-0" style="white-space: pre-line;">{{ member.responsibilities }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% if member.telegram_id %}
|
||||||
|
<div class="mb-2">
|
||||||
|
<small class="text-muted">📱 Telegram: {{ member.telegram_id }}</small>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="text-muted small">
|
||||||
|
<small>Erstellt: {{ member.created_at }}</small>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-footer bg-transparent border-top-0">
|
||||||
|
<div class="d-flex gap-2">
|
||||||
|
<button class="btn btn-sm btn-outline-primary" onclick="editMember({{ member.id }})">
|
||||||
|
✏️ Bearbeiten
|
||||||
|
</button>
|
||||||
|
{% if member.active %}
|
||||||
|
<button class="btn btn-sm btn-outline-warning" onclick="deactivateMember({{ member.id }})">
|
||||||
|
🚫 Deaktivieren
|
||||||
|
</button>
|
||||||
|
{% else %}
|
||||||
|
<button class="btn btn-sm btn-outline-success" onclick="activateMember({{ member.id }})">
|
||||||
|
✅ Aktivieren
|
||||||
|
</button>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="alert alert-info">
|
||||||
|
<strong>ℹ️ Keine Team-Members gefunden</strong><br>
|
||||||
|
Das Self-Learning System wird automatisch Team-Members hinzufügen, wenn neue Absender Emails senden.
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="row mt-4">
|
||||||
|
<div class="col">
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
<h5 class="card-title mb-3">➕ Neues Team-Mitglied hinzufügen</h5>
|
||||||
|
<form method="POST" action="/team/add">
|
||||||
|
<div class="row g-3">
|
||||||
|
<div class="col-md-6">
|
||||||
|
<label class="form-label">Name</label>
|
||||||
|
<input type="text" class="form-control" name="name" required>
|
||||||
|
</div>
|
||||||
|
<div class="col-md-6">
|
||||||
|
<label class="form-label">Email</label>
|
||||||
|
<input type="email" class="form-control" name="email" required>
|
||||||
|
</div>
|
||||||
|
<div class="col-md-6">
|
||||||
|
<label class="form-label">Rolle</label>
|
||||||
|
<input type="text" class="form-control" name="role" required>
|
||||||
|
</div>
|
||||||
|
<div class="col-md-6">
|
||||||
|
<label class="form-label">Telegram Chat ID (optional)</label>
|
||||||
|
<input type="text" class="form-control" name="telegram_chat_id" placeholder="z.B. 123456789">
|
||||||
|
</div>
|
||||||
|
<div class="col-12">
|
||||||
|
<label class="form-label">Aufgaben / Verantwortlichkeiten</label>
|
||||||
|
<textarea class="form-control" name="responsibilities" rows="3" required></textarea>
|
||||||
|
</div>
|
||||||
|
<div class="col-12">
|
||||||
|
<button type="submit" class="btn btn-primary">
|
||||||
|
➕ Team-Member hinzufügen
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
function editMember(id) {
|
||||||
|
alert('Edit-Funktion in Entwicklung - Member ID: ' + id);
|
||||||
|
}
|
||||||
|
|
||||||
|
function deactivateMember(id) {
|
||||||
|
if (confirm('Team-Member wirklich deaktivieren?')) {
|
||||||
|
fetch(`/team/${id}/deactivate`, { method: 'POST' })
|
||||||
|
.then(response => response.json())
|
||||||
|
.then(data => {
|
||||||
|
if (data.success) {
|
||||||
|
location.reload();
|
||||||
|
} else {
|
||||||
|
alert('Fehler: ' + data.error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function activateMember(id) {
|
||||||
|
if (confirm('Team-Member wieder aktivieren?')) {
|
||||||
|
fetch(`/team/${id}/activate`, { method: 'POST' })
|
||||||
|
.then(response => response.json())
|
||||||
|
.then(data => {
|
||||||
|
if (data.success) {
|
||||||
|
location.reload();
|
||||||
|
} else {
|
||||||
|
alert('Fehler: ' + data.error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{% endblock %}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue