commit 1eeea9f83ad9230a5c1f7a75662770eaab0df837 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 21:15:41 2025 +0200 Disable resuming of old time entries. commit 3e3ec2f01cb7943622b819a19179388078ae1315 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 20:59:19 2025 +0200 Refactor db migrations. commit 15a51a569da36c6b7c9e01ab17b6fdbdee6ad994 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 19:58:04 2025 +0200 Apply new style for Time Tracking view. commit 77e5278b303e060d2b03853b06277f8aa567ae68 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 18:06:04 2025 +0200 Allow direct registrations as a Company. commit 188a8772757cbef374243d3a5f29e4440ddecabe Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 18:04:45 2025 +0200 Add email invitation feature. commit d9ebaa02aa01b518960a20dccdd5a327d82f30c6 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 17:12:32 2025 +0200 Apply common style for Company, User, Team management pages. commit 81149caf4d8fc6317e2ab1b4f022b32fc5aa6d22 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 16:44:32 2025 +0200 Move export functions to own module. commit 1a26e19338e73f8849c671471dd15cc3c1b1fe82 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 15:51:15 2025 +0200 Split up models.py. commit 61f1ccd10f721b0ff4dc1eccf30c7a1ee13f204d Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 12:05:28 2025 +0200 Move utility function into own modules. commit 84b341ed35e2c5387819a8b9f9d41eca900ae79f Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 11:44:24 2025 +0200 Refactor auth functions use. commit 923e311e3da5b26d85845c2832b73b7b17c48adb Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 11:35:52 2025 +0200 Refactor route nameing and fix bugs along the way. commit f0a5c4419c340e62a2615c60b2a9de28204d2995 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 10:34:33 2025 +0200 Fix URL endpoints in announcement template. commit b74d74542a1c8dc350749e4788a9464d067a88b5 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 09:25:53 2025 +0200 Move announcements to own module. commit 9563a28021ac46c82c04fe4649b394dbf96f92c7 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 09:16:30 2025 +0200 Combine Company view and edit templates. commit 6687c373e681d54e4deab6b2582fed5cea9aadf6 Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 08:17:42 2025 +0200 Move Users, Company and System Administration to own modules. commit 8b7894a2e3eb84bb059f546648b6b9536fea724e Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 07:40:57 2025 +0200 Move Teams and Projects to own modules. commit d11bf059d99839ecf1f5d7020b8c8c8a2454c00b Author: Jens Luedicke <jens@luedicke.me> Date: Mon Jul 7 07:09:33 2025 +0200 Move Tasks and Sprints to own modules.
129 lines
4.8 KiB
Python
129 lines
4.8 KiB
Python
"""
|
|
Task Management Routes
|
|
Handles all task-related views and operations
|
|
"""
|
|
|
|
from flask import Blueprint, render_template, g, redirect, url_for, flash
|
|
from sqlalchemy import or_
|
|
from models import db, Role, Project, Task, User
|
|
from routes.auth import login_required, role_required, company_required
|
|
|
|
tasks_bp = Blueprint('tasks', __name__, url_prefix='/tasks')
|
|
|
|
|
|
def get_filtered_tasks_for_burndown(user, mode, start_date=None, end_date=None, project_filter=None):
|
|
"""Get filtered tasks for burndown chart"""
|
|
from datetime import datetime, time
|
|
|
|
# Base query - get tasks from user's company
|
|
query = Task.query.join(Project).filter(Project.company_id == user.company_id)
|
|
|
|
# Apply user/team filter
|
|
if mode == 'personal':
|
|
# For personal mode, get tasks assigned to the user or created by them
|
|
query = query.filter(
|
|
(Task.assigned_to_id == user.id) |
|
|
(Task.created_by_id == user.id)
|
|
)
|
|
elif mode == 'team' and user.team_id:
|
|
# For team mode, get tasks from projects assigned to the team
|
|
query = query.filter(Project.team_id == user.team_id)
|
|
|
|
# Apply project filter
|
|
if project_filter:
|
|
if project_filter == 'none':
|
|
# No project filter for tasks - they must belong to a project
|
|
return []
|
|
else:
|
|
try:
|
|
project_id = int(project_filter)
|
|
query = query.filter(Task.project_id == project_id)
|
|
except ValueError:
|
|
pass
|
|
|
|
# Apply date filters - use task creation date and completion date
|
|
if start_date:
|
|
query = query.filter(
|
|
(Task.created_at >= datetime.combine(start_date, time.min)) |
|
|
(Task.completed_date >= start_date)
|
|
)
|
|
if end_date:
|
|
query = query.filter(
|
|
Task.created_at <= datetime.combine(end_date, time.max)
|
|
)
|
|
|
|
return query.order_by(Task.created_at.desc()).all()
|
|
|
|
|
|
@tasks_bp.route('')
|
|
@role_required(Role.TEAM_MEMBER)
|
|
@company_required
|
|
def unified_task_management():
|
|
"""Unified task management interface"""
|
|
|
|
# Get all projects the user has access to (for filtering and task creation)
|
|
if g.user.role in [Role.ADMIN, Role.SUPERVISOR]:
|
|
# Admins and Supervisors can see all company projects
|
|
available_projects = Project.query.filter_by(
|
|
company_id=g.user.company_id,
|
|
is_active=True
|
|
).order_by(Project.name).all()
|
|
elif g.user.team_id:
|
|
# Team members see team projects + unassigned projects
|
|
available_projects = Project.query.filter(
|
|
Project.company_id == g.user.company_id,
|
|
Project.is_active == True,
|
|
or_(Project.team_id == g.user.team_id, Project.team_id == None)
|
|
).order_by(Project.name).all()
|
|
# Filter by actual access permissions
|
|
available_projects = [p for p in available_projects if p.is_user_allowed(g.user)]
|
|
else:
|
|
# Unassigned users see only unassigned projects
|
|
available_projects = Project.query.filter_by(
|
|
company_id=g.user.company_id,
|
|
team_id=None,
|
|
is_active=True
|
|
).order_by(Project.name).all()
|
|
available_projects = [p for p in available_projects if p.is_user_allowed(g.user)]
|
|
|
|
# Get team members for task assignment (company-scoped)
|
|
if g.user.role in [Role.ADMIN, Role.SUPERVISOR]:
|
|
# Admins can assign to anyone in the company
|
|
team_members = User.query.filter_by(
|
|
company_id=g.user.company_id,
|
|
is_blocked=False
|
|
).order_by(User.username).all()
|
|
elif g.user.team_id:
|
|
# Team members can assign to team members + supervisors/admins
|
|
team_members = User.query.filter(
|
|
User.company_id == g.user.company_id,
|
|
User.is_blocked == False,
|
|
or_(
|
|
User.team_id == g.user.team_id,
|
|
User.role.in_([Role.ADMIN, Role.SUPERVISOR])
|
|
)
|
|
).order_by(User.username).all()
|
|
else:
|
|
# Unassigned users can assign to supervisors/admins only
|
|
team_members = User.query.filter(
|
|
User.company_id == g.user.company_id,
|
|
User.is_blocked == False,
|
|
User.role.in_([Role.ADMIN, Role.SUPERVISOR])
|
|
).order_by(User.username).all()
|
|
|
|
# Convert team members to JSON-serializable format
|
|
team_members_data = [{
|
|
'id': member.id,
|
|
'username': member.username,
|
|
'email': member.email,
|
|
'role': member.role.value if member.role else 'Team Member',
|
|
'avatar_url': member.get_avatar_url(32)
|
|
} for member in team_members]
|
|
|
|
return render_template('unified_task_management.html',
|
|
title='Task Management',
|
|
available_projects=available_projects,
|
|
team_members=team_members_data)
|
|
|
|
|