Initial commit: WrestleDesk full project

- Django backend with DRF (clubs, wrestlers, trainers, exercises, templates, trainings, homework, locations, leistungstest)
- Next.js 16 frontend with React, Shadcn UI, Tailwind
- JWT authentication
- Full CRUD for all entities
- Calendar view for trainings
- Homework management system
- Leistungstest tracking
This commit is contained in:
Andrej Spielmann
2026-03-26 13:24:57 +01:00
commit 3fefc550fe
256 changed files with 38295 additions and 0 deletions
@@ -0,0 +1,60 @@
# Generated by Django 4.2.29 on 2026-03-19 09:05
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
initial = True
dependencies = [
('exercises', '0001_initial'),
('wrestlers', '0001_initial'),
('clubs', '0001_initial'),
]
operations = [
migrations.CreateModel(
name='Homework',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('title', models.CharField(max_length=200)),
('description', models.TextField(blank=True)),
('target_group', models.CharField(choices=[('kids', 'Kids'), ('youth', 'Youth'), ('adults', 'Adults'), ('all', 'All')], default='all', max_length=20)),
('due_date', models.DateField(blank=True, null=True)),
('is_active', models.BooleanField(default=True)),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
('club', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='homework_assignments', to='clubs.club')),
('exercises', models.ManyToManyField(related_name='homework_assignments', to='exercises.exercise')),
],
options={
'ordering': ['-created_at'],
},
),
migrations.CreateModel(
name='HomeworkStatus',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('is_completed', models.BooleanField(default=False)),
('completion_date', models.DateField(blank=True, null=True)),
('notes', models.TextField(blank=True)),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
('homework', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='statuses', to='homework.homework')),
('wrestler', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='homework_statuses', to='wrestlers.wrestler')),
],
options={
'unique_together': {('homework', 'wrestler')},
},
),
migrations.AddIndex(
model_name='homework',
index=models.Index(fields=['target_group'], name='homework_ho_target__66652f_idx'),
),
migrations.AddIndex(
model_name='homework',
index=models.Index(fields=['due_date'], name='homework_ho_due_dat_5d3fcb_idx'),
),
]
@@ -0,0 +1,126 @@
# Generated by Django 4.2.29 on 2026-03-19 14:44
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('wrestlers', '0001_initial'),
('clubs', '0001_initial'),
('exercises', '0001_initial'),
('homework', '0001_initial'),
]
operations = [
migrations.CreateModel(
name='HomeworkAssignment',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('due_date', models.DateField(blank=True, null=True)),
('notes', models.TextField(blank=True)),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
],
options={
'ordering': ['-created_at'],
},
),
migrations.CreateModel(
name='HomeworkAssignmentItem',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('is_completed', models.BooleanField(default=False)),
('completion_date', models.DateField(blank=True, null=True)),
],
),
migrations.CreateModel(
name='HomeworkExerciseItem',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('reps', models.IntegerField(blank=True, null=True)),
('time_minutes', models.IntegerField(blank=True, null=True)),
('order', models.IntegerField(default=0)),
],
options={
'ordering': ['homework', 'order'],
},
),
migrations.RemoveIndex(
model_name='homework',
name='homework_ho_target__66652f_idx',
),
migrations.RemoveField(
model_name='homework',
name='exercises',
),
migrations.RemoveField(
model_name='homework',
name='target_group',
),
migrations.AlterField(
model_name='homework',
name='club',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='homework_templates', to='clubs.club'),
),
migrations.AddIndex(
model_name='homework',
index=models.Index(fields=['club'], name='homework_ho_club_id_126bad_idx'),
),
migrations.AddField(
model_name='homeworkexerciseitem',
name='exercise',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='homework_items', to='exercises.exercise'),
),
migrations.AddField(
model_name='homeworkexerciseitem',
name='homework',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='exercise_items', to='homework.homework'),
),
migrations.AddField(
model_name='homeworkassignmentitem',
name='assignment',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='items', to='homework.homeworkassignment'),
),
migrations.AddField(
model_name='homeworkassignmentitem',
name='exercise',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='assignment_items', to='exercises.exercise'),
),
migrations.AddField(
model_name='homeworkassignment',
name='club',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='homework_assignments', to='clubs.club'),
),
migrations.AddField(
model_name='homeworkassignment',
name='homework',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='assignments', to='homework.homework'),
),
migrations.AddField(
model_name='homeworkassignment',
name='wrestler',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='homework_assignments', to='wrestlers.wrestler'),
),
migrations.AlterUniqueTogether(
name='homeworkexerciseitem',
unique_together={('homework', 'exercise')},
),
migrations.AlterUniqueTogether(
name='homeworkassignmentitem',
unique_together={('assignment', 'exercise')},
),
migrations.AddIndex(
model_name='homeworkassignment',
index=models.Index(fields=['wrestler'], name='homework_ho_wrestle_ddebdf_idx'),
),
migrations.AddIndex(
model_name='homeworkassignment',
index=models.Index(fields=['due_date'], name='homework_ho_due_dat_12e964_idx'),
),
migrations.AlterUniqueTogether(
name='homeworkassignment',
unique_together={('homework', 'wrestler')},
),
]
@@ -0,0 +1,21 @@
# Generated by Django 4.2.29 on 2026-03-20 14:29
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('homework', '0002_homeworkassignment_homeworkassignmentitem_and_more'),
]
operations = [
migrations.AddIndex(
model_name='homeworkassignmentitem',
index=models.Index(fields=['assignment', 'is_completed'], name='homework_ho_assignm_791a15_idx'),
),
migrations.AddIndex(
model_name='homeworkassignmentitem',
index=models.Index(fields=['completion_date'], name='homework_ho_complet_55c380_idx'),
),
]
@@ -0,0 +1,23 @@
# Generated by Django 4.2.29 on 2026-03-20 14:40
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('homework', '0003_add_assignment_item_indexes'),
]
operations = [
migrations.AlterField(
model_name='homeworkexerciseitem',
name='reps',
field=models.PositiveIntegerField(blank=True, null=True),
),
migrations.AlterField(
model_name='homeworkexerciseitem',
name='time_minutes',
field=models.PositiveIntegerField(blank=True, null=True),
),
]
@@ -0,0 +1,67 @@
# Generated by Django 4.2.29 on 2026-03-22 12:17
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('wrestlers', '0002_alter_wrestler_license_scan_alter_wrestler_photo'),
('clubs', '0001_initial'),
('trainings', '0005_training_club_and_more'),
('exercises', '0003_alter_exercise_media'),
('homework', '0004_alter_homeworkexerciseitem_reps_and_more'),
]
operations = [
migrations.CreateModel(
name='TrainingHomework',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('created_at', models.DateTimeField(auto_now_add=True)),
('training', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='homework_assignments', to='trainings.training')),
],
options={
'ordering': ['-created_at'],
},
),
migrations.CreateModel(
name='TrainingHomeworkExercise',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('reps', models.PositiveIntegerField(blank=True, null=True)),
('time_minutes', models.PositiveIntegerField(blank=True, null=True)),
('order', models.IntegerField(default=0)),
('exercise', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='training_homework_items', to='exercises.exercise')),
('training_homework', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='exercises', to='homework.traininghomework')),
],
options={
'ordering': ['training_homework', 'order'],
'unique_together': {('training_homework', 'exercise')},
},
),
migrations.CreateModel(
name='TrainingHomeworkAssignment',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('notes', models.TextField(blank=True)),
('is_completed', models.BooleanField(default=False)),
('completion_date', models.DateField(blank=True, null=True)),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
('club', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='training_homework_assignments', to='clubs.club')),
('training_homework', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='assignments', to='homework.traininghomework')),
('wrestler', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='training_homework_assignments', to='wrestlers.wrestler')),
],
options={
'ordering': ['-created_at'],
'indexes': [models.Index(fields=['wrestler'], name='homework_tr_wrestle_63f0d7_idx'), models.Index(fields=['is_completed'], name='homework_tr_is_comp_a157f2_idx'), models.Index(fields=['club'], name='homework_tr_club_id_4648ff_idx')],
'unique_together': {('training_homework', 'wrestler')},
},
),
migrations.AddIndex(
model_name='traininghomework',
index=models.Index(fields=['training'], name='homework_tr_trainin_950ce2_idx'),
),
]
@@ -0,0 +1,66 @@
# Generated by Django 4.2.29 on 2026-03-23 06:45
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('wrestlers', '0002_alter_wrestler_license_scan_alter_wrestler_photo'),
('exercises', '0003_alter_exercise_media'),
('trainings', '0005_training_club_and_more'),
('homework', '0005_traininghomework_traininghomeworkexercise_and_more'),
]
operations = [
migrations.CreateModel(
name='TrainingHomeworkExerciseItem',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('reps', models.PositiveIntegerField(blank=True, null=True)),
('time_minutes', models.PositiveIntegerField(blank=True, null=True)),
('order', models.IntegerField(default=0)),
('is_completed', models.BooleanField(default=False)),
],
options={
'ordering': ['assignment', 'order'],
},
),
migrations.AlterUniqueTogether(
name='traininghomeworkassignment',
unique_together=set(),
),
migrations.AddField(
model_name='traininghomeworkassignment',
name='training',
field=models.ForeignKey(default=1, on_delete=django.db.models.deletion.CASCADE, related_name='homework_assignments', to='trainings.training'),
),
migrations.AlterField(
model_name='traininghomework',
name='training',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='homework_legacy', to='trainings.training'),
),
migrations.AlterUniqueTogether(
name='traininghomeworkassignment',
unique_together={('training', 'wrestler')},
),
migrations.AddIndex(
model_name='traininghomeworkassignment',
index=models.Index(fields=['training'], name='homework_tr_trainin_048980_idx'),
),
migrations.AddField(
model_name='traininghomeworkexerciseitem',
name='assignment',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='exercises', to='homework.traininghomeworkassignment'),
),
migrations.AddField(
model_name='traininghomeworkexerciseitem',
name='exercise',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='training_homework_exercises', to='exercises.exercise'),
),
migrations.RemoveField(
model_name='traininghomeworkassignment',
name='training_homework',
),
]