mirror of
https://github.com/inventree/InvenTree
synced 2024-08-30 18:33:04 +00:00
creating new tab
This commit is contained in:
parent
b803fbae72
commit
7c18ebbbe4
@ -69,6 +69,12 @@
|
|||||||
</li>
|
</li>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if part.purchaseable and roles.purchase_order.view %}
|
{% if part.purchaseable and roles.purchase_order.view %}
|
||||||
|
<li class='list-group-item {% if tab == "order-prices" %}active{% endif %}' title='{% trans "Order Price Information" %}'>
|
||||||
|
<a href='{% url "part-order-prices" part.id %}'>
|
||||||
|
<span class='menu-tab-icon fas fa-dollar-sign' style='width: 20px;'></span>
|
||||||
|
{% trans "Order Price" %}
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
<li class='list-group-item {% if tab == "manufacturers" %}active{% endif %}' title='{% trans "Manufacturers" %}'>
|
<li class='list-group-item {% if tab == "manufacturers" %}active{% endif %}' title='{% trans "Manufacturers" %}'>
|
||||||
<a href='{% url "part-manufacturers" part.id %}'>
|
<a href='{% url "part-manufacturers" part.id %}'>
|
||||||
<span class='menu-tab-icon fas fa-industry'></span>
|
<span class='menu-tab-icon fas fa-industry'></span>
|
||||||
|
219
InvenTree/part/templates/part/order_prices.html
Normal file
219
InvenTree/part/templates/part/order_prices.html
Normal file
@ -0,0 +1,219 @@
|
|||||||
|
{% extends "part/part_base.html" %}
|
||||||
|
{% load static %}
|
||||||
|
{% load i18n %}
|
||||||
|
{% load inventree_extras %}
|
||||||
|
|
||||||
|
{% block menubar %}
|
||||||
|
{% include 'part/navbar.html' with tab='order-prices' %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block heading %}
|
||||||
|
{% trans "Order Price Information" %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block details %}
|
||||||
|
|
||||||
|
<form method="post" action='' class='js-modal-form' enctype="multipart/form-data">
|
||||||
|
{% csrf_token %}
|
||||||
|
{% load crispy_forms_tags %}
|
||||||
|
{% crispy form %}
|
||||||
|
</form>
|
||||||
|
|
||||||
|
|
||||||
|
{% if part.supplier_count > 0 %}
|
||||||
|
<h4>{% trans 'Supplier Pricing' %}</h4>
|
||||||
|
<table class='table table-striped table-condensed table-price-three'>
|
||||||
|
{% if min_total_buy_price %}
|
||||||
|
<tr>
|
||||||
|
<td><b>{% trans 'Unit Cost' %}</b></td>
|
||||||
|
<td>Min: {% include "price.html" with price=min_unit_buy_price %}</td>
|
||||||
|
<td>Max: {% include "price.html" with price=max_unit_buy_price %}</td>
|
||||||
|
</tr>
|
||||||
|
{% if quantity > 1 %}
|
||||||
|
<tr>
|
||||||
|
<td><b>{% trans 'Total Cost' %}</b></td>
|
||||||
|
<td>Min: {% include "price.html" with price=min_total_buy_price %}</td>
|
||||||
|
<td>Max: {% include "price.html" with price=max_total_buy_price %}</td>
|
||||||
|
</tr>
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
<tr>
|
||||||
|
<td colspan='3'>
|
||||||
|
<span class='warning-msg'><i>{% trans 'No supplier pricing available' %}</i></span>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endif %}
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if part.bom_count > 0 %}
|
||||||
|
<h4>{% trans 'BOM Pricing' %}</h4>
|
||||||
|
<table class='table table-striped table-condensed table-price-three'>
|
||||||
|
{% if min_total_bom_price %}
|
||||||
|
<tr>
|
||||||
|
<td><b>{% trans 'Unit Cost' %}</b></td>
|
||||||
|
<td>Min: {% include "price.html" with price=min_unit_bom_price %}</td>
|
||||||
|
<td>Max: {% include "price.html" with price=max_unit_bom_price %}</td>
|
||||||
|
</tr>
|
||||||
|
{% if quantity > 1 %}
|
||||||
|
<tr>
|
||||||
|
<td><b>{% trans 'Total Cost' %}</b></td>
|
||||||
|
<td>Min: {% include "price.html" with price=min_total_bom_price %}</td>
|
||||||
|
<td>Max: {% include "price.html" with price=max_total_bom_price %}</td>
|
||||||
|
</tr>
|
||||||
|
{% endif %}
|
||||||
|
{% if part.has_complete_bom_pricing == False %}
|
||||||
|
<tr>
|
||||||
|
<td colspan='3'>
|
||||||
|
<span class='warning-msg'><i>{% trans 'Note: BOM pricing is incomplete for this part' %}</i></span>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
<tr>
|
||||||
|
<td colspan='3'>
|
||||||
|
<span class='warning-msg'><i>{% trans 'No BOM pricing available' %}</i></span>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endif %}
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if total_part_price %}
|
||||||
|
<h4>{% trans 'Sale Price' %}</h4>
|
||||||
|
<table class='table table-striped table-condensed table-price-two'>
|
||||||
|
<tr>
|
||||||
|
<td><b>{% trans 'Unit Cost' %}</b></td>
|
||||||
|
<td>{% include "price.html" with price=unit_part_price %}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><b>{% trans 'Total Cost' %}</b></td>
|
||||||
|
<td>{% include "price.html" with price=total_part_price %}</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if min_unit_buy_price or min_unit_bom_price %}
|
||||||
|
{% else %}
|
||||||
|
<div class='alert alert-danger alert-block'>
|
||||||
|
{% trans 'No pricing information is available for this part.' %}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
<hr>
|
||||||
|
|
||||||
|
{% if price_history %}
|
||||||
|
<h4>{% trans 'Stock Pricing' %}</h4>
|
||||||
|
{% if price_history|length > 1 %}
|
||||||
|
<div style="max-width: 99%; min-height: 300px">
|
||||||
|
<canvas id="StockPriceChart"></canvas>
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class='alert alert-danger alert-block'>
|
||||||
|
{% trans 'No stock pricing history is available for this part.' %}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
{% block js_ready %}
|
||||||
|
{{ block.super }}
|
||||||
|
|
||||||
|
{% settings_value "INVENTREE_DEFAULT_CURRENCY" as currency %}
|
||||||
|
|
||||||
|
{% if price_history %}
|
||||||
|
var pricedata = {
|
||||||
|
labels: [
|
||||||
|
{% for line in price_history %}'{{ line.date }}',{% endfor %}
|
||||||
|
],
|
||||||
|
datasets: [{
|
||||||
|
label: '{% trans "Single Price" %}',
|
||||||
|
backgroundColor: 'rgba(255, 99, 132, 0.2)',
|
||||||
|
borderColor: 'rgb(255, 99, 132)',
|
||||||
|
yAxisID: 'y',
|
||||||
|
data: [
|
||||||
|
{% for line in price_history %}{{ line.price|stringformat:".2f" }},{% endfor %}
|
||||||
|
],
|
||||||
|
borderWidth: 1,
|
||||||
|
type: 'line'
|
||||||
|
},
|
||||||
|
{% if 'price_diff' in price_history.0 %}
|
||||||
|
{
|
||||||
|
label: '{% trans "Single Price Difference" %}',
|
||||||
|
backgroundColor: 'rgba(68, 157, 68, 0.2)',
|
||||||
|
borderColor: 'rgb(68, 157, 68)',
|
||||||
|
yAxisID: 'y2',
|
||||||
|
data: [
|
||||||
|
{% for line in price_history %}{{ line.price_diff|stringformat:".2f" }},{% endfor %}
|
||||||
|
],
|
||||||
|
borderWidth: 1,
|
||||||
|
type: 'line'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '{% trans "Part Single Price" %}',
|
||||||
|
backgroundColor: 'rgba(70, 127, 155, 0.2)',
|
||||||
|
borderColor: 'rgb(70, 127, 155)',
|
||||||
|
yAxisID: 'y',
|
||||||
|
data: [
|
||||||
|
{% for line in price_history %}{{ line.price_part|stringformat:".2f" }},{% endfor %}
|
||||||
|
],
|
||||||
|
borderWidth: 1,
|
||||||
|
type: 'line'
|
||||||
|
},
|
||||||
|
{% endif %}
|
||||||
|
{
|
||||||
|
label: '{% trans "Quantity" %}',
|
||||||
|
backgroundColor: 'rgba(255, 206, 86, 0.2)',
|
||||||
|
borderColor: 'rgb(255, 206, 86)',
|
||||||
|
yAxisID: 'y1',
|
||||||
|
data: [
|
||||||
|
{% for line in price_history %}{{ line.qty|stringformat:"f" }},{% endfor %}
|
||||||
|
],
|
||||||
|
borderWidth: 1
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
var ctx = document.getElementById('StockPriceChart');
|
||||||
|
var StockPriceChart = new Chart(ctx, {
|
||||||
|
type: 'bar',
|
||||||
|
data: pricedata,
|
||||||
|
options: {
|
||||||
|
responsive: true,
|
||||||
|
maintainAspectRatio: false,
|
||||||
|
plugins: {legend: {position: 'bottom'}},
|
||||||
|
scales: {
|
||||||
|
y: {
|
||||||
|
type: 'linear',
|
||||||
|
position: 'left',
|
||||||
|
grid: {display: false},
|
||||||
|
title: {
|
||||||
|
display: true,
|
||||||
|
text: '{% blocktrans %}Single Price - {{currency}}{% endblocktrans %}'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
y1: {
|
||||||
|
type: 'linear',
|
||||||
|
position: 'right',
|
||||||
|
grid: {display: false},
|
||||||
|
titel: {
|
||||||
|
display: true,
|
||||||
|
text: '{% trans "Quantity" %}',
|
||||||
|
position: 'right'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
y2: {
|
||||||
|
type: 'linear',
|
||||||
|
position: 'left',
|
||||||
|
grid: {display: false},
|
||||||
|
title: {
|
||||||
|
display: true,
|
||||||
|
text: '{% blocktrans %}Single Price Difference- {{currency}}{% endblocktrans %}'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
});
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% endblock %}
|
@ -59,6 +59,7 @@ part_detail_urls = [
|
|||||||
url(r'^bom/?', views.PartDetail.as_view(template_name='part/bom.html'), name='part-bom'),
|
url(r'^bom/?', views.PartDetail.as_view(template_name='part/bom.html'), name='part-bom'),
|
||||||
url(r'^build/?', views.PartDetail.as_view(template_name='part/build.html'), name='part-build'),
|
url(r'^build/?', views.PartDetail.as_view(template_name='part/build.html'), name='part-build'),
|
||||||
url(r'^used/?', views.PartDetail.as_view(template_name='part/used_in.html'), name='part-used-in'),
|
url(r'^used/?', views.PartDetail.as_view(template_name='part/used_in.html'), name='part-used-in'),
|
||||||
|
url(r'^order-prices/', views.PartPricingView.as_view(template_name='part/order_prices.html'), name='part-order-prices'),
|
||||||
url(r'^manufacturers/?', views.PartDetail.as_view(template_name='part/manufacturer.html'), name='part-manufacturers'),
|
url(r'^manufacturers/?', views.PartDetail.as_view(template_name='part/manufacturer.html'), name='part-manufacturers'),
|
||||||
url(r'^suppliers/?', views.PartDetail.as_view(template_name='part/supplier.html'), name='part-suppliers'),
|
url(r'^suppliers/?', views.PartDetail.as_view(template_name='part/supplier.html'), name='part-suppliers'),
|
||||||
url(r'^orders/?', views.PartDetail.as_view(template_name='part/orders.html'), name='part-orders'),
|
url(r'^orders/?', views.PartDetail.as_view(template_name='part/orders.html'), name='part-orders'),
|
||||||
|
@ -784,6 +784,81 @@ class PartDetail(InvenTreeRoleMixin, DetailView):
|
|||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
|
class PartPricingView(PartDetail):
|
||||||
|
""" Detail view for Part object
|
||||||
|
"""
|
||||||
|
context_object_name = 'part'
|
||||||
|
template_name = 'part/order_prices.html'
|
||||||
|
form_class = part_forms.PartPriceForm
|
||||||
|
|
||||||
|
# Add in some extra context information based on query params
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
""" Provide extra context data to template """
|
||||||
|
context = super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
ctx = self.get_pricing(self.get_quantity())
|
||||||
|
ctx['form'] = self.form_class(initial=self.get_initials())
|
||||||
|
|
||||||
|
context.update(ctx)
|
||||||
|
return context
|
||||||
|
|
||||||
|
def get_quantity(self):
|
||||||
|
""" Return set quantity in decimal format """
|
||||||
|
return Decimal(self.request.POST.get('quantity', 1))
|
||||||
|
|
||||||
|
def get_part(self):
|
||||||
|
return self.get_object()
|
||||||
|
|
||||||
|
def get_pricing(self, quantity=1, currency=None):
|
||||||
|
""" returns context with pricing information """
|
||||||
|
ctx = PartPricing.get_pricing(self, quantity, currency)
|
||||||
|
part = self.get_part()
|
||||||
|
# Stock history
|
||||||
|
if part.total_stock > 1:
|
||||||
|
ret = []
|
||||||
|
stock = part.stock_entries(include_variants=False, in_stock=True) # .order_by('purchase_order__date')
|
||||||
|
stock = stock.prefetch_related('purchase_order', 'supplier_part')
|
||||||
|
|
||||||
|
for stock_item in stock:
|
||||||
|
if None in [stock_item.purchase_price, stock_item.quantity]:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# convert purchase price to current currency - only one currency in the graph
|
||||||
|
price = convert_money(stock_item.purchase_price, inventree_settings.currency_code_default())
|
||||||
|
line = {
|
||||||
|
'price': price.amount,
|
||||||
|
'qty': stock_item.quantity
|
||||||
|
}
|
||||||
|
# Supplier Part Name # TODO use in graph
|
||||||
|
if stock_item.supplier_part:
|
||||||
|
line['name'] = stock_item.supplier_part.pretty_name
|
||||||
|
|
||||||
|
if stock_item.supplier_part.unit_pricing and price:
|
||||||
|
line['price_diff'] = price.amount - stock_item.supplier_part.unit_pricing
|
||||||
|
line['price_part'] = stock_item.supplier_part.unit_pricing
|
||||||
|
|
||||||
|
# set date for graph labels
|
||||||
|
if stock_item.purchase_order:
|
||||||
|
line['date'] = stock_item.purchase_order.issue_date.strftime('%d.%m.%Y')
|
||||||
|
else:
|
||||||
|
line['date'] = stock_item.tracking_info.first().date.strftime('%d.%m.%Y')
|
||||||
|
ret.append(line)
|
||||||
|
|
||||||
|
ctx['price_history'] = ret
|
||||||
|
|
||||||
|
return ctx
|
||||||
|
|
||||||
|
def get_initials(self):
|
||||||
|
""" returns initials for form """
|
||||||
|
return {'quantity': self.get_quantity()}
|
||||||
|
|
||||||
|
def post(self, request, *args, **kwargs):
|
||||||
|
self.object = self.get_object()
|
||||||
|
kwargs['object'] = self.object
|
||||||
|
ctx=self.get_context_data(**kwargs)
|
||||||
|
return self.get(request, context=ctx)
|
||||||
|
|
||||||
|
|
||||||
class PartDetailFromIPN(PartDetail):
|
class PartDetailFromIPN(PartDetail):
|
||||||
slug_field = 'IPN'
|
slug_field = 'IPN'
|
||||||
slug_url_kwarg = 'slug'
|
slug_url_kwarg = 'slug'
|
||||||
@ -2040,38 +2115,6 @@ class PartPricing(AjaxView):
|
|||||||
ctx['max_total_bom_price'] = max_bom_price
|
ctx['max_total_bom_price'] = max_bom_price
|
||||||
ctx['max_unit_bom_price'] = max_unit_bom_price
|
ctx['max_unit_bom_price'] = max_unit_bom_price
|
||||||
|
|
||||||
# Stock history
|
|
||||||
if part_settings.part_show_graph and part.total_stock > 1:
|
|
||||||
ret = []
|
|
||||||
stock = part.stock_entries(include_variants=False, in_stock=True) # .order_by('purchase_order__date')
|
|
||||||
stock = stock.prefetch_related('purchase_order', 'supplier_part')
|
|
||||||
|
|
||||||
for stock_item in stock:
|
|
||||||
if None in [stock_item.purchase_price, stock_item.quantity]:
|
|
||||||
continue
|
|
||||||
|
|
||||||
# convert purchase price to current currency - only one currency in the graph
|
|
||||||
price = convert_money(stock_item.purchase_price, inventree_settings.currency_code_default())
|
|
||||||
line = {
|
|
||||||
'price': price.amount,
|
|
||||||
'qty': stock_item.quantity
|
|
||||||
}
|
|
||||||
# Supplier Part Name # TODO use in graph
|
|
||||||
if stock_item.supplier_part:
|
|
||||||
line['name'] = stock_item.supplier_part.pretty_name
|
|
||||||
|
|
||||||
if stock_item.supplier_part.unit_pricing and price:
|
|
||||||
line['price_diff'] = price.amount - stock_item.supplier_part.unit_pricing
|
|
||||||
line['price_part'] = stock_item.supplier_part.unit_pricing
|
|
||||||
|
|
||||||
# set date for graph labels
|
|
||||||
if stock_item.purchase_order:
|
|
||||||
line['date'] = stock_item.purchase_order.issue_date.strftime('%d.%m.%Y')
|
|
||||||
else:
|
|
||||||
line['date'] = stock_item.tracking_info.first().date.strftime('%d.%m.%Y')
|
|
||||||
ret.append(line)
|
|
||||||
|
|
||||||
ctx['price_history'] = ret
|
|
||||||
# part pricing information
|
# part pricing information
|
||||||
part_price = part.get_price(quantity)
|
part_price = part.get_price(quantity)
|
||||||
if part_price is not None:
|
if part_price is not None:
|
||||||
|
Loading…
Reference in New Issue
Block a user