add past and future availabilities to gamecard
This commit is contained in:
@@ -276,6 +276,7 @@ INSTALLED_APPS += [
|
|||||||
"teamsnap",
|
"teamsnap",
|
||||||
"instagen",
|
"instagen",
|
||||||
"gamecard",
|
"gamecard",
|
||||||
|
"gamechanger",
|
||||||
"teamsnap.lineup",
|
"teamsnap.lineup",
|
||||||
"teamsnap.dashboard",
|
"teamsnap.dashboard",
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
{% load static %}
|
{% load static %}
|
||||||
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.8.2/font/bootstrap-icons.css">
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
@@ -101,39 +102,85 @@
|
|||||||
<th class="numbercell"></th>
|
<th class="numbercell"></th>
|
||||||
<th class="condensedNameCell">Available</th>
|
<th class="condensedNameCell">Available</th>
|
||||||
<th class="statscell">AVG/OBP/SLG:PA</th>
|
<th class="statscell">AVG/OBP/SLG:PA</th>
|
||||||
<th class="numbercell"></th>
|
<th class="numbercell">P</th>
|
||||||
<th class="numbercell"></th>
|
<th class="numbercell">C</th>
|
||||||
<th class="numbercell"></th>
|
<th class="numbercell">IF</th>
|
||||||
<th class="numbercell"></th>
|
<th class="numbercell">OF</th>
|
||||||
<th class="numbercell"></th>
|
{% for event in events_future %}
|
||||||
<th class="numbercell"></th>
|
<th class="numbercell">
|
||||||
<th class="numbercell"></th>
|
<span style="transform: rotate(-90deg);display: inline-block">
|
||||||
<th class="numbercell"></th>
|
{{ event.data.start_date|date:'D' }}
|
||||||
<th class="numbercell"></th>
|
</span>
|
||||||
<th class="numbercell"></th>
|
</th>
|
||||||
<th class="numbercell"></th>
|
{% endfor %}
|
||||||
<th class="numbercell"></th>
|
{% for event in events_past %}
|
||||||
|
<th class="numbercell">
|
||||||
|
<span style="transform: rotate(-90deg);display: inline-block">
|
||||||
|
{{ event.data.start_date|date:'D' }}
|
||||||
|
</span>
|
||||||
|
</th>
|
||||||
|
{% endfor %}
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for member in members %}
|
{% for member in members %}
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td class="numbercell"></td>
|
<td class="numbercell"></td>
|
||||||
<td class="numbercell available-status-code-{{ member.availability.status_code }}">{{ member.member.jersey_number }}</td>
|
<td class="numbercell available-status-code-{{ member.availability.status_code }}">{{ member.member.jersey_number }}</td>
|
||||||
<td class="condensedNameCell available-status-code-{{ member.availability.status_code }}">{{ member.member.last_name }}</td>
|
<td class="condensedNameCell available-status-code-{{ member.availability.status_code }}">{{ member.member.last_name }}</td>
|
||||||
<td class="statscell"></td>
|
<td class="statscell"></td>
|
||||||
<td class="numbercell"></td>
|
<td class="numbercell">
|
||||||
<td class="numbercell"></td>
|
{% if "P" in member.member.position %}
|
||||||
<td class="numbercell"></td>
|
<i class="bi bi-check-square-fill"></i>
|
||||||
<td class="numbercell"></td>
|
{% else %}
|
||||||
<td class="numbercell"></td>
|
<i class="bi bi-square"></i>
|
||||||
<td class="numbercell"></td>
|
{% endif %}
|
||||||
<td class="numbercell"></td>
|
<td class="numbercell">
|
||||||
<td class="numbercell"></td>
|
{% if "C" in member.member.position %}
|
||||||
<td class="numbercell"></td>
|
<i class="bi bi-check-square-fill"></i>
|
||||||
<td class="numbercell"></td>
|
{% else %}
|
||||||
<td class="numbercell"></td>
|
<i class="bi bi-square"></i>
|
||||||
<td class="numbercell"></td>
|
{% endif %}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell">
|
||||||
|
{% if "IF" in member.member.position or "1B" in member.member.position %}
|
||||||
|
<i class="bi bi-check-square-fill"></i>
|
||||||
|
{% else %}
|
||||||
|
<i class="bi bi-square"></i>
|
||||||
|
{% endif %}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell">
|
||||||
|
{% if "OF" in member.member.position %}
|
||||||
|
<i class="bi bi-check-square-fill"></i>
|
||||||
|
{% else %}
|
||||||
|
<i class="bi bi-square"></i>
|
||||||
|
{% endif %}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_future.0.data.status_code }}">
|
||||||
|
{{ member.availability_future.0.data.status.0 }}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_future.1.data.status_code }}">
|
||||||
|
{{ member.availability_future.1.data.status.0 }}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_future.2.data.status_code }}">
|
||||||
|
{{ member.availability_future.2.data.status.0 }}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_future.3.data.status_code }}">
|
||||||
|
{{ member.availability_future.3.data.status.0 }}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_past.0.data.status_code }}">
|
||||||
|
{{ member.availability_past.0.data.status.0 }}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_past.1.data.status_code }}">
|
||||||
|
{{ member.availability_past.1.data.status.0 }}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_past.2.data.status_code }}">
|
||||||
|
{{ member.availability_past.2.data.status.0 }}
|
||||||
|
</td>
|
||||||
|
<td class="numbercell available-status-code-{{ member.availability_past.3.data.status_code }}">
|
||||||
|
{{ member.availability_past.3.data.status.0 }}
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
@@ -177,9 +224,11 @@
|
|||||||
VS.
|
VS.
|
||||||
</div>
|
</div>
|
||||||
<div class="" width="100%" style="text-align: right">
|
<div class="" width="100%" style="text-align: right">
|
||||||
|
{% if ts_opponent %}
|
||||||
<img src="{{ ts_opponent.logo.url }}"
|
<img src="{{ ts_opponent.logo.url }}"
|
||||||
width="120px"
|
width="120px"
|
||||||
>
|
>
|
||||||
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,9 +1,12 @@
|
|||||||
|
# TODO Remove VCR
|
||||||
|
import vcr
|
||||||
from django.shortcuts import render
|
from django.shortcuts import render
|
||||||
|
|
||||||
from teamsnap.models import Opponent, Team
|
from teamsnap.models import Opponent, Team
|
||||||
from teamsnap.utils import get_teamsnap_client
|
from teamsnap.utils import get_teamsnap_client
|
||||||
|
|
||||||
|
|
||||||
|
@vcr.use_cassette("gamecard/fixtures/gamecard.yaml", record_mode="new_episodes")
|
||||||
def gamecard(request, team_id, event_id):
|
def gamecard(request, team_id, event_id):
|
||||||
import re
|
import re
|
||||||
|
|
||||||
@@ -20,13 +23,28 @@ def gamecard(request, team_id, event_id):
|
|||||||
ts_bulkload = client.bulk_load(
|
ts_bulkload = client.bulk_load(
|
||||||
team_id=team_id,
|
team_id=team_id,
|
||||||
types=[Event, EventLineup, EventLineupEntry, AvailabilitySummary, Member],
|
types=[Event, EventLineup, EventLineupEntry, AvailabilitySummary, Member],
|
||||||
event__id=event_id,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
ts_event = [
|
ts_events = [e for e in ts_bulkload if isinstance(e, Event)]
|
||||||
i for i in ts_bulkload if isinstance(i, Event) and i.data["id"] == event_id
|
ts_events.sort(key=lambda d: d.data.get("start_date"))
|
||||||
][0]
|
ts_event = [e for e in ts_events if e.data["id"] == event_id][0]
|
||||||
ts_availabilities = Availability.search(client, event_id=ts_event.data["id"])
|
ts_events_future = ts_events[ts_events.index(ts_event) + 1 :]
|
||||||
|
ts_events_past = ts_events[: ts_events.index(ts_event)]
|
||||||
|
|
||||||
|
ts_availabilities = Availability.search(client, team_id=team_id)
|
||||||
|
|
||||||
|
ts_availabilities_future = list(
|
||||||
|
filter(
|
||||||
|
lambda a: a.data["event_id"] in [e.data["id"] for e in ts_events_future],
|
||||||
|
ts_availabilities,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
ts_availabilities_past = list(
|
||||||
|
filter(
|
||||||
|
lambda a: a.data["event_id"] in [e.data["id"] for e in ts_events_past],
|
||||||
|
ts_availabilities,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
ts_lineup_entries = EventLineupEntry.search(client, event_id=event_id)
|
ts_lineup_entries = EventLineupEntry.search(client, event_id=event_id)
|
||||||
|
|
||||||
@@ -41,7 +59,7 @@ def gamecard(request, team_id, event_id):
|
|||||||
|
|
||||||
ts_members = [i for i in ts_bulkload if isinstance(i, Member)]
|
ts_members = [i for i in ts_bulkload if isinstance(i, Member)]
|
||||||
# ts_member_lookup = {m.data["id"]: m for m in ts_members}
|
# ts_member_lookup = {m.data["id"]: m for m in ts_members}
|
||||||
ts_availability_lookup = {m.data["member_id"]: m for m in ts_availabilities}
|
ts_availability_lookup = {m.data["id"]: m for m in ts_availabilities}
|
||||||
ts_lineup_entries_lookup = {m.data["member_id"]: m for m in ts_lineup_entries}
|
ts_lineup_entries_lookup = {m.data["member_id"]: m for m in ts_lineup_entries}
|
||||||
|
|
||||||
members = []
|
members = []
|
||||||
@@ -52,11 +70,27 @@ def gamecard(request, team_id, event_id):
|
|||||||
{
|
{
|
||||||
"member": getattr(member, "data"),
|
"member": getattr(member, "data"),
|
||||||
"availability": getattr(
|
"availability": getattr(
|
||||||
ts_availability_lookup.get(member.data["id"], {}), "data", {}
|
ts_availability_lookup.get(
|
||||||
|
f"{member.data['id']}-{event_id}", {}
|
||||||
|
),
|
||||||
|
"data",
|
||||||
|
{},
|
||||||
),
|
),
|
||||||
"lineup_entry": getattr(
|
"lineup_entry": getattr(
|
||||||
ts_lineup_entries_lookup.get(member.data["id"], {}), "data", {}
|
ts_lineup_entries_lookup.get(member.data["id"], {}), "data", {}
|
||||||
),
|
),
|
||||||
|
"availability_future": list(
|
||||||
|
filter(
|
||||||
|
lambda a: a.data.get("member_id") == member.data["id"],
|
||||||
|
ts_availabilities_future,
|
||||||
|
)
|
||||||
|
)[:4],
|
||||||
|
"availability_past": list(
|
||||||
|
filter(
|
||||||
|
lambda a: a.data.get("member_id") == member.data["id"],
|
||||||
|
ts_availabilities_past,
|
||||||
|
)
|
||||||
|
)[:4],
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -100,10 +134,12 @@ def gamecard(request, team_id, event_id):
|
|||||||
|
|
||||||
context = {
|
context = {
|
||||||
"event": ts_event,
|
"event": ts_event,
|
||||||
|
"events_future": ts_events_future[:4],
|
||||||
|
"events_past": list(reversed(ts_events_past))[:4],
|
||||||
"members": members,
|
"members": members,
|
||||||
"members_startinglineup": members_startinglineup,
|
"members_startinglineup": members_startinglineup,
|
||||||
"members_startingpositiononly": members_startingpositiononly,
|
"members_startingpositiononly": members_startingpositiononly,
|
||||||
"ts_team": Team.objects.get(id=team_id),
|
"ts_team": Team.objects.get(id=team_id),
|
||||||
"ts_opponent": Opponent.objects.get(id=ts_event.data["opponent_id"]),
|
"ts_opponent": Opponent.objects.filter(id=ts_event.data["opponent_id"]).first,
|
||||||
}
|
}
|
||||||
return render(request, "gamecard/gamecard.html", context=context)
|
return render(request, "gamecard/gamecard.html", context=context)
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
[flake8]
|
[flake8]
|
||||||
max-line-length = 120
|
max-line-length = 120
|
||||||
exclude = .tox,.git,*/migrations/*,*/static/CACHE/*,docs,node_modules,venv
|
exclude = .tox,.git,*/migrations/*,*/static/CACHE/*,docs,node_modules,venv
|
||||||
ignore = W503
|
ignore = W503, E203
|
||||||
|
|
||||||
[pycodestyle]
|
[pycodestyle]
|
||||||
max-line-length = 120
|
max-line-length = 120
|
||||||
|
|||||||
@@ -8,7 +8,8 @@ from teamsnap.views import get_teamsnap_client
|
|||||||
def dashboard(request, team_id=None):
|
def dashboard(request, team_id=None):
|
||||||
if not team_id:
|
if not team_id:
|
||||||
return redirect(
|
return redirect(
|
||||||
"teamsnap_dashboard", team_id=request.user.preferences.managed_team_id
|
"teamsnap_dashboard",
|
||||||
|
team_id=request.user.teamsnap_preferences.managed_team_id,
|
||||||
)
|
)
|
||||||
|
|
||||||
from pyteamsnap.api import AvailabilitySummary, Event
|
from pyteamsnap.api import AvailabilitySummary, Event
|
||||||
|
|||||||
Reference in New Issue
Block a user