Minimal working web app
This commit is contained in:
parent
a7a50dcada
commit
e542e3fdc0
|
@ -14,8 +14,9 @@ Including another URLconf
|
||||||
2. Add a URL to urlpatterns: path('blog/', include('blog.urls'))
|
2. Add a URL to urlpatterns: path('blog/', include('blog.urls'))
|
||||||
"""
|
"""
|
||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
from django.urls import path
|
from django.urls import include, path
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
|
path('schedule/', include('schedule.urls')),
|
||||||
path('admin/', admin.site.urls),
|
path('admin/', admin.site.urls),
|
||||||
]
|
]
|
||||||
|
|
10
schedule/converters.py
Normal file
10
schedule/converters.py
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
from datetime import date
|
||||||
|
|
||||||
|
class DateConverter:
|
||||||
|
regex = '[0-9]{4}-[0-9]{2}-[0-9]{2}'
|
||||||
|
|
||||||
|
def to_python(self, value):
|
||||||
|
return date.fromisoformat(value)
|
||||||
|
|
||||||
|
def to_url(self, value):
|
||||||
|
return value.isoformat()
|
45
schedule/migrations/0001_initial.py
Normal file
45
schedule/migrations/0001_initial.py
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
# Generated by Django 3.0.5 on 2020-04-09 13:11
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.db.models.deletion
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
initial = True
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Member',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('member_number', models.PositiveIntegerField(blank=True, null=True, unique=True, verbose_name='aoff member number')),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='PickupDate',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('date', models.DateField(verbose_name='date of pickup')),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Timeslot',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('start', models.TimeField(verbose_name='start of timeslot')),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Pickup',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('date', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='schedule.PickupDate')),
|
||||||
|
('member', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='schedule.Member')),
|
||||||
|
('time', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='schedule.Timeslot')),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
]
|
27
schedule/migrations/0002_auto_20200411_1621.py
Normal file
27
schedule/migrations/0002_auto_20200411_1621.py
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
# Generated by Django 3.0.5 on 2020-04-11 16:21
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('schedule', '0001_initial'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='timeslot',
|
||||||
|
options={'ordering': ['start']},
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='member',
|
||||||
|
name='member_number',
|
||||||
|
field=models.PositiveIntegerField(blank=True, null=True, verbose_name='aoff member number'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='pickupdate',
|
||||||
|
name='date',
|
||||||
|
field=models.DateField(unique=True, verbose_name='date of pickup'),
|
||||||
|
),
|
||||||
|
]
|
|
@ -3,16 +3,24 @@ from django.db import models
|
||||||
# Create your models here.
|
# Create your models here.
|
||||||
|
|
||||||
class PickupDate(models.Model):
|
class PickupDate(models.Model):
|
||||||
date = models.DateField('date of pickup')
|
date = models.DateField('date of pickup', unique=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return str(self.date)
|
||||||
|
|
||||||
class Timeslot(models.Model):
|
class Timeslot(models.Model):
|
||||||
start = models.TimeField('start of timeslot')
|
start = models.TimeField('start of timeslot')
|
||||||
# open-ended for now
|
# open-ended for now
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.start.isoformat(timespec='minutes')
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
ordering = ['start']
|
||||||
|
|
||||||
class Member(models.Model):
|
class Member(models.Model):
|
||||||
member_number = models.PositiveIntegerField(
|
member_number = models.PositiveIntegerField(
|
||||||
verbose_name='aoff member number',
|
verbose_name='aoff member number',
|
||||||
unique=True,
|
|
||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
|
|
37
schedule/templates/schedule/schedule.html
Normal file
37
schedule/templates/schedule/schedule.html
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<title>AOFF</title>
|
||||||
|
|
||||||
|
<h1>Afhentningstidsplan</h1>
|
||||||
|
<p>For at minimere kødannelse bedes medlemmer der skal afhente grøntsager om at vælge hvornår de har tænkt sig at afhente sine grøntsager.
|
||||||
|
Det er hverken bindende eller et krav, men blot en hjælp til vores medlemmer så I kan planlægge afhentning når der er få der skal hente.</p>
|
||||||
|
<p>Det foregår anonymt - andre kan ikke se hvem der henter hvornår.
|
||||||
|
Vi gemmer en såkaldt 'cookie' i din browser med det eneste formål så hvis du kommer tilbage til siden senere i samme browser kan du opdatere eller fjerne dit valgte afhentningstidspunkt.
|
||||||
|
Bemærk at du <em>ikke</em> kan ændre afhentningstidspunkt fra en anden browser eller hvis cookien bliver fjernet.
|
||||||
|
Du kan skrive dit medlemsnummer for at hjælpe administratorer i at rette dine afhentningstidspunkter, men det er <em>ikke</em> et krav.</p>
|
||||||
|
|
||||||
|
<h2>Afhentning {{ date }}</h2>
|
||||||
|
<form action="{% url 'schedule:schedule' date.date %}" method="POST">
|
||||||
|
{% csrf_token %}
|
||||||
|
<label for="member_number">Medlemsnummer (ikke obligatorisk)</label>
|
||||||
|
<input min="0" type="number" placeholder="Mit medlemsnummer" name="member_number" id="member_number" />
|
||||||
|
|
||||||
|
<table>
|
||||||
|
<thead><tr>
|
||||||
|
<th>Tidspunkt</th>
|
||||||
|
<th>Registrerede afhentninger</th>
|
||||||
|
<th></th> {# Action #}
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
{% for timeslot in timeslots %}
|
||||||
|
<tr>
|
||||||
|
<td>{{ timeslot.start | time:"H:i" }}</td>
|
||||||
|
<td>{{ timeslot.sum }} {% if timeslot.own_pickup %}(inklusiv dig){% endif %}</td>
|
||||||
|
<td>{% if timeslot.own_pickup %}
|
||||||
|
<button name="timeslot" value="{{ timeslot.pk }}" formaction="{% url 'schedule:submit_pickup_removal' date.date %}">Fravælg</button>
|
||||||
|
{% else %}
|
||||||
|
<button name="timeslot" value="{{ timeslot.pk }}" formaction="{% url 'schedule:submit_pickup' date.date %}">Vælg</button>
|
||||||
|
{% endif %}
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</table>
|
||||||
|
</form>
|
12
schedule/urls.py
Normal file
12
schedule/urls.py
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
from django.urls import path, register_converter
|
||||||
|
|
||||||
|
from . import views, converters
|
||||||
|
|
||||||
|
register_converter(converters.DateConverter, 'date')
|
||||||
|
|
||||||
|
app_name = 'schedule'
|
||||||
|
urlpatterns = [
|
||||||
|
path('<date:date>/', views.schedule, name='schedule'),
|
||||||
|
path('<date:date>/submit', views.submit_pickup, name='submit_pickup'),
|
||||||
|
path('<date:date>/remove', views.submit_pickup_removal, name='submit_pickup_removal'),
|
||||||
|
]
|
|
@ -1,3 +1,62 @@
|
||||||
from django.shortcuts import render
|
from django.shortcuts import get_object_or_404, render
|
||||||
|
from .models import *
|
||||||
|
from django.http import Http404, HttpResponseRedirect
|
||||||
|
from django.urls import reverse
|
||||||
|
|
||||||
# Create your views here.
|
def schedule(request, date):
|
||||||
|
date = get_object_or_404(PickupDate, date=date)
|
||||||
|
member_number = request.POST and request.POST.get('member_number') \
|
||||||
|
or None
|
||||||
|
member_id = request.session.get('member_id')
|
||||||
|
if member_id is not None:
|
||||||
|
member = get_object_or_404(Member, pk=member_id)
|
||||||
|
elif member_number:
|
||||||
|
member = Member()
|
||||||
|
request.session['member_id'] = member.id
|
||||||
|
request.session.set_expiry(0)
|
||||||
|
else:
|
||||||
|
member = None
|
||||||
|
if member_number and member.member_number != member_number:
|
||||||
|
member.member_number = member_number
|
||||||
|
member.save()
|
||||||
|
timeslots = Timeslot.objects.all()
|
||||||
|
for timeslot in timeslots:
|
||||||
|
timeslot.sum = len(Pickup.objects.filter(time=timeslot))
|
||||||
|
if member:
|
||||||
|
try:
|
||||||
|
timeslot.own_pickup = \
|
||||||
|
Pickup.objects.filter(
|
||||||
|
date=date,
|
||||||
|
time=timeslot,
|
||||||
|
member=member,
|
||||||
|
).exists()
|
||||||
|
except Pickup.DoesNotExist:
|
||||||
|
timeslot.own_pickup = False
|
||||||
|
else:
|
||||||
|
timeslot.own_pickup = False
|
||||||
|
|
||||||
|
return render(request, 'schedule/schedule.html',
|
||||||
|
{ 'date': date,
|
||||||
|
'timeslots': timeslots,
|
||||||
|
})
|
||||||
|
|
||||||
|
def submit_pickup(request, date):
|
||||||
|
date = get_object_or_404(PickupDate, date=date)
|
||||||
|
request.session.set_expiry(0)
|
||||||
|
member_id = request.session.get('member_id')
|
||||||
|
if not member_id:
|
||||||
|
member = Member(member_number=request.POST.get('member_number'))
|
||||||
|
member.save()
|
||||||
|
request.session['member_id'] = member.id
|
||||||
|
else:
|
||||||
|
member = get_object_or_404(Member, pk=member_id)
|
||||||
|
timeslot = get_object_or_404(Timeslot, pk=request.POST.get('timeslot'))
|
||||||
|
Pickup(date=date, time=timeslot, member=member).save()
|
||||||
|
return HttpResponseRedirect(reverse('schedule:schedule', args=(date.date,)))
|
||||||
|
|
||||||
|
def submit_pickup_removal(request, date):
|
||||||
|
date = get_object_or_404(PickupDate, date=date)
|
||||||
|
member = get_object_or_404(Member, pk=request.session.get('member_id'))
|
||||||
|
timeslot = get_object_or_404(Timeslot, pk=request.POST.get('timeslot'))
|
||||||
|
Pickup.objects.filter(date=date, time=timeslot, member=member).delete()
|
||||||
|
return HttpResponseRedirect(reverse('schedule:schedule', args=(date.date,)))
|
||||||
|
|
Loading…
Reference in a new issue