diff --git a/.gitignore b/.gitignore
index 3b614d0..4939baa 100644
--- a/.gitignore
+++ b/.gitignore
@@ -5,6 +5,7 @@
__pycache__
db.sqlite3
media
+/data
# Backup files #
*.bak
diff --git a/benchcoach/settings.py b/benchcoach/settings.py
index 7333e1f..df7fdd6 100644
--- a/benchcoach/settings.py
+++ b/benchcoach/settings.py
@@ -26,8 +26,7 @@ SECRET_KEY = 'django-insecure-qib_j&47o$5l3*gi7y#8#3pjh_88sfdqn@dmp&gx+2)&1nzmor
# SECURITY WARNING: don't run with debug turned on in production!
DEBUG = True
-ALLOWED_HOSTS = ["smithers-ii.local", "127.0.0.1"]
-
+ALLOWED_HOSTS = ["smithers-ii.local", "127.0.0.1", "10.0.1.4", "benchcoach.ascorrea.com"]
# Application definition
@@ -36,6 +35,7 @@ INSTALLED_APPS = [
'teams.apps.TeamsConfig',
'venues.apps.VenuesConfig',
'players.apps.PlayersConfig',
+ 'lineups.apps.LineupsConfig',
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
diff --git a/benchcoach/urls.py b/benchcoach/urls.py
index 7ef45f9..d7b5795 100644
--- a/benchcoach/urls.py
+++ b/benchcoach/urls.py
@@ -26,5 +26,6 @@ urlpatterns = [
path('events/', include('events.urls')),
path('teams/', include('teams.urls')),
path('venues/', include('venues.urls')),
- path('players/', include('players.urls'))
+ path('players/', include('players.urls')),
+ path('lineups/', include('lineups.urls'))
] + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
diff --git a/events/migrations/0003_delete_positioning.py b/events/migrations/0003_delete_positioning.py
new file mode 100644
index 0000000..9f2e51c
--- /dev/null
+++ b/events/migrations/0003_delete_positioning.py
@@ -0,0 +1,16 @@
+# Generated by Django 3.2.6 on 2021-11-11 03:14
+
+from django.db import migrations
+
+
+class Migration(migrations.Migration):
+
+ dependencies = [
+ ('events', '0002_alter_availability_options'),
+ ]
+
+ operations = [
+ migrations.DeleteModel(
+ name='Positioning',
+ ),
+ ]
diff --git a/events/models.py b/events/models.py
index 858d5e0..c474a33 100644
--- a/events/models.py
+++ b/events/models.py
@@ -30,27 +30,5 @@ class Availability(models.Model):
unique_together = ('event', 'player',)
verbose_name_plural = "availabilities"
-class Positioning(models.Model):
- player = models.ForeignKey(Player, on_delete=models.CASCADE)
- event = models.ForeignKey(Event, on_delete=models.CASCADE)
- positions = [
- ('P', 'P'),
- ('C', 'C'),
- ('1B', '1B'),
- ('2B', '2B'),
- ('3B', '3B'),
- ('SS', 'SS'),
- ('LF', 'LF'),
- ('CF', 'CF'),
- ('RF', 'RF'),
- ('DH','DH'),
- ('EH','EH')
- ]
- position = models.CharField(choices=positions, default=None, max_length=2, null=True)
- order = models.IntegerField(default=None, null=True)
-
- class Meta:
- unique_together = ('player', 'event',)
-
class Season(models.Model):
name = models.CharField(max_length=50)
\ No newline at end of file
diff --git a/events/templates/events/schedule.html b/events/templates/events/schedule.html
index f71dd16..295dc52 100644
--- a/events/templates/events/schedule.html
+++ b/events/templates/events/schedule.html
@@ -1,19 +1,14 @@
-
-{% extends 'base.html' %}
-
-{% block title %} {{ title }} {% endblock %}
-
-{% block content %}
+{% extends 'base.html' %}{% block title %} {{ title }} {% endblock %}{% block content %}
{{ title }}
{% for event in events %}
- -
- {{ event.away_team.name }} vs. {{ event.home_team.name }}
+ -
+ {{ event.away_team.name }} vs. {{ event.home_team.name }}
{{ event.start|date:"l, F j, Y g:i A" }}
{{ event.venue.name }}
Edit Event Details
- Edit Lineup
+ Edit Lineup
{% endfor %}
diff --git a/events/views.py b/events/views.py
index f10f598..092ad15 100644
--- a/events/views.py
+++ b/events/views.py
@@ -25,7 +25,7 @@ def edit(request, id=0):
# ...
# redirect to a new URL:
new_event, did_create = Event.objects.update_or_create(pk=id, defaults=form.cleaned_data)
- return render(request, 'success.html', {'call_back':'players list'})
+ return render(request, 'success.html', {'call_back':'schedule'})
# if a GET (or any other method) we'll create a blank form
else:
diff --git a/lineups/__init__.py b/lineups/__init__.py
new file mode 100644
index 0000000..e69de29
diff --git a/lineups/admin.py b/lineups/admin.py
new file mode 100644
index 0000000..8c38f3f
--- /dev/null
+++ b/lineups/admin.py
@@ -0,0 +1,3 @@
+from django.contrib import admin
+
+# Register your models here.
diff --git a/lineups/apps.py b/lineups/apps.py
new file mode 100644
index 0000000..757f8dd
--- /dev/null
+++ b/lineups/apps.py
@@ -0,0 +1,6 @@
+from django.apps import AppConfig
+
+
+class LineupsConfig(AppConfig):
+ default_auto_field = 'django.db.models.BigAutoField'
+ name = 'lineups'
diff --git a/lineups/forms.py b/lineups/forms.py
new file mode 100644
index 0000000..33f59f6
--- /dev/null
+++ b/lineups/forms.py
@@ -0,0 +1,31 @@
+from django import forms
+from .models import Positioning
+from events.models import Event
+from players.models import Player
+from django.forms import modelformset_factory, inlineformset_factory, NumberInput
+from crispy_forms.helper import FormHelper, Layout
+
+class PositioningForm(forms.ModelForm):
+ class Meta:
+ model = Positioning
+ widgets = {
+ 'order': forms.NumberInput(attrs={'class':'input-group-text w-25'}),
+ 'player': forms.Select(attrs={'class': 'form-control'}),
+ 'position': forms.Select(attrs={'class': 'input-group-text w-25'})
+ }
+ exclude = ()
+
+PositioningFormSet = modelformset_factory(
+ model=Positioning,
+ form=PositioningForm,
+ fields = ['player', 'position', 'order'],
+ min_num=9
+)
+
+# class PositioningFormSet(modelformset_factory):
+# class Meta:
+# model = Positioning
+# fields = ['player', 'position', 'order']
+# widgets = {
+# 'order':forms.NumberInput(attrs={'style':'width:6ch'})
+# }
diff --git a/lineups/migrations/0001_initial.py b/lineups/migrations/0001_initial.py
new file mode 100644
index 0000000..33385f1
--- /dev/null
+++ b/lineups/migrations/0001_initial.py
@@ -0,0 +1,30 @@
+# Generated by Django 3.2.6 on 2021-11-11 03:14
+
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+ initial = True
+
+ dependencies = [
+ ('events', '0003_delete_positioning'),
+ ('players', '0003_player_team'),
+ ]
+
+ operations = [
+ migrations.CreateModel(
+ name='Positioning',
+ fields=[
+ ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
+ ('position', models.CharField(choices=[('P', 'P'), ('C', 'C'), ('1B', '1B'), ('2B', '2B'), ('3B', '3B'), ('SS', 'SS'), ('LF', 'LF'), ('CF', 'CF'), ('RF', 'RF'), ('DH', 'DH'), ('EH', 'EH')], default=None, max_length=2, null=True)),
+ ('order', models.IntegerField(default=None, null=True)),
+ ('event', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='events.event')),
+ ('player', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='players.player')),
+ ],
+ options={
+ 'unique_together': {('player', 'event')},
+ },
+ ),
+ ]
diff --git a/lineups/migrations/__init__.py b/lineups/migrations/__init__.py
new file mode 100644
index 0000000..e69de29
diff --git a/lineups/models.py b/lineups/models.py
new file mode 100644
index 0000000..36c0938
--- /dev/null
+++ b/lineups/models.py
@@ -0,0 +1,26 @@
+from django.db import models
+from players.models import Player
+from events.models import Event
+# Create your models here.
+
+class Positioning(models.Model):
+ player = models.ForeignKey(Player, on_delete=models.CASCADE)
+ event = models.ForeignKey(Event, on_delete=models.CASCADE)
+ positions = [
+ ('P', 'P'),
+ ('C', 'C'),
+ ('1B', '1B'),
+ ('2B', '2B'),
+ ('3B', '3B'),
+ ('SS', 'SS'),
+ ('LF', 'LF'),
+ ('CF', 'CF'),
+ ('RF', 'RF'),
+ ('DH','DH'),
+ ('EH','EH')
+ ]
+ position = models.CharField(choices=positions, default=None, max_length=2, null=True)
+ order = models.IntegerField(default=None, null=True)
+
+ class Meta:
+ unique_together = ('player', 'event',)
\ No newline at end of file
diff --git a/lineups/templates/lineups/lineup.html b/lineups/templates/lineups/lineup.html
new file mode 100644
index 0000000..51215f0
--- /dev/null
+++ b/lineups/templates/lineups/lineup.html
@@ -0,0 +1,38 @@
+{% extends 'base.html' %}{% block title %} {{ title }} {% endblock %}{% load crispy_forms_tags %}
+
+{% block content %}
+ {{ title }}
+ {{ event.away_team.name }} vs. {{ event.home_team.name }}
+ {{ event.start|date:"l, F j, Y g:i A" }}
+ {{ event.venue.name }}
+
+
+
+
+
+
+ {% for player in players %}
+ - {{ player.first_name }} {{ player.last_name }}
+ {% endfor %}
+
+
+
+
+
+{% endblock %}
\ No newline at end of file
diff --git a/lineups/tests.py b/lineups/tests.py
new file mode 100644
index 0000000..7ce503c
--- /dev/null
+++ b/lineups/tests.py
@@ -0,0 +1,3 @@
+from django.test import TestCase
+
+# Create your tests here.
diff --git a/lineups/urls.py b/lineups/urls.py
new file mode 100644
index 0000000..051c4a7
--- /dev/null
+++ b/lineups/urls.py
@@ -0,0 +1,9 @@
+from django.contrib import admin
+
+from django.urls import path, include
+
+from . import views
+
+urlpatterns = [
+ path('edit/', views.edit, name="edit lineup"),
+]
\ No newline at end of file
diff --git a/lineups/views.py b/lineups/views.py
new file mode 100644
index 0000000..4740967
--- /dev/null
+++ b/lineups/views.py
@@ -0,0 +1,34 @@
+from django.shortcuts import render, redirect, get_object_or_404
+from django.forms import formset_factory
+from .models import Positioning
+from .forms import PositioningFormSet
+from django.http import HttpResponse
+from django import forms
+from events.models import Event
+from players.models import Player
+
+# Create your views here.
+def edit(request, id):
+
+ if request.method == 'POST':
+ # create a form instance and populate it with data from the request:
+ formset = PositioningFormSet(request.POST)
+ for form in formset:
+ if form.is_valid():
+ # process the data in form.cleaned_data as required
+ # ...
+ # redirect to a new URL:
+ form.cleaned_data.pop('id') #FIXME this is a workaround, not sure why it is necessary
+ new_positioning, did_create = Positioning.objects.update_or_create(id=form['id'].data, defaults=form.cleaned_data)
+ # return render(request, 'success.html', {'call_back':'schedule'})
+ event = Event.objects.get(id=id)
+ players = Player.objects.all()
+ qset = Positioning.objects.filter(event_id=id, order__isnull = False)
+ formset = PositioningFormSet(queryset=qset)
+ for form in formset:
+ for field in form.fields:
+ field
+ return render(request, 'lineups/lineup.html', {'title': 'Lineup',
+ 'event': event,
+ 'players': players,
+ 'positionings_formset':formset})
\ No newline at end of file
diff --git a/players/fixtures/sample_players.yaml b/players/fixtures/sample_players.yaml
index 305ccb9..66890b8 100644
--- a/players/fixtures/sample_players.yaml
+++ b/players/fixtures/sample_players.yaml
@@ -5,7 +5,6 @@
last_name: Tosser
jersey_number: 1
team: 1
- team: 1
- model: players.player
pk: 2
fields:
diff --git a/players/views.py b/players/views.py
index 6dd0e0f..6dbee85 100644
--- a/players/views.py
+++ b/players/views.py
@@ -10,7 +10,12 @@ def root(request):
def list(request):
players = Player.objects.all()
return render(request, 'list.html', {'title': "Players",
- 'items': [(player.id, f"{player.first_name} {player.last_name}") for player in players],
+ 'items': [
+ {'id':player.id,
+ 'title':f"{player.first_name} {player.last_name}",
+ 'subtitle':f"{player.jersey_number}"
+ }
+ for player in players],
'edit_url_name': 'edit player'})
def edit(request, id=0):
diff --git a/teams/views.py b/teams/views.py
index c114ebf..3476936 100644
--- a/teams/views.py
+++ b/teams/views.py
@@ -8,7 +8,13 @@ def root(request):
def list(request):
teams = Team.objects.all()
- return render(request, 'list.html', {'title': "Teams", 'items': [(team.id, f"{team.name}") for team in teams], 'edit_url_name':'edit team'})
+ return render(request, 'list.html', {'title': "Players",
+ 'items': [
+ {'id':team.id,
+ 'title':f"{team.name}"
+ }
+ for team in teams],
+ 'edit_url_name': 'edit team'})
def edit(request, id=0):
# if this is a POST request we need to process the form data
diff --git a/templates/base.html b/templates/base.html
index 2052de3..8455c04 100644
--- a/templates/base.html
+++ b/templates/base.html
@@ -14,7 +14,8 @@
diff --git a/templates/list.html b/templates/list.html
index 9a98336..d5fa3e9 100644
--- a/templates/list.html
+++ b/templates/list.html
@@ -7,8 +7,9 @@
{% for item in items %}
-
-
{{ item.1 }}
- Edit
+ {{ item.title }} {{ item.subtitle }}
+
+ Edit
{% endfor %}
diff --git a/venues/views.py b/venues/views.py
index 9e09cb6..4458eb3 100644
--- a/venues/views.py
+++ b/venues/views.py
@@ -1,5 +1,5 @@
from django.shortcuts import render, redirect, get_object_or_404
-from django.http import HttpResponse
+from django.http import HttpResponse, HttpResponseBadRequest
from .models import Venue
from .forms import VenueForm
@@ -8,7 +8,13 @@ def root(request):
def list(request):
venues = Venue.objects.all()
- return render(request, 'list.html', {'title': "Venues", 'items': [(venue.id, f"{venue.name}") for venue in venues], 'edit_url_name': 'edit venue'})
+ return render(request, 'list.html', {'title': "Venues",
+ 'items': [
+ {'id':venue.id,
+ 'title':f"{venue.name}"
+ }
+ for venue in venues],
+ 'edit_url_name': 'edit venue'})
def edit(request, id=0):
@@ -26,7 +32,8 @@ def edit(request, id=0):
# ...
# redirect to a new URL:
new_venue, did_create = Venue.objects.update_or_create(pk=id, defaults=form.cleaned_data)
- return render(request, 'success.html', {'call_back':'players list'})
+ return render(request, 'success.html', {'call_back':'players list','id':new_venue.id}, status=201 if did_create else 200)
+ return HttpResponseBadRequest()
# if a GET (or any other method) we'll create a blank form
else: