mirror of
				https://github.com/inventree/InvenTree.git
				synced 2025-10-26 02:47:41 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			237 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			237 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| Common database model definitions.
 | |
| These models are 'generic' and do not fit a particular business logic object.
 | |
| """
 | |
| 
 | |
| # -*- coding: utf-8 -*-
 | |
| from __future__ import unicode_literals
 | |
| 
 | |
| import os
 | |
| import decimal
 | |
| 
 | |
| from django.db import models
 | |
| from django.conf import settings
 | |
| from django.utils.translation import ugettext as _
 | |
| from django.core.validators import MinValueValidator, MaxValueValidator
 | |
| from django.core.exceptions import ValidationError
 | |
| 
 | |
| import InvenTree.fields
 | |
| 
 | |
| 
 | |
| class InvenTreeSetting(models.Model):
 | |
|     """
 | |
|     An InvenTreeSetting object is a key:value pair used for storing
 | |
|     single values (e.g. one-off settings values).
 | |
| 
 | |
|     The class provides a way of retrieving the value for a particular key,
 | |
|     even if that key does not exist.
 | |
|     """
 | |
| 
 | |
|     class Meta:
 | |
|         verbose_name = "InvenTree Setting"
 | |
|         verbose_name_plural = "InvenTree Settings"
 | |
| 
 | |
|     @classmethod
 | |
|     def get_setting(cls, key, backup_value=None):
 | |
|         """
 | |
|         Get the value of a particular setting.
 | |
|         If it does not exist, return the backup value (default = None)
 | |
|         """
 | |
| 
 | |
|         try:
 | |
|             setting = InvenTreeSetting.objects.get(key__iexact=key)
 | |
|             return setting.value
 | |
|         except InvenTreeSetting.DoesNotExist:
 | |
|             return backup_value
 | |
| 
 | |
|     @classmethod
 | |
|     def set_setting(cls, key, value, user, create=True):
 | |
|         """
 | |
|         Set the value of a particular setting.
 | |
|         If it does not exist, option to create it.
 | |
| 
 | |
|         Args:
 | |
|             key: settings key
 | |
|             value: New value
 | |
|             user: User object (must be staff member to update a core setting)
 | |
|             create: If True, create a new setting if the specified key does not exist.
 | |
|         """
 | |
| 
 | |
|         if not user.is_staff:
 | |
|             return
 | |
| 
 | |
|         try:
 | |
|             setting = InvenTreeSetting.objects.get(key__iexact=key)
 | |
|         except InvenTreeSetting.DoesNotExist:
 | |
| 
 | |
|             if create:
 | |
|                 setting = InvenTreeSetting(key=key)
 | |
|             else:
 | |
|                 return
 | |
|             
 | |
|         setting.value = value
 | |
|         setting.save()
 | |
| 
 | |
|     key = models.CharField(max_length=50, blank=False, unique=True, help_text=_('Settings key (must be unique - case insensitive'))
 | |
| 
 | |
|     value = models.CharField(max_length=200, blank=True, unique=False, help_text=_('Settings value'))
 | |
| 
 | |
|     description = models.CharField(max_length=200, blank=True, unique=False, help_text=_('Settings description'))
 | |
| 
 | |
|     def validate_unique(self, exclude=None):
 | |
|         """ Ensure that the key:value pair is unique.
 | |
|         In addition to the base validators, this ensures that the 'key'
 | |
|         is unique, using a case-insensitive comparison.
 | |
|         """
 | |
| 
 | |
|         super().validate_unique(exclude)
 | |
| 
 | |
|         try:
 | |
|             setting = InvenTreeSetting.objects.exclude(id=self.id).filter(key__iexact=self.key)
 | |
|             if setting.exists():
 | |
|                 raise ValidationError({'key': _('Key string must be unique')})
 | |
|         except InvenTreeSetting.DoesNotExist:
 | |
|             pass
 | |
| 
 | |
| 
 | |
| class Currency(models.Model):
 | |
|     """
 | |
|     A Currency object represents a particular unit of currency.
 | |
|     Each Currency has a scaling factor which relates it to the base currency.
 | |
|     There must be one (and only one) currency which is selected as the base currency,
 | |
|     and each other currency is calculated relative to it.
 | |
| 
 | |
|     Attributes:
 | |
|         symbol: Currency symbol e.g. $
 | |
|         suffix: Currency suffix e.g. AUD
 | |
|         description: Long-form description e.g. "Australian Dollars"
 | |
|         value: The value of this currency compared to the base currency.
 | |
|         base: True if this currency is the base currency
 | |
| 
 | |
|     """
 | |
| 
 | |
|     symbol = models.CharField(max_length=10, blank=False, unique=False, help_text=_('Currency Symbol e.g. $'))
 | |
| 
 | |
|     suffix = models.CharField(max_length=10, blank=False, unique=True, help_text=_('Currency Suffix e.g. AUD'))
 | |
| 
 | |
|     description = models.CharField(max_length=100, blank=False, help_text=_('Currency Description'))
 | |
| 
 | |
|     value = models.DecimalField(default=1.0, max_digits=10, decimal_places=5, validators=[MinValueValidator(0.00001), MaxValueValidator(100000)], help_text=_('Currency Value'))
 | |
| 
 | |
|     base = models.BooleanField(default=False, help_text=_('Use this currency as the base currency'))
 | |
| 
 | |
|     class Meta:
 | |
|         verbose_name_plural = 'Currencies'
 | |
| 
 | |
|     def __str__(self):
 | |
|         """ Format string for currency representation """
 | |
|         s = "{sym} {suf} - {desc}".format(
 | |
|             sym=self.symbol,
 | |
|             suf=self.suffix,
 | |
|             desc=self.description
 | |
|         )
 | |
| 
 | |
|         if self.base:
 | |
|             s += " (Base)"
 | |
|         
 | |
|         else:
 | |
|             s += " = {v}".format(v=self.value)
 | |
| 
 | |
|         return s
 | |
| 
 | |
|     def save(self, *args, **kwargs):
 | |
|         """ Validate the model before saving
 | |
| 
 | |
|         - Ensure that there is only one base currency!
 | |
|         """
 | |
| 
 | |
|         # If this currency is set as the base currency, ensure no others are
 | |
|         if self.base:
 | |
|             for cur in Currency.objects.filter(base=True).exclude(pk=self.pk):
 | |
|                 cur.base = False
 | |
|                 cur.save()
 | |
| 
 | |
|         # If there are no currencies set as the base currency, set this as base
 | |
|         if not Currency.objects.exclude(pk=self.pk).filter(base=True).exists():
 | |
|             self.base = True
 | |
| 
 | |
|         # If this is the base currency, ensure value is set to unity
 | |
|         if self.base:
 | |
|             self.value = 1.0
 | |
| 
 | |
|         super().save(*args, **kwargs)
 | |
| 
 | |
| 
 | |
| class PriceBreak(models.Model):
 | |
|     """
 | |
|     Represents a PriceBreak model
 | |
|     """
 | |
| 
 | |
|     class Meta:
 | |
|         abstract = True
 | |
| 
 | |
|     quantity = InvenTree.fields.RoundingDecimalField(max_digits=15, decimal_places=5, default=1, validators=[MinValueValidator(1)])
 | |
| 
 | |
|     cost = InvenTree.fields.RoundingDecimalField(max_digits=10, decimal_places=5, validators=[MinValueValidator(0)])
 | |
| 
 | |
|     currency = models.ForeignKey(Currency, blank=True, null=True, on_delete=models.SET_NULL)
 | |
| 
 | |
|     @property
 | |
|     def converted_cost(self):
 | |
|         """
 | |
|         Return the cost of this price break, converted to the base currency
 | |
|         """
 | |
| 
 | |
|         scaler = decimal.Decimal(1.0)
 | |
| 
 | |
|         if self.currency:
 | |
|             scaler = self.currency.value
 | |
| 
 | |
|         return self.cost * scaler
 | |
| 
 | |
| 
 | |
| class ColorTheme(models.Model):
 | |
|     """ Color Theme Setting """
 | |
| 
 | |
|     default_color_theme = ('', _('Default'))
 | |
| 
 | |
|     name = models.CharField(max_length=20,
 | |
|                             default='',
 | |
|                             blank=True)
 | |
| 
 | |
|     user = models.CharField(max_length=150,
 | |
|                             unique=True)
 | |
| 
 | |
|     @classmethod
 | |
|     def get_color_themes_choices(cls):
 | |
|         """ Get all color themes from static folder """
 | |
| 
 | |
|         # Get files list from css/color-themes/ folder
 | |
|         files_list = []
 | |
|         for file in os.listdir(settings.STATIC_COLOR_THEMES_DIR):
 | |
|             files_list.append(os.path.splitext(file))
 | |
| 
 | |
|         # Get color themes choices (CSS sheets)
 | |
|         choices = [(file_name.lower(), _(file_name.replace('-', ' ').title()))
 | |
|                    for file_name, file_ext in files_list
 | |
|                    if file_ext == '.css' and file_name.lower() != 'default']
 | |
| 
 | |
|         # Add default option as empty option
 | |
|         choices.insert(0, cls.default_color_theme)
 | |
| 
 | |
|         return choices
 | |
| 
 | |
|     @classmethod
 | |
|     def is_valid_choice(cls, user_color_theme):
 | |
|         """ Check if color theme is valid choice """
 | |
|         try:
 | |
|             user_color_theme_name = user_color_theme.name
 | |
|         except AttributeError:
 | |
|             return False
 | |
| 
 | |
|         for color_theme in cls.get_color_themes_choices():
 | |
|             if user_color_theme_name == color_theme[0]:
 | |
|                 return True
 | |
| 
 | |
|         return False
 |