diff --git a/scrap_reason_code/README.rst b/scrap_reason_code/README.rst new file mode 100644 index 000000000000..48d1b51938cf --- /dev/null +++ b/scrap_reason_code/README.rst @@ -0,0 +1,118 @@ +================= +Scrap Reason Code +================= + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:2689e0bf5ae11316d48b7c179a2662765f9f8f553b04c3edb4612711a6338a39 + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |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/17.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-17-0/stock-logistics-warehouse-17-0-scrap_reason_code + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runboat-Try%20me-875A7B.png + :target: https://runboat.odoo-community.org/builds?repo=OCA/stock-logistics-warehouse&target_branch=17.0 + :alt: Try me on Runboat + +|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. + +Known issues / Roadmap +====================== + + + +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 to smash 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 +- Lois Rilo + +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 000000000000..69f7babdfb1a --- /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 000000000000..a644eeeaaf7e --- /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": "17.0.1.0.0", + "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 000000000000..5113ca407914 --- /dev/null +++ b/scrap_reason_code/i18n/es.po @@ -0,0 +1,128 @@ +# 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: 2023-07-27 23:12+0000\n" +"Last-Translator: Ivorra78 \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.17\n" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__product_category_ids +msgid "Allowed Product Categories" +msgstr "Categorías de productos permitidos" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__allowed_reason_code_ids +msgid "Allowed Reason Code" +msgstr "Código de motivo autorizado" + +#. 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 +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 +msgid "ID" +msgstr "Id." + +#. module: scrap_reason_code +#: model:ir.model.fields,help:scrap_reason_code.field_scrap_reason_code__product_category_ids +msgid "" +"Indicate the cateogories of products that can use this reason code when " +"doing a scrap. If left empy, this reason code can be used with any product." +msgstr "" +"Indique las categorías de productos que pueden utilizar este código de " +"motivo al realizar un rechazo. Si se deja vacío, este código de motivo puede " +"utilizarse con cualquier producto." + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code____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 "actualizado por última vez 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_move__reason_code_id +#: 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: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" + +#. module: scrap_reason_code +#. odoo-python +#: code:addons/scrap_reason_code/models/stock_scrap.py:0 +#, python-format +msgid "The selected reason code is not allowed for this product category." +msgstr "" +"El código de motivo seleccionado no está permitido para esta categoría de " +"producto." + +#~ msgid "Reason Codes" +#~ msgstr "Códigos de motivo" + +#~ msgid "Reason code" +#~ msgstr "Código de motivo" diff --git a/scrap_reason_code/i18n/fr.po b/scrap_reason_code/i18n/fr.po new file mode 100644 index 000000000000..aeceaa7f63a7 --- /dev/null +++ b/scrap_reason_code/i18n/fr.po @@ -0,0 +1,126 @@ +# 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: 2023-11-10 15:37+0000\n" +"Last-Translator: LESTRAT21 \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.17\n" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__product_category_ids +msgid "Allowed Product Categories" +msgstr "Catégories d'articles autorisées" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__allowed_reason_code_ids +msgid "Allowed Reason Code" +msgstr "Codes des motifs autorisés" + +#. 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 +msgid "Display Name" +msgstr "Nom affiché" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__id +msgid "ID" +msgstr "ID" + +#. module: scrap_reason_code +#: model:ir.model.fields,help:scrap_reason_code.field_scrap_reason_code__product_category_ids +msgid "" +"Indicate the cateogories of products that can use this reason code when " +"doing a scrap. If left empy, this reason code can be used with any product." +msgstr "" +"Catégories de produits qui peuvent utiliser ce motif de rebut (code). Si le " +"champ est laissé vide, ce motif pourra être utilisé pour tous les produits " +"mis au rebut." + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code____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_move__reason_code_id +#: 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: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" + +#. module: scrap_reason_code +#. odoo-python +#: code:addons/scrap_reason_code/models/stock_scrap.py:0 +#, python-format +msgid "The selected reason code is not allowed for this product category." +msgstr "Le motif choisi n'est pas utilisable avec cette catégorie de produits." + +#~ msgid "Reason Codes" +#~ msgstr "Motifs" + +#~ msgid "Reason code" +#~ msgstr "Motif" diff --git a/scrap_reason_code/i18n/it.po b/scrap_reason_code/i18n/it.po new file mode 100644 index 000000000000..51ca3c92c8cc --- /dev/null +++ b/scrap_reason_code/i18n/it.po @@ -0,0 +1,121 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * scrap_reason_code +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 16.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2023-11-21 13:33+0000\n" +"Last-Translator: mymage \n" +"Language-Team: none\n" +"Language: it\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.17\n" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__product_category_ids +msgid "Allowed Product Categories" +msgstr "Categorie prodotto consentite" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__allowed_reason_code_ids +msgid "Allowed Reason Code" +msgstr "Codice causale consentito" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__name +msgid "Code" +msgstr "Codice" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_uid +msgid "Created by" +msgstr "Creato da" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__create_date +msgid "Created on" +msgstr "Creato il" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__description +msgid "Description" +msgstr "Descrizione" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__display_name +msgid "Display Name" +msgstr "Nome visualizzato" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__id +msgid "ID" +msgstr "ID" + +#. module: scrap_reason_code +#: model:ir.model.fields,help:scrap_reason_code.field_scrap_reason_code__product_category_ids +msgid "" +"Indicate the cateogories of products that can use this reason code when " +"doing a scrap. If left empy, this reason code can be used with any product." +msgstr "" +"Indicaere le categorie prodotti che possono utilizzare questo codice causale " +"quando scartano, Se vuoto, questo codice causale può essere utilizzato da " +"tutti i prodotti." + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code____last_update +msgid "Last Modified on" +msgstr "Ultima modifica il" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_uid +msgid "Last Updated by" +msgstr "Ultimo aggiornamento di" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_scrap_reason_code__write_date +msgid "Last Updated on" +msgstr "Ultimo aggiornamento il" + +#. 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_move__reason_code_id +#: 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 "Codice causale" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_scrap +msgid "Scrap" +msgstr "Scarto" + +#. 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 "Ubicazione scarto" + +#. 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 "Codici causale scarto" + +#. module: scrap_reason_code +#: model:ir.model,name:scrap_reason_code.model_stock_move +msgid "Stock Move" +msgstr "Movimento di magazzino" + +#. module: scrap_reason_code +#. odoo-python +#: code:addons/scrap_reason_code/models/stock_scrap.py:0 +#, python-format +msgid "The selected reason code is not allowed for this product category." +msgstr "" +"Il codice causale selezionato non è consentito per questa categoria prodotto." 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 000000000000..496fd0cc34ba --- /dev/null +++ b/scrap_reason_code/i18n/scrap_reason_code.pot @@ -0,0 +1,114 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * scrap_reason_code +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 16.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__product_category_ids +msgid "Allowed Product Categories" +msgstr "" + +#. module: scrap_reason_code +#: model:ir.model.fields,field_description:scrap_reason_code.field_stock_scrap__allowed_reason_code_ids +msgid "Allowed Reason Code" +msgstr "" + +#. 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,help:scrap_reason_code.field_scrap_reason_code__product_category_ids +msgid "" +"Indicate the cateogories of products that can use this reason code when " +"doing a scrap. If left empy, this reason code can be used with any product." +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_move__reason_code_id +#: 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,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 "" + +#. module: scrap_reason_code +#. odoo-python +#: code:addons/scrap_reason_code/models/stock_scrap.py:0 +#, python-format +msgid "The selected reason code is not allowed for this product category." +msgstr "" diff --git a/scrap_reason_code/models/__init__.py b/scrap_reason_code/models/__init__.py new file mode 100644 index 000000000000..b00bb13c9e48 --- /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 000000000000..af4d517e1e6c --- /dev/null +++ b/scrap_reason_code/models/reason_code.py @@ -0,0 +1,26 @@ +# Copyright (C) 2019 IBM Corp. +# Copyright (C) 2019 Open Source Integrators +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# 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)]", + ) + product_category_ids = fields.Many2many( + string="Allowed Product Categories", + comodel_name="product.category", + help="Indicate the cateogories of products that can use this reason code " + "when doing a scrap. If left empy, this reason code can be used " + "with any product.", + ) diff --git a/scrap_reason_code/models/stock_move.py b/scrap_reason_code/models/stock_move.py new file mode 100644 index 000000000000..db41da403a61 --- /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 000000000000..1e873973560d --- /dev/null +++ b/scrap_reason_code/models/stock_scrap.py @@ -0,0 +1,78 @@ +# Copyright (C) 2019 IBM Corp. +# Copyright (C) 2019 Open Source Integrators +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import _, api, fields, models +from odoo.exceptions import ValidationError + + +class StockScrap(models.Model): + _inherit = "stock.scrap" + + reason_code_id = fields.Many2one( + comodel_name="scrap.reason.code", + domain="[('id', 'in', allowed_reason_code_ids)]", + ) + allowed_reason_code_ids = fields.Many2many( + comodel_name="scrap.reason.code", + compute="_compute_allowed_reason_code_ids", + ) + scrap_location_id = fields.Many2one(readonly=True) + + @api.depends("product_id", "product_id.categ_id") + def _compute_allowed_reason_code_ids(self): + for rec in self: + codes = self.env["scrap.reason.code"] + if rec.product_id: + codes = codes.search( + [ + "|", + ("product_category_ids", "=", False), + ("product_category_ids", "in", rec.product_id.categ_id.id), + ] + ) + rec.allowed_reason_code_ids = codes + + @api.constrains("reason_code_id", "product_id") + def _check_reason_code_id(self): + for rec in self: + if ( + rec.reason_code_id + and rec.reason_code_id not in rec.allowed_reason_code_ids + ): + raise ValidationError( + _( + "The selected reason code is not allowed for this product category." + ) + ) + + def _prepare_move_values(self): + res = super()._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().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/pyproject.toml b/scrap_reason_code/pyproject.toml new file mode 100644 index 000000000000..4231d0cccb3d --- /dev/null +++ b/scrap_reason_code/pyproject.toml @@ -0,0 +1,3 @@ +[build-system] +requires = ["whool"] +build-backend = "whool.buildapi" diff --git a/scrap_reason_code/readme/CONFIGURE.md b/scrap_reason_code/readme/CONFIGURE.md new file mode 100644 index 000000000000..9ab956990895 --- /dev/null +++ b/scrap_reason_code/readme/CONFIGURE.md @@ -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.md b/scrap_reason_code/readme/CONTRIBUTORS.md new file mode 100644 index 000000000000..30f060b88636 --- /dev/null +++ b/scrap_reason_code/readme/CONTRIBUTORS.md @@ -0,0 +1,7 @@ +- Michael Allen \<\> +- Bhavesh Odedra \<\> +- Balaji Kannan \<\> +- Serpent Consulting Services Pvt. Ltd. \<\> +- Chandresh Thakkar \<\> +- Hughes Damry \<\> +- Lois Rilo \<\> diff --git a/scrap_reason_code/readme/CREDITS.md b/scrap_reason_code/readme/CREDITS.md new file mode 100644 index 000000000000..f276e4261ed5 --- /dev/null +++ b/scrap_reason_code/readme/CREDITS.md @@ -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.md b/scrap_reason_code/readme/DESCRIPTION.md new file mode 100644 index 000000000000..6f8506f7df7c --- /dev/null +++ b/scrap_reason_code/readme/DESCRIPTION.md @@ -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.md b/scrap_reason_code/readme/ROADMAP.md new file mode 100644 index 000000000000..8b137891791f --- /dev/null +++ b/scrap_reason_code/readme/ROADMAP.md @@ -0,0 +1 @@ + diff --git a/scrap_reason_code/readme/USAGE.md b/scrap_reason_code/readme/USAGE.md new file mode 100644 index 000000000000..2736fe8270e8 --- /dev/null +++ b/scrap_reason_code/readme/USAGE.md @@ -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 000000000000..bb3255b27707 --- /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 000000000000..3a0328b516c4 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 000000000000..06cd24edb67b --- /dev/null +++ b/scrap_reason_code/static/description/index.html @@ -0,0 +1,658 @@ + + + + + + Scrap Reason Code + + + +
+

Scrap Reason Code

+ + +

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

+

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 to smash 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 000000000000..29507951a90c --- /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 000000000000..e755a5cfa4d3 --- /dev/null +++ b/scrap_reason_code/tests/test_scrap_reason_code.py @@ -0,0 +1,221 @@ +# Copyright (C) 2019 IBM Corp. +# Copyright (C) 2019 Open Source Integrators +# Copyright 2023 ForgeFlow S.L. (https://www.forgeflow.com) +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo.exceptions import ValidationError +from odoo.tests.common import TransactionCase + + +class StockScrap(TransactionCase): + def setUp(self): + super().setUp() + + self.stock_location = self.env.ref("stock.stock_location_stock") + self.customer_location = self.env.ref("stock.stock_location_customers") + self.categ_1 = self.env.ref("product.product_category_all") + self.categ_2 = self.env["product.category"].create({"name": "Test category"}) + 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.categ_1.id, + } + ) + self.scrap_product_2 = self.env["product.product"].create( + { + "name": "Scrap Product A", + "type": "product", + "categ_id": self.categ_2.id, + } + ) + + self.reason_code = self.env["scrap.reason.code"].create( + { + "name": "DM300", + "description": "Product is damage", + "location_id": self.scrapped_location.id, + } + ) + self.reason_code_only_categ_2 = self.env["scrap.reason.code"].create( + { + "name": "Test Code 2", + "description": "Test description", + "product_category_ids": [(6, 0, self.categ_2.ids)], + } + ) + + 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_id.id, scrap.id, "Wrong scrap linked to the move." + ) + self.assertEqual( + scrap.scrap_qty, + 5, + "Scrap quantity has been modified and is not " "correct anymore.", + ) + moves = scrap.move_ids + self.assertEqual(moves.mapped("reason_code_id.id"), [self.reason_code.id]) + + scrapped_move.quantity = 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_id.id, scrap2.id, "Wrong scrap linked to the move." + ) + self.assertEqual( + scrap2.scrap_qty, + 5, + "Scrap quantity has been modified and is not " "correct anymore.", + ) + moves = scrap2.move_ids + self.assertEqual(moves.mapped("reason_code_id.id"), [self.reason_code.id]) + + scrapped_move.quantity = 8 + self.assertEqual(scrap2.scrap_qty, 8, "Scrap quantity is not updated.") + + def test_allowed_reason_codes(self): + with self.assertRaises(ValidationError): + self.env["stock.scrap"].create( + { + "product_id": self.scrap_product.id, + "product_uom_id": self.scrap_product_2.uom_id.id, + "scrap_qty": 5, + "reason_code_id": self.reason_code_only_categ_2.id, + } + ) + scrap = self.env["stock.scrap"].create( + { + "product_id": self.scrap_product.id, + "product_uom_id": self.scrap_product.uom_id.id, + "scrap_qty": 5, + "reason_code_id": self.reason_code.id, + } + ) + self.assertEqual(scrap.allowed_reason_code_ids, self.reason_code) + with self.assertRaises(ValidationError): + scrap.write({"reason_code_id": self.reason_code_only_categ_2.id}) + scrap = self.env["stock.scrap"].create( + { + "product_id": self.scrap_product_2.id, + "product_uom_id": self.scrap_product_2.uom_id.id, + "scrap_qty": 5, + "reason_code_id": self.reason_code_only_categ_2.id, + } + ) + with self.assertRaises(ValidationError): + scrap.write({"product_id": self.scrap_product.id}) + self.assertEqual( + scrap.allowed_reason_code_ids, + (self.reason_code + self.reason_code_only_categ_2), + ) + scrap.write({"reason_code_id": self.reason_code.id}) 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 000000000000..acd011acd49d --- /dev/null +++ b/scrap_reason_code/views/reason_code_view.xml @@ -0,0 +1,54 @@ + + + + + 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 000000000000..fcfb962216fa --- /dev/null +++ b/scrap_reason_code/views/stock_move_views.xml @@ -0,0 +1,17 @@ + + + + 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 000000000000..3bb2d8dd4fda --- /dev/null +++ b/scrap_reason_code/views/stock_scrap_views.xml @@ -0,0 +1,38 @@ + + + + stock.scrap.form.reason.code + stock.scrap + + + + + + + + + + stock.scrap.tree.reason.code + stock.scrap + + + + + + + + + + stock.scrap.form2.reason.code + stock.scrap + + + + + + + + +