trytond-sale_margin/sale.py

204 lines
7.5 KiB
Python
Raw Normal View History

2016-04-13 17:17:34 +02:00
# This file is part sale_margin module for Tryton.
# The COPYRIGHT file at the top level of this repository contains
# the full copyright notices and license terms.
2012-07-24 11:56:39 +02:00
from decimal import Decimal
from math import fabs
2012-10-16 17:10:24 +02:00
from trytond.model import fields
2012-07-24 11:56:39 +02:00
from trytond.pyson import Eval
2012-10-16 17:10:24 +02:00
from trytond.pool import Pool, PoolMeta
2016-04-13 17:17:34 +02:00
from trytond.modules.product import price_digits
2012-07-24 11:56:39 +02:00
2012-10-16 17:10:24 +02:00
__all__ = ['Sale', 'SaleLine']
2014-01-17 13:45:34 +01:00
2012-10-16 17:10:24 +02:00
class Sale:
2016-03-29 12:00:38 +02:00
__metaclass__ = PoolMeta
2012-10-16 17:10:24 +02:00
__name__ = 'sale.sale'
2012-07-24 11:56:39 +02:00
margin = fields.Function(fields.Numeric('Margin',
digits=(16, Eval('currency_digits', 2),),
2014-08-30 15:14:54 +02:00
depends=['currency_digits'],
2018-02-01 17:23:58 +01:00
help=('It gives profitability by calculating the difference '
'between the Unit Price and Cost Price.')),
'get_margin')
2012-07-24 11:56:39 +02:00
margin_cache = fields.Numeric('Margin Cache',
digits=(16, Eval('currency_digits', 2)),
readonly=True,
depends=['currency_digits'])
margin_percent = fields.Function(fields.Numeric('Margin (%)',
digits=(16, 4)),
'get_margin_percent')
margin_percent_cache = fields.Numeric('Margin (%) Cache',
digits=(16, 4), readonly=True)
2012-07-24 11:56:39 +02:00
@classmethod
def copy(cls, sales, default=None):
if default is None:
default = {}
default = default.copy()
default['margin_cache'] = None
default['margin_percent_cache'] = None
return super(Sale, cls).copy(sales, default=default)
2012-10-16 17:10:24 +02:00
def get_margin(self, name):
2012-07-24 11:56:39 +02:00
'''
Return the margin of each sales
'''
2012-10-16 17:10:24 +02:00
Currency = Pool().get('currency.currency')
if (self.state in self._states_cached
and self.margin_cache is not None):
return self.margin_cache
margin = sum((l.margin for l in self.lines if l.type == 'line'),
Decimal(0))
return Currency.round(self.currency, margin)
2012-07-24 11:56:39 +02:00
def get_margin_percent(self, name):
Configuration = Pool().get('sale.configuration')
config = Configuration(1)
sale_margin_method = config.sale_margin_method or 'cost_price'
if (self.state in self._states_cached
and self.margin_percent_cache is not None):
return self.margin_percent_cache
price = sum(
Decimal(str(fabs(l.quantity))) * (
getattr(l, sale_margin_method) or Decimal(0))
for l in self.lines if l.type == 'line')
if price:
return (self.margin / price).quantize(Decimal('0.0001'))
else:
return Decimal('1.0')
2012-10-16 17:10:24 +02:00
@classmethod
def store_cache(cls, sales):
super(Sale, cls).store_cache(sales)
2012-10-16 17:10:24 +02:00
for sale in sales:
cls.write([sale], {
2012-07-24 11:56:39 +02:00
'margin_cache': sale.margin,
'margin_percent_cache': sale.margin_percent,
2012-07-24 11:56:39 +02:00
})
2012-10-16 17:10:24 +02:00
class SaleLine:
2016-03-29 12:00:38 +02:00
__metaclass__ = PoolMeta
2012-10-16 17:10:24 +02:00
__name__ = 'sale.line'
2016-04-13 17:17:34 +02:00
cost_price = fields.Numeric('Cost Price', digits=price_digits,
2012-07-24 11:56:39 +02:00
states={
'invisible': Eval('type') != 'line',
'readonly': ~Eval('sale_state').in_(['draft', 'quotation']),
},
depends=['type', 'sale_state'])
2012-07-24 11:56:39 +02:00
margin = fields.Function(fields.Numeric('Margin',
digits=(16, Eval('_parent_sale', {}).get('currency_digits', 2)),
states={
'invisible': ~Eval('type').in_(['line', 'subtotal']),
'readonly': ~Eval('_parent_sale'),
},
depends=['type', 'amount']),
'on_change_with_margin')
margin_percent = fields.Function(fields.Numeric('Margin (%)',
digits=(16, 4), states={
'invisible': ~Eval('type').in_(['line', 'subtotal']),
}, depends=['type']),
'on_change_with_margin_percent')
@classmethod
def __setup__(cls):
super(SaleLine, cls).__setup__()
if hasattr(cls, 'gross_unit_price'):
cls.on_change_with_margin.depends.add('gross_unit_price')
2012-07-24 11:56:39 +02:00
2013-10-04 10:57:14 +02:00
@staticmethod
def default_cost_price():
2014-11-21 08:58:33 +01:00
return Decimal(0)
2013-10-04 10:57:14 +02:00
2015-09-04 00:37:17 +02:00
@fields.depends('product')
2012-10-16 17:10:24 +02:00
def on_change_product(self):
2015-09-04 00:37:17 +02:00
super(SaleLine, self).on_change_product()
if self.product:
2015-09-04 00:37:17 +02:00
cost_price = self.product.cost_price
self.cost_price = cost_price.quantize(
Decimal(1) / 10 ** self.__class__.cost_price.digits[1])
2012-07-24 11:56:39 +02:00
@fields.depends('type', 'quantity', 'cost_price', '_parent_sale.currency',
'_parent_sale.lines', methods=['amount'])
def on_change_with_margin(self, name=None):
2012-07-24 11:56:39 +02:00
'''
Return the margin of each sale lines
'''
2012-10-16 17:10:24 +02:00
Currency = Pool().get('currency.currency')
2014-11-05 17:17:48 +01:00
if not self.sale or not self.sale.currency:
2014-11-21 08:58:33 +01:00
return Decimal(0)
2014-08-30 15:16:39 +02:00
currency = self.sale.currency
2012-10-16 17:10:24 +02:00
if self.type == 'line':
if self.quantity and self.cost_price:
cost = Decimal(str(self.quantity)) * (self.cost_price)
else:
2014-11-21 08:58:33 +01:00
cost = Decimal(0)
self.amount = self.on_change_with_amount()
2014-08-30 15:16:39 +02:00
return Currency.round(currency, self.amount - cost)
elif self.type == 'subtotal':
2014-11-21 08:58:33 +01:00
cost = Decimal(0)
2014-08-30 15:16:39 +02:00
for line2 in self.sale.lines:
if self == line2:
return cost
2014-08-30 15:16:39 +02:00
if line2.type == 'line':
cost2 = Decimal(str(line2.quantity)) * (line2.cost_price or
2014-11-21 08:58:33 +01:00
Decimal(0))
2014-08-30 15:16:39 +02:00
cost += Currency.round(currency, line2.amount - cost2)
elif line2.type == 'subtotal':
2014-11-21 08:58:33 +01:00
cost = Decimal(0)
2012-10-16 17:10:24 +02:00
else:
2014-11-21 08:58:33 +01:00
return Decimal(0)
@fields.depends('type', 'quantity', 'cost_price', 'unit_price',
'_parent_sale.currency', '_parent_sale.lines', methods=['margin'])
def on_change_with_margin_percent(self, name=None):
Configuration = Pool().get('sale.configuration')
config = Configuration(1)
sale_margin_method = config.sale_margin_method or 'cost_price'
if self.type not in ('line', 'subtotal'):
return
self.margin = self.on_change_with_margin()
if not self.margin:
return
if self.type == 'line':
if not self.quantity:
return
if sale_margin_method == 'cost_price' and not self.cost_price:
return Decimal('1.0')
if sale_margin_method == 'unit_price' and not self.unit_price:
return Decimal('1.0')
if sale_margin_method == 'unit_price':
price = self.get_margin_unit_price()
else:
price = self.get_margin_cost_price()
return (self.margin / price).quantize(Decimal('0.0001'))
else:
price = Decimal(0)
for line2 in self.sale.lines:
if self == line2:
if not price:
return Decimal('1.0')
else:
2018-02-01 17:23:58 +01:00
return (
self.margin / price).quantize(Decimal('0.0001'))
if line2.type == 'line':
if sale_margin_method == 'unit_price':
price += line2.get_margin_unit_price()
else:
price += line2.get_margin_cost_price()
elif line2.type == 'subtotal':
price = Decimal(0)
def get_margin_cost_price(self):
return Decimal(str(fabs(self.quantity))) * (self.cost_price or
Decimal(0))
def get_margin_unit_price(self):
return Decimal(str(fabs(self.quantity))) * (self.unit_price or
Decimal(0))