# -*- coding: utf-8 -*- # Part of Odoo. See LICENSE file for full copyright and licensing details. import logging from datetime import datetime from markupsafe import Markup from functools import partial from itertools import groupby from collections import defaultdict import psycopg2 import pytz import re from odoo import api, fields, models, tools, _ from odoo.tools import float_is_zero, float_round, float_repr, float_compare from odoo.exceptions import ValidationError, UserError from odoo.osv.expression import AND import base64 _logger = logging.getLogger(__name__) class PosOrder(models.Model): _name = "pos.order" _inherit = ["portal.mixin"] _description = "Point of Sale Orders" _order = "date_order desc, name desc, id desc" @api.model def _amount_line_tax(self, line, fiscal_position_id): taxes = line.tax_ids.filtered_domain(self.env['account.tax']._check_company_domain(line.order_id.company_id)) taxes = fiscal_position_id.map_tax(taxes) price = line.price_unit * (1 - (line.discount or 0.0) / 100.0) taxes = taxes.compute_all(price, line.order_id.currency_id, line.qty, product=line.product_id, partner=line.order_id.partner_id or False)['taxes'] return sum(tax.get('amount', 0.0) for tax in taxes) @api.model def _order_fields(self, ui_order): process_line = partial(self.env['pos.order.line']._order_line_fields, session_id=ui_order['pos_session_id']) return { 'user_id': ui_order['user_id'] or False, 'session_id': ui_order['pos_session_id'], 'lines': [process_line(l) for l in ui_order['lines']] if ui_order['lines'] else False, 'pos_reference': ui_order['name'], 'sequence_number': ui_order['sequence_number'], 'partner_id': ui_order['partner_id'] or False, 'date_order': ui_order['date_order'].replace('T', ' ')[:19], 'fiscal_position_id': ui_order['fiscal_position_id'], 'pricelist_id': ui_order.get('pricelist_id'), 'amount_paid': ui_order['amount_paid'], 'amount_total': ui_order['amount_total'], 'amount_tax': ui_order['amount_tax'], 'amount_return': ui_order['amount_return'], 'company_id': self.env['pos.session'].browse(ui_order['pos_session_id']).company_id.id, 'to_invoice': ui_order['to_invoice'] if "to_invoice" in ui_order else False, 'shipping_date': ui_order['shipping_date'] if "shipping_date" in ui_order else False, 'is_tipped': ui_order.get('is_tipped', False), 'tip_amount': ui_order.get('tip_amount', 0), 'access_token': ui_order.get('access_token', ''), 'ticket_code': ui_order.get('ticket_code', ''), 'last_order_preparation_change': ui_order.get('last_order_preparation_change', '{}'), } @api.model def _payment_fields(self, order, ui_paymentline): return { 'amount': ui_paymentline['amount'] or 0.0, 'payment_date': ui_paymentline['name'], 'payment_method_id': ui_paymentline['payment_method_id'], 'card_type': ui_paymentline.get('card_type'), 'cardholder_name': ui_paymentline.get('cardholder_name'), 'transaction_id': ui_paymentline.get('transaction_id'), 'payment_status': ui_paymentline.get('payment_status'), 'ticket': ui_paymentline.get('ticket'), 'pos_order_id': order.id, } # This deals with orders that belong to a closed session. In order # to recover from this situation we create a new rescue session, # making it obvious that something went wrong. # A new, separate, rescue session is preferred for every such recovery, # to avoid adding unrelated orders to live sessions. def _get_valid_session(self, order): PosSession = self.env['pos.session'] closed_session = PosSession.browse(order['pos_session_id']) _logger.warning('session %s (ID: %s) was closed but received order %s (total: %s) belonging to it', closed_session.name, closed_session.id, order['name'], order['amount_total']) rescue_session = PosSession.search([ ('state', 'not in', ('closed', 'closing_control')), ('rescue', '=', True), ('config_id', '=', closed_session.config_id.id), ], limit=1) if rescue_session: _logger.warning('reusing recovery session %s for saving order %s', rescue_session.name, order['name']) return rescue_session _logger.warning('attempting to create recovery session for saving order %s', order['name']) new_session = PosSession.create({ 'config_id': closed_session.config_id.id, 'name': _('(RESCUE FOR %(session)s)', session=closed_session.name), 'rescue': True, # avoid conflict with live sessions }) # bypass opening_control (necessary when using cash control) new_session.action_pos_session_open() if new_session.config_id.cash_control and new_session.rescue: last_session = self.env['pos.session'].search([('config_id', '=', new_session.config_id.id), ('id', '!=', new_session.id)], limit=1) new_session.cash_register_balance_start = last_session.cash_register_balance_end_real return new_session @api.depends('sequence_number', 'session_id') def _compute_tracking_number(self): for record in self: record.tracking_number = str((record.session_id.id % 10) * 100 + record.sequence_number % 100).zfill(3) @api.model def _process_order(self, order, draft, existing_order): """Create or update an pos.order from a given dictionary. :param dict order: dictionary representing the order. :param bool draft: Indicate that the pos_order is not validated yet. :param existing_order: order to be updated or False. :type existing_order: pos.order. :returns: id of created/updated pos.order :rtype: int """ order = order['data'] pos_session = self.env['pos.session'].browse(order['pos_session_id']) if pos_session.state == 'closing_control' or pos_session.state == 'closed': order['pos_session_id'] = self._get_valid_session(order).id if order.get('partner_id'): partner_id = self.env['res.partner'].browse(order['partner_id']) if not partner_id.exists(): order.update({ "partner_id": False, "to_invoice": False, }) pos_order = False if not existing_order: pos_order = self.create(self._order_fields(order)) else: pos_order = existing_order pos_order.lines.unlink() order['user_id'] = pos_order.user_id.id pos_order.write(self._order_fields(order)) pos_order._link_combo_items(order) pos_order = pos_order.with_company(pos_order.company_id) self = self.with_company(pos_order.company_id) self._process_payment_lines(order, pos_order, pos_session, draft) return pos_order._process_saved_order(draft) def _link_combo_items(self, order_vals): self.ensure_one() lines = [l[2] for l in order_vals['lines'] if l[2].get('combo_parent_id') or l[2].get('combo_line_ids')] uuid_by_cid = {line['id']: line['uuid'] for line in lines} line_by_uuid = {line.uuid: line for line in self.lines.filtered_domain([("uuid", "in", [line['uuid'] for line in lines])])} for line in lines: if line.get('combo_parent_id'): line_by_uuid[line['uuid']].combo_parent_id = line_by_uuid[uuid_by_cid[line['combo_parent_id']]] def _process_saved_order(self, draft): self.ensure_one() if not draft: try: self.action_pos_order_paid() except psycopg2.DatabaseError: # do not hide transactional errors, the order(s) won't be saved! raise except Exception as e: _logger.error('Could not fully process the POS Order: %s', tools.ustr(e)) self._create_order_picking() self._compute_total_cost_in_real_time() if self.to_invoice and self.state == 'paid': self._generate_pos_order_invoice() return self.id def _clean_payment_lines(self): self.ensure_one() self.payment_ids.unlink() def _process_payment_lines(self, pos_order, order, pos_session, draft): """Create account.bank.statement.lines from the dictionary given to the parent function. If the payment_line is an updated version of an existing one, the existing payment_line will first be removed before making a new one. :param pos_order: dictionary representing the order. :type pos_order: dict. :param order: Order object the payment lines should belong to. :type order: pos.order :param pos_session: PoS session the order was created in. :type pos_session: pos.session :param draft: Indicate that the pos_order is not validated yet. :type draft: bool. """ prec_acc = order.currency_id.decimal_places order._clean_payment_lines() for payments in pos_order['statement_ids']: order.add_payment(self._payment_fields(order, payments[2])) order.amount_paid = sum(order.payment_ids.mapped('amount')) if not draft and not float_is_zero(pos_order['amount_return'], prec_acc): cash_payment_method = pos_session.payment_method_ids.filtered('is_cash_count')[:1] if not cash_payment_method: raise UserError(_("No cash statement found for this session. Unable to record returned cash.")) return_payment_vals = { 'name': _('return'), 'pos_order_id': order.id, 'amount': -pos_order['amount_return'], 'payment_date': fields.Datetime.now(), 'payment_method_id': cash_payment_method.id, 'is_change': True, } order.add_payment(return_payment_vals) def _prepare_tax_base_line_values(self, sign=1): """ Convert pos order lines into dictionaries that would be used to compute taxes later. :param sign: An optional parameter to force the sign of amounts. :return: A list of python dictionaries (see '_convert_to_tax_base_line_dict' in account.tax). """ self.ensure_one() return self.lines._prepare_tax_base_line_values(sign=sign) @api.model def _get_invoice_lines_values(self, line_values, pos_order_line): return { 'product_id': line_values['product'].id, 'quantity': line_values['quantity'], 'discount': line_values['discount'], 'price_unit': line_values['price_unit'], 'name': line_values['name'], 'tax_ids': [(6, 0, line_values['taxes'].ids)], 'product_uom_id': line_values['uom'].id, } def _prepare_invoice_lines(self): """ Prepare a list of orm commands containing the dictionaries to fill the 'invoice_line_ids' field when creating an invoice. :return: A list of Command.create to fill 'invoice_line_ids' when calling account.move.create. """ sign = 1 if self.amount_total >= 0 else -1 line_values_list = self._prepare_tax_base_line_values(sign=sign) invoice_lines = [] for line_values in line_values_list: line = line_values['record'] invoice_lines_values = self._get_invoice_lines_values(line_values, line) invoice_lines.append((0, None, invoice_lines_values)) if line.order_id.pricelist_id.discount_policy == 'without_discount' and float_compare(line.price_unit, line.product_id.lst_price, precision_rounding=self.currency_id.rounding) < 0: invoice_lines.append((0, None, { 'name': _('Price discount from %s -> %s', float_repr(line.product_id.lst_price, self.currency_id.decimal_places), float_repr(line.price_unit, self.currency_id.decimal_places)), 'display_type': 'line_note', })) if line.customer_note: invoice_lines.append((0, None, { 'name': line.customer_note, 'display_type': 'line_note', })) return invoice_lines def _get_pos_anglo_saxon_price_unit(self, product, partner_id, quantity): moves = self.filtered(lambda o: o.partner_id.id == partner_id)\ .mapped('picking_ids.move_ids')\ ._filter_anglo_saxon_moves(product)\ .sorted(lambda x: x.date) price_unit = product.with_company(self.company_id)._compute_average_price(0, quantity, moves) return price_unit name = fields.Char(string='Order Ref', required=True, readonly=True, copy=False, default='/') last_order_preparation_change = fields.Char(string='Last preparation change', help="Last printed state of the order") date_order = fields.Datetime(string='Date', readonly=True, index=True, default=fields.Datetime.now) user_id = fields.Many2one( comodel_name='res.users', string='Responsible', help="Person who uses the cash register. It can be a reliever, a student or an interim employee.", default=lambda self: self.env.uid, ) amount_tax = fields.Float(string='Taxes', digits=0, readonly=True, required=True) amount_total = fields.Float(string='Total', digits=0, readonly=True, required=True) amount_paid = fields.Float(string='Paid', digits=0, required=True) amount_return = fields.Float(string='Returned', digits=0, required=True, readonly=True) margin = fields.Monetary(string="Margin", compute='_compute_margin') margin_percent = fields.Float(string="Margin (%)", compute='_compute_margin', digits=(12, 4)) is_total_cost_computed = fields.Boolean(compute='_compute_is_total_cost_computed', help="Allows to know if all the total cost of the order lines have already been computed") lines = fields.One2many('pos.order.line', 'order_id', string='Order Lines', copy=True) company_id = fields.Many2one('res.company', string='Company', required=True, readonly=True, index=True) country_code = fields.Char(related='company_id.account_fiscal_country_id.code') pricelist_id = fields.Many2one('product.pricelist', string='Pricelist') partner_id = fields.Many2one('res.partner', string='Customer', change_default=True, index='btree_not_null') sequence_number = fields.Integer(string='Sequence Number', help='A session-unique sequence number for the order', default=1) session_id = fields.Many2one( 'pos.session', string='Session', required=True, index=True, domain="[('state', '=', 'opened')]") config_id = fields.Many2one('pos.config', related='session_id.config_id', string="Point of Sale", readonly=False, store=True) currency_id = fields.Many2one('res.currency', related='config_id.currency_id', string="Currency") currency_rate = fields.Float("Currency Rate", compute='_compute_currency_rate', compute_sudo=True, store=True, digits=0, readonly=True, help='The rate of the currency to the currency of rate applicable at the date of the order') state = fields.Selection( [('draft', 'New'), ('cancel', 'Cancelled'), ('paid', 'Paid'), ('done', 'Posted'), ('invoiced', 'Invoiced')], 'Status', readonly=True, copy=False, default='draft', index=True) account_move = fields.Many2one('account.move', string='Invoice', readonly=True, copy=False, index="btree_not_null") picking_ids = fields.One2many('stock.picking', 'pos_order_id') picking_count = fields.Integer(compute='_compute_picking_count') failed_pickings = fields.Boolean(compute='_compute_picking_count') picking_type_id = fields.Many2one('stock.picking.type', related='session_id.config_id.picking_type_id', string="Operation Type", readonly=False) procurement_group_id = fields.Many2one('procurement.group', 'Procurement Group', copy=False) note = fields.Text(string='Internal Notes') nb_print = fields.Integer(string='Number of Print', readonly=True, copy=False, default=0) pos_reference = fields.Char(string='Receipt Number', readonly=True, copy=False, index=True) sale_journal = fields.Many2one('account.journal', related='session_id.config_id.journal_id', string='Sales Journal', store=True, readonly=True, ondelete='restrict') fiscal_position_id = fields.Many2one( comodel_name='account.fiscal.position', string='Fiscal Position', readonly=False, ) payment_ids = fields.One2many('pos.payment', 'pos_order_id', string='Payments', readonly=True) session_move_id = fields.Many2one('account.move', string='Session Journal Entry', related='session_id.move_id', readonly=True, copy=False) to_invoice = fields.Boolean('To invoice', copy=False) shipping_date = fields.Date('Shipping Date') is_invoiced = fields.Boolean('Is Invoiced', compute='_compute_is_invoiced') is_tipped = fields.Boolean('Is this already tipped?', readonly=True) tip_amount = fields.Float(string='Tip Amount', digits=0, readonly=True) refund_orders_count = fields.Integer('Number of Refund Orders', compute='_compute_refund_related_fields') is_refunded = fields.Boolean(compute='_compute_refund_related_fields') refunded_order_ids = fields.Many2many('pos.order', compute='_compute_refund_related_fields') has_refundable_lines = fields.Boolean('Has Refundable Lines', compute='_compute_has_refundable_lines') refunded_orders_count = fields.Integer(compute='_compute_refund_related_fields') ticket_code = fields.Char(help='5 digits alphanumeric code to be used by portal user to request an invoice') tracking_number = fields.Char(string="Order Number", compute='_compute_tracking_number', search='_search_tracking_number') def _search_tracking_number(self, operator, value): #search is made over the pos_reference field #The pos_reference field is like 'Order 00001-001-0001' if operator in ['ilike', '='] and isinstance(value, str): if value[0] == '%' and value[-1] == '%': value = value[1:-1] value = value.zfill(3) search = '% ____' + value[0] + '-___-__' + value[1:] return [('pos_reference', operator, search or '')] else: raise NotImplementedError(_("Unsupported search operation")) @api.depends('lines.refund_orderline_ids', 'lines.refunded_orderline_id') def _compute_refund_related_fields(self): for order in self: order.refund_orders_count = len(order.mapped('lines.refund_orderline_ids.order_id')) order.is_refunded = order.refund_orders_count > 0 order.refunded_order_ids = order.mapped('lines.refunded_orderline_id.order_id') order.refunded_orders_count = len(order.refunded_order_ids) @api.depends('lines.refunded_qty', 'lines.qty') def _compute_has_refundable_lines(self): digits = self.env['decimal.precision'].precision_get('Product Unit of Measure') for order in self: order.has_refundable_lines = any([float_compare(line.qty, line.refunded_qty, digits) > 0 for line in order.lines]) @api.depends('account_move') def _compute_is_invoiced(self): for order in self: order.is_invoiced = bool(order.account_move) @api.depends('picking_ids', 'picking_ids.state') def _compute_picking_count(self): for order in self: order.picking_count = len(order.picking_ids) order.failed_pickings = bool(order.picking_ids.filtered(lambda p: p.state != 'done')) @api.depends('date_order', 'company_id', 'currency_id', 'company_id.currency_id') def _compute_currency_rate(self): for order in self: order.currency_rate = self.env['res.currency']._get_conversion_rate(order.company_id.currency_id, order.currency_id, order.company_id, order.date_order.date()) @api.depends('lines.is_total_cost_computed') def _compute_is_total_cost_computed(self): for order in self: order.is_total_cost_computed = not False in order.lines.mapped('is_total_cost_computed') def _compute_total_cost_in_real_time(self): """ Compute the total cost of the order when it's processed by the server. It will compute the total cost of all the lines if it's possible. If a margin of one of the order's lines cannot be computed (because of session_id.update_stock_at_closing), then the margin of said order is not computed (it will be computed when closing the session). """ for order in self: lines = order.lines if not order._should_create_picking_real_time(): storable_fifo_avco_lines = lines.filtered(lambda l: l._is_product_storable_fifo_avco()) lines -= storable_fifo_avco_lines stock_moves = order.picking_ids.move_ids lines._compute_total_cost(stock_moves) def _compute_total_cost_at_session_closing(self, stock_moves): """ Compute the margin at the end of the session. This method should be called to compute the remaining lines margin containing a storable product with a fifo/avco cost method and then compute the order margin """ for order in self: storable_fifo_avco_lines = order.lines.filtered(lambda l: l._is_product_storable_fifo_avco()) storable_fifo_avco_lines._compute_total_cost(stock_moves) @api.depends('lines.margin', 'is_total_cost_computed') def _compute_margin(self): for order in self: if order.is_total_cost_computed: order.margin = sum(order.lines.mapped('margin')) amount_untaxed = order.currency_id.round(sum(line.price_subtotal for line in order.lines)) order.margin_percent = not float_is_zero(amount_untaxed, precision_rounding=order.currency_id.rounding) and order.margin / amount_untaxed or 0 else: order.margin = 0 order.margin_percent = 0 @api.onchange('payment_ids', 'lines') def _onchange_amount_all(self): for order in self: if not order.currency_id: raise UserError(_("You can't: create a pos order from the backend interface, or unset the pricelist, or create a pos.order in a python test with Form tool, or edit the form view in studio if no PoS order exist")) currency = order.currency_id order.amount_paid = sum(payment.amount for payment in order.payment_ids) order.amount_return = sum(payment.amount < 0 and payment.amount or 0 for payment in order.payment_ids) order.amount_tax = currency.round(sum(self._amount_line_tax(line, order.fiscal_position_id) for line in order.lines)) amount_untaxed = currency.round(sum(line.price_subtotal for line in order.lines)) order.amount_total = order.amount_tax + amount_untaxed def _compute_batch_amount_all(self): """ Does essentially the same thing as `_onchange_amount_all` but only for actually existing records It is intended as a helper method , not as a business one Practical to be used for migrations """ amounts = {order_id: {'paid': 0, 'return': 0, 'taxed': 0, 'taxes': 0} for order_id in self.ids} for pos_order, amount in self.env['pos.payment']._read_group([('pos_order_id', 'in', self.ids)], ['pos_order_id'], ['amount:sum']): amounts[pos_order.id]['paid'] = amount for pos_order, amount in self.env['pos.payment']._read_group(['&', ('pos_order_id', 'in', self.ids), ('amount', '<', 0)], ['pos_order_id'], ['amount:sum']): amounts[pos_order.id]['return'] = amount for order, price_subtotal, price_subtotal_incl in self.env['pos.order.line']._read_group([('order_id', 'in', self.ids)], ['order_id'], ['price_subtotal:sum', 'price_subtotal_incl:sum']): amounts[order.id]['taxed'] = price_subtotal_incl amounts[order.id]['taxes'] = price_subtotal_incl - price_subtotal for order in self: order.write({ 'amount_paid': amounts[order.id]['paid'], 'amount_return': amounts[order.id]['return'], 'amount_tax': order.currency_id.round(amounts[order.id]['taxes']), 'amount_total': order.currency_id.round(amounts[order.id]['taxed']) }) @api.onchange('partner_id') def _onchange_partner_id(self): if self.partner_id: self.pricelist_id = self.partner_id.property_product_pricelist.id @api.ondelete(at_uninstall=False) def _unlink_except_draft_or_cancel(self): for pos_order in self.filtered(lambda pos_order: pos_order.state not in ['draft', 'cancel']): raise UserError(_('In order to delete a sale, it must be new or cancelled.')) @api.model_create_multi def create(self, vals_list): for vals in vals_list: session = self.env['pos.session'].browse(vals['session_id']) vals = self._complete_values_from_session(session, vals) return super().create(vals_list) @api.model def _complete_values_from_session(self, session, values): if values.get('state') and values['state'] == 'paid' and not values.get('name'): values['name'] = self._compute_order_name() values.setdefault('pricelist_id', session.config_id.pricelist_id.id) values.setdefault('fiscal_position_id', session.config_id.default_fiscal_position_id.id) values.setdefault('company_id', session.config_id.company_id.id) return values def write(self, vals): for order in self: if vals.get('state') and vals['state'] == 'paid' and order.name == '/': vals['name'] = self._compute_order_name() return super(PosOrder, self).write(vals) def _compute_order_name(self): if len(self.refunded_order_ids) != 0: return ','.join(self.refunded_order_ids.mapped('name')) + _(' REFUND') else: return self.session_id.config_id.sequence_id._next() def action_stock_picking(self): self.ensure_one() action = self.env['ir.actions.act_window']._for_xml_id('stock.action_picking_tree_ready') action['display_name'] = _('Pickings') action['context'] = {} action['domain'] = [('id', 'in', self.picking_ids.ids)] return action def action_view_invoice(self): return { 'name': _('Customer Invoice'), 'view_mode': 'form', 'view_id': self.env.ref('account.view_move_form').id, 'res_model': 'account.move', 'context': "{'move_type':'out_invoice'}", 'type': 'ir.actions.act_window', 'res_id': self.account_move.id, } def action_view_refund_orders(self): return { 'name': _('Refund Orders'), 'view_mode': 'tree,form', 'res_model': 'pos.order', 'type': 'ir.actions.act_window', 'domain': [('id', 'in', self.mapped('lines.refund_orderline_ids.order_id').ids)], } def action_view_refunded_orders(self): return { 'name': _('Refunded Orders'), 'view_mode': 'tree,form', 'res_model': 'pos.order', 'type': 'ir.actions.act_window', 'domain': [('id', 'in', self.refunded_order_ids.ids)], } def _is_pos_order_paid(self): return float_is_zero(self._get_rounded_amount(self.amount_total) - self.amount_paid, precision_rounding=self.currency_id.rounding) def _get_rounded_amount(self, amount, force_round=False): # TODO: add support for mix of cash and non-cash payments when both cash_rounding and only_round_cash_method are True if self.config_id.cash_rounding \ and (force_round or (not self.config_id.only_round_cash_method \ or any(p.payment_method_id.is_cash_count for p in self.payment_ids))): amount = float_round(amount, precision_rounding=self.config_id.rounding_method.rounding, rounding_method=self.config_id.rounding_method.rounding_method) currency = self.currency_id return currency.round(amount) if currency else amount def _get_partner_bank_id(self): bank_partner_id = False has_pay_later = any(not pm.journal_id for pm in self.payment_ids.mapped('payment_method_id')) if has_pay_later: if self.amount_total <= 0 and self.partner_id.bank_ids: bank_partner_id = self.partner_id.bank_ids[0].id elif self.amount_total >= 0 and self.company_id.partner_id.bank_ids: bank_partner_id = self.company_id.partner_id.bank_ids[0].id return bank_partner_id def _create_invoice(self, move_vals): self.ensure_one() new_move = self.env['account.move'].sudo().with_company(self.company_id).with_context(default_move_type=move_vals['move_type']).create(move_vals) message = _("This invoice has been created from the point of sale session: %s", self._get_html_link()) new_move.message_post(body=message) if self.config_id.cash_rounding: with self.env['account.move']._check_balanced({'records': new_move}): rounding_applied = float_round(self.amount_paid - self.amount_total, precision_rounding=new_move.currency_id.rounding) rounding_line = new_move.line_ids.filtered(lambda line: line.display_type == 'rounding') if rounding_line and rounding_line.debit > 0: rounding_line_difference = rounding_line.debit + rounding_applied elif rounding_line and rounding_line.credit > 0: rounding_line_difference = -rounding_line.credit + rounding_applied else: rounding_line_difference = rounding_applied if rounding_applied: if rounding_applied > 0.0: account_id = new_move.invoice_cash_rounding_id.loss_account_id.id else: account_id = new_move.invoice_cash_rounding_id.profit_account_id.id if rounding_line: if rounding_line_difference: rounding_line.with_context(skip_invoice_sync=True).write({ 'debit': rounding_applied < 0.0 and -rounding_applied or 0.0, 'credit': rounding_applied > 0.0 and rounding_applied or 0.0, 'account_id': account_id, 'price_unit': rounding_applied, }) else: self.env['account.move.line'].with_context(skip_invoice_sync=True).create({ 'balance': -rounding_applied, 'quantity': 1.0, 'partner_id': new_move.partner_id.id, 'move_id': new_move.id, 'currency_id': new_move.currency_id.id, 'company_id': new_move.company_id.id, 'company_currency_id': new_move.company_id.currency_id.id, 'display_type': 'rounding', 'sequence': 9999, 'name': self.config_id.rounding_method.name, 'account_id': account_id, }) else: if rounding_line: rounding_line.with_context(skip_invoice_sync=True).unlink() if rounding_line_difference: existing_terms_line = new_move.line_ids.filtered( lambda line: line.account_id.account_type in ('asset_receivable', 'liability_payable')) existing_terms_line_new_val = float_round( existing_terms_line.balance + rounding_line_difference, precision_rounding=new_move.currency_id.rounding) existing_terms_line.with_context(skip_invoice_sync=True).balance = existing_terms_line_new_val return new_move def action_pos_order_paid(self): self.ensure_one() # TODO: add support for mix of cash and non-cash payments when both cash_rounding and only_round_cash_method are True if not self.config_id.cash_rounding \ or self.config_id.only_round_cash_method \ and not any(p.payment_method_id.is_cash_count for p in self.payment_ids): total = self.amount_total else: total = float_round(self.amount_total, precision_rounding=self.config_id.rounding_method.rounding, rounding_method=self.config_id.rounding_method.rounding_method) isPaid = float_is_zero(total - self.amount_paid, precision_rounding=self.currency_id.rounding) if not isPaid and not self.config_id.cash_rounding: raise UserError(_("Order %s is not fully paid.", self.name)) elif not isPaid and self.config_id.cash_rounding: currency = self.currency_id if self.config_id.rounding_method.rounding_method == "HALF-UP": maxDiff = currency.round(self.config_id.rounding_method.rounding / 2) else: maxDiff = currency.round(self.config_id.rounding_method.rounding) diff = currency.round(self.amount_total - self.amount_paid) if not abs(diff) <= maxDiff: raise UserError(_("Order %s is not fully paid.", self.name)) self.write({'state': 'paid'}) return True def _prepare_invoice_vals(self): self.ensure_one() timezone = pytz.timezone(self._context.get('tz') or self.env.user.tz or 'UTC') invoice_date = fields.Datetime.now() if self.session_id.state == 'closed' else self.date_order pos_refunded_invoice_ids = [] for orderline in self.lines: if orderline.refunded_orderline_id and orderline.refunded_orderline_id.order_id.account_move: pos_refunded_invoice_ids.append(orderline.refunded_orderline_id.order_id.account_move.id) vals = { 'invoice_origin': self.name, 'pos_refunded_invoice_ids': pos_refunded_invoice_ids, 'journal_id': self.session_id.config_id.invoice_journal_id.id, 'move_type': 'out_invoice' if self.amount_total >= 0 else 'out_refund', 'ref': self.name, 'partner_id': self.partner_id.id, 'partner_bank_id': self._get_partner_bank_id(), 'currency_id': self.currency_id.id, 'invoice_user_id': self.user_id.id, 'invoice_date': invoice_date.astimezone(timezone).date(), 'fiscal_position_id': self.fiscal_position_id.id, 'invoice_line_ids': self._prepare_invoice_lines(), 'invoice_payment_term_id': self.partner_id.property_payment_term_id.id or False, 'invoice_cash_rounding_id': self.config_id.rounding_method.id if self.config_id.cash_rounding and (not self.config_id.only_round_cash_method or any(p.payment_method_id.is_cash_count for p in self.payment_ids)) else False } if self.refunded_order_ids.account_move: vals['ref'] = _('Reversal of: %s', self.refunded_order_ids.account_move.name) vals['reversed_entry_id'] = self.refunded_order_ids.account_move.id if self.note: vals.update({'narration': self.note}) return vals def _prepare_aml_values_list_per_nature(self): self.ensure_one() sign = 1 if self.amount_total < 0 else -1 commercial_partner = self.partner_id.commercial_partner_id company_currency = self.company_id.currency_id rate = self.currency_id._get_conversion_rate(self.currency_id, company_currency, self.company_id, self.date_order) # Concert each order line to a dictionary containing business values. Also, prepare for taxes computation. base_line_vals_list = self._prepare_tax_base_line_values(sign=-1) tax_results = self.env['account.tax']._compute_taxes(base_line_vals_list) total_balance = 0.0 total_amount_currency = 0.0 aml_vals_list_per_nature = defaultdict(list) # Create the tax lines for tax_line_vals in tax_results['tax_lines_to_add']: tax_rep = self.env['account.tax.repartition.line'].browse(tax_line_vals['tax_repartition_line_id']) amount_currency = tax_line_vals['tax_amount'] balance = company_currency.round(amount_currency * rate) aml_vals_list_per_nature['tax'].append({ 'name': tax_rep.tax_id.name, 'account_id': tax_line_vals['account_id'], 'partner_id': tax_line_vals['partner_id'], 'currency_id': tax_line_vals['currency_id'], 'tax_repartition_line_id': tax_line_vals['tax_repartition_line_id'], 'tax_ids': tax_line_vals['tax_ids'], 'tax_tag_ids': tax_line_vals['tax_tag_ids'], 'group_tax_id': None if tax_rep.tax_id.id == tax_line_vals['tax_id'] else tax_line_vals['tax_id'], 'amount_currency': amount_currency, 'balance': balance, }) total_amount_currency += amount_currency total_balance += balance # Create the aml values for order lines. for base_line_vals, update_base_line_vals in tax_results['base_lines_to_update']: order_line = base_line_vals['record'] amount_currency = update_base_line_vals['price_subtotal'] balance = company_currency.round(amount_currency * rate) aml_vals_list_per_nature['product'].append({ 'name': order_line.full_product_name, 'account_id': base_line_vals['account'].id, 'partner_id': base_line_vals['partner'].id, 'currency_id': base_line_vals['currency'].id, 'tax_ids': [(6, 0, base_line_vals['taxes'].ids)], 'tax_tag_ids': update_base_line_vals['tax_tag_ids'], 'amount_currency': amount_currency, 'balance': balance, }) total_amount_currency += amount_currency total_balance += balance # Cash rounding. cash_rounding = self.config_id.rounding_method if self.config_id.cash_rounding and cash_rounding and not self.config_id.only_round_cash_method: amount_currency = cash_rounding.compute_difference(self.currency_id, total_amount_currency) if not self.currency_id.is_zero(amount_currency): balance = company_currency.round(amount_currency * rate) if cash_rounding.strategy == 'biggest_tax': biggest_tax_aml_vals = None for aml_vals in aml_vals_list_per_nature['tax']: if not biggest_tax_aml_vals or float_compare(-sign * aml_vals['amount_currency'], -sign * biggest_tax_aml_vals['amount_currency'], precision_rounding=self.currency_id.rounding) > 0: biggest_tax_aml_vals = aml_vals if biggest_tax_aml_vals: biggest_tax_aml_vals['amount_currency'] += amount_currency biggest_tax_aml_vals['balance'] += balance elif cash_rounding.strategy == 'add_invoice_line': if -sign * amount_currency > 0.0 and cash_rounding.loss_account_id: account_id = cash_rounding.loss_account_id.id else: account_id = cash_rounding.profit_account_id.id aml_vals_list_per_nature['cash_rounding'].append({ 'name': cash_rounding.name, 'account_id': account_id, 'partner_id': commercial_partner.id, 'currency_id': self.currency_id.id, 'amount_currency': amount_currency, 'balance': balance, 'display_type': 'rounding', }) # Stock. if self.company_id.anglo_saxon_accounting and self.picking_ids.ids: stock_moves = self.env['stock.move'].sudo().search([ ('picking_id', 'in', self.picking_ids.ids), ('product_id.categ_id.property_valuation', '=', 'real_time') ]) for stock_move in stock_moves: expense_account = stock_move.product_id._get_product_accounts()['expense'] stock_output_account = stock_move.product_id.categ_id.property_stock_account_output_categ_id balance = -sum(stock_move.stock_valuation_layer_ids.mapped('value')) aml_vals_list_per_nature['stock'].append({ 'name': _("Stock input for %s", stock_move.product_id.name), 'account_id': expense_account.id, 'partner_id': commercial_partner.id, 'currency_id': self.company_id.currency_id.id, 'amount_currency': balance, 'balance': balance, }) aml_vals_list_per_nature['stock'].append({ 'name': _("Stock output for %s", stock_move.product_id.name), 'account_id': stock_output_account.id, 'partner_id': commercial_partner.id, 'currency_id': self.company_id.currency_id.id, 'amount_currency': -balance, 'balance': -balance, }) # sort self.payment_ids by is_split_transaction: for payment_id in self.payment_ids: is_split_transaction = payment_id.payment_method_id.split_transactions if is_split_transaction: reversed_move_receivable_account_id = self.partner_id.property_account_receivable_id else: reversed_move_receivable_account_id = payment_id.payment_method_id.receivable_account_id or self.company_id.account_default_pos_receivable_account_id aml_vals_entry_found = [aml_entry for aml_entry in aml_vals_list_per_nature['payment_terms'] if aml_entry['account_id'] == reversed_move_receivable_account_id.id and not aml_entry['partner_id']] if aml_vals_entry_found and not is_split_transaction: aml_vals_entry_found[0]['amount_currency'] += self.session_id._amount_converter(payment_id.amount, self.date_order, False) aml_vals_entry_found[0]['balance'] += payment_id.amount else: aml_vals_list_per_nature['payment_terms'].append({ 'partner_id': commercial_partner.id if is_split_transaction else False, 'name': f"{reversed_move_receivable_account_id.code} {reversed_move_receivable_account_id.code}", 'account_id': reversed_move_receivable_account_id.id, 'currency_id': self.currency_id.id, 'amount_currency': payment_id.amount, 'balance': self.session_id._amount_converter(payment_id.amount, self.date_order, False), }) return aml_vals_list_per_nature def _create_misc_reversal_move(self, payment_moves): """ Create a misc move to reverse this POS order and "remove" it from the POS closing entry. This is done by taking data from the order and using it to somewhat replicate the resulting entry in order to reverse partially the movements done ine the POS closing entry. """ aml_values_list_per_nature = self._prepare_aml_values_list_per_nature() move_lines = [] for aml_values_list in aml_values_list_per_nature.values(): for aml_values in aml_values_list: aml_values['balance'] = -aml_values['balance'] aml_values['amount_currency'] = -aml_values['amount_currency'] move_lines.append(aml_values) # Make a move with all the lines. reversal_entry = self.env['account.move'].with_context( default_journal_id=self.config_id.journal_id.id, skip_invoice_sync=True, skip_invoice_line_sync=True, ).create({ 'journal_id': self.config_id.journal_id.id, 'date': fields.Date.context_today(self), 'ref': _('Reversal of POS closing entry %s for order %s from session %s', self.session_move_id.name, self.name, self.session_id.name), 'invoice_line_ids': [(0, 0, aml_value) for aml_value in move_lines], }) reversal_entry.action_post() # Reconcile the new receivable line with the lines from the payment move. pos_account_receivable = self.company_id.account_default_pos_receivable_account_id reversal_entry_receivable = reversal_entry.line_ids.filtered(lambda l: l.account_id == pos_account_receivable) payment_receivable = payment_moves.line_ids.filtered(lambda l: l.account_id == pos_account_receivable) (reversal_entry_receivable | payment_receivable).reconcile() def action_pos_order_invoice(self): if len(self.company_id) > 1: raise UserError(_("You cannot invoice orders belonging to different companies.")) self.write({'to_invoice': True}) res = self._generate_pos_order_invoice() if self.company_id.anglo_saxon_accounting and self.session_id.update_stock_at_closing and self.session_id.state != 'closed': self._create_order_picking() return res def _generate_pos_order_invoice(self): moves = self.env['account.move'] for order in self: # Force company for all SUPERUSER_ID action if order.account_move: moves += order.account_move continue if not order.partner_id: raise UserError(_('Please provide a partner for the sale.')) move_vals = order._prepare_invoice_vals() new_move = order._create_invoice(move_vals) order.write({'account_move': new_move.id, 'state': 'invoiced'}) new_move.sudo().with_company(order.company_id).with_context(skip_invoice_sync=True)._post() # Send and Print if self.env.context.get('generate_pdf', True): template = self.env.ref(new_move._get_mail_template()) new_move.with_context(skip_invoice_sync=True)._generate_pdf_and_send_invoice(template) moves += new_move payment_moves = order._apply_invoice_payments() if order.session_id.state == 'closed': # If the session isn't closed this isn't needed. # If a client requires the invoice later, we need to revers the amount from the closing entry, by making a new entry for that. order._create_misc_reversal_move(payment_moves) if not moves: return {} return { 'name': _('Customer Invoice'), 'view_mode': 'form', 'view_id': self.env.ref('account.view_move_form').id, 'res_model': 'account.move', 'context': "{'move_type':'out_invoice'}", 'type': 'ir.actions.act_window', 'nodestroy': True, 'target': 'current', 'res_id': moves and moves.ids[0] or False, } # this method is unused, and so is the state 'cancel' def action_pos_order_cancel(self): return self.write({'state': 'cancel'}) def _apply_invoice_payments(self): receivable_account = self.env["res.partner"]._find_accounting_partner(self.partner_id).with_company(self.company_id).property_account_receivable_id payment_moves = self.payment_ids.sudo().with_company(self.company_id)._create_payment_moves() if receivable_account.reconcile: invoice_receivables = self.account_move.line_ids.filtered(lambda line: line.account_id == receivable_account and not line.reconciled) if invoice_receivables: payment_receivables = payment_moves.mapped('line_ids').filtered(lambda line: line.account_id == receivable_account and line.partner_id) (invoice_receivables | payment_receivables).sudo().with_company(self.company_id).reconcile() return payment_moves @api.model def create_from_ui(self, orders, draft=False): """ Create and update Orders from the frontend PoS application. Create new orders and update orders that are in draft status. If an order already exists with a status different from 'draft'it will be discarded, otherwise it will be saved to the database. If saved with 'draft' status the order can be overwritten later by this function. :param orders: dictionary with the orders to be created. :type orders: dict. :param draft: Indicate if the orders are meant to be finalized or temporarily saved. :type draft: bool. :Returns: list -- list of db-ids for the created and updated orders. """ order_ids = [] for order in orders: existing_draft_order = None if 'server_id' in order['data'] and order['data']['server_id']: # if the server id exists, it must only search based on the id existing_draft_order = self.env['pos.order'].search(['&', ('id', '=', order['data']['server_id']), ('state', '=', 'draft')], limit=1) # if there is no draft order, skip processing this order if not existing_draft_order: continue if not existing_draft_order: existing_draft_order = self.env['pos.order'].search(['&', ('pos_reference', '=', order['data']['name']), ('state', '=', 'draft')], limit=1) if existing_draft_order: order_ids.append(self._process_order(order, draft, existing_draft_order)) else: existing_orders = self.env['pos.order'].search([('pos_reference', '=', order['data']['name'])]) if all(not self._is_the_same_order(order['data'], existing_order) for existing_order in existing_orders): order_ids.append(self._process_order(order, draft, False)) return self.env['pos.order'].search_read(domain=[('id', 'in', order_ids)], fields=['id', 'pos_reference', 'account_move'], load=False) def _is_the_same_order(self, data, existing_order): received_payments = [(p[2]['amount'], p[2]['payment_method_id']) for p in data['statement_ids']] existing_payments = [(p.amount, p.payment_method_id.id) for p in existing_order.payment_ids] for amount, payment_method in received_payments: if not any( float_is_zero(amount - ex_amount, precision_rounding=existing_order.currency_id.rounding) and payment_method == ex_payment_method for ex_amount, ex_payment_method in existing_payments ): return False if len(data['lines']) != len(existing_order.lines): return False received_lines = sorted([(l[2]['product_id'], l[2]['qty'], l[2]['price_unit']) for l in data['lines']]) existing_lines = sorted([(l.product_id.id, l.qty, l.price_unit) for l in existing_order.lines]) if received_lines != existing_lines: return False return True def _should_create_picking_real_time(self): return not self.session_id.update_stock_at_closing or (self.company_id.anglo_saxon_accounting and self.to_invoice) def _create_order_picking(self): self.ensure_one() if self.shipping_date: self.lines._launch_stock_rule_from_pos_order_lines() else: if self._should_create_picking_real_time(): picking_type = self.config_id.picking_type_id if self.partner_id.property_stock_customer: destination_id = self.partner_id.property_stock_customer.id elif not picking_type or not picking_type.default_location_dest_id: destination_id = self.env['stock.warehouse']._get_partner_locations()[0].id else: destination_id = picking_type.default_location_dest_id.id pickings = self.env['stock.picking']._create_picking_from_pos_order_lines(destination_id, self.lines, picking_type, self.partner_id) pickings.write({'pos_session_id': self.session_id.id, 'pos_order_id': self.id, 'origin': self.name}) def add_payment(self, data): """Create a new payment for the order""" self.ensure_one() self.env['pos.payment'].create(data) self.amount_paid = sum(self.payment_ids.mapped('amount')) def _prepare_refund_values(self, current_session): self.ensure_one() return { 'name': self.name + _(' REFUND'), 'session_id': current_session.id, 'date_order': fields.Datetime.now(), 'pos_reference': self.pos_reference, 'lines': False, 'amount_tax': -self.amount_tax, 'amount_total': -self.amount_total, 'amount_paid': 0, 'is_total_cost_computed': False } def _prepare_mail_values(self, name, client, ticket): message = Markup( _("

Dear %(client_name)s,
Here is your electronic ticket for the %(pos_name)s.

") ) % { 'client_name': client['name'], 'pos_name': name, } return { 'subject': _('Receipt %s', name), 'body_html': message, 'author_id': self.env.user.partner_id.id, 'email_from': self.env.company.email or self.env.user.email_formatted, 'email_to': client['email'], 'attachment_ids': self._add_mail_attachment(name, ticket), } def _refund(self): """ Create a copy of order to refund them. return The newly created refund orders. """ refund_orders = self.env['pos.order'] for order in self: # When a refund is performed, we are creating it in a session having the same config as the original # order. It can be the same session, or if it has been closed the new one that has been opened. current_session = order.session_id.config_id.current_session_id if not current_session: raise UserError(_('To return product(s), you need to open a session in the POS %s', order.session_id.config_id.display_name)) refund_order = order.copy( order._prepare_refund_values(current_session) ) for line in order.lines: PosOrderLineLot = self.env['pos.pack.operation.lot'] for pack_lot in line.pack_lot_ids: PosOrderLineLot += pack_lot.copy() line.copy(line._prepare_refund_data(refund_order, PosOrderLineLot)) refund_orders |= refund_order return refund_orders def refund(self): return { 'name': _('Return Products'), 'view_mode': 'form', 'res_model': 'pos.order', 'res_id': self._refund().ids[0], 'view_id': False, 'context': self.env.context, 'type': 'ir.actions.act_window', 'target': 'current', } def _add_mail_attachment(self, name, ticket): filename = 'Receipt-' + name + '.jpg' receipt = self.env['ir.attachment'].create({ 'name': filename, 'type': 'binary', 'datas': ticket, 'res_model': 'pos.order', 'res_id': self.ids[0], 'mimetype': 'image/jpeg', }) attachment = [(4, receipt.id)] if self.mapped('account_move'): report = self.env['ir.actions.report']._render_qweb_pdf("account.account_invoices", self.account_move.ids[0]) filename = name + '.pdf' invoice = self.env['ir.attachment'].create({ 'name': filename, 'type': 'binary', 'datas': base64.b64encode(report[0]), 'res_model': 'pos.order', 'res_id': self.ids[0], 'mimetype': 'application/x-pdf' }) attachment += [(4, invoice.id)] return attachment def action_receipt_to_customer(self, name, client, ticket): if not self: return False if not client.get('email'): return False mail = self.env['mail.mail'].sudo().create(self._prepare_mail_values(name, client, ticket)) mail.send() def is_already_paid(self): return self.state == "paid" @api.model def remove_from_ui(self, server_ids): """ Remove orders from the frontend PoS application Remove orders from the server by id. :param server_ids: list of the id's of orders to remove from the server. :type server_ids: list. :returns: list -- list of db-ids for the removed orders. """ orders = self.search([('id', 'in', server_ids), ('state', '=', 'draft')]) orders.write({'state': 'cancel'}) # TODO Looks like delete cascade is a better solution. orders.mapped('payment_ids').sudo().unlink() orders.sudo().unlink() return orders.ids @api.model def search_paid_order_ids(self, config_id, domain, limit, offset): """Search for 'paid' orders that satisfy the given domain, limit and offset.""" default_domain = [('state', '!=', 'draft'), ('state', '!=', 'cancel')] if domain == []: real_domain = AND([[['config_id', '=', config_id]], default_domain]) else: real_domain = AND([domain, default_domain]) orders = self.search(real_domain, limit=limit, offset=offset) # We clean here the orders that does not have the same currency. # As we cannot use currency_id in the domain (because it is not a stored field), # we must do it after the search. pos_config = self.env['pos.config'].browse(config_id) orders = orders.filtered(lambda order: order.currency_id == pos_config.currency_id) orderlines = self.env['pos.order.line'].search(['|', ('refunded_orderline_id.order_id', 'in', orders.ids), ('order_id', 'in', orders.ids)]) # We will return to the frontend the ids and the date of their last modification # so that it can compare to the last time it fetched the orders and can ask to fetch # orders that are not up-to-date. # The date of their last modification is either the last time one of its orderline has changed, # or the last time a refunded orderline related to it has changed. orders_info = defaultdict(lambda: datetime.min) for orderline in orderlines: key_order = orderline.order_id.id if orderline.order_id in orders \ else orderline.refunded_orderline_id.order_id.id if orders_info[key_order] < orderline.write_date: orders_info[key_order] = orderline.write_date totalCount = self.search_count(real_domain) return {'ordersInfo': list(orders_info.items())[::-1], 'totalCount': totalCount} def _export_for_ui(self, order): timezone = pytz.timezone(self._context.get('tz') or self.env.user.tz or 'UTC') return { 'lines': [[0, 0, line] for line in order.lines.export_for_ui()], 'statement_ids': [[0, 0, payment] for payment in order.payment_ids.export_for_ui()], 'name': order.pos_reference, 'uid': re.search('([0-9-]){14}', order.pos_reference).group(0), 'amount_paid': order.amount_paid, 'amount_total': order.amount_total, 'amount_tax': order.amount_tax, 'amount_return': order.amount_return, 'pos_session_id': order.session_id.id, 'pricelist_id': order.pricelist_id.id, 'partner_id': order.partner_id.id, 'user_id': order.user_id.id, 'sequence_number': order.sequence_number, 'date_order': str(order.date_order.astimezone(timezone)), 'fiscal_position_id': order.fiscal_position_id.id, 'to_invoice': order.to_invoice, 'shipping_date': order.shipping_date, 'state': order.state, 'account_move': order.account_move.id, 'id': order.id, 'is_tipped': order.is_tipped, 'tip_amount': order.tip_amount, 'access_token': order.access_token, 'ticket_code': order.ticket_code, 'last_order_preparation_change': order.last_order_preparation_change, } @api.model def export_for_ui_shared_order(self, config_id): config = self.env['pos.config'].browse(config_id) orders = self.env['pos.order'].search(['&', ('state', '=', 'draft'), '|', ('config_id', '=', config_id), ('config_id', 'in', config.trusted_config_ids.ids)]) return orders.export_for_ui() def export_for_ui(self): """ Returns a list of dict with each item having similar signature as the return of `export_as_JSON` of models.Order. This is useful for back-and-forth communication between the pos frontend and backend. """ return self.mapped(self._export_for_ui) if self else [] def _send_order(self): # This function is made to be overriden by pos_self_order_preparation_display pass class PosOrderLine(models.Model): _name = "pos.order.line" _description = "Point of Sale Order Lines" _rec_name = "product_id" def _order_line_fields(self, line, session_id=None): if line and 'name' not in line[2]: session = self.env['pos.session'].browse(session_id).exists() if session_id else None if session and session.config_id.sequence_line_id: # set name based on the sequence specified on the config line[2]['name'] = session.config_id.sequence_line_id._next() else: # fallback on any pos.order.line sequence line[2]['name'] = self.env['ir.sequence'].next_by_code('pos.order.line') if line and 'tax_ids' not in line[2]: product = self.env['product.product'].browse(line[2]['product_id']) line[2]['tax_ids'] = [(6, 0, [x.id for x in product.taxes_id])] # Clean up fields sent by the JS line = [ line[0], line[1], {k: v for k, v in line[2].items() if self._is_field_accepted(k)} ] return line company_id = fields.Many2one('res.company', string='Company', related="order_id.company_id", store=True) name = fields.Char(string='Line No', required=True, copy=False) skip_change = fields.Boolean('Skip line when sending ticket to kitchen printers.') notice = fields.Char(string='Discount Notice') product_id = fields.Many2one('product.product', string='Product', domain=[('sale_ok', '=', True)], required=True, change_default=True) attribute_value_ids = fields.Many2many('product.template.attribute.value', string="Selected Attributes") custom_attribute_value_ids = fields.One2many( comodel_name='product.attribute.custom.value', inverse_name='pos_order_line_id', string="Custom Values", store=True, readonly=False) price_unit = fields.Float(string='Unit Price', digits=0) qty = fields.Float('Quantity', digits='Product Unit of Measure', default=1) price_subtotal = fields.Float(string='Subtotal w/o Tax', digits=0, readonly=True, required=True) price_subtotal_incl = fields.Float(string='Subtotal', digits=0, readonly=True, required=True) price_extra = fields.Float(string="Price extra") margin = fields.Monetary(string="Margin", compute='_compute_margin') margin_percent = fields.Float(string="Margin (%)", compute='_compute_margin', digits=(12, 4)) total_cost = fields.Float(string='Total cost', digits='Product Price', readonly=True) is_total_cost_computed = fields.Boolean(help="Allows to know if the total cost has already been computed or not") discount = fields.Float(string='Discount (%)', digits=0, default=0.0) order_id = fields.Many2one('pos.order', string='Order Ref', ondelete='cascade', required=True, index=True) tax_ids = fields.Many2many('account.tax', string='Taxes', readonly=True) tax_ids_after_fiscal_position = fields.Many2many('account.tax', compute='_get_tax_ids_after_fiscal_position', string='Taxes to Apply') pack_lot_ids = fields.One2many('pos.pack.operation.lot', 'pos_order_line_id', string='Lot/serial Number') product_uom_id = fields.Many2one('uom.uom', string='Product UoM', related='product_id.uom_id') currency_id = fields.Many2one('res.currency', related='order_id.currency_id') full_product_name = fields.Char('Full Product Name') customer_note = fields.Char('Customer Note') refund_orderline_ids = fields.One2many('pos.order.line', 'refunded_orderline_id', 'Refund Order Lines', help='Orderlines in this field are the lines that refunded this orderline.') refunded_orderline_id = fields.Many2one('pos.order.line', 'Refunded Order Line', help='If this orderline is a refund, then the refunded orderline is specified in this field.') refunded_qty = fields.Float('Refunded Quantity', compute='_compute_refund_qty', help='Number of items refunded in this orderline.') uuid = fields.Char(string='Uuid', readonly=True, copy=False) combo_parent_id = fields.Many2one('pos.order.line', string='Combo Parent') combo_line_ids = fields.One2many('pos.order.line', 'combo_parent_id', string='Combo Lines') @api.model def _is_field_accepted(self, field): return field in self._fields and not field in ['combo_parent_id', 'combo_line_ids'] @api.depends('refund_orderline_ids') def _compute_refund_qty(self): for orderline in self: orderline.refunded_qty = -sum(orderline.mapped('refund_orderline_ids.qty')) def _prepare_refund_data(self, refund_order, PosOrderLineLot): """ This prepares data for refund order line. Inheritance may inject more data here @param refund_order: the pre-created refund order @type refund_order: pos.order @param PosOrderLineLot: the pre-created Pack operation Lot @type PosOrderLineLot: pos.pack.operation.lot @return: dictionary of data which is for creating a refund order line from the original line @rtype: dict """ self.ensure_one() return { 'name': self.name + _(' REFUND'), 'qty': -(self.qty - self.refunded_qty), 'order_id': refund_order.id, 'price_subtotal': -self.price_subtotal, 'price_subtotal_incl': -self.price_subtotal_incl, 'pack_lot_ids': PosOrderLineLot, 'is_total_cost_computed': False, 'refunded_orderline_id': self.id, } @api.model_create_multi def create(self, vals_list): for vals in vals_list: if vals.get('order_id') and not vals.get('name'): # set name based on the sequence specified on the config config = self.env['pos.order'].browse(vals['order_id']).session_id.config_id if config.sequence_line_id: vals['name'] = config.sequence_line_id._next() if not vals.get('name'): # fallback on any pos.order sequence vals['name'] = self.env['ir.sequence'].next_by_code('pos.order.line') return super().create(vals_list) def write(self, values): if values.get('pack_lot_line_ids'): for pl in values.get('pack_lot_ids'): if pl[2].get('server_id'): pl[2]['id'] = pl[2]['server_id'] del pl[2]['server_id'] return super().write(values) @api.ondelete(at_uninstall=False) def _unlink_except_order_state(self): if self.filtered(lambda x: x.order_id.state not in ["draft", "cancel"]): raise UserError(_("You can only unlink PoS order lines that are related to orders in new or cancelled state.")) @api.onchange('price_unit', 'tax_ids', 'qty', 'discount', 'product_id') def _onchange_amount_line_all(self): for line in self: res = line._compute_amount_line_all() line.update(res) def _compute_amount_line_all(self): self.ensure_one() fpos = self.order_id.fiscal_position_id tax_ids_after_fiscal_position = fpos.map_tax(self.tax_ids) price = self.price_unit * (1 - (self.discount or 0.0) / 100.0) taxes = tax_ids_after_fiscal_position.compute_all(price, self.order_id.currency_id, self.qty, product=self.product_id, partner=self.order_id.partner_id) return { 'price_subtotal_incl': taxes['total_included'], 'price_subtotal': taxes['total_excluded'], } @api.onchange('product_id') def _onchange_product_id(self): if self.product_id: price = self.order_id.pricelist_id._get_product_price( self.product_id, self.qty or 1.0, currency=self.currency_id ) self.tax_ids = self.product_id.taxes_id.filtered_domain(self.env['account.tax']._check_company_domain(self.company_id)) tax_ids_after_fiscal_position = self.order_id.fiscal_position_id.map_tax(self.tax_ids) self.price_unit = self.env['account.tax']._fix_tax_included_price_company(price, self.tax_ids, tax_ids_after_fiscal_position, self.company_id) self._onchange_qty() @api.onchange('qty', 'discount', 'price_unit', 'tax_ids') def _onchange_qty(self): if self.product_id: price = self.price_unit * (1 - (self.discount or 0.0) / 100.0) self.price_subtotal = self.price_subtotal_incl = price * self.qty if (self.tax_ids): taxes = self.tax_ids.compute_all(price, self.order_id.currency_id, self.qty, product=self.product_id, partner=False) self.price_subtotal = taxes['total_excluded'] self.price_subtotal_incl = taxes['total_included'] @api.depends('order_id', 'order_id.fiscal_position_id') def _get_tax_ids_after_fiscal_position(self): for line in self: line.tax_ids_after_fiscal_position = line.order_id.fiscal_position_id.map_tax(line.tax_ids) def _export_for_ui(self, orderline): return { 'id': orderline.id, 'qty': orderline.qty, 'attribute_value_ids': orderline.attribute_value_ids.ids, 'custom_attribute_value_ids': orderline.custom_attribute_value_ids.read(['id', 'name', 'custom_product_template_attribute_value_id', 'custom_value'], load=False), 'price_unit': orderline.price_unit, 'skip_change': orderline.skip_change, 'uuid': orderline.uuid, 'price_subtotal': orderline.price_subtotal, 'price_subtotal_incl': orderline.price_subtotal_incl, 'product_id': orderline.product_id.id, 'discount': orderline.discount, 'tax_ids': [[6, False, orderline.tax_ids.mapped(lambda tax: tax.id)]], 'pack_lot_ids': [[0, 0, lot] for lot in orderline.pack_lot_ids.export_for_ui()], 'customer_note': orderline.customer_note, 'refunded_qty': orderline.refunded_qty, 'price_extra': orderline.price_extra, 'full_product_name': orderline.full_product_name, 'refunded_orderline_id': orderline.refunded_orderline_id.id, 'combo_parent_id': orderline.combo_parent_id.id, 'combo_line_ids': orderline.combo_line_ids.mapped('id'), } def export_for_ui(self): return self.mapped(self._export_for_ui) if self else [] def _get_procurement_group(self): return self.order_id.procurement_group_id def _prepare_procurement_group_vals(self): return { 'name': self.order_id.name, 'move_type': self.order_id.config_id.picking_policy, 'pos_order_id': self.order_id.id, 'partner_id': self.order_id.partner_id.id, } def _prepare_procurement_values(self, group_id=False): """ Prepare specific key for moves or other components that will be created from a stock rule coming from a sale order line. This method could be override in order to add other custom key that could be used in move/po creation. """ self.ensure_one() # Use the delivery date if there is else use date_order and lead time if self.order_id.shipping_date: date_deadline = self.order_id.shipping_date else: date_deadline = self.order_id.date_order values = { 'group_id': group_id, 'date_planned': date_deadline, 'date_deadline': date_deadline, 'route_ids': self.order_id.config_id.route_id, 'warehouse_id': self.order_id.config_id.warehouse_id or False, 'partner_id': self.order_id.partner_id.id, 'product_description_variants': self.full_product_name, 'company_id': self.order_id.company_id, } return values def _launch_stock_rule_from_pos_order_lines(self): procurements = [] for line in self: line = line.with_company(line.company_id) if not line.product_id.type in ('consu','product'): continue group_id = line._get_procurement_group() if not group_id: group_id = self.env['procurement.group'].create(line._prepare_procurement_group_vals()) line.order_id.procurement_group_id = group_id values = line._prepare_procurement_values(group_id=group_id) product_qty = line.qty procurement_uom = line.product_id.uom_id procurements.append(self.env['procurement.group'].Procurement( line.product_id, product_qty, procurement_uom, line.order_id.partner_id.property_stock_customer, line.name, line.order_id.name, line.order_id.company_id, values)) if procurements: self.env['procurement.group'].run(procurements) # This next block is currently needed only because the scheduler trigger is done by picking confirmation rather than stock.move confirmation orders = self.mapped('order_id') for order in orders: pickings_to_confirm = order.picking_ids if pickings_to_confirm: # Trigger the Scheduler for Pickings pickings_to_confirm.action_confirm() tracked_lines = order.lines.filtered(lambda l: l.product_id.tracking != 'none') lines_by_tracked_product = groupby(sorted(tracked_lines, key=lambda l: l.product_id.id), key=lambda l: l.product_id.id) for product_id, lines in lines_by_tracked_product: lines = self.env['pos.order.line'].concat(*lines) moves = pickings_to_confirm.move_ids.filtered(lambda m: m.product_id.id == product_id) moves.move_line_ids.unlink() moves._add_mls_related_to_order(lines, are_qties_done=False) moves._recompute_state() return True def _is_product_storable_fifo_avco(self): self.ensure_one() return self.product_id.type == 'product' and self.product_id.cost_method in ['fifo', 'average'] def _compute_total_cost(self, stock_moves): """ Compute the total cost of the order lines. :param stock_moves: recordset of `stock.move`, used for fifo/avco lines """ for line in self.filtered(lambda l: not l.is_total_cost_computed): product = line.product_id if line._is_product_storable_fifo_avco() and stock_moves: product_cost = product._compute_average_price(0, line.qty, line._get_stock_moves_to_consider(stock_moves, product)) else: product_cost = product.standard_price line.total_cost = line.qty * product.cost_currency_id._convert( from_amount=product_cost, to_currency=line.currency_id, company=line.company_id or self.env.company, date=line.order_id.date_order or fields.Date.today(), round=False, ) line.is_total_cost_computed = True def _get_stock_moves_to_consider(self, stock_moves, product): self.ensure_one() return stock_moves.filtered(lambda ml: ml.product_id.id == product.id) @api.depends('price_subtotal', 'total_cost') def _compute_margin(self): for line in self: line.margin = line.price_subtotal - line.total_cost line.margin_percent = not float_is_zero(line.price_subtotal, precision_rounding=line.currency_id.rounding) and line.margin / line.price_subtotal or 0 def _prepare_tax_base_line_values(self, sign=1): """ Convert pos order lines into dictionaries that would be used to compute taxes later. :param sign: An optional parameter to force the sign of amounts. :return: A list of python dictionaries (see '_convert_to_tax_base_line_dict' in account.tax). """ base_line_vals_list = [] for line in self: commercial_partner = self.order_id.partner_id.commercial_partner_id fiscal_position = self.order_id.fiscal_position_id line = line.with_company(self.order_id.company_id) account = line.product_id._get_product_accounts()['income'] or self.order_id.config_id.journal_id.default_account_id if not account: raise UserError(_( "Please define income account for this product: '%s' (id:%d).", line.product_id.name, line.product_id.id, )) if fiscal_position: account = fiscal_position.map_account(account) is_refund = line.qty * line.price_unit < 0 product_name = line.product_id\ .with_context(lang=line.order_id.partner_id.lang or self.env.user.lang)\ .get_product_multiline_description_sale() base_line_vals_list.append( { **self.env['account.tax']._convert_to_tax_base_line_dict( line, partner=commercial_partner, currency=self.order_id.currency_id, product=line.product_id, taxes=line.tax_ids_after_fiscal_position, price_unit=line.price_unit, quantity=sign * line.qty, price_subtotal=sign * line.price_subtotal, discount=line.discount, account=account, is_refund=is_refund, ), 'uom': line.product_uom_id, 'name': product_name, } ) return base_line_vals_list class PosOrderLineLot(models.Model): _name = "pos.pack.operation.lot" _description = "Specify product lot/serial number in pos order line" _rec_name = "lot_name" pos_order_line_id = fields.Many2one('pos.order.line') order_id = fields.Many2one('pos.order', related="pos_order_line_id.order_id", readonly=False) lot_name = fields.Char('Lot Name') product_id = fields.Many2one('product.product', related='pos_order_line_id.product_id', readonly=False) def _export_for_ui(self, lot): return { 'lot_name': lot.lot_name, 'order_line': lot.pos_order_line_id.id, } def export_for_ui(self): return self.mapped(self._export_for_ui) if self else [] class AccountCashRounding(models.Model): _inherit = 'account.cash.rounding' @api.constrains('rounding', 'rounding_method', 'strategy') def _check_session_state(self): open_session = self.env['pos.session'].search([('config_id.rounding_method', 'in', self.ids), ('state', '!=', 'closed')], limit=1) if open_session: raise ValidationError( _("You are not allowed to change the cash rounding configuration while a pos session using it is already opened."))