commit 727e0ad6afb6fa14b8b790fd1b09c30946aeb49c Author: Suherdy Yacob Date: Mon Jan 12 16:43:00 2026 +0700 first commit diff --git a/README.rst b/README.rst new file mode 100755 index 0000000..a0956e6 --- /dev/null +++ b/README.rst @@ -0,0 +1,180 @@ +======================= +Stock Disallow Negative +======================= + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:13f62e491b453ad71831420b6e1b7d6c30a989c98c56dee76cb674305cc80263 + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |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--workflow-lightgray.png?logo=github + :target: https://github.com/OCA/stock-logistics-workflow/tree/18.0/stock_no_negative + :alt: OCA/stock-logistics-workflow +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/stock-logistics-workflow-18-0/stock-logistics-workflow-18-0-stock_no_negative + :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-workflow&target_branch=18.0 + :alt: Try me on Runboat + +|badge1| |badge2| |badge3| |badge4| |badge5| + +By default, Odoo allows negative stock. The advantage of negative stock +is that, if some stock levels are wrong in the ERP, you will not be +blocked when validating the picking for a customer... so you will still +be able to ship the products on time (it's an example !). The problem is +that, after you forced the stock level to negative, you are supposed to +fix the stock level later via an inventory ; but this action is often +forgotten by users, so you end up with negative stock levels in your ERP +and it can stay like this forever (or at least until the next full +inventory). + +If you disallow negative stock in Odoo with this module, you will be +blocked when trying to validate a stock operation that will set the +stock level of a product and/or location as negative. So you will have +to fix the wrong stock level of that product without delay, in order to +validate the stock operation in Odoo...you can't forget it anymore ! + +**Table of contents** + +.. contents:: + :local: + +Configuration +============= + +By default, the stockable products will not be allowed to have a +negative stock. If you want to make some exceptions for some products, +product categories or locations, you can activate the option *Allow +Negative Stock*: + +For products: + +1. Go to *Inventory / Master Data / Products* and in the tab *General + Information* activate this option. + +For product categories: + +1. Go to *Inventory / Configuration / Products / Product Categories* and + activate this option. + +For individual locations: + +1. Go to *Inventory / Configuration / Settings* and activate the option + *Storage Locations*. +2. Go to *Inventory / Configuration / Warehouse Management / Locations* + and activate the option the option *Allow Negative Stock* for the + locations you choose. + +Usage +===== + +When you validate a stock operation (a stock move, a picking, a +manufacturing order, etc.) that will set the stock level of a stockable +product as negative, you will be blocked by an error message. The +consumable products can still have a negative stock level. + +Changelog +========= + +16.0.1.0.0 (2022-10-22) +----------------------- + +- [16.0][MIG] stock_no_negative + +15.0.1.0.0 (2021-12-22) +----------------------- + +- [15.0][MIG] stock_no_negative + +14.0.1.0.0 (2020-12-14) +----------------------- + +- [14.0][MIG] stock_no_negative + +13.0.1.0.0 (2020-01-03) +----------------------- + +- [13.0][MIG] stock_no_negative Remove all decorators @api.multi + +11.0.1.1.0 (2018-12-13) +----------------------- + +- Add the ability to allow negative stock for individual stock + locations. + +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 +------- + +* Akretion + +Contributors +------------ + +- Alexis de Lattre + +- Vishnu Vanneri + +- Serpent Consulting Services Pvt. Ltd. + +- `ForgeFlow S.L. `__: + + - Jordi Ballester + - Joan Mateu + +- `Tecnativa `__: + + - Pedro M. Baeza + +- `Spacefoot `__: + + - Quentin Delcourte + +- Vishnu Vanneri + +- `OERP Canada `__: + + - Foram Darji + +- `Dynapps `__: + + - Bert Van Groenendael + +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. + +This module is part of the `OCA/stock-logistics-workflow `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/__init__.py b/__init__.py new file mode 100755 index 0000000..31c4688 --- /dev/null +++ b/__init__.py @@ -0,0 +1,5 @@ +# ?? 2015-2016 Akretion (http://www.akretion.com) +# @author Alexis de Lattre +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import models diff --git a/__manifest__.py b/__manifest__.py new file mode 100755 index 0000000..68993c3 --- /dev/null +++ b/__manifest__.py @@ -0,0 +1,17 @@ +# ?? 2015-2016 Akretion (http://www.akretion.com) +# @author Alexis de Lattre +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + + +{ + "name": "Stock Disallow Negative", + "version": "18.0.1.0.3", + "category": "Inventory, Logistic, Storage", + "license": "AGPL-3", + "summary": "Disallow negative stock levels by default", + "author": "Akretion,Odoo Community Association (OCA)", + "website": "https://github.com/OCA/stock-logistics-workflow", + "depends": ["stock"], + "data": ["views/product_product_views.xml", "views/stock_location_views.xml"], + "installable": True, +} diff --git a/__pycache__/__init__.cpython-310.pyc b/__pycache__/__init__.cpython-310.pyc new file mode 100755 index 0000000..4ac0473 Binary files /dev/null and b/__pycache__/__init__.cpython-310.pyc differ diff --git a/__pycache__/__init__.cpython-312.pyc b/__pycache__/__init__.cpython-312.pyc new file mode 100644 index 0000000..a87b207 Binary files /dev/null and b/__pycache__/__init__.cpython-312.pyc differ diff --git a/i18n/ar.po b/i18n/ar.po new file mode 100755 index 0000000..efebe65 --- /dev/null +++ b/i18n/ar.po @@ -0,0 +1,101 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 17.0+e\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2023-11-21 14:44+0000\n" +"PO-Revision-Date: 2023-11-21 14:44+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Permitir Stock Negativo" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Permitir niveles de stock negativos para los productos en stock adjuntos a " +"esta categoría. Las opciones no se aplican a los productos adjuntos a " +"subcategorías de esta categoría." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" +"Permitir niveles de stock negativos para los productos en stock ubicados en " +"esta ubicación." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and " +"that this product is a stockable product, then the validation of the related " +"stock moves will be blocked if the stock level becomes negative with the " +"stock move." +msgstr "" +"Si esta opción no está activa en este producto ni en su categoría de " +"producto y este producto es un producto almacenable, entonces la validación " +"de los movimientos de stock relacionados se bloqueará si el nivel de stock " +"se vuelve negativo con el movimiento de stock." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Ubicaciones de inventario" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Categoría de producto" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Quants" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock " +"location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"No se puede validar esta operación de stock porque el nivel de stock del " +"producto '%s'%s se volvería negativo (%s) en la ubicación de stock '%s' y no " +"se permite stock negativo para este producto y/o ubicación." + +#~ msgid " lot {}" +#~ msgstr " lote {}" diff --git a/i18n/de.po b/i18n/de.po new file mode 100755 index 0000000..672c817 --- /dev/null +++ b/i18n/de.po @@ -0,0 +1,99 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 17.0+e\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2023-11-21 14:50+0000\n" +"PO-Revision-Date: 2023-11-21 14:50+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Negative Lagerbestände erlauben" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Negative Lagerbestände für lagerbare Produkte dieser Kategorie erlauben. " +"Diese Option gilt nicht für Produkte, die zu Unterkategorien dieser " +"Kategorie gehören." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" +"Negative Lagerbestände für die lagerfähigen Produkte erlauben, die mit " +"diesem Ort verbunden sind." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and " +"that this product is a stockable product, then the validation of the related " +"stock moves will be blocked if the stock level becomes negative with the " +"stock move." +msgstr "" +"Wenn diese Option weder für dieses Produkt noch für seine Produktkategorie " +"aktiviert ist und es sich bei diesem Produkt um ein lagerfähiges Produkt " +"handelt, dann wird die Validierung der damit verbundenen Lagerbewegungen " +"blockiert, wenn der Lagerbestand durch die Lagerbewegung negativ wird." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Lagerorte" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "Produkt" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Produktkategorie" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Quanten" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock " +"location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"Diese Aktion kann nicht ausgeführt werden, da der Lagerbestand von " +"'{name}'{name_lot} im Lager '{complete_name}' negativ werden würde " +"({q_quantity}). Negative Lagerbestände sind für dieses Produkt oder in " +"diesem Lager nicht möglich." diff --git a/i18n/es.po b/i18n/es.po new file mode 100755 index 0000000..17aa3b1 --- /dev/null +++ b/i18n/es.po @@ -0,0 +1,103 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 17.0+e\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2023-11-21 14:51+0000\n" +"PO-Revision-Date: 2025-03-11 02:06+0000\n" +"Last-Translator: MartinLG3 \n" +"Language-Team: \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 5.10.2\n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr " lote %(name)s" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Permitir Stock Negativo" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Permitir niveles de stock negativos para los productos en stock adjuntos a " +"esta categoría. Las opciones no se aplican a los productos adjuntos a " +"subcategorías de esta categoría." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" +"Permitir niveles de stock negativos para los productos en stock ubicados en " +"esta ubicación." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and " +"that this product is a stockable product, then the validation of the related " +"stock moves will be blocked if the stock level becomes negative with the " +"stock move." +msgstr "" +"Si esta opción no está activa en este producto ni en su categoría de " +"producto y este producto es un producto almacenable, entonces la validación " +"de los movimientos de stock relacionados se bloqueará si el nivel de stock " +"se vuelve negativo con el movimiento de stock." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Ubicaciones de inventario" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "Producto" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Categoría de producto" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Cants" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock " +"location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"No puede validar esta operación de stock porque el nivel de stock del " +"producto '{name}'{name_lot} sería negativo ({q_quantity}) en la ubicación de " +"stock '{complete_name}' y el stock negativo no está permitido para este " +"producto y/o ubicación." + +#~ msgid " lot {}" +#~ msgstr " lote {}" diff --git a/i18n/fr.po b/i18n/fr.po new file mode 100755 index 0000000..cdf7c70 --- /dev/null +++ b/i18n/fr.po @@ -0,0 +1,100 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 17.0+e\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2023-11-21 14:49+0000\n" +"PO-Revision-Date: 2024-04-23 06:38+0000\n" +"Last-Translator: \"Denis Roussel (ACSONE)\" \n" +"Language-Team: \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: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Autoriser le stock négatif" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Autorise les niveaux de stock négatif pour les articles stockables attachés " +"à cette catégorie. Cette option ne s'applique pas aux articles attachés à " +"des sous-catégories de cette catégorie." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" +"Autorise les niveaux de stock négatif pour les articles stockables attachés " +"à cette catégorie. Cette option ne s'applique pas aux articles attachés à " +"des sous-catégories de cette catégorie." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and " +"that this product is a stockable product, then the validation of the related " +"stock moves will be blocked if the stock level becomes negative with the " +"stock move." +msgstr "" +"Si cette option n'est pas activée sur cet article ni sur la catégorie à " +"laquelle il est rattaché et que cet article est un produit stockable, alors " +"la validation des mouvements de stock sera bloquée si le niveau de stock " +"devient négatif avec ce mouvement de stock." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Emplacements d'inventaire" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Catégorie d'article" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Quants" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock " +"location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"Impossible de valider cette opération car le niveau de stock de ce produit " +"{name}'{name_lot} deviendrait négatif({q_quantity}) dans l'emplacement du " +"stock '{complete_name}'. Un stock négatif n'est pas permis pour ce produit." diff --git a/i18n/hr.po b/i18n/hr.po new file mode 100755 index 0000000..d13a32d --- /dev/null +++ b/i18n/hr.po @@ -0,0 +1,98 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 17.0+e\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2023-11-21 14:49+0000\n" +"PO-Revision-Date: 2023-11-21 14:49+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Dozvoli negativnu zalihu" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Dozvoli negativna stanja uskladištivih proizvoda pridruženih ovoj " +"kategoriji. Opcije ne se odnose na proizvode iz pordeđenih kategorija ovoj." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "Dozvoli negativnu zalihu za uskladištive proizvode na ovoj lokaciji." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and " +"that this product is a stockable product, then the validation of the related " +"stock moves will be blocked if the stock level becomes negative with the " +"stock move." +msgstr "" +"Ako ova opcija nije aktivna na ovom proizvodu niti na kategoriji, i ovo je " +"uskladištivi proizvod, tada će potvrđivanje skladišnih kretanja biti " +"blokirano ako tim kretanjem zaliha ulazi u minus." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Skladišne lokacije" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "Proizvod" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Kategorija proizvoda" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Količine" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock " +"location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"Nije moguće potvrditi ovu skladišnu operaciju jer bi količine proizvoda " +"'{name}'{name_lot} na stanju postale negativne ({q_quantity}) na skladišnoj " +"lokaciji '{complete_name}' a negativna zaliha nije dozvoljena za taj " +"proizvod i/ili lokaciju." + +#~ msgid " lot {}" +#~ msgstr " lote {}" diff --git a/i18n/it.po b/i18n/it.po new file mode 100755 index 0000000..8434612 --- /dev/null +++ b/i18n/it.po @@ -0,0 +1,102 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 17.0+e\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2023-11-21 14:50+0000\n" +"PO-Revision-Date: 2025-02-12 16:06+0000\n" +"Last-Translator: mymage \n" +"Language-Team: \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 5.6.2\n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr " lotto %(name)s" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Consenti giacenze negative" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Consente giacenze negative per prodotti stoccabili associati a questa " +"categoria. Questa opzione non si applica ai prodotti associati alle sotto " +"categorie della categoria." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" +"Consente giacenze negative per prodottistoccabili associati a questa " +"ubicazione." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and " +"that this product is a stockable product, then the validation of the related " +"stock moves will be blocked if the stock level becomes negative with the " +"stock move." +msgstr "" +"Se questa opzione non è attiva nè in questo prodotto nè nella sua categoria " +"e il prodotto è stoccabile, se la giacenza diventa negativa con il movimento " +"di magazzino allora la validazione del relativo movimento verrà bloccata ." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Ubicazioni di inventario" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "Prodotto" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Categoria prodotto" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Quanti" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock " +"location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"Questa operazione di magazzino non può essere validata perché la giacenza " +"del prodotto '{name}'{name_lot} diventerebbe negativa ({q_quantity}) " +"nell'ubicazione '{complete_name}' e per questo prodotto e/o ubicazione non è " +"consentata la giacenza negativa." + +#~ msgid " lot {}" +#~ msgstr " lotto {}" diff --git a/i18n/nl.po b/i18n/nl.po new file mode 100755 index 0000000..b8739a1 --- /dev/null +++ b/i18n/nl.po @@ -0,0 +1,84 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"Last-Translator: Automatically generated\n" +"Language-Team: none\n" +"Language: nl\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" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and" +" that this product is a stockable product, then the validation of the " +"related stock moves will be blocked if the stock level becomes negative with" +" the stock move." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock" +" location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" diff --git a/i18n/nl_NL.po b/i18n/nl_NL.po new file mode 100755 index 0000000..fd581b8 --- /dev/null +++ b/i18n/nl_NL.po @@ -0,0 +1,99 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2025-04-14 22:24+0000\n" +"Last-Translator: Bosd \n" +"Language-Team: none\n" +"Language: nl_NL\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 5.10.4\n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr " lot %(name)s" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Negatieve Voorraad Toestaan" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Staat negatieve voorraadniveaus toe voor de voorraadproducten die aan deze " +"categorie zijn gekoppeld. De optie is niet van toepassing op producten die " +"aan subcategorieën van deze categorie zijn gekoppeld." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" +"Staat negatieve voorraadniveaus toe voor de voorraadproducten die aan deze " +"locatie zijn gekoppeld." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and" +" that this product is a stockable product, then the validation of the " +"related stock moves will be blocked if the stock level becomes negative with" +" the stock move." +msgstr "" +"Als deze optie niet actief is voor dit product of de productcategorie en dit " +"product een voorraadproduct is, dan wordt de validatie van de gerelateerde " +"voorraadmutaties geblokkeerd als het voorraadniveau negatief wordt met de " +"voorraadmutatie." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Voorraadlocaties" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "Product" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Productcategorie" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Quantiteiten" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock" +" location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"U kunt deze voorraadhandeling niet valideren omdat het voorraadniveau van " +"het product '{name}'{name_lot} negatief zou worden ({q_quantity}) op de " +"voorraadlocatie '{complete_name}' en negatieve voorraad niet is toegestaan " +"voor dit product en/of locatie." diff --git a/i18n/pt.po b/i18n/pt.po new file mode 100755 index 0000000..17e8deb --- /dev/null +++ b/i18n/pt.po @@ -0,0 +1,100 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 17.0+e\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2023-11-21 14:50+0000\n" +"PO-Revision-Date: 2025-08-26 11:24+0000\n" +"Last-Translator: Pedro Castro Silva \n" +"Language-Team: \n" +"Language: pt\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 5.10.4\n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr " lote %(name)s" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Permitir Stock Negativo" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Permitir níveis de stock negativos para os produtos armazenáveis ligados a " +"esta categoria. As opções não se aplicam aos produtos vinculados a sub-" +"categorias desta categoria." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" +"Permitir níveis de stock negativos para os produtos armazenáveis anexados a " +"este local." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and " +"that this product is a stockable product, then the validation of the related " +"stock moves will be blocked if the stock level becomes negative with the " +"stock move." +msgstr "" +"Se esta opção não estiver ativa neste produto nem na sua categoria e se este " +"for um produto armazenável, então a validação dos movimentos de stock " +"relacionados será bloqueada se o nível de stock se tornar negativo após o " +"movimento de stock." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Localizações de Inventário" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "Produto" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Categoria do Artigo" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Quants" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock " +"location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"Não pode validar esta operação de stock porque o nível de stock do produto " +"'{name}'{name_lot} ficaria negativo ({q_quantity}) na localização de stock " +"'{complete_name}' e um stock negativo não é permitido para este produto e/ou " +"local." diff --git a/i18n/pt_BR.po b/i18n/pt_BR.po new file mode 100755 index 0000000..101a92c --- /dev/null +++ b/i18n/pt_BR.po @@ -0,0 +1,84 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"Last-Translator: Automatically generated\n" +"Language-Team: none\n" +"Language: pt_BR\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" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and" +" that this product is a stockable product, then the validation of the " +"related stock moves will be blocked if the stock level becomes negative with" +" the stock move." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock" +" location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" diff --git a/i18n/stock_no_negative.pot b/i18n/stock_no_negative.pot new file mode 100755 index 0000000..1f0edb1 --- /dev/null +++ b/i18n/stock_no_negative.pot @@ -0,0 +1,83 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.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: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and" +" that this product is a stockable product, then the validation of the " +"related stock moves will be blocked if the stock level becomes negative with" +" the stock move." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock" +" location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" diff --git a/i18n/tr.po b/i18n/tr.po new file mode 100755 index 0000000..b6daf73 --- /dev/null +++ b/i18n/tr.po @@ -0,0 +1,94 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2025-06-24 12:25+0000\n" +"Last-Translator: Betül Öğmen \n" +"Language-Team: none\n" +"Language: tr\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 5.10.4\n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "Stok Negatif Olabilir" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" +"Bu kategorideki stoklanabilir ürünlerin stokları negatif olabilir. Bu " +"seçenek alt kategorilere uygulanmaz." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "Bu konumdaki stoklanabilir ürünlerin stokları negatif olabilir." + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and" +" that this product is a stockable product, then the validation of the " +"related stock moves will be blocked if the stock level becomes negative with" +" the stock move." +msgstr "" +"Eğer bu seçenek üründe veya ürünün kategorisinde etkinse ve ürün " +"stoklanabilir bir ürünse, stok seviyesini negatif yapacak seviye stok " +"hareketlerinin onaylanması engellenir." + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "Stok Konumları" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "Ürün" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "Ürün Kategorisi" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "Stoklar" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock" +" location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" +"Bu stok hareketini onaylayamazsınız çünkü '{name}'{name_lot} ürünü stok " +"seviyesi negatif ({q_quantity}) olacak '{complete_name}' konumunda ve ürün " +"miktarı bu ürün ve/veya konum için negatif stok olamaz." diff --git a/i18n/zh_CN.po b/i18n/zh_CN.po new file mode 100755 index 0000000..13a18c6 --- /dev/null +++ b/i18n/zh_CN.po @@ -0,0 +1,84 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * stock_no_negative +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"Last-Translator: Automatically generated\n" +"Language-Team: none\n" +"Language: zh_CN\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=1; plural=0;\n" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid " lot %(name)s" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,field_description:stock_no_negative.field_product_category__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_product_template__allow_negative_stock +#: model:ir.model.fields,field_description:stock_no_negative.field_stock_location__allow_negative_stock +msgid "Allow Negative Stock" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_category__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"category. The options doesn't apply to products attached to sub-categories " +"of this category." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_stock_location__allow_negative_stock +msgid "" +"Allow negative stock levels for the stockable products attached to this " +"location." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model.fields,help:stock_no_negative.field_product_product__allow_negative_stock +#: model:ir.model.fields,help:stock_no_negative.field_product_template__allow_negative_stock +msgid "" +"If this option is not active on this product nor on its product category and" +" that this product is a stockable product, then the validation of the " +"related stock moves will be blocked if the stock level becomes negative with" +" the stock move." +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_location +msgid "Inventory Locations" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_template +msgid "Product" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_product_category +msgid "Product Category" +msgstr "" + +#. module: stock_no_negative +#: model:ir.model,name:stock_no_negative.model_stock_quant +msgid "Quants" +msgstr "" + +#. module: stock_no_negative +#. odoo-python +#: code:addons/stock_no_negative/models/stock_quant.py:0 +msgid "" +"You cannot validate this stock operation because the stock level of the " +"product '{name}'{name_lot} would become negative ({q_quantity}) on the stock" +" location '{complete_name}' and negative stock is not allowed for this " +"product and/or location." +msgstr "" diff --git a/models/__init__.py b/models/__init__.py new file mode 100755 index 0000000..8d84cb1 --- /dev/null +++ b/models/__init__.py @@ -0,0 +1,7 @@ +# ?? 2015-2016 Akretion (http://www.akretion.com) +# @author Alexis de Lattre +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import product +from . import stock_quant +from . import stock_location diff --git a/models/__pycache__/__init__.cpython-310.pyc b/models/__pycache__/__init__.cpython-310.pyc new file mode 100755 index 0000000..aa9bada Binary files /dev/null and b/models/__pycache__/__init__.cpython-310.pyc differ diff --git a/models/__pycache__/__init__.cpython-312.pyc b/models/__pycache__/__init__.cpython-312.pyc new file mode 100644 index 0000000..96a122f Binary files /dev/null and b/models/__pycache__/__init__.cpython-312.pyc differ diff --git a/models/__pycache__/product.cpython-310.pyc b/models/__pycache__/product.cpython-310.pyc new file mode 100755 index 0000000..e6e88a4 Binary files /dev/null and b/models/__pycache__/product.cpython-310.pyc differ diff --git a/models/__pycache__/product.cpython-312.pyc b/models/__pycache__/product.cpython-312.pyc new file mode 100644 index 0000000..dd01a63 Binary files /dev/null and b/models/__pycache__/product.cpython-312.pyc differ diff --git a/models/__pycache__/stock_location.cpython-310.pyc b/models/__pycache__/stock_location.cpython-310.pyc new file mode 100755 index 0000000..7671cad Binary files /dev/null and b/models/__pycache__/stock_location.cpython-310.pyc differ diff --git a/models/__pycache__/stock_location.cpython-312.pyc b/models/__pycache__/stock_location.cpython-312.pyc new file mode 100644 index 0000000..e7e4161 Binary files /dev/null and b/models/__pycache__/stock_location.cpython-312.pyc differ diff --git a/models/__pycache__/stock_quant.cpython-310.pyc b/models/__pycache__/stock_quant.cpython-310.pyc new file mode 100755 index 0000000..7e122aa Binary files /dev/null and b/models/__pycache__/stock_quant.cpython-310.pyc differ diff --git a/models/__pycache__/stock_quant.cpython-312.pyc b/models/__pycache__/stock_quant.cpython-312.pyc new file mode 100644 index 0000000..688edfd Binary files /dev/null and b/models/__pycache__/stock_quant.cpython-312.pyc differ diff --git a/models/product.py b/models/product.py new file mode 100755 index 0000000..86d6371 --- /dev/null +++ b/models/product.py @@ -0,0 +1,26 @@ +# Copyright 2015-2016 Akretion (http://www.akretion.com) +# @author Alexis de Lattre +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import fields, models + + +class ProductCategory(models.Model): + _inherit = "product.category" + + allow_negative_stock = fields.Boolean( + help="Allow negative stock levels for the stockable products " + "attached to this category. The options doesn't apply to products " + "attached to sub-categories of this category.", + ) + + +class ProductTemplate(models.Model): + _inherit = "product.template" + + allow_negative_stock = fields.Boolean( + help="If this option is not active on this product nor on its " + "product category and that this product is a stockable product, " + "then the validation of the related stock moves will be blocked if " + "the stock level becomes negative with the stock move.", + ) diff --git a/models/stock_location.py b/models/stock_location.py new file mode 100755 index 0000000..1dc3c9b --- /dev/null +++ b/models/stock_location.py @@ -0,0 +1,14 @@ +# Copyright 2018 ForgeFlow (https://www.forgeflow.com) +# @author Jordi Ballester +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import fields, models + + +class StockLocation(models.Model): + _inherit = "stock.location" + + allow_negative_stock = fields.Boolean( + help="Allow negative stock levels for the stockable products " + "attached to this location.", + ) diff --git a/models/stock_quant.py b/models/stock_quant.py new file mode 100755 index 0000000..ce25d11 --- /dev/null +++ b/models/stock_quant.py @@ -0,0 +1,67 @@ +# Copyright 2015-2017 Akretion (http://www.akretion.com) +# @author Alexis de Lattre +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo import _, api, models +from odoo.exceptions import ValidationError +from odoo.tools import config, float_compare + + +class StockQuant(models.Model): + _inherit = "stock.quant" + + @api.constrains("product_id", "quantity") + def check_negative_qty(self): + # To provide an option to skip the check when necessary. + # e.g. mrp_subcontracting_skip_no_negative - passes the context + # for subcontracting receipts. + # Also allows negative stock for subcontracting locations as they + # are expected to have negative stock in certain Odoo 18 workflows. + if self.env.context.get("skip_negative_qty_check"): + return + p = self.env["decimal.precision"].precision_get("Product Unit of Measure") + check_negative_qty = ( + config["test_enable"] and self.env.context.get("test_stock_no_negative") + ) or not config["test_enable"] + if not check_negative_qty: + return + + for quant in self: + disallowed_by_product = ( + not quant.product_id.allow_negative_stock + and not quant.product_id.categ_id.allow_negative_stock + ) + disallowed_by_location = not quant.location_id.allow_negative_stock + + # Allow negative stock for subcontracting locations + # Check if mrp_subcontracting is installed and location has is_subcontracting_location field + is_subcontracting_location = False + if hasattr(quant.location_id, 'is_subcontracting_location'): + is_subcontracting_location = quant.location_id.is_subcontracting_location + + if ( + float_compare(quant.quantity, 0, precision_digits=p) == -1 + and quant.product_id.is_storable + and quant.location_id.usage in ["internal", "transit"] + and disallowed_by_product + and disallowed_by_location + and not is_subcontracting_location + ): + msg_add = "" + if quant.lot_id: + msg_add = _(" lot %(name)s", name=quant.lot_id.display_name) + raise ValidationError( + _( + "You cannot validate this stock operation because the " + "stock level of the product '{name}'{name_lot} would " + "become negative " + "({q_quantity}) on the stock location '{complete_name}' " + "and negative stock is " + "not allowed for this product and/or location." + ).format( + name=quant.product_id.display_name, + name_lot=msg_add, + q_quantity=quant.quantity, + complete_name=quant.location_id.complete_name, + ) + ) diff --git a/pyproject.toml b/pyproject.toml new file mode 100755 index 0000000..4231d0c --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,3 @@ +[build-system] +requires = ["whool"] +build-backend = "whool.buildapi" diff --git a/readme/CONFIGURE.md b/readme/CONFIGURE.md new file mode 100755 index 0000000..75739fc --- /dev/null +++ b/readme/CONFIGURE.md @@ -0,0 +1,22 @@ +By default, the stockable products will not be allowed to have a +negative stock. If you want to make some exceptions for some products, +product categories or locations, you can activate the option *Allow +Negative Stock*: + +For products: + +1. Go to *Inventory / Master Data / Products* and in the tab *General + Information* activate this option. + +For product categories: + +1. Go to *Inventory / Configuration / Products / Product Categories* + and activate this option. + +For individual locations: + +1. Go to *Inventory / Configuration / Settings* and activate the option + *Storage Locations*. +2. Go to *Inventory / Configuration / Warehouse Management / Locations* + and activate the option the option *Allow Negative Stock* for the + locations you choose. diff --git a/readme/CONTRIBUTORS.md b/readme/CONTRIBUTORS.md new file mode 100755 index 0000000..0f244d0 --- /dev/null +++ b/readme/CONTRIBUTORS.md @@ -0,0 +1,23 @@ +- Alexis de Lattre \<\> + +- Vishnu Vanneri \<\> + +- Serpent Consulting Services Pvt. Ltd. \<\> + +- [ForgeFlow S.L.](contact@forgeflow.com): + - Jordi Ballester + - Joan Mateu + +- [Tecnativa](https://www.tecnativa.com): + - Pedro M. Baeza + +- [Spacefoot](https://www.spacefoot.com): + - Quentin Delcourte + +- Vishnu Vanneri \<\> + +- [OERP Canada](https://www.oerp.ca/): + - Foram Darji \<\> + +- [Dynapps](https://www.dynapps.eu/): + - Bert Van Groenendael \<\> diff --git a/readme/DESCRIPTION.md b/readme/DESCRIPTION.md new file mode 100755 index 0000000..519c5dd --- /dev/null +++ b/readme/DESCRIPTION.md @@ -0,0 +1,15 @@ +By default, Odoo allows negative stock. The advantage of negative stock +is that, if some stock levels are wrong in the ERP, you will not be +blocked when validating the picking for a customer... so you will still +be able to ship the products on time (it's an example !). The problem is +that, after you forced the stock level to negative, you are supposed to +fix the stock level later via an inventory ; but this action is often +forgotten by users, so you end up with negative stock levels in your ERP +and it can stay like this forever (or at least until the next full +inventory). + +If you disallow negative stock in Odoo with this module, you will be +blocked when trying to validate a stock operation that will set the +stock level of a product and/or location as negative. So you will have +to fix the wrong stock level of that product without delay, in order to +validate the stock operation in Odoo...you can't forget it anymore ! diff --git a/readme/HISTORY.md b/readme/HISTORY.md new file mode 100755 index 0000000..e18fa7a --- /dev/null +++ b/readme/HISTORY.md @@ -0,0 +1,24 @@ +## 18.0.1.0.3 (2024-12-18) + +- Allow negative stock for subcontracting locations to support Odoo 18 subcontracting workflows + +## 16.0.1.0.0 (2022-10-22) + +- \[16.0\]\[MIG\] stock_no_negative + +## 15.0.1.0.0 (2021-12-22) + +- \[15.0\]\[MIG\] stock_no_negative + +## 14.0.1.0.0 (2020-12-14) + +- \[14.0\]\[MIG\] stock_no_negative + +## 13.0.1.0.0 (2020-01-03) + +- \[13.0\]\[MIG\] stock_no_negative Remove all decorators @api.multi + +## 11.0.1.1.0 (2018-12-13) + +- Add the ability to allow negative stock for individual stock + locations. diff --git a/readme/USAGE.md b/readme/USAGE.md new file mode 100755 index 0000000..e6c1807 --- /dev/null +++ b/readme/USAGE.md @@ -0,0 +1,4 @@ +When you validate a stock operation (a stock move, a picking, a +manufacturing order, etc.) that will set the stock level of a stockable +product as negative, you will be blocked by an error message. The +consumable products can still have a negative stock level. diff --git a/static/description/icon.png b/static/description/icon.png new file mode 100755 index 0000000..3a0328b Binary files /dev/null and b/static/description/icon.png differ diff --git a/static/description/index.html b/static/description/index.html new file mode 100755 index 0000000..867858e --- /dev/null +++ b/static/description/index.html @@ -0,0 +1,536 @@ + + + + + +Stock Disallow Negative + + + +
+

Stock Disallow Negative

+ + +

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

+

By default, Odoo allows negative stock. The advantage of negative stock +is that, if some stock levels are wrong in the ERP, you will not be +blocked when validating the picking for a customer… so you will still +be able to ship the products on time (it’s an example !). The problem is +that, after you forced the stock level to negative, you are supposed to +fix the stock level later via an inventory ; but this action is often +forgotten by users, so you end up with negative stock levels in your ERP +and it can stay like this forever (or at least until the next full +inventory).

+

If you disallow negative stock in Odoo with this module, you will be +blocked when trying to validate a stock operation that will set the +stock level of a product and/or location as negative. So you will have +to fix the wrong stock level of that product without delay, in order to +validate the stock operation in Odoo…you can’t forget it anymore !

+

Table of contents

+ +
+

Configuration

+

By default, the stockable products will not be allowed to have a +negative stock. If you want to make some exceptions for some products, +product categories or locations, you can activate the option Allow +Negative Stock:

+

For products:

+
    +
  1. Go to Inventory / Master Data / Products and in the tab General +Information activate this option.
  2. +
+

For product categories:

+
    +
  1. Go to Inventory / Configuration / Products / Product Categories and +activate this option.
  2. +
+

For individual locations:

+
    +
  1. Go to Inventory / Configuration / Settings and activate the option +Storage Locations.
  2. +
  3. Go to Inventory / Configuration / Warehouse Management / Locations +and activate the option the option Allow Negative Stock for the +locations you choose.
  4. +
+
+
+

Usage

+

When you validate a stock operation (a stock move, a picking, a +manufacturing order, etc.) that will set the stock level of a stockable +product as negative, you will be blocked by an error message. The +consumable products can still have a negative stock level.

+
+
+

Changelog

+
+

16.0.1.0.0 (2022-10-22)

+
    +
  • [16.0][MIG] stock_no_negative
  • +
+
+
+

15.0.1.0.0 (2021-12-22)

+
    +
  • [15.0][MIG] stock_no_negative
  • +
+
+
+

14.0.1.0.0 (2020-12-14)

+
    +
  • [14.0][MIG] stock_no_negative
  • +
+
+
+

13.0.1.0.0 (2020-01-03)

+
    +
  • [13.0][MIG] stock_no_negative Remove all decorators @api.multi
  • +
+
+
+

11.0.1.1.0 (2018-12-13)

+
    +
  • Add the ability to allow negative stock for individual stock +locations.
  • +
+
+
+
+

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

+
    +
  • Akretion
  • +
+
+
+

Contributors

+ +
+
+

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.

+

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

+

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

+
+
+
+ + diff --git a/tests/__init__.py b/tests/__init__.py new file mode 100755 index 0000000..2195ba1 --- /dev/null +++ b/tests/__init__.py @@ -0,0 +1,8 @@ +# ?? 2015-2016 Akretion (http://www.akretion.com) +# @author Alexis de Lattre +# ?? 2016 ForgeFlow S.L. +# (http://www.forgeflow.com) +# ?? 2016 Serpent Consulting Services Pvt. Ltd. () +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import test_stock_no_negative diff --git a/tests/__pycache__/test_stock_no_negative.cpython-312.pyc b/tests/__pycache__/test_stock_no_negative.cpython-312.pyc new file mode 100644 index 0000000..a2fe58f Binary files /dev/null and b/tests/__pycache__/test_stock_no_negative.cpython-312.pyc differ diff --git a/tests/test_stock_no_negative.py b/tests/test_stock_no_negative.py new file mode 100755 index 0000000..13b19d6 --- /dev/null +++ b/tests/test_stock_no_negative.py @@ -0,0 +1,273 @@ +# Copyright 2015-2016 Akretion (http://www.akretion.com) - Alexis de Lattre +# Copyright 2016 ForgeFlow (http://www.forgeflow.com) +# Copyright 2016 Serpent Consulting Services () +# Copyright 2018 Tecnativa - Pedro M. Baeza +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + + +from odoo.exceptions import UserError, ValidationError +from odoo.tests.common import TransactionCase + + +class TestStockNoNegative(TransactionCase): + @classmethod + def setUpClass(cls): + super().setUpClass() + cls.product_model = cls.env["product.product"] + cls.product_ctg_model = cls.env["product.category"] + cls.lot_model = cls.env["stock.lot"] + cls.picking_type_id = cls.env.ref("stock.picking_type_out") + cls.location_id = cls.env.ref("stock.stock_location_stock") + cls.location_dest_id = cls.env.ref("stock.stock_location_customers") + # Create product category + cls.product_ctg = cls._create_product_category(cls) + # Create a Product + cls.product = cls._create_product(cls, "test_product1") + # Create a Product With Lot + cls.product_with_lot = cls._create_product_with_lot(cls, "test_lot_product1") + # Create Lot + cls.lot1 = cls._create_lot(cls, "lot1") + cls._create_picking(cls) + cls._create_picking_with_lot(cls) + + def _create_product_category(self): + product_ctg = self.product_ctg_model.create( + {"name": "test_product_ctg", "allow_negative_stock": False} + ) + return product_ctg + + def _create_product(self, name): + product = self.product_model.create( + { + "name": name, + "categ_id": self.product_ctg.id, + "is_storable": True, + "type": "consu", + "allow_negative_stock": False, + } + ) + return product + + def _create_product_with_lot(self, name): + product = self.product_model.create( + { + "name": name, + "categ_id": self.product_ctg.id, + "is_storable": True, + "type": "consu", + "tracking": "lot", + "allow_negative_stock": False, + } + ) + return product + + def _create_lot(self, name): + lot = self.lot_model.create( + { + "name": name, + "product_id": self.product_with_lot.id, + "company_id": self.env.company.id, + } + ) + return lot + + def _create_picking(self): + self.stock_picking = ( + self.env["stock.picking"] + .with_context(test_stock_no_negative=True) + .create( + { + "picking_type_id": self.picking_type_id.id, + "move_type": "direct", + "location_id": self.location_id.id, + "location_dest_id": self.location_dest_id.id, + } + ) + ) + + self.stock_move = self.env["stock.move"].create( + { + "name": "Test Move", + "product_id": self.product.id, + "product_uom_qty": 100.0, + "product_uom": self.product.uom_id.id, + "picking_id": self.stock_picking.id, + "state": "draft", + "location_id": self.location_id.id, + "location_dest_id": self.location_dest_id.id, + "quantity": 100.0, + } + ) + + def _create_picking_with_lot(self): + self.stock_picking_with_lot = ( + self.env["stock.picking"] + .with_context(test_stock_no_negative=True) + .create( + { + "picking_type_id": self.picking_type_id.id, + "move_type": "direct", + "location_id": self.location_id.id, + "location_dest_id": self.location_dest_id.id, + } + ) + ) + + self.stock_move_with_lot = self.env["stock.move"].create( + { + "name": "Test Move", + "product_id": self.product_with_lot.id, + "product_uom_qty": 100.0, + "product_uom": self.product_with_lot.uom_id.id, + "picking_id": self.stock_picking_with_lot.id, + "state": "draft", + "location_id": self.location_id.id, + "location_dest_id": self.location_dest_id.id, + } + ) + + def test_check_constrains(self): + """Assert that constraint is raised when user + tries to validate the stock operation which would + make the stock level of the product negative""" + self.stock_picking.action_confirm() + with self.assertRaises(ValidationError): + self.stock_picking.button_validate() + + def test_check_constrains_with_lot(self): + """Assert that constraint is raised when user + tries to validate the stock operation which would + make the stock level of the product negative with + a product with lot""" + self.stock_picking_with_lot.action_confirm() + self.stock_move_line_with_lot = self.env["stock.move.line"].create( + { + "product_id": self.product_with_lot.id, + "quantity": 100.0, + "picking_id": self.stock_picking_with_lot.id, + "move_id": self.stock_move_with_lot.id, + "location_id": self.location_id.id, + "location_dest_id": self.location_dest_id.id, + "lot_id": self.lot1.id, + } + ) + with self.assertRaises(ValidationError): + self.stock_picking_with_lot.button_validate() + + def test_true_allow_negative_stock_product(self): + """Assert that negative stock levels are allowed when + the allow_negative_stock is set active in the product""" + self.product.allow_negative_stock = True + self.stock_picking.action_confirm() + self.stock_picking.button_validate() + quant = self.env["stock.quant"].search( + [ + ("product_id", "=", self.product.id), + ("location_id", "=", self.location_id.id), + ] + ) + self.assertEqual(quant.quantity, -100) + + def test_true_allow_negative_stock_location(self): + """Assert that negative stock levels are allowed when + the allow_negative_stock is set active in the product""" + self.product.allow_negative_stock = False + self.location_id.allow_negative_stock = True + self.stock_picking.action_confirm() + self.stock_picking.button_validate() + quant = self.env["stock.quant"].search( + [ + ("product_id", "=", self.product.id), + ("location_id", "=", self.location_id.id), + ] + ) + self.assertEqual(quant.quantity, -100) + + def test_true_allow_negative_stock_product_with_lot(self): + """Assert that negative stock levels are allowed when + the allow_negative_stock is set active in the product with lot""" + self.product_with_lot.allow_negative_stock = True + self.stock_picking_with_lot.action_confirm() + with self.assertRaises(UserError): + self.stock_picking_with_lot.button_validate() + # create Detail Operations (move line with lot) + self.stock_move_line_with_lot = self.env["stock.move.line"].create( + { + "product_id": self.product_with_lot.id, + "quantity": 100.0, + "picking_id": self.stock_picking_with_lot.id, + "move_id": self.stock_move_with_lot.id, + "location_id": self.location_id.id, + "location_dest_id": self.location_dest_id.id, + "lot_id": self.lot1.id, + } + ) + self.stock_picking_with_lot.button_validate() + quant = self.env["stock.quant"].search( + [ + ("product_id", "=", self.product_with_lot.id), + ("location_id", "=", self.location_id.id), + ("lot_id", "=", self.lot1.id), + ] + ) + self.assertEqual(quant.quantity, -100) + + def test_allow_negative_stock_subcontracting_location(self): + """Assert that negative stock levels are allowed in subcontracting locations + even when allow_negative_stock is False on product and location""" + # Skip test if mrp_subcontracting is not installed + if not hasattr(self.env['stock.location'], 'is_subcontracting_location'): + self.skipTest("mrp_subcontracting module not installed") + + # Create a subcontracting location + subcontracting_location = self.env['stock.location'].create({ + 'name': 'Test Subcontracting Location', + 'usage': 'internal', + 'is_subcontracting_location': True, + 'allow_negative_stock': False, + }) + + # Create picking from subcontracting location to customer + stock_picking_subcontract = ( + self.env["stock.picking"] + .with_context(test_stock_no_negative=True) + .create( + { + "picking_type_id": self.picking_type_id.id, + "move_type": "direct", + "location_id": subcontracting_location.id, + "location_dest_id": self.location_dest_id.id, + } + ) + ) + + stock_move_subcontract = self.env["stock.move"].create( + { + "name": "Test Subcontract Move", + "product_id": self.product.id, + "product_uom_qty": 100.0, + "product_uom": self.product.uom_id.id, + "picking_id": stock_picking_subcontract.id, + "state": "draft", + "location_id": subcontracting_location.id, + "location_dest_id": self.location_dest_id.id, + "quantity": 100.0, + } + ) + + # Ensure product and location don't allow negative stock + self.product.allow_negative_stock = False + subcontracting_location.allow_negative_stock = False + + # This should not raise ValidationError because it's a subcontracting location + stock_picking_subcontract.action_confirm() + stock_picking_subcontract.button_validate() + + # Verify negative stock is allowed + quant = self.env["stock.quant"].search( + [ + ("product_id", "=", self.product.id), + ("location_id", "=", subcontracting_location.id), + ] + ) + self.assertEqual(quant.quantity, -100) diff --git a/views/product_product_views.xml b/views/product_product_views.xml new file mode 100755 index 0000000..5256371 --- /dev/null +++ b/views/product_product_views.xml @@ -0,0 +1,28 @@ + + + + + stock_no_negative.product.template.form + product.template + + + + + + + + + stock_no_negative.product.category.form + product.category + + + + + + + + diff --git a/views/stock_location_views.xml b/views/stock_location_views.xml new file mode 100755 index 0000000..ac26b1c --- /dev/null +++ b/views/stock_location_views.xml @@ -0,0 +1,21 @@ + + + + + stock.location.form.allow_negative_stock + stock.location + + + + + + + +