diff --git a/addons/sales_discount/__init__.py b/addons/sales_discount/__init__.py new file mode 100644 index 00000000..14d55858 --- /dev/null +++ b/addons/sales_discount/__init__.py @@ -0,0 +1,4 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from . import models +from . import report diff --git a/addons/sales_discount/__manifest__.py b/addons/sales_discount/__manifest__.py new file mode 100644 index 00000000..2720a13c --- /dev/null +++ b/addons/sales_discount/__manifest__.py @@ -0,0 +1,27 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +{ + 'name': "Sale Discount", + 'category': 'Sale', + 'description': """ + Customized module for amending discounts + """, + 'summary': 'Global Discount on Sale Order', + 'author': 'FlectraHQ', + 'website': 'https://flectrahq.com', + 'version': '1.0', + 'depends': ['sale_management', 'sale_stock', 'account_discount'], + 'data': [ + 'security/ir.model.access.csv', + 'data/sale_discount_data.xml', + 'views/res_config_settings_views.xml', + 'views/sale_discount_config_view.xml', + 'views/sale_view.xml', + 'report/sale_order_report_view.xml', + ], + 'demo': [ + 'demo/sale_order.xml', + ], + 'installable': True, + 'auto_install': False, +} diff --git a/addons/sales_discount/data/sale_discount_data.xml b/addons/sales_discount/data/sale_discount_data.xml new file mode 100644 index 00000000..b7b15e7b --- /dev/null +++ b/addons/sales_discount/data/sale_discount_data.xml @@ -0,0 +1,22 @@ + + + + + + 1000 + 25 + + + + + 500 + 15 + + + + + 100 + 10 + + + diff --git a/addons/sales_discount/demo/sale_order.xml b/addons/sales_discount/demo/sale_order.xml new file mode 100644 index 00000000..e32a097f --- /dev/null +++ b/addons/sales_discount/demo/sale_order.xml @@ -0,0 +1,36 @@ + + + + + + per + 10 + + + + draft + + + + + + + Headset USB + + 10 + + 65 + 10 + + + + + Laptop E5023 + + 20 + + 2950.00 + 10 + + + diff --git a/addons/sales_discount/i18n/sales_discount.pot b/addons/sales_discount/i18n/sales_discount.pot new file mode 100644 index 00000000..dd5c766e --- /dev/null +++ b/addons/sales_discount/i18n/sales_discount.pot @@ -0,0 +1,247 @@ +# Translation of Flectra Server. +# This file contains the translation of the following modules: +# * sales_discount +# +msgid "" +msgstr "" +"Project-Id-Version: Flectra Server 1.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2018-01-04 11:42+0000\n" +"PO-Revision-Date: 2018-01-04 11:42+0000\n" +"Last-Translator: <>\n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.inherit_view_order_form +msgid "(Update Discount)" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.inherit_view_order_form +msgid "(update)" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.custom_report_sale_layout +msgid "Discount" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.custom_report_sale_layout +msgid "Gross Amount" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.custom_report_sale_layout +msgid "Taxes" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.custom_report_sale_layout +msgid "Total" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.custom_report_sale_layout +msgid "Untaxed Amount" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.res_config_settings_view_discount_form +msgid "Allow global discount on the sale order" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale_discount_config.py:21 +#, python-format +msgid "Assigned group already exist!" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_create_uid +msgid "Created by" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_create_date +msgid "Created on" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_order_discount +#: model:ir.ui.view,arch_db:sales_discount.inherit_view_order_form +msgid "Discount" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_order_discount_per +msgid "Discount (%)" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale.py:121 +#, python-format +msgid "Discount (%s) should be less than Gross Amount (%s)." +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_order_discount_amount +msgid "Discount Amount" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_order_discount_method +msgid "Discount Method" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale.py:118 +#, python-format +msgid "Discount should be less than Gross Amount" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_display_name +msgid "Display Name" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_res_config_settings_global_discount_apply +msgid "Do you want to set global discount limit?" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_res_config_settings_global_discount_fix_amount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_fix_amount +msgid "Fix Amount" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale_discount_config.py:30 +#, python-format +msgid "Fix amount (%s) is greater than configuration Amount (%s)!" +msgstr "" + +#. module: sales_discount +#: selection:sale.order,discount_method:0 +msgid "Fixed" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_res_config_settings_global_discount_per_so_invoice_line +msgid "Global Discounts" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_order_gross_amount +msgid "Gross Amount" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_group_id +msgid "Groups" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_id +msgid "ID" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.inherit_view_order_form +msgid "Invoices" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config___last_update +msgid "Last Modified on" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_write_uid +msgid "Last Updated by" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_write_date +msgid "Last Updated on" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_sale_discount_config_percentage +#: selection:sale.order,discount_method:0 +msgid "Percentage" +msgstr "" + +#. module: sales_discount +#: model:ir.model.fields,field_description:sales_discount.field_res_config_settings_global_discount_percentage +msgid "Percentage (%)" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale_discount_config.py:44 +#, python-format +msgid "Percentage (%s) is greater than configuration Percentage (%s)!" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale.py:94 +#, python-format +msgid "Percentage should be between 0% to 100%" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale_discount_config.py:38 +#, python-format +msgid "Percentage should be between 0% to 100%!" +msgstr "" + +#. module: sales_discount +#: model:ir.model,name:sales_discount.model_sale_order +msgid "Quotation" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.sale_discount_config_view_form +msgid "Sale Discount Configuration" +msgstr "" + +#. module: sales_discount +#: model:ir.ui.view,arch_db:sales_discount.res_config_settings_view_discount_form +msgid "Sale Order Discount" +msgstr "" + +#. module: sales_discount +#: model:ir.actions.act_window,name:sales_discount.action_sale_discount_config +#: model:ir.ui.menu,name:sales_discount.menu_sale_order +msgid "Sales Discounts" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale.py:97 +#: code:addons/sales_discount/models/sale.py:106 +#, python-format +msgid "You are not allowed to apply Discount Percentage (%s) more than configured Discount Percentage (%s) in configuration setting!" +msgstr "" + +#. module: sales_discount +#: code:addons/sales_discount/models/sale.py:127 +#: code:addons/sales_discount/models/sale.py:136 +#, python-format +msgid "You're not allowed to apply Discount Amount (%s) more than configured amount (%s) in configuration setting!" +msgstr "" + +#. module: sales_discount +#: model:ir.model,name:sales_discount.model_res_config_settings +msgid "res.config.settings" +msgstr "" + +#. module: sales_discount +#: model:ir.model,name:sales_discount.model_sale_discount_config +msgid "sale.discount.config" +msgstr "" + diff --git a/addons/sales_discount/models/__init__.py b/addons/sales_discount/models/__init__.py new file mode 100644 index 00000000..02acd6a9 --- /dev/null +++ b/addons/sales_discount/models/__init__.py @@ -0,0 +1,5 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from . import res_config_settings +from . import sale_discount_config +from . import sale diff --git a/addons/sales_discount/models/res_config_settings.py b/addons/sales_discount/models/res_config_settings.py new file mode 100644 index 00000000..a84481cd --- /dev/null +++ b/addons/sales_discount/models/res_config_settings.py @@ -0,0 +1,61 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from flectra import api, fields, models + + +class ResConfigSettings(models.TransientModel): + _inherit = 'res.config.settings' + + global_discount_per_so_invoice_line = fields.Boolean( + "Global Discounts", + implied_group='sales_discount.global_discount_per_so_invoice_line') + global_discount_apply = fields.Boolean( + "Do you want to set global discount limit?", + implied_group='sales_discount.global_discount_apply') + global_discount_fix_amount = fields.Integer( + 'Fix Amount', + implied_group='sales_discount.global_discount_apply') + global_discount_percentage = fields.Integer( + 'Percentage (%)', + implied_group='sales_discount.global_discount_percentage') + + @api.onchange('global_discount_per_so_invoice_line') + def onchange_global_discount_per_so_invoice_line(self): + if not self.global_discount_per_so_invoice_line: + self.global_discount_apply = False + + @api.onchange('global_discount_apply') + def onchange_global_discount_apply(self): + if not self.global_discount_apply: + self.global_discount_fix_amount = False + + @api.model + def get_values(self): + res = super(ResConfigSettings, self).get_values() + res.update( + global_discount_per_so_invoice_line=self.env[ + 'ir.config_parameter'].sudo().get_param( + 'global_discount_per_so_invoice_line'), + global_discount_apply=self.env[ + 'ir.config_parameter'].sudo().get_param( + 'global_discount_apply'), + global_discount_fix_amount=int(self.env[ + 'ir.config_parameter'].sudo().get_param( + 'global_discount_fix_amount')), + global_discount_percentage=int(self.env[ + 'ir.config_parameter'].sudo().get_param( + 'global_discount_percentage')), + ) + return res + + def set_values(self): + super(ResConfigSettings, self).set_values() + params = self.env['ir.config_parameter'].sudo() + params.set_param('global_discount_per_so_invoice_line', + self.global_discount_per_so_invoice_line) + params.set_param('global_discount_apply', + self.global_discount_apply) + params.set_param('global_discount_fix_amount', + self.global_discount_fix_amount) + params.set_param('global_discount_percentage', + self.global_discount_percentage) diff --git a/addons/sales_discount/models/sale.py b/addons/sales_discount/models/sale.py new file mode 100644 index 00000000..7a61ad7b --- /dev/null +++ b/addons/sales_discount/models/sale.py @@ -0,0 +1,154 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from flectra import api, fields, models, _ +from flectra.tools.misc import formatLang +from flectra.exceptions import Warning + + +class SaleOrder(models.Model): + _inherit = "sale.order" + + @api.multi + @api.depends('discount_amount', 'discount_per', 'amount_untaxed') + def _get_discount(self): + total_discount = 0.0 + for record in self: + for so_line_id in record.order_line: + total_price = \ + (so_line_id.product_uom_qty * so_line_id.price_unit) + total_discount += (total_price * so_line_id.discount) / 100 + record.discount = record.pricelist_id.currency_id.round(total_discount) + + @api.multi + @api.depends('order_line', 'discount_per', 'discount_amount') + def _get_total_amount(self): + for order_id in self: + order_id.gross_amount = sum( + [line_id.product_uom_qty * + line_id.price_unit for line_id in order_id.order_line]) + + discount_method = fields.Selection( + [('fixed', 'Fixed'), ('per', 'Percentage')], string="Discount Method") + discount_amount = fields.Float(string="Discount Amount") + discount_per = fields.Float(string="Discount (%)") + discount = fields.Monetary( + string='Discount', store=True, readonly=True, compute='_get_discount', + track_visibility='always') + gross_amount = fields.Float(string="Gross Amount", + compute='_get_total_amount', store=True) + + @api.multi + def calculate_discount(self): + self._check_constrains() + for line in self.order_line: + line.write({'discount': 0.0}) + amount_untaxed = self.amount_untaxed + if self.discount_method == 'per': + for line in self.order_line: + line.write({'discount': line.discount + self.discount_per}) + else: + for line in self.order_line: + discount_value_ratio = \ + (self.discount_amount * + line.price_subtotal) / amount_untaxed + discount_per_ratio = \ + (discount_value_ratio * 100) / line.price_subtotal + line.write({'discount': line.discount + discount_per_ratio}) + + @api.onchange('discount_method') + def onchange_discount_method(self): + self.discount_amount = 0.0 + self.discount_per = 0.0 + if self.discount_method and not self.order_line: + raise Warning('No Sale Order Line(s) were found!') + + @api.constrains('discount_per', 'discount_amount', 'order_line') + def _check_constrains(self): + self.onchange_discount_per() + self.onchange_discount_amount() + + @api.multi + def get_maximum_per_amount(self): + sale_dis_config_obj = self.env['sale.discount.config'] + max_percentage = 0 + max_amount = 0 + check_group = False + for groups_id in self.env.user.groups_id: + sale_dis_config_id = \ + sale_dis_config_obj.search([('group_id', '=', groups_id.id)]) + if sale_dis_config_id: + check_group = True + if sale_dis_config_id.percentage > max_percentage: + max_percentage = sale_dis_config_id.percentage + if sale_dis_config_id.fix_amount > max_amount: + max_amount = sale_dis_config_id.fix_amount + return {'max_percentage': max_percentage, + 'max_amount': max_amount, 'check_group': check_group} + + @api.onchange('discount_per') + def onchange_discount_per(self): + values = self.get_maximum_per_amount() + if self.discount_method == 'per' and ( + self.discount_per > 100 or self.discount_per < 0) and \ + values.get('check_group', False): + raise Warning(_("Percentage should be between 0% to 100%")) + if self.discount_per > values.get('max_percentage', False) and \ + values.get('check_group', False): + raise Warning(_("You are not allowed to apply Discount Percentage" + " (%s) more than configured Discount Percentage " + "(%s) in configuration setting!") % ( + formatLang(self.env, self.discount_per, digits=2), + formatLang(self.env, values['max_percentage'], digits=2))) + config_id = self.env['res.config.settings'].search( + [], order='id desc', limit=1) + if config_id and config_id.global_discount_apply: + if config_id.global_discount_percentage < self.discount_per: + raise Warning(_("You are not allowed to apply Discount " + "Percentage (%s) more than configured " + "Discount Percentage (%s) in configuration " + "setting!") % ( + formatLang(self.env, self.discount_per, digits=2), + formatLang(self.env, config_id.global_discount_percentage, + digits=2))) + + @api.onchange('discount_amount') + def onchange_discount_amount(self): + values = self.get_maximum_per_amount() + if self.discount < 0: + raise Warning(_("Discount should be less than Gross Amount")) + discount = self.discount or self.discount_amount + if discount > self.gross_amount: + raise Warning(_("Discount (%s) should be less than " + "Gross Amount (%s).") % ( + formatLang(self.env, discount, digits=2), + formatLang(self.env, self.gross_amount, digits=2))) + if self.discount_amount > values.get('max_amount', False) \ + and values.get('check_group', False): + raise Warning(_("You're not allowed to apply Discount Amount " + "(%s) more than configured amount (%s) in " + "configuration setting!") % ( + formatLang(self.env, self.discount_amount, digits=2), + formatLang(self.env, values['max_amount'], digits=2))) + config_id = self.env['res.config.settings'].search( + [], order='id desc', limit=1) + if config_id and config_id.global_discount_apply: + if config_id.global_discount_fix_amount < self.discount_amount: + raise Warning(_("You're not allowed to apply Discount " + "Amount (%s) more than configured amount " + "(%s) in configuration setting!") % ( + formatLang(self.env, self.discount_amount, digits=2), + formatLang(self.env, config_id.global_discount_fix_amount, + digits=2))) + + @api.multi + def _prepare_invoice(self): + sale_order = self.env['sale.order'].browse( + self._context.get('active_ids', [])) + invoice_vals = super(SaleOrder, self)._prepare_invoice() + invoice_vals.update({ + 'discount_method': sale_order.discount_method, + 'discount_amount': sale_order.discount_amount, + 'discount_per': sale_order.discount_per, + 'discount': sale_order.discount, + }) + return invoice_vals diff --git a/addons/sales_discount/models/sale_discount_config.py b/addons/sales_discount/models/sale_discount_config.py new file mode 100644 index 00000000..55507a16 --- /dev/null +++ b/addons/sales_discount/models/sale_discount_config.py @@ -0,0 +1,49 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from flectra import api, fields, models, _ +from flectra.tools.misc import formatLang + + +class SaleDiscountConfig(models.Model): + _name = 'sale.discount.config' + + group_id = fields.Many2one('res.groups', 'Groups', required=True) + fix_amount = fields.Float('Fix Amount', required=True) + percentage = fields.Float('Percentage', required=True) + + @api.constrains('group_id') + def _check_already_exist(self): + check_name = self.search( + [('id', '!=', self.id), + ('group_id.name', '=', self.group_id.name)]) + if check_name: + raise ValueError( + _("Assigned group already exist!")) + + @api.constrains('fix_amount') + def _check_fix_amount_value(self): + config_id = self.env['res.config.settings'].search( + [], order='id desc', limit=1) + if config_id and config_id.global_discount_apply \ + and config_id.global_discount_fix_amount < self.fix_amount: + raise ValueError( + _("Fix amount (%s) is greater than configuration Amount (%s)!" + ) % (formatLang( + self.env, self.fix_amount, digits=2), formatLang( + self.env, config_id.global_discount_fix_amount, digits=2))) + + @api.constrains('percentage') + def _check_percentage(self): + if self.percentage < 0 or self.percentage > 100: + raise ValueError(_("Percentage should be between 0% to 100%!")) + config_id = self.env[ + 'res.config.settings'].search([], order='id desc', limit=1) + if config_id and config_id.global_discount_apply \ + and config_id.global_discount_percentage < self.percentage: + raise ValueError( + _("Percentage (%s) is greater than configuration Percentage " + "(%s)!") % (formatLang( + self.env, self.percentage, digits=2), + formatLang(self.env, + config_id.global_discount_percentage, + digits=2))) diff --git a/addons/sales_discount/report/sale_order_report_view.xml b/addons/sales_discount/report/sale_order_report_view.xml new file mode 100644 index 00000000..82062614 --- /dev/null +++ b/addons/sales_discount/report/sale_order_report_view.xml @@ -0,0 +1,43 @@ + + + + \ No newline at end of file diff --git a/addons/sales_discount/security/ir.model.access.csv b/addons/sales_discount/security/ir.model.access.csv new file mode 100644 index 00000000..bb67f931 --- /dev/null +++ b/addons/sales_discount/security/ir.model.access.csv @@ -0,0 +1,3 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_sale_discount_config,access_sale_discount_config,model_sale_discount_config,base.group_system,1,1,1,1 +access_sale_discount_config_all,access_sale_discount_config_all,model_sale_discount_config,,1,0,0,0 diff --git a/addons/sales_discount/static/description/icon.png b/addons/sales_discount/static/description/icon.png new file mode 100644 index 00000000..c386810f Binary files /dev/null and b/addons/sales_discount/static/description/icon.png differ diff --git a/addons/sales_discount/tests/__init__.py b/addons/sales_discount/tests/__init__.py new file mode 100644 index 00000000..7da69e87 --- /dev/null +++ b/addons/sales_discount/tests/__init__.py @@ -0,0 +1,5 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from . import discount_common +from . import test_01_sale_order_discount +from . import test_02_so_discount_invoice diff --git a/addons/sales_discount/tests/discount_common.py b/addons/sales_discount/tests/discount_common.py new file mode 100644 index 00000000..eb4dd8c8 --- /dev/null +++ b/addons/sales_discount/tests/discount_common.py @@ -0,0 +1,30 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from flectra.tests.common import TransactionCase + + +class TestDiscountCommon(TransactionCase): + def setUp(self): + super(TestDiscountCommon, self).setUp() + self.config = self.env['res.config.settings'].create({ + 'global_discount_per_so_invoice_line': True, + 'global_discount_apply': True, + 'global_discount_fix_amount': 5000, + 'global_discount_percentage': 50, + }) + self.config.onchange_global_discount_per_so_invoice_line() + self.config.onchange_global_discount_apply() + self.config.set_values() + self.disc_config_1 = self.env['sale.discount.config'].create({ + 'group_id': self.env.ref('sales_team.group_sale_manager').id, + 'fix_amount': 3000.0, + 'percentage': 20.0, + }) + self.partner_id = self.env.ref('base.res_partner_3') + self.partner_invoice_id = self.env.ref('base.res_partner_address_11') + self.user_id = self.env.ref('base.user_root') + self.partner_shipping_id = self.env.ref('base.res_partner_address_11') + self.pricelist_id = self.env.ref('product.list0') + self.product_1 = self.env.ref('product.product_delivery_01') + self.product_uom = self.env.ref('product.product_uom_unit') + self.product_2 = self.env.ref('product.product_product_25') diff --git a/addons/sales_discount/tests/test_01_sale_order_discount.py b/addons/sales_discount/tests/test_01_sale_order_discount.py new file mode 100644 index 00000000..4aa3481e --- /dev/null +++ b/addons/sales_discount/tests/test_01_sale_order_discount.py @@ -0,0 +1,81 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from .discount_common import TestDiscountCommon +import logging + + +class TestSODiscount(TestDiscountCommon): + def setUp(self): + super(TestSODiscount, self).setUp() + + def discount_01_set_fixamount(self): + sale_order = self.env['sale.order'].create({ + 'partner_id': self.partner_id.id, + 'partner_invoice_id': self.partner_invoice_id.id, + 'partner_shipping_id': self.partner_shipping_id.id, + 'pricelist_id': self.pricelist_id.id, + }) + sale_order.onchange_discount_method() + self.env['sale.order.line'].create({ + 'name': self.product_1.name, + 'product_id': self.product_1.id, + 'product_uom_qty': 20, + 'product_uom': self.product_1.uom_id.id, + 'price_unit': self.product_1.list_price, + 'order_id': sale_order.id, + }) + self.env['sale.order.line'].create({ + 'name': self.product_2.name, + 'product_id': self.product_2.id, + 'product_uom_qty': 20, + 'product_uom': self.product_2.uom_id.id, + 'price_unit': self.product_2.list_price, + 'order_id': sale_order.id, + }) + sale_order.write({ + 'discount_method': 'fixed', + 'discount_amount': 100, + }) + self.assertTrue(sale_order, 'Sale Order: no sale order created') + logging.info('Successful: Sale Order Created!') + sale_order.calculate_discount() + return sale_order + + def discount_02_set_percentages(self): + sale_order = self.env['sale.order'].create({ + 'partner_id': self.partner_id.id, + 'partner_invoice_id': self.partner_invoice_id.id, + 'partner_shipping_id': self.partner_shipping_id.id, + 'pricelist_id': self.pricelist_id.id, + }) + sale_order.onchange_discount_method() + sale_order.write({ + 'discount_method': 'per', + 'discount_per': 10, + }) + self.env['sale.order.line'].create({ + 'name': self.product_1.name, + 'product_id': self.product_1.id, + 'product_uom_qty': 10, + 'product_uom': self.product_1.uom_id.id, + 'price_unit': self.product_1.list_price, + 'order_id': sale_order.id, + 'discount': 10, + }) + self.env['sale.order.line'].create({ + 'name': self.product_2.name, + 'product_id': self.product_2.id, + 'product_uom_qty': 20, + 'product_uom': self.product_2.uom_id.id, + 'price_unit': self.product_2.list_price, + 'order_id': sale_order.id, + 'discount': 10, + }) + self.assertTrue(sale_order, 'Sale Order: no sale order created') + logging.info('Successful: Sale Order Created!') + sale_order.calculate_discount() + sale_order.write({ + 'discount_method': 'per', + 'discount_per': 20.0, + }) + return sale_order diff --git a/addons/sales_discount/tests/test_02_so_discount_invoice.py b/addons/sales_discount/tests/test_02_so_discount_invoice.py new file mode 100644 index 00000000..0532209c --- /dev/null +++ b/addons/sales_discount/tests/test_02_so_discount_invoice.py @@ -0,0 +1,17 @@ +# Part of Flectra See LICENSE file for full copyright and licensing details. + +from .test_01_sale_order_discount import TestSODiscount + + +class TestSODiscountInvoice(TestSODiscount): + def setUp(self): + super(TestSODiscountInvoice, self).setUp() + + def test_01_so_dp_fixed_amount(self): + so = self.discount_01_set_fixamount() + so._prepare_invoice() + + def test_02_so_percentage_discount(self): + so = self.discount_02_set_percentages() + so._prepare_invoice() + diff --git a/addons/sales_discount/views/res_config_settings_views.xml b/addons/sales_discount/views/res_config_settings_views.xml new file mode 100644 index 00000000..6ae8dac3 --- /dev/null +++ b/addons/sales_discount/views/res_config_settings_views.xml @@ -0,0 +1,45 @@ + + + + res.config.settings.view.form.inherit.discount.sale + res.config.settings + + + + +

Sale Order Discount

+
+
+
+ +
+
+
+
+
+
+
+
+
diff --git a/addons/sales_discount/views/sale_discount_config_view.xml b/addons/sales_discount/views/sale_discount_config_view.xml new file mode 100644 index 00000000..e1607d9b --- /dev/null +++ b/addons/sales_discount/views/sale_discount_config_view.xml @@ -0,0 +1,26 @@ + + + + sale.discount.config + sale.discount.config + + + + + + + + + + + + Sales Discounts + sale.discount.config + form + tree + + + + diff --git a/addons/sales_discount/views/sale_view.xml b/addons/sales_discount/views/sale_view.xml new file mode 100644 index 00000000..2b7fbf52 --- /dev/null +++ b/addons/sales_discount/views/sale_view.xml @@ -0,0 +1,44 @@ + + + + + sale.order.inherit.form + sale.order + + + + + + + +
+ + + + + +
+
+
+
+