""" Django views for interacting with Part app """ # -*- coding: utf-8 -*- from __future__ import unicode_literals from django.core.files.base import ContentFile from django.core.exceptions import ValidationError from django.db import transaction from django.db.utils import IntegrityError from django.shortcuts import get_object_or_404 from django.shortcuts import HttpResponseRedirect from django.utils.translation import gettext_lazy as _ from django.urls import reverse, reverse_lazy from django.views.generic import DetailView, ListView, FormView, UpdateView from django.forms.models import model_to_dict from django.forms import HiddenInput, CheckboxInput from django.conf import settings from moneyed import CURRENCIES from djmoney.contrib.exchange.models import convert_money from PIL import Image import requests import os import io from rapidfuzz import fuzz from decimal import Decimal, InvalidOperation from .models import PartCategory, Part, PartAttachment, PartRelated from .models import PartParameterTemplate, PartParameter from .models import PartCategoryParameterTemplate from .models import BomItem from .models import match_part_names from .models import PartTestTemplate from .models import PartSellPriceBreak from common.models import InvenTreeSetting from company.models import SupplierPart import common.settings as inventree_settings from . import forms as part_forms from .bom import MakeBomTemplate, BomUploadManager, ExportBom, IsValidBOMFormat from .admin import PartResource from InvenTree.views import AjaxView, AjaxCreateView, AjaxUpdateView, AjaxDeleteView from InvenTree.views import QRCodeView from InvenTree.views import InvenTreeRoleMixin from InvenTree.helpers import DownloadFile, str2bool class PartIndex(InvenTreeRoleMixin, ListView): """ View for displaying list of Part objects """ model = Part template_name = 'part/category.html' context_object_name = 'parts' def get_queryset(self): return Part.objects.all().select_related('category') def get_context_data(self, **kwargs): context = super(PartIndex, self).get_context_data(**kwargs).copy() # View top-level categories children = PartCategory.objects.filter(parent=None) context['children'] = children context['category_count'] = PartCategory.objects.count() context['part_count'] = Part.objects.count() return context class PartRelatedCreate(AjaxCreateView): """ View for creating a new PartRelated object - The view only makes sense if a Part object is passed to it """ model = PartRelated form_class = part_forms.CreatePartRelatedForm ajax_form_title = _("Add Related Part") ajax_template_name = "modal_form.html" def get_initial(self): """ Set parent part as part_1 field """ initials = {} part_id = self.request.GET.get('part', None) if part_id: try: initials['part_1'] = Part.objects.get(pk=part_id) except (Part.DoesNotExist, ValueError): pass return initials def get_form(self): """ Create a form to upload a new PartRelated - Hide the 'part_1' field (parent part) - Display parts which are not yet related """ form = super(AjaxCreateView, self).get_form() form.fields['part_1'].widget = HiddenInput() try: # Get parent part parent_part = self.get_initial()['part_1'] # Get existing related parts related_parts = [related_part[1].pk for related_part in parent_part.get_related_parts()] # Build updated choice list excluding # - parts already related to parent part # - the parent part itself updated_choices = [] for choice in form.fields["part_2"].choices: if (choice[0] not in related_parts) and (choice[0] != parent_part.pk): updated_choices.append(choice) # Update choices for related part form.fields['part_2'].choices = updated_choices except KeyError: pass return form class PartRelatedDelete(AjaxDeleteView): """ View for deleting a PartRelated object """ model = PartRelated ajax_form_title = _("Delete Related Part") context_object_name = "related" # Explicit role requirement role_required = 'part.change' 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 form_class = part_forms.EditPartAttachmentForm ajax_form_title = _("Add part attachment") ajax_template_name = "modal_form.html" def save(self, form, **kwargs): """ Record the user that uploaded this attachment """ attachment = form.save(commit=False) attachment.user = self.request.user attachment.save() def get_data(self): return { 'success': _('Added attachment') } 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= """ initials = super(AjaxCreateView, self).get_initial() # TODO - If the proper part was not sent, return an error message try: initials['part'] = Part.objects.get(id=self.request.GET.get('part', None)) except (ValueError, Part.DoesNotExist): pass 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 """ model = PartAttachment form_class = part_forms.EditPartAttachmentForm ajax_template_name = 'modal_form.html' ajax_form_title = _('Edit attachment') def get_data(self): return { 'success': _('Part attachment updated') } 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 ajax_form_title = _("Delete Part Attachment") ajax_template_name = "attachment_delete.html" context_object_name = "attachment" role_required = 'part.change' def get_data(self): return { 'danger': _('Deleted part attachment') } 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") class PartSetCategory(AjaxUpdateView): """ View for settings the part category for multiple parts at once """ ajax_template_name = 'part/set_category.html' ajax_form_title = _('Set Part Category') form_class = part_forms.SetPartCategoryForm role_required = 'part.change' 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 = [] return self.renderJsonResponse(request, form=self.get_form(), context=self.get_context_data()) 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: self.set_category() return self.renderJsonResponse(request, data=data, form=self.get_form(), context=self.get_context_data()) @transaction.atomic def set_category(self): for part in self.parts: part.set_category(self.category) 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 class MakePartVariant(AjaxCreateView): """ View for creating a new variant based on an existing template Part - Part is provided in the URL '/part//make_variant/' - Automatically copy relevent data (BOM, etc, etc) """ model = Part form_class = part_forms.EditPartForm ajax_form_title = _('Create Variant') 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 # form.fields['variant_of'].widget = HiddenInput() # 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() 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() bom_copy = str2bool(request.POST.get('bom_copy', False)) parameters_copy = str2bool(request.POST.get('parameters_copy', False)) # Copy relevent information from the template part part.deep_copy(part_template, bom=bom_copy, parameters=parameters_copy) 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 initials['bom_copy'] = InvenTreeSetting.get_setting('PART_COPY_BOM') initials['parameters_copy'] = InvenTreeSetting.get_setting('PART_COPY_PARAMETERS') return initials class PartDuplicate(AjaxCreateView): """ View for duplicating an existing Part object. - Part is provided in the URL '/part//copy/' - Option for 'deep-copy' which will duplicate all BOM items (default = True) """ model = Part form_class = part_forms.EditPartForm ajax_form_title = _("Duplicate Part") ajax_template_name = "part/copy_part.html" def get_data(self): return { 'success': _('Copied part') } def get_part_to_copy(self): try: return Part.objects.get(id=self.kwargs['pk']) except (Part.DoesNotExist, ValueError): return None def get_context_data(self): return { 'part': self.get_part_to_copy() } def get_form(self): form = super(AjaxCreateView, self).get_form() # 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() return form def post(self, request, *args, **kwargs): """ Capture the POST request for part duplication - 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! """ 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: # Display the first five closest matches context['matches'] = matches[:5] # 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: msg = _('Possible matches exist - confirm creation of new part') form.add_error('confirm_creation', msg) form.pre_form_warning = msg valid = False data = { 'form_valid': valid } if valid: # Create the new Part part = form.save(commit=False) part.creation_user = request.user part.save() data['pk'] = part.pk data['text'] = str(part) bom_copy = str2bool(request.POST.get('bom_copy', False)) parameters_copy = str2bool(request.POST.get('parameters_copy', False)) original = self.get_part_to_copy() if original: part.deep_copy(original, bom=bom_copy, parameters=parameters_copy) 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: initials = super(AjaxCreateView, self).get_initial() initials['bom_copy'] = str2bool(InvenTreeSetting.get_setting('PART_COPY_BOM', True)) initials['parameters_copy'] = str2bool(InvenTreeSetting.get_setting('PART_COPY_PARAMETERS', True)) return initials class PartCreate(AjaxCreateView): """ View for creating a new Part object. Options for providing initial conditions: - Provide a category object as initial data """ model = Part form_class = part_forms.EditPartForm ajax_form_title = _('Create New Part') ajax_template_name = 'part/create_part.html' def get_data(self): return { 'success': _("Created new part"), } def get_category_id(self): return self.request.GET.get('category', None) def get_context_data(self, **kwargs): """ Provide extra context information for the form to display: - Add category information (if provided) """ context = super(PartCreate, self).get_context_data(**kwargs) # Add category information to the page cat_id = self.get_category_id() if cat_id: try: context['category'] = PartCategory.objects.get(pk=cat_id) except (PartCategory.DoesNotExist, ValueError): pass return context def get_form(self): """ Create Form for making new Part object. Remove the 'default_supplier' field as there are not yet any matching SupplierPart objects """ form = super(AjaxCreateView, self).get_form() # Hide the "default expiry" field if the feature is not enabled if not inventree_settings.stock_expiry_enabled(): form.fields['default_expiry'].widget = HiddenInput() # Hide the default_supplier field (there are no matching supplier parts yet!) form.fields['default_supplier'].widget = HiddenInput() # Display category templates widgets form.fields['selected_category_templates'].widget = CheckboxInput() form.fields['parent_category_templates'].widget = CheckboxInput() return form def post(self, request, *args, **kwargs): form = self.get_form() context = {} valid = form.is_valid() name = request.POST.get('name', None) if name: matches = match_part_names(name) if len(matches) > 0: # Limit to the top 5 matches (to prevent clutter) context['matches'] = matches[:5] # 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: msg = _('Possible matches exist - confirm creation of new part') form.add_error('confirm_creation', msg) form.pre_form_warning = msg valid = False data = { 'form_valid': valid } if valid: # Create the new Part part = form.save(commit=False) # Record the user who created this part part.creation_user = request.user # Store category templates settings add_category_templates = { 'main': form.cleaned_data['selected_category_templates'], 'parent': form.cleaned_data['parent_category_templates'], } # Save part and pass category template settings part.save(**{'add_category_templates': add_category_templates}) data['pk'] = part.pk data['text'] = str(part) try: data['url'] = part.get_absolute_url() except AttributeError: pass return self.renderJsonResponse(request, form, data, context=context) def get_initial(self): """ Get initial data for the new Part object: - If a category is provided, pre-fill the Category field """ initials = super(PartCreate, self).get_initial() if self.get_category_id(): try: category = PartCategory.objects.get(pk=self.get_category_id()) initials['category'] = category initials['keywords'] = category.default_keywords except (PartCategory.DoesNotExist, ValueError): pass # 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) # Automatically create part parameters from category templates initials['selected_category_templates'] = str2bool(InvenTreeSetting.get_setting('PART_CATEGORY_PARAMETERS', False)) initials['parent_category_templates'] = initials['selected_category_templates'] return initials 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 role_required = 'part.change' fields = ['notes'] def get_success_url(self): """ Return the success URL for this form """ return reverse('part-notes', kwargs={'pk': self.get_object().id}) def get_context_data(self, **kwargs): part = self.get_object() context = super().get_context_data(**kwargs) context['editing'] = str2bool(self.request.GET.get('edit', '')) ctx = part.get_context_data(self.request) context.update(ctx) return context class PartDetail(InvenTreeRoleMixin, DetailView): """ Detail view for Part object """ context_object_name = 'part' queryset = Part.objects.all().select_related('category') template_name = 'part/detail.html' # Add in some extra context information based on query params def get_context_data(self, **kwargs): """ Provide extra context data to template - If '?editing=True', set 'editing_enabled' context variable """ context = super().get_context_data(**kwargs) part = self.get_object() if str2bool(self.request.GET.get('edit', '')): # Allow BOM editing if the part is active context['editing_enabled'] = 1 if part.active else 0 else: context['editing_enabled'] = 0 ctx = part.get_context_data(self.request) context.update(**ctx) 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 # BOM Information for Pie-Chart bom_items = [{'name': str(a.sub_part), 'price': a.sub_part.get_price_range(quantity), 'q': a.quantity} for a in part.bom_items.all()] if [True for a in bom_items if len(set(a['price']))==2]: ctx['bom_parts'] = [{ 'name': a['name'], 'min_price': str((a['price'][0] * a['q'])/ quantity), 'max_price': str((a['price'][1] * a['q']) / quantity)} for a in bom_items] ctx['bom_pie_min'] = True else: ctx['bom_parts'] = [{ 'name':a['name'], 'price': str((a['price'][0] * a['q']) / quantity)} for a in bom_items] 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): 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 except queryset.model.MultipleObjectsReturned: pass 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) class PartQRCode(QRCodeView): """ View for displaying a QR code for a Part object """ ajax_form_title = _("Part QR Code") role_required = 'part.view' 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 class PartImageDownloadFromURL(AjaxUpdateView): """ View for downloading an image from a provided URL """ model = Part ajax_template_name = 'image_download.html' form_class = part_forms.PartImageDownloadForm ajax_form_title = _('Download Image') def validate(self, part, form): """ Validate that the image data are correct. - Try to download the image! """ # First ensure that the normal validation routines pass if not form.is_valid(): return # We can now extract a valid URL from the form data url = form.cleaned_data.get('url', None) # Download the file response = requests.get(url, stream=True) # Look at response header, reject if too large content_length = response.headers.get('Content-Length', '0') try: content_length = int(content_length) except (ValueError): # If we cannot extract meaningful length, just assume it's "small enough" content_length = 0 # TODO: Factor this out into a configurable setting MAX_IMG_LENGTH = 10 * 1024 * 1024 if content_length > MAX_IMG_LENGTH: form.add_error('url', _('Image size exceeds maximum allowable size for download')) return self.response = response # Check for valid response code if not response.status_code == 200: form.add_error('url', _('Invalid response: {code}').format(code=response.status_code)) return response.raw.decode_content = True try: self.image = Image.open(response.raw).convert() self.image.verify() except: form.add_error('url', _("Supplied URL is not a valid image file")) return def save(self, part, form, **kwargs): """ Save the downloaded image to the part """ fmt = self.image.format if not fmt: fmt = 'PNG' buffer = io.BytesIO() self.image.save(buffer, format=fmt) # Construct a simplified name for the image filename = f"part_{part.pk}_image.{fmt.lower()}" part.image.save( filename, ContentFile(buffer.getvalue()), ) class PartImageUpload(AjaxUpdateView): """ View for uploading a new Part image """ model = Part ajax_template_name = 'modal_form.html' ajax_form_title = _('Upload Part Image') form_class = part_forms.PartImageForm def get_data(self): return { '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') fields = [ 'image', ] 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) class PartEdit(AjaxUpdateView): """ View for editing Part object """ model = Part form_class = part_forms.EditPartForm ajax_template_name = 'modal_form.html' ajax_form_title = _('Edit Part Properties') context_object_name = 'part' 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() # Hide the "default expiry" field if the feature is not enabled if not inventree_settings.stock_expiry_enabled(): form.fields['default_expiry'].widget = HiddenInput() part = self.get_object() form.fields['default_supplier'].queryset = SupplierPart.objects.filter(part=part) # Check if IPN can be edited ipn_edit_enable = InvenTreeSetting.get_setting('PART_ALLOW_EDIT_IPN') if not ipn_edit_enable and not self.request.user.is_superuser: # Admin can still change IPN form.fields['IPN'].disabled = True return form class BomDuplicate(AjaxUpdateView): """ View for duplicating BOM from a parent item. """ model = Part context_object_name = 'part' ajax_form_title = _('Duplicate BOM') ajax_template_name = 'part/bom_duplicate.html' form_class = part_forms.BomDuplicateForm def get_form(self): form = super().get_form() # Limit choices to parents of the current part parents = self.get_object().get_ancestors() form.fields['parent'].queryset = parents return form def get_initial(self): initials = super().get_initial() parents = self.get_object().get_ancestors() if parents.count() == 1: initials['parent'] = parents[0] return initials def validate(self, part, form): confirm = str2bool(form.cleaned_data.get('confirm', False)) if not confirm: form.add_error('confirm', _('Confirm duplication of BOM from parent')) def save(self, part, form): """ Duplicate BOM from the specified parent """ parent = form.cleaned_data.get('parent', None) clear = str2bool(form.cleaned_data.get('clear', True)) if parent: part.copy_bom_from(parent, clear=clear) class BomValidate(AjaxUpdateView): """ Modal form view for validating a part BOM """ model = Part ajax_form_title = _("Validate BOM") 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 validate(self, part, form, **kwargs): confirm = str2bool(form.cleaned_data.get('validate', False)) if not confirm: form.add_error('validate', _('Confirm that the BOM is valid')) def save(self, part, form, **kwargs): """ Mark the BOM as validated """ part.validate_bom(self.request.user) def get_data(self): return { 'success': _('Validated Bill of Materials') } class BomUpload(InvenTreeRoleMixin, FormView): """ 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. """ template_name = 'part/bom_upload/upload_file.html' # Context data passed to the forms (initially empty, extracted from uploaded file) bom_headers = [] bom_columns = [] bom_rows = [] missing_columns = [] allowed_parts = [] role_required = ('part.change', 'part.add') def get_success_url(self): part = self.get_object() return reverse('upload-bom', kwargs={'pk': part.id}) def get_form_class(self): # Default form is the starting point return part_forms.BomUploadSelectFile def get_context_data(self, *args, **kwargs): ctx = super().get_context_data(*args, **kwargs) # 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, 'column': self.bom_columns[idx] }) rows.append({ 'index': row.get('index', -1), 'data': data, 'part_match': row.get('part_match', None), 'part_options': row.get('part_options', self.allowed_parts), # User-input (passed between client and server) 'quantity': row.get('quantity', None), 'description': row.get('description', ''), 'part_name': row.get('part_name', ''), 'part': row.get('part', None), 'reference': row.get('reference', ''), 'notes': row.get('notes', ''), 'errors': row.get('errors', ''), }) ctx['part'] = self.part ctx['bom_headers'] = BomUploadManager.HEADERS ctx['bom_columns'] = self.bom_columns ctx['bom_rows'] = rows ctx['missing_columns'] = self.missing_columns ctx['allowed_parts_list'] = self.allowed_parts return ctx def getAllowedParts(self): """ Return a queryset of parts which are allowed to be added to this BOM. """ return self.part.get_allowed_bom_items() def get(self, request, *args, **kwargs): """ Perform the initial 'GET' request. Initially returns a form for file upload """ self.request = request # A valid Part object must be supplied. This is the 'parent' part for the BOM self.part = get_object_or_404(Part, pk=self.kwargs['pk']) self.form = self.get_form() form_class = self.get_form_class() form = self.get_form(form_class) return self.render_to_response(self.get_context_data(form=form)) def handleBomFileUpload(self): """ 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. """ bom_file = self.request.FILES.get('bom_file', None) manager = None bom_file_valid = False if bom_file is None: self.form.add_error('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 for k, v in errors.items(): self.form.add_error(k, v) if bom_file_valid: # BOM file is valid? Proceed to the next step! form = None self.template_name = 'part/bom_upload/select_fields.html' self.extractDataFromFile(manager) else: form = self.form return self.render_to_response(self.get_context_data(form=form)) def getColumnIndex(self, name): """ Return the index of the column with the given name. It named column is not found, return -1 """ try: idx = list(self.column_selections.values()).index(name) except ValueError: idx = -1 return idx def preFillSelections(self): """ 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. """ # 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') q_idx = self.getColumnIndex('Quantity') r_idx = self.getColumnIndex('Reference') o_idx = self.getColumnIndex('Overage') n_idx = self.getColumnIndex('Note') for row in self.bom_rows: """ 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 # A list of potential Part matches part_options = self.allowed_parts # Check if there is a column corresponding to "quantity" if q_idx >= 0: q_val = row['data'][q_idx] 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 # Check if there is a column corresponding to "Part Name" if p_idx >= 0: part_name = row['data'][p_idx] row['part_name'] = part_name matches = [] for part in self.allowed_parts: ratio = fuzz.partial_ratio(part.name + part.description, part_name) matches.append({'part': part, 'match': ratio}) # Sort matches by the 'strength' of the match ratio if len(matches) > 0: matches = sorted(matches, key=lambda item: item['match'], reverse=True) part_options = [m['part'] for m in matches] # Check if there is a column corresponding to "Part IPN" if i_idx >= 0: row['part_ipn'] = row['data'][i_idx] # Check if there is a column corresponding to "Overage" field if o_idx >= 0: row['overage'] = row['data'][o_idx] # Check if there is a column corresponding to "Reference" field if r_idx >= 0: row['reference'] = row['data'][r_idx] # Check if there is a column corresponding to "Note" field if n_idx >= 0: 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 # Unless found, the 'part_match' is blank row['part_match'] = None 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 part.IPN and row['part_ipn'].lower() == str(part.IPN.lower())] # Check for single match if len(part_matches) == 1: row['part_match'] = part_matches[0] continue except KeyError: pass def extractDataFromFile(self, bom): """ Read data from the BOM file """ self.bom_columns = bom.columns() self.bom_rows = bom.rows() 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_ - Column name at idx as provided in the uploaded file col_guess_ - Column guess at idx as selected in the BOM row__col - Cell data as provided in the uploaded file """ # Map the columns self.column_names = {} self.column_selections = {} self.row_data = {} for item in self.request.POST: value = self.request.POST[item] # Column names as passed as col_name_ where idx is an integer # Extract the column names if item.startswith('col_name_'): try: col_id = int(item.replace('col_name_', '')) except ValueError: continue col_name = value self.column_names[col_id] = col_name # 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 col_name = value self.column_selections[col_id] = value # Extract the row data if item.startswith('row_'): # Item should be of the format row__col_ s = item.split('_') if len(s) < 4: continue # Ignore row/col IDs which are not correct numeric values try: row_id = int(s[1]) col_id = int(s[3]) except ValueError: continue 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()): value = row[col_idx] items.append(value) self.bom_rows.append({ 'index': row_idx, 'data': items, 'errors': {}, }) # Construct the column data self.bom_columns = [] # Track any duplicate column selections self.duplicates = False for col in self.col_ids: if col in self.column_selections: guess = self.column_selections[col] else: guess = None header = ({ 'name': self.column_names[col], 'guess': guess }) if guess: n = list(self.column_selections.values()).count(self.column_selections[col]) if n > 1: header['duplicate'] = True self.duplicates = True self.bom_columns.append(header) # Are there any missing columns? self.missing_columns = [] # Check that all required fields are present for col in BomUploadManager.REQUIRED_HEADERS: if col not in self.column_selections.values(): self.missing_columns.append(col) # Check that at least one of the part match field is present part_match_found = False for col in BomUploadManager.PART_MATCH_HEADERS: if col in self.column_selections.values(): part_match_found = True break # If not, notify user if not part_match_found: for col in BomUploadManager.PART_MATCH_HEADERS: self.missing_columns.append(col) 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. """ # Extract POST data self.getTableDataFromPost() valid = len(self.missing_columns) == 0 and not self.duplicates if valid: # Try to extract meaningful data self.preFillSelections() self.template_name = 'part/bom_upload/select_parts.html' else: self.template_name = 'part/bom_upload/select_fields.html' return self.render_to_response(self.get_context_data(form=None)) def handlePartSelection(self): # Extract basic table data from POST request self.getTableDataFromPost() # 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 q = Decimal(1) try: q = Decimal(value) if q < 0: row['errors']['quantity'] = _('Quantity must be greater than zero') 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): row['errors']['quantity'] = _('Enter a valid quantity') row['quantity'] = q except ValueError: continue # Extract part from each row if key.startswith('part_'): 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 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') # 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 # Are there any errors after form handling? valid = True for row in self.bom_rows: # Has a part been selected for the given row? part = row.get('part', None) if part is None: row['errors']['part'] = _('Select a part') 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') # Has a quantity been specified? if row.get('quantity', None) is None: row['errors']['quantity'] = _('Specify quantity') errors = row.get('errors', []) if len(errors) > 0: valid = False self.template_name = 'part/bom_upload/select_parts.html' ctx = self.get_context_data(form=None) if valid: self.part.clear_bom() # Generate new BOM items for row in self.bom_rows: part = row.get('part') quantity = row.get('quantity') reference = row.get('reference', '') notes = row.get('notes', '') # 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})) else: ctx['form_errors'] = True return self.render_to_response(ctx) def getRowByIndex(self, idx): for row in self.bom_rows: if row['index'] == idx: return row return None def post(self, request, *args, **kwargs): """ Perform the various 'POST' requests required. """ self.request = request self.part = get_object_or_404(Part, pk=self.kwargs['pk']) self.allowed_parts = self.getAllowedParts() self.form = self.get_form(self.get_form_class()) # Did the user POST a file named bom_file? form_step = request.POST.get('form_step', None) if form_step == 'select_file': return self.handleBomFileUpload() elif form_step == 'select_fields': return self.handleFieldSelection() elif form_step == 'select_parts': return self.handlePartSelection() return self.render_to_response(self.get_context_data(form=self.form)) class PartExport(AjaxView): """ Export a CSV file containing information on multiple parts """ role_required = 'part.view' def get_parts(self, request): """ Extract part list from the POST parameters. Parts can be supplied as: - Part category - List of part PK values """ # 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() # Also optionally filter by explicit list of part IDs part_ids = request.GET.get('parts', '') parts = [] for pk in part_ids.split(','): try: 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', ) return part_list def get(self, request, *args, **kwargs): parts = self.get_parts(request) dataset = PartResource().export(queryset=parts) csv = dataset.export('csv') return DownloadFile(csv, 'InvenTree_Parts.csv') 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) class BomDownload(AjaxView): """ Provide raw download of a BOM file. - File format should be passed as a query param e.g. ?format=csv """ role_required = 'part.view' model = Part def get(self, request, *args, **kwargs): part = get_object_or_404(Part, pk=self.kwargs['pk']) export_format = request.GET.get('file_format', 'csv') cascade = str2bool(request.GET.get('cascade', False)) parameter_data = str2bool(request.GET.get('parameter_data', False)) stock_data = str2bool(request.GET.get('stock_data', False)) supplier_data = str2bool(request.GET.get('supplier_data', False)) manufacturer_data = str2bool(request.GET.get('manufacturer_data', False)) levels = request.GET.get('levels', None) if levels is not None: try: levels = int(levels) if levels <= 0: levels = None except ValueError: levels = None if not IsValidBOMFormat(export_format): export_format = 'csv' return ExportBom(part, fmt=export_format, cascade=cascade, max_levels=levels, parameter_data=parameter_data, stock_data=stock_data, supplier_data=supplier_data, manufacturer_data=manufacturer_data, ) def get_data(self): return { 'info': 'Exported BOM' } 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") role_required = 'part.view' def get(self, request, *args, **kwargs): return self.renderJsonResponse(request, self.form_class()) 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)) levels = request.POST.get('levels', None) parameter_data = str2bool(request.POST.get('parameter_data', False)) stock_data = str2bool(request.POST.get('stock_data', False)) supplier_data = str2bool(request.POST.get('supplier_data', False)) manufacturer_data = str2bool(request.POST.get('manufacturer_data', False)) 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) url += '¶meter_data=' + str(parameter_data) url += '&stock_data=' + str(stock_data) url += '&supplier_data=' + str(supplier_data) url += '&manufacturer_data=' + str(manufacturer_data) if levels: url += '&levels=' + str(levels) data = { 'form_valid': part is not None, 'url': url, } return self.renderJsonResponse(request, self.form_class(), data=data) class PartDelete(AjaxDeleteView): """ View to delete a Part object """ model = Part ajax_template_name = 'part/partial_delete.html' ajax_form_title = _('Confirm Part Deletion') context_object_name = 'part' success_url = '/part/' def get_data(self): return { 'danger': _('Part was deleted'), } class PartPricing(AjaxView): """ View for inspecting part pricing information """ model = Part ajax_template_name = "part/part_pricing.html" ajax_form_title = _("Part Pricing") form_class = part_forms.PartPriceForm role_required = ['sales_order.view', 'part.view'] def get_quantity(self): """ Return set quantity in decimal format """ return Decimal(self.request.POST.get('quantity', 1)) def get_part(self): try: return Part.objects.get(id=self.kwargs['pk']) except Part.DoesNotExist: return None def get_pricing(self, quantity=1, currency=None): """ returns context with pricing information """ if quantity <= 0: quantity = 1 # TODO - Capacity for price comparison in different currencies currency = None # Currency scaler scaler = Decimal(1.0) part = self.get_part() ctx = { 'part': part, 'quantity': quantity, 'currency': currency, } if part is None: return ctx # Supplier pricing information if part.supplier_count > 0: buy_price = part.get_supplier_price_range(quantity) if buy_price is not None: min_buy_price, max_buy_price = buy_price min_buy_price /= scaler max_buy_price /= scaler min_unit_buy_price = round(min_buy_price / quantity, 3) max_unit_buy_price = round(max_buy_price / quantity, 3) min_buy_price = round(min_buy_price, 3) max_buy_price = round(max_buy_price, 3) if min_buy_price: ctx['min_total_buy_price'] = min_buy_price ctx['min_unit_buy_price'] = min_unit_buy_price if max_buy_price: ctx['max_total_buy_price'] = max_buy_price ctx['max_unit_buy_price'] = max_unit_buy_price # BOM pricing information if part.bom_count > 0: bom_price = part.get_bom_price_range(quantity) if bom_price is not None: min_bom_price, max_bom_price = bom_price min_bom_price /= scaler max_bom_price /= scaler min_unit_bom_price = round(min_bom_price / quantity, 3) max_unit_bom_price = round(max_bom_price / quantity, 3) min_bom_price = round(min_bom_price, 3) max_bom_price = round(max_bom_price, 3) if min_bom_price: ctx['min_total_bom_price'] = min_bom_price ctx['min_unit_bom_price'] = min_unit_bom_price if max_bom_price: ctx['max_total_bom_price'] = max_bom_price ctx['max_unit_bom_price'] = max_unit_bom_price # part pricing information part_price = part.get_price(quantity) if part_price is not None: ctx['total_part_price'] = round(part_price, 3) ctx['unit_part_price'] = round(part_price / quantity, 3) return ctx def get_initials(self): """ returns initials for form """ return {'quantity': self.get_quantity()} def get(self, request, *args, **kwargs): init = self.get_initials() qty = self.get_quantity() return self.renderJsonResponse(request, self.form_class(initial=init), context=self.get_pricing(qty)) def post(self, request, *args, **kwargs): currency = None quantity = self.get_quantity() # Retain quantity value set by user form = self.form_class(initial=self.get_initials()) # TODO - How to handle pricing in different currencies? currency = None # check if data is set try: data = self.data except AttributeError: data = {} # Always mark the form as 'invalid' (the user may wish to keep getting pricing data) data['form_valid'] = False return self.renderJsonResponse(request, form, data=data, context=self.get_pricing(quantity, currency)) class PartParameterTemplateCreate(AjaxCreateView): """ View for creating a new PartParameterTemplate """ model = PartParameterTemplate form_class = part_forms.EditPartParameterTemplateForm ajax_form_title = _('Create Part Parameter Template') class PartParameterTemplateEdit(AjaxUpdateView): """ View for editing a PartParameterTemplate """ model = PartParameterTemplate form_class = part_forms.EditPartParameterTemplateForm ajax_form_title = _('Edit Part Parameter Template') class PartParameterTemplateDelete(AjaxDeleteView): """ View for deleting an existing PartParameterTemplate """ model = PartParameterTemplate ajax_form_title = _("Delete Part Parameter Template") class PartParameterCreate(AjaxCreateView): """ View for creating a new PartParameter """ model = PartParameter form_class = part_forms.EditPartParameterForm ajax_form_title = _('Create Part Parameter') 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 class PartParameterEdit(AjaxUpdateView): """ View for editing a PartParameter """ model = PartParameter form_class = part_forms.EditPartParameterForm ajax_form_title = _('Edit Part Parameter') def get_form(self): form = super().get_form() return form class PartParameterDelete(AjaxDeleteView): """ View for deleting a PartParameter """ model = PartParameter ajax_template_name = 'part/param_delete.html' ajax_form_title = _('Delete Part Parameter') class CategoryDetail(InvenTreeRoleMixin, DetailView): """ Detail view for PartCategory """ model = PartCategory context_object_name = 'category' queryset = PartCategory.objects.all().prefetch_related('children') template_name = 'part/category_partlist.html' def get_context_data(self, **kwargs): context = super(CategoryDetail, self).get_context_data(**kwargs).copy() try: context['part_count'] = kwargs['object'].partcount() except KeyError: context['part_count'] = 0 return context class CategoryParametric(CategoryDetail): """ Parametric view for PartCategory """ template_name = 'part/category_parametric.html' def get_context_data(self, **kwargs): context = super(CategoryParametric, self).get_context_data(**kwargs).copy() # Get current category category = kwargs.get('object', None) if category: cascade = kwargs.get('cascade', True) # Prefetch parts parameters parts_parameters = category.prefetch_parts_parameters(cascade=cascade) # Get table headers (unique parameters names) context['headers'] = category.get_unique_parameters(cascade=cascade, prefetch=parts_parameters) # Insert part information context['headers'].insert(0, 'description') context['headers'].insert(0, 'part') # Get parameters data context['parameters'] = category.get_parts_parameters(cascade=cascade, prefetch=parts_parameters) return context class CategoryEdit(AjaxUpdateView): """ Update view to edit a PartCategory """ model = PartCategory form_class = part_forms.EditCategoryForm ajax_template_name = 'modal_form.html' ajax_form_title = _('Edit Part Category') def get_context_data(self, **kwargs): context = super(CategoryEdit, self).get_context_data(**kwargs).copy() try: context['category'] = self.get_object() except: pass return context 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 class CategoryDelete(AjaxDeleteView): """ Delete view to delete a PartCategory """ model = PartCategory ajax_template_name = 'part/category_delete.html' ajax_form_title = _('Delete Part Category') context_object_name = 'category' success_url = '/part/' def get_data(self): return { 'danger': _('Part category was deleted'), } class CategoryCreate(AjaxCreateView): """ Create view to make a new PartCategory """ model = PartCategory ajax_form_action = reverse_lazy('category-create') ajax_form_title = _('Create new part category') ajax_template_name = 'modal_form.html' form_class = part_forms.EditCategoryForm def get_context_data(self, **kwargs): """ Add extra context data to template. - If parent category provided, pass the category details to the template """ context = super(CategoryCreate, self).get_context_data(**kwargs).copy() parent_id = self.request.GET.get('category', None) if parent_id: try: context['category'] = PartCategory.objects.get(pk=parent_id) except PartCategory.DoesNotExist: pass return context def get_initial(self): """ Get initial data for new PartCategory - If parent provided, pre-fill the parent category """ initials = super(CategoryCreate, self).get_initial().copy() parent_id = self.request.GET.get('category', None) if parent_id: try: initials['parent'] = PartCategory.objects.get(pk=parent_id) except PartCategory.DoesNotExist: pass return initials class CategoryParameterTemplateCreate(AjaxCreateView): """ View for creating a new PartCategoryParameterTemplate """ model = PartCategoryParameterTemplate form_class = part_forms.EditCategoryParameterTemplateForm ajax_form_title = _('Create Category Parameter Template') def get_initial(self): """ Get initial data for Category """ initials = super().get_initial() category_id = self.kwargs.get('pk', None) if category_id: try: initials['category'] = PartCategory.objects.get(pk=category_id) except (PartCategory.DoesNotExist, ValueError): pass return initials def get_form(self): """ Create a form to upload a new CategoryParameterTemplate - Hide the 'category' field (parent part) - Display parameter templates which are not yet related """ form = super(AjaxCreateView, self).get_form() form.fields['category'].widget = HiddenInput() if form.is_valid(): form.cleaned_data['category'] = self.kwargs.get('pk', None) try: # Get selected category category = self.get_initial()['category'] # Get existing parameter templates parameters = [template.parameter_template.pk for template in category.get_parameter_templates()] # Exclude templates already linked to category updated_choices = [] for choice in form.fields["parameter_template"].choices: if (choice[0] not in parameters): updated_choices.append(choice) # Update choices for parameter templates form.fields['parameter_template'].choices = updated_choices except KeyError: pass return form def post(self, request, *args, **kwargs): """ Capture the POST request - If the add_to_all_categories object is set, link parameter template to all categories - If the add_to_same_level_categories object is set, link parameter template to same level categories """ form = self.get_form() valid = form.is_valid() if valid: add_to_same_level_categories = form.cleaned_data['add_to_same_level_categories'] add_to_all_categories = form.cleaned_data['add_to_all_categories'] selected_category = PartCategory.objects.get(pk=int(self.kwargs['pk'])) parameter_template = form.cleaned_data['parameter_template'] default_value = form.cleaned_data['default_value'] categories = PartCategory.objects.all() if add_to_same_level_categories and not add_to_all_categories: # Get level level = selected_category.level # Filter same level categories categories = categories.filter(level=level) if add_to_same_level_categories or add_to_all_categories: # Add parameter template and default value to categories for category in categories: # Skip selected category (will be processed in the post call) if category.pk != selected_category.pk: try: cat_template = PartCategoryParameterTemplate.objects.create(category=category, parameter_template=parameter_template, default_value=default_value) cat_template.save() except IntegrityError: # Parameter template is already linked to category pass return super().post(request, *args, **kwargs) class CategoryParameterTemplateEdit(AjaxUpdateView): """ View for editing a PartCategoryParameterTemplate """ model = PartCategoryParameterTemplate form_class = part_forms.EditCategoryParameterTemplateForm ajax_form_title = _('Edit Category Parameter Template') def get_object(self): try: self.object = self.model.objects.get(pk=self.kwargs['pid']) except: return None return self.object def get_form(self): """ Create a form to upload a new CategoryParameterTemplate - Hide the 'category' field (parent part) - Display parameter templates which are not yet related """ form = super(AjaxUpdateView, self).get_form() form.fields['category'].widget = HiddenInput() form.fields['add_to_all_categories'].widget = HiddenInput() form.fields['add_to_same_level_categories'].widget = HiddenInput() if form.is_valid(): form.cleaned_data['category'] = self.kwargs.get('pk', None) try: # Get selected category category = PartCategory.objects.get(pk=self.kwargs.get('pk', None)) # Get selected template selected_template = self.get_object().parameter_template # Get existing parameter templates parameters = [template.parameter_template.pk for template in category.get_parameter_templates() if template.parameter_template.pk != selected_template.pk] # Exclude templates already linked to category updated_choices = [] for choice in form.fields["parameter_template"].choices: if (choice[0] not in parameters): updated_choices.append(choice) # Update choices for parameter templates form.fields['parameter_template'].choices = updated_choices # Set initial choice to current template form.fields['parameter_template'].initial = selected_template except KeyError: pass return form class CategoryParameterTemplateDelete(AjaxDeleteView): """ View for deleting an existing PartCategoryParameterTemplate """ model = PartCategoryParameterTemplate ajax_form_title = _("Delete Category Parameter Template") def get_object(self): try: self.object = self.model.objects.get(pk=self.kwargs['pid']) except: return None return self.object class BomItemCreate(AjaxCreateView): """ Create view for making a new BomItem object """ model = BomItem form_class = part_forms.EditBomItemForm ajax_template_name = 'modal_form.html' ajax_form_title = _('Create BOM Item') 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 """ form = super(AjaxCreateView, self).get_form() part_id = form['part'].value() # Construct a queryset for the part field part_query = Part.objects.filter(active=True) # Construct a queryset for the sub_part field sub_part_query = Part.objects.filter( component=True, active=True ) try: part = Part.objects.get(id=part_id) # Hide the 'part' field form.fields['part'].widget = HiddenInput() # Exclude the part from its own BOM sub_part_query = sub_part_query.exclude(id=part.id) # Eliminate any options that are already in the BOM! sub_part_query = sub_part_query.exclude(id__in=[item.id for item in part.getRequiredParts()]) except (ValueError, Part.DoesNotExist): pass # Set the querysets for the fields form.fields['part'].queryset = part_query form.fields['sub_part'].queryset = sub_part_query return form def get_initial(self): """ Provide initial data for the BomItem: - If 'parent' provided, set the parent part field """ # 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: try: initials['part'] = Part.objects.get(pk=parent_id) except Part.DoesNotExist: pass return initials class BomItemEdit(AjaxUpdateView): """ Update view for editing BomItem """ model = BomItem form_class = part_forms.EditBomItemForm ajax_template_name = 'modal_form.html' ajax_form_title = _('Edit BOM item') 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 """ item = self.get_object() form = super().get_form() part_id = form['part'].value() try: part = Part.objects.get(pk=part_id) # Construct a queryset query = Part.objects.filter(component=True) # Limit to "active" items, *unless* the currently selected item is not active if item.sub_part.active: query = query.filter(active=True) # Prevent the parent part from being selected 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.getRequiredParts()] 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 class BomItemDelete(AjaxDeleteView): """ Delete view for removing BomItem """ model = BomItem ajax_template_name = 'part/bom-delete.html' context_object_name = 'item' ajax_form_title = _('Confim BOM item deletion') class PartSalePriceBreakCreate(AjaxCreateView): """ View for creating a sale price break for a part """ model = PartSellPriceBreak form_class = part_forms.EditPartSalePriceBreakForm ajax_form_title = _('Add Price Break') def get_data(self): return { 'success': _('Added new price break') } def get_part(self): try: part = Part.objects.get(id=self.request.GET.get('part')) except (ValueError, Part.DoesNotExist): part = None if part is None: try: part = Part.objects.get(id=self.request.POST.get('part')) except (ValueError, Part.DoesNotExist): part = None return part def get_form(self): form = super(AjaxCreateView, self).get_form() form.fields['part'].widget = HiddenInput() return form def get_initial(self): initials = super(AjaxCreateView, self).get_initial() initials['part'] = self.get_part() default_currency = InvenTreeSetting.get_setting('INVENTREE_DEFAULT_CURRENCY') currency = CURRENCIES.get(default_currency, None) if currency is not None: initials['price'] = [1.0, currency] return initials class PartSalePriceBreakEdit(AjaxUpdateView): """ View for editing a sale price break """ model = PartSellPriceBreak form_class = part_forms.EditPartSalePriceBreakForm ajax_form_title = _('Edit Price Break') def get_form(self): form = super().get_form() form.fields['part'].widget = HiddenInput() return form class PartSalePriceBreakDelete(AjaxDeleteView): """ View for deleting a sale price break """ model = PartSellPriceBreak ajax_form_title = _("Delete Price Break") ajax_template_name = "modal_delete_form.html"