diff --git a/scrap_reason_code/README.rst b/scrap_reason_code/README.rst new file mode 100644 index 000000000..b50801f17 --- /dev/null +++ b/scrap_reason_code/README.rst @@ -0,0 +1,109 @@ +================= +Scrap Reason Code +================= + +.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png + :target: https://odoo-community.org/page/development-status + :alt: Beta +.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png + :target: http://www.gnu.org/licenses/agpl-3.0-standalone.html + :alt: License: AGPL-3 +.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fstock--logistics--warehouse-lightgray.png?logo=github + :target: https://github.com/OCA/stock-logistics-warehouse/tree/15.0/scrap_reason_code + :alt: OCA/stock-logistics-warehouse +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/stock-logistics-warehouse-15-0/stock-logistics-warehouse-15-0-scrap_reason_code + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png + :target: https://runbot.odoo-community.org/runbot/153/15.0 + :alt: Try me on Runbot + +|badge1| |badge2| |badge3| |badge4| |badge5| + +Adds a reason code for scrapping operations and an interface for the user +to create scrap codes + +**Table of contents** + +.. contents:: + :local: + +Configuration +============= + +Go to Inventory > Configuration > Scrap Reason Codes + +Create a required scrap reason code and provide scrap location. + +Usage +===== + +- Go to Inventory > Operations > Scrap +- Create a scarp order and select reason code. +- A scrap location will be readonly and auto fill based on selected reason + code. + +Bug Tracker +=========== + +Bugs are tracked on `GitHub Issues `_. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us smashing it by providing a detailed and welcomed +`feedback `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +~~~~~~~ + +* Open Source Integrators + +Contributors +~~~~~~~~~~~~ + +* Michael Allen +* Bhavesh Odedra +* Balaji Kannan +* Serpent Consulting Services Pvt. Ltd. +* Chandresh Thakkar +* Hughes Damry + +Other credits +~~~~~~~~~~~~~ + +The development of this module has been financially supported by: + +* Open Source Integrators + +Maintainers +~~~~~~~~~~~ + +This module is maintained by the OCA. + +.. image:: https://odoo-community.org/logo.png + :alt: Odoo Community Association + :target: https://odoo-community.org + +OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use. + +.. |maintainer-bodedra| image:: https://github.com/bodedra.png?size=40px + :target: https://github.com/bodedra + :alt: bodedra + +Current `maintainer `__: + +|maintainer-bodedra| + +This module is part of the `OCA/stock-logistics-warehouse `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/scrap_reason_code/__init__.py b/scrap_reason_code/__init__.py new file mode 100644 index 000000000..69f7babdf --- /dev/null +++ b/scrap_reason_code/__init__.py @@ -0,0 +1,3 @@ +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import models diff --git a/scrap_reason_code/__manifest__.py b/scrap_reason_code/__manifest__.py new file mode 100644 index 000000000..6ddffce50 --- /dev/null +++ b/scrap_reason_code/__manifest__.py @@ -0,0 +1,20 @@ +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +{ + "name": "Scrap Reason Code", + "version": "16.0.1.0.1", + "license": "AGPL-3", + "summary": "Reason code for scrapping", + "author": "Open Source Integrators, Odoo Community Association (OCA)", + "website": "https://github.com/OCA/stock-logistics-warehouse", + "category": "Warehouse Management", + "depends": ["stock"], + "data": [ + "security/ir.model.access.csv", + "views/reason_code_view.xml", + "views/stock_scrap_views.xml", + "views/stock_move_views.xml", + ], + "maintainers": ["bodedra"], + "installable": True, +} diff --git a/scrap_reason_code/i18n/es.po b/scrap_reason_code/i18n/es.po new file mode 100644 index 000000000..de725fbf3 --- /dev/null +++ b/scrap_reason_code/i18n/es.po @@ -0,0 +1,108 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * scrap_reason_code +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 14.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2022-03-25 20:17+0000\n" +"Last-Translator: David Alonso (Solvos) \n" +"Language-Team: none\n" +"Language: es\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=2; plural=n != 1;\n" +"X-Generator: Weblate 4.3.2\n" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__name +msgid "Code" +msgstr "Código" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_uid +msgid "Created by" +msgstr "Creado por" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_date +msgid "Created on" +msgstr "Creado en" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__description +msgid "Description" +msgstr "Descripción" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__display_name +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move__display_name +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__display_name +msgid "Display Name" +msgstr "Nombre a mostrar" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__id +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move__id +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__id +msgid "ID" +msgstr "Id." + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code____last_update +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move____last_update +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap____last_update +msgid "Last Modified on" +msgstr "Últ. modificación en" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_uid +msgid "Last Updated by" +msgstr "Últ. modificación por" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_date +msgid "Last Updated on" +msgstr "Últ. actualización en" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__reason_code_id +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.stock_reason_code_form +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.view_scrap_reason_code_form +msgid "Reason Code" +msgstr "Código del motivo" + +#. module: scrap_reason_code +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.view_scrap_reason_code_list +msgid "Reason Codes" +msgstr "Códigos de motivo" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move__reason_code_id +msgid "Reason code" +msgstr "Código de motivo" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_scrap +msgid "Scrap" +msgstr "Desecho" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__location_id +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__scrap_location_id +msgid "Scrap Location" +msgstr "Ubicación de desecho" + +#. module: scrap_reason_code +#: model:ir.actions.act_window,name:scrap_reason_code.open_view_scrap_reason_code_form +#: model:ir.ui.menu,name:scrap_reason_code.menu_view_scrap_reason_code_form +msgid "Scrap Reason Codes" +msgstr "Códigos de motivo de desecho" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_move +msgid "Stock Move" +msgstr "Movimiento de almacén" diff --git a/scrap_reason_code/i18n/fr.po b/scrap_reason_code/i18n/fr.po new file mode 100644 index 000000000..a6ccecaae --- /dev/null +++ b/scrap_reason_code/i18n/fr.po @@ -0,0 +1,108 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * scrap_reason_code +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 14.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2021-05-12 08:47+0000\n" +"Last-Translator: Rémi \n" +"Language-Team: none\n" +"Language: fr\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=2; plural=n > 1;\n" +"X-Generator: Weblate 4.3.2\n" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__name +msgid "Code" +msgstr "Code" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_uid +msgid "Created by" +msgstr "Créateur" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_date +msgid "Created on" +msgstr "Date de création" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__description +msgid "Description" +msgstr "Description" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__display_name +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move__display_name +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__display_name +msgid "Display Name" +msgstr "Nom affiché" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__id +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move__id +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__id +msgid "ID" +msgstr "ID" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code____last_update +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move____last_update +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap____last_update +msgid "Last Modified on" +msgstr "Date de dernière modification" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_uid +msgid "Last Updated by" +msgstr "Dernière modification par" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_date +msgid "Last Updated on" +msgstr "Date de dernière modification" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__reason_code_id +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.stock_reason_code_form +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.view_scrap_reason_code_form +msgid "Reason Code" +msgstr "Motif" + +#. module: scrap_reason_code +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.view_scrap_reason_code_list +msgid "Reason Codes" +msgstr "Motifs" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move__reason_code_id +msgid "Reason code" +msgstr "Motif" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_scrap +msgid "Scrap" +msgstr "Rebut" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__location_id +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__scrap_location_id +msgid "Scrap Location" +msgstr "Emplacement de Rebut" + +#. module: scrap_reason_code +#: model:ir.actions.act_window,name:scrap_reason_code.open_view_scrap_reason_code_form +#: model:ir.ui.menu,name:scrap_reason_code.menu_view_scrap_reason_code_form +msgid "Scrap Reason Codes" +msgstr "Motifs de Rebut" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_move +msgid "Stock Move" +msgstr "Opération" diff --git a/scrap_reason_code/i18n/scrap_reason_code.pot b/scrap_reason_code/i18n/scrap_reason_code.pot new file mode 100644 index 000000000..e6023fc27 --- /dev/null +++ b/scrap_reason_code/i18n/scrap_reason_code.pot @@ -0,0 +1,94 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * scrap_reason_code +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 15.0\n" +"Report-Msgid-Bugs-To: \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: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__name +msgid "Code" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_uid +msgid "Created by" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_date +msgid "Created on" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__description +msgid "Description" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__display_name +msgid "Display Name" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__id +msgid "ID" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code____last_update +msgid "Last Modified on" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_uid +msgid "Last Updated by" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_date +msgid "Last Updated on" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__reason_code_id +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.stock_reason_code_form +#: model_terms:ir.ui.view,arch_db:scrap_reason_code.view_scrap_reason_code_form +msgid "Reason Code" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_move__reason_code_id +msgid "Reason code" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_scrap +msgid "Scrap" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__location_id +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__scrap_location_id +msgid "Scrap Location" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.actions.act_window,name:scrap_reason_code.open_view_scrap_reason_code_form +#: model:ir.ui.menu,name:scrap_reason_code.menu_view_scrap_reason_code_form +msgid "Scrap Reason Codes" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_move +msgid "Stock Move" +msgstr "" diff --git a/scrap_reason_code/models/__init__.py b/scrap_reason_code/models/__init__.py new file mode 100644 index 000000000..b00bb13c9 --- /dev/null +++ b/scrap_reason_code/models/__init__.py @@ -0,0 +1,5 @@ +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import reason_code +from . import stock_move +from . import stock_scrap diff --git a/scrap_reason_code/models/reason_code.py b/scrap_reason_code/models/reason_code.py new file mode 100644 index 000000000..a412f79f5 --- /dev/null +++ b/scrap_reason_code/models/reason_code.py @@ -0,0 +1,18 @@ +# Copyright (C) 2019 IBM Corp. +# Copyright (C) 2019 Open Source Integrators +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import fields, models + + +class ScrapReasonCode(models.Model): + _name = "scrap.reason.code" + _description = "Reason Code" + + name = fields.Char("Code", required=True) + description = fields.Text() + location_id = fields.Many2one( + "stock.location", + string="Scrap Location", + domain="[('scrap_location', '=', True)]", + ) diff --git a/scrap_reason_code/models/stock_move.py b/scrap_reason_code/models/stock_move.py new file mode 100644 index 000000000..db41da403 --- /dev/null +++ b/scrap_reason_code/models/stock_move.py @@ -0,0 +1,11 @@ +# Copyright (C) 2019 IBM Corp. +# Copyright (C) 2019 Open Source Integrators +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import fields, models + + +class StockMove(models.Model): + _inherit = "stock.move" + + reason_code_id = fields.Many2one("scrap.reason.code") diff --git a/scrap_reason_code/models/stock_scrap.py b/scrap_reason_code/models/stock_scrap.py new file mode 100644 index 000000000..6f00f178c --- /dev/null +++ b/scrap_reason_code/models/stock_scrap.py @@ -0,0 +1,44 @@ +# Copyright (C) 2019 IBM Corp. +# Copyright (C) 2019 Open Source Integrators +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import api, fields, models + + +class StockScrap(models.Model): + _inherit = "stock.scrap" + + reason_code_id = fields.Many2one( + "scrap.reason.code", states={"done": [("readonly", True)]} + ) + scrap_location_id = fields.Many2one(readonly=True) + + def _prepare_move_values(self): + res = super(StockScrap, self)._prepare_move_values() + res["reason_code_id"] = self.reason_code_id.id + return res + + @api.onchange("reason_code_id") + def _onchange_reason_code_id(self): + if self.reason_code_id.location_id: + self.scrap_location_id = self.reason_code_id.location_id + + def _update_scrap_reason_code_location(self, vals): + if "reason_code_id" in vals: + location_id = ( + self.env["scrap.reason.code"] + .browse(vals.get("reason_code_id")) + .location_id.id + ) + if location_id: + vals.update({"scrap_location_id": location_id}) + + def write(self, vals): + self._update_scrap_reason_code_location(vals) + return super(StockScrap, self).write(vals) + + @api.model_create_multi + def create(self, vals_list): + for vals in vals_list: + self._update_scrap_reason_code_location(vals) + return super().create(vals_list) diff --git a/scrap_reason_code/readme/CONFIGURE.rst b/scrap_reason_code/readme/CONFIGURE.rst new file mode 100644 index 000000000..7e06896eb --- /dev/null +++ b/scrap_reason_code/readme/CONFIGURE.rst @@ -0,0 +1,3 @@ +Go to Inventory > Configuration > Scrap Reason Codes + +Create a required scrap reason code and provide scrap location. diff --git a/scrap_reason_code/readme/CONTRIBUTORS.rst b/scrap_reason_code/readme/CONTRIBUTORS.rst new file mode 100644 index 000000000..ad224128e --- /dev/null +++ b/scrap_reason_code/readme/CONTRIBUTORS.rst @@ -0,0 +1,6 @@ +* Michael Allen +* Bhavesh Odedra +* Balaji Kannan +* Serpent Consulting Services Pvt. Ltd. +* Chandresh Thakkar +* Hughes Damry diff --git a/scrap_reason_code/readme/CREDITS.rst b/scrap_reason_code/readme/CREDITS.rst new file mode 100644 index 000000000..8209266d9 --- /dev/null +++ b/scrap_reason_code/readme/CREDITS.rst @@ -0,0 +1,3 @@ +The development of this module has been financially supported by: + +* Open Source Integrators diff --git a/scrap_reason_code/readme/DESCRIPTION.rst b/scrap_reason_code/readme/DESCRIPTION.rst new file mode 100644 index 000000000..6905032da --- /dev/null +++ b/scrap_reason_code/readme/DESCRIPTION.rst @@ -0,0 +1,2 @@ +Adds a reason code for scrapping operations and an interface for the user +to create scrap codes diff --git a/scrap_reason_code/readme/ROADMAP.rst b/scrap_reason_code/readme/ROADMAP.rst new file mode 100644 index 000000000..e69de29bb diff --git a/scrap_reason_code/readme/USAGE.rst b/scrap_reason_code/readme/USAGE.rst new file mode 100644 index 000000000..b2ebead93 --- /dev/null +++ b/scrap_reason_code/readme/USAGE.rst @@ -0,0 +1,4 @@ +- Go to Inventory > Operations > Scrap +- Create a scarp order and select reason code. +- A scrap location will be readonly and auto fill based on selected reason + code. diff --git a/scrap_reason_code/security/ir.model.access.csv b/scrap_reason_code/security/ir.model.access.csv new file mode 100644 index 000000000..bb3255b27 --- /dev/null +++ b/scrap_reason_code/security/ir.model.access.csv @@ -0,0 +1,4 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_scrap_reason_code_user,scrap.reason.code,model_scrap_reason_code,base.group_user,1,0,0,0 +access_scrap_reason_code_stock_user,scrap.reason.code,model_scrap_reason_code,stock.group_stock_user,1,1,0,0 +access_scrap_reason_code_stock_manager,scrap.reason.code,model_scrap_reason_code,stock.group_stock_manager,1,1,1,1 diff --git a/scrap_reason_code/static/description/icon.png b/scrap_reason_code/static/description/icon.png new file mode 100644 index 000000000..3a0328b51 Binary files /dev/null and b/scrap_reason_code/static/description/icon.png differ diff --git a/scrap_reason_code/static/description/index.html b/scrap_reason_code/static/description/index.html new file mode 100644 index 000000000..4bdfaab38 --- /dev/null +++ b/scrap_reason_code/static/description/index.html @@ -0,0 +1,451 @@ + + + + + + +Scrap Reason Code + + + +
+

Scrap Reason Code

+ + +

Beta License: AGPL-3 OCA/stock-logistics-warehouse Translate me on Weblate Try me on Runbot

+

Adds a reason code for scrapping operations and an interface for the user +to create scrap codes

+

Table of contents

+ +
+

Configuration

+

Go to Inventory > Configuration > Scrap Reason Codes

+

Create a required scrap reason code and provide scrap location.

+
+
+

Usage

+
    +
  • Go to Inventory > Operations > Scrap
  • +
  • Create a scarp order and select reason code.
  • +
  • A scrap location will be readonly and auto fill based on selected reason +code.
  • +
+
+
+

Bug Tracker

+

Bugs are tracked on GitHub Issues. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us smashing it by providing a detailed and welcomed +feedback.

+

Do not contact contributors directly about support or help with technical issues.

+
+
+

Credits

+
+

Authors

+
    +
  • Open Source Integrators
  • +
+
+
+

Contributors

+ +
+
+

Other credits

+

The development of this module has been financially supported by:

+
    +
  • Open Source Integrators
  • +
+
+
+

Maintainers

+

This module is maintained by the OCA.

+Odoo Community Association +

OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use.

+

Current maintainer:

+

bodedra

+

This module is part of the OCA/stock-logistics-warehouse project on GitHub.

+

You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

+
+
+
+ + diff --git a/scrap_reason_code/tests/__init__.py b/scrap_reason_code/tests/__init__.py new file mode 100644 index 000000000..29507951a --- /dev/null +++ b/scrap_reason_code/tests/__init__.py @@ -0,0 +1,3 @@ +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import test_scrap_reason_code diff --git a/scrap_reason_code/tests/test_scrap_reason_code.py b/scrap_reason_code/tests/test_scrap_reason_code.py new file mode 100644 index 000000000..27851bf8a --- /dev/null +++ b/scrap_reason_code/tests/test_scrap_reason_code.py @@ -0,0 +1,166 @@ +# Copyright (C) 2019 IBM Corp. +# Copyright (C) 2019 Open Source Integrators +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo.tests.common import TransactionCase + + +class StockScrap(TransactionCase): + def setUp(self): + super(StockScrap, self).setUp() + + self.stock_location = self.env.ref("stock.stock_location_stock") + self.customer_location = self.env.ref("stock.stock_location_customers") + stock_location_locations_virtual = self.env["stock.location"].create( + {"name": "Virtual Locations", "usage": "view", "posz": 1} + ) + self.scrapped_location = self.env["stock.location"].create( + { + "name": "Scrapped", + "location_id": stock_location_locations_virtual.id, + "scrap_location": True, + "usage": "inventory", + } + ) + + self.scrap_product = self.env["product.product"].create( + { + "name": "Scrap Product A", + "type": "product", + "categ_id": self.env.ref("product.product_category_all").id, + } + ) + + self.reason_code = self.env["scrap.reason.code"].create( + { + "name": "DM300", + "description": "Product is damage", + "location_id": self.scrapped_location.id, + } + ) + + self.uom_unit = self.env.ref("uom.product_uom_unit") + + def test_scrap_reason_code(self): + """Scrap the product of a picking. Then modify the + done linked stock move and ensure the scrap quantity is also + updated and verify scrap reason code + """ + self.env["stock.quant"]._update_available_quantity( + self.scrap_product, self.stock_location, 10 + ) + partner = self.env["res.partner"].create({"name": "BOdedra"}) + picking = self.env["stock.picking"].create( + { + "name": "A single picking with one move to scrap", + "location_id": self.stock_location.id, + "location_dest_id": self.customer_location.id, + "partner_id": partner.id, + "picking_type_id": self.env.ref("stock.picking_type_out").id, + } + ) + move1 = self.env["stock.move"].create( + { + "name": "A move to confirm and scrap its product", + "location_id": self.stock_location.id, + "location_dest_id": self.customer_location.id, + "product_id": self.scrap_product.id, + "product_uom": self.uom_unit.id, + "product_uom_qty": 1.0, + "picking_id": picking.id, + } + ) + move1._action_confirm() + + self.assertEqual(move1.state, "assigned") + scrap = self.env["stock.scrap"].create( + { + "product_id": self.scrap_product.id, + "product_uom_id": self.scrap_product.uom_id.id, + "scrap_qty": 5, + "picking_id": picking.id, + "reason_code_id": self.reason_code.id, + } + ) + scrap._onchange_reason_code_id() + scrap.do_scrap() + self.assertEqual(len(picking.move_ids), 2) + scrapped_move = picking.move_ids.filtered(lambda m: m.state == "done") + self.assertTrue(scrapped_move, "No scrapped move created.") + self.assertEqual( + scrapped_move.scrap_ids.ids, [scrap.id], "Wrong scrap linked to the move." + ) + self.assertEqual( + scrap.scrap_qty, + 5, + "Scrap quantity has been modified and is not " "correct anymore.", + ) + move = scrap.move_id + self.assertEqual(move.reason_code_id.id, self.reason_code.id) + + scrapped_move.quantity_done = 8 + self.assertEqual(scrap.scrap_qty, 8, "Scrap quantity is not updated.") + + def test_scrap_reason_code_write(self): + """Scrap the product of a picking2. Then modify the + done linked stock move and ensure the scrap quantity is also + updated and verify scrap reason code + """ + self.env["stock.quant"]._update_available_quantity( + self.scrap_product, self.stock_location, 10 + ) + partner2 = self.env["res.partner"].create({"name": "BOdedra 2"}) + picking2 = self.env["stock.picking"].create( + { + "name": "A single picking with one move to scrap 2", + "location_id": self.stock_location.id, + "location_dest_id": self.customer_location.id, + "partner_id": partner2.id, + "picking_type_id": self.env.ref("stock.picking_type_out").id, + } + ) + move2 = self.env["stock.move"].create( + { + "name": "A move to confirm and scrap its product", + "location_id": self.stock_location.id, + "location_dest_id": self.customer_location.id, + "product_id": self.scrap_product.id, + "product_uom": self.uom_unit.id, + "product_uom_qty": 1.0, + "picking_id": picking2.id, + } + ) + move2._action_confirm() + + self.assertEqual(move2.state, "assigned") + scrap2 = self.env["stock.scrap"].create( + { + "product_id": self.scrap_product.id, + "product_uom_id": self.scrap_product.uom_id.id, + "scrap_qty": 5, + "picking_id": picking2.id, + } + ) + scrap2.write( + { + "reason_code_id": self.reason_code.id, + } + ) + scrap2._onchange_reason_code_id() + scrap2.do_scrap() + self.assertEqual(len(picking2.move_ids), 2) + scrapped_move = picking2.move_ids.filtered(lambda m: m.state == "done") + self.assertTrue(scrapped_move, "No scrapped move created.") + self.assertEqual( + scrapped_move.scrap_ids.ids, [scrap2.id], "Wrong scrap linked to the move." + ) + self.assertEqual( + scrap2.scrap_qty, + 5, + "Scrap quantity has been modified and is not " "correct anymore.", + ) + move = scrap2.move_id + self.assertEqual(move.reason_code_id.id, self.reason_code.id) + + scrapped_move.quantity_done = 8 + self.assertEqual(scrap2.scrap_qty, 8, "Scrap quantity is not updated.") diff --git a/scrap_reason_code/views/reason_code_view.xml b/scrap_reason_code/views/reason_code_view.xml new file mode 100644 index 000000000..478614e91 --- /dev/null +++ b/scrap_reason_code/views/reason_code_view.xml @@ -0,0 +1,45 @@ + + + + + scrap.reason.code.form + scrap.reason.code + +
+ + + + + + + + + +
+
+
+ + scrap.reason.code.list + scrap.reason.code + + + + + + + + + + Scrap Reason Codes + scrap.reason.code + tree,form + + +
diff --git a/scrap_reason_code/views/stock_move_views.xml b/scrap_reason_code/views/stock_move_views.xml new file mode 100644 index 000000000..a2a77af18 --- /dev/null +++ b/scrap_reason_code/views/stock_move_views.xml @@ -0,0 +1,20 @@ + + + + stock.reason.code.form + stock.move + + + + + + + + + + + diff --git a/scrap_reason_code/views/stock_scrap_views.xml b/scrap_reason_code/views/stock_scrap_views.xml new file mode 100644 index 000000000..4b60ee1f3 --- /dev/null +++ b/scrap_reason_code/views/stock_scrap_views.xml @@ -0,0 +1,34 @@ + + + + stock.scrap.form.reason.code + stock.scrap + + + + + + + + + stock.scrap.tree.reason.code + stock.scrap + + + + + + + + + stock.scrap.form2.reason.code + stock.scrap + + + + + + + + diff --git a/setup/scrap_reason_code/odoo/addons/scrap_reason_code b/setup/scrap_reason_code/odoo/addons/scrap_reason_code new file mode 120000 index 000000000..0ed8b62d4 --- /dev/null +++ b/setup/scrap_reason_code/odoo/addons/scrap_reason_code @@ -0,0 +1 @@ +../../../../scrap_reason_code \ No newline at end of file diff --git a/setup/scrap_reason_code/setup.py b/setup/scrap_reason_code/setup.py new file mode 100644 index 000000000..28c57bb64 --- /dev/null +++ b/setup/scrap_reason_code/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +)