# -*- coding: utf-8 -*- from odoo import fields, models, api from odoo.tools import float_is_zero class AccountMove(models.Model): _inherit = 'account.move' stock_move_id = fields.Many2one('stock.move', string='Stock Move', index='btree_not_null') stock_valuation_layer_ids = fields.One2many('stock.valuation.layer', 'account_move_id', string='Stock Valuation Layer') def _compute_show_reset_to_draft_button(self): super()._compute_show_reset_to_draft_button() for move in self: if move.sudo().line_ids.stock_valuation_layer_ids: move.show_reset_to_draft_button = False # ------------------------------------------------------------------------- # OVERRIDE METHODS # ------------------------------------------------------------------------- def _get_lines_onchange_currency(self): # OVERRIDE return self.line_ids.filtered(lambda l: l.display_type != 'cogs') def copy_data(self, default=None): # OVERRIDE # Don't keep anglo-saxon lines when copying a journal entry. res = super().copy_data(default=default) if not self._context.get('move_reverse_cancel'): for copy_vals in res: if 'line_ids' in copy_vals: copy_vals['line_ids'] = [line_vals for line_vals in copy_vals['line_ids'] if line_vals[0] != 0 or line_vals[2].get('display_type') != 'cogs'] return res def _post(self, soft=True): # OVERRIDE # Don't change anything on moves used to cancel another ones. if self._context.get('move_reverse_cancel'): return super()._post(soft) # Create additional COGS lines for customer invoices. self.env['account.move.line'].create(self._stock_account_prepare_anglo_saxon_out_lines_vals()) # Post entries. posted = super()._post(soft) # Reconcile COGS lines in case of anglo-saxon accounting with perpetual valuation. if not self.env.context.get('skip_cogs_reconciliation'): posted._stock_account_anglo_saxon_reconcile_valuation() return posted def button_draft(self): res = super(AccountMove, self).button_draft() # Unlink the COGS lines generated during the 'post' method. self.mapped('line_ids').filtered(lambda line: line.display_type == 'cogs').unlink() return res def button_cancel(self): # OVERRIDE res = super(AccountMove, self).button_cancel() # Unlink the COGS lines generated during the 'post' method. # In most cases it shouldn't be necessary since they should be unlinked with 'button_draft'. # However, since it can be called in RPC, better be safe. self.mapped('line_ids').filtered(lambda line: line.display_type == 'cogs').unlink() return res # ------------------------------------------------------------------------- # COGS METHODS # ------------------------------------------------------------------------- def _stock_account_prepare_anglo_saxon_out_lines_vals(self): ''' Prepare values used to create the journal items (account.move.line) corresponding to the Cost of Good Sold lines (COGS) for customer invoices. Example: Buy a product having a cost of 9 being a storable product and having a perpetual valuation in FIFO. Sell this product at a price of 10. The customer invoice's journal entries looks like: Account | Debit | Credit --------------------------------------------------------------- 200000 Product Sales | | 10.0 --------------------------------------------------------------- 101200 Account Receivable | 10.0 | --------------------------------------------------------------- This method computes values used to make two additional journal items: --------------------------------------------------------------- 220000 Expenses | 9.0 | --------------------------------------------------------------- 101130 Stock Interim Account (Delivered) | | 9.0 --------------------------------------------------------------- Note: COGS are only generated for customer invoices except refund made to cancel an invoice. :return: A list of Python dictionary to be passed to env['account.move.line'].create. ''' lines_vals_list = [] price_unit_prec = self.env['decimal.precision'].precision_get('Product Price') for move in self: # Make the loop multi-company safe when accessing models like product.product move = move.with_company(move.company_id) if not move.is_sale_document(include_receipts=True) or not move.company_id.anglo_saxon_accounting: continue for line in move.invoice_line_ids: # Filter out lines being not eligible for COGS. if not line._eligible_for_cogs(): continue # Retrieve accounts needed to generate the COGS. accounts = line.product_id.product_tmpl_id.get_product_accounts(fiscal_pos=move.fiscal_position_id) debit_interim_account = accounts['stock_output'] credit_expense_account = accounts['expense'] or move.journal_id.default_account_id if not debit_interim_account or not credit_expense_account: continue # Compute accounting fields. sign = -1 if move.move_type == 'out_refund' else 1 price_unit = line._stock_account_get_anglo_saxon_price_unit() amount_currency = sign * line.quantity * price_unit if move.currency_id.is_zero(amount_currency) or float_is_zero(price_unit, precision_digits=price_unit_prec): continue # Add interim account line. lines_vals_list.append({ 'name': line.name[:64], 'move_id': move.id, 'partner_id': move.commercial_partner_id.id, 'product_id': line.product_id.id, 'product_uom_id': line.product_uom_id.id, 'quantity': line.quantity, 'price_unit': price_unit, 'amount_currency': -amount_currency, 'account_id': debit_interim_account.id, 'display_type': 'cogs', 'tax_ids': [], 'cogs_origin_id': line.id, }) # Add expense account line. lines_vals_list.append({ 'name': line.name[:64], 'move_id': move.id, 'partner_id': move.commercial_partner_id.id, 'product_id': line.product_id.id, 'product_uom_id': line.product_uom_id.id, 'quantity': line.quantity, 'price_unit': -price_unit, 'amount_currency': amount_currency, 'account_id': credit_expense_account.id, 'analytic_distribution': line.analytic_distribution, 'display_type': 'cogs', 'tax_ids': [], 'cogs_origin_id': line.id, }) return lines_vals_list def _stock_account_get_last_step_stock_moves(self): """ To be overridden for customer invoices and vendor bills in order to return the stock moves related to the invoices in self. """ return self.env['stock.move'] def _stock_account_anglo_saxon_reconcile_valuation(self, product=False): """ Reconciles the entries made in the interim accounts in anglosaxon accounting, reconciling stock valuation move lines with the invoice's. """ for move in self: if not move.is_invoice(): continue if not move.company_id.anglo_saxon_accounting: continue stock_moves = move._stock_account_get_last_step_stock_moves() # In case we return a return, we have to provide the related AMLs so all can be reconciled stock_moves |= stock_moves.origin_returned_move_id if not stock_moves: continue products = product or move.mapped('invoice_line_ids.product_id') for prod in products: if prod.valuation != 'real_time': continue # We first get the invoices move lines (taking the invoice and the previous ones into account)... product_accounts = prod.product_tmpl_id._get_product_accounts() if move.is_sale_document(): product_interim_account = product_accounts['stock_output'] else: product_interim_account = product_accounts['stock_input'] if product_interim_account.reconcile: # Search for anglo-saxon lines linked to the product in the journal entry. product_account_moves = move.line_ids.filtered( lambda line: line.product_id == prod and line.account_id == product_interim_account and not line.reconciled) # Search for anglo-saxon lines linked to the product in the stock moves. product_stock_moves = stock_moves._get_all_related_sm(prod) product_account_moves |= product_stock_moves._get_all_related_aml().filtered( lambda line: line.account_id == product_interim_account and not line.reconciled and line.move_id.state == "posted" ) stock_aml = product_account_moves.filtered(lambda aml: aml.move_id.sudo().stock_valuation_layer_ids.stock_move_id) invoice_aml = product_account_moves.filtered(lambda aml: aml.move_id == move) correction_amls = product_account_moves - stock_aml - invoice_aml # Reconcile. if correction_amls: if sum(correction_amls.mapped('balance')) > 0 or all(aml.is_same_currency for aml in correction_amls): product_account_moves.with_context(no_exchange_difference=True).reconcile() else: (invoice_aml | correction_amls).with_context(no_exchange_difference=True).reconcile() (invoice_aml.filtered(lambda aml: not aml.reconciled) | stock_aml).with_context(no_exchange_difference=True).reconcile() else: product_account_moves.reconcile() def _get_invoiced_lot_values(self): return [] class AccountMoveLine(models.Model): _inherit = 'account.move.line' stock_valuation_layer_ids = fields.One2many('stock.valuation.layer', 'account_move_line_id', string='Stock Valuation Layer') cogs_origin_id = fields.Many2one( # technical field used to keep track in the originating line of the anglo-saxon lines comodel_name="account.move.line", copy=False, index="btree_not_null", ) def _compute_account_id(self): super()._compute_account_id() input_lines = self.filtered(lambda line: ( line._can_use_stock_accounts() and line.move_id.company_id.anglo_saxon_accounting and line.move_id.is_purchase_document() )) for line in input_lines: line = line.with_company(line.move_id.journal_id.company_id) fiscal_position = line.move_id.fiscal_position_id accounts = line.product_id.product_tmpl_id.get_product_accounts(fiscal_pos=fiscal_position) if accounts['stock_input']: line.account_id = accounts['stock_input'] def _eligible_for_cogs(self): self.ensure_one() return self.product_id.type == 'product' and self.product_id.valuation == 'real_time' def _get_gross_unit_price(self): price_unit = self.price_subtotal / self.quantity return -price_unit if self.move_id.move_type == 'in_refund' else price_unit def _get_stock_valuation_layers(self, move): valued_moves = self._get_valued_in_moves() if move.move_type == 'in_refund': valued_moves = valued_moves.filtered(lambda stock_move: stock_move._is_out()) else: valued_moves = valued_moves.filtered(lambda stock_move: stock_move._is_in()) return valued_moves.stock_valuation_layer_ids def _get_valued_in_moves(self): return self.env['stock.move'] def _can_use_stock_accounts(self): return self.product_id.type == 'product' and self.product_id.categ_id.property_valuation == 'real_time' def _stock_account_get_anglo_saxon_price_unit(self): self.ensure_one() if not self.product_id: return self.price_unit original_line = self.move_id.reversed_entry_id.line_ids.filtered( lambda l: l.display_type == 'cogs' and l.product_id == self.product_id and l.product_uom_id == self.product_uom_id and l.price_unit >= 0) original_line = original_line and original_line[0] return original_line.price_unit if original_line \ else self.product_id.with_company(self.company_id)._stock_account_get_anglo_saxon_price_unit(uom=self.product_uom_id) @api.onchange('product_id') def _inverse_product_id(self): super(AccountMoveLine, self.filtered(lambda l: l.display_type != 'cogs'))._inverse_product_id()