Construct table of "shipments"

This commit is contained in:
Oliver 2021-10-26 00:17:17 +11:00
parent e9e4d13541
commit e7c25126a4
5 changed files with 170 additions and 7 deletions

View File

@ -810,7 +810,7 @@ order_api_urls = [
])), ])),
url(r'^shipment/', include([ url(r'^shipment/', include([
url(r'^(?P<pk>\d)+/', include([ url(r'^(?P<pk>\d+)/', include([
url(r'^.*$', SOShipmentDetail.as_view(), name='api-so-shipment-detail'), url(r'^.*$', SOShipmentDetail.as_view(), name='api-so-shipment-detail'),
])), ])),
url(r'^.*$', SOShipmentList.as_view(), name='api-so-shipment-list'), url(r'^.*$', SOShipmentList.as_view(), name='api-so-shipment-list'),

View File

@ -41,6 +41,9 @@ def add_shipment(apps, schema_editor):
order=order, order=order,
) )
if order.status == SalesOrderStatus.SHIPPED:
shipment.shipment_date = order.shipment_date
shipment.save() shipment.save()
# Iterate through each allocation associated with this order # Iterate through each allocation associated with this order

View File

@ -33,6 +33,32 @@
</div> </div>
</div> </div>
<div class='panel panel-default panel-inventree panel-hidden' id='panel-order-shipments'>
{% if order.is_pending %}
<div class='panel-heading'>
<h4>{% trans "Pending Shipments" %}</h4>
</div>
<div class='panel-content'>
{% if roles.sales_order.change %}
<div id='pending-shipment-toolbar' class='btn-group' style='float: right;'>
<div class='btn-group'>
<button type='button' class='btn btn-success' id='new-shipment'>
<span class='fas fa-plus-circle'></span> {% trans "New Shipment" %}
</button>
</div>
</div>
{% endif %}
<table class='table table-striped table-condensed' id='pending-shipments-table' data-toolbar='#pending-shipment-toolbar'></table>
</div>
{% endif %}
<div class='panel-heading'>
<h4>{% trans "Completed Shipments" %}</h4>
</div>
<div class='panel-content'>
<table class='table table-striped table-condensed' id='completed-shipments-table'></table>
</div>
</div>
<div class='panel panel-default panel-inventree panel-hidden' id='panel-order-builds'> <div class='panel panel-default panel-inventree panel-hidden' id='panel-order-builds'>
<div class='panel-heading'> <div class='panel-heading'>
<h4>{% trans "Build Orders" %}</h4> <h4>{% trans "Build Orders" %}</h4>
@ -79,6 +105,23 @@
{% block js_ready %} {% block js_ready %}
{{ block.super }} {{ block.super }}
// Callback when the "shipments" panel is first loaded
onPanelLoad('order-shipments', function() {
{% if order.is_pending %}
loadSalesOrderShipmentTable('#pending-shipments-table', {
order: {{ order.pk }},
shipped: false,
});
{% endif %}
loadSalesOrderShipmentTable('#completed-shipments-table', {
order: {{ order.pk }},
shipped: true,
});
});
$('#edit-notes').click(function() { $('#edit-notes').click(function() {
constructForm('{% url "api-so-detail" order.pk %}', { constructForm('{% url "api-so-detail" order.pk %}', {
fields: { fields: {

View File

@ -16,6 +16,13 @@
</a> </a>
</li> </li>
<li class='list-group-item' title='{% trans "Shipments" %}'>
<a href='#' id='select-order-shipments' class='nav-toggle'>
<span class='fas fa-truck sidebar-icon'></span>
{% trans "Shipments" %}
</a>
</li>
<li class='list-group-item' title='{% trans "Build Orders" %}'> <li class='list-group-item' title='{% trans "Build Orders" %}'>
<a href='#' id='select-order-builds' class='nav-toggle'> <a href='#' id='select-order-builds' class='nav-toggle'>
<span class='fas fa-tools sidebar-icon'></span> <span class='fas fa-tools sidebar-icon'></span>

View File

@ -26,6 +26,7 @@
loadPurchaseOrderTable, loadPurchaseOrderTable,
loadSalesOrderAllocationTable, loadSalesOrderAllocationTable,
loadSalesOrderLineItemTable, loadSalesOrderLineItemTable,
loadSalesOrderShipmentTable,
loadSalesOrderTable, loadSalesOrderTable,
newPurchaseOrderFromOrderWizard, newPurchaseOrderFromOrderWizard,
newSupplierPartFromOrderWizard, newSupplierPartFromOrderWizard,
@ -1100,6 +1101,117 @@ function loadSalesOrderTable(table, options) {
} }
/*
* Load a table displaying Shipment information against a particular order
*/
function loadSalesOrderShipmentTable(table, options={}) {
options.table = table;
options.params = options.params || {};
// Filter by order
options.params.order = options.order;
// Filter by "shipped" status
options.params.shipped = options.shipped || false;
var filters = loadTableFilters('salesordershipment');
for (var key in options.params) {
filters[key] = options.params[key];
}
var todo = "Setup filter list for this table";
function makeShipmentActions(row) {
// Construct "actions" for the given shipment row
var pk = row.pk;
var html = `<div class='btn-group float-right' role='group'>`;
html += makeIconButton('fa-edit icon-blue', 'button-shipment-edit', pk, '{% trans "Edit shipment" %}');
html += `</div>`;
return html;
}
function setupShipmentCallbacks() {
// Setup action button callbacks
$(table).find('.button-shipment-edit').click(function() {
var pk = $(this).attr('pk');
constructForm(`/api/order/so/shipment/${pk}/`, {
fields: {
reference: {},
},
title: '{% trans "Edit Shipment" %}',
onSuccess: function() {
$(table).bootstrapTable('refresh');
}
});
});
}
$(table).inventreeTable({
url: '{% url "api-so-shipment-list" %}',
queryParams: filters,
original: options.params,
name: options.name || 'salesordershipment',
search: false,
paginationVAlign: 'bottom',
showColumns: true,
detailView: true,
detailViewByClick: false,
detailFilter: function(index, row) {
return row.allocations.length > 0;
},
detailFormatter: function(index, row, element) {
return showAllocationSubTable(index, row, element, options);
},
onPostBody: setupShipmentCallbacks,
formatNoMatches: function() {
return '{% trans "No matching shipments found" %}';
},
columns: [
{
visible: false,
checkbox: true,
switchable: false,
},
{
field: 'reference',
title: '{% trans "Reference" %}',
switchable: false,
},
{
field: 'shipment_date',
title: '{% trans "Shipment Date" %}',
visible: options.shipped,
switchable: false,
},
{
field: 'notes',
title: '{% trans "Notes" %}',
visible: false,
// TODO: Implement 'notes' field
},
{
title: '',
switchable: false,
formatter: function(value, row) {
return makeShipmentActions(row);
}
}
],
});
}
function loadSalesOrderAllocationTable(table, options={}) { function loadSalesOrderAllocationTable(table, options={}) {
/** /**
* Load a table with SalesOrderAllocation items * Load a table with SalesOrderAllocation items
@ -1123,7 +1235,7 @@ function loadSalesOrderAllocationTable(table, options={}) {
$(table).inventreeTable({ $(table).inventreeTable({
url: '{% url "api-so-allocation-list" %}', url: '{% url "api-so-allocation-list" %}',
queryParams: filters, queryParams: filters,
name: 'salesorderallocation', name: options.name || 'salesorderallocation',
groupBy: false, groupBy: false,
search: false, search: false,
paginationVAlign: 'bottom', paginationVAlign: 'bottom',
@ -1198,16 +1310,14 @@ function showAllocationSubTable(index, row, element, options) {
// Construct a sub-table element // Construct a sub-table element
var html = ` var html = `
<div class='sub-table'> <div class='sub-table'>
<table class='table table-striped table-condensed' id='allocation-table-${row.pk}'> <table class='table table-striped table-condensed' id='allocation-table-${row.pk}'></table>
</table>
</div>`; </div>`;
element.html(html); element.html(html);
var table = $(`#allocation-table-${row.pk}`); var table = $(`#allocation-table-${row.pk}`);
// Is the parent SalesOrder pending? var shipped = options.shipped;
var pending = options.status == {{ SalesOrderStatus.PENDING }};
function setupCallbacks() { function setupCallbacks() {
// Add callbacks for 'edit' buttons // Add callbacks for 'edit' buttons
@ -1300,7 +1410,7 @@ function showAllocationSubTable(index, row, element, options) {
var html = `<div class='btn-group float-right' role='group'>`; var html = `<div class='btn-group float-right' role='group'>`;
var pk = row.pk; var pk = row.pk;
if (pending) { if (!shipped) {
html += makeIconButton('fa-edit icon-blue', 'button-allocation-edit', pk, '{% trans "Edit stock allocation" %}'); html += makeIconButton('fa-edit icon-blue', 'button-allocation-edit', pk, '{% trans "Edit stock allocation" %}');
html += makeIconButton('fa-trash-alt icon-red', 'button-allocation-delete', pk, '{% trans "Delete stock allocation" %}'); html += makeIconButton('fa-trash-alt icon-red', 'button-allocation-delete', pk, '{% trans "Delete stock allocation" %}');
} }