346 lines
13 KiB
Python
346 lines
13 KiB
Python
from django import forms
|
|
from django.forms import widgets
|
|
from django.core.exceptions import ValidationError
|
|
from parts import models as parts_models
|
|
from shimatta_modules.EngineeringNumberConverter import EngineeringNumberConverter
|
|
import uuid
|
|
from django.urls import reverse
|
|
|
|
from crispy_forms.helper import FormHelper
|
|
from crispy_forms.layout import Layout, Fieldset, Row, Column
|
|
|
|
class AutoCompleteWidget(widgets.Input):
|
|
template_name = 'widgets/autocomplete-foreign-key.html'
|
|
|
|
def __init__(self, api_search_url, image_field_name, foreign_model, name_field_name, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
self.image_field_name = image_field_name
|
|
self.foreign_model = foreign_model
|
|
self.api_search_url = api_search_url
|
|
self.name_field_name = name_field_name
|
|
|
|
def get_context(self, name, value, attrs):
|
|
context = super().get_context(name, value, attrs)
|
|
if value is not None:
|
|
try:
|
|
try:
|
|
my_id = uuid.UUID(value)
|
|
except:
|
|
my_id = int(value)
|
|
instance = self.foreign_model.objects.get(id=my_id)
|
|
except Exception as ex:
|
|
instance = None
|
|
else:
|
|
instance = None
|
|
|
|
image = None
|
|
if instance is not None and self.image_field_name is not None:
|
|
image = getattr(instance, self.image_field_name)
|
|
|
|
display_name = None
|
|
if instance:
|
|
display_name = getattr(instance, self.name_field_name)
|
|
|
|
context['custom'] = {
|
|
'search_url': reverse(self.api_search_url),
|
|
'image_field_name': self.image_field_name,
|
|
'current_instance': instance,
|
|
'image': image,
|
|
'name_field_name': self.name_field_name,
|
|
'name': display_name,
|
|
}
|
|
return context
|
|
|
|
class AutocompleteForeingKeyField(forms.UUIDField):
|
|
def __init__(self, foreign_model=None, api_search_url=None, image_field_name='image', name_field_name='name', **kwargs):
|
|
super().__init__(**kwargs)
|
|
self.widget = AutoCompleteWidget(api_search_url, image_field_name, foreign_model, name_field_name)
|
|
self.foreign_model = foreign_model
|
|
|
|
|
|
def clean(self, value):
|
|
try:
|
|
pre_cleaned_uuid = super().clean(value)
|
|
except Exception as ex:
|
|
try:
|
|
pre_cleaned_uuid = int(value)
|
|
except:
|
|
raise ex
|
|
if pre_cleaned_uuid is None and not self.required:
|
|
return None
|
|
try:
|
|
obj = self.foreign_model.objects.get(id=pre_cleaned_uuid)
|
|
except self.foreign_model.DoesNotExist:
|
|
raise ValidationError('Given element does not exist')
|
|
return obj
|
|
|
|
|
|
class MyTestForm(forms.Form):
|
|
pass
|
|
|
|
class AddSubStorageForm(forms.Form):
|
|
storage_name = forms.CharField(label="storage_name", initial='')
|
|
responsible = forms.CharField(label='responsible_user')
|
|
|
|
class DeleteStockForm(forms.Form):
|
|
stock_uuid = forms.UUIDField()
|
|
|
|
class EditWatermarkForm(forms.Form):
|
|
stock_uuid = forms.UUIDField()
|
|
watermark_active = forms.BooleanField(required=False) #If it is false, the webbrowser won't send it at all. Therefore we have to set it to required=False
|
|
watermark = forms.IntegerField(min_value=0)
|
|
|
|
def clean(self):
|
|
cleaned_data = super().clean()
|
|
id = cleaned_data.get("stock_uuid")
|
|
|
|
if not id:
|
|
raise ValidationError("No stock UUID given")
|
|
|
|
stock = None
|
|
try:
|
|
stock = parts_models.Stock.objects.get(id=id)
|
|
except:
|
|
raise ValidationError("Stock with uuid %s does not exist" % (id))
|
|
cleaned_data['stock'] = stock
|
|
|
|
return cleaned_data
|
|
|
|
def save(self):
|
|
stock = self.cleaned_data['stock']
|
|
active = self.cleaned_data['watermark_active']
|
|
watermark = self.cleaned_data['watermark']
|
|
|
|
if not active:
|
|
watermark = -1
|
|
|
|
stock.watermark = watermark
|
|
stock.save()
|
|
|
|
class EditStockAmountForm(forms.Form):
|
|
stock_uuid = forms.UUIDField()
|
|
amount = forms.IntegerField(min_value=0)
|
|
|
|
def clean(self):
|
|
cleaned_data = super().clean()
|
|
id = cleaned_data.get("stock_uuid")
|
|
|
|
if not id:
|
|
raise ValidationError("No stock UUID given")
|
|
|
|
stock = None
|
|
try:
|
|
stock = parts_models.Stock.objects.get(id=id)
|
|
except:
|
|
raise ValidationError("Stock with uuid %s does not exist" % (id))
|
|
cleaned_data['stock'] = stock
|
|
|
|
return cleaned_data
|
|
|
|
def save(self, increase: bool):
|
|
stock = self.cleaned_data['stock']
|
|
amount = self.cleaned_data['amount']
|
|
|
|
if not increase:
|
|
amount = -amount
|
|
|
|
return stock.atomic_increment(amount)
|
|
|
|
class AddStockForm(forms.Form):
|
|
watermark_active = forms.BooleanField(required=False)
|
|
watermark = forms.IntegerField(min_value=0, required=True, initial=0)
|
|
amount = forms.IntegerField(min_value=0, required=True, initial=1)
|
|
component_uuid = forms.UUIDField(required=True)
|
|
lot = forms.CharField(max_length=255, required=False)
|
|
|
|
def clean(self):
|
|
cleaned_data = super().clean()
|
|
|
|
id = cleaned_data.get('component_uuid')
|
|
if not id:
|
|
raise ValidationError('No valid component selected!')
|
|
component = None
|
|
try:
|
|
component = parts_models.Component.objects.get(id=id)
|
|
except:
|
|
raise ValidationError("Invalid component selected!")
|
|
cleaned_data['component'] = component
|
|
|
|
return cleaned_data
|
|
|
|
def save(self, storage):
|
|
component = self.cleaned_data.get('component')
|
|
amount = self.cleaned_data.get('amount')
|
|
watermark = -1
|
|
|
|
if self.cleaned_data.get('watermark_active'):
|
|
watermark = self.cleaned_data.get('watermark')
|
|
|
|
new_stock = parts_models.Stock.objects.create(storage=storage, component=component, watermark=watermark, amount=amount, lot=self.cleaned_data['lot'])
|
|
new_stock.save()
|
|
|
|
class ComponentForm(forms.ModelForm):
|
|
manufacturer = AutocompleteForeingKeyField(api_search_url='manufacturer-list', foreign_model=parts_models.Manufacturer, required=False)
|
|
component_type = AutocompleteForeingKeyField(api_search_url='componenttype-list', foreign_model=parts_models.ComponentType, name_field_name='class_name', required=False, image_field_name=None)
|
|
package = AutocompleteForeingKeyField(api_search_url='package-list', foreign_model=parts_models.Package, required=False)
|
|
pref_distri = AutocompleteForeingKeyField(api_search_url='distributor-list', foreign_model=parts_models.Distributor, required=False)
|
|
class Meta:
|
|
model = parts_models.Component
|
|
fields = '__all__'
|
|
|
|
class PackageForm(forms.ModelForm):
|
|
class Meta:
|
|
model = parts_models.Package
|
|
fields = '__all__'
|
|
|
|
class DistributorForm(forms.ModelForm):
|
|
class Meta:
|
|
model = parts_models.Distributor
|
|
fields = '__all__'
|
|
class ManufacturerForm(forms.ModelForm):
|
|
class Meta:
|
|
model = parts_models.Manufacturer
|
|
fields = '__all__'
|
|
|
|
class DistributorNumberCreateForm(forms.ModelForm):
|
|
distributor = AutocompleteForeingKeyField(api_search_url='distributor-list', foreign_model=parts_models.Distributor, required=True)
|
|
|
|
class Meta:
|
|
model = parts_models.DistributorNum
|
|
fields = ['distributor', 'distributor_part_number']
|
|
|
|
class DistributorNumberDeleteForm(forms.Form):
|
|
distributor_num = forms.UUIDField(required=True)
|
|
|
|
def clean_distributor_num(self):
|
|
my_uuid = self.cleaned_data['distributor_num']
|
|
try:
|
|
distributor_number = parts_models.DistributorNum.objects.get(id=my_uuid)
|
|
except:
|
|
raise ValidationError('distributor number invalid')
|
|
return distributor_number
|
|
|
|
class ComponentParameterDeleteForm(forms.Form):
|
|
param_num = forms.UUIDField(required=True)
|
|
|
|
def clean_param_num(self):
|
|
my_uuid = self.cleaned_data['param_num']
|
|
try:
|
|
param = parts_models.ComponentParameter.objects.get(id=my_uuid)
|
|
except:
|
|
raise ValidationError('Parameter Number Invalid')
|
|
return param
|
|
|
|
class AdvancedComponentSearchForm(forms.Form):
|
|
name = forms.CharField(max_length=255, label='Component Name', required=False)
|
|
package = AutocompleteForeingKeyField(required=False, api_search_url='package-list', foreign_model=parts_models.Package)
|
|
package_pin_count = forms.IntegerField(min_value=0, label='Pin Count', required=False)
|
|
distributor_num = forms.CharField(max_length=100, label='Distributor Part Number', required=False)
|
|
distributor = AutocompleteForeingKeyField(required=False, api_search_url='distributor-list', foreign_model=parts_models.Distributor)
|
|
component_type = AutocompleteForeingKeyField(required=False, api_search_url='componenttype-list', foreign_model=parts_models.ComponentType, image_field_name=None, name_field_name='class_name')
|
|
manufacturer = AutocompleteForeingKeyField(required=False, api_search_url='manufacturer-list', foreign_model=parts_models.Manufacturer)
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
self.helper = FormHelper()
|
|
self.helper.form_tag = False
|
|
self.helper.disable_csrf = True
|
|
self.helper.layout = Layout(
|
|
Row(
|
|
Column('name'),
|
|
),
|
|
Row(
|
|
Column('package'),
|
|
Column('package_pin_count'),
|
|
),
|
|
Row(
|
|
Column('component_type'),
|
|
Column('manufacturer'),
|
|
),
|
|
Row(
|
|
Column('distributor'),
|
|
Column('distributor_num'),
|
|
),
|
|
)
|
|
|
|
|
|
PARAMETER_COMPARISON_TYPES = (
|
|
('eq', '=='),
|
|
('lte', '<='),
|
|
('gte', '>='),
|
|
)
|
|
|
|
class ComponentParameterSearchForm(forms.Form):
|
|
parameter = AutocompleteForeingKeyField(required=True, foreign_model=parts_models.ComponentParameterType, api_search_url='componentparametertype-list', image_field_name=None, name_field_name='parameter_name')
|
|
value = forms.CharField(max_length=100, required=False)
|
|
compare_method = forms.ChoiceField(choices=PARAMETER_COMPARISON_TYPES, required=True, initial=1)
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
self.helper = FormHelper()
|
|
self.helper.form_tag = False
|
|
self.helper.disable_csrf = True
|
|
self.helper.layout = Layout(
|
|
Row(
|
|
Column('parameter'),
|
|
Column('compare_method'),
|
|
Column('value')
|
|
)
|
|
)
|
|
|
|
def clean(self):
|
|
cleaned_data = super().clean()
|
|
parameter = cleaned_data.get('parameter')
|
|
value = cleaned_data.get('value')
|
|
|
|
if value != '' or value != None:
|
|
value = value.strip()
|
|
|
|
if value == '' or value == None:
|
|
cleaned_data['value'] = None
|
|
value = None
|
|
|
|
if parameter and value is not None and value != '':
|
|
if parameter.parameter_type != 'F':
|
|
try:
|
|
cleaned_data['value'] = EngineeringNumberConverter.engineering_to_number(value)
|
|
except:
|
|
raise ValidationError('Cannot convert value to number')
|
|
|
|
return cleaned_data
|
|
|
|
|
|
class ComponentParameterCreateForm(forms.Form):
|
|
parameter_type = AutocompleteForeingKeyField(required=True, foreign_model=parts_models.ComponentParameterType, api_search_url='componentparametertype-list', image_field_name=None, name_field_name='descriptive_name')
|
|
value = forms.CharField(required=True, max_length=256)
|
|
|
|
def clean(self):
|
|
data = super().clean()
|
|
ptype = data.get('parameter_type')
|
|
value = data.get('value')
|
|
|
|
if not ptype:
|
|
raise ValidationError('No valid parameter type selected')
|
|
|
|
if not value:
|
|
raise ValidationError('No valid parameter value')
|
|
|
|
if ptype.parameter_type == 'E' or ptype.parameter_type == 'I' or ptype.parameter_type == 'N':
|
|
try:
|
|
number = EngineeringNumberConverter.engineering_to_number(value)
|
|
except:
|
|
raise ValidationError('Cannot convert value to number')
|
|
data['number_value'] = number
|
|
else:
|
|
pass
|
|
|
|
def save(self, component):
|
|
param_type = self.cleaned_data['parameter_type']
|
|
if param_type.parameter_type == 'F':
|
|
text_value = self.cleaned_data['value']
|
|
value = 0
|
|
else:
|
|
text_value = ''
|
|
value = self.cleaned_data['number_value']
|
|
parts_models.ComponentParameter.objects.create(parameter_type=param_type, component=component, value=value, text_value=text_value)
|