-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'pkv/invoice-form-and-payments' into pkv/staging-deploy
- Loading branch information
Showing
8 changed files
with
309 additions
and
14 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
36 changes: 36 additions & 0 deletions
36
commcare_connect/opportunity/migrations/0057_paymentinvoice_payment_invoice.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
# Generated by Django 4.2.5 on 2024-09-05 08:08 | ||
|
||
from django.db import migrations, models | ||
import django.db.models.deletion | ||
|
||
|
||
class Migration(migrations.Migration): | ||
dependencies = [ | ||
("opportunity", "0056_payment_organization"), | ||
] | ||
|
||
operations = [ | ||
migrations.CreateModel( | ||
name="PaymentInvoice", | ||
fields=[ | ||
("id", models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")), | ||
("amount", models.PositiveIntegerField()), | ||
("date", models.DateField()), | ||
("invoice_number", models.CharField(max_length=50)), | ||
( | ||
"opportunity", | ||
models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="opportunity.opportunity"), | ||
), | ||
], | ||
options={ | ||
"unique_together": {("opportunity", "invoice_number")}, | ||
}, | ||
), | ||
migrations.AddField( | ||
model_name="payment", | ||
name="invoice", | ||
field=models.OneToOneField( | ||
blank=True, null=True, on_delete=django.db.models.deletion.DO_NOTHING, to="opportunity.paymentinvoice" | ||
), | ||
), | ||
] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
103 changes: 103 additions & 0 deletions
103
commcare_connect/templates/opportunity/invoice_list.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,103 @@ | ||
{% extends "opportunity/base.html" %} | ||
{% load django_tables2 %} | ||
{% load static %} | ||
{% load crispy_forms_tags %} | ||
|
||
{% block title %}{{ request.org }} - Invoices{% endblock title %} | ||
|
||
{% block breadcrumbs_inner %} | ||
{{ block.super }} | ||
<li class="breadcrumb-item"> | ||
<a href="{% url 'opportunity:detail' org_slug=request.org.slug pk=opportunity.pk %}"> | ||
{{ opportunity.name }} | ||
</a> | ||
</li> | ||
<li class="breadcrumb-item active" aria-current="page">Invoices</li> | ||
{% endblock %} | ||
|
||
{% block content %} | ||
<h2>Invoices</h2> | ||
<hr /> | ||
{% if not request.org_membership.is_program_manager %} | ||
<button type="button" class="btn btn-primary mb-2" data-bs-toggle="modal" data-bs-target="#invoiceModal"> | ||
Add New Invoice | ||
</button> | ||
{% endif %} | ||
|
||
<form hx-get="{% url "opportunity:invoice_table" org_slug=request.org.slug pk=opportunity.pk %}" | ||
hx-trigger="load, change, newInvoice from:body" | ||
hx-target="#invoiceTable"> | ||
<select class="form-select mb-2" id="filterSelect" name="filter"> | ||
<option selected value="">Filter Invoices</option> | ||
<option value="paid">Paid</option> | ||
<option value="pending">Pending</option> | ||
</select> | ||
</form> | ||
|
||
{% if request.org_membership.is_program_manager %} | ||
<form x-data="{ | ||
selectAll: false, | ||
selected: [], | ||
toggleSelectAll() { | ||
this.selectAll = !this.selectAll; | ||
if(this.selectAll) | ||
this.selected = JSON.parse(document.getElementById('invoice_ids').textContent); | ||
else | ||
this.selected = []; | ||
} | ||
}" | ||
hx-post="{% url "opportunity:invoice_approve" org_slug=request.org.slug pk=opportunity.pk %}" | ||
hx-swap="none" | ||
> | ||
{% csrf_token %} | ||
{% endif %} | ||
|
||
<div id="invoiceTable"> | ||
{% include "tables/table_placeholder.html" with num_cols=4 %} | ||
</div> | ||
|
||
{% if request.org_membership.is_program_manager %} | ||
<button type="submit" class="btn btn-primary mt-2">Pay</button> | ||
</form> | ||
{% endif %} | ||
{% endblock content %} | ||
|
||
{% if not request.org_membership.is_program_manager %} | ||
{% block modal %} | ||
<div class="modal fade" id="invoiceModal" tabindex="-1" role="dialog" aria-labelledby="invoiceModalLabel" aria-hidden="true"> | ||
<div class="modal-dialog" role="document"> | ||
<div class="modal-content"> | ||
<div class="modal-header"> | ||
<h5 class="modal-title" id="invoiceModalLabel">Create Invoice</h5> | ||
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button> | ||
</div> | ||
<form hx-post="{% url "opportunity:invoice_create" org_slug=request.org.slug pk=opportunity.pk %}" | ||
hx-trigger="submit" | ||
hx-headers='{"X-CSRFToken": "{{ csrf_token }}"}' | ||
hx-target="#formFields"> | ||
<div class="modal-body"> | ||
<div id="formFields">{% crispy form %}</div> | ||
</div> | ||
<div class="modal-footer"> | ||
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Close</button> | ||
<button type="submit" class="btn btn-primary">Save changes</button> | ||
</div> | ||
</form> | ||
</div> | ||
</div> | ||
</div> | ||
{% endblock modal %} | ||
{% endif %} | ||
|
||
{% block inline_javascript %} | ||
<script> | ||
window.addEventListener('DOMContentLoaded', () => { | ||
const filter = document.querySelector("#filterSelect"); | ||
filter.addEventListener("change", (event) => { | ||
const url = new URL(window.location); | ||
url.searchParams.set("filter", event.target.value) | ||
history.pushState(null, '', url); | ||
}); | ||
}); | ||
</script> | ||
{% endblock inline_javascript %} |
Oops, something went wrong.