Files
2026-03-14 09:42:12 +00:00

183 lines
6.9 KiB
Python

# This file is part of Tryton. The COPYRIGHT file at the top level of
# this repository contains the full copyright notices and license terms.
from decimal import ROUND_DOWN, ROUND_HALF_EVEN, Decimal
from operator import itemgetter
from trytond.model import ModelView, Workflow, fields
from trytond.modules.product import price_digits, round_price
from trytond.modules.stock_shipment_cost import ShipmentCostMixin
from trytond.pool import Pool, PoolMeta
from trytond.pyson import Eval
class ShipmentIn(ShipmentCostMixin, metaclass=PoolMeta):
__name__ = 'stock.shipment.in'
@fields.depends('state')
def on_change_with_shipment_cost_readonly(self, name=None):
return self.state != 'draft'
@property
def shipment_cost_moves(self):
return [
m for m in self.incoming_moves
if m.state not in {'done', 'cancelled'}]
def allocate_cost_by_value(self):
pool = Pool()
Currency = pool.get('currency.currency')
Move = pool.get('stock.move')
cost = self._get_shipment_cost()
if not cost:
return
moves = self.shipment_cost_moves
sum_value = 0
unit_prices = {}
for move in moves:
unit_price = Currency.compute(move.currency, move.unit_price,
self.company.currency, round=False)
unit_prices[move.id] = unit_price
sum_value += unit_price * Decimal(str(move.quantity))
costs = []
digit = Move.unit_price.digits[1]
exp = Decimal(str(10.0 ** -digit))
difference = cost
for move in moves:
quantity = Decimal(str(move.quantity))
if not sum_value:
move_cost = cost / Decimal(len(moves))
else:
move_cost = cost * quantity * unit_prices[move.id] / sum_value
unit_shipment_cost = round_price(
move_cost / quantity, rounding=ROUND_DOWN)
costs.append({
'unit_shipment_cost': unit_shipment_cost,
'difference': move_cost - (unit_shipment_cost * quantity),
'move': move,
})
difference -= unit_shipment_cost * quantity
costs.sort(key=itemgetter('difference'))
for cost in costs:
move = cost['move']
quantity = Decimal(str(move.quantity))
if exp * quantity < difference:
cost['unit_shipment_cost'] += exp
difference -= exp * quantity
if difference < exp:
break
for cost in costs:
move = cost['move']
unit_shipment_cost = Currency.compute(
self.company.currency, cost['unit_shipment_cost'],
move.currency, round=False)
unit_shipment_cost = round_price(
unit_shipment_cost, rounding=ROUND_HALF_EVEN)
move.unit_price += unit_shipment_cost
move.unit_shipment_cost = unit_shipment_cost
Move.save(moves)
@classmethod
@ModelView.button
@Workflow.transition('received')
def receive(cls, shipments):
Carrier = Pool().get('carrier')
for shipment in shipments:
shipment.cost = shipment.cost_used
shipment.cost_currency = shipment.cost_currency_used
if shipment.carrier:
allocation_method = \
shipment.carrier.carrier_cost_allocation_method
else:
allocation_method = \
Carrier.default_carrier_cost_allocation_method()
getattr(shipment, 'allocate_cost_by_%s' % allocation_method)()
super().receive(shipments)
@classmethod
@ModelView.button
@Workflow.transition('cancelled')
def cancel(cls, shipments):
for shipment in shipments:
shipment.cost = None
shipment.cost_currency = None
cls.save(shipments)
super().cancel(shipments)
class Move(metaclass=PoolMeta):
__name__ = 'stock.move'
unit_shipment_cost = fields.Numeric(
"Unit Shipment Cost", digits=price_digits, readonly=True,
states={
'invisible': ~Eval('unit_shipment_cost'),
})
def _compute_unit_price(self, unit_price):
if self.unit_shipment_cost:
unit_price -= self.unit_shipment_cost
unit_price = super()._compute_unit_price(unit_price)
if self.unit_shipment_cost:
unit_price += self.unit_shipment_cost
return unit_price
def _compute_component_unit_price(self, unit_price):
if self.unit_shipment_cost:
unit_price -= self.unit_shipment_cost
unit_price = super()._compute_component_unit_price(unit_price)
if self.unit_shipment_cost:
unit_price += self.unit_shipment_cost
return unit_price
# Split the shipment cost if account_stock_continental is activated
def _get_account_stock_move_lines(self, type_):
pool = Pool()
AccountMoveLine = pool.get('account.move.line')
Currency = pool.get('currency.currency')
move_lines = super()._get_account_stock_move_lines(type_)
if (type_.startswith('in_')
and self.unit_shipment_cost
and self.shipment
and self.shipment.carrier):
shipment_cost = Currency.compute(self.currency,
Decimal(str(self.quantity)) * self.unit_shipment_cost,
self.company.currency)
shipment_cost_account = \
self.shipment.carrier.carrier_product.account_expense_used
account = self.product.account_stock_in_used
for move_line in move_lines:
if move_line.account == account:
move_line.credit -= shipment_cost
shipment_cost_line = AccountMoveLine(
debit=Decimal('0'),
credit=shipment_cost,
account=shipment_cost_account,
)
move_lines.append(shipment_cost_line)
break
else:
raise AssertionError('missing account_stock_supplier')
return move_lines
# Remove shipment cost if account_stock_anglo_saxon is activated
@classmethod
def _get_anglo_saxon_move(cls, moves, quantity, type_):
pool = Pool()
Currency = pool.get('currency.currency')
Uom = pool.get('product.uom')
for move, qty, cost_price in super()._get_anglo_saxon_move(
moves, quantity, type_):
if (type_.startswith('in_')
and move.unit_shipment_cost):
shipment_cost = Uom.compute_price(
move.unit, move.unit_shipment_cost,
move.product.default_uom)
shipment_cost = Currency.compute(move.currency,
shipment_cost, move.company.currency)
cost_price -= shipment_cost
yield move, qty, cost_price