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

122 lines
3.8 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 sql.operators import Equal
from trytond.model import Exclude, ModelSQL, ModelView, Workflow, fields
from trytond.pool import Pool, PoolMeta
from trytond.pyson import Bool, Eval, If
from trytond.sql.functions import DateRange
from trytond.sql.operators import RangeOverlap
class SubscriptionService(metaclass=PoolMeta):
__name__ = 'sale.subscription.service'
asset_lots = fields.Many2Many(
'sale.subscription.service-stock.lot.asset',
'service', 'lot', "Asset Lots",
domain=[
('product.type', '=', 'assets'),
])
asset_lots_available = fields.Many2Many(
'sale.subscription.service-stock.lot.asset',
'service', 'lot', "Available Asset Lots", readonly=True,
domain=[
('product.type', '=', 'assets'),
],
filter=[
('subscribed', '=', None),
])
class SubscriptionServiceStockLot(ModelSQL):
__name__ = 'sale.subscription.service-stock.lot.asset'
service = fields.Many2One(
'sale.subscription.service', "Service",
ondelete='CASCADE', required=True)
lot = fields.Many2One(
'stock.lot', "Lot", ondelete='CASCADE', required=True,
domain=[
('product.type', '=', 'assets'),
])
class Subscription(metaclass=PoolMeta):
__name__ = 'sale.subscription'
@classmethod
@ModelView.button
@Workflow.transition('cancelled')
def cancel(cls, subscriptions):
pool = Pool()
SubscriptionLine = pool.get('sale.subscription.line')
sub_lines = [l for s in subscriptions for l in s.lines if l.asset_lot]
SubscriptionLine.write(sub_lines, {'asset_lot': None})
super().cancel(subscriptions)
@classmethod
@ModelView.button
@Workflow.transition('running')
def run(cls, subscriptions):
pool = Pool()
Line = pool.get('sale.subscription.line')
super().run(subscriptions)
lines = [l for s in subscriptions for l in s.lines]
Line._validate(lines, ['asset_lot'])
class SubscriptionLine(metaclass=PoolMeta):
__name__ = 'sale.subscription.line'
asset_lot = fields.Many2One('stock.lot', "Asset Lot",
domain=[
('subscription_services', '=', Eval('service', -1)),
],
states={
'required': ((Eval('subscription_state') == 'running')
& Eval('asset_lot_required')),
'invisible': ~Eval('asset_lot_required'),
'readonly': Eval('subscription_state') != 'draft',
})
asset_lot_required = fields.Function(
fields.Boolean("Asset Lot Required"),
'on_change_with_asset_lot_required')
@classmethod
def __setup__(cls):
super().__setup__()
t = cls.__table__()
cls._sql_constraints += [
('dates_asset_lot_overlap',
Exclude(t,
(t.asset_lot, Equal),
(DateRange(t.start_date, t.end_date, '[)'), RangeOverlap),
),
'sale_subscription_asset.msg_asset_line_overlap'),
]
cls.quantity.domain = [
cls.quantity.domain,
If(Bool(Eval('asset_lot')),
('quantity', '=', 1),
()),
]
@fields.depends('service')
def on_change_with_asset_lot_required(self, name=None):
if not self.service:
return False
return bool(self.service.asset_lots)
@classmethod
def copy(cls, lines, default=None):
if default is None:
default = {}
else:
default = default.copy()
default.setdefault('asset_lot')
return super().copy(lines, default)