sale_loyalty/models/sale_order_line.py

120 lines
5.6 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo import api, fields, models
class SaleOrderLine(models.Model):
_inherit = 'sale.order.line'
is_reward_line = fields.Boolean(
string="Is a program reward line", compute='_compute_is_reward_line')
reward_id = fields.Many2one(
comodel_name='loyalty.reward', ondelete='restrict', readonly=True)
coupon_id = fields.Many2one(
comodel_name='loyalty.card', ondelete='restrict', readonly=True)
reward_identifier_code = fields.Char(
help="Technical field used to link multiple reward lines from the same reward together.")
points_cost = fields.Float(help="How much point this reward costs on the loyalty card.")
def _compute_name(self):
# Avoid computing the name for reward lines
reward = self.filtered('reward_id')
super(SaleOrderLine, self - reward)._compute_name()
@api.depends('reward_id')
def _compute_is_reward_line(self):
for line in self:
line.is_reward_line = bool(line.reward_id)
def _compute_tax_id(self):
reward_lines = self.filtered('is_reward_line')
super(SaleOrderLine, self - reward_lines)._compute_tax_id()
# Discount reward line is split per tax, the discount is set on the line but not on the product
# as the product is the generic discount line.
# In case of a free product, retrieving the tax on the line instead of the product won't affect the behavior.
for line in reward_lines:
line = line.with_company(line.company_id)
fpos = line.order_id.fiscal_position_id or line.order_id.fiscal_position_id._get_fiscal_position(line.order_partner_id)
# If company_id is set, always filter taxes by the company
taxes = line.tax_id.filtered(lambda r: not line.company_id or r.company_id == line.company_id)
line.tax_id = fpos.map_tax(taxes)
def _get_display_price(self):
# A product created from a promotion does not have a list_price.
# The price_unit of a reward order line is computed by the promotion, so it can be used directly
if self.is_reward_line and self.reward_id.reward_type != 'product':
return self.price_unit
return super()._get_display_price()
def _is_not_sellable_line(self):
return self.is_reward_line or super()._is_not_sellable_line()
def _reset_loyalty(self, complete=False):
"""
Reset the line(s) to a state which does not impact reward computation.
If complete is set to True we also remove the coupon and reward from the line(s).
This option should be used when the line will be unlinked.
Returns self
"""
vals = {
'points_cost': 0,
'price_unit': 0,
}
if complete:
vals.update({
'coupon_id': False,
'reward_id': False,
})
self.write(vals)
return self
@api.model_create_multi
def create(self, vals_list):
res = super().create(vals_list)
# Update our coupon points if the order is in a confirmed state
for line in res:
if line.coupon_id and line.points_cost and line.state == 'sale':
line.coupon_id.points -= line.points_cost
return res
def write(self, vals):
cost_in_vals = 'points_cost' in vals
if cost_in_vals:
previous_cost = {l: l.points_cost for l in self}
res = super().write(vals)
if cost_in_vals:
# Update our coupon points if the order is in a confirmed state
for line in self:
if previous_cost[line] != line.points_cost and line.state == 'sale':
line.coupon_id.points += (previous_cost[line] - line.points_cost)
return res
def unlink(self):
# Remove related reward lines
reward_coupon_set = {(l.reward_id, l.coupon_id, l.reward_identifier_code) for l in self if l.reward_id}
related_lines = self.env['sale.order.line']
related_lines |= self.order_id.order_line.filtered(lambda l: (l.reward_id, l.coupon_id, l.reward_identifier_code) in reward_coupon_set)
# Remove the line's coupon from order if it is the last line using that coupon
coupons_to_unlink = self.env['loyalty.card']
for line in self:
if line.coupon_id:
# 2 cases:
# case 1: coupon has been applied directly
# case 2: coupon was created from a program
if line.coupon_id in line.order_id.applied_coupon_ids:
line.order_id.applied_coupon_ids -= line.coupon_id
elif line.coupon_id.order_id == line.order_id and line.coupon_id.program_id.applies_on == 'current' and\
not any(oLine.coupon_id == line.coupon_id and oLine not in related_lines for oLine in line.order_id.order_line):
# ondelete='restrict' would prevent deletion of the coupon unlink after unlinking lines
coupons_to_unlink |= line.coupon_id
line.order_id.code_enabled_rule_ids = line.order_id.code_enabled_rule_ids.filtered(lambda r: r.program_id != line.coupon_id.program_id)
# Give back the points if the order is confirmed, points are given back if the order is cancelled but in this case we need to do it directly
for line in related_lines:
if line.state == 'sale':
line.coupon_id.points += line.points_cost
res = super(SaleOrderLine, self | related_lines).unlink()
coupons_to_unlink.sudo().unlink()
return res