mirror of
https://github.com/bringout/oca-ocb-sale.git
synced 2026-04-26 18:52:02 +02:00
3762 lines
158 KiB
Python
3762 lines
158 KiB
Python
# -*- coding: utf-8 -*-
|
|
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
|
|
|
import inspect
|
|
import logging
|
|
import base64
|
|
import io
|
|
|
|
from PIL import Image
|
|
from contextlib import contextmanager
|
|
from unittest.mock import patch
|
|
from unittest import skip
|
|
from odoo import Command, api
|
|
|
|
from odoo.tools import DEFAULT_SERVER_DATE_FORMAT
|
|
from odoo.tests import tagged, loaded_demo_data
|
|
from odoo.addons.account.tests.common import TestTaxCommon, AccountTestInvoicingHttpCommon
|
|
from odoo.addons.point_of_sale.tests.common_setup_methods import setup_product_combo_items
|
|
from datetime import date, timedelta
|
|
from odoo.addons.point_of_sale.tests.common import archive_products
|
|
from odoo.exceptions import UserError
|
|
from freezegun import freeze_time
|
|
|
|
_logger = logging.getLogger(__name__)
|
|
|
|
|
|
def _create_image(color: int | str = 0, dims=(1920, 1080), format='JPEG'):
|
|
f = io.BytesIO()
|
|
Image.new('RGB', dims, color).save(f, format)
|
|
f.seek(0)
|
|
return base64.b64encode(f.read())
|
|
|
|
|
|
class TestPointOfSaleHttpCommon(AccountTestInvoicingHttpCommon):
|
|
|
|
@classmethod
|
|
def _get_main_company(cls):
|
|
return cls.company_data['company']
|
|
|
|
def _get_url(self, pos_config=None):
|
|
pos_config = pos_config or self.main_pos_config
|
|
return f"/pos/ui/{pos_config.id}"
|
|
|
|
def get_method_additional_tags(self, test_method):
|
|
additional_tags = super().get_method_additional_tags(test_method)
|
|
method_source = inspect.getsource(test_method)
|
|
if "self.start_pos_tour" in method_source:
|
|
additional_tags.append("is_tour")
|
|
return additional_tags
|
|
|
|
def start_pos_tour(self, tour_name, login="pos_user", **kwargs):
|
|
self.start_tour(self._get_url(pos_config=kwargs.get('pos_config')), tour_name, login=login, **kwargs)
|
|
|
|
@contextmanager
|
|
def with_new_session(self, config=None, user=None):
|
|
config = config or self.main_pos_config
|
|
user = user or self.pos_user
|
|
config.with_user(user).open_ui()
|
|
session = config.current_session_id
|
|
yield session
|
|
session.post_closing_cash_details(0)
|
|
session.close_session_from_ui()
|
|
|
|
@classmethod
|
|
def setUpClass(cls):
|
|
super().setUpClass()
|
|
|
|
env = cls.env
|
|
cls.env.user.group_ids += env.ref('point_of_sale.group_pos_manager')
|
|
journal_obj = env['account.journal']
|
|
account_obj = env['account.account']
|
|
main_company = cls._get_main_company()
|
|
|
|
cls.account_receivable = account_obj.create({'code': 'X1012',
|
|
'name': 'Account Receivable - Test',
|
|
'account_type': 'asset_receivable',
|
|
'reconcile': True})
|
|
env.company.account_default_pos_receivable_account_id = cls.account_receivable
|
|
env['ir.default'].set('res.partner', 'property_account_receivable_id', cls.account_receivable.id, company_id=main_company.id)
|
|
# Pricelists are set below, do not take demo data into account
|
|
env['res.partner'].sudo().invalidate_model(['property_product_pricelist', 'specific_property_product_pricelist'])
|
|
# remove the all specific values for all companies only for test
|
|
env.cr.execute('UPDATE res_partner SET specific_property_product_pricelist = NULL')
|
|
|
|
# Create user.
|
|
cls.pos_user = cls.env['res.users'].create({
|
|
'name': 'A simple PoS man!',
|
|
'login': 'pos_user',
|
|
'password': 'pos_user',
|
|
'group_ids': [
|
|
(4, cls.env.ref('base.group_user').id),
|
|
(4, cls.env.ref('point_of_sale.group_pos_user').id),
|
|
(4, cls.env.ref('stock.group_stock_user').id),
|
|
],
|
|
'tz': 'America/New_York',
|
|
})
|
|
cls.pos_admin = cls.env['res.users'].create({
|
|
'name': 'A powerful PoS man!',
|
|
'login': 'pos_admin',
|
|
'password': 'pos_admin',
|
|
'group_ids': [
|
|
(4, cls.env.ref('point_of_sale.group_pos_manager').id),
|
|
],
|
|
'tz': 'America/New_York',
|
|
})
|
|
|
|
cls.pos_user.partner_id.email = 'pos_user@test.com'
|
|
cls.pos_admin.partner_id.email = 'pos_admin@test.com'
|
|
|
|
cls.bank_journal = journal_obj.create({
|
|
'name': 'Bank Test',
|
|
'type': 'bank',
|
|
'company_id': main_company.id,
|
|
'code': 'BNK',
|
|
'sequence': 10,
|
|
})
|
|
|
|
cls.bank_payment_method = env['pos.payment.method'].create({
|
|
'name': 'Bank',
|
|
'journal_id': cls.bank_journal.id,
|
|
'outstanding_account_id': cls.inbound_payment_method_line.payment_account_id.id,
|
|
})
|
|
env['pos.config'].search([]).unlink()
|
|
cls.main_pos_config = env['pos.config'].create({
|
|
'name': 'Shop',
|
|
'module_pos_restaurant': False,
|
|
})
|
|
|
|
env['res.partner'].create({
|
|
'name': 'Acme Corporation',
|
|
})
|
|
|
|
cash_journal = journal_obj.create({
|
|
'name': 'Cash Test',
|
|
'type': 'cash',
|
|
'company_id': main_company.id,
|
|
'code': 'CSH',
|
|
'sequence': 10,
|
|
})
|
|
|
|
archive_products(env)
|
|
|
|
cls.tip = env.ref('point_of_sale.product_product_tip')
|
|
|
|
cls.pos_desk_misc_test = env['pos.category'].create({
|
|
'name': 'Misc test',
|
|
})
|
|
cls.pos_cat_chair_test = env['pos.category'].create({
|
|
'name': 'Chair test',
|
|
})
|
|
cls.pos_cat_desk_test = env['pos.category'].create({
|
|
'name': 'Desk test',
|
|
})
|
|
|
|
# test an extra price on an attribute
|
|
cls.whiteboard_pen = env['product.template'].create({
|
|
'name': 'Whiteboard Pen',
|
|
'available_in_pos': True,
|
|
'list_price': 1.20,
|
|
'taxes_id': False,
|
|
'weight': 0.01,
|
|
'to_weight': True,
|
|
'pos_categ_ids': [(4, cls.pos_desk_misc_test.id)],
|
|
})
|
|
cls.wall_shelf = env['product.template'].create({
|
|
'name': 'Wall Shelf Unit',
|
|
'available_in_pos': True,
|
|
'list_price': 1.98,
|
|
'taxes_id': False,
|
|
'barcode': '2100005000000',
|
|
})
|
|
cls.small_shelf = env['product.template'].create({
|
|
'name': 'Small Shelf',
|
|
'available_in_pos': True,
|
|
'list_price': 2.83,
|
|
'taxes_id': False,
|
|
})
|
|
cls.magnetic_board = env['product.template'].create({
|
|
'name': 'Magnetic Board',
|
|
'available_in_pos': True,
|
|
'list_price': 1.98,
|
|
'taxes_id': False,
|
|
'barcode': '2305000000004',
|
|
})
|
|
cls.monitor_stand = env['product.template'].create({
|
|
'name': 'Monitor Stand',
|
|
'available_in_pos': True,
|
|
'list_price': 3.19,
|
|
'taxes_id': False,
|
|
'barcode': '0123456789', # No pattern in barcode nomenclature
|
|
})
|
|
cls.desk_pad = env['product.template'].create({
|
|
'name': 'Desk Pad',
|
|
'available_in_pos': True,
|
|
'list_price': 1.98,
|
|
'taxes_id': False,
|
|
'pos_categ_ids': [(4, cls.pos_cat_desk_test.id)],
|
|
})
|
|
cls.letter_tray = env['product.template'].create({
|
|
'name': 'Letter Tray',
|
|
'available_in_pos': True,
|
|
'list_price': 4.80,
|
|
'taxes_id': False,
|
|
'categ_id': env.ref('product.product_category_services').id,
|
|
'pos_categ_ids': [(4, cls.pos_cat_chair_test.id)],
|
|
})
|
|
cls.desk_organizer = env['product.template'].create({
|
|
'name': 'Desk Organizer',
|
|
'available_in_pos': True,
|
|
'list_price': 5.10,
|
|
'taxes_id': False,
|
|
'barcode': '2300002000007',
|
|
})
|
|
cls.configurable_chair = env['product.template'].create({
|
|
'name': 'Configurable Chair',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
})
|
|
|
|
attribute = env['product.attribute'].create({
|
|
'name': 'add 2',
|
|
})
|
|
attribute_value = env['product.attribute.value'].create({
|
|
'name': 'add 2',
|
|
'attribute_id': attribute.id,
|
|
})
|
|
line = env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': cls.whiteboard_pen.id,
|
|
'attribute_id': attribute.id,
|
|
'value_ids': [(6, 0, attribute_value.ids)]
|
|
})
|
|
line.product_template_value_ids[0].price_extra = 2
|
|
|
|
cls.chair_color_attribute = env['product.attribute'].create({
|
|
'name': 'Color',
|
|
'display_type': 'color',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
cls.chair_color_red = env['product.attribute.value'].create({
|
|
'name': 'Red',
|
|
'attribute_id': cls.chair_color_attribute.id,
|
|
'html_color': '#ff0000',
|
|
})
|
|
chair_color_blue = env['product.attribute.value'].create({
|
|
'name': 'Blue',
|
|
'attribute_id': cls.chair_color_attribute.id,
|
|
'html_color': '#0000ff',
|
|
})
|
|
chair_color_line = env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': cls.configurable_chair.id,
|
|
'attribute_id': cls.chair_color_attribute.id,
|
|
'value_ids': [(6, 0, [cls.chair_color_red.id, chair_color_blue.id])]
|
|
})
|
|
chair_color_line.product_template_value_ids[0].price_extra = 1
|
|
|
|
chair_legs_attribute = env['product.attribute'].create({
|
|
'name': 'Chair Legs',
|
|
'display_type': 'select',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
chair_legs_metal = env['product.attribute.value'].create({
|
|
'name': 'Metal',
|
|
'attribute_id': chair_legs_attribute.id,
|
|
})
|
|
chair_legs_wood = env['product.attribute.value'].create({
|
|
'name': 'Wood',
|
|
'attribute_id': chair_legs_attribute.id,
|
|
})
|
|
env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': cls.configurable_chair.id,
|
|
'attribute_id': chair_legs_attribute.id,
|
|
'value_ids': [(6, 0, [chair_legs_metal.id, chair_legs_wood.id])]
|
|
})
|
|
|
|
cls.chair_fabrics_attribute = env['product.attribute'].create({
|
|
'name': 'Fabrics',
|
|
'display_type': 'radio',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
chair_fabrics_leather = env['product.attribute.value'].create({
|
|
'name': 'Leather',
|
|
'attribute_id': cls.chair_fabrics_attribute.id,
|
|
})
|
|
cls.chair_fabrics_wool = env['product.attribute.value'].create({
|
|
'name': 'wool',
|
|
'attribute_id': cls.chair_fabrics_attribute.id,
|
|
})
|
|
cls.chair_fabrics_other = env['product.attribute.value'].create({
|
|
'name': 'Other',
|
|
'attribute_id': cls.chair_fabrics_attribute.id,
|
|
'is_custom': True,
|
|
})
|
|
env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': cls.configurable_chair.id,
|
|
'attribute_id': cls.chair_fabrics_attribute.id,
|
|
'value_ids': [(6, 0, [chair_fabrics_leather.id, cls.chair_fabrics_wool.id, cls.chair_fabrics_other.id])]
|
|
})
|
|
chair_color_line.product_template_value_ids[1].is_custom = True
|
|
|
|
cls.chair_addons_attribute = env['product.attribute'].create({
|
|
'name': 'Add-ons',
|
|
'display_type': 'multi',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
cls.chair_addon_cushion = env['product.attribute.value'].create({
|
|
'name': 'Cushion',
|
|
'attribute_id': cls.chair_addons_attribute.id,
|
|
})
|
|
cls.chair_addon_cupholder = env['product.attribute.value'].create({
|
|
'name': 'Cup Holder',
|
|
'attribute_id': cls.chair_addons_attribute.id,
|
|
})
|
|
cls.chair_addon_headrest = env['product.attribute.value'].create({
|
|
'name': 'Headrest',
|
|
'attribute_id': cls.chair_addons_attribute.id,
|
|
})
|
|
env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': cls.configurable_chair.id,
|
|
'attribute_id': cls.chair_addons_attribute.id,
|
|
'value_ids': [(6, 0, [cls.chair_addon_cushion.id, cls.chair_addon_cupholder.id, cls.chair_addon_headrest.id])]
|
|
})
|
|
|
|
fixed_pricelist = env['product.pricelist'].create({
|
|
'name': 'Fixed',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 1,
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 2,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.wall_shelf.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 13.95, # test for issues like in 7f260ab517ebde634fc274e928eb062463f0d88f
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.small_shelf.product_variant_id.id,
|
|
})],
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'Percentage',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'percentage',
|
|
'percent_price': 100,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.wall_shelf.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'percentage',
|
|
'percent_price': 99,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.small_shelf.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'percentage',
|
|
'percent_price': 0,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.magnetic_board.product_variant_id.id,
|
|
})],
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'Formula',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'formula',
|
|
'price_discount': 6,
|
|
'price_surcharge': 5,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.wall_shelf.product_variant_id.id,
|
|
}), (0, 0, {
|
|
# .99 prices
|
|
'compute_price': 'formula',
|
|
'price_surcharge': -0.01,
|
|
'price_round': 1,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.small_shelf.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'formula',
|
|
'price_min_margin': 10,
|
|
'price_max_margin': 100,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.magnetic_board.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'formula',
|
|
'price_surcharge': 10,
|
|
'price_max_margin': 5,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.monitor_stand.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'formula',
|
|
'price_discount': -100,
|
|
'price_min_margin': 5,
|
|
'price_max_margin': 20,
|
|
'applied_on': '0_product_variant',
|
|
'product_id': cls.desk_pad.product_variant_id.id,
|
|
})],
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'min_quantity ordering',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 1,
|
|
'applied_on': '0_product_variant',
|
|
'min_quantity': 2,
|
|
'product_id': cls.wall_shelf.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 2,
|
|
'applied_on': '0_product_variant',
|
|
'min_quantity': 1,
|
|
'product_id': cls.wall_shelf.product_variant_id.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 1,
|
|
'applied_on': '0_product_variant',
|
|
'min_quantity': 5,
|
|
'product_id': cls.monitor_stand.product_variant_id.id,
|
|
})],
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'Product template',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 1,
|
|
'applied_on': '1_product',
|
|
'product_tmpl_id': cls.wall_shelf.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 2,
|
|
})],
|
|
})
|
|
|
|
product_category_3 = env['product.category'].create({
|
|
'name': 'Services',
|
|
'parent_id': env.ref('product.product_category_services').id,
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
# no category has precedence over category
|
|
'name': 'Category vs no category',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 1,
|
|
'applied_on': '2_product_category',
|
|
'categ_id': product_category_3.id,
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 2,
|
|
})],
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'Category',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 2,
|
|
'applied_on': '2_product_category',
|
|
'categ_id': env.ref('product.product_category_services').id,
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 1,
|
|
'applied_on': '2_product_category',
|
|
'categ_id': product_category_3.id,
|
|
})],
|
|
})
|
|
|
|
today = date.today()
|
|
one_week_ago = today - timedelta(weeks=1)
|
|
two_weeks_ago = today - timedelta(weeks=2)
|
|
one_week_from_now = today + timedelta(weeks=1)
|
|
two_weeks_from_now = today + timedelta(weeks=2)
|
|
|
|
public_pricelist = env['product.pricelist'].create({
|
|
'name': 'Public Pricelist',
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'Dates',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 1,
|
|
'date_start': two_weeks_ago.strftime(DEFAULT_SERVER_DATE_FORMAT),
|
|
'date_end': one_week_ago.strftime(DEFAULT_SERVER_DATE_FORMAT),
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 2,
|
|
'date_start': today.strftime(DEFAULT_SERVER_DATE_FORMAT),
|
|
'date_end': one_week_from_now.strftime(DEFAULT_SERVER_DATE_FORMAT),
|
|
}), (0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 3,
|
|
'date_start': one_week_from_now.strftime(DEFAULT_SERVER_DATE_FORMAT),
|
|
'date_end': two_weeks_from_now.strftime(DEFAULT_SERVER_DATE_FORMAT),
|
|
})],
|
|
})
|
|
|
|
cost_base_pricelist = env['product.pricelist'].create({
|
|
'name': 'Cost base',
|
|
'item_ids': [(0, 0, {
|
|
'base': 'standard_price',
|
|
'compute_price': 'percentage',
|
|
'percent_price': 55,
|
|
})],
|
|
})
|
|
|
|
pricelist_base_pricelist = env['product.pricelist'].create({
|
|
'name': 'Pricelist base',
|
|
'item_ids': [(0, 0, {
|
|
'base': 'pricelist',
|
|
'base_pricelist_id': cost_base_pricelist.id,
|
|
'compute_price': 'percentage',
|
|
'percent_price': 15,
|
|
})],
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'Pricelist base 2',
|
|
'item_ids': [(0, 0, {
|
|
'base': 'pricelist',
|
|
'base_pricelist_id': pricelist_base_pricelist.id,
|
|
'compute_price': 'percentage',
|
|
'percent_price': 3,
|
|
})],
|
|
})
|
|
|
|
env['product.pricelist'].create({
|
|
'name': 'Pricelist base rounding',
|
|
'item_ids': [(0, 0, {
|
|
'base': 'pricelist',
|
|
'base_pricelist_id': fixed_pricelist.id,
|
|
'compute_price': 'percentage',
|
|
'percent_price': 0.01,
|
|
})],
|
|
})
|
|
|
|
excluded_pricelist = env['product.pricelist'].create({
|
|
'name': 'Not loaded'
|
|
})
|
|
res_partner_18 = env['res.partner'].create({
|
|
'name': 'Lumber Inc',
|
|
'is_company': True,
|
|
})
|
|
res_partner_18.property_product_pricelist = excluded_pricelist
|
|
|
|
test_sale_journal = journal_obj.create({'name': 'Sales Journal - Test',
|
|
'code': 'TSJ',
|
|
'type': 'sale',
|
|
'company_id': main_company.id})
|
|
|
|
all_pricelists = env['product.pricelist'].search([
|
|
('id', '!=', excluded_pricelist.id),
|
|
'|', ('company_id', '=', main_company.id), ('company_id', '=', False)
|
|
])
|
|
all_pricelists.write(dict(currency_id=main_company.currency_id.id))
|
|
|
|
FP_POS_2M = env['account.fiscal.position'].create({
|
|
'name': "FP-POS-2M",
|
|
})
|
|
|
|
src_tax = env['account.tax'].create({'name': "SRC", 'amount': 10})
|
|
env['account.tax'].create({'name': "DST", 'amount': 5, 'fiscal_position_ids': [Command.link(FP_POS_2M.id)], 'original_tax_ids': [Command.link(src_tax.id)]})
|
|
env['account.tax'].create({'name': "DST2", 'amount': 10, 'fiscal_position_ids': [Command.link(FP_POS_2M.id)], 'original_tax_ids': [Command.link(src_tax.id)]})
|
|
|
|
cls.letter_tray.taxes_id = [(6, 0, [src_tax.id])]
|
|
|
|
cls.main_pos_config.write({
|
|
'tax_regime_selection': True,
|
|
'fiscal_position_ids': FP_POS_2M,
|
|
'journal_id': test_sale_journal.id,
|
|
'invoice_journal_id': test_sale_journal.id,
|
|
'payment_method_ids': [(0, 0, { 'name': 'Cash',
|
|
'journal_id': cash_journal.id,
|
|
'receivable_account_id': cls.account_receivable.id,
|
|
})],
|
|
'use_pricelist': True,
|
|
'pricelist_id': public_pricelist.id,
|
|
'available_pricelist_ids': [(4, pricelist.id) for pricelist in all_pricelists],
|
|
})
|
|
|
|
cls.printer = cls.env['pos.printer'].create({
|
|
'name': 'Printer',
|
|
'printer_type': 'epson_epos',
|
|
'epson_printer_ip': '0.0.0.0',
|
|
})
|
|
|
|
# Set customers
|
|
# Unlink some data partners that makes the test crash
|
|
for xmlid in [
|
|
"l10n_us_hr_payroll.res_partner_taxation_va",
|
|
"l10n_us_hr_payroll.res_partner_revenue_dc",
|
|
"l10n_us_hr_payroll.res_partner_pfl_dc",
|
|
"l10n_us_hr_payroll.res_partner_revenue_or",
|
|
"l10n_us_hr_payroll.res_partner_dcbs_or",
|
|
"l10n_us_hr_payroll.res_partner_employment_or",
|
|
"l10n_us_hr_payroll.res_partner_revenue_nc",
|
|
"l10n_us_hr_payroll.res_partner_state_tax_commission_id",
|
|
"l10n_us_hr_payroll.res_partner_department_taxes_vt",
|
|
"l10n_us_hr_payroll.res_partner_department_revenue_il",
|
|
"l10n_us_hr_payroll.res_partner_department_revenue_az",
|
|
]:
|
|
partner = cls.env.ref(xmlid, raise_if_not_found=False)
|
|
if partner:
|
|
partner.unlink()
|
|
|
|
partners = cls.env['res.partner'].create([
|
|
{'name': 'Partner Test 1'},
|
|
{'name': 'Partner Test 2'},
|
|
{'name': 'Partner Test 3'},
|
|
{
|
|
'name': 'Partner Full',
|
|
'email': 'partner.full@example.com',
|
|
'street': '77 Santa Barbara Rd',
|
|
'city': 'Pleasant Hill',
|
|
'state_id': cls.env.ref('base.state_us_5').id,
|
|
'zip': '94523',
|
|
'country_id': cls.env.ref('base.us').id,
|
|
}
|
|
])
|
|
cls.partner_test_1 = partners[0]
|
|
cls.partner_test_2 = partners[1]
|
|
cls.partner_test_3 = partners[2]
|
|
cls.partner_full = partners[3]
|
|
|
|
# Change the default sale pricelist of customers,
|
|
# so the js tests can expect deterministically this pricelist when selecting a customer.
|
|
# bad hack only for test
|
|
env['ir.default'].set("res.partner", "specific_property_product_pricelist", public_pricelist.id, company_id=main_company.id)
|
|
|
|
|
|
@tagged('post_install', '-at_install')
|
|
class TestUi(TestPointOfSaleHttpCommon):
|
|
def test_01_pos_basic_order(self):
|
|
self.start_pos_tour('pos_pricelist')
|
|
|
|
def test_pos_basic_order_02_decimal_order_quantity(self):
|
|
self.start_pos_tour('pos_basic_order_02_decimal_order_quantity')
|
|
|
|
def test_pos_basic_order_03_tax_position(self):
|
|
self.start_pos_tour('pos_basic_order_03_tax_position')
|
|
|
|
def test_floating_order_tour(self):
|
|
self.start_pos_tour('FloatingOrderTour')
|
|
|
|
def test_product_screen_tour(self):
|
|
self.whiteboard_pen.write({
|
|
'is_favorite': True
|
|
})
|
|
self.start_pos_tour('ProductScreenTour')
|
|
|
|
def test_payment_screen_tour(self):
|
|
self.start_pos_tour('PaymentScreenTour')
|
|
|
|
def test_receipt_screen_tour(self):
|
|
self.tip.write({
|
|
'taxes_id': False
|
|
})
|
|
self.main_pos_config.write({
|
|
'iface_tipproduct': True,
|
|
'tip_product_id': self.tip.id,
|
|
'ship_later': True
|
|
})
|
|
self.start_pos_tour('ReceiptScreenTour')
|
|
for order in self.env['pos.order'].search([]):
|
|
self.assertEqual(order.state, 'paid', "Validated order has payment of " + str(order.amount_paid) + " and total of " + str(order.amount_total))
|
|
|
|
# check if email from ReceiptScreenTour is properly sent
|
|
email_count = self.env['mail.mail'].search_count([('email_to', '=', 'test@receiptscreen.com')])
|
|
self.assertEqual(email_count, 1)
|
|
|
|
@skip('Temporary to fast merge new valuation')
|
|
def test_02_pos_with_invoiced(self):
|
|
self.pos_user.write({
|
|
'group_ids': [
|
|
(4, self.env.ref('account.group_account_invoice').id),
|
|
]
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'ChromeTour', login="pos_user")
|
|
n_invoiced = self.env['pos.order'].search_count([('account_move', '!=', False)])
|
|
n_paid = self.env['pos.order'].search_count([('state', '=', 'paid')])
|
|
self.assertEqual(n_invoiced, 1, 'There should be 1 invoiced order.')
|
|
self.assertEqual(n_paid, 2, 'There should be 2 paid order.')
|
|
last_order = self.env['pos.order'].search([], limit=1, order="id desc")
|
|
self.assertEqual(last_order.lines[0].price_subtotal, 30.0)
|
|
self.assertEqual(last_order.lines[0].price_subtotal_incl, 30.0)
|
|
# Check if session name contains config name as prefix
|
|
self.assertEqual(self.main_pos_config.name in last_order.session_id.name, True)
|
|
|
|
def test_03_pos_with_lots(self):
|
|
|
|
# open a session, the /pos/ui controller will redirect to it
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
|
|
self.monitor_stand.tracking = 'lot'
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_03_pos_with_lots', login="pos_user")
|
|
|
|
def test_04_product_configurator(self):
|
|
# Making one attribute inactive to verify that it doesn't show
|
|
configurable_product = self.env['product.product'].search([('name', '=', 'Configurable Chair'), ('available_in_pos', '=', 'True')], limit=1)
|
|
fabrics_line = configurable_product.attribute_line_ids[2]
|
|
fabrics_line.product_template_value_ids[1].ptav_active = False
|
|
self.pos_user.write({
|
|
'group_ids': [
|
|
(4, self.env.ref('stock.group_stock_manager').id),
|
|
]
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('ProductConfiguratorTour')
|
|
|
|
def test_optional_product(self):
|
|
# optional product in pos
|
|
image = _create_image(color="orange")
|
|
|
|
self.small_shelf.write({'image_1920': image})
|
|
|
|
self.desk_pad.write({'pos_optional_product_ids': [
|
|
Command.set([ self.small_shelf.id ])
|
|
]})
|
|
|
|
self.letter_tray.write({
|
|
'pos_optional_product_ids': [Command.set([self.configurable_chair.id])],
|
|
'barcode': 'lettertray'
|
|
})
|
|
# Case 1: Images ON → images must be visible in optional product dialog
|
|
self.main_pos_config.show_product_images = True
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_optional_product')
|
|
|
|
# Case 2: Images OFF → product images should not be shown
|
|
self.main_pos_config.show_product_images = False
|
|
self.start_pos_tour('test_optional_product_image_not_display')
|
|
|
|
@skip('Temporary to fast merge new valuation')
|
|
def test_05_ticket_screen(self):
|
|
self.pos_user.write({
|
|
'group_ids': [
|
|
(4, self.env.ref('account.group_account_invoice').id),
|
|
]
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'TicketScreenTour', login="pos_user")
|
|
|
|
def test_product_information_screen_admin(self):
|
|
'''Consider this test method to contain a test tour with miscellaneous tests/checks that require admin access.
|
|
'''
|
|
self.product_a.available_in_pos = True
|
|
self.pos_admin.write({
|
|
'group_ids': [Command.link(self.env.ref('product.group_product_manager').id)],
|
|
})
|
|
self.main_pos_config.write({
|
|
'is_margins_costs_accessible_to_every_user': True,
|
|
})
|
|
self.assertFalse(self.product_a.is_storable)
|
|
self.main_pos_config.with_user(self.pos_admin).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'CheckProductInformation', login="pos_admin")
|
|
|
|
def test_fixed_tax_negative_qty(self):
|
|
""" Assert the negative amount of a negative-quantity orderline
|
|
with zero-amount product with fixed tax.
|
|
"""
|
|
|
|
# setup the zero-amount product
|
|
tax_received_account = self.env['account.account'].create({
|
|
'name': 'TAX_BASE',
|
|
'code': 'TBASE',
|
|
'account_type': 'asset_current',
|
|
})
|
|
fixed_tax = self.env['account.tax'].create({
|
|
'name': 'fixed amount tax',
|
|
'amount_type': 'fixed',
|
|
'amount': 1,
|
|
'invoice_repartition_line_ids': [
|
|
(0, 0, {'repartition_type': 'base'}),
|
|
(0, 0, {
|
|
'repartition_type': 'tax',
|
|
'account_id': tax_received_account.id,
|
|
}),
|
|
],
|
|
'price_include_override': 'tax_excluded',
|
|
})
|
|
zero_amount_product = self.env['product.product'].create({
|
|
'name': 'Zero Amount Product',
|
|
'available_in_pos': True,
|
|
'list_price': 0,
|
|
'taxes_id': [(6, 0, [fixed_tax.id])],
|
|
'categ_id': self.env.ref('product.product_category_services').id,
|
|
})
|
|
|
|
# Make an order with the zero-amount product from the frontend.
|
|
# We need to do this because of the fix in the "compute_all" port.
|
|
self.main_pos_config.write({'iface_tax_included': 'total'})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'FixedTaxNegativeQty', login="pos_user")
|
|
pos_session = self.main_pos_config.current_session_id
|
|
|
|
# Close the session and check the session journal entry.
|
|
pos_session.action_pos_session_validate()
|
|
|
|
lines = pos_session.move_id.line_ids.sorted('balance')
|
|
|
|
# order in the tour is paid using the bank payment method.
|
|
bank_pm = self.main_pos_config.payment_method_ids.filtered(lambda pm: pm.name == 'Bank')
|
|
|
|
self.assertEqual(lines[0].account_id, bank_pm.receivable_account_id or self.env.company.account_default_pos_receivable_account_id)
|
|
self.assertAlmostEqual(lines[0].balance, -1)
|
|
self.assertEqual(lines[1].account_id, self.env.company.income_account_id)
|
|
self.assertAlmostEqual(lines[1].balance, 0)
|
|
self.assertEqual(lines[2].account_id, tax_received_account)
|
|
self.assertAlmostEqual(lines[2].balance, 1)
|
|
|
|
def test_change_without_cash_method(self):
|
|
#create bank payment method
|
|
bank_pm = self.env['pos.payment.method'].create({
|
|
'name': 'Bank',
|
|
'receivable_account_id': self.env.company.account_default_pos_receivable_account_id.id,
|
|
'is_cash_count': False,
|
|
'split_transactions': False,
|
|
'company_id': self.env.company.id,
|
|
})
|
|
self.main_pos_config.write({'payment_method_ids': [(6, 0, bank_pm.ids)], 'ship_later': True})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'PaymentScreenTour2', login="pos_user")
|
|
|
|
def test_rounding_up(self):
|
|
rouding_method = self.env['account.cash.rounding'].create({
|
|
'name': 'Rounding up',
|
|
'rounding': 0.05,
|
|
'rounding_method': 'UP',
|
|
})
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'Product Test',
|
|
'available_in_pos': True,
|
|
'list_price': 1.96,
|
|
'taxes_id': False,
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'rounding_method': rouding_method.id,
|
|
'cash_rounding': True,
|
|
'only_round_cash_method': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'PaymentScreenRoundingUp', login="pos_user")
|
|
|
|
def test_rounding_down(self):
|
|
rouding_method = self.env['account.cash.rounding'].create({
|
|
'name': 'Rounding down',
|
|
'rounding': 0.05,
|
|
'rounding_method': 'DOWN',
|
|
})
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'Product Test',
|
|
'available_in_pos': True,
|
|
'list_price': 1.98,
|
|
'taxes_id': False,
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'rounding_method': rouding_method.id,
|
|
'cash_rounding': True,
|
|
'only_round_cash_method': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'PaymentScreenRoundingDown', login="pos_user")
|
|
self.env["pos.order"].search([('state', '=', 'draft')]).write({'state': 'cancel'})
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'PaymentScreenTotalDueWithOverPayment', login="pos_user")
|
|
|
|
def test_pos_closing_cash_details(self):
|
|
"""Test cash difference *loss* at closing.
|
|
"""
|
|
self.main_pos_config.open_ui()
|
|
current_session = self.main_pos_config.current_session_id
|
|
current_session.post_closing_cash_details(0)
|
|
current_session.close_session_from_ui()
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'CashClosingDetails', login="pos_user")
|
|
self.assertEqual(self.main_pos_config.last_session_closing_cash, 50.0)
|
|
cash_diff_line = self.env['account.bank.statement.line'].search([
|
|
('payment_ref', 'ilike', 'Cash difference observed during the counting (Loss)')
|
|
])
|
|
self.assertAlmostEqual(cash_diff_line.amount, -1.00)
|
|
|
|
def test_cash_payments_should_reflect_on_next_opening(self):
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'OrderPaidInCash', login="pos_user")
|
|
self.assertEqual(self.main_pos_config.last_session_closing_cash, 25.0)
|
|
|
|
def test_pos_session_statistics_display(self):
|
|
"""Test that POS session statistics are properly displayed in the UI."""
|
|
# For testing `opening_cash` and `paid_orders` in dashboard
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'SessionStatisticsDisplay', login="pos_user")
|
|
|
|
# For testing `draft_orders`
|
|
self.env['pos.order'].create({
|
|
'config_id': self.main_pos_config.id,
|
|
'session_id': self.main_pos_config.current_session_id.id,
|
|
'company_id': self.main_pos_config.company_id.id,
|
|
'amount_total': 10.0,
|
|
'amount_paid': 10.0,
|
|
'amount_tax': 0.0,
|
|
'amount_return': 0.0,
|
|
'to_invoice': False,
|
|
'partner_id': False,
|
|
'pricelist_id': self.main_pos_config.pricelist_id.id,
|
|
'pos_reference': '1000-004-00001',
|
|
'name': 'Order 1001',
|
|
'state': 'draft',
|
|
'lines': [(0, 0, {
|
|
'product_id': self.desk_pad.product_variant_id.id,
|
|
'price_unit': 10.00,
|
|
'discount': 0,
|
|
'qty': 1,
|
|
'tax_ids': False,
|
|
'price_subtotal': 10.00,
|
|
'price_subtotal_incl': 10.00,
|
|
})],
|
|
})
|
|
|
|
dashboard_statistics = self.main_pos_config.statistics_for_current_session
|
|
|
|
self.assertTrue(dashboard_statistics['date']['is_started'])
|
|
self.assertEqual(dashboard_statistics['cash']['raw_opening_cash'], 100.0)
|
|
self.assertEqual(dashboard_statistics['orders']['paid']['amount'], 45.0)
|
|
self.assertEqual(dashboard_statistics['orders']['paid']['count'], 2)
|
|
self.assertEqual(dashboard_statistics['orders']['draft']['amount'], 10.0)
|
|
self.assertEqual(dashboard_statistics['orders']['draft']['count'], 1)
|
|
|
|
def test_tax_control_button_visiblity(self):
|
|
self.main_pos_config.write({
|
|
'tax_regime_selection': False,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_tax_control_button_visiblity')
|
|
|
|
def test_fiscal_position_no_tax(self):
|
|
#create a tax of 15% with price included
|
|
tax = self.env['account.tax'].create({
|
|
'name': 'Tax 15%',
|
|
'amount': 15,
|
|
'price_include_override': 'tax_included',
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
})
|
|
|
|
#create a product with the tax
|
|
self.product = self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'taxes_id': [(6, 0, [tax.id])],
|
|
'list_price': 100,
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
#create a fiscal position that map the tax to no tax
|
|
fiscal_position = self.env['account.fiscal.position'].create({
|
|
'name': 'No Tax',
|
|
})
|
|
|
|
pricelist = self.env['product.pricelist'].create({
|
|
'name': 'Test Pricelist',
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'tax_regime_selection': True,
|
|
'fiscal_position_ids': [(6, 0, [fiscal_position.id])],
|
|
'available_pricelist_ids': [(6, 0, [pricelist.id])],
|
|
'pricelist_id': pricelist.id,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'FiscalPositionNoTax', login="pos_user")
|
|
|
|
def test_fiscal_position_inclusive_and_exclusive_tax(self):
|
|
""" Test the mapping of fiscal position for both Tax Inclusive ans Tax Exclusive"""
|
|
# create a fiscal position that map the tax
|
|
fiscal_position_1 = self.env['account.fiscal.position'].create({
|
|
'name': 'Incl. to Incl.',
|
|
})
|
|
fiscal_position_2 = self.env['account.fiscal.position'].create({
|
|
'name': 'Incl. to Excl.',
|
|
})
|
|
fiscal_position_3 = self.env['account.fiscal.position'].create({
|
|
'name': 'Excl. to Excl.',
|
|
})
|
|
fiscal_position_4 = self.env['account.fiscal.position'].create({
|
|
'name': 'Excl. to Incl.',
|
|
})
|
|
|
|
# create a tax with price included
|
|
tax_inclusive_1 = self.env['account.tax'].create({
|
|
'name': 'Tax incl.20%',
|
|
'amount': 20,
|
|
'price_include_override': 'tax_included',
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
})
|
|
tax_exclusive_1 = self.env['account.tax'].create({
|
|
'name': 'Tax excl.20%',
|
|
'amount': 20,
|
|
'price_include_override': 'tax_excluded',
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
})
|
|
self.env['account.tax'].create({
|
|
'name': 'Tax incl.10%',
|
|
'amount': 10,
|
|
'price_include_override': 'tax_included',
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'fiscal_position_ids': [Command.set((fiscal_position_1 | fiscal_position_4).ids)],
|
|
'original_tax_ids': [Command.set((tax_inclusive_1 | tax_exclusive_1).ids)],
|
|
})
|
|
self.env['account.tax'].create({
|
|
'name': 'Tax excl.10%',
|
|
'amount': 10,
|
|
'price_include_override': 'tax_excluded',
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'fiscal_position_ids': [Command.set((fiscal_position_2 | fiscal_position_3).ids)],
|
|
'original_tax_ids': [Command.set((tax_inclusive_1 | tax_exclusive_1).ids)],
|
|
})
|
|
self.test_product_1 = self.env['product.product'].create({
|
|
'name': 'Test Product 1',
|
|
'available_in_pos': True,
|
|
'list_price': 100,
|
|
'taxes_id': [(6, 0, [tax_inclusive_1.id])],
|
|
})
|
|
|
|
self.test_product_2 = self.env['product.product'].create({
|
|
'name': 'Test Product 2',
|
|
'available_in_pos': True,
|
|
'list_price': 100,
|
|
'taxes_id': [(6, 0, [tax_exclusive_1.id])],
|
|
})
|
|
|
|
# add the fiscal position to the PoS
|
|
self.main_pos_config.write({
|
|
'tax_regime_selection': True,
|
|
'fiscal_position_ids': [(6, 0, [
|
|
fiscal_position_1.id,
|
|
fiscal_position_2.id,
|
|
fiscal_position_3.id,
|
|
fiscal_position_4.id,
|
|
])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'FiscalPositionIncl', login="pos_user")
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'FiscalPositionExcl', login="pos_user")
|
|
|
|
def test_06_pos_discount_display_with_multiple_pricelist(self):
|
|
""" Test the discount display on the POS screen when multiple pricelists are used."""
|
|
test_product = self.env['product.template'].create({
|
|
'name': 'Test Product',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
})
|
|
|
|
base_pricelist = self.env['product.pricelist'].create({
|
|
'name': 'base_pricelist',
|
|
})
|
|
|
|
self.env['product.pricelist.item'].create({
|
|
'pricelist_id': base_pricelist.id,
|
|
'product_tmpl_id': test_product.id,
|
|
'compute_price': 'percentage',
|
|
'applied_on': '1_product',
|
|
'percent_price': 30,
|
|
})
|
|
|
|
special_pricelist = self.env['product.pricelist'].create({
|
|
'name': 'special_pricelist',
|
|
})
|
|
self.env['product.pricelist.item'].create({
|
|
'pricelist_id': special_pricelist.id,
|
|
'base': 'pricelist',
|
|
'base_pricelist_id': base_pricelist.id,
|
|
'compute_price': 'percentage',
|
|
'applied_on': '3_global',
|
|
'percent_price': 10,
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'pricelist_id': base_pricelist.id,
|
|
'available_pricelist_ids': [(6, 0, [base_pricelist.id, special_pricelist.id])],
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'ReceiptScreenDiscountWithPricelistTour', login="pos_user")
|
|
|
|
def test_07_product_combo(self):
|
|
self.env['decimal.precision'].search([('name', '=', 'Product Price')]).digits = 4
|
|
setup_product_combo_items(self)
|
|
combo_product_sofa = self.env["product.template"].create(
|
|
{
|
|
"name": "Combo product Sofa",
|
|
"is_storable": True,
|
|
"available_in_pos": True,
|
|
"list_price": 40,
|
|
}
|
|
)
|
|
sofa_size_attribute = self.env['product.attribute'].create({
|
|
'name': 'Size',
|
|
'display_type': 'radio',
|
|
'create_variant': 'always',
|
|
})
|
|
sofa_color_attribute = self.env['product.attribute'].create({
|
|
'name': 'Color',
|
|
'display_type': 'radio',
|
|
'create_variant': 'always',
|
|
})
|
|
sofa_size_L = self.env['product.attribute.value'].create({
|
|
'name': 'L',
|
|
'attribute_id': sofa_size_attribute.id,
|
|
})
|
|
sofa_size_M = self.env['product.attribute.value'].create({
|
|
'name': 'M',
|
|
'attribute_id': sofa_size_attribute.id,
|
|
})
|
|
sofa_color_red = self.env['product.attribute.value'].create({
|
|
'name': 'red',
|
|
'attribute_id': sofa_color_attribute.id,
|
|
})
|
|
sofa_color_blue = self.env['product.attribute.value'].create({
|
|
'name': 'blue',
|
|
'attribute_id': sofa_color_attribute.id,
|
|
})
|
|
|
|
product_attribute_size = self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': combo_product_sofa.id,
|
|
'attribute_id': sofa_size_attribute.id,
|
|
'value_ids': [Command.set([sofa_size_M.id, sofa_size_L.id])],
|
|
|
|
})
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': combo_product_sofa.id,
|
|
'attribute_id': sofa_color_attribute.id,
|
|
'value_ids': [Command.set([sofa_color_red.id, sofa_color_blue.id])],
|
|
|
|
})
|
|
product_attribute_size.product_template_value_ids[0].price_extra = 50
|
|
product_attribute_size.product_template_value_ids[1].price_extra = 100
|
|
self.sofa_combo = self.env["product.combo"].create(
|
|
{
|
|
"name": "Chairs Combo",
|
|
"combo_item_ids": [
|
|
Command.create({
|
|
"product_id": combo_product_sofa.product_variant_ids[0].id,
|
|
"extra_price": 5,
|
|
}),
|
|
Command.create({
|
|
"product_id": combo_product_sofa.product_variant_ids[1].id,
|
|
"extra_price": 10,
|
|
}),
|
|
],
|
|
},
|
|
)
|
|
self.sofa_combo = self.env["product.product"].create(
|
|
{
|
|
"available_in_pos": True,
|
|
"list_price": 20,
|
|
"name": "Sofa Combo",
|
|
"type": "combo",
|
|
"uom_id": self.env.ref("uom.product_uom_unit").id,
|
|
"combo_ids": [
|
|
Command.set([self.sofa_combo.id]),
|
|
],
|
|
},
|
|
)
|
|
self.office_combo.write({
|
|
'lst_price': 50,
|
|
'barcode': 'SuperCombo',
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('ProductComboPriceTaxIncludedTour')
|
|
order = self.env['pos.order'].search([])
|
|
self.assertEqual(len(order.lines), 4, "There should be 4 order lines - 1 combo parent and 3 combo lines")
|
|
# check that the combo lines are correctly linked to each other
|
|
parent_line_id = self.env['pos.order.line'].search([('product_id.name', '=', 'Office Combo'), ('order_id', '=', order.id)])
|
|
combo_line_ids = self.env['pos.order.line'].search([('product_id.name', '!=', 'Office Combo'), ('order_id', '=', order.id)])
|
|
self.assertEqual(parent_line_id.combo_line_ids, combo_line_ids, "The combo parent should have 3 combo lines")
|
|
self.assertEqual(order.lines[1].price_unit, 10.33)
|
|
self.assertEqual(order.lines[2].price_unit, 18.67)
|
|
self.assertEqual(order.lines[3].price_unit, 30.00)
|
|
# In the future we might want to test also if:
|
|
# - the combo lines are correctly stored in and restored from local storage
|
|
# - the combo lines are correctly shared between the pos configs ( in cross ordering )
|
|
|
|
def test_07_product_combo_max_free_qty(self):
|
|
""" Test the max free quantity of a product combo."""
|
|
setup_product_combo_items(self)
|
|
self.office_combo.combo_ids[0].write({
|
|
'qty_free': 2,
|
|
'qty_max': 2,
|
|
})
|
|
self.office_combo.combo_ids[1].write({
|
|
'qty_free': 2,
|
|
'qty_max': 5,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('ProductComboMaxFreeQtyTour')
|
|
|
|
def test_line_configurators(self):
|
|
setup_product_combo_items(self)
|
|
self.env['product.combo.item'].create({
|
|
'combo_id': self.desks_combo.id,
|
|
'product_id': self.configurable_chair.product_variant_id.id,
|
|
'extra_price': 0,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_line_configurators_product')
|
|
self.start_pos_tour('test_line_configurators_combo')
|
|
|
|
def test_07_pos_barcodes_scan(self):
|
|
barcode_rule = self.env.ref("point_of_sale.barcode_rule_client")
|
|
barcode_rule.pattern = barcode_rule.pattern + "|234"
|
|
# should in theory be changed in the JS code to `|^234`
|
|
# If not, it will fail as it will mistakenly match with the product barcode "0123456789"
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'BarcodeScanningTour', login="pos_user")
|
|
|
|
def test_08_show_tax_excluded(self):
|
|
# define a tax included tax record
|
|
tax = self.env['account.tax'].create({
|
|
'name': 'Tax 10% Included',
|
|
'amount_type': 'percent',
|
|
'amount': 10,
|
|
'price_include_override': 'tax_included',
|
|
})
|
|
|
|
# define a product record with the tax
|
|
self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'list_price': 110,
|
|
'taxes_id': [(6, 0, [tax.id])],
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
# set Tax-Excluded Price
|
|
self.main_pos_config.write({
|
|
'iface_tax_included': 'subtotal'
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'ShowTaxExcludedTour', login="pos_user")
|
|
|
|
def test_chrome_without_cash_move_permission(self):
|
|
self.env.user.write({'group_ids': [
|
|
Command.set(
|
|
[
|
|
self.env.ref('base.group_user').id,
|
|
self.env.ref('point_of_sale.group_pos_user').id,
|
|
]
|
|
)
|
|
]})
|
|
self.main_pos_config.open_ui()
|
|
self.start_pos_tour('chrome_without_cash_move_permission', login="accountman")
|
|
|
|
def test_09_pos_barcodes_scan_product_packaging(self):
|
|
pack_of_10 = self.env['uom.uom'].create({
|
|
'name': 'Pack of 10',
|
|
'relative_factor': 10,
|
|
'relative_uom_id': self.env.ref('uom.product_uom_unit').id,
|
|
'is_pos_groupable': True,
|
|
})
|
|
product = self.env['product.product'].create({
|
|
'name': 'Packaging Product',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
'barcode': '12345601',
|
|
'uom_ids': [Command.link(pack_of_10.id)],
|
|
})
|
|
self.env['product.uom'].create({
|
|
'barcode': '12345610',
|
|
'product_id': product.id,
|
|
'uom_id': pack_of_10.id,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'BarcodeScanningProductPackagingTour', login="pos_user")
|
|
|
|
def test_GS1_pos_barcodes_scan(self):
|
|
barcodes_gs1_nomenclature = self.env.ref("barcodes_gs1_nomenclature.default_gs1_nomenclature")
|
|
default_nomenclature_id = self.env.ref("barcodes.default_barcode_nomenclature")
|
|
self.main_pos_config.company_id.write({
|
|
'nomenclature_id': barcodes_gs1_nomenclature.id
|
|
})
|
|
self.main_pos_config.write({
|
|
'fallback_nomenclature_id': default_nomenclature_id
|
|
})
|
|
self.env['product.product'].create({
|
|
'name': 'Product 1',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
'barcode': '08431673020125',
|
|
})
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'Product 2',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
'barcode': '08431673020126',
|
|
})
|
|
|
|
# 3760171283370 can be parsed with GS1 rules but it's not GS1
|
|
self.env['product.product'].create({
|
|
'name': 'Product 3',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
'barcode': '3760171283370',
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'GS1BarcodeScanningTour', login="pos_user")
|
|
|
|
def test_refund_order_with_fp_tax_included(self):
|
|
# create a fiscal position
|
|
self.fiscal_position = self.env['account.fiscal.position'].create({
|
|
'name': 'No Tax',
|
|
})
|
|
#create a tax of 15% tax included
|
|
self.tax1 = self.env['account.tax'].create({
|
|
'name': 'Tax 1',
|
|
'amount': 15,
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'price_include_override': 'tax_included',
|
|
})
|
|
#create a tax of 0%
|
|
self.tax2 = self.env['account.tax'].create({
|
|
'name': 'Tax 2',
|
|
'amount': 0,
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'price_include_override': 'tax_included',
|
|
'fiscal_position_ids': self.fiscal_position,
|
|
'original_tax_ids': self.tax1,
|
|
})
|
|
|
|
self.product_test = self.env['product.product'].create({
|
|
'name': 'Product Test',
|
|
'is_storable': True,
|
|
'available_in_pos': True,
|
|
'list_price': 100,
|
|
'taxes_id': [(6, 0, self.tax1.ids)],
|
|
})
|
|
|
|
#add the fiscal position to the PoS
|
|
self.main_pos_config.write({
|
|
'fiscal_position_ids': [(4, self.fiscal_position.id)],
|
|
'tax_regime_selection': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'FiscalPositionNoTaxRefund', login="pos_user")
|
|
order = self.env['pos.order'].search([])
|
|
self.assertTrue(order[0].name == order[1].name + " REFUND")
|
|
|
|
def test_customer_display_popup(self):
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'customer_display_shows_qr_popup', login="pos_user")
|
|
|
|
def test_lot_refund(self):
|
|
|
|
self.product1 = self.env['product.product'].create({
|
|
'name': 'Product A',
|
|
'is_storable': True,
|
|
'tracking': 'serial',
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'LotRefundTour', login="pos_user")
|
|
|
|
def test_receipt_tracking_method(self):
|
|
self.product_a = self.env['product.product'].create({
|
|
'name': 'Product A',
|
|
'is_storable': True,
|
|
'tracking': 'lot',
|
|
'available_in_pos': True,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'ReceiptTrackingMethodTour', login="pos_user")
|
|
|
|
def test_printed_receipt_tour(self):
|
|
self.main_pos_config.write({
|
|
'basic_receipt': True,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour("point_of_sale.test_printed_receipt_tour")
|
|
|
|
def test_limited_product_pricelist_loading(self):
|
|
self.env['ir.config_parameter'].sudo().set_param('point_of_sale.limited_product_count', '1')
|
|
|
|
limited_category = self.env['pos.category'].create({
|
|
'name': 'Limited Category',
|
|
})
|
|
product_1 = self.env['product.product'].create({
|
|
'name': 'Test Product 1',
|
|
'list_price': 100,
|
|
'barcode': '0100100',
|
|
'taxes_id': False,
|
|
'pos_categ_ids': [(4, limited_category.id)],
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
color_attribute = self.env['product.attribute'].create({
|
|
'name': 'Color',
|
|
'sequence': 4,
|
|
'value_ids': [(0, 0, {
|
|
'name': 'White',
|
|
'sequence': 1,
|
|
}), (0, 0, {
|
|
'name': 'Red',
|
|
'sequence': 2,
|
|
'default_extra_price': 50,
|
|
})],
|
|
})
|
|
|
|
product_2_template = self.env['product.template'].create({
|
|
'name': 'Test Product 2',
|
|
'list_price': 200,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
'pos_categ_ids': [(4, limited_category.id)],
|
|
'tracking': 'lot',
|
|
'attribute_line_ids': [(0, 0, {
|
|
'attribute_id': color_attribute.id,
|
|
'value_ids': [(6, 0, color_attribute.value_ids.ids)]
|
|
})],
|
|
})
|
|
|
|
# Check that two product variant are created
|
|
self.assertEqual(product_2_template.product_variant_count, 2)
|
|
product_2_template.product_variant_ids[0].write({'barcode': '0100201'})
|
|
product_2_template.product_variant_ids[1].write({'barcode': '0100202'})
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'Test Product 3',
|
|
'list_price': 300,
|
|
'barcode': '0100300',
|
|
'taxes_id': False,
|
|
'pos_categ_ids': [(4, limited_category.id)],
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
pricelist_item = self.env['product.pricelist.item'].create([{
|
|
'applied_on': '3_global',
|
|
'fixed_price': 50,
|
|
}, {
|
|
'applied_on': '1_product',
|
|
'product_tmpl_id': product_2_template.id,
|
|
'fixed_price': 100,
|
|
}, {
|
|
'applied_on': '0_product_variant',
|
|
'product_id': product_1.id,
|
|
'fixed_price': 80,
|
|
'min_quantity': 1,
|
|
}, {
|
|
'applied_on': '0_product_variant',
|
|
'product_id': product_1.id,
|
|
'fixed_price': 70,
|
|
'min_quantity': 2,
|
|
}, {
|
|
'applied_on': '0_product_variant',
|
|
'product_id': product_2_template.product_variant_ids[1].id,
|
|
'fixed_price': 120,
|
|
}])
|
|
self.main_pos_config.write({
|
|
'iface_available_categ_ids': [],
|
|
'limit_categories': True,
|
|
})
|
|
self.main_pos_config.pricelist_id.write({'item_ids': [(6, 0, pricelist_item.ids)]})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'limitedProductPricelistLoading', login="pos_user")
|
|
|
|
def test_restricted_categories_combo_product(self):
|
|
"""
|
|
Ensure combo choices product are always loaded if parent is in allowed categories, even when restricted categories are configured:
|
|
- These combo choices should be visible when configuring the parent combo product but not be visible as product that we can directly sell inside POS
|
|
- These combo choices should appear on the preparation ticket changes
|
|
"""
|
|
pos_restricted_categ = self.env["pos.category"].create({
|
|
"name": "Restricted product",
|
|
})
|
|
pos_other_categ = self.env["pos.category"].create({
|
|
"name": "Other products",
|
|
})
|
|
self.env['pos.printer'].create({
|
|
'name': 'Printer',
|
|
'printer_type': 'epson_epos',
|
|
'epson_printer_ip': '0.0.0.0',
|
|
'product_categories_ids': [Command.set(self.env['pos.category'].search([]).ids)],
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'is_order_printer': True,
|
|
'printer_ids': [Command.set(self.env['pos.printer'].search([]).ids)],
|
|
})
|
|
self.main_pos_config.write({
|
|
"limit_categories": True,
|
|
"iface_available_categ_ids": [(6, 0, [pos_restricted_categ.id])],
|
|
})
|
|
setup_product_combo_items(self)
|
|
self.office_combo.pos_categ_ids = [(6, 0, [pos_restricted_categ.id])]
|
|
self.office_combo.combo_ids = [(6, 0, [self.desks_combo.id])]
|
|
self.desks_combo.combo_item_ids[0].product_id.pos_categ_ids = [(6, 0, [pos_restricted_categ.id])]
|
|
self.desks_combo.combo_item_ids[1].product_id.pos_categ_ids = [(6, 0, [pos_other_categ.id])]
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_restricted_categories_combo_product', login="pos_user")
|
|
|
|
def test_printer_restricts_to_allowed_categories_for_combo(self):
|
|
setup_product_combo_items(self)
|
|
self.printer.write({
|
|
'product_categories_ids': [Command.set(self.env['pos.category'].search([('name', '=', 'Category 2')]).ids)],
|
|
})
|
|
self.office_combo.write({
|
|
'pos_categ_ids': [Command.set(self.env['pos.category'].search([('name', '=', 'Category 1')]).ids)],
|
|
})
|
|
self.main_pos_config.write({
|
|
'is_order_printer': True,
|
|
'printer_ids': [Command.set(self.printer.ids)],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_printer_restricts_to_allowed_categories_for_combo', login="pos_user")
|
|
|
|
def test_printer_not_linked_to_any_combo_category(self):
|
|
setup_product_combo_items(self)
|
|
new_category = self.env['pos.category'].create({
|
|
'name': 'New Category',
|
|
})
|
|
self.wall_shelf.write({
|
|
'pos_categ_ids': [Command.set(new_category.ids)],
|
|
})
|
|
self.printer.write({
|
|
'product_categories_ids': [Command.set(new_category.ids)],
|
|
})
|
|
self.main_pos_config.write({
|
|
'is_order_printer': True,
|
|
'printer_ids': [Command.set(self.printer.ids)],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_printer_not_linked_to_any_combo_category', login="pos_user")
|
|
|
|
def test_multi_product_options(self):
|
|
self.pos_user.write({
|
|
'group_ids': [
|
|
(4, self.env.ref('stock.group_stock_manager').id),
|
|
]
|
|
})
|
|
product_a = self.env['product.product'].create({
|
|
'name': 'Product A',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
})
|
|
|
|
chair_multi_attribute = self.env['product.attribute'].create({
|
|
'name': 'Multi',
|
|
'display_type': 'multi',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
chair_multi_value_1 = self.env['product.attribute.value'].create({
|
|
'name': 'Value 1',
|
|
'attribute_id': chair_multi_attribute.id,
|
|
})
|
|
chair_multi_value_2 = self.env['product.attribute.value'].create({
|
|
'name': 'Value 2',
|
|
'attribute_id': chair_multi_attribute.id,
|
|
})
|
|
self.chair_multi_line = self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product_a.product_tmpl_id.id,
|
|
'attribute_id': chair_multi_attribute.id,
|
|
'value_ids': [(6, 0, [chair_multi_value_1.id, chair_multi_value_2.id])]
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'MultiProductOptionsTour', login="pos_user")
|
|
|
|
def test_translate_product_name(self):
|
|
self.env['res.lang']._activate_lang('fr_FR')
|
|
self.pos_user.write({'lang': 'fr_FR'})
|
|
|
|
product = self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
})
|
|
product.update_field_translations('name', {'fr_FR': 'Testez le produit'})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'TranslateProductNameTour', login="pos_user")
|
|
|
|
def test_properly_display_price(self):
|
|
"""Make sure that when the decimal separator is a comma, the shown orderline price is correct.
|
|
"""
|
|
lang = self.env['res.lang'].search([('code', '=', self.pos_user.lang)])
|
|
lang.write({'thousands_sep': '.', 'decimal_point': ','})
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'list_price': 1_453.53,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, "DecimalCommaOrderlinePrice", login="pos_user")
|
|
|
|
def test_res_partner_scan_barcode(self):
|
|
# default Customer Barcodes pattern is '042'
|
|
self.env['res.partner'].create({
|
|
'name': 'John Doe',
|
|
'barcode': '0421234567890',
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'BarcodeScanPartnerTour', login="pos_user")
|
|
|
|
def test_allow_order_modification_after_validation_error(self):
|
|
"""
|
|
User error as a result of validation should block the order.
|
|
Taking action by order modification should be allowed.
|
|
"""
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'list_price': 10.00,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
def sync_from_ui_patch(*_args, **_kwargs):
|
|
raise UserError('Test Error')
|
|
|
|
with patch.object(self.env.registry.models['pos.order'], "sync_from_ui", sync_from_ui_patch):
|
|
# If there is problem in the tour, remove the log catcher to debug.
|
|
with self.assertLogs(level="WARNING") as log_catcher:
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'OrderModificationAfterValidationError', login="pos_user")
|
|
|
|
warning_outputs = [o for o in log_catcher.output if 'WARNING' in o]
|
|
self.assertEqual(len(warning_outputs), 1, "Exactly one warning should be logged")
|
|
|
|
def test_customer_display(self):
|
|
self.start_tour(f"/pos_customer_display/{self.main_pos_config.id}/{self.main_pos_config.access_token}", 'CustomerDisplayTour', login="pos_user")
|
|
|
|
def test_customer_display_scroll(self):
|
|
self.start_tour(f"/pos_customer_display/{self.main_pos_config.id}/{self.main_pos_config.access_token}", 'CustomerDisplayTourScroll', login="pos_user")
|
|
|
|
def test_customer_display_with_qr(self):
|
|
self.start_tour(f"/pos_customer_display/{self.main_pos_config.id}/{self.main_pos_config.access_token}", 'CustomerDisplayTourWithQr', login="pos_user")
|
|
|
|
def test_combo_refund_different_qty(self):
|
|
setup_product_combo_items(self)
|
|
self.desks_combo.write({
|
|
'qty_free': 2,
|
|
'qty_max': 2,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_combo_refund_different_qty')
|
|
|
|
def test_order_refund_flow(self):
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_order_refund_flow')
|
|
|
|
def test_refund_few_quantities(self):
|
|
""" Test to check that refund works with quantities of less than 0.5 """
|
|
self.env['product.product'].create({
|
|
'name': 'Sugar',
|
|
'list_price': 3,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
'uom_id': self.env.ref('uom.product_uom_kgm').id,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'RefundFewQuantities', login="pos_user")
|
|
|
|
def test_refund_multiple_products_amounts_compliance(self):
|
|
test_product = self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'list_price': 10.00,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
current_session = self.main_pos_config.current_session_id
|
|
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'refund_multiple_products_amounts_compliance', login="pos_user")
|
|
|
|
refund_order = current_session.order_ids.filtered(lambda order: order.is_refund)
|
|
self.assertEqual(refund_order.lines[0].price_subtotal, 2 * test_product.list_price)
|
|
total_cash_payment = sum(current_session.mapped('order_ids.payment_ids').filtered(
|
|
lambda payment: payment.payment_method_id.type == 'cash').mapped('amount')
|
|
)
|
|
current_session.post_closing_cash_details(total_cash_payment)
|
|
current_session.close_session_from_ui()
|
|
self.assertEqual(current_session.state, 'closed')
|
|
report_refund_order, report_order = self.env['report.pos.order'].sudo().search([('order_id', 'in', current_session.order_ids.ids)])
|
|
self.assertEqual(report_order.margin, 20.0)
|
|
self.assertEqual(report_refund_order.margin, -20.0)
|
|
self.assertEqual(report_order.price_total, 20.0)
|
|
self.assertEqual(report_refund_order.price_total, -20.0)
|
|
|
|
def test_product_combo_price(self):
|
|
""" Check that the combo has the expected price """
|
|
self.desk_organizer.product_variant_id.write({"lst_price": 7})
|
|
self.desk_pad.product_variant_id.write({"lst_price": 2.5})
|
|
self.whiteboard_pen.product_variant_id.write({"lst_price": 1.5})
|
|
|
|
combos = self.env["product.combo"].create([
|
|
{
|
|
"name": product.name,
|
|
"combo_item_ids": [
|
|
Command.create({
|
|
"product_id": product.id, "extra_price": 0
|
|
})
|
|
]
|
|
}
|
|
for product in (self.desk_organizer.product_variant_id, self.desk_pad.product_variant_id, self.whiteboard_pen.product_variant_id)
|
|
])
|
|
|
|
self.env["product.product"].create(
|
|
{
|
|
"available_in_pos": True,
|
|
"list_price": 7,
|
|
"standard_price": 10,
|
|
"name": "Desk Combo",
|
|
"type": "combo",
|
|
"taxes_id": False,
|
|
"combo_ids": [
|
|
(6, 0, [combo.id for combo in combos])
|
|
],
|
|
}
|
|
)
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'ProductComboPriceCheckTour', login="pos_user")
|
|
order = self.env['pos.order'].search([], limit=1)
|
|
self.assertEqual(order.lines.filtered(lambda l: l.product_id.type == 'combo').margin, 0)
|
|
self.assertEqual(order.lines.filtered(lambda l: l.product_id.type == 'combo').margin_percent, 0)
|
|
|
|
def test_customer_display_as_public(self):
|
|
self.main_pos_config.customer_display_bg_img = b'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAIAAACQd1PeAAAADElEQVR4nGNgYGAAAAAEAAH2FzhVAAAAAElFTkSuQmCC'
|
|
response = self.url_open(f"/web/image/pos.config/{self.main_pos_config.id}/customer_display_bg_img")
|
|
self.assertEqual(response.status_code, 200)
|
|
self.assertTrue('Shop.png' in response.headers['Content-Disposition'])
|
|
|
|
def test_customer_all_fields_displayed(self):
|
|
"""
|
|
Verify that all the field of a partner can be displayed in the partner list.
|
|
Also verify that all these fields can be searched.
|
|
"""
|
|
self.env["res.partner"].create({
|
|
"name": "John Doe",
|
|
"street": "1 street of astreet",
|
|
"city": "Acity",
|
|
"state_id": self.env.ref("base.state_us_30").id, # Ohio
|
|
"country_id": self.env.ref("base.us").id,
|
|
"zip": "26432685463",
|
|
"phone": "9898989899",
|
|
"email": "john@doe.com"
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('PosCustomerAllFieldsDisplayed')
|
|
|
|
def test_product_combo_change_fp(self):
|
|
"""
|
|
Verify than when the fiscal position is changed,
|
|
the price of the combo doesn't change and taxes are well taken into account
|
|
"""
|
|
fiscal_position = self.env['account.fiscal.position'].create({
|
|
'name': 'test fp',
|
|
})
|
|
|
|
tax_1 = self.env['account.tax'].create({
|
|
'name': 'Tax 10%',
|
|
'amount': 10,
|
|
'price_include_override': 'tax_included',
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
})
|
|
|
|
self.env['account.tax'].create({
|
|
'name': 'Tax 5%',
|
|
'amount': 5,
|
|
'price_include_override': 'tax_included',
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'fiscal_position_ids': [Command.link(fiscal_position.id)],
|
|
'original_tax_ids': [Command.link(tax_1.id)],
|
|
})
|
|
|
|
setup_product_combo_items(self)
|
|
self.office_combo.write({'list_price': 50, 'taxes_id': [(6, 0, [tax_1.id])]})
|
|
for combo in self.office_combo.combo_ids: # Set the tax to all the products of the combo
|
|
for item in combo.combo_item_ids:
|
|
item.product_id.taxes_id = [(6, 0, [tax_1.id])]
|
|
|
|
self.main_pos_config.write({
|
|
'tax_regime_selection': True,
|
|
'fiscal_position_ids': [(6, 0, [fiscal_position.id])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'ProductComboChangeFP', login="pos_user")
|
|
|
|
def test_product_combo_change_pricelist(self):
|
|
"""
|
|
Verify than when we change the pricelist, the combo price is updated
|
|
"""
|
|
setup_product_combo_items(self)
|
|
|
|
sale_10_pl = self.env['product.pricelist'].create({
|
|
'name': 'sale 10%',
|
|
})
|
|
self.env['product.pricelist.item'].create({
|
|
'pricelist_id': sale_10_pl.id,
|
|
'compute_price': 'percentage',
|
|
'applied_on': '3_global',
|
|
'percent_price': 10,
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'available_pricelist_ids': [(4, sale_10_pl.id)],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'ProductComboChangePricelist', login="pos_user")
|
|
|
|
def test_product_combo_discount(self):
|
|
"""
|
|
Verify that the combo product applies the correct discount and updates prices accordingly
|
|
"""
|
|
setup_product_combo_items(self)
|
|
self.office_combo.write({'list_price': 100})
|
|
self.office_combo.combo_ids.combo_item_ids.product_id.taxes_id = False
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(
|
|
f"/pos/ui?config_id={self.main_pos_config.id}",
|
|
'ProductComboDiscountTour',
|
|
login="pos_user",
|
|
)
|
|
|
|
def test_combo_item_image_display(self):
|
|
""" when `show_product_images` is enabled, verify combo item product images should appear in the POS UI. When disabled, the UI should not display
|
|
the product image for combo items.
|
|
"""
|
|
|
|
setup_product_combo_items(self)
|
|
image = _create_image(color="orange")
|
|
|
|
for combo in self.office_combo.combo_ids:
|
|
for item in combo.combo_item_ids:
|
|
item.product_id.image_1920 = image
|
|
|
|
# Case 1: Images ON → images must be visible in combo items
|
|
self.main_pos_config.show_product_images = True
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'test_combo_item_image_display', login="pos_user")
|
|
|
|
# Case 2: Images OFF → product images should not be shown
|
|
self.main_pos_config.show_product_images = False
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'test_combo_item_image_not_display', login="pos_user")
|
|
|
|
def test_product_categories_order(self):
|
|
""" Verify that the order of categories doesnt change in the frontend """
|
|
self.env['pos.category'].search([]).write({'sequence': 100})
|
|
aaa_catg = self.env['pos.category'].create({
|
|
'name': 'AAA',
|
|
'parent_id': False,
|
|
'sequence': 1,
|
|
})
|
|
aac_catg = self.env['pos.category'].create({
|
|
'name': 'AAC',
|
|
'parent_id': False,
|
|
'sequence': 3,
|
|
})
|
|
parentA = self.env['pos.category'].create({
|
|
'name': 'AAB',
|
|
'parent_id': False,
|
|
'sequence': 2,
|
|
})
|
|
parentB = self.env['pos.category'].create({
|
|
'name': 'AAX',
|
|
'parent_id': parentA.id,
|
|
})
|
|
aay_catg = self.env['pos.category'].create({
|
|
'name': 'AAY',
|
|
'parent_id': parentB.id,
|
|
})
|
|
# Add a product that belongs to both parent and child categories.
|
|
# It's presence is checked during the tour to make sure app doesn't crash.
|
|
self.env['product.product'].create({
|
|
'name': 'Product in AAB and AAX',
|
|
'pos_categ_ids': [(6, 0, [parentA.id, parentB.id])],
|
|
'available_in_pos': True,
|
|
})
|
|
self.env['product.product'].create([
|
|
{
|
|
'name': 'Product in AAA Catg',
|
|
'pos_categ_ids': [(6, 0, [aaa_catg.id])],
|
|
'available_in_pos': True,
|
|
},
|
|
{
|
|
'name': 'Product in AAC Catg',
|
|
'pos_categ_ids': [(6, 0, [aac_catg.id])],
|
|
'available_in_pos': True,
|
|
},
|
|
{
|
|
'name': 'Product in AAY Catg',
|
|
'pos_categ_ids': [(6, 0, [aay_catg.id])],
|
|
'available_in_pos': True,
|
|
},
|
|
])
|
|
self.main_pos_config.with_user(self.pos_admin).open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'PosCategoriesOrder', login="pos_admin")
|
|
|
|
def test_product_with_dynamic_attributes(self):
|
|
dynamic_attribute = self.env['product.attribute'].create({
|
|
'name': 'Dynamic Attribute',
|
|
'create_variant': 'dynamic',
|
|
})
|
|
value_1 = self.env['product.attribute.value'].create({
|
|
'name': 'Test 1',
|
|
'attribute_id': dynamic_attribute.id,
|
|
})
|
|
value_2 = self.env['product.attribute.value'].create({
|
|
'name': 'Test 2',
|
|
'default_extra_price': 10,
|
|
'attribute_id': dynamic_attribute.id,
|
|
})
|
|
product_template = self.env['product.template'].create({
|
|
'name': 'Dynamic Product',
|
|
'uom_id': self.env.ref('uom.product_uom_unit').id,
|
|
'is_storable': True,
|
|
'available_in_pos': True,
|
|
})
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product_template.id,
|
|
'attribute_id': dynamic_attribute.id,
|
|
'value_ids': [Command.set([value_1.id, value_2.id])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_admin).open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'PosProductWithDynamicAttributes', login="pos_admin")
|
|
|
|
def test_autofill_cash_count(self):
|
|
"""Make sure that when the decimal separator is a comma, the shown orderline price is correct.
|
|
"""
|
|
lang = self.env['res.lang'].search([('code', '=', self.pos_user.lang)])
|
|
lang.write({'thousands_sep': '.', 'decimal_point': ','})
|
|
self.env["product.product"].create(
|
|
{
|
|
"available_in_pos": True,
|
|
"list_price": 123456,
|
|
"name": "Test Expensive",
|
|
"taxes_id": False
|
|
}
|
|
)
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, "AutofillCashCount", login="pos_user")
|
|
|
|
def test_product_search_2(self):
|
|
self.env['product.product'].create({
|
|
'name': 'Test chair 1',
|
|
'available_in_pos': True,
|
|
})
|
|
self.env['product.product'].create({
|
|
'name': 'Test CHAIR 2',
|
|
'available_in_pos': True,
|
|
})
|
|
self.env['product.product'].create({
|
|
'name': 'Test sofa',
|
|
'available_in_pos': True,
|
|
"default_code": "CHAIR_01",
|
|
})
|
|
self.env['product.product'].create({
|
|
'name': 'clémentine',
|
|
'available_in_pos': True,
|
|
})
|
|
self.main_pos_config.open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'SearchProducts', login="pos_user")
|
|
|
|
def test_lot(self):
|
|
self.product1 = self.env['product.product'].create({
|
|
'name': 'Product A',
|
|
'is_storable': True,
|
|
'tracking': 'serial',
|
|
'available_in_pos': True,
|
|
})
|
|
product2 = self.env['product.product'].create({
|
|
'name': 'Product B',
|
|
'is_storable': True,
|
|
'tracking': 'lot',
|
|
'available_in_pos': True,
|
|
})
|
|
self.env['stock.quant'].with_context(inventory_mode=True).create({
|
|
'product_id': product2.id,
|
|
'inventory_quantity': 1,
|
|
'location_id': self.env.user._get_default_warehouse_id().lot_stock_id.id,
|
|
'lot_id': self.env['stock.lot'].create({'name': '1001', 'product_id': product2.id}).id,
|
|
}).sudo().action_apply_inventory()
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'LotTour', login="pos_user")
|
|
two_last_orders = self.env['pos.order'].search([], order='id desc', limit=2)
|
|
order_lot_id = [lot_id.lot_name for lot_id in two_last_orders[1].lines.pack_lot_ids]
|
|
refund_lot_id = [lot_id.lot_name for lot_id in two_last_orders[0].lines.pack_lot_ids]
|
|
self.assertEqual(order_lot_id, refund_lot_id, "In the refund we should find the same lot as in the original order")
|
|
self.assertEqual(two_last_orders[0].state, 'paid')
|
|
self.assertEqual(two_last_orders[1].state, 'paid')
|
|
self.main_pos_config.current_session_id.order_ids.filtered(
|
|
lambda o: o.state != 'paid').state = 'cancel'
|
|
|
|
self.main_pos_config.current_session_id.action_pos_session_closing_control()
|
|
self.assertEqual(
|
|
two_last_orders[0].picking_ids.move_line_ids.owner_id.id,
|
|
two_last_orders[1].picking_ids.move_line_ids.owner_id.id,
|
|
"The owner of the refund is not the same as the owner of the original order")
|
|
|
|
def test_only_existing_lots(self):
|
|
product = self.env['product.product'].create({
|
|
'name': 'Product with existing lots',
|
|
'is_storable': True,
|
|
'tracking': 'lot',
|
|
'available_in_pos': True,
|
|
})
|
|
self.env['stock.quant'].with_context(inventory_mode=True).create([{
|
|
'product_id': product.id,
|
|
'inventory_quantity': 1,
|
|
'location_id': self.env.user._get_default_warehouse_id().lot_stock_id.id,
|
|
'lot_id': self.env['stock.lot'].create({'name': '1001', 'product_id': product.id}).id,
|
|
}, {
|
|
'product_id': product.id,
|
|
'inventory_quantity': 1,
|
|
'location_id': self.env.user._get_default_warehouse_id().lot_stock_id.id,
|
|
'lot_id': self.env['stock.lot'].create({'name': '1002', 'product_id': product.id}).id,
|
|
}]).sudo().action_apply_inventory()
|
|
|
|
self.main_pos_config.picking_type_id.write({
|
|
"use_create_lots": False,
|
|
"use_existing_lots": True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_only_existing_lots', login="pos_user")
|
|
|
|
def test_order_with_existing_serial(self):
|
|
product = self.env['product.product'].create({
|
|
'name': 'Serial Product',
|
|
'is_storable': True,
|
|
'tracking': 'serial',
|
|
'available_in_pos': True,
|
|
})
|
|
for sn in ["SN1", "SN2"]:
|
|
self.env['stock.quant'].create({
|
|
'product_id': product.id,
|
|
'inventory_quantity': 1,
|
|
'location_id': self.env.user._get_default_warehouse_id().lot_stock_id.id,
|
|
'lot_id': self.env['stock.lot'].create({'name': sn, 'product_id': product.id}).id,
|
|
}).sudo().action_apply_inventory()
|
|
self.env['stock.picking.type'].search([('name', '=', 'PoS Orders')]).use_create_lots = False
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour("test_order_with_existing_serial")
|
|
|
|
def test_product_search(self):
|
|
"""Verify that the product search works correctly"""
|
|
product_with_variant = self.env['product.template'].create({
|
|
'name': 'Product with Variant',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
'barcode': '1234567',
|
|
})
|
|
|
|
color_attribute = self.env['product.attribute'].create({
|
|
'name': 'Color always',
|
|
'create_variant': 'always',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Red',
|
|
'sequence': 1,
|
|
}), (0, 0, {
|
|
'name': 'Blue',
|
|
'sequence': 2,
|
|
})],
|
|
})
|
|
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product_with_variant.id,
|
|
'attribute_id': color_attribute.id,
|
|
'value_ids': [(6, 0, color_attribute.value_ids.ids)]
|
|
})
|
|
product_with_variant.product_variant_ids[0].write({
|
|
"barcode": "variant_barcode_1",
|
|
"default_code": "VARIANT_1"
|
|
})
|
|
product_with_variant.product_variant_ids[1].write({
|
|
"barcode": "variant_barcode_2",
|
|
"default_code": "VARIANT_2"
|
|
})
|
|
|
|
self.env['product.product'].create([
|
|
{
|
|
'name': 'Test Product 1',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
'barcode': '1234567890123',
|
|
'default_code': 'TESTPROD1',
|
|
},
|
|
{
|
|
'name': 'Test Product 2',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
'barcode': '1234567890124',
|
|
'default_code': 'TESTPROD2',
|
|
},
|
|
{
|
|
'name': 'Apple',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
},
|
|
{
|
|
'name': 'galaxy',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
},
|
|
{
|
|
'name': '1234567890123',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
},
|
|
])
|
|
|
|
att_color = self.env['product.attribute'].create({'name': 'Color', 'sequence': 1})
|
|
|
|
att_color_values = self.env['product.attribute.value'].create([
|
|
{'name': 'galaxy variant', 'attribute_id': att_color.id, 'sequence': 1},
|
|
{'name': 'blue', 'attribute_id': att_color.id, 'sequence': 2},
|
|
])
|
|
|
|
self.env['product.template'].create({
|
|
'name': 'Test Product variant',
|
|
'attribute_line_ids': [
|
|
Command.create({
|
|
'attribute_id': att_color.id,
|
|
'value_ids': [Command.set(att_color_values.mapped('id'))],
|
|
}),
|
|
],
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'ProductSearchTour', login="pos_user")
|
|
|
|
def test_customer_popup(self):
|
|
"""Verify that the customer popup search & inifnite scroll work properly"""
|
|
self.env["res.partner"].create([{"name": "Z partner to search"}, {"name": "Z partner to scroll"}])
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'CustomerPopupTour', login="pos_user")
|
|
|
|
def test_pricelist_multi_items_different_qty_thresholds(self):
|
|
""" Having multiple pricelist items for the same product tmpl with ascending `min_quantity`
|
|
values, prefer the "latest available"- that is, the one with greater `min_quantity`.
|
|
"""
|
|
product = self.env['product.product'].create({
|
|
'name': 'tpmcapi product',
|
|
'list_price': 1.0,
|
|
'available_in_pos': True,
|
|
'taxes_id': False,
|
|
})
|
|
self.main_pos_config.pricelist_id.write({
|
|
'item_ids': [Command.create({
|
|
'display_applied_on': '1_product',
|
|
'product_tmpl_id': product.product_tmpl_id.id,
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 10.0,
|
|
'min_quantity': 3,
|
|
}), Command.create({
|
|
'display_applied_on': '1_product',
|
|
'product_tmpl_id': product.product_tmpl_id.id,
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 20.0,
|
|
'min_quantity': 2,
|
|
})],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(
|
|
f'/pos/ui/{self.main_pos_config.id}',
|
|
'test_pricelist_multi_items_different_qty_thresholds',
|
|
login='pos_user'
|
|
)
|
|
|
|
def test_tracking_number_closing_session(self):
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'test_tracking_number_closing_session', login="pos_user")
|
|
|
|
# Change should be given in cash
|
|
cash_payment_method = self.main_pos_config.payment_method_ids.filtered(lambda p: p.is_cash_count)
|
|
last_order = self.main_pos_config.current_session_id.order_ids[-1]
|
|
self.assertRecordValues(last_order.payment_ids.sorted(), [
|
|
{'amount': -18.02, 'payment_method_id': cash_payment_method.id, 'is_change': True},
|
|
{'amount': 20.0, 'payment_method_id': self.bank_payment_method.id, 'is_change': False},
|
|
])
|
|
|
|
# References should not have gaps
|
|
references = self.env['pos.order'].search([], order="pos_reference").mapped("pos_reference")
|
|
for i in range(len(references) - 1):
|
|
self.assertEqual(int(references[i + 1].split('-')[-1]), int(references[i].split('-')[-1]) + 1, "There is a gap in the pos references")
|
|
|
|
def test_reload_page_before_payment_with_customer_account(self):
|
|
self.customer_account_payment_method = self.env['pos.payment.method'].create({
|
|
'name': 'Customer Account',
|
|
'split_transactions': True,
|
|
})
|
|
self.main_pos_config.write({'payment_method_ids': [(6, 0, self.customer_account_payment_method.ids)]})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(
|
|
f'/pos/ui/{self.main_pos_config.id}',
|
|
'test_reload_page_before_payment_with_customer_account',
|
|
login='pos_user'
|
|
)
|
|
|
|
def test_product_card_qty_precision(self):
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'ProductCardUoMPrecision', login="pos_user")
|
|
|
|
@freeze_time("2025-06-15 11:09")
|
|
def test_cash_in_out(self):
|
|
self.pos_user.write({
|
|
'group_ids': [
|
|
(4, self.env.ref('account.group_account_basic').id),
|
|
]
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui/{self.main_pos_config.id}", 'test_cash_in_out', login="pos_user")
|
|
|
|
self.assertEqual(len(self.main_pos_config.current_session_id.statement_line_ids), 1, "There should be one cash in/out statement line")
|
|
self.assertEqual(self.main_pos_config.current_session_id.statement_line_ids[0].amount, -5, "The cash in/out amount should be -5")
|
|
|
|
def test_reuse_empty_floating_order(self):
|
|
""" Verify that after a payment, POS should reuse an existing empty floating order if available, instead of always creating new ones """
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'test_reuse_empty_floating_order', login="pos_user")
|
|
|
|
def test_add_multiple_serials_at_once(self):
|
|
self.product_a = self.env['product.product'].create({
|
|
'name': 'Product A',
|
|
'is_storable': True,
|
|
'tracking': 'serial',
|
|
'available_in_pos': True,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, "AddMultipleSerialsAtOnce", login="pos_user")
|
|
|
|
def test_order_and_invoice_amounts(self):
|
|
payment_term = self.env['account.payment.term'].create({
|
|
'name': "early_payment_term",
|
|
'discount_percentage': 10,
|
|
'discount_days': 10,
|
|
'early_discount': True,
|
|
'early_pay_discount_computation': 'mixed',
|
|
'line_ids': [Command.create({
|
|
'value': 'percent',
|
|
'nb_days': 0,
|
|
'value_amount': 100,
|
|
})]
|
|
})
|
|
self.partner_test_1.property_payment_term_id = payment_term.id
|
|
|
|
tax = self.env['account.tax'].create({
|
|
'name': 'Tax 10%',
|
|
'amount': 10,
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
})
|
|
self.env['product.product'].create({
|
|
'name': 'Product Test',
|
|
'available_in_pos': True,
|
|
'list_price': 1000,
|
|
'taxes_id': [(6, 0, [tax.id])],
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'PaymentScreenInvoiceOrder', login="pos_user")
|
|
|
|
order = self.env['pos.order'].search([('partner_id', '=', self.partner_test_1.id)], limit=1)
|
|
self.assertTrue(order)
|
|
|
|
self.assertEqual(order.partner_id, self.partner_test_1)
|
|
|
|
invoice = self.env['account.move'].search([('invoice_origin', '=', order.pos_reference)], limit=1)
|
|
self.assertTrue(invoice)
|
|
self.assertFalse(invoice.invoice_payment_term_id)
|
|
|
|
self.assertAlmostEqual(order.amount_total, invoice.amount_total, places=2, msg="Order and Invoice amounts do not match.")
|
|
|
|
def test_pricelist_parent_category_rule(self):
|
|
parent_category = self.env['product.category'].create({
|
|
'name': 'Parent Category',
|
|
})
|
|
child_category = self.env['product.category'].create({
|
|
'name': 'Child Category',
|
|
'parent_id': parent_category.id,
|
|
})
|
|
self.env['product.product'].create({
|
|
'name': 'Product with child category',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
'categ_id': child_category.id,
|
|
})
|
|
|
|
pricelist = self.env['product.pricelist'].create({
|
|
'name': 'Test pricelist on category',
|
|
'item_ids': [(0, 0, {
|
|
'compute_price': 'fixed',
|
|
'fixed_price': 50,
|
|
'applied_on': '2_product_category',
|
|
'categ_id': parent_category.id,
|
|
})],
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'pricelist_id': pricelist.id,
|
|
'available_pricelist_ids': [(6, 0, [pricelist.id])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'test_pricelist_parent_category_rule', login="pos_user")
|
|
|
|
def test_product_create_update_from_frontend(self):
|
|
''' This test verifies product creation and updates product details from the POS frontend. '''
|
|
self.pos_admin.write({
|
|
'group_ids': [Command.link(self.env.ref('base.group_system').id)],
|
|
})
|
|
self.env['pos.category'].search([('id', '!=', self.pos_cat_chair_test.id)]).write({'sequence': 100})
|
|
self.pos_cat_chair_test.write({'sequence': 1})
|
|
self.main_pos_config.with_user(self.pos_admin).open_ui()
|
|
self.start_tour('/pos/ui/%d' % self.main_pos_config.id, 'test_product_create_update_from_frontend', login='pos_admin')
|
|
|
|
# In the frontend, a product was created during the tour with the following details:
|
|
# - Product name: Test Frontend Product
|
|
# - Barcode: 710535977349
|
|
# - List price: 20.0
|
|
|
|
# Ensure that the original product created in the frontend ('Test Frontend Product') has been edited to ('Test Frontend Product Edited').
|
|
frontend_created_product = self.env['product.product'].search_count([('name', '=', 'Test Frontend Product')])
|
|
frontend_created_product_edited = self.env['product.product'].search([('name', '=', 'Test Frontend Product Edited')])
|
|
|
|
self.assertEqual(frontend_created_product, 0)
|
|
self.assertEqual(frontend_created_product_edited.name, 'Test Frontend Product Edited')
|
|
self.assertEqual(frontend_created_product_edited.barcode, '710535977348')
|
|
self.assertEqual(frontend_created_product_edited.list_price, 50.0)
|
|
|
|
def test_one_attribute_value_scan_barcode(self):
|
|
product = self.env['product.template'].create({
|
|
'name': 'Product Test',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
'barcode': '1234567',
|
|
})
|
|
|
|
size_attribute = self.env['product.attribute'].create({
|
|
'name': 'Size never',
|
|
'create_variant': 'no_variant',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Large',
|
|
})],
|
|
})
|
|
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': size_attribute.id,
|
|
'value_ids': [(6, 0, size_attribute.value_ids.ids)]
|
|
})
|
|
|
|
color_attribute = self.env['product.attribute'].create({
|
|
'name': 'Color always',
|
|
'create_variant': 'always',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Red',
|
|
'sequence': 1,
|
|
}), (0, 0, {
|
|
'name': 'Blue',
|
|
'sequence': 2,
|
|
})],
|
|
})
|
|
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': color_attribute.id,
|
|
'value_ids': [(6, 0, color_attribute.value_ids.ids)]
|
|
})
|
|
|
|
product.product_variant_ids[0].barcode = '1234567'
|
|
product.product_variant_ids[1].barcode = '1234568'
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_one_attribute_value_scan_barcode', login="pos_user")
|
|
|
|
def test_fiscal_position_tax_group_labels(self):
|
|
fiscal_position = self.env['account.fiscal.position'].create({
|
|
'name': 'Fiscal Position Test',
|
|
})
|
|
tax_1 = self.env['account.tax'].create({
|
|
'name': 'Tax 15%',
|
|
'amount': 15,
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'tax_group_id': self.env['account.tax.group'].create({
|
|
'name': 'Tax Group 15%',
|
|
'company_id': self.env.company.id,
|
|
'pos_receipt_label': 'Tax Group 1',
|
|
}).id,
|
|
})
|
|
|
|
tax_2 = self.env['account.tax'].create({
|
|
'name': 'Tax 5%',
|
|
'amount': 5,
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'tax_group_id': self.env['account.tax.group'].create({
|
|
'name': 'Tax Group 5%',
|
|
'company_id': self.env.company.id,
|
|
'pos_receipt_label': 'Tax Group 2',
|
|
}).id,
|
|
'fiscal_position_ids': [Command.link(fiscal_position.id)],
|
|
'original_tax_ids': [Command.link(tax_1.id)],
|
|
})
|
|
|
|
self.product = self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'taxes_id': [(6, 0, [tax_1.id])],
|
|
'list_price': 100,
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
'tax_regime_selection': True,
|
|
'fiscal_position_ids': [(6, 0, [fiscal_position.id])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_fiscal_position_tax_group_labels')
|
|
orders = self.main_pos_config.current_session_id.order_ids
|
|
|
|
self.assertEqual(orders[0].fiscal_position_id.id, fiscal_position.id)
|
|
self.assertEqual(orders[0].lines.tax_ids_after_fiscal_position.id, tax_2.id)
|
|
self.assertEqual(orders[0].amount_total, 105)
|
|
self.assertFalse(orders[1].fiscal_position_id)
|
|
self.assertEqual(orders[1].lines.tax_ids_after_fiscal_position.id, tax_1.id)
|
|
self.assertEqual(orders[1].amount_total, 115)
|
|
|
|
def test_draft_orders_not_syncing(self):
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_draft_orders_not_syncing', login="pos_user")
|
|
n_draft_order = self.env['pos.order'].search_count([('state', '=', 'draft')], limit=1)
|
|
self.assertEqual(n_draft_order, 0, 'There should be no draft orders created')
|
|
|
|
def test_product_long_press(self):
|
|
""" Test the long press on product to open the product info """
|
|
archive_products(self.env)
|
|
self.main_pos_config.company_id.country_id.vat_label = 'Should stay VAT even after editing vat_label'
|
|
group_tax = self.env['account.tax'].create({
|
|
'name': 'Parent Tax',
|
|
'amount_type': 'group',
|
|
'children_tax_ids': [(0, 0, {
|
|
'name': 'Child Tax 1',
|
|
'amount': 10,
|
|
}), (0, 0, {
|
|
'name': 'Child Tax 2',
|
|
'amount': 5,
|
|
})],
|
|
})
|
|
self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'list_price': 100,
|
|
'taxes_id': [(6, 0, [group_tax.id])],
|
|
'available_in_pos': True,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'test_product_long_press', login="pos_user")
|
|
|
|
def test_zero_decimal_places_currency(self):
|
|
zero_decimal_currency = self.env['res.currency'].create({
|
|
'name': 'ZeroDecimalCurrency',
|
|
'symbol': 'ZDC',
|
|
'rounding': 1.0,
|
|
'decimal_places': 0,
|
|
})
|
|
|
|
self.env.user.company_id.currency_id = zero_decimal_currency
|
|
self.main_pos_config.available_pricelist_ids.write({'currency_id': zero_decimal_currency.id})
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'Test Product',
|
|
'list_price': 100,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_zero_decimal_places_currency', login="pos_user")
|
|
|
|
def test_barcode_search_attributes_preset(self):
|
|
product = self.env['product.template'].create({
|
|
'name': 'Product with Attributes',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
})
|
|
|
|
# Product template to force UI reset (acts as a delay)
|
|
self.env['product.template'].create({
|
|
'name': 'Product without Attributes',
|
|
'available_in_pos': True,
|
|
'list_price': 20,
|
|
'taxes_id': False,
|
|
'barcode': '987654321',
|
|
})
|
|
|
|
attribute_1, attribute_2, attribute_3, attribute_4 = self.env['product.attribute'].create([{
|
|
'name': 'Attribute 1',
|
|
'create_variant': 'always',
|
|
'display_type': 'radio',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Value 1',
|
|
}), (0, 0, {
|
|
'name': 'Value 2',
|
|
})],
|
|
}, {
|
|
'name': 'Attribute 2',
|
|
'create_variant': 'always',
|
|
'display_type': 'pills',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Value 3',
|
|
}), (0, 0, {
|
|
'name': 'Value 4',
|
|
})],
|
|
}, {
|
|
'name': 'Attribute 3',
|
|
'create_variant': 'always',
|
|
'display_type': 'select',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Value 5',
|
|
}), (0, 0, {
|
|
'name': 'Value 6',
|
|
})],
|
|
}, {
|
|
'name': 'Attribute 4',
|
|
'create_variant': 'always',
|
|
'display_type': 'color',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Value 7',
|
|
}), (0, 0, {
|
|
'name': 'Value 8',
|
|
})],
|
|
}])
|
|
|
|
self.env['product.template.attribute.line'].create([{
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': attribute_1.id,
|
|
'value_ids': [(6, 0, attribute_1.value_ids.ids)],
|
|
'sequence': 1,
|
|
}, {
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': attribute_2.id,
|
|
'value_ids': [(6, 0, attribute_2.value_ids.ids)],
|
|
'sequence': 2,
|
|
}, {
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': attribute_3.id,
|
|
'value_ids': [(6, 0, attribute_3.value_ids.ids)],
|
|
'sequence': 3,
|
|
}, {
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': attribute_4.id,
|
|
'value_ids': [(6, 0, attribute_4.value_ids.ids)],
|
|
'sequence': 4,
|
|
}])
|
|
|
|
for p in product.product_variant_ids:
|
|
p.write({
|
|
'barcode': f'1234{"".join(p.product_template_attribute_value_ids.mapped(lambda ptav: ptav.name[-1]))}',
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_barcode_search_attributes_preset', login="pos_user")
|
|
|
|
def test_auto_validate_force_done(self):
|
|
self.main_pos_config.write({
|
|
'auto_validate_terminal_payment': True
|
|
})
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_auto_validate_force_done', login="pos_user")
|
|
|
|
def test_pos_ui_round_globally(self):
|
|
self.main_pos_config.company_id.tax_calculation_rounding_method = 'round_globally'
|
|
tax_16 = self.env['account.tax'].create({
|
|
'name': 'Tax 16%',
|
|
'amount': 16,
|
|
})
|
|
self.env['product.product'].create([{
|
|
'name': 'Test Product 1',
|
|
'list_price': 7051.73,
|
|
'taxes_id': [(6, 0, [tax_16.id])],
|
|
'available_in_pos': True,
|
|
}, {
|
|
'name': 'Test Product 2',
|
|
'list_price': 352.59,
|
|
'taxes_id': [(6, 0, [tax_16.id])],
|
|
'available_in_pos': True,
|
|
}])
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_pos_ui_round_globally', login="pos_user")
|
|
|
|
pos_session = self.main_pos_config.current_session_id
|
|
self.assertEqual(pos_session.order_ids[0].payment_ids[0].amount, 7771.01)
|
|
|
|
# Close the session and check the session journal entry.
|
|
pos_session.action_pos_session_validate()
|
|
|
|
lines = pos_session.move_id.line_ids.sorted('balance')
|
|
|
|
self.assertEqual(len(lines), 5, "There should be 5 lines in the session journal entry")
|
|
self.assertAlmostEqual(lines[0].balance, -7051.73)
|
|
self.assertAlmostEqual(lines[1].balance, -1128.28)
|
|
self.assertAlmostEqual(lines[2].balance, 56.41)
|
|
self.assertAlmostEqual(lines[3].balance, 352.59)
|
|
self.assertAlmostEqual(lines[4].balance, 7771.01)
|
|
|
|
def test_ctrl_number_ignored(self):
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_ctrl_number_ignored', login="pos_user")
|
|
|
|
def test_click_all_orders_keep_customer(self):
|
|
"""Verify that clicking on 'All Orders' keeps the customer selected."""
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_click_all_orders_keep_customer', login="pos_user")
|
|
|
|
def test_quantity_package_of_non_basic_unit(self):
|
|
test_uom_unit = self.env['uom.uom'].create({
|
|
"name": "test unit uom",
|
|
"relative_factor": "1.0",
|
|
})
|
|
pack_of_12_unit = self.env['uom.uom'].create({
|
|
'name': 'Pack of 12 unit',
|
|
'relative_factor': 12,
|
|
'relative_uom_id': test_uom_unit.id,
|
|
'is_pos_groupable': True,
|
|
})
|
|
product_cord = self.env['product.product'].create({
|
|
'name': 'Cord',
|
|
'is_storable': True,
|
|
'available_in_pos': True,
|
|
'uom_id': test_uom_unit.id,
|
|
'uom_ids': [pack_of_12_unit.id],
|
|
'lst_price': 10.0,
|
|
})
|
|
self.env['product.uom'].create({
|
|
'barcode': '555555',
|
|
'product_id': product_cord.id,
|
|
'uom_id': pack_of_12_unit.id,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_quantity_package_of_non_basic_unit', login="pos_user")
|
|
|
|
def test_attribute_order(self):
|
|
product = self.env['product.template'].create({
|
|
'name': 'Product Test',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
})
|
|
|
|
attribute_3 = self.env['product.attribute'].create({
|
|
'name': 'Attribute 3',
|
|
'create_variant': 'no_variant',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Value 3',
|
|
}), (0, 0, {
|
|
'name': 'Value 4',
|
|
})],
|
|
})
|
|
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': attribute_3.id,
|
|
'value_ids': [(6, 0, attribute_3.value_ids.ids)],
|
|
'sequence': 3,
|
|
})
|
|
|
|
attribute_2 = self.env['product.attribute'].create({
|
|
'name': 'Attribute 2',
|
|
'create_variant': 'no_variant',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Value 2',
|
|
})],
|
|
})
|
|
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': attribute_2.id,
|
|
'value_ids': [(6, 0, attribute_2.value_ids.ids)],
|
|
'sequence': 2,
|
|
})
|
|
|
|
attribute_1 = self.env['product.attribute'].create({
|
|
'name': 'Attribute 1',
|
|
'create_variant': 'no_variant',
|
|
'value_ids': [(0, 0, {
|
|
'name': 'Value 1',
|
|
})],
|
|
})
|
|
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': attribute_1.id,
|
|
'value_ids': [(6, 0, attribute_1.value_ids.ids)],
|
|
'sequence': 1,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_attribute_order', login="pos_user")
|
|
|
|
def test_preset_timing_retail(self):
|
|
"""
|
|
Test to set order preset hour inside a tour
|
|
"""
|
|
self.preset_dine_in = self.env['pos.preset'].create({
|
|
'name': 'Dine in',
|
|
})
|
|
self.preset_delivery = self.env['pos.preset'].create({
|
|
'name': 'Delivery',
|
|
'identification': 'address',
|
|
})
|
|
self.main_pos_config.write({
|
|
'use_presets': True,
|
|
'default_preset_id': self.preset_dine_in.id,
|
|
'available_preset_ids': [(6, 0, [self.preset_delivery.id])],
|
|
})
|
|
self.pos_user.street = 'Rue de Ramillies'
|
|
resource_calendar = self.env['resource.calendar'].create({
|
|
'name': 'Takeaway',
|
|
'attendance_ids': [(0, 0, {
|
|
'name': 'Takeaway',
|
|
'dayofweek': str(day),
|
|
'hour_from': 0,
|
|
'hour_to': 24,
|
|
'day_period': 'morning',
|
|
}) for day in range(7)],
|
|
})
|
|
self.preset_delivery.write({
|
|
'use_timing': True,
|
|
'resource_calendar_id': resource_calendar
|
|
})
|
|
self.start_pos_tour('test_preset_timing_retail')
|
|
|
|
def test_pricelists_in_pos(self):
|
|
pos_limited_category = self.env['pos.category'].create({'name': 'Limited Category'})
|
|
pos_category = self.env['pos.category'].create({'name': 'test_pricelists_in_pos'})
|
|
product_category = self.env['product.category'].create({'name': 'test_pricelists_in_pos'})
|
|
orange_category = self.env['product.category'].create({'name': 'Orange Category'})
|
|
|
|
def generate_pricelist_items(pricelist, fixed_price, product=None, product_tmpl=None, product_category=None):
|
|
applied_on = '0_product_variant' if product else '1_product' if product_tmpl else '2_product_category' if product_category else '3_global'
|
|
return self.env['product.pricelist.item'].create({
|
|
'pricelist_id': pricelist.id,
|
|
'product_id': product.id if product else False,
|
|
'product_tmpl_id': product_tmpl.id if product_tmpl else False,
|
|
'categ_id': product_category.id if product_category else False,
|
|
'compute_price': 'fixed',
|
|
'applied_on': applied_on,
|
|
'fixed_price': fixed_price,
|
|
})
|
|
|
|
def generate_product_template_with_attributes(name, price, pos_category=None, product_category=None):
|
|
size_attribute = self.env['product.attribute'].create({
|
|
'name': 'Size',
|
|
'sequence': 4,
|
|
'value_ids': [(0, 0, {
|
|
'name': 'BIG',
|
|
'sequence': 1,
|
|
}), (0, 0, {
|
|
'name': 'MEDIUM',
|
|
'sequence': 2,
|
|
}), (0, 0, {
|
|
'name': 'SMALL',
|
|
'sequence': 3,
|
|
})],
|
|
})
|
|
|
|
product_tmpl = self.env['product.template'].create({
|
|
'name': name.capitalize(),
|
|
'available_in_pos': True,
|
|
'categ_id': product_category.id if product_category else False,
|
|
'pos_categ_ids': [(4, pos_category.id)] if pos_category else False,
|
|
'list_price': price,
|
|
'taxes_id': False,
|
|
'attribute_line_ids': [(0, 0, {
|
|
'attribute_id': size_attribute.id,
|
|
'value_ids': [(6, 0, size_attribute.value_ids.ids)]
|
|
})],
|
|
})
|
|
|
|
for index, variant in enumerate(product_tmpl.product_variant_ids):
|
|
variant.write({'barcode': f'{name}_{index}'})
|
|
|
|
return product_tmpl
|
|
|
|
banana = generate_product_template_with_attributes('banana', 10.00, pos_category)
|
|
apple = generate_product_template_with_attributes('apple', 5.00, False, product_category)
|
|
pear = generate_product_template_with_attributes('pear', 2.00)
|
|
lime = generate_product_template_with_attributes('lime', 1.00)
|
|
orange = generate_product_template_with_attributes('orange', 3.00, False, orange_category)
|
|
kiwi = generate_product_template_with_attributes('kiwi', 4.00)
|
|
|
|
test_pricelist = self.env['product.pricelist'].create({
|
|
'name': 'Test Pricelist',
|
|
})
|
|
|
|
percentage_pricelist = self.env['product.pricelist'].create({
|
|
'name': 'Percentage Pricelist',
|
|
})
|
|
|
|
generate_pricelist_items(test_pricelist, 20, False, banana)
|
|
generate_pricelist_items(test_pricelist, 100, banana.product_variant_ids[0])
|
|
generate_pricelist_items(test_pricelist, 150, banana.product_variant_ids[1])
|
|
generate_pricelist_items(test_pricelist, 500, False, False, product_category)
|
|
generate_pricelist_items(test_pricelist, 1000, False, False, orange_category)
|
|
generate_pricelist_items(test_pricelist, 100, apple.product_variant_ids[0])
|
|
generate_pricelist_items(test_pricelist, 20, pear.product_variant_ids[0])
|
|
generate_pricelist_items(test_pricelist, 40, pear.product_variant_ids[1])
|
|
generate_pricelist_items(test_pricelist, 60, pear.product_variant_ids[2])
|
|
generate_pricelist_items(test_pricelist, 100, False, lime)
|
|
generate_pricelist_items(test_pricelist, 200, lime.product_variant_ids[1])
|
|
generate_pricelist_items(test_pricelist, 400, lime.product_variant_ids[2])
|
|
generate_pricelist_items(test_pricelist, 600, orange.product_variant_ids[1])
|
|
generate_pricelist_items(test_pricelist, 500, orange.product_variant_ids[2])
|
|
generate_pricelist_items(test_pricelist, 10)
|
|
generate_pricelist_items(test_pricelist, 20, kiwi.product_variant_ids[0])
|
|
|
|
self.env['product.pricelist.item'].create({
|
|
'pricelist_id': percentage_pricelist.id,
|
|
'base': 'pricelist',
|
|
'base_pricelist_id': test_pricelist.id,
|
|
'compute_price': 'percentage',
|
|
'percent_price': 50,
|
|
'applied_on': '3_global',
|
|
})
|
|
|
|
self.main_pos_config.write({
|
|
"limit_categories": True,
|
|
"iface_available_categ_ids": [(6, 0, [pos_limited_category.id])],
|
|
'available_pricelist_ids': [(6, 0, [test_pricelist.id, percentage_pricelist.id])],
|
|
'pricelist_id': test_pricelist.id,
|
|
})
|
|
|
|
load_product_from_pos_stats = {'count': 0, 'items': {}}
|
|
product_template = self.env.registry.models['product.template']
|
|
|
|
# Test product exclusion
|
|
cherry = generate_product_template_with_attributes('cherry', 2.00)
|
|
color_attribute = self.env['product.attribute'].create({
|
|
'name': 'Color',
|
|
'sequence': 5,
|
|
'value_ids': [(0, 0, {
|
|
'name': 'RED',
|
|
'sequence': 1,
|
|
}), (0, 0, {
|
|
'name': 'GREEN',
|
|
'sequence': 2,
|
|
}), (0, 0, {
|
|
'name': 'BLUE',
|
|
'sequence': 3,
|
|
})],
|
|
})
|
|
cherry.attribute_line_ids = [(0, 0, {
|
|
'attribute_id': color_attribute.id,
|
|
'value_ids': [(6, 0, color_attribute.value_ids.ids)]
|
|
})]
|
|
color_attribute = cherry.attribute_line_ids.filtered(lambda l: l.attribute_id.name == 'Color')
|
|
first_color_value = color_attribute.product_template_value_ids.filtered(lambda v: v.attribute_id.name == 'Color' and v.name == 'RED')
|
|
first_size_value = cherry.product_variant_ids.product_template_attribute_value_ids.filtered(lambda v: v.attribute_id.name == 'Size' and v.name == 'BIG')
|
|
first_color_value.exclude_for = [(0, 0, {
|
|
'product_tmpl_id': cherry.id,
|
|
'value_ids': first_size_value.ids,
|
|
'product_template_attribute_value_id': first_size_value.id
|
|
})]
|
|
for index, variant in enumerate(cherry.product_variant_ids):
|
|
variant.write({'barcode': f'cherry_{index}'})
|
|
|
|
@api.model
|
|
def load_product_from_pos_patch(self, config_id, domain, offset=0, limit=0):
|
|
load_product_from_pos_stats['count'] += 1
|
|
result = super(product_template, self).load_product_from_pos(config_id, domain, offset, limit)
|
|
lowered_name = result['product.template'][0]['display_name'].lower()
|
|
load_product_from_pos_stats['items'][lowered_name] = len(result['product.pricelist.item'])
|
|
return result
|
|
|
|
with patch.object(product_template, "load_product_from_pos", load_product_from_pos_patch):
|
|
self.start_pos_tour('test_pricelists_in_pos')
|
|
|
|
# Should load 6 different products, since 6 products were created
|
|
self.assertEqual(load_product_from_pos_stats['count'], 7)
|
|
|
|
# Length of loaded pricelist items should correspond to the number of items linked
|
|
# to the product template or product variant
|
|
# Global rules are loaded at starting of the PoS
|
|
self.assertEqual(load_product_from_pos_stats['items']['banana'], 3, "Banana should have 3 pricelist items")
|
|
self.assertEqual(load_product_from_pos_stats['items']['apple'], 1, "Apple should have 1 pricelist item")
|
|
self.assertEqual(load_product_from_pos_stats['items']['pear'], 3, "Pear should have 3 pricelist items")
|
|
self.assertEqual(load_product_from_pos_stats['items']['lime'], 3, "Lime should have 3 pricelist items")
|
|
self.assertEqual(load_product_from_pos_stats['items']['orange'], 2, "Orange should have 2 pricelist items")
|
|
self.assertEqual(load_product_from_pos_stats['items']['kiwi'], 1, "Kiwi should have 1 pricelist item")
|
|
|
|
def test_available_children_categories(self):
|
|
parent_categ = self.env['pos.category'].create({
|
|
'name': 'Parent Category',
|
|
})
|
|
children_categs = self.env['pos.category'].create([{
|
|
'name': 'Child Category 1',
|
|
'parent_id': parent_categ.id,
|
|
}, {
|
|
'name': 'Child Category 2',
|
|
'parent_id': parent_categ.id,
|
|
}])
|
|
self.env['product.product'].create([{
|
|
'name': 'parent product',
|
|
'pos_categ_ids': [(6, 0, [parent_categ.id])],
|
|
'available_in_pos': True,
|
|
}, {
|
|
'name': 'child product 1',
|
|
'pos_categ_ids': [(6, 0, [parent_categ.id, children_categs[0].id])],
|
|
'available_in_pos': True,
|
|
}, {
|
|
'name': 'child product 2',
|
|
'pos_categ_ids': [(6, 0, [parent_categ.id, children_categs[1].id])],
|
|
'available_in_pos': True,
|
|
}])
|
|
self.main_pos_config.write({
|
|
'limit_categories': True,
|
|
'iface_available_categ_ids': [(6, 0, [parent_categ.id, children_categs[1].id])],
|
|
})
|
|
self.main_pos_config.open_ui()
|
|
loaded_data = self.main_pos_config.current_session_id.load_data([])
|
|
category_id = [category['id'] for category in loaded_data['pos.category']]
|
|
self.assertNotIn(children_categs[0].id, category_id, "Child category is unavailable and shouldn't appear in the POS")
|
|
self.assertIn(children_categs[1].id, category_id, "Child category is available and should appear in the POS")
|
|
|
|
def test_available_product_uom_ids(self):
|
|
# Making sure that all of the non-special products that are included in the `load_data` are the ones created in this method.
|
|
self.env['product.template'].search([]).write({'is_favorite': False})
|
|
|
|
self.env['ir.config_parameter'].sudo().set_param('point_of_sale.limited_product_count', '2')
|
|
uom = self.env['uom.uom'].create({
|
|
'name': 'Random UOM',
|
|
'relative_uom_id': self.env.ref('uom.product_uom_unit').id,
|
|
})
|
|
product_one, product_two, product_three = self.env['product.product'].create([{
|
|
'name': "product_one",
|
|
'available_in_pos': True,
|
|
'is_favorite': True,
|
|
},
|
|
{
|
|
'name': "product_two",
|
|
'available_in_pos': True,
|
|
'is_favorite': True,
|
|
},
|
|
{
|
|
'name': "product_three",
|
|
'available_in_pos': True,
|
|
}])
|
|
|
|
_, _, product_uom_three = self.env['product.uom'].create([{
|
|
'barcode': "product_one_barcode",
|
|
'uom_id': uom.id,
|
|
'product_id': product_one.id,
|
|
},
|
|
{
|
|
'barcode': "product_two_barcode",
|
|
'uom_id': uom.id,
|
|
'product_id': product_two.id,
|
|
},
|
|
{
|
|
'barcode': "product_three_barcode",
|
|
'uom_id': uom.id,
|
|
'product_id': product_three.id,
|
|
},
|
|
])
|
|
|
|
self.env['product.template'].flush_model()
|
|
self.main_pos_config.open_ui()
|
|
loaded_data = self.main_pos_config.current_session_id.load_data([])
|
|
loaded_product_uoms = [loaded_product_uom['id'] for loaded_product_uom in loaded_data['product.uom']]
|
|
|
|
self.assertNotIn(product_uom_three.id, loaded_product_uoms, f"Product UOM {product_uom_three} shouldn't be loaded as its product {product_three} is not included in the results")
|
|
|
|
def test_pos_order_shipping_date(self):
|
|
self.env['res.partner'].create({
|
|
'name': 'Partner Test with Address',
|
|
'street': 'test street',
|
|
'zip': '1234',
|
|
'city': 'test city',
|
|
'country_id': self.env.ref('base.us').id
|
|
})
|
|
self.main_pos_config.write({'ship_later': True})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_tour(
|
|
f"/pos/ui?config_id={self.main_pos_config.id}",
|
|
"test_pos_order_shipping_date",
|
|
login="pos_user",
|
|
)
|
|
|
|
def test_fast_payment_validation_from_product_screen_without_automatic_receipt_printing(self):
|
|
self.preset_delivery = self.env['pos.preset'].create({
|
|
'name': 'Delivery',
|
|
'identification': 'address',
|
|
})
|
|
self.main_pos_config.write({
|
|
'use_fast_payment': True,
|
|
'use_presets': True,
|
|
'fast_payment_method_ids': [(6, 0, self.bank_payment_method.ids)],
|
|
'default_preset_id': self.preset_delivery.id,
|
|
'available_preset_ids': [(6, 0, [self.preset_delivery.id])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_fast_payment_validation_from_product_screen_without_automatic_receipt_printing')
|
|
order1 = self.main_pos_config.current_session_id.order_ids[0]
|
|
order2 = self.main_pos_config.current_session_id.order_ids[1]
|
|
self.assertEqual(order1.state, 'paid', "The order should be paid after the fast payment validation")
|
|
self.assertEqual(len(order1.payment_ids), 1, "There should be one payment line used for the fast payment")
|
|
self.assertEqual(order1.payment_ids.payment_method_id, self.bank_payment_method, "The payment method used should be the bank payment method")
|
|
self.assertEqual(order2.state, 'paid', "The order should be paid")
|
|
self.assertEqual(len(order2.payment_ids), 1, "There should be one payment line")
|
|
self.assertEqual(order2.payment_ids.payment_method_id, self.bank_payment_method, "The payment method used should be the bank payment method")
|
|
|
|
def test_fast_payment_validation_from_product_screen_with_automatic_receipt_printing(self):
|
|
self.main_pos_config.write({
|
|
'use_fast_payment': True,
|
|
'fast_payment_method_ids': [(6, 0, self.bank_payment_method.ids)],
|
|
'iface_print_auto': True,
|
|
'iface_print_skip_screen': True,
|
|
'other_devices': True,
|
|
'epson_printer_ip': '127.0.0.1:8069/receipt_receiver',
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_fast_payment_validation_from_product_screen_with_automatic_receipt_printing')
|
|
order1 = self.main_pos_config.current_session_id.order_ids[0]
|
|
order2 = self.main_pos_config.current_session_id.order_ids[1]
|
|
self.assertEqual(order1.state, 'paid', "The order should be paid after the fast payment validation")
|
|
self.assertEqual(len(order1.payment_ids), 1, "There should be one payment line used for the fast payment")
|
|
self.assertEqual(order1.payment_ids.payment_method_id, self.bank_payment_method, "The payment method used should be the bank payment method")
|
|
self.assertEqual(order2.state, 'paid', "The order should be paid")
|
|
self.assertEqual(len(order2.payment_ids), 1, "There should be one payment line")
|
|
self.assertEqual(order2.payment_ids.payment_method_id, self.bank_payment_method, "The payment method used should be the bank payment method")
|
|
|
|
def test_consistent_refund_process_between_frontend_and_backend(self):
|
|
"""
|
|
Ensure that the partial refund process is consistent between the frontend and backend.
|
|
This includes validating the refund order creation, amount, state, and payment processing.
|
|
"""
|
|
# Open POS UI with the POS user
|
|
pricelists = self.env['product.pricelist'].create([
|
|
{'name': 'Test Pricelist'},
|
|
{'name': 'Percentage Pricelist'},
|
|
])
|
|
self.main_pos_config.write({
|
|
'available_pricelist_ids': [Command.set(pricelists.ids)],
|
|
'pricelist_id': pricelists[0].id,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
|
|
# Run the POS tour simulating a partial refund
|
|
self.start_pos_tour('test_consistent_refund_process_between_frontend_and_backend')
|
|
|
|
# Fetch orders created in the current POS session
|
|
orders = self.env['pos.order'].search([
|
|
('session_id', '=', self.main_pos_config.current_session_id.id)
|
|
])
|
|
self.assertEqual(len(orders), 2, "Expected two orders: original and refund.")
|
|
order, refund_order = orders[0], orders[1]
|
|
self.assertEqual(
|
|
refund_order.pricelist_id.id,
|
|
order.pricelist_id.id,
|
|
"Refund order pricelist should be the original order's pricelist."
|
|
)
|
|
|
|
# Perform refund on order and retrieve the resulting draft refund order
|
|
refund_action = orders[1].refund()
|
|
refund_order = self.env['pos.order'].browse(refund_action['res_id'])
|
|
|
|
# Validate the refund order is in draft and has correct negative total
|
|
self.assertEqual(refund_order.state, 'draft', "Refund order should be in draft state.")
|
|
self.assertEqual(refund_order.amount_total, -4, "Refund order total should be -4.")
|
|
|
|
# Create a payment for the refund using the configured bank method
|
|
payment_context = {
|
|
"active_ids": refund_order.ids,
|
|
"active_id": refund_order.id
|
|
}
|
|
refund_payment = self.env['pos.make.payment'].with_context(**payment_context).create({
|
|
'amount': refund_order.amount_total,
|
|
'payment_method_id': self.bank_payment_method.id,
|
|
})
|
|
|
|
# Validate and finalize the refund payment
|
|
refund_payment.with_context(**payment_context).check()
|
|
self.assertEqual(refund_order.state, 'paid', "Refund order should be marked as paid.")
|
|
|
|
def test_paid_order_with_archived_product_loads(self):
|
|
""" Test that a paid order with archived products can be loaded in the POS. """
|
|
|
|
archived_product = self.env['product.product'].create({
|
|
'name': 'Archived Product',
|
|
'available_in_pos': True,
|
|
'list_price': 10.0,
|
|
'taxes_id': False,
|
|
'active': False, # Archived product
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.env['pos.order'].create({
|
|
'config_id': self.main_pos_config.id,
|
|
'session_id': self.main_pos_config.current_session_id.id,
|
|
'company_id': self.main_pos_config.company_id.id,
|
|
'amount_total': 10.0,
|
|
'amount_paid': 10.0,
|
|
'amount_tax': 0.0,
|
|
'amount_return': 0.0,
|
|
'to_invoice': False,
|
|
'partner_id': False,
|
|
'pricelist_id': self.main_pos_config.pricelist_id.id,
|
|
'pos_reference': '1000-004-00002',
|
|
'name': 'Order 0002',
|
|
'state': 'paid',
|
|
'lines': [(0, 0, {
|
|
'name': 'Line 0001',
|
|
'product_id': archived_product.id,
|
|
'price_unit': 10.00,
|
|
'discount': 0,
|
|
'qty': 1,
|
|
'tax_ids': False,
|
|
'price_subtotal': 10.00,
|
|
'price_subtotal_incl': 10.00,
|
|
})],
|
|
})
|
|
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'test_paid_order_with_archived_product_loads', login="pos_user")
|
|
|
|
def test_delete_line(self):
|
|
""" Test that deleting a line in the POS through the popup works correctly. """
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_delete_line')
|
|
|
|
def test_order_invoice_search(self):
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.pos_user.group_ids = [Command.link(self.env.ref('account.group_account_invoice').id)]
|
|
self.start_tour("/pos/ui/%d" % self.main_pos_config.id, 'test_order_invoice_search', login="pos_user")
|
|
|
|
def test_automatic_receipt_printing(self):
|
|
self.main_pos_config.write({
|
|
'iface_print_auto': True,
|
|
'iface_print_skip_screen': True,
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_automatic_receipt_printing', login="pos_user")
|
|
|
|
def test_load_pos_demo_data(self):
|
|
""" Test that the demo data can be loaded by admin but not by user. """
|
|
|
|
if loaded_demo_data(self.env):
|
|
self.skipTest('Cannot test with demo data.')
|
|
|
|
# archive existing product records
|
|
archive_products(self.env)
|
|
|
|
# cannot load by pos user
|
|
self.start_pos_tour('test_load_pos_demo_data_by_pos_user', login='pos_user')
|
|
products = self.env['product.template'].search_count([('available_in_pos', '=', True)])
|
|
self.assertFalse(products, 'Demo data should not be loaded by user.')
|
|
|
|
# Member role with POS Administrator access
|
|
self.pos_user.write({'group_ids': [
|
|
Command.set(
|
|
[
|
|
self.env.ref('base.group_user').id,
|
|
self.env.ref('point_of_sale.group_pos_manager').id,
|
|
self.env.ref('account.group_account_manager').id,
|
|
]
|
|
)
|
|
]})
|
|
self.start_pos_tour('test_load_pos_demo_data_with_member_role', login='pos_user')
|
|
products = self.env['product.template'].search_count([('available_in_pos', '=', True)])
|
|
self.assertFalse(products, 'Demo data should not be loaded by user with member role.')
|
|
|
|
def test_combo_variant_mix(self):
|
|
color_attribute = self.env['product.attribute'].create({
|
|
'name': 'Color',
|
|
'value_ids': [
|
|
Command.create({'name': 'Red'}),
|
|
Command.create({'name': 'Blue'})
|
|
],
|
|
'create_variant': 'no_variant',
|
|
})
|
|
size_attribute = self.env['product.attribute'].create({
|
|
'name': 'Size',
|
|
'value_ids': [
|
|
Command.create({'name': 'Small'}),
|
|
Command.create({'name': 'Large'})
|
|
],
|
|
'create_variant': 'always',
|
|
})
|
|
|
|
product_template = self.env['product.template'].create({
|
|
'name': 'Test Product',
|
|
'available_in_pos': True,
|
|
'list_price': 10,
|
|
'taxes_id': False,
|
|
'attribute_line_ids': [
|
|
Command.create({
|
|
'attribute_id': color_attribute.id,
|
|
'value_ids': [Command.link(id) for id in color_attribute.value_ids.ids]
|
|
}),
|
|
Command.create({
|
|
'attribute_id': size_attribute.id,
|
|
'value_ids': [Command.link(id) for id in size_attribute.value_ids.ids]
|
|
})
|
|
]
|
|
})
|
|
|
|
combo = self.env['product.combo'].create({
|
|
'name': 'Test Combo',
|
|
'combo_item_ids': [
|
|
Command.create({
|
|
'product_id': product_template.product_variant_ids[0].id,
|
|
'extra_price': 0,
|
|
}),
|
|
Command.create({
|
|
'product_id': product_template.product_variant_ids[1].id,
|
|
'extra_price': 0,
|
|
}),
|
|
]
|
|
})
|
|
self.env['product.template'].create({
|
|
'name': 'Test Product Combo',
|
|
'available_in_pos': True,
|
|
'list_price': 20,
|
|
'taxes_id': False,
|
|
'type': 'combo',
|
|
'combo_ids': [Command.link(combo.id)],
|
|
})
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_combo_variant_mix', login="pos_user")
|
|
|
|
def test_cross_exclusion_attribute_values(self):
|
|
""" If you create a product with two attributes and 2 values for each attribute, and you exclude one value of the first attribute with one value of the second attribute
|
|
and vice versa, you should still be able to select the other values of the attributes. """
|
|
self.attribute_1 = self.env['product.attribute'].create({
|
|
'name': 'attribute_1',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
|
|
self.attribute_2 = self.env['product.attribute'].create({
|
|
'name': 'attribute_2',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
|
|
self.attribute_1_value_1 = self.env['product.attribute.value'].create({
|
|
'name': 'attribute_1_value_1',
|
|
'attribute_id': self.attribute_1.id,
|
|
})
|
|
self.attribute_1_value_2 = self.env['product.attribute.value'].create({
|
|
'name': 'attribute_1_value_2',
|
|
'attribute_id': self.attribute_1.id,
|
|
})
|
|
self.attribute_2_value_1 = self.env['product.attribute.value'].create({
|
|
'name': 'attribute_2_value_1',
|
|
'attribute_id': self.attribute_2.id,
|
|
})
|
|
self.attribute_2_value_2 = self.env['product.attribute.value'].create({
|
|
'name': 'attribute_2_value_2',
|
|
'attribute_id': self.attribute_2.id,
|
|
})
|
|
|
|
self.test_product_1 = self.env['product.template'].create({
|
|
'name': 'Test Product 1',
|
|
'available_in_pos': True,
|
|
'list_price': 10.0,
|
|
'attribute_line_ids': [
|
|
(0, 0, {
|
|
'attribute_id': self.attribute_1.id,
|
|
'value_ids': [(6, 0, [self.attribute_1_value_1.id, self.attribute_1_value_2.id])],
|
|
}),
|
|
(0, 0, {
|
|
'attribute_id': self.attribute_2.id,
|
|
'value_ids': [(6, 0, [self.attribute_2_value_1.id, self.attribute_2_value_2.id])],
|
|
}),
|
|
],
|
|
})
|
|
|
|
# Test the exclusion of attribute values
|
|
ptav_1_1 = self.test_product_1.attribute_line_ids.filtered(lambda l: l.attribute_id.id == self.attribute_1.id).product_template_value_ids.filtered(lambda v: v.product_attribute_value_id.id == self.attribute_1_value_1.id)
|
|
ptav_1_2 = self.test_product_1.attribute_line_ids.filtered(lambda l: l.attribute_id.id == self.attribute_1.id).product_template_value_ids.filtered(lambda v: v.product_attribute_value_id.id == self.attribute_1_value_2.id)
|
|
ptav_2_2 = self.test_product_1.attribute_line_ids.filtered(lambda l: l.attribute_id.id == self.attribute_2.id).product_template_value_ids.filtered(lambda v: v.product_attribute_value_id.id == self.attribute_2_value_2.id)
|
|
ptav_2_1 = self.test_product_1.attribute_line_ids.filtered(lambda l: l.attribute_id.id == self.attribute_2.id).product_template_value_ids.filtered(lambda v: v.product_attribute_value_id.id == self.attribute_2_value_1.id)
|
|
self.env['product.template.attribute.exclusion'].create({
|
|
'product_tmpl_id': self.test_product_1.id,
|
|
'product_template_attribute_value_id': ptav_1_1.id,
|
|
'value_ids': [Command.set([ptav_2_1.id])],
|
|
})
|
|
|
|
self.env['product.template.attribute.exclusion'].create({
|
|
'product_tmpl_id': self.test_product_1.id,
|
|
'product_template_attribute_value_id': ptav_1_2.id,
|
|
'value_ids': [Command.set([ptav_2_2.id])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_cross_exclusion_attribute_values')
|
|
|
|
def test_custom_attribute_alone_displayed(self):
|
|
"""
|
|
Tests that if product configurator will be shown if any of the
|
|
attributes have a free text field, even if there is only one
|
|
possible selection for every attributes.
|
|
"""
|
|
attribute_custom = self.env['product.attribute'].create({
|
|
'name': 'Custom',
|
|
'display_type': 'radio',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
attribute_value_custom = self.env['product.attribute.value'].create({
|
|
'name': 'Custom',
|
|
'attribute_id': attribute_custom.id,
|
|
'is_custom': True,
|
|
})
|
|
self.test_product_1 = self.env['product.template'].create({
|
|
'name': 'Only Custom',
|
|
'available_in_pos': True,
|
|
'list_price': 10.0,
|
|
'attribute_line_ids': [
|
|
Command.create({
|
|
'attribute_id': attribute_custom.id,
|
|
'value_ids': [Command.set([attribute_value_custom.id])],
|
|
}),
|
|
],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_custom_attribute_alone_displayed')
|
|
|
|
def test_preset_customer_selection(self):
|
|
self.preset_delivery = self.env['pos.preset'].create({
|
|
'name': 'Delivery',
|
|
'identification': 'address',
|
|
})
|
|
self.env['res.partner'].create({
|
|
'name': 'Test Partner',
|
|
'street': '123 Test Street',
|
|
'city': 'Test City',
|
|
'zip': '12345',
|
|
'country_id': self.env['res.country'].search([], limit=1).id,
|
|
})
|
|
self.main_pos_config.write({
|
|
'use_presets': True,
|
|
'default_preset_id': self.preset_delivery.id,
|
|
'available_preset_ids': [(6, 0, [self.preset_delivery.id])],
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_preset_customer_selection')
|
|
|
|
def test_pos_large_amount_confirmation_dialog(self):
|
|
"""Test that the Large amount confirmation dialog appears
|
|
and closes properly after clicking 'OK'."""
|
|
self.env['product.product'].create({
|
|
'name': 'Overpay Test Product',
|
|
'list_price': 1.0,
|
|
'available_in_pos': True,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_pos_large_amount_confirmation_dialog')
|
|
|
|
def test_product_info_product_inventory(self):
|
|
""" Test that the product variant inventory info is correctly displayed in the POS. """
|
|
size_attribute = self.env['product.attribute'].create({
|
|
'name': 'Size',
|
|
'value_ids': [
|
|
Command.create({'name': 'Small'}),
|
|
Command.create({'name': 'Large'})
|
|
],
|
|
'create_variant': 'always',
|
|
})
|
|
|
|
product_template = self.env['product.template'].create({
|
|
'name': 'Test Product',
|
|
'available_in_pos': True,
|
|
'is_storable': True,
|
|
'attribute_line_ids': [
|
|
Command.create({
|
|
'attribute_id': size_attribute.id,
|
|
'value_ids': [Command.link(id) for id in size_attribute.value_ids.ids]
|
|
})
|
|
]
|
|
})
|
|
|
|
for variant in range(len(product_template.product_variant_ids)):
|
|
self.env['stock.quant']._update_available_quantity(product_template.product_variant_ids[variant], self.main_pos_config.warehouse_id.lot_stock_id, (variant + 1) * 100)
|
|
product_template.product_variant_ids[variant].write({'barcode': f'product_variant_{variant}'})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_product_info_product_inventory')
|
|
|
|
def test_add_money_button_with_different_decimal_separator(self):
|
|
"""
|
|
Tests that the buttons such as +10 or +50 work even in languages such as
|
|
french that have ',' as a decimal separator.
|
|
"""
|
|
lang = self.env['res.lang'].search([('code', '=', self.pos_user.lang)])
|
|
lang.write({'thousands_sep': '.', 'decimal_point': ','})
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'test_add_money_button_with_different_decimal_separator', login="pos_user")
|
|
|
|
def test_sync_from_ui_one_by_one(self):
|
|
"""
|
|
Sync from UI is now syncing orders one by one.
|
|
sync_from_ui should be called 6 times in this tour (6 orders created).
|
|
"""
|
|
|
|
pos_order = self.env.registry.models['pos.order']
|
|
sync_counter = {'count': 0}
|
|
|
|
@api.model
|
|
def sync_from_ui_patch(self, orders):
|
|
sync_counter['count'] += 1
|
|
return super(pos_order, self).sync_from_ui(orders)
|
|
|
|
with patch.object(pos_order, "sync_from_ui", sync_from_ui_patch):
|
|
self.start_pos_tour("test_sync_from_ui_one_by_one", login="pos_user")
|
|
self.assertEqual(sync_counter['count'], 6)
|
|
|
|
def test_lot_refund_lower_qty(self):
|
|
product = self.env['product.product'].create({
|
|
'name': 'Serial Product',
|
|
'is_storable': True,
|
|
'tracking': 'serial',
|
|
'available_in_pos': True,
|
|
})
|
|
for sn in ["SN1", "SN2"]:
|
|
self.env['stock.quant'].create({
|
|
'product_id': product.id,
|
|
'inventory_quantity': 1,
|
|
'location_id': self.env.user._get_default_warehouse_id().lot_stock_id.id,
|
|
'lot_id': self.env['stock.lot'].create({'name': sn, 'product_id': product.id}).id,
|
|
}).sudo().action_apply_inventory()
|
|
self.env['stock.picking.type'].search([('name', '=', 'PoS Orders')]).use_create_lots = False
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour("test_lot_refund_lower_qty")
|
|
|
|
def test_exclusion_attribute_values(self):
|
|
chair_fabrics_other_ptav = self.configurable_chair.attribute_line_ids.filtered(lambda l: l.attribute_id.id == self.chair_fabrics_attribute.id).product_template_value_ids.filtered(lambda v: v.product_attribute_value_id.id == self.chair_fabrics_other.id)
|
|
chair_fabrics_wool_ptav = self.configurable_chair.attribute_line_ids.filtered(lambda l: l.attribute_id.id == self.chair_fabrics_attribute.id).product_template_value_ids.filtered(lambda v: v.product_attribute_value_id.id == self.chair_fabrics_wool.id)
|
|
chair_color_red_ptav = self.configurable_chair.attribute_line_ids.filtered(lambda l: l.attribute_id.id == self.chair_color_attribute.id).product_template_value_ids.filtered(lambda v: v.product_attribute_value_id.id == self.chair_color_red.id)
|
|
|
|
# Test the exclusion of attribute values
|
|
self.env['product.template.attribute.exclusion'].create({
|
|
'product_tmpl_id': self.configurable_chair.id,
|
|
'product_template_attribute_value_id': chair_color_red_ptav.id,
|
|
'value_ids': [Command.set([chair_fabrics_other_ptav.id])],
|
|
})
|
|
|
|
# # Test the exclusion of attribute values in the opposite direction
|
|
self.env['product.template.attribute.exclusion'].create({
|
|
'product_tmpl_id': self.configurable_chair.id,
|
|
'product_template_attribute_value_id': chair_fabrics_wool_ptav.id,
|
|
'value_ids': [Command.set([chair_color_red_ptav.id])],
|
|
})
|
|
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_exclusion_attribute_values')
|
|
|
|
def test_lot_tracking_without_lot_creation(self):
|
|
pricelist = self.env['product.pricelist'].create({
|
|
'name': 'Test Pricelist',
|
|
})
|
|
self.main_pos_config.write({
|
|
'available_pricelist_ids': [(6, 0, [pricelist.id])],
|
|
'pricelist_id': pricelist.id,
|
|
})
|
|
self.main_pos_config.picking_type_id.write({
|
|
"use_create_lots": False,
|
|
"use_existing_lots": False,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.monitor_stand.tracking = 'lot'
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_lot_tracking_without_lot_creation', login="pos_user")
|
|
|
|
def test_refund_line_keep_attributes(self):
|
|
"""
|
|
Tests that when refunding an order that has lines that are variants, the new line keeps
|
|
this variant and displays it.
|
|
"""
|
|
product_test = self.env['product.product'].create({
|
|
'name': 'Donut',
|
|
'list_price': 10,
|
|
'available_in_pos': True,
|
|
'taxes_id': False,
|
|
})
|
|
attribute = self.env['product.attribute'].create({
|
|
'name': 'Flavor',
|
|
'create_variant': 'always',
|
|
})
|
|
attribute_value_1 = self.env['product.attribute.value'].create({
|
|
'name': 'Sugar',
|
|
'attribute_id': attribute.id,
|
|
})
|
|
attribute_value_2 = self.env['product.attribute.value'].create({
|
|
'name': 'Chocolate',
|
|
'attribute_id': attribute.id,
|
|
})
|
|
self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product_test.product_tmpl_id.id,
|
|
'attribute_id': attribute.id,
|
|
'value_ids': [(6, 0, [attribute_value_1.id, attribute_value_2.id])],
|
|
})
|
|
self.start_pos_tour("test_refund_line_keep_attributes")
|
|
|
|
def test_set_opening_note_without_cash_method(self):
|
|
cash_method = self.main_pos_config.payment_method_ids.filtered(lambda pm: pm.is_cash_count)
|
|
self.main_pos_config.payment_method_ids -= cash_method
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
current_session = self.main_pos_config.current_session_id
|
|
self.start_pos_tour('test_set_opening_note_without_cash_method')
|
|
self.assertEqual(current_session.opening_notes, 'Opening Notes')
|
|
|
|
def test_orderline_merge_with_higher_price_precision(self):
|
|
""" Test that orderline merging works correctly when product price has a higher precision than the currency. """
|
|
self.env['decimal.precision'].search([('name', '=', 'Product Price')]).digits = 3
|
|
|
|
self.env['product.product'].create({
|
|
'name': 'High Precision Product',
|
|
'list_price': 8.245,
|
|
'taxes_id': False,
|
|
'available_in_pos': True,
|
|
})
|
|
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_orderline_merge_with_higher_price_precision', login="pos_user")
|
|
|
|
def test_product_configurator_price(self):
|
|
""" Test that the product configurator displays the correct price when selecting attributes that impact the price. """
|
|
self.env['product.template'].search([('available_in_pos', '=', True)]).active = False
|
|
tax_10 = self.env['account.tax'].create({
|
|
'name': 'Tax 10%',
|
|
'amount': 10,
|
|
})
|
|
fiscal_position = self.env['account.fiscal.position'].create({
|
|
'name': 'Include to Exclude',
|
|
})
|
|
tax_10 = self.env['account.tax'].create({
|
|
'name': 'Tax 10 Excluded',
|
|
'amount': 10,
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'price_include_override': 'tax_excluded',
|
|
})
|
|
self.env['account.tax'].create({
|
|
'name': 'Tax 10 Included',
|
|
'amount': 10,
|
|
'amount_type': 'percent',
|
|
'type_tax_use': 'sale',
|
|
'price_include_override': 'tax_included',
|
|
'fiscal_position_ids': fiscal_position,
|
|
'original_tax_ids': tax_10,
|
|
})
|
|
product = self.env['product.template'].create({
|
|
'name': 'Configurable Product',
|
|
'available_in_pos': True,
|
|
'list_price': 10.0,
|
|
'taxes_id': [(6, 0, [tax_10.id])],
|
|
})
|
|
size_attribute = self.env['product.attribute'].create({
|
|
'name': 'Size',
|
|
'create_variant': 'always',
|
|
})
|
|
color_attribute = self.env['product.attribute'].create({
|
|
'name': 'Color',
|
|
'create_variant': 'no_variant',
|
|
})
|
|
small_size_value, large_size_value = self.env['product.attribute.value'].create([{
|
|
'name': 'Small',
|
|
'attribute_id': size_attribute.id,
|
|
}, {
|
|
'name': 'Large',
|
|
'attribute_id': size_attribute.id,
|
|
}])
|
|
red_color_value, blue_color_value = self.env['product.attribute.value'].create([{
|
|
'name': 'Red',
|
|
'attribute_id': color_attribute.id,
|
|
}, {
|
|
'name': 'Blue',
|
|
'attribute_id': color_attribute.id,
|
|
}])
|
|
size_line = self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': size_attribute.id,
|
|
'value_ids': [(6, 0, [small_size_value.id, large_size_value.id])],
|
|
})
|
|
size_line.product_template_value_ids[1].price_extra = 1
|
|
color_line = self.env['product.template.attribute.line'].create({
|
|
'product_tmpl_id': product.id,
|
|
'attribute_id': color_attribute.id,
|
|
'value_ids': [(6, 0, [red_color_value.id, blue_color_value.id])],
|
|
})
|
|
color_line.product_template_value_ids[0].price_extra = 2
|
|
color_line.product_template_value_ids[1].price_extra = 3
|
|
|
|
pricelist_1, pricelist_2 = self.env['product.pricelist'].create([{
|
|
'name': 'Pricelist 1',
|
|
}, {
|
|
'name': 'Pricelist 2',
|
|
'item_ids': [(0, 0, {
|
|
'applied_on': '1_product',
|
|
'product_tmpl_id': product.id,
|
|
'fixed_price': 20.0,
|
|
})],
|
|
}])
|
|
self.main_pos_config.write({
|
|
'available_pricelist_ids': [(6, 0, [pricelist_1.id, pricelist_2.id])],
|
|
'pricelist_id': pricelist_1.id,
|
|
'tax_regime_selection': True,
|
|
'fiscal_position_ids': [(6, 0, [fiscal_position.id])],
|
|
})
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_product_configurator_price', login="pos_user")
|
|
|
|
def test_combo_no_free_item(self):
|
|
""" Test a product combo with no free item allowed. """
|
|
setup_product_combo_items(self)
|
|
for combo_item in self.office_combo.combo_ids:
|
|
combo_item.write({
|
|
'qty_free': 0,
|
|
'qty_max': 5,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
self.start_pos_tour('test_combo_no_free_item')
|
|
|
|
def test_not_available_pricelist_not_set_on_order(self):
|
|
""" Test that when the pricelist is not available, it is not set on the order """
|
|
not_available_pricelist, available_pricelist = self.env['product.pricelist'].create([{
|
|
'name': 'Not Available Pricelist',
|
|
}, {
|
|
'name': 'Available Pricelist',
|
|
}])
|
|
|
|
self.main_pos_config.write({
|
|
'available_pricelist_ids': [(4, available_pricelist.id)],
|
|
'pricelist_id': available_pricelist.id,
|
|
})
|
|
self.main_pos_config.with_user(self.pos_user).open_ui()
|
|
pos_session = self.main_pos_config.current_session_id
|
|
|
|
partner = self.env['res.partner'].create({
|
|
'name': 'AA Customer',
|
|
'property_product_pricelist': not_available_pricelist.id,
|
|
})
|
|
|
|
order = self.env['pos.order'].create({
|
|
'company_id': self.env.company.id,
|
|
'session_id': pos_session.id,
|
|
'partner_id': partner.id,
|
|
'config_id': self.main_pos_config.id,
|
|
'lines': [(0, 0, {
|
|
'name': 'OL/0001',
|
|
'product_id': self.wall_shelf.product_variant_ids[0].id,
|
|
'price_unit': 10.00,
|
|
'discount': 0,
|
|
'qty': 1,
|
|
'tax_ids': False,
|
|
'price_subtotal': 10.00,
|
|
'price_subtotal_incl': 10.00,
|
|
})],
|
|
'pricelist_id': not_available_pricelist.id,
|
|
'amount_paid': 10.00,
|
|
'amount_total': 10.00,
|
|
'amount_tax': 0.0,
|
|
'amount_return': 0.0,
|
|
'to_invoice': False,
|
|
'pos_reference': 'Test/0001',
|
|
})
|
|
order.action_pos_order_paid()
|
|
|
|
self.start_tour(f"/pos/ui?config_id={self.main_pos_config.id}", 'test_not_available_pricelist_not_set_on_order', login="pos_user")
|
|
|
|
created_order = self.env['pos.order'].search([('partner_id', '=', partner.id)], limit=1)
|
|
self.assertNotEqual(created_order.pricelist_id, not_available_pricelist)
|
|
|
|
def test_payment_screen_tip_scenario(self):
|
|
self.main_pos_config.write({
|
|
'iface_tipproduct': True,
|
|
'tip_product_id': self.tip.id,
|
|
})
|
|
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'test_payment_screen_tip_scenario', login="pos_user")
|
|
|
|
|
|
# This class just runs the same tests as above but with mobile emulation
|
|
class MobileTestUi(TestUi):
|
|
browser_size = '375x667'
|
|
touch_enabled = True
|
|
allow_inherited_tests_method = True
|
|
|
|
|
|
class TestTaxCommonPOS(TestPointOfSaleHttpCommon, TestTaxCommon):
|
|
@classmethod
|
|
def setUpClass(cls):
|
|
super().setUpClass()
|
|
cls.partner_a.name = "AAAAAA" # The POS only load the first 100 partners
|
|
|
|
def create_base_line_product(self, base_line, **kwargs):
|
|
return self.env['product.product'].create({
|
|
**kwargs,
|
|
'available_in_pos': True,
|
|
'list_price': base_line['price_unit'],
|
|
'taxes_id': [Command.set(base_line['tax_ids'].ids)],
|
|
'pos_categ_ids': [Command.set(self.pos_desk_misc_test.ids)],
|
|
})
|
|
|
|
def ensure_products_on_document(self, document, product_prefix):
|
|
for i, base_line in enumerate(document['lines'], start=1):
|
|
base_line['product_id'] = self.create_base_line_product(base_line, name=f'{product_prefix}_{i}')
|
|
|
|
def assert_pos_order_totals(self, order, expected_values):
|
|
expected_amounts = {}
|
|
if 'tax_amount_currency' in expected_values:
|
|
expected_amounts['amount_tax'] = expected_values['tax_amount_currency']
|
|
if 'total_amount_currency' in expected_values:
|
|
expected_amounts['amount_total'] = expected_values['total_amount_currency']
|
|
self.assertRecordValues(order, [expected_amounts])
|
|
|
|
def assert_pos_orders_and_invoices(self, tour, tests_with_orders):
|
|
if self.main_pos_config.current_session_id:
|
|
self.main_pos_config.current_session_id.post_closing_cash_details(0)
|
|
self.main_pos_config.current_session_id.close_session_from_ui()
|
|
|
|
self.start_pos_tour(tour)
|
|
orders = self.env['pos.order'].search([('session_id', '=', self.main_pos_config.current_session_id.id)], limit=len(tests_with_orders))
|
|
for index, (order, (test_code, _document, _soft_checking, _amount_type, _amount, expected_values)) in enumerate(zip(orders, tests_with_orders)):
|
|
with self.subTest(test_code=test_code, index=index):
|
|
self.assert_pos_order_totals(order, expected_values)
|
|
if order.account_move:
|
|
self.assert_invoice_totals(order.account_move, expected_values)
|