2019-04-27 12:18:07 +00:00
|
|
|
"""
|
|
|
|
Django views for interacting with Part app
|
|
|
|
"""
|
|
|
|
|
2018-04-16 14:32:02 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
from __future__ import unicode_literals
|
2018-04-15 11:29:24 +00:00
|
|
|
|
2019-06-27 12:46:11 +00:00
|
|
|
from django.core.exceptions import ValidationError
|
2019-09-17 04:17:49 +00:00
|
|
|
from django.db import transaction
|
2018-04-15 15:02:17 +00:00
|
|
|
from django.shortcuts import get_object_or_404
|
2019-07-10 02:27:19 +00:00
|
|
|
from django.shortcuts import HttpResponseRedirect
|
2019-06-25 09:15:39 +00:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2019-07-02 09:02:19 +00:00
|
|
|
from django.urls import reverse, reverse_lazy
|
2020-02-01 02:36:09 +00:00
|
|
|
from django.views.generic import DetailView, ListView, FormView, UpdateView
|
2019-04-18 13:12:29 +00:00
|
|
|
from django.forms.models import model_to_dict
|
2019-05-11 02:29:02 +00:00
|
|
|
from django.forms import HiddenInput, CheckboxInput
|
2020-02-10 12:48:45 +00:00
|
|
|
from django.conf import settings
|
|
|
|
|
|
|
|
import os
|
2018-04-13 12:36:59 +00:00
|
|
|
|
2020-03-22 18:54:36 +00:00
|
|
|
from rapidfuzz import fuzz
|
2020-08-18 04:01:01 +00:00
|
|
|
from decimal import Decimal, InvalidOperation
|
2019-07-07 01:44:17 +00:00
|
|
|
|
2019-05-02 07:29:21 +00:00
|
|
|
from .models import PartCategory, Part, PartAttachment
|
2019-08-20 04:14:21 +00:00
|
|
|
from .models import PartParameterTemplate, PartParameter
|
2019-05-02 07:29:21 +00:00
|
|
|
from .models import BomItem
|
2019-05-11 01:55:17 +00:00
|
|
|
from .models import match_part_names
|
2020-05-17 06:50:34 +00:00
|
|
|
from .models import PartTestTemplate
|
2018-04-22 11:54:12 +00:00
|
|
|
|
2020-02-03 10:14:06 +00:00
|
|
|
from common.models import Currency, InvenTreeSetting
|
2019-05-18 10:24:09 +00:00
|
|
|
from company.models import SupplierPart
|
|
|
|
|
2019-05-10 23:51:45 +00:00
|
|
|
from . import forms as part_forms
|
2019-09-15 12:21:12 +00:00
|
|
|
from .bom import MakeBomTemplate, BomUploadManager, ExportBom, IsValidBOMFormat
|
2018-04-15 15:02:17 +00:00
|
|
|
|
2019-09-15 09:52:28 +00:00
|
|
|
from .admin import PartResource
|
|
|
|
|
2019-04-16 12:32:43 +00:00
|
|
|
from InvenTree.views import AjaxView, AjaxCreateView, AjaxUpdateView, AjaxDeleteView
|
2019-05-04 08:46:57 +00:00
|
|
|
from InvenTree.views import QRCodeView
|
2018-04-13 12:36:59 +00:00
|
|
|
|
2019-06-27 12:46:11 +00:00
|
|
|
from InvenTree.helpers import DownloadFile, str2bool
|
2018-04-27 15:16:47 +00:00
|
|
|
|
2019-04-16 12:32:43 +00:00
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
class PartIndex(ListView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" View for displaying list of Part objects
|
|
|
|
"""
|
2018-04-14 15:18:12 +00:00
|
|
|
model = Part
|
2018-05-04 08:53:39 +00:00
|
|
|
template_name = 'part/category.html'
|
2018-04-14 15:18:12 +00:00
|
|
|
context_object_name = 'parts'
|
2018-04-13 12:36:59 +00:00
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
def get_queryset(self):
|
2019-05-20 14:31:34 +00:00
|
|
|
return Part.objects.all().select_related('category')
|
2018-04-13 12:36:59 +00:00
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
def get_context_data(self, **kwargs):
|
2018-04-13 14:08:30 +00:00
|
|
|
|
2018-04-15 10:10:49 +00:00
|
|
|
context = super(PartIndex, self).get_context_data(**kwargs).copy()
|
2018-04-13 14:08:30 +00:00
|
|
|
|
2018-04-15 01:40:03 +00:00
|
|
|
# View top-level categories
|
|
|
|
children = PartCategory.objects.filter(parent=None)
|
2018-04-13 14:08:30 +00:00
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
context['children'] = children
|
2019-09-27 00:04:20 +00:00
|
|
|
context['category_count'] = PartCategory.objects.count()
|
|
|
|
context['part_count'] = Part.objects.count()
|
2018-04-14 11:58:01 +00:00
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
return context
|
2018-04-13 14:08:30 +00:00
|
|
|
|
|
|
|
|
2019-05-02 07:29:21 +00:00
|
|
|
class PartAttachmentCreate(AjaxCreateView):
|
|
|
|
""" View for creating a new PartAttachment object
|
|
|
|
|
|
|
|
- The view only makes sense if a Part object is passed to it
|
|
|
|
"""
|
|
|
|
model = PartAttachment
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditPartAttachmentForm
|
2020-02-11 23:25:46 +00:00
|
|
|
ajax_form_title = _("Add part attachment")
|
2019-05-02 07:29:21 +00:00
|
|
|
ajax_template_name = "modal_form.html"
|
|
|
|
|
2020-05-12 11:40:42 +00:00
|
|
|
def post_save(self):
|
|
|
|
""" Record the user that uploaded the attachment """
|
|
|
|
self.object.user = self.request.user
|
|
|
|
self.object.save()
|
|
|
|
|
2019-05-02 07:29:21 +00:00
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-10 11:10:06 +00:00
|
|
|
'success': _('Added attachment')
|
2019-05-02 07:29:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def get_initial(self):
|
|
|
|
""" Get initial data for new PartAttachment object.
|
|
|
|
|
|
|
|
- Client should have requested this form with a parent part in mind
|
|
|
|
- e.g. ?part=<pk>
|
|
|
|
"""
|
|
|
|
|
|
|
|
initials = super(AjaxCreateView, self).get_initial()
|
|
|
|
|
|
|
|
# TODO - If the proper part was not sent, return an error message
|
2019-09-03 12:45:11 +00:00
|
|
|
try:
|
|
|
|
initials['part'] = Part.objects.get(id=self.request.GET.get('part', None))
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
|
|
pass
|
2019-05-02 07:29:21 +00:00
|
|
|
|
|
|
|
return initials
|
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
""" Create a form to upload a new PartAttachment
|
|
|
|
|
|
|
|
- Hide the 'part' field
|
|
|
|
"""
|
|
|
|
|
|
|
|
form = super(AjaxCreateView, self).get_form()
|
|
|
|
|
|
|
|
form.fields['part'].widget = HiddenInput()
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
|
|
|
|
|
|
|
class PartAttachmentEdit(AjaxUpdateView):
|
|
|
|
""" View for editing a PartAttachment object """
|
2020-03-22 08:55:46 +00:00
|
|
|
|
2019-05-02 07:29:21 +00:00
|
|
|
model = PartAttachment
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditPartAttachmentForm
|
2019-05-02 07:29:21 +00:00
|
|
|
ajax_template_name = 'modal_form.html'
|
2020-02-11 23:25:46 +00:00
|
|
|
ajax_form_title = _('Edit attachment')
|
2019-05-02 07:29:21 +00:00
|
|
|
|
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-10 11:10:06 +00:00
|
|
|
'success': _('Part attachment updated')
|
2019-05-02 07:29:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
form = super(AjaxUpdateView, self).get_form()
|
|
|
|
|
|
|
|
form.fields['part'].widget = HiddenInput()
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
|
|
|
|
|
|
|
class PartAttachmentDelete(AjaxDeleteView):
|
|
|
|
""" View for deleting a PartAttachment """
|
|
|
|
|
|
|
|
model = PartAttachment
|
2020-02-11 23:25:46 +00:00
|
|
|
ajax_form_title = _("Delete Part Attachment")
|
2020-05-06 23:57:54 +00:00
|
|
|
ajax_template_name = "attachment_delete.html"
|
2019-05-02 07:29:21 +00:00
|
|
|
context_object_name = "attachment"
|
|
|
|
|
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-11 23:25:46 +00:00
|
|
|
'danger': _('Deleted part attachment')
|
2019-05-02 07:29:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-17 06:50:34 +00:00
|
|
|
class PartTestTemplateCreate(AjaxCreateView):
|
|
|
|
""" View for creating a PartTestTemplate """
|
|
|
|
|
|
|
|
model = PartTestTemplate
|
|
|
|
form_class = part_forms.EditPartTestTemplateForm
|
|
|
|
ajax_form_title = _("Create Test Template")
|
|
|
|
|
|
|
|
def get_initial(self):
|
|
|
|
|
|
|
|
initials = super().get_initial()
|
|
|
|
|
|
|
|
try:
|
|
|
|
part_id = self.request.GET.get('part', None)
|
|
|
|
initials['part'] = Part.objects.get(pk=part_id)
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
|
|
pass
|
|
|
|
|
|
|
|
return initials
|
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
|
|
|
|
form = super().get_form()
|
|
|
|
form.fields['part'].widget = HiddenInput()
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
|
|
|
|
|
|
|
class PartTestTemplateEdit(AjaxUpdateView):
|
|
|
|
""" View for editing a PartTestTemplate """
|
|
|
|
|
|
|
|
model = PartTestTemplate
|
|
|
|
form_class = part_forms.EditPartTestTemplateForm
|
|
|
|
ajax_form_title = _("Edit Test Template")
|
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
|
|
|
|
form = super().get_form()
|
|
|
|
form.fields['part'].widget = HiddenInput()
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
|
|
|
|
|
|
|
class PartTestTemplateDelete(AjaxDeleteView):
|
|
|
|
""" View for deleting a PartTestTemplate """
|
|
|
|
|
|
|
|
model = PartTestTemplate
|
|
|
|
ajax_form_title = _("Delete Test Template")
|
|
|
|
|
|
|
|
|
2019-09-17 04:06:11 +00:00
|
|
|
class PartSetCategory(AjaxUpdateView):
|
2019-06-25 09:15:39 +00:00
|
|
|
""" View for settings the part category for multiple parts at once """
|
|
|
|
|
|
|
|
ajax_template_name = 'part/set_category.html'
|
2020-02-11 23:25:46 +00:00
|
|
|
ajax_form_title = _('Set Part Category')
|
2019-09-17 04:06:11 +00:00
|
|
|
form_class = part_forms.SetPartCategoryForm
|
2019-06-25 09:15:39 +00:00
|
|
|
|
|
|
|
category = None
|
|
|
|
parts = []
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
""" Respond to a GET request to this view """
|
|
|
|
|
|
|
|
self.request = request
|
|
|
|
|
|
|
|
if 'parts[]' in request.GET:
|
|
|
|
self.parts = Part.objects.filter(id__in=request.GET.getlist('parts[]'))
|
|
|
|
else:
|
|
|
|
self.parts = []
|
|
|
|
|
2019-09-17 04:06:11 +00:00
|
|
|
return self.renderJsonResponse(request, form=self.get_form(), context=self.get_context_data())
|
2019-06-25 09:15:39 +00:00
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
""" Respond to a POST request to this view """
|
|
|
|
|
|
|
|
self.parts = []
|
|
|
|
|
|
|
|
for item in request.POST:
|
|
|
|
if item.startswith('part_id_'):
|
|
|
|
pk = item.replace('part_id_', '')
|
|
|
|
|
|
|
|
try:
|
|
|
|
part = Part.objects.get(pk=pk)
|
|
|
|
except (Part.DoesNotExist, ValueError):
|
|
|
|
continue
|
|
|
|
|
|
|
|
self.parts.append(part)
|
|
|
|
|
|
|
|
self.category = None
|
|
|
|
|
|
|
|
if 'part_category' in request.POST:
|
|
|
|
pk = request.POST['part_category']
|
|
|
|
|
|
|
|
try:
|
|
|
|
self.category = PartCategory.objects.get(pk=pk)
|
|
|
|
except (PartCategory.DoesNotExist, ValueError):
|
|
|
|
self.category = None
|
|
|
|
|
|
|
|
valid = self.category is not None
|
|
|
|
|
|
|
|
data = {
|
|
|
|
'form_valid': valid,
|
|
|
|
'success': _('Set category for {n} parts'.format(n=len(self.parts)))
|
|
|
|
}
|
|
|
|
|
|
|
|
if valid:
|
2019-09-17 04:17:49 +00:00
|
|
|
self.set_category()
|
2019-06-25 09:15:39 +00:00
|
|
|
|
2019-09-17 04:06:11 +00:00
|
|
|
return self.renderJsonResponse(request, data=data, form=self.get_form(), context=self.get_context_data())
|
2019-06-25 09:15:39 +00:00
|
|
|
|
2019-09-17 04:17:49 +00:00
|
|
|
@transaction.atomic
|
|
|
|
def set_category(self):
|
|
|
|
for part in self.parts:
|
|
|
|
part.set_category(self.category)
|
|
|
|
|
2019-06-25 09:15:39 +00:00
|
|
|
def get_context_data(self):
|
|
|
|
""" Return context data for rendering in the form """
|
|
|
|
ctx = {}
|
|
|
|
|
|
|
|
ctx['parts'] = self.parts
|
|
|
|
ctx['categories'] = PartCategory.objects.all()
|
|
|
|
ctx['category'] = self.category
|
|
|
|
|
|
|
|
return ctx
|
2019-06-25 09:16:24 +00:00
|
|
|
|
2019-06-25 09:15:39 +00:00
|
|
|
|
2019-05-26 06:00:27 +00:00
|
|
|
class MakePartVariant(AjaxCreateView):
|
2019-05-26 06:05:54 +00:00
|
|
|
""" View for creating a new variant based on an existing template Part
|
2019-05-26 06:00:27 +00:00
|
|
|
|
|
|
|
- Part <pk> is provided in the URL '/part/<pk>/make_variant/'
|
|
|
|
- Automatically copy relevent data (BOM, etc, etc)
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
model = Part
|
|
|
|
form_class = part_forms.EditPartForm
|
|
|
|
|
2020-02-11 23:25:46 +00:00
|
|
|
ajax_form_title = _('Create Variant')
|
2019-05-26 06:00:27 +00:00
|
|
|
ajax_template_name = 'part/variant_part.html'
|
|
|
|
|
|
|
|
def get_part_template(self):
|
|
|
|
return get_object_or_404(Part, id=self.kwargs['pk'])
|
|
|
|
|
|
|
|
def get_context_data(self):
|
|
|
|
return {
|
|
|
|
'part': self.get_part_template(),
|
|
|
|
}
|
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
form = super(AjaxCreateView, self).get_form()
|
|
|
|
|
|
|
|
# Hide some variant-related fields
|
2020-09-02 13:57:51 +00:00
|
|
|
# form.fields['variant_of'].widget = HiddenInput()
|
2019-05-26 06:00:27 +00:00
|
|
|
|
2020-09-05 10:10:18 +00:00
|
|
|
# Force display of the 'bom_copy' widget
|
|
|
|
form.fields['bom_copy'].widget = CheckboxInput()
|
|
|
|
|
|
|
|
# Force display of the 'parameters_copy' widget
|
|
|
|
form.fields['parameters_copy'].widget = CheckboxInput()
|
|
|
|
|
2019-05-26 06:00:27 +00:00
|
|
|
return form
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
form = self.get_form()
|
|
|
|
context = self.get_context_data()
|
|
|
|
part_template = self.get_part_template()
|
|
|
|
|
|
|
|
valid = form.is_valid()
|
|
|
|
|
|
|
|
data = {
|
|
|
|
'form_valid': valid,
|
|
|
|
}
|
|
|
|
|
|
|
|
if valid:
|
|
|
|
# Create the new part variant
|
|
|
|
part = form.save(commit=False)
|
|
|
|
part.variant_of = part_template
|
|
|
|
part.is_template = False
|
|
|
|
|
|
|
|
part.save()
|
|
|
|
|
|
|
|
data['pk'] = part.pk
|
|
|
|
data['text'] = str(part)
|
|
|
|
data['url'] = part.get_absolute_url()
|
|
|
|
|
2020-09-05 10:10:18 +00:00
|
|
|
bom_copy = str2bool(request.POST.get('bom_copy', False))
|
|
|
|
parameters_copy = str2bool(request.POST.get('parameters_copy', False))
|
|
|
|
|
2019-05-26 06:00:27 +00:00
|
|
|
# Copy relevent information from the template part
|
2020-09-05 10:10:18 +00:00
|
|
|
part.deepCopy(part_template, bom=bom_copy, parameters=parameters_copy)
|
2019-05-26 06:00:27 +00:00
|
|
|
|
|
|
|
return self.renderJsonResponse(request, form, data, context=context)
|
|
|
|
|
|
|
|
def get_initial(self):
|
|
|
|
|
|
|
|
part_template = self.get_part_template()
|
|
|
|
|
|
|
|
initials = model_to_dict(part_template)
|
|
|
|
initials['is_template'] = False
|
|
|
|
initials['variant_of'] = part_template
|
|
|
|
|
|
|
|
return initials
|
|
|
|
|
|
|
|
|
2019-05-13 11:41:32 +00:00
|
|
|
class PartDuplicate(AjaxCreateView):
|
|
|
|
""" View for duplicating an existing Part object.
|
|
|
|
|
|
|
|
- Part <pk> is provided in the URL '/part/<pk>/copy/'
|
|
|
|
- Option for 'deep-copy' which will duplicate all BOM items (default = True)
|
|
|
|
"""
|
|
|
|
|
|
|
|
model = Part
|
|
|
|
form_class = part_forms.EditPartForm
|
|
|
|
|
2020-02-11 23:25:46 +00:00
|
|
|
ajax_form_title = _("Duplicate Part")
|
2019-05-13 11:41:32 +00:00
|
|
|
ajax_template_name = "part/copy_part.html"
|
|
|
|
|
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-10 11:10:06 +00:00
|
|
|
'success': _('Copied part')
|
2019-05-13 11:41:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def get_part_to_copy(self):
|
|
|
|
try:
|
|
|
|
return Part.objects.get(id=self.kwargs['pk'])
|
2019-08-08 13:32:34 +00:00
|
|
|
except (Part.DoesNotExist, ValueError):
|
2019-05-13 11:41:32 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
def get_context_data(self):
|
|
|
|
return {
|
|
|
|
'part': self.get_part_to_copy()
|
|
|
|
}
|
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
form = super(AjaxCreateView, self).get_form()
|
|
|
|
|
2020-09-04 19:02:12 +00:00
|
|
|
# Force display of the 'bom_copy' widget
|
|
|
|
form.fields['bom_copy'].widget = CheckboxInput()
|
|
|
|
|
|
|
|
# Force display of the 'parameters_copy' widget
|
|
|
|
form.fields['parameters_copy'].widget = CheckboxInput()
|
2019-05-13 11:41:32 +00:00
|
|
|
|
|
|
|
return form
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
""" Capture the POST request for part duplication
|
|
|
|
|
2020-09-04 19:02:12 +00:00
|
|
|
- If the bom_copy object is set, copy all the BOM items too!
|
|
|
|
- If the parameters_copy object is set, copy all the parameters too!
|
2019-05-13 11:41:32 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
form = self.get_form()
|
|
|
|
|
|
|
|
context = self.get_context_data()
|
|
|
|
|
|
|
|
valid = form.is_valid()
|
|
|
|
|
|
|
|
name = request.POST.get('name', None)
|
|
|
|
|
|
|
|
if name:
|
|
|
|
matches = match_part_names(name)
|
|
|
|
|
|
|
|
if len(matches) > 0:
|
|
|
|
context['matches'] = matches
|
|
|
|
|
|
|
|
# Enforce display of the checkbox
|
|
|
|
form.fields['confirm_creation'].widget = CheckboxInput()
|
|
|
|
|
|
|
|
# Check if the user has checked the 'confirm_creation' input
|
|
|
|
confirmed = str2bool(request.POST.get('confirm_creation', False))
|
|
|
|
|
|
|
|
if not confirmed:
|
|
|
|
form.errors['confirm_creation'] = ['Possible matches exist - confirm creation of new part']
|
|
|
|
|
|
|
|
form.pre_form_warning = 'Possible matches exist - confirm creation of new part'
|
|
|
|
valid = False
|
|
|
|
|
|
|
|
data = {
|
|
|
|
'form_valid': valid
|
|
|
|
}
|
|
|
|
|
|
|
|
if valid:
|
|
|
|
# Create the new Part
|
2020-03-18 10:53:02 +00:00
|
|
|
part = form.save(commit=False)
|
|
|
|
|
|
|
|
part.creation_user = request.user
|
|
|
|
part.save()
|
2019-05-13 11:41:32 +00:00
|
|
|
|
|
|
|
data['pk'] = part.pk
|
2019-05-14 13:04:00 +00:00
|
|
|
data['text'] = str(part)
|
2019-05-13 11:41:32 +00:00
|
|
|
|
2020-09-04 19:02:12 +00:00
|
|
|
bom_copy = str2bool(request.POST.get('bom_copy', False))
|
|
|
|
parameters_copy = str2bool(request.POST.get('parameters_copy', False))
|
2019-05-13 11:41:32 +00:00
|
|
|
|
|
|
|
original = self.get_part_to_copy()
|
|
|
|
|
|
|
|
if original:
|
2020-09-04 19:02:12 +00:00
|
|
|
part.deepCopy(original, bom=bom_copy, parameters=parameters_copy)
|
2019-05-13 11:41:32 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
data['url'] = part.get_absolute_url()
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if valid:
|
|
|
|
pass
|
|
|
|
|
|
|
|
return self.renderJsonResponse(request, form, data, context=context)
|
|
|
|
|
|
|
|
def get_initial(self):
|
|
|
|
""" Get initial data based on the Part to be copied from.
|
|
|
|
"""
|
|
|
|
|
|
|
|
part = self.get_part_to_copy()
|
|
|
|
|
|
|
|
if part:
|
|
|
|
initials = model_to_dict(part)
|
|
|
|
else:
|
2019-05-13 11:54:52 +00:00
|
|
|
initials = super(AjaxCreateView, self).get_initial()
|
2019-05-13 11:41:32 +00:00
|
|
|
|
2020-09-04 19:02:12 +00:00
|
|
|
initials['bom_copy'] = str2bool(InvenTreeSetting.get_setting('part_deep_copy', True))
|
|
|
|
# Create new entry in InvenTree/common/kvp.yaml?
|
|
|
|
initials['parameters_copy'] = str2bool(InvenTreeSetting.get_setting('part_deep_copy', True))
|
2020-02-03 10:14:06 +00:00
|
|
|
|
2019-05-13 11:41:32 +00:00
|
|
|
return initials
|
|
|
|
|
|
|
|
|
2018-04-26 08:22:41 +00:00
|
|
|
class PartCreate(AjaxCreateView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" View for creating a new Part object.
|
|
|
|
|
|
|
|
Options for providing initial conditions:
|
|
|
|
|
|
|
|
- Provide a category object as initial data
|
2018-04-15 00:08:44 +00:00
|
|
|
"""
|
|
|
|
model = Part
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditPartForm
|
2019-04-18 13:47:04 +00:00
|
|
|
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Create new part')
|
2019-05-11 01:55:17 +00:00
|
|
|
ajax_template_name = 'part/create_part.html'
|
2018-04-26 08:22:41 +00:00
|
|
|
|
2018-04-29 14:23:02 +00:00
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-10 11:10:06 +00:00
|
|
|
'success': _("Created new part"),
|
2018-04-29 14:23:02 +00:00
|
|
|
}
|
|
|
|
|
2018-04-15 00:08:44 +00:00
|
|
|
def get_category_id(self):
|
|
|
|
return self.request.GET.get('category', None)
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Provide extra context information for the form to display:
|
|
|
|
|
|
|
|
- Add category information (if provided)
|
|
|
|
"""
|
2018-04-15 00:08:44 +00:00
|
|
|
context = super(PartCreate, self).get_context_data(**kwargs)
|
|
|
|
|
|
|
|
# Add category information to the page
|
|
|
|
cat_id = self.get_category_id()
|
|
|
|
|
|
|
|
if cat_id:
|
2019-04-28 01:09:19 +00:00
|
|
|
try:
|
|
|
|
context['category'] = PartCategory.objects.get(pk=cat_id)
|
2019-08-08 13:32:34 +00:00
|
|
|
except (PartCategory.DoesNotExist, ValueError):
|
2019-04-28 01:09:19 +00:00
|
|
|
pass
|
2018-04-15 00:08:44 +00:00
|
|
|
|
|
|
|
return context
|
|
|
|
|
2019-04-28 13:55:21 +00:00
|
|
|
def get_form(self):
|
2019-04-28 13:57:29 +00:00
|
|
|
""" Create Form for making new Part object.
|
|
|
|
Remove the 'default_supplier' field as there are not yet any matching SupplierPart objects
|
|
|
|
"""
|
2019-04-28 13:55:21 +00:00
|
|
|
form = super(AjaxCreateView, self).get_form()
|
|
|
|
|
|
|
|
# Hide the default_supplier field (there are no matching supplier parts yet!)
|
2019-04-30 23:40:49 +00:00
|
|
|
form.fields['default_supplier'].widget = HiddenInput()
|
2019-04-28 13:57:29 +00:00
|
|
|
|
2019-04-28 13:55:21 +00:00
|
|
|
return form
|
|
|
|
|
2019-05-11 01:55:17 +00:00
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
form = self.get_form()
|
|
|
|
|
|
|
|
context = {}
|
2019-05-11 02:29:02 +00:00
|
|
|
|
|
|
|
valid = form.is_valid()
|
|
|
|
|
|
|
|
name = request.POST.get('name', None)
|
2019-05-11 01:55:17 +00:00
|
|
|
|
|
|
|
if name:
|
|
|
|
matches = match_part_names(name)
|
|
|
|
|
|
|
|
if len(matches) > 0:
|
|
|
|
context['matches'] = matches
|
|
|
|
|
2019-05-11 08:07:37 +00:00
|
|
|
# Enforce display of the checkbox
|
|
|
|
form.fields['confirm_creation'].widget = CheckboxInput()
|
|
|
|
|
2019-05-11 02:29:02 +00:00
|
|
|
# Check if the user has checked the 'confirm_creation' input
|
2019-05-11 08:06:17 +00:00
|
|
|
confirmed = str2bool(request.POST.get('confirm_creation', False))
|
2019-05-11 02:29:02 +00:00
|
|
|
|
|
|
|
if not confirmed:
|
|
|
|
form.errors['confirm_creation'] = ['Possible matches exist - confirm creation of new part']
|
2019-05-11 08:07:37 +00:00
|
|
|
|
|
|
|
form.pre_form_warning = 'Possible matches exist - confirm creation of new part'
|
2019-05-11 02:29:02 +00:00
|
|
|
valid = False
|
2019-05-11 01:55:17 +00:00
|
|
|
|
|
|
|
data = {
|
2019-05-11 02:29:02 +00:00
|
|
|
'form_valid': valid
|
2019-05-11 01:55:17 +00:00
|
|
|
}
|
|
|
|
|
2019-05-11 02:29:02 +00:00
|
|
|
if valid:
|
|
|
|
# Create the new Part
|
2020-03-18 10:53:02 +00:00
|
|
|
part = form.save(commit=False)
|
|
|
|
|
|
|
|
# Record the user who created this part
|
|
|
|
part.creation_user = request.user
|
|
|
|
|
|
|
|
part.save()
|
2019-05-11 02:29:02 +00:00
|
|
|
|
|
|
|
data['pk'] = part.pk
|
2019-05-14 13:04:00 +00:00
|
|
|
data['text'] = str(part)
|
2019-05-11 02:29:02 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
data['url'] = part.get_absolute_url()
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
|
2019-05-11 01:55:17 +00:00
|
|
|
return self.renderJsonResponse(request, form, data, context=context)
|
|
|
|
|
2018-04-15 00:08:44 +00:00
|
|
|
def get_initial(self):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Get initial data for the new Part object:
|
|
|
|
|
|
|
|
- If a category is provided, pre-fill the Category field
|
|
|
|
"""
|
2018-04-15 00:08:44 +00:00
|
|
|
|
2019-05-13 11:41:32 +00:00
|
|
|
initials = super(PartCreate, self).get_initial()
|
2018-04-15 00:08:44 +00:00
|
|
|
|
|
|
|
if self.get_category_id():
|
2019-04-28 01:09:19 +00:00
|
|
|
try:
|
2019-05-14 07:32:29 +00:00
|
|
|
category = PartCategory.objects.get(pk=self.get_category_id())
|
|
|
|
initials['category'] = category
|
|
|
|
initials['keywords'] = category.default_keywords
|
2019-08-08 13:32:34 +00:00
|
|
|
except (PartCategory.DoesNotExist, ValueError):
|
2019-04-28 01:09:19 +00:00
|
|
|
pass
|
2019-07-07 03:06:59 +00:00
|
|
|
|
|
|
|
# Allow initial data to be passed through as arguments
|
|
|
|
for label in ['name', 'IPN', 'description', 'revision', 'keywords']:
|
|
|
|
if label in self.request.GET:
|
|
|
|
initials[label] = self.request.GET.get(label)
|
2018-04-15 00:08:44 +00:00
|
|
|
|
|
|
|
return initials
|
|
|
|
|
|
|
|
|
2020-02-01 02:36:09 +00:00
|
|
|
class PartNotes(UpdateView):
|
|
|
|
""" View for editing the 'notes' field of a Part object.
|
|
|
|
Presents a live markdown editor.
|
|
|
|
"""
|
|
|
|
|
|
|
|
context_object_name = 'part'
|
|
|
|
# form_class = part_forms.EditNotesForm
|
|
|
|
template_name = 'part/notes.html'
|
|
|
|
model = Part
|
|
|
|
|
|
|
|
fields = ['notes']
|
|
|
|
|
2020-02-01 11:25:35 +00:00
|
|
|
def get_success_url(self):
|
|
|
|
""" Return the success URL for this form """
|
|
|
|
|
|
|
|
return reverse('part-notes', kwargs={'pk': self.get_object().id})
|
|
|
|
|
2020-02-01 02:36:09 +00:00
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
|
|
|
|
part = self.get_object()
|
|
|
|
|
|
|
|
ctx = super().get_context_data(**kwargs)
|
|
|
|
|
|
|
|
ctx['editing'] = str2bool(self.request.GET.get('edit', ''))
|
|
|
|
|
|
|
|
ctx['starred'] = part.isStarredBy(self.request.user)
|
|
|
|
ctx['disabled'] = not part.active
|
|
|
|
|
|
|
|
return ctx
|
|
|
|
|
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
class PartDetail(DetailView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Detail view for Part object
|
|
|
|
"""
|
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
context_object_name = 'part'
|
2019-05-20 14:31:34 +00:00
|
|
|
queryset = Part.objects.all().select_related('category')
|
2018-04-14 15:18:12 +00:00
|
|
|
template_name = 'part/detail.html'
|
2018-04-13 14:46:18 +00:00
|
|
|
|
2019-04-15 08:32:15 +00:00
|
|
|
# Add in some extra context information based on query params
|
|
|
|
def get_context_data(self, **kwargs):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Provide extra context data to template
|
|
|
|
|
|
|
|
- If '?editing=True', set 'editing_enabled' context variable
|
|
|
|
"""
|
2019-04-15 08:32:15 +00:00
|
|
|
context = super(PartDetail, self).get_context_data(**kwargs)
|
2019-06-28 00:00:23 +00:00
|
|
|
|
|
|
|
part = self.get_object()
|
2019-04-15 08:32:15 +00:00
|
|
|
|
2019-04-27 12:18:07 +00:00
|
|
|
if str2bool(self.request.GET.get('edit', '')):
|
2019-06-28 00:00:23 +00:00
|
|
|
# Allow BOM editing if the part is active
|
|
|
|
context['editing_enabled'] = 1 if part.active else 0
|
2019-04-15 08:32:15 +00:00
|
|
|
else:
|
2019-04-15 08:41:48 +00:00
|
|
|
context['editing_enabled'] = 0
|
2019-04-15 08:32:15 +00:00
|
|
|
|
2019-05-05 00:54:21 +00:00
|
|
|
context['starred'] = part.isStarredBy(self.request.user)
|
2019-06-18 12:54:32 +00:00
|
|
|
context['disabled'] = not part.active
|
2019-05-05 00:54:21 +00:00
|
|
|
|
2019-04-15 08:32:15 +00:00
|
|
|
return context
|
|
|
|
|
2018-04-13 14:46:18 +00:00
|
|
|
|
2020-09-04 22:20:17 +00:00
|
|
|
class PartDetailFromIPN(PartDetail):
|
|
|
|
slug_field = 'IPN'
|
|
|
|
slug_url_kwarg = 'slug'
|
|
|
|
|
|
|
|
def get_object(self):
|
|
|
|
""" Return Part object which IPN field matches the slug value """
|
|
|
|
queryset = self.get_queryset()
|
|
|
|
# Get slug
|
|
|
|
slug = self.kwargs.get(self.slug_url_kwarg)
|
|
|
|
|
|
|
|
if slug is not None:
|
|
|
|
slug_field = self.get_slug_field()
|
|
|
|
# Filter by the slug value
|
|
|
|
queryset = queryset.filter(**{slug_field: slug})
|
|
|
|
|
|
|
|
try:
|
|
|
|
# Get unique part from queryset
|
|
|
|
part = queryset.get()
|
|
|
|
# Return Part object
|
|
|
|
return part
|
2020-09-07 13:40:37 +00:00
|
|
|
except queryset.model.MultipleObjectsReturned:
|
|
|
|
pass
|
2020-09-04 22:20:17 +00:00
|
|
|
except queryset.model.DoesNotExist:
|
|
|
|
pass
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
""" Attempt to match slug to a Part, else redirect to PartIndex view """
|
|
|
|
self.object = self.get_object()
|
|
|
|
|
|
|
|
if not self.object:
|
|
|
|
return HttpResponseRedirect(reverse('part-index'))
|
|
|
|
|
|
|
|
return super(PartDetailFromIPN, self).get(request, *args, **kwargs)
|
|
|
|
|
|
|
|
|
2019-05-04 08:46:57 +00:00
|
|
|
class PartQRCode(QRCodeView):
|
|
|
|
""" View for displaying a QR code for a Part object """
|
|
|
|
|
2020-02-11 23:25:46 +00:00
|
|
|
ajax_form_title = _("Part QR Code")
|
2019-05-04 08:46:57 +00:00
|
|
|
|
|
|
|
def get_qr_data(self):
|
|
|
|
""" Generate QR code data for the Part """
|
|
|
|
|
|
|
|
try:
|
|
|
|
part = Part.objects.get(id=self.pk)
|
|
|
|
return part.format_barcode()
|
|
|
|
except Part.DoesNotExist:
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2020-02-10 11:57:36 +00:00
|
|
|
class PartImageUpload(AjaxUpdateView):
|
|
|
|
""" View for uploading a new Part image """
|
2018-04-29 02:25:07 +00:00
|
|
|
|
|
|
|
model = Part
|
|
|
|
ajax_template_name = 'modal_form.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Upload Part Image')
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.PartImageForm
|
2018-04-29 02:25:07 +00:00
|
|
|
|
2018-05-03 13:40:27 +00:00
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-10 11:10:06 +00:00
|
|
|
'success': _('Updated part image'),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class PartImageSelect(AjaxUpdateView):
|
|
|
|
""" View for selecting Part image from existing images. """
|
|
|
|
|
|
|
|
model = Part
|
|
|
|
ajax_template_name = 'part/select_image.html'
|
|
|
|
ajax_form_title = _('Select Part Image')
|
|
|
|
|
2020-02-10 11:57:36 +00:00
|
|
|
fields = [
|
|
|
|
'image',
|
|
|
|
]
|
|
|
|
|
2020-02-10 12:48:45 +00:00
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
part = self.get_object()
|
|
|
|
form = self.get_form()
|
|
|
|
|
|
|
|
img = request.POST.get('image', '')
|
|
|
|
|
|
|
|
img = os.path.basename(img)
|
|
|
|
|
|
|
|
data = {}
|
|
|
|
|
|
|
|
if img:
|
|
|
|
img_path = os.path.join(settings.MEDIA_ROOT, 'part_images', img)
|
|
|
|
|
|
|
|
# Ensure that the image already exists
|
|
|
|
if os.path.exists(img_path):
|
|
|
|
|
|
|
|
part.image = os.path.join('part_images', img)
|
|
|
|
part.save()
|
|
|
|
|
|
|
|
data['success'] = _('Updated part image')
|
|
|
|
|
|
|
|
if 'success' not in data:
|
|
|
|
data['error'] = _('Part image not found')
|
|
|
|
|
|
|
|
return self.renderJsonResponse(request, form, data)
|
2018-05-03 13:40:27 +00:00
|
|
|
|
2018-04-29 02:25:07 +00:00
|
|
|
|
2018-04-26 13:28:27 +00:00
|
|
|
class PartEdit(AjaxUpdateView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" View for editing Part object """
|
|
|
|
|
2018-04-14 15:18:12 +00:00
|
|
|
model = Part
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditPartForm
|
2018-04-26 13:28:27 +00:00
|
|
|
ajax_template_name = 'modal_form.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Edit Part Properties')
|
2019-04-16 11:25:20 +00:00
|
|
|
context_object_name = 'part'
|
2018-04-14 06:26:26 +00:00
|
|
|
|
2019-04-28 13:50:35 +00:00
|
|
|
def get_form(self):
|
|
|
|
""" Create form for Part editing.
|
|
|
|
Overrides default get_form() method to limit the choices
|
|
|
|
for the 'default_supplier' field to SupplierParts that reference this part
|
|
|
|
"""
|
|
|
|
|
|
|
|
form = super(AjaxUpdateView, self).get_form()
|
|
|
|
|
|
|
|
part = self.get_object()
|
|
|
|
|
|
|
|
form.fields['default_supplier'].queryset = SupplierPart.objects.filter(part=part)
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
|
|
|
|
2019-05-12 06:27:50 +00:00
|
|
|
class BomValidate(AjaxUpdateView):
|
|
|
|
""" Modal form view for validating a part BOM """
|
|
|
|
|
|
|
|
model = Part
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _("Validate BOM")
|
2019-05-12 06:27:50 +00:00
|
|
|
ajax_template_name = 'part/bom_validate.html'
|
|
|
|
context_object_name = 'part'
|
|
|
|
form_class = part_forms.BomValidateForm
|
|
|
|
|
|
|
|
def get_context(self):
|
|
|
|
return {
|
|
|
|
'part': self.get_object(),
|
|
|
|
}
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
form = self.get_form()
|
|
|
|
|
|
|
|
return self.renderJsonResponse(request, form, context=self.get_context())
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
form = self.get_form()
|
|
|
|
part = self.get_object()
|
|
|
|
|
|
|
|
confirmed = str2bool(request.POST.get('validate', False))
|
|
|
|
|
|
|
|
if confirmed:
|
|
|
|
part.validate_bom(request.user)
|
|
|
|
else:
|
|
|
|
form.errors['validate'] = ['Confirm that the BOM is valid']
|
|
|
|
|
|
|
|
data = {
|
|
|
|
'form_valid': confirmed
|
|
|
|
}
|
|
|
|
|
|
|
|
return self.renderJsonResponse(request, form, data, context=self.get_context())
|
|
|
|
|
|
|
|
|
2019-07-02 09:02:19 +00:00
|
|
|
class BomUpload(FormView):
|
2019-05-24 13:56:36 +00:00
|
|
|
""" View for uploading a BOM file, and handling BOM data importing.
|
|
|
|
|
|
|
|
The BOM upload process is as follows:
|
|
|
|
|
|
|
|
1. (Client) Select and upload BOM file
|
|
|
|
2. (Server) Verify that supplied file is a file compatible with tablib library
|
|
|
|
3. (Server) Introspect data file, try to find sensible columns / values / etc
|
|
|
|
4. (Server) Send suggestions back to the client
|
|
|
|
5. (Client) Makes choices based on suggestions:
|
|
|
|
- Accept automatic matching to parts found in database
|
|
|
|
- Accept suggestions for 'partial' or 'fuzzy' matches
|
|
|
|
- Create new parts in case of parts not being available
|
|
|
|
6. (Client) Sends updated dataset back to server
|
|
|
|
7. (Server) Check POST data for validity, sanity checking, etc.
|
|
|
|
8. (Server) Respond to POST request
|
|
|
|
- If data are valid, proceed to 9.
|
|
|
|
- If data not valid, return to 4.
|
|
|
|
9. (Server) Send confirmation form to user
|
|
|
|
- Display the actions which will occur
|
|
|
|
- Provide final "CONFIRM" button
|
|
|
|
10. (Client) Confirm final changes
|
|
|
|
11. (Server) Apply changes to database, update BOM items.
|
|
|
|
|
|
|
|
During these steps, data are passed between the server/client as JSON objects.
|
|
|
|
"""
|
|
|
|
|
2019-07-10 03:38:14 +00:00
|
|
|
template_name = 'part/bom_upload/upload_file.html'
|
2019-07-02 09:20:45 +00:00
|
|
|
|
|
|
|
# Context data passed to the forms (initially empty, extracted from uploaded file)
|
|
|
|
bom_headers = []
|
|
|
|
bom_columns = []
|
|
|
|
bom_rows = []
|
|
|
|
missing_columns = []
|
2019-07-07 01:22:01 +00:00
|
|
|
allowed_parts = []
|
2019-07-02 09:20:45 +00:00
|
|
|
|
2019-07-02 09:02:19 +00:00
|
|
|
def get_success_url(self):
|
|
|
|
part = self.get_object()
|
|
|
|
return reverse('upload-bom', kwargs={'pk': part.id})
|
2019-06-27 13:49:01 +00:00
|
|
|
|
|
|
|
def get_form_class(self):
|
|
|
|
|
2019-08-14 03:54:16 +00:00
|
|
|
# Default form is the starting point
|
|
|
|
return part_forms.BomUploadSelectFile
|
2019-06-27 11:17:33 +00:00
|
|
|
|
2019-07-02 09:02:19 +00:00
|
|
|
def get_context_data(self, *args, **kwargs):
|
|
|
|
|
|
|
|
ctx = super().get_context_data(*args, **kwargs)
|
|
|
|
|
2019-07-06 23:50:59 +00:00
|
|
|
# Give each row item access to the column it is in
|
|
|
|
# This provides for much simpler template rendering
|
|
|
|
|
|
|
|
rows = []
|
|
|
|
for row in self.bom_rows:
|
|
|
|
row_data = row['data']
|
|
|
|
|
|
|
|
data = []
|
|
|
|
|
|
|
|
for idx, item in enumerate(row_data):
|
|
|
|
|
|
|
|
data.append({
|
|
|
|
'cell': item,
|
|
|
|
'idx': idx,
|
2019-07-09 09:21:54 +00:00
|
|
|
'column': self.bom_columns[idx]
|
2019-07-06 23:50:59 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
rows.append({
|
|
|
|
'index': row.get('index', -1),
|
|
|
|
'data': data,
|
2020-08-17 19:10:24 +00:00
|
|
|
'part_match': row.get('part_match', None),
|
2019-07-09 23:22:38 +00:00
|
|
|
'part_options': row.get('part_options', self.allowed_parts),
|
2019-07-07 03:06:59 +00:00
|
|
|
|
|
|
|
# User-input (passed between client and server)
|
2019-07-06 23:50:59 +00:00
|
|
|
'quantity': row.get('quantity', None),
|
2019-07-07 03:06:59 +00:00
|
|
|
'description': row.get('description', ''),
|
2019-07-07 23:33:44 +00:00
|
|
|
'part_name': row.get('part_name', ''),
|
2019-07-07 03:06:59 +00:00
|
|
|
'part': row.get('part', None),
|
|
|
|
'reference': row.get('reference', ''),
|
|
|
|
'notes': row.get('notes', ''),
|
2019-07-09 09:45:36 +00:00
|
|
|
'errors': row.get('errors', ''),
|
2019-07-06 23:50:59 +00:00
|
|
|
})
|
|
|
|
|
2019-07-02 09:02:19 +00:00
|
|
|
ctx['part'] = self.part
|
2019-07-03 10:08:49 +00:00
|
|
|
ctx['bom_headers'] = BomUploadManager.HEADERS
|
2019-07-02 09:20:45 +00:00
|
|
|
ctx['bom_columns'] = self.bom_columns
|
2019-07-06 23:50:59 +00:00
|
|
|
ctx['bom_rows'] = rows
|
2019-07-02 09:20:45 +00:00
|
|
|
ctx['missing_columns'] = self.missing_columns
|
2019-07-07 01:22:01 +00:00
|
|
|
ctx['allowed_parts_list'] = self.allowed_parts
|
2019-05-24 13:56:36 +00:00
|
|
|
|
2019-06-27 12:16:24 +00:00
|
|
|
return ctx
|
2019-05-24 13:56:36 +00:00
|
|
|
|
2019-07-07 01:22:01 +00:00
|
|
|
def getAllowedParts(self):
|
|
|
|
""" Return a queryset of parts which are allowed to be added to this BOM.
|
|
|
|
"""
|
|
|
|
|
|
|
|
return self.part.get_allowed_bom_items()
|
|
|
|
|
2019-05-24 13:56:36 +00:00
|
|
|
def get(self, request, *args, **kwargs):
|
2019-06-27 14:15:23 +00:00
|
|
|
""" Perform the initial 'GET' request.
|
2019-05-24 13:56:36 +00:00
|
|
|
|
|
|
|
Initially returns a form for file upload """
|
|
|
|
|
2019-05-24 14:18:04 +00:00
|
|
|
self.request = request
|
|
|
|
|
2019-05-24 13:56:36 +00:00
|
|
|
# A valid Part object must be supplied. This is the 'parent' part for the BOM
|
2019-06-27 12:16:24 +00:00
|
|
|
self.part = get_object_or_404(Part, pk=self.kwargs['pk'])
|
2019-05-24 13:56:36 +00:00
|
|
|
|
2019-06-27 12:16:24 +00:00
|
|
|
self.form = self.get_form()
|
2019-05-24 13:56:36 +00:00
|
|
|
|
2019-07-02 09:02:19 +00:00
|
|
|
form_class = self.get_form_class()
|
|
|
|
form = self.get_form(form_class)
|
|
|
|
return self.render_to_response(self.get_context_data(form=form))
|
2019-05-24 13:56:36 +00:00
|
|
|
|
2019-06-27 13:49:01 +00:00
|
|
|
def handleBomFileUpload(self):
|
2019-06-28 09:40:27 +00:00
|
|
|
""" Process a BOM file upload form.
|
|
|
|
|
|
|
|
This function validates that the uploaded file was valid,
|
|
|
|
and contains tabulated data that can be extracted.
|
|
|
|
If the file does not satisfy these requirements,
|
|
|
|
the "upload file" form is again shown to the user.
|
|
|
|
"""
|
2019-06-27 13:49:01 +00:00
|
|
|
|
|
|
|
bom_file = self.request.FILES.get('bom_file', None)
|
|
|
|
|
|
|
|
manager = None
|
|
|
|
bom_file_valid = False
|
|
|
|
|
|
|
|
if bom_file is None:
|
|
|
|
self.form.errors['bom_file'] = [_('No BOM file provided')]
|
|
|
|
else:
|
|
|
|
# Create a BomUploadManager object - will perform initial data validation
|
|
|
|
# (and raise a ValidationError if there is something wrong with the file)
|
|
|
|
try:
|
|
|
|
manager = BomUploadManager(bom_file)
|
|
|
|
bom_file_valid = True
|
|
|
|
except ValidationError as e:
|
|
|
|
errors = e.error_dict
|
|
|
|
|
2019-06-27 14:15:23 +00:00
|
|
|
for k, v in errors.items():
|
2019-06-27 13:49:01 +00:00
|
|
|
self.form.errors[k] = v
|
|
|
|
|
2019-07-02 09:20:45 +00:00
|
|
|
if bom_file_valid:
|
2019-06-27 13:49:01 +00:00
|
|
|
# BOM file is valid? Proceed to the next step!
|
2019-08-14 03:54:16 +00:00
|
|
|
form = None
|
2019-07-02 09:20:45 +00:00
|
|
|
self.template_name = 'part/bom_upload/select_fields.html'
|
2019-06-28 09:40:27 +00:00
|
|
|
|
2019-07-02 09:20:45 +00:00
|
|
|
self.extractDataFromFile(manager)
|
2019-06-27 13:49:01 +00:00
|
|
|
else:
|
|
|
|
form = self.form
|
|
|
|
|
2019-07-02 09:07:59 +00:00
|
|
|
return self.render_to_response(self.get_context_data(form=form))
|
2019-06-27 12:16:24 +00:00
|
|
|
|
2019-07-07 03:06:59 +00:00
|
|
|
def getColumnIndex(self, name):
|
2019-07-10 03:38:14 +00:00
|
|
|
""" Return the index of the column with the given name.
|
2019-07-07 03:06:59 +00:00
|
|
|
It named column is not found, return -1
|
2019-07-03 11:45:56 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
2019-07-07 03:06:59 +00:00
|
|
|
idx = list(self.column_selections.values()).index(name)
|
2019-07-03 11:45:56 +00:00
|
|
|
except ValueError:
|
2019-07-07 03:06:59 +00:00
|
|
|
idx = -1
|
2019-07-03 11:45:56 +00:00
|
|
|
|
2019-07-07 03:06:59 +00:00
|
|
|
return idx
|
|
|
|
|
|
|
|
def preFillSelections(self):
|
2019-07-07 23:33:44 +00:00
|
|
|
""" Once data columns have been selected, attempt to pre-select the proper data from the database.
|
|
|
|
This function is called once the field selection has been validated.
|
|
|
|
The pre-fill data are then passed through to the part selection form.
|
2019-07-07 03:06:59 +00:00
|
|
|
"""
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Fields prefixed with "Part_" can be used to do "smart matching" against Part objects in the database
|
|
|
|
k_idx = self.getColumnIndex('Part_ID')
|
|
|
|
p_idx = self.getColumnIndex('Part_Name')
|
|
|
|
i_idx = self.getColumnIndex('Part_IPN')
|
|
|
|
|
2019-07-07 03:06:59 +00:00
|
|
|
q_idx = self.getColumnIndex('Quantity')
|
|
|
|
r_idx = self.getColumnIndex('Reference')
|
2020-08-18 04:01:01 +00:00
|
|
|
o_idx = self.getColumnIndex('Overage')
|
|
|
|
n_idx = self.getColumnIndex('Note')
|
2019-07-07 01:44:17 +00:00
|
|
|
|
2019-07-03 11:45:56 +00:00
|
|
|
for row in self.bom_rows:
|
2020-08-18 04:01:01 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
Iterate through each row in the uploaded data,
|
|
|
|
and see if we can match the row to a "Part" object in the database.
|
|
|
|
|
|
|
|
There are three potential ways to match, based on the uploaded data:
|
|
|
|
|
|
|
|
a) Use the PK (primary key) field for the part, uploaded in the "Part_ID" field
|
|
|
|
b) Use the IPN (internal part number) field for the part, uploaded in the "Part_IPN" field
|
|
|
|
c) Use the name of the part, uploaded in the "Part_Name" field
|
|
|
|
|
|
|
|
Notes:
|
|
|
|
- If using the Part_ID field, we can do an exact match against the PK field
|
|
|
|
- If using the Part_IPN field, we can do an exact match against the IPN field
|
|
|
|
- If using the Part_Name field, we can use fuzzy string matching to match "close" values
|
|
|
|
|
|
|
|
We also extract other information from the row, for the other non-matched fields:
|
|
|
|
- Quantity
|
|
|
|
- Reference
|
|
|
|
- Overage
|
|
|
|
- Note
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Initially use a quantity of zero
|
|
|
|
quantity = Decimal(0)
|
|
|
|
|
|
|
|
# Initially we do not have a part to reference
|
|
|
|
exact_match_part = None
|
2019-07-03 11:45:56 +00:00
|
|
|
|
2020-08-18 04:01:40 +00:00
|
|
|
# A list of potential Part matches
|
2020-08-18 04:01:01 +00:00
|
|
|
part_options = self.allowed_parts
|
2019-07-03 11:45:56 +00:00
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Check if there is a column corresponding to "quantity"
|
2019-07-03 11:45:56 +00:00
|
|
|
if q_idx >= 0:
|
|
|
|
q_val = row['data'][q_idx]
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
if q_val:
|
|
|
|
try:
|
|
|
|
# Attempt to extract a valid quantity from the field
|
|
|
|
quantity = Decimal(q_val)
|
|
|
|
except (ValueError, InvalidOperation):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Store the 'quantity' value
|
|
|
|
row['quantity'] = quantity
|
|
|
|
|
|
|
|
# Check if there is a column corresponding to "PK"
|
|
|
|
if k_idx >= 0:
|
|
|
|
pk = row['data'][k_idx]
|
|
|
|
|
|
|
|
if pk:
|
|
|
|
try:
|
|
|
|
# Attempt Part lookup based on PK value
|
|
|
|
exact_match_part = Part.objects.get(pk=pk)
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
|
|
exact_match_part = None
|
2019-07-03 11:45:56 +00:00
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Check if there is a column corresponding to "Part Name"
|
2019-07-07 01:44:17 +00:00
|
|
|
if p_idx >= 0:
|
2019-07-07 23:33:44 +00:00
|
|
|
part_name = row['data'][p_idx]
|
|
|
|
|
|
|
|
row['part_name'] = part_name
|
2019-07-07 01:44:17 +00:00
|
|
|
|
|
|
|
matches = []
|
|
|
|
|
|
|
|
for part in self.allowed_parts:
|
2019-07-07 23:33:44 +00:00
|
|
|
ratio = fuzz.partial_ratio(part.name + part.description, part_name)
|
2019-07-07 01:44:17 +00:00
|
|
|
matches.append({'part': part, 'match': ratio})
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Sort matches by the 'strength' of the match ratio
|
2019-07-07 01:44:17 +00:00
|
|
|
if len(matches) > 0:
|
|
|
|
matches = sorted(matches, key=lambda item: item['match'], reverse=True)
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
part_options = [m['part'] for m in matches]
|
|
|
|
|
|
|
|
# Check if there is a column corresponding to "Part IPN"
|
2020-08-17 21:35:38 +00:00
|
|
|
if i_idx >= 0:
|
|
|
|
row['part_ipn'] = row['data'][i_idx]
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Check if there is a column corresponding to "Overage" field
|
|
|
|
if o_idx >= 0:
|
|
|
|
row['overage'] = row['data'][o_idx]
|
2019-07-07 03:06:59 +00:00
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Check if there is a column corresponding to "Reference" field
|
2019-07-07 03:06:59 +00:00
|
|
|
if r_idx >= 0:
|
|
|
|
row['reference'] = row['data'][r_idx]
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Check if there is a column corresponding to "Note" field
|
2019-07-07 03:06:59 +00:00
|
|
|
if n_idx >= 0:
|
2020-08-18 04:01:01 +00:00
|
|
|
row['note'] = row['data'][n_idx]
|
|
|
|
|
|
|
|
# Supply list of part options for each row, sorted by how closely they match the part name
|
|
|
|
row['part_options'] = part_options
|
2020-08-17 19:10:24 +00:00
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
# Unless found, the 'part_match' is blank
|
|
|
|
row['part_match'] = None
|
2020-08-17 21:35:38 +00:00
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
if exact_match_part:
|
|
|
|
# If there is an exact match based on PK, use that
|
|
|
|
row['part_match'] = exact_match_part
|
|
|
|
else:
|
|
|
|
# Otherwise, check to see if there is a matching IPN
|
|
|
|
try:
|
|
|
|
if row['part_ipn']:
|
|
|
|
part_matches = [part for part in self.allowed_parts if row['part_ipn'].lower() == part.IPN.lower()]
|
|
|
|
|
|
|
|
# Check for single match
|
|
|
|
if len(part_matches) == 1:
|
|
|
|
row['part_match'] = part_matches[0]
|
2020-08-17 21:35:38 +00:00
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
continue
|
|
|
|
except KeyError:
|
|
|
|
pass
|
2020-08-17 21:35:38 +00:00
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
print(row, row['part_match'], len(row['part_options']))
|
2019-07-03 11:45:56 +00:00
|
|
|
|
2019-07-02 09:20:45 +00:00
|
|
|
def extractDataFromFile(self, bom):
|
|
|
|
""" Read data from the BOM file """
|
|
|
|
|
|
|
|
self.bom_columns = bom.columns()
|
|
|
|
self.bom_rows = bom.rows()
|
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
def getTableDataFromPost(self):
|
|
|
|
""" Extract table cell data from POST request.
|
|
|
|
These data are used to maintain state between sessions.
|
|
|
|
|
|
|
|
Table data keys are as follows:
|
|
|
|
|
|
|
|
col_name_<idx> - Column name at idx as provided in the uploaded file
|
|
|
|
col_guess_<idx> - Column guess at idx as selected in the BOM
|
|
|
|
row_<x>_col<y> - Cell data as provided in the uploaded file
|
2019-07-02 09:20:45 +00:00
|
|
|
|
2019-06-28 09:40:27 +00:00
|
|
|
"""
|
2019-05-24 14:18:04 +00:00
|
|
|
|
2019-06-29 09:56:04 +00:00
|
|
|
# Map the columns
|
2019-07-03 11:45:56 +00:00
|
|
|
self.column_names = {}
|
|
|
|
self.column_selections = {}
|
2019-06-28 09:58:56 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
self.row_data = {}
|
2019-06-28 09:58:56 +00:00
|
|
|
|
|
|
|
for item in self.request.POST:
|
|
|
|
value = self.request.POST[item]
|
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
# Column names as passed as col_name_<idx> where idx is an integer
|
|
|
|
|
2019-06-28 09:58:56 +00:00
|
|
|
# Extract the column names
|
|
|
|
if item.startswith('col_name_'):
|
2019-07-09 09:21:54 +00:00
|
|
|
try:
|
|
|
|
col_id = int(item.replace('col_name_', ''))
|
|
|
|
except ValueError:
|
|
|
|
continue
|
2019-06-28 09:58:56 +00:00
|
|
|
col_name = value
|
|
|
|
|
2019-07-03 11:45:56 +00:00
|
|
|
self.column_names[col_id] = col_name
|
2019-06-28 09:58:56 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
# Extract the column selections (in the 'select fields' view)
|
|
|
|
if item.startswith('col_guess_'):
|
|
|
|
|
|
|
|
try:
|
|
|
|
col_id = int(item.replace('col_guess_', ''))
|
|
|
|
except ValueError:
|
|
|
|
continue
|
2019-06-28 09:58:56 +00:00
|
|
|
|
|
|
|
col_name = value
|
|
|
|
|
2019-07-03 11:45:56 +00:00
|
|
|
self.column_selections[col_id] = value
|
2019-06-28 09:58:56 +00:00
|
|
|
|
|
|
|
# Extract the row data
|
|
|
|
if item.startswith('row_'):
|
|
|
|
# Item should be of the format row_<r>_col_<c>
|
|
|
|
s = item.split('_')
|
|
|
|
|
|
|
|
if len(s) < 4:
|
|
|
|
continue
|
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
# Ignore row/col IDs which are not correct numeric values
|
|
|
|
try:
|
|
|
|
row_id = int(s[1])
|
|
|
|
col_id = int(s[3])
|
|
|
|
except ValueError:
|
|
|
|
continue
|
2019-06-28 09:58:56 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
if row_id not in self.row_data:
|
|
|
|
self.row_data[row_id] = {}
|
|
|
|
|
|
|
|
self.row_data[row_id][col_id] = value
|
|
|
|
|
|
|
|
self.col_ids = sorted(self.column_names.keys())
|
|
|
|
|
|
|
|
# Re-construct the data table
|
|
|
|
self.bom_rows = []
|
|
|
|
|
|
|
|
for row_idx in sorted(self.row_data.keys()):
|
|
|
|
row = self.row_data[row_idx]
|
|
|
|
items = []
|
|
|
|
|
|
|
|
for col_idx in sorted(row.keys()):
|
2019-06-28 09:58:56 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
value = row[col_idx]
|
|
|
|
items.append(value)
|
2019-06-28 10:16:17 +00:00
|
|
|
|
2019-07-09 09:45:36 +00:00
|
|
|
self.bom_rows.append({
|
|
|
|
'index': row_idx,
|
|
|
|
'data': items,
|
|
|
|
'errors': {},
|
|
|
|
})
|
2019-07-09 09:21:54 +00:00
|
|
|
|
|
|
|
# Construct the column data
|
2019-07-03 10:08:49 +00:00
|
|
|
self.bom_columns = []
|
2019-06-28 10:16:17 +00:00
|
|
|
|
2019-07-03 11:19:31 +00:00
|
|
|
# Track any duplicate column selections
|
2019-07-09 09:21:54 +00:00
|
|
|
self.duplicates = False
|
|
|
|
|
|
|
|
for col in self.col_ids:
|
2019-07-03 11:19:31 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
if col in self.column_selections:
|
|
|
|
guess = self.column_selections[col]
|
|
|
|
else:
|
|
|
|
guess = None
|
2019-06-28 10:16:17 +00:00
|
|
|
|
|
|
|
header = ({
|
2019-07-03 11:45:56 +00:00
|
|
|
'name': self.column_names[col],
|
2019-07-09 09:21:54 +00:00
|
|
|
'guess': guess
|
2019-06-28 10:16:17 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
if guess:
|
2019-07-03 11:45:56 +00:00
|
|
|
n = list(self.column_selections.values()).count(self.column_selections[col])
|
2019-06-28 10:16:17 +00:00
|
|
|
if n > 1:
|
|
|
|
header['duplicate'] = True
|
2019-07-09 09:21:54 +00:00
|
|
|
self.duplicates = True
|
2019-06-28 10:16:17 +00:00
|
|
|
|
2019-07-03 10:08:49 +00:00
|
|
|
self.bom_columns.append(header)
|
2019-06-28 10:16:17 +00:00
|
|
|
|
|
|
|
# Are there any missing columns?
|
2019-07-03 10:08:49 +00:00
|
|
|
self.missing_columns = []
|
2019-06-28 10:16:17 +00:00
|
|
|
|
|
|
|
for col in BomUploadManager.REQUIRED_HEADERS:
|
2019-07-03 11:45:56 +00:00
|
|
|
if col not in self.column_selections.values():
|
2019-07-03 10:08:49 +00:00
|
|
|
self.missing_columns.append(col)
|
2019-06-28 10:16:17 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
def handleFieldSelection(self):
|
|
|
|
""" Handle the output of the field selection form.
|
|
|
|
Here the user is presented with the raw data and must select the
|
|
|
|
column names and which rows to process.
|
|
|
|
"""
|
2019-06-28 10:21:21 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
# Extract POST data
|
|
|
|
self.getTableDataFromPost()
|
2019-06-28 10:21:21 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
valid = len(self.missing_columns) == 0 and not self.duplicates
|
2019-07-03 11:19:31 +00:00
|
|
|
|
|
|
|
if valid:
|
2019-07-03 11:45:56 +00:00
|
|
|
# Try to extract meaningful data
|
|
|
|
self.preFillSelections()
|
2019-07-10 03:38:14 +00:00
|
|
|
self.template_name = 'part/bom_upload/select_parts.html'
|
2019-07-03 11:19:31 +00:00
|
|
|
else:
|
|
|
|
self.template_name = 'part/bom_upload/select_fields.html'
|
2019-06-28 10:21:21 +00:00
|
|
|
|
2019-08-14 03:54:16 +00:00
|
|
|
return self.render_to_response(self.get_context_data(form=None))
|
2019-05-24 14:18:04 +00:00
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
def handlePartSelection(self):
|
|
|
|
|
2019-07-09 09:45:36 +00:00
|
|
|
# Extract basic table data from POST request
|
2019-07-09 09:21:54 +00:00
|
|
|
self.getTableDataFromPost()
|
|
|
|
|
2019-07-09 09:45:36 +00:00
|
|
|
# Keep track of the parts that have been selected
|
|
|
|
parts = {}
|
|
|
|
|
|
|
|
# Extract other data (part selections, etc)
|
|
|
|
for key in self.request.POST:
|
|
|
|
value = self.request.POST[key]
|
|
|
|
|
|
|
|
# Extract quantity from each row
|
|
|
|
if key.startswith('quantity_'):
|
|
|
|
try:
|
|
|
|
row_id = int(key.replace('quantity_', ''))
|
|
|
|
|
|
|
|
row = self.getRowByIndex(row_id)
|
|
|
|
|
|
|
|
if row is None:
|
|
|
|
continue
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
q = Decimal(1)
|
2019-07-09 09:45:36 +00:00
|
|
|
|
|
|
|
try:
|
2020-08-18 04:01:01 +00:00
|
|
|
q = Decimal(value)
|
|
|
|
if q < 0:
|
2019-07-09 09:45:36 +00:00
|
|
|
row['errors']['quantity'] = _('Quantity must be greater than zero')
|
2020-08-18 04:01:01 +00:00
|
|
|
|
|
|
|
if 'part' in row.keys():
|
|
|
|
if row['part'].trackable:
|
|
|
|
# Trackable parts must use integer quantities
|
|
|
|
if not q == int(q):
|
|
|
|
row['errors']['quantity'] = _('Quantity must be integer value for trackable parts')
|
|
|
|
|
|
|
|
except (ValueError, InvalidOperation):
|
2019-07-09 09:45:36 +00:00
|
|
|
row['errors']['quantity'] = _('Enter a valid quantity')
|
|
|
|
|
|
|
|
row['quantity'] = q
|
|
|
|
|
|
|
|
except ValueError:
|
|
|
|
continue
|
|
|
|
|
2019-07-09 23:22:38 +00:00
|
|
|
# Extract part from each row
|
|
|
|
if key.startswith('part_'):
|
2020-08-18 04:01:01 +00:00
|
|
|
|
2019-07-09 23:22:38 +00:00
|
|
|
try:
|
|
|
|
row_id = int(key.replace('part_', ''))
|
|
|
|
|
|
|
|
row = self.getRowByIndex(row_id)
|
|
|
|
|
|
|
|
if row is None:
|
|
|
|
continue
|
|
|
|
except ValueError:
|
|
|
|
# Row ID non integer value
|
|
|
|
continue
|
|
|
|
|
|
|
|
try:
|
|
|
|
part_id = int(value)
|
|
|
|
part = Part.objects.get(id=part_id)
|
|
|
|
except ValueError:
|
|
|
|
row['errors']['part'] = _('Select valid part')
|
|
|
|
continue
|
|
|
|
except Part.DoesNotExist:
|
|
|
|
row['errors']['part'] = _('Select valid part')
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Keep track of how many of each part we have seen
|
|
|
|
if part_id in parts:
|
|
|
|
parts[part_id]['quantity'] += 1
|
|
|
|
row['errors']['part'] = _('Duplicate part selected')
|
|
|
|
else:
|
|
|
|
parts[part_id] = {
|
|
|
|
'part': part,
|
|
|
|
'quantity': 1,
|
|
|
|
}
|
|
|
|
|
|
|
|
row['part'] = part
|
|
|
|
|
2020-08-18 04:01:01 +00:00
|
|
|
if part.trackable:
|
|
|
|
# For trackable parts, ensure the quantity is an integer value!
|
|
|
|
if 'quantity' in row.keys():
|
|
|
|
q = row['quantity']
|
|
|
|
|
|
|
|
if not q == int(q):
|
|
|
|
row['errors']['quantity'] = _('Quantity must be integer value for trackable parts')
|
|
|
|
|
2019-07-09 23:40:04 +00:00
|
|
|
# Extract other fields which do not require further validation
|
|
|
|
for field in ['reference', 'notes']:
|
|
|
|
if key.startswith(field + '_'):
|
|
|
|
try:
|
|
|
|
row_id = int(key.replace(field + '_', ''))
|
|
|
|
|
|
|
|
row = self.getRowByIndex(row_id)
|
|
|
|
|
|
|
|
if row:
|
|
|
|
row[field] = value
|
|
|
|
except:
|
|
|
|
continue
|
|
|
|
|
2019-07-09 23:22:38 +00:00
|
|
|
# Are there any errors after form handling?
|
|
|
|
valid = True
|
|
|
|
|
|
|
|
for row in self.bom_rows:
|
2019-07-10 02:27:19 +00:00
|
|
|
# Has a part been selected for the given row?
|
2020-08-18 04:17:59 +00:00
|
|
|
part = row.get('part', None)
|
|
|
|
|
|
|
|
if part is None:
|
2019-07-10 02:27:19 +00:00
|
|
|
row['errors']['part'] = _('Select a part')
|
2020-08-18 04:17:59 +00:00
|
|
|
else:
|
|
|
|
# Will the selected part result in a recursive BOM?
|
|
|
|
try:
|
|
|
|
part.checkAddToBOM(self.part)
|
|
|
|
except ValidationError:
|
|
|
|
row['errors']['part'] = _('Selected part creates a circular BOM')
|
2019-07-10 02:27:19 +00:00
|
|
|
|
|
|
|
# Has a quantity been specified?
|
|
|
|
if row.get('quantity', None) is None:
|
|
|
|
row['errors']['quantity'] = _('Specify quantity')
|
|
|
|
|
2019-07-09 23:22:38 +00:00
|
|
|
errors = row.get('errors', [])
|
|
|
|
|
|
|
|
if len(errors) > 0:
|
|
|
|
valid = False
|
|
|
|
|
2019-07-09 09:21:54 +00:00
|
|
|
self.template_name = 'part/bom_upload/select_parts.html'
|
|
|
|
|
2019-07-10 02:04:24 +00:00
|
|
|
ctx = self.get_context_data(form=None)
|
|
|
|
|
|
|
|
if valid:
|
2019-07-10 02:27:19 +00:00
|
|
|
self.part.clear_bom()
|
|
|
|
|
|
|
|
# Generate new BOM items
|
|
|
|
for row in self.bom_rows:
|
|
|
|
part = row.get('part')
|
|
|
|
quantity = row.get('quantity')
|
2019-07-10 03:30:51 +00:00
|
|
|
reference = row.get('reference', '')
|
|
|
|
notes = row.get('notes', '')
|
2019-07-10 02:27:19 +00:00
|
|
|
|
|
|
|
# Create a new BOM item!
|
|
|
|
item = BomItem(
|
|
|
|
part=self.part,
|
|
|
|
sub_part=part,
|
|
|
|
quantity=quantity,
|
|
|
|
reference=reference,
|
|
|
|
note=notes
|
|
|
|
)
|
|
|
|
|
|
|
|
item.save()
|
|
|
|
|
|
|
|
# Redirect to the BOM view
|
|
|
|
return HttpResponseRedirect(reverse('part-bom', kwargs={'pk': self.part.id}))
|
2019-07-10 02:04:24 +00:00
|
|
|
else:
|
|
|
|
ctx['form_errors'] = True
|
|
|
|
|
|
|
|
return self.render_to_response(ctx)
|
2019-07-09 09:21:54 +00:00
|
|
|
|
2019-07-09 09:45:36 +00:00
|
|
|
def getRowByIndex(self, idx):
|
|
|
|
|
|
|
|
for row in self.bom_rows:
|
|
|
|
if row['index'] == idx:
|
|
|
|
return row
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2019-05-24 13:56:36 +00:00
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
""" Perform the various 'POST' requests required.
|
|
|
|
"""
|
|
|
|
|
2019-05-24 14:18:04 +00:00
|
|
|
self.request = request
|
|
|
|
|
2019-06-27 12:16:24 +00:00
|
|
|
self.part = get_object_or_404(Part, pk=self.kwargs['pk'])
|
2019-07-07 01:22:01 +00:00
|
|
|
self.allowed_parts = self.getAllowedParts()
|
2019-07-02 09:07:59 +00:00
|
|
|
self.form = self.get_form(self.get_form_class())
|
2019-06-27 12:16:24 +00:00
|
|
|
|
2019-05-24 14:18:04 +00:00
|
|
|
# Did the user POST a file named bom_file?
|
2019-06-27 13:49:01 +00:00
|
|
|
|
|
|
|
form_step = request.POST.get('form_step', None)
|
2019-05-24 13:56:36 +00:00
|
|
|
|
2019-06-27 13:49:01 +00:00
|
|
|
if form_step == 'select_file':
|
|
|
|
return self.handleBomFileUpload()
|
|
|
|
elif form_step == 'select_fields':
|
|
|
|
return self.handleFieldSelection()
|
2019-07-09 09:21:54 +00:00
|
|
|
elif form_step == 'select_parts':
|
|
|
|
return self.handlePartSelection()
|
2019-05-24 14:18:04 +00:00
|
|
|
|
2019-07-02 09:07:59 +00:00
|
|
|
return self.render_to_response(self.get_context_data(form=self.form))
|
2019-06-27 12:16:24 +00:00
|
|
|
|
|
|
|
|
2019-08-06 23:52:49 +00:00
|
|
|
class PartExport(AjaxView):
|
|
|
|
""" Export a CSV file containing information on multiple parts """
|
|
|
|
|
2019-09-09 09:37:59 +00:00
|
|
|
def get_parts(self, request):
|
|
|
|
""" Extract part list from the POST parameters.
|
|
|
|
Parts can be supplied as:
|
2019-08-06 23:52:49 +00:00
|
|
|
|
2019-09-09 09:37:59 +00:00
|
|
|
- Part category
|
2019-09-09 09:59:56 +00:00
|
|
|
- List of part PK values
|
2019-09-09 09:37:59 +00:00
|
|
|
"""
|
|
|
|
|
2019-09-09 09:59:56 +00:00
|
|
|
# Filter by part category
|
|
|
|
cat_id = request.GET.get('category', None)
|
|
|
|
|
|
|
|
part_list = None
|
|
|
|
|
|
|
|
if cat_id is not None:
|
|
|
|
try:
|
|
|
|
category = PartCategory.objects.get(pk=cat_id)
|
|
|
|
part_list = category.get_parts()
|
|
|
|
except (ValueError, PartCategory.DoesNotExist):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Backup - All parts
|
|
|
|
if part_list is None:
|
|
|
|
part_list = Part.objects.all()
|
2019-09-09 09:37:59 +00:00
|
|
|
|
2019-09-09 09:59:56 +00:00
|
|
|
# Also optionally filter by explicit list of part IDs
|
2019-09-09 09:37:59 +00:00
|
|
|
part_ids = request.GET.get('parts', '')
|
|
|
|
parts = []
|
|
|
|
|
|
|
|
for pk in part_ids.split(','):
|
2019-08-06 23:52:49 +00:00
|
|
|
try:
|
2019-09-09 09:37:59 +00:00
|
|
|
parts.append(int(pk))
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if len(parts) > 0:
|
|
|
|
part_list = part_list.filter(pk__in=parts)
|
|
|
|
|
|
|
|
# Prefetch related fields to reduce DB hits
|
|
|
|
part_list = part_list.prefetch_related(
|
|
|
|
'category',
|
|
|
|
'used_in',
|
|
|
|
'builds',
|
|
|
|
'supplier_parts__purchase_order_line_items',
|
|
|
|
'stock_items__allocations',
|
2019-09-09 10:02:52 +00:00
|
|
|
)
|
2019-09-09 09:37:59 +00:00
|
|
|
|
|
|
|
return part_list
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
parts = self.get_parts(request)
|
2019-08-06 23:52:49 +00:00
|
|
|
|
2019-09-15 09:52:28 +00:00
|
|
|
dataset = PartResource().export(queryset=parts)
|
|
|
|
|
|
|
|
csv = dataset.export('csv')
|
2019-08-06 23:52:49 +00:00
|
|
|
return DownloadFile(csv, 'InvenTree_Parts.csv')
|
|
|
|
|
|
|
|
|
2019-06-27 12:15:58 +00:00
|
|
|
class BomUploadTemplate(AjaxView):
|
|
|
|
"""
|
|
|
|
Provide a BOM upload template file for download.
|
|
|
|
- Generates a template file in the provided format e.g. ?format=csv
|
|
|
|
"""
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
export_format = request.GET.get('format', 'csv')
|
|
|
|
|
|
|
|
return MakeBomTemplate(export_format)
|
2019-05-24 13:56:36 +00:00
|
|
|
|
|
|
|
|
2019-04-16 11:25:20 +00:00
|
|
|
class BomDownload(AjaxView):
|
|
|
|
"""
|
|
|
|
Provide raw download of a BOM file.
|
|
|
|
- File format should be passed as a query param e.g. ?format=csv
|
|
|
|
"""
|
|
|
|
|
2019-04-13 11:04:23 +00:00
|
|
|
model = Part
|
|
|
|
|
2019-04-13 12:22:04 +00:00
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
part = get_object_or_404(Part, pk=self.kwargs['pk'])
|
|
|
|
|
2020-02-11 10:43:17 +00:00
|
|
|
export_format = request.GET.get('file_format', 'csv')
|
2019-04-16 11:46:12 +00:00
|
|
|
|
2020-02-11 11:32:36 +00:00
|
|
|
cascade = str2bool(request.GET.get('cascade', False))
|
|
|
|
|
2020-08-20 18:53:27 +00:00
|
|
|
parameter_data = str2bool(request.GET.get('parameter_data', False))
|
|
|
|
|
|
|
|
stock_data = str2bool(request.GET.get('stock_data', False))
|
|
|
|
|
2020-08-19 04:05:16 +00:00
|
|
|
supplier_data = str2bool(request.GET.get('supplier_data', False))
|
|
|
|
|
2020-08-15 22:29:36 +00:00
|
|
|
levels = request.GET.get('levels', None)
|
|
|
|
|
|
|
|
if levels is not None:
|
|
|
|
try:
|
|
|
|
levels = int(levels)
|
|
|
|
|
|
|
|
if levels <= 0:
|
|
|
|
levels = None
|
|
|
|
|
|
|
|
except ValueError:
|
|
|
|
levels = None
|
|
|
|
|
2019-09-15 12:21:12 +00:00
|
|
|
if not IsValidBOMFormat(export_format):
|
|
|
|
export_format = 'csv'
|
2019-04-13 12:22:04 +00:00
|
|
|
|
2020-08-20 19:53:03 +00:00
|
|
|
return ExportBom(part,
|
|
|
|
fmt=export_format,
|
|
|
|
cascade=cascade,
|
|
|
|
max_levels=levels,
|
|
|
|
parameter_data=parameter_data,
|
|
|
|
stock_data=stock_data,
|
|
|
|
supplier_data=supplier_data)
|
2019-04-13 11:04:23 +00:00
|
|
|
|
|
|
|
def get_data(self):
|
|
|
|
return {
|
|
|
|
'info': 'Exported BOM'
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-02-11 10:43:17 +00:00
|
|
|
class BomExport(AjaxView):
|
|
|
|
""" Provide a simple form to allow the user to select BOM download options.
|
|
|
|
"""
|
|
|
|
|
|
|
|
model = Part
|
|
|
|
form_class = part_forms.BomExportForm
|
|
|
|
ajax_form_title = _("Export Bill of Materials")
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
2020-02-11 11:32:36 +00:00
|
|
|
return self.renderJsonResponse(request, self.form_class())
|
2020-02-11 10:43:17 +00:00
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
# Extract POSTed form data
|
|
|
|
fmt = request.POST.get('file_format', 'csv').lower()
|
|
|
|
cascade = str2bool(request.POST.get('cascading', False))
|
2020-08-15 22:29:36 +00:00
|
|
|
levels = request.POST.get('levels', None)
|
2020-08-20 18:53:27 +00:00
|
|
|
parameter_data = str2bool(request.POST.get('parameter_data', False))
|
|
|
|
stock_data = str2bool(request.POST.get('stock_data', False))
|
2020-08-19 04:05:16 +00:00
|
|
|
supplier_data = str2bool(request.POST.get('supplier_data', False))
|
2020-02-11 10:43:17 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
part = Part.objects.get(pk=self.kwargs['pk'])
|
|
|
|
except:
|
|
|
|
part = None
|
|
|
|
|
|
|
|
# Format a URL to redirect to
|
|
|
|
if part:
|
|
|
|
url = reverse('bom-download', kwargs={'pk': part.pk})
|
|
|
|
else:
|
|
|
|
url = ''
|
|
|
|
|
|
|
|
url += '?file_format=' + fmt
|
|
|
|
url += '&cascade=' + str(cascade)
|
2020-08-20 18:53:27 +00:00
|
|
|
url += '¶meter_data=' + str(parameter_data)
|
|
|
|
url += '&stock_data=' + str(stock_data)
|
2020-08-19 04:05:16 +00:00
|
|
|
url += '&supplier_data=' + str(supplier_data)
|
2020-02-11 10:43:17 +00:00
|
|
|
|
2020-08-15 22:29:36 +00:00
|
|
|
if levels:
|
|
|
|
url += '&levels=' + str(levels)
|
|
|
|
|
2020-02-11 10:43:17 +00:00
|
|
|
data = {
|
|
|
|
'form_valid': part is not None,
|
|
|
|
'url': url,
|
|
|
|
}
|
|
|
|
|
|
|
|
return self.renderJsonResponse(request, self.form_class(), data=data)
|
|
|
|
|
|
|
|
|
2018-04-26 14:06:44 +00:00
|
|
|
class PartDelete(AjaxDeleteView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" View to delete a Part object """
|
|
|
|
|
2018-04-14 22:45:50 +00:00
|
|
|
model = Part
|
2018-04-26 14:06:44 +00:00
|
|
|
ajax_template_name = 'part/partial_delete.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Confirm Part Deletion')
|
2019-04-16 11:25:20 +00:00
|
|
|
context_object_name = 'part'
|
2018-04-14 22:45:50 +00:00
|
|
|
|
|
|
|
success_url = '/part/'
|
|
|
|
|
2018-04-29 14:23:02 +00:00
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-10 11:10:06 +00:00
|
|
|
'danger': _('Part was deleted'),
|
2018-04-29 14:23:02 +00:00
|
|
|
}
|
2018-04-14 22:45:50 +00:00
|
|
|
|
2018-04-15 01:40:03 +00:00
|
|
|
|
2019-05-18 12:58:11 +00:00
|
|
|
class PartPricing(AjaxView):
|
|
|
|
""" View for inspecting part pricing information """
|
|
|
|
|
|
|
|
model = Part
|
|
|
|
ajax_template_name = "part/part_pricing.html"
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _("Part Pricing")
|
2019-05-18 12:58:11 +00:00
|
|
|
form_class = part_forms.PartPriceForm
|
|
|
|
|
|
|
|
def get_part(self):
|
|
|
|
try:
|
|
|
|
return Part.objects.get(id=self.kwargs['pk'])
|
|
|
|
except Part.DoesNotExist:
|
|
|
|
return None
|
|
|
|
|
2019-09-03 12:28:53 +00:00
|
|
|
def get_pricing(self, quantity=1, currency=None):
|
2019-05-18 12:58:11 +00:00
|
|
|
|
2019-05-19 21:53:23 +00:00
|
|
|
try:
|
|
|
|
quantity = int(quantity)
|
|
|
|
except ValueError:
|
|
|
|
quantity = 1
|
|
|
|
|
|
|
|
if quantity < 1:
|
|
|
|
quantity = 1
|
|
|
|
|
2019-09-03 12:33:50 +00:00
|
|
|
if currency is None:
|
|
|
|
# No currency selected? Try to select a default one
|
|
|
|
try:
|
|
|
|
currency = Currency.objects.get(base=1)
|
|
|
|
except Currency.DoesNotExist:
|
|
|
|
currency = None
|
|
|
|
|
2019-09-03 12:28:53 +00:00
|
|
|
# Currency scaler
|
|
|
|
scaler = Decimal(1.0)
|
|
|
|
|
|
|
|
if currency is not None:
|
|
|
|
scaler = Decimal(currency.value)
|
|
|
|
|
2019-05-18 12:58:11 +00:00
|
|
|
part = self.get_part()
|
|
|
|
|
|
|
|
ctx = {
|
|
|
|
'part': part,
|
2019-09-03 12:28:53 +00:00
|
|
|
'quantity': quantity,
|
|
|
|
'currency': currency,
|
2019-05-18 12:58:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if part is None:
|
|
|
|
return ctx
|
|
|
|
|
2019-05-18 13:44:43 +00:00
|
|
|
# Supplier pricing information
|
|
|
|
if part.supplier_count > 0:
|
2019-05-20 13:53:39 +00:00
|
|
|
buy_price = part.get_supplier_price_range(quantity)
|
2019-05-18 12:58:11 +00:00
|
|
|
|
2019-05-20 13:53:39 +00:00
|
|
|
if buy_price is not None:
|
|
|
|
min_buy_price, max_buy_price = buy_price
|
2019-05-18 12:58:11 +00:00
|
|
|
|
2019-09-03 12:28:53 +00:00
|
|
|
min_buy_price /= scaler
|
|
|
|
max_buy_price /= scaler
|
|
|
|
|
|
|
|
min_buy_price = round(min_buy_price, 3)
|
|
|
|
max_buy_price = round(max_buy_price, 3)
|
|
|
|
|
2019-05-20 13:53:39 +00:00
|
|
|
if min_buy_price:
|
|
|
|
ctx['min_total_buy_price'] = min_buy_price
|
|
|
|
ctx['min_unit_buy_price'] = min_buy_price / quantity
|
|
|
|
|
|
|
|
if max_buy_price:
|
|
|
|
ctx['max_total_buy_price'] = max_buy_price
|
|
|
|
ctx['max_unit_buy_price'] = max_buy_price / quantity
|
2019-05-18 13:44:43 +00:00
|
|
|
|
|
|
|
# BOM pricing information
|
|
|
|
if part.bom_count > 0:
|
|
|
|
|
2019-05-20 13:53:39 +00:00
|
|
|
bom_price = part.get_bom_price_range(quantity)
|
2019-05-18 13:44:43 +00:00
|
|
|
|
2019-05-20 13:53:39 +00:00
|
|
|
if bom_price is not None:
|
|
|
|
min_bom_price, max_bom_price = bom_price
|
|
|
|
|
2019-09-03 12:28:53 +00:00
|
|
|
min_bom_price /= scaler
|
|
|
|
max_bom_price /= scaler
|
|
|
|
|
|
|
|
min_bom_price = round(min_bom_price, 3)
|
|
|
|
max_bom_price = round(max_bom_price, 3)
|
|
|
|
|
2019-05-20 13:53:39 +00:00
|
|
|
if min_bom_price:
|
|
|
|
ctx['min_total_bom_price'] = min_bom_price
|
|
|
|
ctx['min_unit_bom_price'] = min_bom_price / quantity
|
|
|
|
|
|
|
|
if max_bom_price:
|
|
|
|
ctx['max_total_bom_price'] = max_bom_price
|
|
|
|
ctx['max_unit_bom_price'] = max_bom_price / quantity
|
2019-05-18 12:58:11 +00:00
|
|
|
|
|
|
|
return ctx
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
|
|
|
|
return self.renderJsonResponse(request, self.form_class(), context=self.get_pricing())
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
|
2019-09-03 12:28:53 +00:00
|
|
|
currency = None
|
|
|
|
|
2019-05-18 12:58:11 +00:00
|
|
|
try:
|
|
|
|
quantity = int(self.request.POST.get('quantity', 1))
|
|
|
|
except ValueError:
|
|
|
|
quantity = 1
|
|
|
|
|
2019-09-03 12:28:53 +00:00
|
|
|
try:
|
|
|
|
currency_id = int(self.request.POST.get('currency', None))
|
|
|
|
|
|
|
|
if currency_id:
|
|
|
|
currency = Currency.objects.get(pk=currency_id)
|
|
|
|
except (ValueError, Currency.DoesNotExist):
|
2019-09-03 12:33:50 +00:00
|
|
|
currency = None
|
2019-09-03 12:28:53 +00:00
|
|
|
|
2019-05-18 12:58:11 +00:00
|
|
|
# Always mark the form as 'invalid' (the user may wish to keep getting pricing data)
|
|
|
|
data = {
|
|
|
|
'form_valid': False,
|
|
|
|
}
|
|
|
|
|
2019-09-03 12:28:53 +00:00
|
|
|
return self.renderJsonResponse(request, self.form_class(), data=data, context=self.get_pricing(quantity, currency))
|
2019-05-18 12:58:11 +00:00
|
|
|
|
|
|
|
|
2019-08-20 04:33:18 +00:00
|
|
|
class PartParameterTemplateCreate(AjaxCreateView):
|
|
|
|
""" View for creating a new PartParameterTemplate """
|
|
|
|
|
|
|
|
model = PartParameterTemplate
|
|
|
|
form_class = part_forms.EditPartParameterTemplateForm
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Create Part Parameter Template')
|
2019-08-20 04:33:18 +00:00
|
|
|
|
|
|
|
|
2019-09-07 10:28:38 +00:00
|
|
|
class PartParameterTemplateEdit(AjaxUpdateView):
|
|
|
|
""" View for editing a PartParameterTemplate """
|
|
|
|
|
|
|
|
model = PartParameterTemplate
|
|
|
|
form_class = part_forms.EditPartParameterTemplateForm
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Edit Part Parameter Template')
|
2019-09-07 10:28:38 +00:00
|
|
|
|
|
|
|
|
2019-09-07 10:30:51 +00:00
|
|
|
class PartParameterTemplateDelete(AjaxDeleteView):
|
|
|
|
""" View for deleting an existing PartParameterTemplate """
|
|
|
|
|
|
|
|
model = PartParameterTemplate
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _("Delete Part Parameter Template")
|
2019-09-07 10:30:51 +00:00
|
|
|
|
|
|
|
|
2019-08-20 04:14:21 +00:00
|
|
|
class PartParameterCreate(AjaxCreateView):
|
|
|
|
""" View for creating a new PartParameter """
|
|
|
|
|
|
|
|
model = PartParameter
|
|
|
|
form_class = part_forms.EditPartParameterForm
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Create Part Parameter')
|
2019-08-20 04:14:21 +00:00
|
|
|
|
2019-08-20 04:28:15 +00:00
|
|
|
def get_initial(self):
|
|
|
|
|
|
|
|
initials = {}
|
|
|
|
|
|
|
|
part_id = self.request.GET.get('part', None)
|
|
|
|
|
|
|
|
if part_id:
|
|
|
|
try:
|
|
|
|
initials['part'] = Part.objects.get(pk=part_id)
|
|
|
|
except (Part.DoesNotExist, ValueError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
return initials
|
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
""" Return the form object.
|
|
|
|
|
|
|
|
- Hide the 'Part' field (specified in URL)
|
|
|
|
- Limit the 'Template' options (to avoid duplicates)
|
|
|
|
"""
|
|
|
|
|
|
|
|
form = super().get_form()
|
|
|
|
|
|
|
|
part_id = self.request.GET.get('part', None)
|
|
|
|
|
|
|
|
if part_id:
|
|
|
|
try:
|
|
|
|
part = Part.objects.get(pk=part_id)
|
|
|
|
|
|
|
|
form.fields['part'].widget = HiddenInput()
|
|
|
|
|
|
|
|
query = form.fields['template'].queryset
|
|
|
|
|
|
|
|
query = query.exclude(id__in=[param.template.id for param in part.parameters.all()])
|
|
|
|
|
|
|
|
form.fields['template'].queryset = query
|
|
|
|
|
|
|
|
except (Part.DoesNotExist, ValueError):
|
|
|
|
pass
|
|
|
|
|
|
|
|
return form
|
2019-08-20 04:14:21 +00:00
|
|
|
|
|
|
|
|
2019-08-21 03:11:19 +00:00
|
|
|
class PartParameterEdit(AjaxUpdateView):
|
|
|
|
""" View for editing a PartParameter """
|
|
|
|
|
|
|
|
model = PartParameter
|
|
|
|
form_class = part_forms.EditPartParameterForm
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Edit Part Parameter')
|
2019-08-21 03:11:19 +00:00
|
|
|
|
|
|
|
def get_form(self):
|
|
|
|
|
|
|
|
form = super().get_form()
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
|
|
|
|
2019-08-28 09:39:47 +00:00
|
|
|
class PartParameterDelete(AjaxDeleteView):
|
|
|
|
""" View for deleting a PartParameter """
|
|
|
|
|
|
|
|
model = PartParameter
|
|
|
|
ajax_template_name = 'part/param_delete.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Delete Part Parameter')
|
2019-08-28 09:39:47 +00:00
|
|
|
|
|
|
|
|
2018-04-15 01:40:03 +00:00
|
|
|
class CategoryDetail(DetailView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Detail view for PartCategory """
|
2018-04-15 01:40:03 +00:00
|
|
|
model = PartCategory
|
|
|
|
context_object_name = 'category'
|
2019-05-20 14:31:34 +00:00
|
|
|
queryset = PartCategory.objects.all().prefetch_related('children')
|
2018-05-04 08:53:39 +00:00
|
|
|
template_name = 'part/category.html'
|
2018-04-15 01:40:03 +00:00
|
|
|
|
|
|
|
|
2018-04-27 10:42:12 +00:00
|
|
|
class CategoryEdit(AjaxUpdateView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Update view to edit a PartCategory """
|
2018-04-15 01:40:03 +00:00
|
|
|
model = PartCategory
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditCategoryForm
|
2018-04-27 10:42:12 +00:00
|
|
|
ajax_template_name = 'modal_form.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Edit Part Category')
|
2018-04-15 01:40:03 +00:00
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(CategoryEdit, self).get_context_data(**kwargs).copy()
|
|
|
|
|
2019-04-28 01:09:19 +00:00
|
|
|
try:
|
2019-05-08 07:57:31 +00:00
|
|
|
context['category'] = self.get_object()
|
2019-04-28 01:09:19 +00:00
|
|
|
except:
|
|
|
|
pass
|
2018-04-15 01:40:03 +00:00
|
|
|
|
|
|
|
return context
|
|
|
|
|
2019-05-08 07:57:31 +00:00
|
|
|
def get_form(self):
|
|
|
|
""" Customize form data for PartCategory editing.
|
|
|
|
|
|
|
|
Limit the choices for 'parent' field to those which make sense
|
|
|
|
"""
|
|
|
|
|
|
|
|
form = super(AjaxUpdateView, self).get_form()
|
|
|
|
|
|
|
|
category = self.get_object()
|
|
|
|
|
|
|
|
# Remove any invalid choices for the parent category part
|
|
|
|
parent_choices = PartCategory.objects.all()
|
|
|
|
parent_choices = parent_choices.exclude(id__in=category.getUniqueChildren())
|
|
|
|
|
|
|
|
form.fields['parent'].queryset = parent_choices
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
2018-04-15 01:40:03 +00:00
|
|
|
|
2018-04-27 10:42:12 +00:00
|
|
|
class CategoryDelete(AjaxDeleteView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Delete view to delete a PartCategory """
|
2018-04-15 01:40:03 +00:00
|
|
|
model = PartCategory
|
2019-04-27 23:00:54 +00:00
|
|
|
ajax_template_name = 'part/category_delete.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Delete Part Category')
|
2018-04-15 01:40:03 +00:00
|
|
|
context_object_name = 'category'
|
2018-04-15 15:02:17 +00:00
|
|
|
success_url = '/part/'
|
2018-04-15 01:40:03 +00:00
|
|
|
|
2018-04-29 14:23:02 +00:00
|
|
|
def get_data(self):
|
|
|
|
return {
|
2020-02-10 11:10:06 +00:00
|
|
|
'danger': _('Part category was deleted'),
|
2018-04-29 14:23:02 +00:00
|
|
|
}
|
|
|
|
|
2018-04-15 01:40:03 +00:00
|
|
|
|
2018-04-25 04:10:56 +00:00
|
|
|
class CategoryCreate(AjaxCreateView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Create view to make a new PartCategory """
|
2018-04-15 01:40:03 +00:00
|
|
|
model = PartCategory
|
2018-04-25 04:10:56 +00:00
|
|
|
ajax_form_action = reverse_lazy('category-create')
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Create new part category')
|
2018-04-25 04:10:56 +00:00
|
|
|
ajax_template_name = 'modal_form.html'
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditCategoryForm
|
2018-04-15 01:40:03 +00:00
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Add extra context data to template.
|
|
|
|
|
|
|
|
- If parent category provided, pass the category details to the template
|
|
|
|
"""
|
2018-04-15 01:40:03 +00:00
|
|
|
context = super(CategoryCreate, self).get_context_data(**kwargs).copy()
|
|
|
|
|
|
|
|
parent_id = self.request.GET.get('category', None)
|
|
|
|
|
|
|
|
if parent_id:
|
2019-04-28 01:09:19 +00:00
|
|
|
try:
|
|
|
|
context['category'] = PartCategory.objects.get(pk=parent_id)
|
|
|
|
except PartCategory.DoesNotExist:
|
|
|
|
pass
|
2018-04-15 01:40:03 +00:00
|
|
|
|
|
|
|
return context
|
|
|
|
|
|
|
|
def get_initial(self):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Get initial data for new PartCategory
|
|
|
|
|
|
|
|
- If parent provided, pre-fill the parent category
|
|
|
|
"""
|
2018-04-15 01:40:03 +00:00
|
|
|
initials = super(CategoryCreate, self).get_initial().copy()
|
|
|
|
|
|
|
|
parent_id = self.request.GET.get('category', None)
|
|
|
|
|
|
|
|
if parent_id:
|
2019-04-28 01:09:19 +00:00
|
|
|
try:
|
|
|
|
initials['parent'] = PartCategory.objects.get(pk=parent_id)
|
|
|
|
except PartCategory.DoesNotExist:
|
|
|
|
pass
|
2018-04-15 01:40:03 +00:00
|
|
|
|
|
|
|
return initials
|
2018-04-15 11:29:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
class BomItemDetail(DetailView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Detail view for BomItem """
|
2018-04-15 15:02:17 +00:00
|
|
|
context_object_name = 'item'
|
2018-04-15 11:29:24 +00:00
|
|
|
queryset = BomItem.objects.all()
|
|
|
|
template_name = 'part/bom-detail.html'
|
|
|
|
|
|
|
|
|
2018-04-25 23:26:43 +00:00
|
|
|
class BomItemCreate(AjaxCreateView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Create view for making a new BomItem object """
|
2018-04-15 11:29:24 +00:00
|
|
|
model = BomItem
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditBomItemForm
|
2018-04-25 23:26:43 +00:00
|
|
|
ajax_template_name = 'modal_form.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Create BOM item')
|
2018-04-15 11:29:24 +00:00
|
|
|
|
2019-04-29 14:16:20 +00:00
|
|
|
def get_form(self):
|
|
|
|
""" Override get_form() method to reduce Part selection options.
|
|
|
|
|
|
|
|
- Do not allow part to be added to its own BOM
|
|
|
|
- Remove any Part items that are already in the BOM
|
|
|
|
"""
|
2019-04-29 14:18:58 +00:00
|
|
|
|
2019-04-29 14:16:20 +00:00
|
|
|
form = super(AjaxCreateView, self).get_form()
|
|
|
|
|
|
|
|
part_id = form['part'].value()
|
|
|
|
|
|
|
|
try:
|
|
|
|
part = Part.objects.get(id=part_id)
|
|
|
|
|
2019-06-28 00:00:23 +00:00
|
|
|
# Only allow active parts to be selected
|
|
|
|
query = form.fields['part'].queryset.filter(active=True)
|
|
|
|
form.fields['part'].queryset = query
|
|
|
|
|
2019-04-29 14:16:20 +00:00
|
|
|
# Don't allow selection of sub_part objects which are already added to the Bom!
|
|
|
|
query = form.fields['sub_part'].queryset
|
|
|
|
|
|
|
|
# Don't allow a part to be added to its own BOM
|
|
|
|
query = query.exclude(id=part.id)
|
2019-06-28 00:00:23 +00:00
|
|
|
query = query.filter(active=True)
|
2019-04-29 14:16:20 +00:00
|
|
|
|
|
|
|
# Eliminate any options that are already in the BOM!
|
2019-04-29 14:18:58 +00:00
|
|
|
query = query.exclude(id__in=[item.id for item in part.required_parts()])
|
2019-04-29 14:16:20 +00:00
|
|
|
|
|
|
|
form.fields['sub_part'].queryset = query
|
2019-06-28 00:20:37 +00:00
|
|
|
|
|
|
|
form.fields['part'].widget = HiddenInput()
|
|
|
|
|
2019-09-13 06:26:44 +00:00
|
|
|
except (ValueError, Part.DoesNotExist):
|
2019-04-29 14:16:20 +00:00
|
|
|
pass
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
2018-04-15 11:29:24 +00:00
|
|
|
def get_initial(self):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Provide initial data for the BomItem:
|
|
|
|
|
|
|
|
- If 'parent' provided, set the parent part field
|
|
|
|
"""
|
|
|
|
|
2018-04-15 11:29:24 +00:00
|
|
|
# Look for initial values
|
|
|
|
initials = super(BomItemCreate, self).get_initial().copy()
|
|
|
|
|
|
|
|
# Parent part for this item?
|
|
|
|
parent_id = self.request.GET.get('parent', None)
|
|
|
|
|
|
|
|
if parent_id:
|
2019-04-28 01:09:19 +00:00
|
|
|
try:
|
|
|
|
initials['part'] = Part.objects.get(pk=parent_id)
|
|
|
|
except Part.DoesNotExist:
|
|
|
|
pass
|
2018-04-15 11:29:24 +00:00
|
|
|
|
|
|
|
return initials
|
|
|
|
|
|
|
|
|
2018-04-25 23:26:43 +00:00
|
|
|
class BomItemEdit(AjaxUpdateView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Update view for editing BomItem """
|
|
|
|
|
2018-04-15 11:29:24 +00:00
|
|
|
model = BomItem
|
2019-05-10 23:51:45 +00:00
|
|
|
form_class = part_forms.EditBomItemForm
|
2018-04-25 23:26:43 +00:00
|
|
|
ajax_template_name = 'modal_form.html'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Edit BOM item')
|
2018-04-15 11:29:24 +00:00
|
|
|
|
2019-09-13 06:26:44 +00:00
|
|
|
def get_form(self):
|
|
|
|
""" Override get_form() method to filter part selection options
|
|
|
|
|
|
|
|
- Do not allow part to be added to its own BOM
|
|
|
|
- Remove any part items that are already in the BOM
|
|
|
|
"""
|
|
|
|
|
|
|
|
form = super().get_form()
|
|
|
|
|
|
|
|
part_id = form['part'].value()
|
|
|
|
|
|
|
|
try:
|
|
|
|
part = Part.objects.get(pk=part_id)
|
|
|
|
|
|
|
|
query = form.fields['sub_part'].queryset
|
|
|
|
|
|
|
|
# Reduce the available selection options
|
|
|
|
query = query.exclude(pk=part_id)
|
|
|
|
|
|
|
|
# Eliminate any options that are already in the BOM,
|
|
|
|
# *except* for the item which is already selected
|
|
|
|
try:
|
|
|
|
sub_part_id = int(form['sub_part'].value())
|
|
|
|
except ValueError:
|
|
|
|
sub_part_id = -1
|
|
|
|
|
|
|
|
existing = [item.pk for item in part.required_parts()]
|
|
|
|
|
|
|
|
if sub_part_id in existing:
|
|
|
|
existing.remove(sub_part_id)
|
|
|
|
|
|
|
|
query = query.exclude(id__in=existing)
|
|
|
|
|
|
|
|
form.fields['sub_part'].queryset = query
|
|
|
|
|
|
|
|
except (ValueError, Part.DoesNotExist):
|
|
|
|
pass
|
|
|
|
|
|
|
|
return form
|
|
|
|
|
2018-04-15 11:29:24 +00:00
|
|
|
|
2018-04-26 14:54:01 +00:00
|
|
|
class BomItemDelete(AjaxDeleteView):
|
2019-04-27 12:18:07 +00:00
|
|
|
""" Delete view for removing BomItem """
|
2018-04-15 11:29:24 +00:00
|
|
|
model = BomItem
|
2019-04-27 23:00:54 +00:00
|
|
|
ajax_template_name = 'part/bom-delete.html'
|
2018-04-15 11:29:24 +00:00
|
|
|
context_object_name = 'item'
|
2020-02-10 11:10:06 +00:00
|
|
|
ajax_form_title = _('Confim BOM item deletion')
|