1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
|
from odoo import fields, models, api
from datetime import datetime
class PromotionProgramLine(models.Model):
_name = "promotion.program.line"
name = fields.Char(string="Name")
image = fields.Binary(string="Image")
product_id = fields.Many2one(comodel_name="product.product", string="Product Variant")
program_id = fields.Many2one(comodel_name="promotion.program", string="Program")
discount_type = fields.Selection(selection=[
("percentage", "Percentage"),
("fixed_price", "Fixed Price"),
], string="Discount Type")
discount_amount = fields.Float(string="Discount Amount")
promotion_type = fields.Selection(selection=[
("special_price", "Special Price"),
("bundling", "Bundling"),
("discount_loading", "Discount Loading"),
("merchandise", "Merchandise")
], string="Promotion Type")
minimum_purchase_qty = fields.Integer(string="Minimum Purchase Qty", help="Minimum Qty to applied discount loading")
applies_multiply = fields.Boolean(string="Applies Multiply", help="Is applies multiply")
limit_qty = fields.Integer(string="Limit Qty", help="Limit Qty product in promotion")
limit_qty_user = fields.Integer(string="Limit Qty / User", help="Limit Qty per User")
limit_qty_transaction = fields.Integer(string="Limit Qty / Transaction", help="Limit Qty per Transaction")
line_free_item = fields.One2many(comodel_name="promotion.program.free_item", inverse_name="line_id", string="Line Free Item")
display_on_homepage = fields.Boolean(string="Display on Homepage")
order_line_ids = fields.One2many('sale.order.line', 'program_line_id')
@api.onchange('product_id')
def _onchange_product_id(self):
if self.product_id and not self.name:
self.name = self.product_id.display_name
self._discount_loading_auto()
@api.onchange('promotion_type')
def onchange_promotion_type(self):
self._discount_loading_auto()
def _discount_loading_auto(self):
program_line = self
product = program_line.product_id
promotion_type = program_line.promotion_type
if promotion_type != 'discount_loading' or not product:
return
line = self.browse(self.ids)
line.product_id = self.product_id.id
line.promotion_type = self.promotion_type
product_added = False
line_free_item = program_line.line_free_item
for line in line_free_item:
if line.product_id.id == product.id:
product_added = True
continue
line.unlink()
if not product_added:
data = {'product_id': product.id,
'qty': 1, 'line_id': program_line.id}
line_free_item.create(data)
def calculate_price(self, price):
initial_price = price['price']
if self.discount_type == 'percentage':
price['discount_percentage'] = self.discount_amount
price['price_discount'] = initial_price - (initial_price * self.discount_amount / 100)
elif self.discount_type == 'fixed_price':
price['price_discount'] = self.discount_amount
price['discount_percentage'] = round((initial_price - self.discount_amount) / initial_price * 100)
return price
def get_active_promotions(self, product_id):
current_time = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
return self.search([
('program_id.start_time', '<=', current_time),
('program_id.end_time', '>=', current_time),
('product_id', '=', product_id)
])
def _get_remaining_qty(self, user):
remaining_qty = {
'all': self.limit_qty,
'user': self.limit_qty_user,
'transaction': self.limit_qty_transaction
}
for order in self.order_line_ids:
parent_order = order.order_id
if parent_order.state != 'cancel':
remaining_qty['all'] -= order.product_uom_qty
if user and parent_order.partner_id.id == user['partner_id']:
remaining_qty['user'] -= order.product_uom_qty
if remaining_qty['all'] < remaining_qty['user']:
remaining_qty['user'] = remaining_qty['all']
if remaining_qty['user'] < remaining_qty['transaction']:
remaining_qty['transaction'] = remaining_qty['user']
return remaining_qty
def _get_remaining_time(self):
calculate_time = self.program_id.end_time - datetime.now()
return round(calculate_time.total_seconds())
def _res_limit_qty(self):
return {
'all': self.limit_qty,
'user': self.limit_qty_user,
'transaction': self.limit_qty_transaction,
}
def _res_promotion_type(self):
return {
'value': self.promotion_type,
'label': dict(self._fields['promotion_type'].selection).get(self.promotion_type)
}
def format(self, user = None):
ir_attachment = self.env['ir.attachment']
product_price = self.product_id.calculate_website_price()
limit_qty = self._res_limit_qty()
remaining_qty = self._get_remaining_qty(user)
percent_remaining = 0
if limit_qty['all'] > 0:
percent_remaining = (limit_qty['all'] - remaining_qty['all']) / limit_qty['all'] * 100
return {
'id': self.id,
'name': self.name,
'image': ir_attachment.api_image('promotion.program.line', 'image', self.id),
'minimum_purchase_qty': self.minimum_purchase_qty,
'applies_multiply': self.applies_multiply,
'remaining_time': self._get_remaining_time(),
'type': self._res_promotion_type(),
'limit_qty': limit_qty,
'remaining_qty': remaining_qty,
'used_percentage': percent_remaining,
'price': self.calculate_price(price=product_price)
}
def res_format(self, user):
data = [x.format(user) for x in self]
return data
def res_format_cart(self, user):
data = self.format(user)
return data
|