mirror of
https://github.com/nottinghamtec/PyRIGS.git
synced 2026-02-08 07:59:42 +00:00
Compare commits
11 Commits
revert-594
...
6747567bc1
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6747567bc1 | ||
| 3afa4deb52 | |||
|
|
9f910af7eb | ||
|
|
6c32db3998 | ||
|
|
c2ef469d5d | ||
|
|
e5c7e24941 | ||
| be7b595edb | |||
| 6d53df0c8b | |||
| 732a6e5c1e | |||
|
|
c6823bb9ac | ||
|
|
ec000beee8 |
5
.github/workflows/django.yml
vendored
5
.github/workflows/django.yml
vendored
@@ -15,10 +15,13 @@ jobs:
|
|||||||
PYTHONDONTWRITEBYTECODE: 1
|
PYTHONDONTWRITEBYTECODE: 1
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
- name: Install build dependencies
|
||||||
|
run: |
|
||||||
|
sudo apt-get install libcairo2-dev
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: 3.9
|
python-version: "3.10"
|
||||||
cache: 'pipenv'
|
cache: 'pipenv'
|
||||||
- name: Install Dependencies
|
- name: Install Dependencies
|
||||||
run: |
|
run: |
|
||||||
|
|||||||
2
Pipfile
2
Pipfile
@@ -47,7 +47,7 @@ python-dateutil = "~=2.8.1"
|
|||||||
pytoml = "~=0.1.21"
|
pytoml = "~=0.1.21"
|
||||||
pytz = "~=2020.5"
|
pytz = "~=2020.5"
|
||||||
reportlab = "*"
|
reportlab = "*"
|
||||||
requests = "~=2.32.0"
|
requests = "~=2.32.3"
|
||||||
retrying = "~=1.3.3"
|
retrying = "~=1.3.3"
|
||||||
simplejson = "~=3.17.2"
|
simplejson = "~=3.17.2"
|
||||||
six = "~=1.15.0"
|
six = "~=1.15.0"
|
||||||
|
|||||||
1465
Pipfile.lock
generated
1465
Pipfile.lock
generated
File diff suppressed because it is too large
Load Diff
@@ -224,6 +224,8 @@ USE_L10N = True
|
|||||||
|
|
||||||
USE_TZ = True
|
USE_TZ = True
|
||||||
|
|
||||||
|
USE_THOUSAND_SEPARATOR = False
|
||||||
|
|
||||||
# Need to allow seconds as datetime-local input type spits out a time that has seconds
|
# Need to allow seconds as datetime-local input type spits out a time that has seconds
|
||||||
DATETIME_INPUT_FORMATS = ('%Y-%m-%dT%H:%M', '%Y-%m-%dT%H:%M:%S')
|
DATETIME_INPUT_FORMATS = ('%Y-%m-%dT%H:%M', '%Y-%m-%dT%H:%M:%S')
|
||||||
|
|
||||||
|
|||||||
@@ -154,9 +154,9 @@ class AssociateAdmin(VersionAdmin):
|
|||||||
|
|
||||||
@admin.register(models.Profile)
|
@admin.register(models.Profile)
|
||||||
class ProfileAdmin(UserAdmin, AssociateAdmin):
|
class ProfileAdmin(UserAdmin, AssociateAdmin):
|
||||||
list_display = ('username', 'name', 'is_approved', 'is_superuser', 'is_supervisor', 'number_of_events', 'last_login')
|
list_display = ('username', 'name', 'is_approved', 'is_superuser', 'is_supervisor', 'number_of_events', 'last_login', 'date_joined')
|
||||||
list_display_links = ['username']
|
list_display_links = ['username']
|
||||||
list_filter = UserAdmin.list_filter + ('is_approved',)
|
list_filter = UserAdmin.list_filter + ('is_approved', 'date_joined')
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
(None, {'fields': ('username', 'password')}),
|
(None, {'fields': ('username', 'password')}),
|
||||||
(_('Personal info'), {
|
(_('Personal info'), {
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
from datetime import datetime
|
from datetime import datetime, timedelta
|
||||||
|
|
||||||
import simplejson
|
import simplejson
|
||||||
from django import forms
|
from django import forms
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core import serializers
|
from django.core import serializers
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
from django.utils.html import format_html
|
||||||
from reversion import revisions as reversion
|
from reversion import revisions as reversion
|
||||||
|
|
||||||
from RIGS import models
|
from RIGS import models
|
||||||
@@ -97,6 +98,9 @@ class EventForm(forms.ModelForm):
|
|||||||
raise forms.ValidationError(
|
raise forms.ValidationError(
|
||||||
'You haven\'t provided any client contact details. Please add a person or organisation.',
|
'You haven\'t provided any client contact details. Please add a person or organisation.',
|
||||||
code='contact')
|
code='contact')
|
||||||
|
access = self.cleaned_data.get("access_at")
|
||||||
|
if 'warn-access' not in self.data and access is not None and access.date() < (self.cleaned_data.get("start_date") - timedelta(days=7)):
|
||||||
|
raise forms.ValidationError(format_html("Are you sure about that? Your access time seems a bit optimistic. If you're sure, save again. <input type='hidden' id='warn-access' name='warn-access' value='0'/>"), code='access_sanity')
|
||||||
return super().clean()
|
return super().clean()
|
||||||
|
|
||||||
def save(self, commit=True):
|
def save(self, commit=True):
|
||||||
|
|||||||
@@ -254,7 +254,7 @@ class Command(BaseCommand):
|
|||||||
new_invoice.void = True
|
new_invoice.void = True
|
||||||
elif random.randint(0, 2) > 1: # 1 in 3 have been paid
|
elif random.randint(0, 2) > 1: # 1 in 3 have been paid
|
||||||
models.Payment.objects.create(invoice=new_invoice, amount=new_invoice.balance,
|
models.Payment.objects.create(invoice=new_invoice, amount=new_invoice.balance,
|
||||||
date=datetime.date.today())
|
date=datetime.date.today(), method=random.choice(models.Payment.METHODS)[0])
|
||||||
if i == 1 or random.randint(0, 5) > 0: # Event 1 and 1 in 5 have a RA
|
if i == 1 or random.randint(0, 5) > 0: # Event 1 and 1 in 5 have a RA
|
||||||
models.RiskAssessment.objects.create(event=new_event, supervisor_consulted=bool(random.getrandbits(1)),
|
models.RiskAssessment.objects.create(event=new_event, supervisor_consulted=bool(random.getrandbits(1)),
|
||||||
nonstandard_equipment=bool(random.getrandbits(1)),
|
nonstandard_equipment=bool(random.getrandbits(1)),
|
||||||
|
|||||||
@@ -45,6 +45,7 @@
|
|||||||
Invoices <span class="badge {% if todo == 0 %}badge-success{% else %}badge-danger{% endif %} badge-pill">{{ todo }}</span>
|
Invoices <span class="badge {% if todo == 0 %}badge-success{% else %}badge-danger{% endif %} badge-pill">{{ todo }}</span>
|
||||||
</a>
|
</a>
|
||||||
<div class="dropdown-menu" aria-labelledby="navbarDropdownInvoices">
|
<div class="dropdown-menu" aria-labelledby="navbarDropdownInvoices">
|
||||||
|
<a class="dropdown-item" href="{% url 'invoice_dashboard' %}"><span class="fas fa-chart-line"></span> Dashboard</a>
|
||||||
{% if perms.RIGS.add_invoice %}
|
{% if perms.RIGS.add_invoice %}
|
||||||
<a class="dropdown-item text-nowrap" href="{% url 'invoice_waiting' %}"><span class="fas fa-briefcase text-danger"></span> Waiting <span class="badge {% if waiting == 0 %}badge-success{% else %}badge-danger{% endif %} badge-pill">{{ waiting }}</span></a>
|
<a class="dropdown-item text-nowrap" href="{% url 'invoice_waiting' %}"><span class="fas fa-briefcase text-danger"></span> Waiting <span class="badge {% if waiting == 0 %}badge-success{% else %}badge-danger{% endif %} badge-pill">{{ waiting }}</span></a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|||||||
106
RIGS/templates/invoice_dashboard.html
Normal file
106
RIGS/templates/invoice_dashboard.html
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
{% extends 'base_rigs.html' %}
|
||||||
|
|
||||||
|
{% load humanize %}
|
||||||
|
{% block content %}
|
||||||
|
|
||||||
|
<form method="GET" action="{% url 'invoice_dashboard' %}">
|
||||||
|
<div class="form-row">
|
||||||
|
<div class="form-group col-md-4">
|
||||||
|
<label for="time_filter">Time Filter</label>
|
||||||
|
<select id="time_filter" name="time_filter" class="form-control">
|
||||||
|
<option value="week" {% if time_filter == 'week' %}selected{% endif %}>Last Week (7 days)</option>
|
||||||
|
<option value="month" {% if time_filter == 'month' %}selected{% endif %}>Last Month (30 days)</option>
|
||||||
|
<option value="year" {% if time_filter == 'year' %}selected{% endif %}>Last Year</option>
|
||||||
|
<option value="all" {% if time_filter == 'all' %}selected{% endif %}>All Time</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
$('#time_filter').change(function () {
|
||||||
|
$(this).closest('form').submit();
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<h3>Overview</h3>
|
||||||
|
|
||||||
|
<!-- big cards in 2x2 grid with total_outstanding, total_events, total_invoices and total_payments, different backgrounds -->
|
||||||
|
|
||||||
|
<div class="card-deck">
|
||||||
|
<div class="card">
|
||||||
|
<a href="{% url 'invoice_waiting' %}" class="text-decoration-none text-white">
|
||||||
|
<div class="card-body bg-primary">
|
||||||
|
<h5 class="card-title text-center">Total Waiting</h5>
|
||||||
|
<p class="card-text text-center h3"><strong>£{{ total_waiting|floatformat:"2g" }}</strong></p>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="card">
|
||||||
|
<a href="{% url 'invoice_list' %}" class="text-decoration-none text-dark">
|
||||||
|
<div class="card-body bg-info">
|
||||||
|
<h5 class="card-title text-center">Total Outstanding</h5>
|
||||||
|
<p class="card-text text-center h3"><strong>£{{ total_outstanding|floatformat:"2g" }}</strong></p>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body bg-danger">
|
||||||
|
<h5 class="card-title text-center">Total Events</h5>
|
||||||
|
<p class="card-text text-center h3"><strong>{{ total_events }}</strong></p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body bg-success">
|
||||||
|
<h5 class="card-title text-center">Total Invoices</h5>
|
||||||
|
<p class="card-text text-center h3"><strong>{{ total_invoices }}</strong></p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<br />
|
||||||
|
|
||||||
|
<h3>Payments</h3>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
<h4>Sources</h4>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
{% for source in payment_methods %}
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
<h5 class="card-title"><strong>{{ source.method }}</strong></h5>
|
||||||
|
<p class="card-text h3">£{{ source.total|floatformat:"2g" }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
<h4>Total</h4>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
<h5 class="card-title text-center">Total Income</h5>
|
||||||
|
<p class="card-text text-center h3"><strong>£{{ total_income|floatformat:"2g" }}</strong></p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
<h4>Invoice Payment Time</h4>
|
||||||
|
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
<div class="card">
|
||||||
|
<div class="card-body">
|
||||||
|
<h5 class="card-title text-center">Average Time to Pay</h5>
|
||||||
|
<p class="card-text text-center h3"><strong>{{ mean_invoice_to_payment|floatformat:"2g" }} days</strong></p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% endblock %}
|
||||||
@@ -31,7 +31,7 @@
|
|||||||
{% for event in object_list %}
|
{% for event in object_list %}
|
||||||
<tr class="{{event.status_color}}">
|
<tr class="{{event.status_color}}">
|
||||||
<th scope="row"><a href="{% url 'event_detail' event.pk %}">{{ event.display_id }}</a><br>
|
<th scope="row"><a href="{% url 'event_detail' event.pk %}">{{ event.display_id }}</a><br>
|
||||||
<span class="text-muted">{{ event.get_status_display }}</span></th>
|
<span class="{% if event.get_status_display == 'Cancelled' %}text-danger{% endif %}">{{ event.get_status_display }}</span></th>
|
||||||
<td>{{ event.start_date }}</td>
|
<td>{{ event.start_date }}</td>
|
||||||
<td>
|
<td>
|
||||||
{{ event.name }}
|
{{ event.name }}
|
||||||
|
|||||||
@@ -127,7 +127,7 @@ class TestEventCreate(BaseRigboardTest):
|
|||||||
|
|
||||||
# Fix it
|
# Fix it
|
||||||
self.page.end_date = datetime.date(2020, 1, 11)
|
self.page.end_date = datetime.date(2020, 1, 11)
|
||||||
self.page.access_at = datetime.datetime(2020, 1, 1, 9)
|
self.page.access_at = datetime.datetime(2020, 1, 8, 9)
|
||||||
self.page.dry_hire = True
|
self.page.dry_hire = True
|
||||||
self.page.status = "Booked"
|
self.page.status = "Booked"
|
||||||
self.page.collected_by = "Fred"
|
self.page.collected_by = "Fred"
|
||||||
|
|||||||
@@ -115,7 +115,8 @@ urlpatterns = [
|
|||||||
path('event/webhook/', views.RecieveForumWebhook.as_view(), name='webhook_recieve'),
|
path('event/webhook/', views.RecieveForumWebhook.as_view(), name='webhook_recieve'),
|
||||||
|
|
||||||
# Finance
|
# Finance
|
||||||
path('invoice/', permission_required_with_403('RIGS.view_invoice')(views.InvoiceIndex.as_view()),
|
path('invoice/', permission_required_with_403('RIGS.view_invoice')(views.InvoiceDashboard.as_view()), name='invoice_dashboard'),
|
||||||
|
path('invoice/outstanding', permission_required_with_403('RIGS.view_invoice')(views.InvoiceOutstanding.as_view()),
|
||||||
name='invoice_list'),
|
name='invoice_list'),
|
||||||
path('invoice/archive/', permission_required_with_403('RIGS.view_invoice')(views.InvoiceArchive.as_view()),
|
path('invoice/archive/', permission_required_with_403('RIGS.view_invoice')(views.InvoiceArchive.as_view()),
|
||||||
name='invoice_archive'),
|
name='invoice_archive'),
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import reversion
|
|||||||
from django import forms
|
from django import forms
|
||||||
from django.contrib import messages
|
from django.contrib import messages
|
||||||
from django.db import transaction
|
from django.db import transaction
|
||||||
from django.db.models import Q
|
from django.db.models import Sum
|
||||||
from django.http import Http404, HttpResponseRedirect
|
from django.http import Http404, HttpResponseRedirect
|
||||||
from django.http import HttpResponse
|
from django.http import HttpResponse
|
||||||
from django.shortcuts import get_object_or_404
|
from django.shortcuts import get_object_or_404
|
||||||
@@ -18,8 +18,76 @@ from RIGS import models
|
|||||||
|
|
||||||
forms.DateField.widget = forms.DateInput(attrs={'type': 'date'})
|
forms.DateField.widget = forms.DateInput(attrs={'type': 'date'})
|
||||||
|
|
||||||
|
TIME_FILTERS = ["all", "year", "month", "week"]
|
||||||
|
|
||||||
class InvoiceIndex(generic.ListView):
|
|
||||||
|
def days_between(d1, d2):
|
||||||
|
diff = d2 - d1
|
||||||
|
return diff.total_seconds() / datetime.timedelta(days=1).total_seconds()
|
||||||
|
|
||||||
|
|
||||||
|
class InvoiceDashboard(generic.TemplateView):
|
||||||
|
template_name = 'invoice_dashboard.html'
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
context = super().get_context_data(**kwargs)
|
||||||
|
context['page_title'] = "Invoice Dashboard"
|
||||||
|
context['description'] = "Overview of financial status of TEC rigs."
|
||||||
|
|
||||||
|
time_filter = self.request.GET.get('time_filter', 'all')
|
||||||
|
|
||||||
|
if time_filter not in TIME_FILTERS:
|
||||||
|
time_filter = 'all'
|
||||||
|
|
||||||
|
if time_filter == 'all':
|
||||||
|
context['events'] = models.Event.objects.filter(is_rig=True)
|
||||||
|
context['invoices'] = models.Invoice.objects.all()
|
||||||
|
context['payments'] = models.Payment.objects.all()
|
||||||
|
elif time_filter == 'year':
|
||||||
|
context['events'] = models.Event.objects.filter(is_rig=True, start_date__gte=datetime.date.today() - datetime.timedelta(days=365))
|
||||||
|
context['invoices'] = models.Invoice.objects.filter(invoice_date__gte=datetime.date.today() - datetime.timedelta(days=365))
|
||||||
|
context['payments'] = models.Payment.objects.filter(date__gte=datetime.date.today() - datetime.timedelta(days=365))
|
||||||
|
elif time_filter == 'month':
|
||||||
|
context['events'] = models.Event.objects.filter(is_rig=True, start_date__gte=datetime.date.today() - datetime.timedelta(days=30))
|
||||||
|
context['invoices'] = models.Invoice.objects.filter(invoice_date__gte=datetime.date.today() - datetime.timedelta(days=30))
|
||||||
|
context['payments'] = models.Payment.objects.filter(date__gte=datetime.date.today() - datetime.timedelta(days=30))
|
||||||
|
elif time_filter == 'week':
|
||||||
|
context['events'] = models.Event.objects.filter(is_rig=True, start_date__gte=datetime.date.today() - datetime.timedelta(days=7))
|
||||||
|
context['invoices'] = models.Invoice.objects.filter(invoice_date__gte=datetime.date.today() - datetime.timedelta(days=7))
|
||||||
|
context['payments'] = models.Payment.objects.filter(date__gte=datetime.date.today() - datetime.timedelta(days=7))
|
||||||
|
|
||||||
|
context["time_filter"] = time_filter
|
||||||
|
|
||||||
|
context['total_outstanding'] = sum([i.balance for i in models.Invoice.objects.outstanding_invoices()])
|
||||||
|
context['total_waiting'] = sum([i.sum_total for i in models.Event.objects.waiting_invoices()])
|
||||||
|
context['total_events'] = len(context['events'])
|
||||||
|
context['total_invoices'] = len(context['invoices'])
|
||||||
|
context['total_payments'] = len(context['payments'])
|
||||||
|
|
||||||
|
payment_methods = dict(models.Payment.METHODS)
|
||||||
|
|
||||||
|
context['payment_methods'] = context["payments"].values('method').annotate(total=Sum('amount')).order_by('method')
|
||||||
|
|
||||||
|
for method in context['payment_methods']:
|
||||||
|
method['method'] = payment_methods.get(method['method'], f"Unknown method ({method['method']})")
|
||||||
|
|
||||||
|
context["total_income"] = sum([i['total'] for i in context['payment_methods']])
|
||||||
|
|
||||||
|
payments = context['payments']
|
||||||
|
mean_duration = 0
|
||||||
|
|
||||||
|
for payment in payments:
|
||||||
|
mean_duration += days_between(payment.invoice.invoice_date, payment.date)
|
||||||
|
|
||||||
|
if len(payments) > 0:
|
||||||
|
mean_duration /= len(payments)
|
||||||
|
|
||||||
|
context['mean_invoice_to_payment'] = mean_duration
|
||||||
|
|
||||||
|
return context
|
||||||
|
|
||||||
|
|
||||||
|
class InvoiceOutstanding(generic.ListView):
|
||||||
model = models.Invoice
|
model = models.Invoice
|
||||||
template_name = 'invoice_list.html'
|
template_name = 'invoice_list.html'
|
||||||
|
|
||||||
|
|||||||
@@ -374,7 +374,7 @@ def generate_label(pk):
|
|||||||
barcode = Code39(str(obj.asset_id), writer=ImageWriter())
|
barcode = Code39(str(obj.asset_id), writer=ImageWriter())
|
||||||
|
|
||||||
logo_size = (200, 200)
|
logo_size = (200, 200)
|
||||||
image.paste(logo.resize(logo_size, Image.ANTIALIAS), box=(5, 5))
|
image.paste(logo.resize(logo_size, Image.LANCZOS), box=(5, 5))
|
||||||
barcode_image = barcode.render(writer_options={"quiet_zone": 0, "write_text": False})
|
barcode_image = barcode.render(writer_options={"quiet_zone": 0, "write_text": False})
|
||||||
width, height = barcode_image.size
|
width, height = barcode_image.size
|
||||||
image.paste(barcode_image.crop((0, 0, width, 100)), (int(((size[0] + logo_size[0]) - width) / 2), 40))
|
image.paste(barcode_image.crop((0, 0, width, 100)), (int(((size[0] + logo_size[0]) - width) / 2), 40))
|
||||||
|
|||||||
@@ -9,3 +9,4 @@ $theme-colors: (
|
|||||||
"primary": #3A52A2
|
"primary": #3A52A2
|
||||||
) !default;
|
) !default;
|
||||||
$enable-shadows: true;
|
$enable-shadows: true;
|
||||||
|
$alert-color-level: 10;
|
||||||
|
|||||||
@@ -35,8 +35,8 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
<nav class="navbar navbar-expand-lg navbar-dark bg-dark" role="navigation">
|
<nav class="navbar navbar-expand-lg navbar-dark bg-dark" role="navigation">
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<a class="navbar-brand" style="position: absolute; left:0.5em; top: 2px;" href="{% if request.user.is_authenticated %}https://rigs.nottinghamtec.co.uk{%else%}https://nottinghamtec.co.uk{%endif%}">
|
<a class="navbar-brand" href="{% if request.user.is_authenticated %}https://rigs.nottinghamtec.co.uk{%else%}https://nottinghamtec.co.uk{%endif%}">
|
||||||
<img src="{% static 'imgs/logo.webp' %}" width="40" height="40" alt="TEC's Logo: Serif 'TEC' vertically next to a blue box with the words 'PA and Lighting', surrounded by graduated rings" id="logo">
|
<img src="{% static 'imgs/logo.webp' %}" class="mr-auto" style="max-height: 40px; position: absolute; left: 0.5em; top: 0;" alt="TEC's Logo: Serif 'TEC' vertically next to a blue box with the words 'PA and Lighting', surrounded by graduated rings" id="logo">
|
||||||
</a>
|
</a>
|
||||||
{% block titleheader %}
|
{% block titleheader %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|||||||
@@ -9,9 +9,11 @@
|
|||||||
<h1 class="col-sm-12 pb-3">R<small class="text-muted">ig</small> I<small class="text-muted">nformation</small> G<small class="text-muted">athering</small> S<small class="text-muted">ystem</small></h1>
|
<h1 class="col-sm-12 pb-3">R<small class="text-muted">ig</small> I<small class="text-muted">nformation</small> G<small class="text-muted">athering</small> S<small class="text-muted">ystem</small></h1>
|
||||||
<h2 class="col-sm-12 pb-3">Welcome back {{ user.get_full_name }}, there {%if rig_count == 1 %}is one rig coming up{%else%}are {{ rig_count|apnumber }} rigs coming up.{%endif%}</h2>
|
<h2 class="col-sm-12 pb-3">Welcome back {{ user.get_full_name }}, there {%if rig_count == 1 %}is one rig coming up{%else%}are {{ rig_count|apnumber }} rigs coming up.{%endif%}</h2>
|
||||||
{% if now %}
|
{% if now %}
|
||||||
<div class="col-sm-12 alert alert-primary rounded-0 mx-auto">
|
<div class="col-sm-12">
|
||||||
{% for event in now %}
|
{% for event in now %}
|
||||||
Event {{ event }} is happening today! <a href="{% url 'event_checkin' event.pk %}" class="btn btn-success btn-sm modal-href align-baseline {% if request.user.current_event %}disabled{%endif%}"><span class="fas fa-user-clock"></span> <span class="d-none d-sm-inline">Check In</span></a><br/>
|
<div class="alert alert-primary rounded-0">
|
||||||
|
Event <a href="{% url 'event_detail' event.pk %}" class="text-danger">{{ event }}</a> is happening today! <a href="{% url 'event_checkin' event.pk %}" class="btn btn-success btn-sm modal-href align-baseline {% if request.user.current_event %}disabled{%endif%}"><span class="fas fa-user-clock"></span> <span class="d-none d-sm-inline">Check In</span></a><br/>
|
||||||
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|||||||
Reference in New Issue
Block a user