summaryrefslogtreecommitdiff
path: root/indoteknik_custom/models/promotion_program_line.py
blob: 71418d6c305b5d3b851c39f3ad8d1fb6755d2641 (plain)
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
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
from odoo import fields, models, api
from datetime import datetime
import math


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(string="Promotion Type", 
        selection=[
            ("special_price", "Special Price"),
            ("bundling", "Bundling"),
            ("discount_loading", "Discount Loading"),
            ("merchandise", "Merchandise")
        ], 
        help='- Special Price: Potongan harga barang.\n'
            '- Bundling: Menggabungkan beberapa produk menjadi satu paket.\n'
            '- Discount Loading: Semakin banyak unit yang dibeli maka semakin murah harga yang dibayar.\n'
            '- Merchandise: Pemberian barang gratis.'
    )
    minimum_purchase_qty = fields.Integer(string="Minimum Purchase Qty", help="Minimum unit barang yang perlu dibeli untuk dapat menggunakan promosi")
    applies_multiply = fields.Boolean(string="Applies Multiply", help="Diterapkan berlipat ganda")
    limit_qty = fields.Integer(string="Limit Qty", help="Kuota promosi keseluruhan")
    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, qty = 0):
        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
        
        qty_can_buy = min(remaining_qty['user'], qty)
        if self.limit_qty_transaction > 0:
            qty_can_buy = min(qty_can_buy, self.limit_qty_transaction)
        
        multiplier = 0
        if qty_can_buy > self.minimum_purchase_qty:
            multiplier = 1
        
        if self.applies_multiply and qty_can_buy > 0:
            multiplier = math.floor(qty_can_buy / self.minimum_purchase_qty)

        response = {
            '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),
            'items': [{
                'product_id': line.product_id.id,
                'quantity': line.qty * multiplier
            } for line in self.line_free_item]
        }

        return response
    
    def res_format(self, user):
        data = [x.format(user) for x in self]
        return data