Merge branch 'master' into training

This commit is contained in:
2021-10-20 20:22:59 +01:00
12 changed files with 61 additions and 37 deletions

View File

@@ -82,7 +82,7 @@ class EventRiskAssessmentList(generic.ListView):
template_name = 'hs_object_list.html' template_name = 'hs_object_list.html'
def get_queryset(self): def get_queryset(self):
return self.model.objects.order_by('reviewed_at').select_related('event') return self.model.objects.exclude(event__status=models.Event.CANCELLED).order_by('reviewed_at').select_related('event')
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super(EventRiskAssessmentList, self).get_context_data(**kwargs) context = super(EventRiskAssessmentList, self).get_context_data(**kwargs)
@@ -187,6 +187,9 @@ class EventChecklistList(generic.ListView):
model = models.EventChecklist model = models.EventChecklist
template_name = 'hs_object_list.html' template_name = 'hs_object_list.html'
def get_queryset(self):
return self.model.objects.exclude(event__status=models.Event.CANCELLED).order_by('reviewed_at').select_related('event')
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super(EventChecklistList, self).get_context_data(**kwargs) context = super(EventChecklistList, self).get_context_data(**kwargs)
context['title'] = 'Event Checklist' context['title'] = 'Event Checklist'
@@ -215,7 +218,7 @@ class HSList(generic.ListView):
template_name = 'hs_list.html' template_name = 'hs_list.html'
def get_queryset(self): def get_queryset(self):
return models.Event.objects.all().order_by('-start_date').select_related('riskassessment').prefetch_related('checklists') return models.Event.objects.all().exclude(status=models.Event.CANCELLED).order_by('-start_date').select_related('riskassessment').prefetch_related('checklists')
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super(HSList, self).get_context_data(**kwargs) context = super(HSList, self).get_context_data(**kwargs)

View File

@@ -370,6 +370,9 @@ class Event(models.Model, RevisionMixin):
@property @property
def vat(self): def vat(self):
# No VAT is owed on internal transfers
if self.internal:
return 0
return Decimal(self.sum_total * self.vat_rate.rate).quantize(Decimal('.01')) return Decimal(self.sum_total * self.vat_rate.rate).quantize(Decimal('.01'))
""" """

View File

@@ -8,7 +8,7 @@
<div class="row"> <div class="row">
<div class="col-12 text-right my-3"> <div class="col-12 text-right my-3">
{% button 'edit' url='ec_edit' pk=object.pk %} {% button 'edit' url='ec_edit' pk=object.pk %}
{% button 'view' url='event_detail' pk=object.pk text="Event" %} {% button 'view' url='event_detail' pk=object.event.pk text="Event" %}
{% include 'partials/review_status.html' with perm=perms.RIGS.review_eventchecklist review='ec_review' %} {% include 'partials/review_status.html' with perm=perms.RIGS.review_eventchecklist review='ec_review' %}
</div> </div>
</div> </div>

View File

@@ -190,19 +190,21 @@
{% endif %} {% endif %}
</para> </para>
</td> </td>
<td>£ {{ item.cost|floatformat:2 }}</td> <td>£{{ item.cost|floatformat:2 }}</td>
<td>{{ item.quantity }}</td> <td>{{ item.quantity }}</td>
<td>£ {{ item.total_cost|floatformat:2 }}</td> <td>£{{ item.total_cost|floatformat:2 }}</td>
</tr> </tr>
{% endfor %} {% endfor %}
</blockTable> </blockTable>
<keepTogether> <keepTogether>
<blockTable style="totalTable" colWidths="300,115,80"> <blockTable style="totalTable" colWidths="300,115,80">
{% if object.vat > 0 %}
<tr> <tr>
<td>{% if quote %}VAT Registration Number: 170734807{% endif %}</td> <td>{% if quote %}VAT Registration Number: 170734807</td>
<td>Total (ex. VAT)</td> <td>Total (ex. VAT){% endif %}</td>
<td>£ {{ object.sum_total|floatformat:2 }}</td> <td>£ {{ object.sum_total|floatformat:2 }}</td>
</tr> </tr>
{% endif %}
<tr> <tr>
<td> <td>
{% if quote %} {% if quote %}
@@ -211,8 +213,10 @@
</para> </para>
{% endif %} {% endif %}
</td> </td>
{% if object.vat > 0 %}
<td>VAT @ {{ object.vat_rate.as_percent|floatformat:2 }}%</td> <td>VAT @ {{ object.vat_rate.as_percent|floatformat:2 }}%</td>
<td>£ {{ object.vat|floatformat:2 }}</td> <td>£{{ object.vat|floatformat:2 }}</td>
{% endif %}
</tr> </tr>
<tr> <tr>
<td> <td>
@@ -224,7 +228,7 @@
</td> </td>
{% if invoice %} {% if invoice %}
<td>Total</td> <td>Total</td>
<td>£ {{ object.total|floatformat:2 }}</td> <td>£{{ object.total|floatformat:2 }}</td>
{% else %} {% else %}
<td> <td>
<para> <para>
@@ -233,7 +237,7 @@
</td> </td>
<td> <td>
<para> <para>
<b>£ {{ object.total|floatformat:2 }}</b> <b>£{{ object.total|floatformat:2 }}</b>
</para> </para>
</td> </td>
{% endif %} {% endif %}
@@ -267,7 +271,7 @@
<tr> <tr>
<td>{{ payment.get_method_display }}</td> <td>{{ payment.get_method_display }}</td>
<td>{{ payment.date }}</td> <td>{{ payment.date }}</td>
<td>£ {{ payment.amount|floatformat:2 }}</td> <td>£{{ payment.amount|floatformat:2 }}</td>
</tr> </tr>
{% endfor %} {% endfor %}
</blockTable> </blockTable>
@@ -275,18 +279,18 @@
<tr> <tr>
<td></td> <td></td>
<td>Payment Total</td> <td>Payment Total</td>
<td>£ {{ object.invoice.payment_total|floatformat:2 }}</td> <td>£{{ object.invoice.payment_total|floatformat:2 }}</td>
</tr> </tr>
<tr> <tr>
<td></td> <td></td>
<td> <td>
<para> <para>
<b>Balance</b> (ex. VAT) <b>Balance</b> {% if object.vat > 0 %}(ex. VAT){% endif %}
</para> </para>
</td> </td>
<td> <td>
<para> <para>
<b>£ {{ object.invoice.balance|floatformat:2 }}</b> <b>£{{ object.invoice.balance|floatformat:2 }}</b>
</para> </para>
</td> </td>
</tr> </tr>
@@ -316,7 +320,7 @@
<tr> <tr>
<td>General Enquires and 24 Hour Emergency Contact: 0115 84 68720</td> <td>General Enquires and 24 Hour Emergency Contact: 0115 84 68720</td>
</tr> </tr>
{% else %} {% elif object.vat > 0 %}
<tr> <tr>
<td> <td>
<para>VAT Registration Number: 170734807</para> <para>VAT Registration Number: 170734807</para>

View File

@@ -4,10 +4,11 @@
{% block content %} {% block content %}
<div class="table-responsive"> <div class="table-responsive">
<table class="table mb-0"> <table class="table mb-0 table-sm">
<thead> <thead>
<tr> <tr>
<th scope="col">Event</th> <th scope="col">Event</th>
<th scope="col">MIC</th>
<th scope="col">Dates</th> <th scope="col">Dates</th>
<th scope="col">RA</th> <th scope="col">RA</th>
<th scope="col">Checklists</th> <th scope="col">Checklists</th>
@@ -16,7 +17,8 @@
<tbody> <tbody>
{% for event in object_list %} {% for event in object_list %}
<tr id="event_row"> <tr id="event_row">
<th scope="row" id="event_number"><a href="{% url 'event_detail' event.pk %}">{{ event }}</a></th> <th scope="row" id="event_number"><a href="{% url 'event_detail' event.pk %}">{{ event }}</a><br><small>{{ event.get_status_display }}</small></th>
<td>{% if event.mic is not None %}<a href="{% url 'profile_detail' event.mic.pk %}">{% else %}<span class="text-danger">{% endif %}{{ event.mic }}{% if event.mic is not None %}</a>{% else %}</span>{%endif%}</td>
<!--Dates--> <!--Dates-->
<td id="event_dates"> <td id="event_dates">
<span><strong>{{ event.start_date|date:"D d/m/Y" }}</strong></span> <span><strong>{{ event.start_date|date:"D d/m/Y" }}</strong></span>

View File

@@ -15,7 +15,7 @@
<div class="row"> <div class="row">
<div class="col-12"> <div class="col-12">
<div class="table-responsive"> <div class="table-responsive">
<table class="table mb-0"> <table class="table mb-0 table-sm">
<thead> <thead>
<tr> <tr>
<th scope="col">Event</th> <th scope="col">Event</th>
@@ -32,7 +32,7 @@
{% for object in object_list %} {% for object in object_list %}
<tr class="{% if object.reviewed_by %}table-success{%endif%}"> <tr class="{% if object.reviewed_by %}table-success{%endif%}">
{# General #} {# General #}
<th scope="row"><a href="{% url 'event_detail' object.event.pk %}">{{ object.event }}</a></th> <th scope="row"><a href="{% url 'event_detail' object.event.pk %}">{{ object.event }}</a><br><small>{{ object.event.get_status_display }}</small></th>
{% for field in object_list.0.fieldz %} {% for field in object_list.0.fieldz %}
<td>{{ object|get_field:field }}</td> <td>{{ object|get_field:field }}</td>
{% endfor %} {% endfor %}

View File

@@ -6,11 +6,11 @@
</div> </div>
</th> </th>
{% if perms.RIGS.view_event %} {% if perms.RIGS.view_event %}
<td>£&nbsp;<span class="cost">{{item.cost|floatformat:2}}</span></td> <td>£<span class="cost">{{item.cost|floatformat:2}}</span></td>
{% endif %} {% endif %}
<td class="quantity">{{item.quantity}}</td> <td class="quantity">{{item.quantity}}</td>
{% if perms.RIGS.view_event %} {% if perms.RIGS.view_event %}
<td>£&nbsp;<span class="sub-total" data-subtotal="{{item.total_cost}}">{{item.total_cost|floatformat:2}}</span></td> <td>£<span class="sub-total" data-subtotal="{{item.total_cost}}">{{item.total_cost|floatformat:2}}</span></td>
{% endif %} {% endif %}
{% if edit %} {% if edit %}
<td class="vert-align text-right"> <td class="vert-align text-right">

View File

@@ -27,12 +27,13 @@
{% endfor %} {% endfor %}
</tbody> </tbody>
{% if auth or perms.RIGS.view_event %} {% if auth or perms.RIGS.view_event %}
<tfoot> <tfoot style="font-weight: bold">
<tr> <tr>
<td rowspan="3" colspan="2"></td> <td rowspan="3" colspan="2"></td>
<td>Total (ex. VAT)</td> <td>Total {% if object.vat > 0 or not object.pk %}(ex. VAT){% endif %}</td>
<td colspan="2">£ <span id="sumtotal">{{object.sum_total|default:0|floatformat:2}}</span></td> <td colspan="2">£<span id="sumtotal">{{object.sum_total|default:0|floatformat:2}}</span></td>
</tr> </tr>
{% if object.vat > 0 or not object.pk %}
<tr> <tr>
{% if not object.pk %} {% if not object.pk %}
<td id="vat-rate" data-rate="{{currentVAT.rate}}">VAT @ <td id="vat-rate" data-rate="{{currentVAT.rate}}">VAT @
@@ -41,12 +42,13 @@
<td id="vat-rate" data-rate="{{object.vat_rate.rate}}">VAT @ <td id="vat-rate" data-rate="{{object.vat_rate.rate}}">VAT @
{{object.vat_rate.as_percent|floatformat|default:"TBD"}}%</td> {{object.vat_rate.as_percent|floatformat|default:"TBD"}}%</td>
{% endif %} {% endif %}
<td colspan="2">£ <span id="vat">{{object.vat|default:0|floatformat:2}}</span></td> <td colspan="2">£<span id="vat">{{object.vat|default:0|floatformat:2}}</span></td>
</tr> </tr>
<tr> <tr>
<td>Total</td> <td>Total</td>
<td colspan="2">£ <span id="total">{{object.total|default:0|floatformat:2}}</span></td> <td colspan="2">£<span id="total">{{object.total|default:0|floatformat:2}}</span></td>
</tr> </tr>
{% endif %}
</tfoot> </tfoot>
{% endif %} {% endif %}
</table> </table>
@@ -59,9 +61,9 @@
<em class="description"></em> <em class="description"></em>
</div> </div>
</td> </td>
<td>£&nbsp;<span class="cost"></span></td> <td>£<span class="cost"></span></td>
<td class="quantity"></td> <td class="quantity"></td>
<td>£&nbsp;<span class="sub-total"></span></td> <td>£<span class="sub-total"></span></td>
{% if edit %} {% if edit %}
<td class="vert-align text-right"> <td class="vert-align text-right">
<div class="btn-group" role="group" aria-label="Action buttons"> <div class="btn-group" role="group" aria-label="Action buttons">

8
assets/converters.py Normal file
View File

@@ -0,0 +1,8 @@
class AssetIDConverter: # Forces lowercase to uppercase
regex = '[^/]+'
def to_python(self, value):
return str(value).upper()
def to_url(self, value):
return str(value).upper()

View File

@@ -1,22 +1,24 @@
from django.contrib.auth.decorators import login_required from django.contrib.auth.decorators import login_required
from django.urls import path from django.urls import path, register_converter
from django.views.decorators.clickjacking import xframe_options_exempt from django.views.decorators.clickjacking import xframe_options_exempt
from PyRIGS.decorators import has_oembed, permission_required_with_403 from PyRIGS.decorators import has_oembed, permission_required_with_403
from PyRIGS.views import OEmbedView from PyRIGS.views import OEmbedView
from assets import views from . import views, converters
register_converter(converters.AssetIDConverter, 'asset')
urlpatterns = [ urlpatterns = [
path('', login_required(views.AssetList.as_view()), name='asset_index'), path('', login_required(views.AssetList.as_view()), name='asset_index'),
path('asset/list/', login_required(views.AssetList.as_view()), name='asset_list'), path('asset/list/', login_required(views.AssetList.as_view()), name='asset_list'),
path('asset/id/<str:pk>/', has_oembed(oembed_view="asset_oembed")(views.AssetDetail.as_view()), name='asset_detail'), path('asset/id/<asset:pk>/', has_oembed(oembed_view="asset_oembed")(views.AssetDetail.as_view()), name='asset_detail'),
path('asset/create/', permission_required_with_403('assets.add_asset') path('asset/create/', permission_required_with_403('assets.add_asset')
(views.AssetCreate.as_view()), name='asset_create'), (views.AssetCreate.as_view()), name='asset_create'),
path('asset/id/<str:pk>/edit/', permission_required_with_403('assets.change_asset') path('asset/id/<asset:pk>/edit/', permission_required_with_403('assets.change_asset')
(views.AssetEdit.as_view()), name='asset_update'), (views.AssetEdit.as_view()), name='asset_update'),
path('asset/id/<str:pk>/duplicate/', permission_required_with_403('assets.add_asset') path('asset/id/<asset:pk>/duplicate/', permission_required_with_403('assets.add_asset')
(views.AssetDuplicate.as_view()), name='asset_duplicate'), (views.AssetDuplicate.as_view()), name='asset_duplicate'),
path('asset/id/<str:pk>/label', login_required(views.GenerateLabel.as_view()), name='generate_label'), path('asset/id/<asset:pk>/label', login_required(views.GenerateLabel.as_view()), name='generate_label'),
path('cabletype/list/', login_required(views.CableTypeList.as_view()), name='cable_type_list'), path('cabletype/list/', login_required(views.CableTypeList.as_view()), name='cable_type_list'),
path('cabletype/create/', permission_required_with_403('assets.add_cable_type')(views.CableTypeCreate.as_view()), name='cable_type_create'), path('cabletype/create/', permission_required_with_403('assets.add_cable_type')(views.CableTypeCreate.as_view()), name='cable_type_create'),

View File

@@ -48,9 +48,9 @@ class AssetList(LoginRequiredMixin, generic.ListView):
queryset = self.model.objects.all() queryset = self.model.objects.all()
elif len(query_string) >= 3: elif len(query_string) >= 3:
queryset = self.model.objects.filter( queryset = self.model.objects.filter(
Q(asset_id__exact=query_string) | Q(description__icontains=query_string) | Q(serial_number__exact=query_string)) Q(asset_id__exact=query_string.upper()) | Q(description__icontains=query_string) | Q(serial_number__exact=query_string))
else: else:
queryset = self.model.objects.filter(Q(asset_id__exact=query_string)) queryset = self.model.objects.filter(Q(asset_id__exact=query_string.upper()))
if form.cleaned_data['category']: if form.cleaned_data['category']:
queryset = queryset.filter(category__in=form.cleaned_data['category']) queryset = queryset.filter(category__in=form.cleaned_data['category'])

View File

@@ -1,7 +1,7 @@
<div class="modal-dialog modal-lg"> <div class="modal-dialog modal-lg">
<div class="modal-content"> <div class="modal-content">
<div class="modal-header"> <div class="modal-header">
<h4 class="modal-title">{{page_title}}{% block title %}{% endblock %}</h4> <h4 class="modal-title">{{page_title|safe}}{% block title %}{% endblock %}</h4>
<button type="button" class="close" data-dismiss="modal" aria-label="Close"> <button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">&times;</span> <span aria-hidden="true">&times;</span>
</button> </button>