From 727e0ad6afb6fa14b8b790fd1b09c30946aeb49c Mon Sep 17 00:00:00 2001 From: Suherdy Yacob Date: Mon, 12 Jan 2026 16:43:00 +0700 Subject: [PATCH] first commit --- README.rst | 180 ++++++ __init__.py | 5 + __manifest__.py | 17 + __pycache__/__init__.cpython-310.pyc | Bin 0 -> 216 bytes __pycache__/__init__.cpython-312.pyc | Bin 0 -> 207 bytes i18n/ar.po | 101 ++++ i18n/de.po | 99 ++++ i18n/es.po | 103 ++++ i18n/fr.po | 100 ++++ i18n/hr.po | 98 ++++ i18n/it.po | 102 ++++ i18n/nl.po | 84 +++ i18n/nl_NL.po | 99 ++++ i18n/pt.po | 100 ++++ i18n/pt_BR.po | 84 +++ i18n/stock_no_negative.pot | 83 +++ i18n/tr.po | 94 +++ i18n/zh_CN.po | 84 +++ models/__init__.py | 7 + models/__pycache__/__init__.cpython-310.pyc | Bin 0 -> 295 bytes models/__pycache__/__init__.cpython-312.pyc | Bin 0 -> 288 bytes models/__pycache__/product.cpython-310.pyc | Bin 0 -> 1060 bytes models/__pycache__/product.cpython-312.pyc | Bin 0 -> 1258 bytes .../stock_location.cpython-310.pyc | Bin 0 -> 598 bytes .../stock_location.cpython-312.pyc | Bin 0 -> 673 bytes .../__pycache__/stock_quant.cpython-310.pyc | Bin 0 -> 1771 bytes .../__pycache__/stock_quant.cpython-312.pyc | Bin 0 -> 2999 bytes models/product.py | 26 + models/stock_location.py | 14 + models/stock_quant.py | 67 +++ pyproject.toml | 3 + readme/CONFIGURE.md | 22 + readme/CONTRIBUTORS.md | 23 + readme/DESCRIPTION.md | 15 + readme/HISTORY.md | 24 + readme/USAGE.md | 4 + static/description/icon.png | Bin 0 -> 9455 bytes static/description/index.html | 536 ++++++++++++++++++ tests/__init__.py | 8 + .../test_stock_no_negative.cpython-312.pyc | Bin 0 -> 13047 bytes tests/test_stock_no_negative.py | 273 +++++++++ views/product_product_views.xml | 28 + views/stock_location_views.xml | 21 + 43 files changed, 2504 insertions(+) create mode 100755 README.rst create mode 100755 __init__.py create mode 100755 __manifest__.py create mode 100755 __pycache__/__init__.cpython-310.pyc create mode 100644 __pycache__/__init__.cpython-312.pyc create mode 100755 i18n/ar.po create mode 100755 i18n/de.po create mode 100755 i18n/es.po create mode 100755 i18n/fr.po create mode 100755 i18n/hr.po create mode 100755 i18n/it.po create mode 100755 i18n/nl.po create mode 100755 i18n/nl_NL.po create mode 100755 i18n/pt.po create mode 100755 i18n/pt_BR.po create mode 100755 i18n/stock_no_negative.pot create mode 100755 i18n/tr.po create mode 100755 i18n/zh_CN.po create mode 100755 models/__init__.py create mode 100755 models/__pycache__/__init__.cpython-310.pyc create mode 100644 models/__pycache__/__init__.cpython-312.pyc create mode 100755 models/__pycache__/product.cpython-310.pyc create mode 100644 models/__pycache__/product.cpython-312.pyc create mode 100755 models/__pycache__/stock_location.cpython-310.pyc create mode 100644 models/__pycache__/stock_location.cpython-312.pyc create mode 100755 models/__pycache__/stock_quant.cpython-310.pyc create mode 100644 models/__pycache__/stock_quant.cpython-312.pyc create mode 100755 models/product.py create mode 100755 models/stock_location.py create mode 100755 models/stock_quant.py create mode 100755 pyproject.toml create mode 100755 readme/CONFIGURE.md create mode 100755 readme/CONTRIBUTORS.md create mode 100755 readme/DESCRIPTION.md create mode 100755 readme/HISTORY.md create mode 100755 readme/USAGE.md create mode 100755 static/description/icon.png create mode 100755 static/description/index.html create mode 100755 tests/__init__.py create mode 100644 tests/__pycache__/test_stock_no_negative.cpython-312.pyc create mode 100755 tests/test_stock_no_negative.py create mode 100755 views/product_product_views.xml create mode 100755 views/stock_location_views.xml 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 0000000000000000000000000000000000000000..4ac047357b19ce032c28131275d60e583c09ae5c GIT binary patch literal 216 zcmYjLK?=e!5KL+j5eofAFAd&2i1q~{coQMSY%s+pTbfqr6a0l&zu_}__2d^kx%K40 z&g?Ag&SH|x8R5RU*(d31H2;Xm6jUK40tUQdjvqK@5baqud2{5*O-?ec71EnYq}v25Eo0KyU?Oj0O*6DR$rt? g>CIWS<<$r&OJBB9=5;^JRutYVNJ$c zY`OU cE8+ke4YIiy@~ literal 0 HcmV?d00001 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 0000000000000000000000000000000000000000..aa9badad9867db1dfd6bd2705cade8bb3acd2b7e GIT binary patch literal 295 zcmYk0J5Izf5Qgm}JOsOlV~{2m9R-3t0YainV;PS{B4SVQ11U$~7PMR`EfrVTiW#(@g?FhGsoVl0libH3@*ab zEym*W&YRhU&SE!$dR+2=!6G+-wsvli%%jE*Np51jfasSNbzluNDoHD$C0fRjZSR-aGn2WmjWx*_J zfiRZU>P| literal 0 HcmV?d00001 diff --git a/models/__pycache__/product.cpython-310.pyc b/models/__pycache__/product.cpython-310.pyc new file mode 100755 index 0000000000000000000000000000000000000000..e6e88a44a5bab83f016cd86925d12b597f4a1e7b GIT binary patch literal 1060 zcmZ`&J#P~+81{W7QAE{JAtnZIC=%qv)&W5a17aYmM5kL$&e`d?`0hPyCs9~oVcMdwXMo?d$W;`S*a3A8xVRLR=hUH%B;d z!aYR_FYyXL@jdd2aGwXC2oHpR5qu1i5Rs5ah(sL{@xfa%8UDm*GV$xdLJGxAJt%++ zWikivHoQWU?(zXn1W7zj5}$iXAi|5tCHvwf10Et}RPUc^;LFUuWLBJm-h4QyT40*> z$hQ-vU`{4Zt9>j|J$wP61S{)%%sn>kADcFh-4eZzr<&pSJkFsIu~}j`-%K|w zgZXj6DppQ2m`Wa>Zfp)^hT6oy0iGPi*%E0A#yP$>ZWz-Nu5!e!jB#K1YNI2}Ho|`C zd^IAW7y7<8{?p+9m0LTT?MQkiib|n~*OJ;E`SrT1Un>GdaY{f5vPNJi+v0W?!Yxe^ zlh*XwTuDh;iTdWu-i$y#*7{`E5|8kZaNL5fn35drgtU#6f@KusqSr#Uvjc}6;^4Zz+plJmK^;2v l8hdE>qw465`)-W}O~h?f_2_A9xMv=Q7?<}TJPhuS{{r3ZfLL+*o_wz1U&xHQv38Ooc># z091jBh9(Lvzk-Smxk93%L6LF@DHXGRr>79C7wK~R!A9t- z*|e+EIQtPe+lV2C1xm1w6U(tS!WBZf&@2($RMp_{{R13h4l{^ABXFaQoeF zCaG_g?l9(psnSg(<3fcd+^&Wi?Jq)P8*$`ghJ1@*-?k8UHmagpaQ$W0dwUnxzV8-J zzf;bx=8}yUhAVu=4KrGg(o#~+j2u(p35!EL%fREPtIxeN_ z0OE$5lQ9)B0~rEsB>_%yZtlt&RVH!_>e!5hAOlVY0+b48vAOjL;lgPMqlD|Xs@{xs zbPQ#P7V<8%dnG?*nm%Vt)&8@_n{SQTsmoyg9GpFK7fgK-ySL0XV literal 0 HcmV?d00001 diff --git a/models/__pycache__/stock_location.cpython-310.pyc b/models/__pycache__/stock_location.cpython-310.pyc new file mode 100755 index 0000000000000000000000000000000000000000..7671cada1382473f585dd27b381386f94cd06eff GIT binary patch literal 598 zcmYjPy^ho{5VjrfUbMPJ6o`t#6;hf+bQA~{8X7o(u1jP2Y#+(tc$U~Xw6xGr@D8-R z0*~UBidUe4ad!8(k!IwXZ~S?F4D0oZpnQM!wfjj4`IVj9!qM5FhKFbjG1QTPwsc@E zqvRzq%oJaUDJ;7xJ{K)F{G3#a-jbta_c0LHqYbNIFt=xk8K12mSj)R2}L z+7^~yEi&~t+AjB$X?-3+zkdZ<#UA{pJ0ZKZJNj>*I|mxw4cS+_l_PEskQSA&%@t)#?eCYR5rQhu0} zJ0PX(eP_o$jtsj=Coce;RX*KQxiWdIOwN`4Amb@!&E`+vbTHT^OnBYw>YdVXu^H4> z`C3EWo94|fcHlJ*u>k|{=y9Vbq#2Yk;6pPrN)I=0H2ZlUfSZoCyD_h`L=AcCCE=7a uMpu95_*u0`JoC>DjCna}@j7pAhW{75Kdm!fKG{#>DgGT2GW%VAzxW5q)1}4$ literal 0 HcmV?d00001 diff --git a/models/__pycache__/stock_location.cpython-312.pyc b/models/__pycache__/stock_location.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..e7e41619ba53ea2967846144a449682e2adc6608 GIT binary patch literal 673 zcmYjOy>8S%5T0H8oYRTBAdw;pAhw}!6N@IIKqrv`D2@n3f#haoyc_JW_Fn9D(4|5m z4-l#g6*US^z$;MEAy-ILbdfFyDHXFmJ7J31Z+`cinQz1408rlN+v!&V@XH#V^jfeO zBbb8#K@?;pBpD4U0Sb>n&?69zq%@;0g?__2ED-IJd;{gbUv_}uo7iMuM|9DDuw2KuxPhiUS-FAg%prvQ9ROkHo=3dnNAt;t$sa)2)cyk$ C*0EFo literal 0 HcmV?d00001 diff --git a/models/__pycache__/stock_quant.cpython-310.pyc b/models/__pycache__/stock_quant.cpython-310.pyc new file mode 100755 index 0000000000000000000000000000000000000000..7e122aae978aebb3fdc1b5d711c52f93b5be666e GIT binary patch literal 1771 zcmZWqPirGL6qhub@nmc#+1)hDQVMfu$a)JlrI#K`yU>3Wx?xLbp@w#tjAUn=nbAa& zv+E>=()7|pzC&{C$LQD4wNThwq4d(jrcWBj@zOEUJUzX6@2B5;dKTSogy6aQ<53i>ENu7ceozaDhrZ#3dP$(i?i%>PSILf9N~eD}u5$Y?a|K#OO=JDD!_o z%;(-@?ZO*IEZ9fA)+_LZdY-|F@zOdsK`9t7R1cet@6sY?sm?|9xs*cohzT-Ljq zt*qQXwhI&0Qm|>Jlbo5*sd7Eb@JhTE$RUhRVW>Z0I!6;U!B-IS8p}0?_ML{2N9f9f z;CPNTW%LmeZ!azVYh1^Twu9#6oT#1VyU)-MU(LO9Z{kma4iXXtZTAi*9kzCYK&CK!S+i&!zeUTAP#R%gX)dd@FfA?9O6A(1({5Y{9q&A@(vtUVO;IbI{{6j}##vfbuzAuH8ISd0 zu3{HA7B!cy?&1TUrBh`$`C_l&C%mxet!hyk@$Q-RJ8xbI#GJ=3#I#`cdr|WEmZ13Y z*>Q4PL-SrNs#VV-Yl7~c*}^Wk=1Jo=PAhijqrnJZE&Q}7#0$>ik&w*^ZXv-Z1JD%< zr~Mw44}d}wT9+20iSD47?i&$%4~)mF6Vrn0ruj>4yfN3Ndq+JiWdPnJvF0|+T3?lcLxCq(fG~%eG9xSSV|6N$$}rV{igaBqG^?CoTQBXhU60n z^6zMH2pJrx=^>YF)}N)BICxs7bz1c^(a+i7`Ai>*Dub>y5KM^oKOAIJI93ThM5P9f zjMYjWG!1UC;V|~=ncRV3T^QWq&uM1-qEy0Mmcy4f%=wb^6Z!va|L7PtzXI7$%P5Br#Ev0K93zLGC2BR0~%x z3E1QnP>^fs0YDQ$X(+g8d8Hh5N7l;p9vDg66m6=N6+%RDMAos>?)BP+TDn}Ool9xE zw%M@CK7X0<+BSaGq&;$4EkvQ3y0cYe{Ac#B!ueZ?Y_le9coTlqoaJNNI~;~B+5xV% Gz5f6-PZEm& literal 0 HcmV?d00001 diff --git a/models/__pycache__/stock_quant.cpython-312.pyc b/models/__pycache__/stock_quant.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..688edfde82d176882c76cccaf430ae5f9b888aab GIT binary patch literal 2999 zcma)8O>7%Q6rS~ZZO2|Gw(G=ho1abFxZac`ErPTOB?q8sg=o_fK&mCW+IlC+y6atc zb{*;vO9fS;l`0Xqq?{-xKTOvb~G9Ws89&Zqf{ zT#j>PRtpq^xnMDr3o*b4jswMi1t=V33keUj-EXi>ITx-3#NY!u5LwJfn3YQ^4ipUq zYNp8Gp0{L8Rb)#w^kWDaNc7=A-q07+MI2qw4B3+MMzJI#m~UeP;BO5YFA-rKK#*e; zkn<@_j`acPFSy&@?f}SE6n4Yc)KTIat?68#!ifPKp0|wrg)?PYw^kx1mB%mA0S=dt zp_KEMq$)V%s#L3zXSRWN5t;6`Q6ud-SOtY`lb%>;E$xkIU+~Ugn^IvmTG)uGJOxYf z?WEqpZEh`*Y_a8CWvq}lulT9`B?gT!PeJ_O>^Kkq5@UuL@Z=gisQOlYrpx>lu*SUo zHUL$2mF;vO1(K0ZemrG4OjvgI=zh|kM4=-})lU#&|J#~V0@Z-E!?W+~s|3CDL4BPq z3P}&4-4ooDHzHnXPj2^0XqCCu<=q!0d?5>f$z1=GSa(R{!mhTbClTv!p2Dc#vuXEA zWSQdsxaueEomGysKdlB3bG@$c)gZwcY}F zVfA?VRM_v$w)>|7vehKC;ti%&yxyhAvtfU`l6&6dAB?wT0uy~j6GgVQq)D&G9Ua+ zK2kSBs%}A~%Nh<6XWdjS3|69oW>~^VR+o!Vqy^VcrIp56qb%fQT_=;v%?bEQR#kY%C~FG!H;Pd3q9aQz8$%>Rf0yVz6LEQHDLu=SK1g&j)W|wFKe3d4pf8%12uQxQ6erAMCZkC zT=XM)6>*3{Qi?d-rh=pYT2bu9EYz2Ake&p*WMOs@S~%JcJsBcefKh}(CS4Mb&rxy- z;t)AFxJVz4h)04Bf=jX~TNXlb>QAdQqD2fx=-FEaax;W0oAM&W0Ya1bAo{*wUQx}G zCRf~;pa4|`$gYdBg=kd}L)dRZZ2>bSoNyDgO^D&@D{=9Zv#Gihfgoy@eInXbJsznOXC5$hY=|64qD zBWlMpwPtBuUn_@=tXXYTd%t(Bcr z)=p(>qqBE=j@QSYb;f4wu^DIVtUY$N790IN0OEr;;HUa0`WN!|!*^q^{NUS+y;^Ux zIk=fT=nhBhXK_-*$x)ns_WrK^wP1Zmdj06fNAC>|-R$`aeye@0 zIg@kt}-vm2xaK3b>`>~RLh?13?7VA>v-t`BGH zd$XUUA;ha@*& zB?*TmiKIkXgGA;fS_L%sDoVsg)t4YrEkuhj4wGAFk{YJ!rkgHb0(6k*(FASiqv$qN zFf*jN0_r{emrva1c3htS;M{xX)?T|gy2*__^gYid8Y~EmyHXQ^I-L?D*l#F?;a)3u zNko6ea1T|E!%KNsqCcc&^NxjGtz{UR*}U@f5@7Bm{e^9^nIwuf!@ojDo7 +# 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 0000000000000000000000000000000000000000..3a0328b516c4980e8e44cdb63fd945757ddd132d GIT binary patch literal 9455 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 z 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 Xviia!U7SGha1wx#SCgwmn*{w2TRX*I literal 0 HcmV?d00001 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 0000000000000000000000000000000000000000..a2fe58f190a0ee0d5c6307cf3310d71ca1c15d34 GIT binary patch literal 13047 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 N5BmR;!C%t&zX6cWN^k%G literal 0 HcmV?d00001 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 + + + + + + + +