From cf41528e89be99a05aa9367878166dc0f8529271 Mon Sep 17 00:00:00 2001 From: abelsrzz Date: Mon, 16 Mar 2026 20:51:34 +0100 Subject: [PATCH 1/4] [ADD] repair_group This module allows grouping multiple repair orders by selecting completed stock pickings (delivery notes) from a specific customer. --- repair_group/README.rst | 170 +++++ repair_group/__init__.py | 3 + repair_group/__manifest__.py | 32 + repair_group/data/repair_group_sequence.xml | 14 + repair_group/data/stock_settings.xml | 4 + repair_group/i18n/es.po | 639 ++++++++++++++++++ repair_group/i18n/gl.po | 639 ++++++++++++++++++ repair_group/i18n/repair_group.pot | 621 +++++++++++++++++ repair_group/models/__init__.py | 5 + repair_group/models/repair_group.py | 204 ++++++ repair_group/models/repair_order.py | 64 ++ repair_group/models/res_config_settings.py | 38 ++ repair_group/models/sale_order.py | 101 +++ repair_group/models/stock_picking.py | 51 ++ repair_group/pyproject.toml | 3 + repair_group/readme/CONFIGURE.md | 14 + repair_group/readme/CONTRIBUTORS.md | 1 + repair_group/readme/DESCRIPTION.md | 20 + repair_group/readme/USAGE.md | 35 + repair_group/security/ir.model.access.csv | 4 + repair_group/static/description/icon.png | Bin 0 -> 9455 bytes repair_group/static/description/index.html | 521 ++++++++++++++ repair_group/tests/__init__.py | 4 + repair_group/tests/test_repair_group.py | 134 ++++ repair_group/views/repair_groups.xml | 168 +++++ repair_group/views/repair_groups_menu.xml | 12 + repair_group/views/repair_order_views.xml | 44 ++ repair_group/views/sale_order_views.xml | 99 +++ repair_group/views/stock_picking_views.xml | 31 + .../views/wizard_repair_group_quotation.xml | 44 ++ repair_group/wizards/__init__.py | 1 + .../wizards/wizard_repair_group_quotation.py | 105 +++ 32 files changed, 3825 insertions(+) create mode 100644 repair_group/README.rst create mode 100644 repair_group/__init__.py create mode 100644 repair_group/__manifest__.py create mode 100644 repair_group/data/repair_group_sequence.xml create mode 100644 repair_group/data/stock_settings.xml create mode 100644 repair_group/i18n/es.po create mode 100644 repair_group/i18n/gl.po create mode 100755 repair_group/i18n/repair_group.pot create mode 100644 repair_group/models/__init__.py create mode 100644 repair_group/models/repair_group.py create mode 100644 repair_group/models/repair_order.py create mode 100644 repair_group/models/res_config_settings.py create mode 100644 repair_group/models/sale_order.py create mode 100644 repair_group/models/stock_picking.py create mode 100644 repair_group/pyproject.toml create mode 100644 repair_group/readme/CONFIGURE.md create mode 100644 repair_group/readme/CONTRIBUTORS.md create mode 100644 repair_group/readme/DESCRIPTION.md create mode 100644 repair_group/readme/USAGE.md create mode 100644 repair_group/security/ir.model.access.csv create mode 100644 repair_group/static/description/icon.png create mode 100644 repair_group/static/description/index.html create mode 100644 repair_group/tests/__init__.py create mode 100644 repair_group/tests/test_repair_group.py create mode 100644 repair_group/views/repair_groups.xml create mode 100644 repair_group/views/repair_groups_menu.xml create mode 100644 repair_group/views/repair_order_views.xml create mode 100644 repair_group/views/sale_order_views.xml create mode 100644 repair_group/views/stock_picking_views.xml create mode 100644 repair_group/views/wizard_repair_group_quotation.xml create mode 100644 repair_group/wizards/__init__.py create mode 100644 repair_group/wizards/wizard_repair_group_quotation.py diff --git a/repair_group/README.rst b/repair_group/README.rst new file mode 100644 index 00000000..5b5c5644 --- /dev/null +++ b/repair_group/README.rst @@ -0,0 +1,170 @@ +============ +Repair Group +============ + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:167a01bd8fdc1b8a648ca6697401ffc5a3239fa4724743c37385817532413871 + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |badge1| image:: https://img.shields.io/badge/maturity-Alpha-red.png + :target: https://odoo-community.org/page/development-status + :alt: Alpha +.. |badge2| image:: https://img.shields.io/badge/licence-LGPL--3-blue.png + :target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html + :alt: License: LGPL-3 +.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Frepair-lightgray.png?logo=github + :target: https://github.com/OCA/repair/tree/18.0/repair_group + :alt: OCA/repair +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/repair-18-0/repair-18-0-repair_group + :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/repair&target_branch=18.0 + :alt: Try me on Runboat + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module allows grouping multiple repair orders by selecting +completed stock pickings (delivery notes) from a specific customer. + +**Key features:** + +- Create a **Repair Group** from one or more done stock pickings + belonging to the same customer (consignment/owner-tracked pickings). +- Automatically generate individual repair orders for each product in + the selected pickings. +- Supervisor approval workflow: technicians request approval and + supervisors validate or reject each repair order. +- Merge individual repair quotations (sale orders) into a single + consolidated sale order per group using a selection wizard. +- Full traceability: each repair order is linked to its origin picking + move line and to the repair group. +- State tracking across the full lifecycle: *Draft → Repairs Created → + Quotation Requested → Under Repair → Repaired* (or *Cancelled*). +- Mail thread and activity tracking on repair groups. +- Smart button on stock pickings to navigate directly to the associated + repair group. + +.. IMPORTANT:: + This is an alpha version, the data model and design can change at any time without warning. + Only for development or testing purpose, do not use in production. + `More details on development status `_ + +**Table of contents** + +.. contents:: + :local: + +Configuration +============= + +Consignment (Stock Ownership Tracking) +-------------------------------------- + +This module requires the **Consignment** feature (*Inventory → +Configuration → Settings → Traceability → Consignment*) to be enabled so +that stock pickings track the owner of the goods. + +The feature is **enabled automatically** when this module is installed +via a post-install hook. No manual configuration is required. + +Supervisors +----------- + +Each repair group has a **Supervisor** field. Assign the user +responsible for approving repair orders in the group. By default, the +user who creates the group is set as supervisor. + +Usage +===== + +Creating a Repair Group from a Stock Picking +-------------------------------------------- + +1. Open a **done** stock picking whose products are owned by a customer + (Consignment must be enabled — this module enables it automatically + on installation). +2. Click **Create Repair Group** in the picking form header. +3. The system creates a new repair group pre-filled with the customer + and the picking. You are redirected to the repair group form. + +Repair Group Workflow +--------------------- + +1. **Draft**: Add additional pickings if needed. Click **Create + Repairs** to generate one repair order per move line. +2. **Repairs Created**: Review the generated repair orders in the + *Repair Orders* tab. Click **Request Quotations** to move forward, or + **Return to Draft** to undo. +3. **Quotation Requested**: Technicians fill in repair details and + generate quotations from each repair order. Once all quotations + exist, click **Create Complete Sale Order** to open the wizard. +4. **Wizard – Select Quotations**: Choose which quotations to merge and + click **Create Complete Sale Order**. The system consolidates the + selected lines into a single sale order. +5. **Quotation Requested (with main order)**: Review the consolidated + order. Click **Confirm Complete Sale Order** to confirm it and move + to *Under Repair*, or **Cancel Complete Sale Order** to discard it. +6. **Under Repair**: Repairs are being performed. Click **End Repair + Group** once all repair orders are in *Done* state. +7. **Repaired**: The group is complete. + +Supervisor Approval +------------------- + +On each repair order that belongs to a group, the responsible user can +click **Request Approval** to submit it for supervisor validation. The +supervisor (configured on the repair group) can then **Validate** or +**Reject** the repair directly from the repair order form. + +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 +------- + +* Grupo Isonor + +Contributors +------------ + +- Abel Suárez abel@suarezmuinho.com> + +Maintainers +----------- + +This module is maintained by the OCA. + +.. image:: https://odoo-community.org/logo.png + :alt: Odoo Community Association + :target: https://odoo-community.org + +OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use. + +.. |maintainer-abelsrzz| image:: https://github.com/abelsrzz.png?size=40px + :target: https://github.com/abelsrzz + :alt: abelsrzz + +Current `maintainer `__: + +|maintainer-abelsrzz| + +This module is part of the `OCA/repair `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/repair_group/__init__.py b/repair_group/__init__.py new file mode 100644 index 00000000..f9751317 --- /dev/null +++ b/repair_group/__init__.py @@ -0,0 +1,3 @@ +from . import models +from . import wizards +from .models.res_config_settings import post_init_hook diff --git a/repair_group/__manifest__.py b/repair_group/__manifest__.py new file mode 100644 index 00000000..8297e311 --- /dev/null +++ b/repair_group/__manifest__.py @@ -0,0 +1,32 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +{ + "name": "Repair Group", + "summary": "Create a group of repairs selecting Stock pickings", + "version": "18.0.1.0.0", + "category": "Repair", + "website": "https://github.com/OCA/repair", + "author": "Grupo Isonor, Odoo Community Association (OCA)", + "maintainers": ["abelsrzz"], + "license": "LGPL-3", + "development_status": "Alpha", + "depends": [ + "repair", + "sale", + "stock", + ], + "data": [ + "security/ir.model.access.csv", + "data/repair_group_sequence.xml", + "views/wizard_repair_group_quotation.xml", + "views/repair_groups.xml", + "views/repair_groups_menu.xml", + "views/stock_picking_views.xml", + "views/sale_order_views.xml", + "views/repair_order_views.xml", + ], + "application": False, + "installable": True, + "post_init_hook": "post_init_hook", +} diff --git a/repair_group/data/repair_group_sequence.xml b/repair_group/data/repair_group_sequence.xml new file mode 100644 index 00000000..52e0211c --- /dev/null +++ b/repair_group/data/repair_group_sequence.xml @@ -0,0 +1,14 @@ + + + + + Repair Group + repair.group + RG/ + 5 + 1 + 1 + + + diff --git a/repair_group/data/stock_settings.xml b/repair_group/data/stock_settings.xml new file mode 100644 index 00000000..b366c674 --- /dev/null +++ b/repair_group/data/stock_settings.xml @@ -0,0 +1,4 @@ + + + + diff --git a/repair_group/i18n/es.po b/repair_group/i18n/es.po new file mode 100644 index 00000000..750171e9 --- /dev/null +++ b/repair_group/i18n/es.po @@ -0,0 +1,639 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * repair_group +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2026-03-16 19:36+0000\n" +"PO-Revision-Date: 2026-03-16 19:36+0200\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: es_ES\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_order__state +msgid "" +"* The 'New' status is used when a user is encoding a new and unconfirmed repair order.\n" +"* The 'Confirmed' status is used when a user confirms the repair order.\n" +"* The 'Under Repair' status is used when the repair is ongoing.\n" +"* The 'Repaired' status is set when repairing is completed.\n" +"* The 'Cancelled' status is used when user cancel repair order." +msgstr "" +"* El estado 'Nuevo' se utiliza cuando un usuario registra una nueva orden de reparación sin confirmar.\n" +"* El estado 'Confirmado' se utiliza cuando un usuario confirma la orden de reparación.\n" +"* El estado 'En Reparación' se utiliza cuando la reparación está en curso.\n" +"* El estado 'Reparado' se establece cuando se completa la reparación.\n" +"* El estado 'Cancelado' se utiliza cuando el usuario cancela la orden de reparación." + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +msgid "" +"
\n" +" Main Sale Order:" +msgstr "" +"
\n" +" Pedido de Venta Principal:" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_needaction +msgid "Action Needed" +msgstr "Acción requerida" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_ids +msgid "Activities" +msgstr "Actividades" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_exception_decoration +msgid "Activity Exception Decoration" +msgstr "Decoración de Actividad de Excepción" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_state +msgid "Activity State" +msgstr "Estado de la actividad" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_type_icon +msgid "Activity Type Icon" +msgstr "Icono de tipo de actvidad" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Add internal notes." +msgstr "Añadir notas internas." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_attachment_count +msgid "Attachment Count" +msgstr "Contador de adjuntos." + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Cancel Complete Sale Order" +msgstr "Cancelar Pedido de Venta Completo" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Cancel" +msgstr "Cancelar" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__cancel +msgid "Cancelled" +msgstr "Cancelado" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Cannot finish the group. The following repairs are not finished: {}" +msgstr "No se puede finalizar el grupo. Las siguientes reparaciones no están finalizadas: {}" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__carrier_id +msgid "Carrier" +msgstr "Transportista" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/wizards/wizard_repair_group_quotation.py:0 +msgid "Combined order from Repair Group: {}" +msgstr "Pedido combinado desde Grupo de Reparación: {}" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__company_id +msgid "Company" +msgstr "Compañía" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_res_config_settings +msgid "Config Settings" +msgstr "Ajustes de Configuración" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Confirm Complete Sale Order" +msgstr "Confirmar Presupuesto Completo" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_res_config_settings__group_stock_tracking_owner +msgid "Consignment" +msgstr "Consignación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Create Complete Sale Order" +msgstr "Crear Presupuesto Completo" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_picking_form_extended +msgid "Create repair group" +msgstr "Crear grupo de reparación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Create repairs" +msgstr "Crear reparaciones" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__create_uid +msgid "Created by" +msgstr "Creado por" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__create_date +msgid "Created on" +msgstr "Creado en" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__currency_id +msgid "Currency" +msgstr "Moneda" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__picking_ids +msgid "Delivery note" +msgstr "Albarán" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Delivery notes" +msgstr "Albaranes" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__display_name +msgid "Display Name" +msgstr "Nombre para mostrar" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "End Repair Group" +msgstr "Finalizar Grupo de Reparaciones" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__scheduled_datetime +msgid "Estimated date" +msgstr "Fecha estimada" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_follower_ids +msgid "Followers" +msgstr "Seguidores" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_partner_ids +msgid "Followers (Partners)" +msgstr "Seguidores (Contactos)" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_type_icon +msgid "Font awesome icon e.g. fa-tasks" +msgstr "Icono de Font Awesome p. ej. fa-tasks" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__complete_sale_order_id +msgid "General Sale Order" +msgstr "Venta general" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__has_message +msgid "Has Message" +msgstr "Tiene mensaje" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__id +msgid "ID" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_exception_icon +msgid "Icon" +msgstr "Icono" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_exception_icon +msgid "Icon to indicate an exception activity." +msgstr "Icono para indicar una actividad de excepción." + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_needaction +msgid "If checked, new messages require your attention." +msgstr "Si está marcada, hay nuevos mensajes que requieren su atención." + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_has_error +msgid "If checked, some messages have a delivery error." +msgstr "Si está marcada, algunos mensajes tienen error de envío." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__selected +msgid "Include" +msgstr "Incluir" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__internal_notes +msgid "Internal Notes" +msgstr "Notas Internas" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Internal notes" +msgstr "Notas internas" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_is_follower +msgid "Is Follower" +msgstr "Es seguidor" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__is_repair_main_order +msgid "Is Repair Main Order" +msgstr "Es Pedido Principal de Reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__write_uid +msgid "Last Updated by" +msgstr "Última actualización por" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__write_date +msgid "Last Updated on" +msgstr "Última actualización el" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__main_repair_sale_id +msgid "Main Repair Sale Order" +msgstr "Pedido de Venta Principal de Reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_has_error +msgid "Message Delivery error" +msgstr "Error de envío de mensaje" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_ids +msgid "Messages" +msgstr "Mensajes" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__my_activity_date_deadline +msgid "My Activity Deadline" +msgstr "Fecha límite de mi actividad" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__draft +msgid "New" +msgstr "Nuevo" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_date_deadline +msgid "Next Activity Deadline" +msgstr "Próxima fecha límite de la actividad" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_summary +msgid "Next Activity Summary" +msgstr "Próximo resumen de la actividad" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_type_id +msgid "Next Activity Type" +msgstr "Próximo tipo de la actividad" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "No sale orders found to merge." +msgstr "No se encontraron ventas que fusionar." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_needaction_counter +msgid "Number of Actions" +msgstr "Número de acciones" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_has_error_counter +msgid "Number of errors" +msgstr "Número de errores" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_needaction_counter +msgid "Number of messages requiring action" +msgstr "Número de mensajes que requieren acción" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_has_error_counter +msgid "Number of messages with delivery error" +msgstr "Número de mensajes con error de envío" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "" +"Only the supervisor and the responsible can request approval for this " +"repair." +msgstr "" +"Solo el supervisor y el responsable pueden solicitar aprobación para esta " +"reparación." + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Only the supervisor can approve this repair." +msgstr "Solo el supervisor puede aprobar esta reparación." + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Only the supervisor can reject this repair." +msgstr "Solo el supervisor puede rechazar esta reparación." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__partner_id +msgid "Partner" +msgstr "Contacto" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/wizards/wizard_repair_group_quotation.py:0 +msgid "Please select at least one quotation to merge." +msgstr "Por favor, seleccione al menos un presupuesto para fusionar." + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Print" +msgstr "Imprimir" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__product_id +msgid "Product" +msgstr "Producto" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_stock_move_line +msgid "Product Moves (Stock Move Line)" +msgstr "Movimiento de producto (Líneas de transferencia)" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__sale_order_id +msgid "Quotation" +msgstr "Presupuesto" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__quotation_requested +msgid "Quotation Requested" +msgstr "Presupuesto solicitado" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Quotation requested for this repair order." +msgstr "Presupuesto solicitado para esta orden de reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__line_ids +msgid "Quotations" +msgstr "Presupuestos" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Quotations where requested." +msgstr "Los presupuestos fueron solicitados." + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Reject" +msgstr "Rechazar" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_repair_order__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_sale_order__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_stock_picking__repair_group_id +#: model_terms:ir.ui.view,arch_db:repair_group.view_picking_form_extended +msgid "Repair Group" +msgstr "Grupo de Reparación" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_init +msgid "Repair Group Init" +msgstr "Inicialización de Grupo de Reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__name +msgid "Repair Group Reference" +msgstr "Referencia de Grupo de Reparación" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_order +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__repair_order_id +#: model:ir.model.fields,field_description:repair_group.field_stock_move_line__repair_order_id +msgid "Repair Order" +msgstr "Orden de reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__repair_order_ids +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Repair Orders" +msgstr "Órdenes de reparación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Repair group" +msgstr "Grupo de reparación" + +#. module: repair_group +#: model:ir.actions.act_window,name:repair_group.action_repair_group_tree +#: model:ir.ui.menu,name:repair_group.repair_group_menu +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form_filter +msgid "Repair groups" +msgstr "Grupos de reparación" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Repair rejected by supervisor. Please review it." +msgstr "Reparación rechazada por el supervisor. Por favor, revíselo." + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__done +msgid "Repaired" +msgstr "Reparado" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__repairs_created +msgid "Repairs Created" +msgstr "Reparaciones creadas" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Request Approval" +msgstr "Solicitar Aprobación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Request Quotations" +msgstr "Solicitar Presupuestos" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_user_id +msgid "Responsible User" +msgstr "Usuario responsable" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Return to draft" +msgstr "Volver a Borrador" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__sale_order_ids +msgid "Sale Orders" +msgstr "Pedidos de Venta" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_sale_order +msgid "Sales Order" +msgstr "Pedido de Venta" + +#. module: repair_group +#: model:ir.actions.act_window,name:repair_group.action_repair_group_quotation_wizard +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Select Quotations to Merge" +msgstr "Seleccionar presupuestos a fusionar" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__source_sale_order_ids +msgid "Source Sale Orders" +msgstr "Pedidos de Venta de Origen" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__state +#: model:ir.model.fields,field_description:repair_group.field_repair_order__state +msgid "Status" +msgstr "Estado" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_state +msgid "" +"Status based on activities\n" +"Overdue: Due date is already passed\n" +"Today: Activity date is today\n" +"Planned: Future activities." +msgstr "" +"Estado basado en actividades\n" +"Vencida: la fecha límite ya ha pasado\n" +"Hoy: la fecha límite es hoy\n" +"Planificada: actividades futuras." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__supervisor_id +msgid "Supervisor" +msgstr "Supervisor" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +msgid "" +"This quotation is part of a Repair Group.\n" +" Please use the Complete Sale Order of the group to manage it.\n" +"
\n" +" Group:" +msgstr "" +"Esta cotización es parte de un Grupo de Reparación.\n" +" Por favor, utilice el Pedido de Venta Completo del grupo para gestionarlo.\n" +"
\n" +" Grupo:" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "This repair order is not part of a group." +msgstr "Esta orden de reparación no pertenece a ningún grupo." + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_order__state__to_approve +msgid "To Approve" +msgstr "Por Aprobar" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__amount_total +msgid "Total" +msgstr "Total" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_stock_picking +msgid "Transfer" +msgstr "Transferencia" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_exception_decoration +msgid "Type of the exception activity on record." +msgstr "Tipo de actividad de excepción en el registro." + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__under_repair +msgid "Under Repair" +msgstr "Bajo reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__user_id +msgid "User" +msgstr "Usuario" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Validate" +msgstr "Validar" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__website_message_ids +msgid "Website Messages" +msgstr "Mensajes de sitio web" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__website_message_ids +msgid "Website communication history" +msgstr "Historico de comunicación del sitio web" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__wizard_id +msgid "Wizard" +msgstr "Asistente" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_quotation_wizard_line +msgid "Wizard line for quotation selection" +msgstr "Línea del asistente para selección de presupuestos" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_quotation_wizard +msgid "Wizard to select quotations to merge" +msgstr "Asistente para seleccionar presupuestos a fusionar" diff --git a/repair_group/i18n/gl.po b/repair_group/i18n/gl.po new file mode 100644 index 00000000..4198ba7e --- /dev/null +++ b/repair_group/i18n/gl.po @@ -0,0 +1,639 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * repair_group +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2026-03-16 19:36+0000\n" +"PO-Revision-Date: 2026-03-16 19:36+0100\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: gl_ES\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_order__state +msgid "" +"* The 'New' status is used when a user is encoding a new and unconfirmed repair order.\n" +"* The 'Confirmed' status is used when a user confirms the repair order.\n" +"* The 'Under Repair' status is used when the repair is ongoing.\n" +"* The 'Repaired' status is set when repairing is completed.\n" +"* The 'Cancelled' status is used when user cancel repair order." +msgstr "" +"* O estado 'Novo' utilízase cando un usuario rexistra unha nova orde de reparación sen confirmar.\n" +"* O estado 'Confirmado' utilízase cando un usuario confirma a orde de reparación.\n" +"* O estado 'En Reparación' utilízase cando a reparación está en curso.\n" +"* O estado 'Reparado' establécese cando se completa a reparación.\n" +"* O estado 'Cancelado' utilízase cando o usuario cancela a orde de reparación." + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +msgid "" +"
\n" +" Main Sale Order:" +msgstr "" +"
\n" +" Pedido de Venda Principal:" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_needaction +msgid "Action Needed" +msgstr "Acción requirida" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_ids +msgid "Activities" +msgstr "Actividades" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_exception_decoration +msgid "Activity Exception Decoration" +msgstr "Decoración de actividade de excepción" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_state +msgid "Activity State" +msgstr "Estado da actividade" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_type_icon +msgid "Activity Type Icon" +msgstr "Icona de tipo de actividade" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Add internal notes." +msgstr "Engadir notas internas." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_attachment_count +msgid "Attachment Count" +msgstr "Número de anexos" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Cancel Complete Sale Order" +msgstr "Cancelar pedido de venda completo" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Cancel" +msgstr "Cancelar" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__cancel +msgid "Cancelled" +msgstr "Cancelado" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Cannot finish the group. The following repairs are not finished: {}" +msgstr "Non se pode finalizar o grupo. As seguintes reparacións non están finalizadas: {}" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__carrier_id +msgid "Carrier" +msgstr "Transportista" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/wizards/wizard_repair_group_quotation.py:0 +msgid "Combined order from Repair Group: {}" +msgstr "Pedido combinado do Grupo de Reparación: {}" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__company_id +msgid "Company" +msgstr "Empresa" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_res_config_settings +msgid "Config Settings" +msgstr "Axustes de configuración" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Confirm Complete Sale Order" +msgstr "Confirmar pedido de venda completo" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_res_config_settings__group_stock_tracking_owner +msgid "Consignment" +msgstr "Consignación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Create Complete Sale Order" +msgstr "Crear pedido de venda completo" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_picking_form_extended +msgid "Create repair group" +msgstr "Crear grupo de reparación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Create repairs" +msgstr "Crear reparacións" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__create_uid +msgid "Created by" +msgstr "Creado por" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__create_date +msgid "Created on" +msgstr "Creado o" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__currency_id +msgid "Currency" +msgstr "Moeda" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__picking_ids +msgid "Delivery note" +msgstr "Albará" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Delivery notes" +msgstr "Albarás" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__display_name +msgid "Display Name" +msgstr "Nome para mostrar" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "End Repair Group" +msgstr "Finalizar grupo de reparacións" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__scheduled_datetime +msgid "Estimated date" +msgstr "Data estimada" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_follower_ids +msgid "Followers" +msgstr "Seguidores" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_partner_ids +msgid "Followers (Partners)" +msgstr "Seguidores (contactos)" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_type_icon +msgid "Font awesome icon e.g. fa-tasks" +msgstr "Icona Font Awesome p. ex. fa-tasks" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__complete_sale_order_id +msgid "General Sale Order" +msgstr "Pedido de venda xeral" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__has_message +msgid "Has Message" +msgstr "Ten mensaxe" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__id +msgid "ID" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_exception_icon +msgid "Icon" +msgstr "Icona" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_exception_icon +msgid "Icon to indicate an exception activity." +msgstr "Icona para indicar unha actividade de excepción." + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_needaction +msgid "If checked, new messages require your attention." +msgstr "Se está marcada, hai novos mensaxes que requiren a súa atención." + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_has_error +msgid "If checked, some messages have a delivery error." +msgstr "Se está marcada, algúns mensaxes teñen erro de envío." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__selected +msgid "Include" +msgstr "Incluír" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__internal_notes +msgid "Internal Notes" +msgstr "Notas internas" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Internal notes" +msgstr "Notas internas" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_is_follower +msgid "Is Follower" +msgstr "É seguidor" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__is_repair_main_order +msgid "Is Repair Main Order" +msgstr "É pedido principal de reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__write_uid +msgid "Last Updated by" +msgstr "Última actualización por" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__write_date +msgid "Last Updated on" +msgstr "Última actualización o" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__main_repair_sale_id +msgid "Main Repair Sale Order" +msgstr "Pedido de venda principal de reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_has_error +msgid "Message Delivery error" +msgstr "Erro de envío de mensaxe" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_ids +msgid "Messages" +msgstr "Mensaxes" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__my_activity_date_deadline +msgid "My Activity Deadline" +msgstr "Data límite da miña actividade" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__draft +msgid "New" +msgstr "Novo" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_date_deadline +msgid "Next Activity Deadline" +msgstr "Próxima data límite da actividade" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_summary +msgid "Next Activity Summary" +msgstr "Próximo resumo da actividade" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_type_id +msgid "Next Activity Type" +msgstr "Próximo tipo de actividade" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "No sale orders found to merge." +msgstr "Non se atoparon pedidos de venda para fusionar." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_needaction_counter +msgid "Number of Actions" +msgstr "Número de accións" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_has_error_counter +msgid "Number of errors" +msgstr "Número de erros" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_needaction_counter +msgid "Number of messages requiring action" +msgstr "Número de mensaxes que requiren acción" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_has_error_counter +msgid "Number of messages with delivery error" +msgstr "Número de mensaxes con erro de envío" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "" +"Only the supervisor and the responsible can request approval for this " +"repair." +msgstr "" +"Só o supervisor e o responsable poden solicitar aprobación para esta " +"reparación." + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Only the supervisor can approve this repair." +msgstr "Só o supervisor pode aprobar esta reparación." + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Only the supervisor can reject this repair." +msgstr "Só o supervisor pode rexeitar esta reparación." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__partner_id +msgid "Partner" +msgstr "Contacto" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/wizards/wizard_repair_group_quotation.py:0 +msgid "Please select at least one quotation to merge." +msgstr "Por favor, seleccione polo menos un orzamento para fusionar." + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Print" +msgstr "Imprimir" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__product_id +msgid "Product" +msgstr "Produto" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_stock_move_line +msgid "Product Moves (Stock Move Line)" +msgstr "Movementos de produto (liñas de transferencia)" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__sale_order_id +msgid "Quotation" +msgstr "Orzamento" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__quotation_requested +msgid "Quotation Requested" +msgstr "Orzamento solicitado" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Quotation requested for this repair order." +msgstr "Orzamento solicitado para esta orde de reparación." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__line_ids +msgid "Quotations" +msgstr "Orzamentos" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Quotations where requested." +msgstr "Os orzamentos foron solicitados." + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Reject" +msgstr "Rexeitar" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_repair_order__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_sale_order__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_stock_picking__repair_group_id +#: model_terms:ir.ui.view,arch_db:repair_group.view_picking_form_extended +msgid "Repair Group" +msgstr "Grupo de reparación" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_init +msgid "Repair Group Init" +msgstr "Inicialización do grupo de reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__name +msgid "Repair Group Reference" +msgstr "Referencia do grupo de reparación" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_order +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__repair_order_id +#: model:ir.model.fields,field_description:repair_group.field_stock_move_line__repair_order_id +msgid "Repair Order" +msgstr "Orde de reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__repair_order_ids +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Repair Orders" +msgstr "Ordes de reparación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Repair group" +msgstr "Grupo de reparación" + +#. module: repair_group +#: model:ir.actions.act_window,name:repair_group.action_repair_group_tree +#: model:ir.ui.menu,name:repair_group.repair_group_menu +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form_filter +msgid "Repair groups" +msgstr "Grupos de reparación" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Repair rejected by supervisor. Please review it." +msgstr "Reparación rexeitada polo supervisor. Por favor, revíseo." + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__done +msgid "Repaired" +msgstr "Reparado" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__repairs_created +msgid "Repairs Created" +msgstr "Reparacións creadas" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Request Approval" +msgstr "Solicitar aprobación" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Request Quotations" +msgstr "Solicitar orzamentos" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_user_id +msgid "Responsible User" +msgstr "Usuario responsable" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Return to draft" +msgstr "Volver a borrador" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__sale_order_ids +msgid "Sale Orders" +msgstr "Pedidos de venda" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_sale_order +msgid "Sales Order" +msgstr "Pedido de venda" + +#. module: repair_group +#: model:ir.actions.act_window,name:repair_group.action_repair_group_quotation_wizard +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Select Quotations to Merge" +msgstr "Seleccionar orzamentos a fusionar" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__source_sale_order_ids +msgid "Source Sale Orders" +msgstr "Pedidos de venda de orixe" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__state +#: model:ir.model.fields,field_description:repair_group.field_repair_order__state +msgid "Status" +msgstr "Estado" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_state +msgid "" +"Status based on activities\n" +"Overdue: Due date is already passed\n" +"Today: Activity date is today\n" +"Planned: Future activities." +msgstr "" +"Estado baseado en actividades\n" +"Vencida: a data límite xa pasou\n" +"Hoxe: a data límite é hoxe\n" +"Planificada: actividades futuras." + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__supervisor_id +msgid "Supervisor" +msgstr "Supervisor" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +msgid "" +"This quotation is part of a Repair Group.\n" +" Please use the Complete Sale Order of the group to manage it.\n" +"
\n" +" Group:" +msgstr "" +"Esta cotización é parte dun Grupo de Reparación.\n" +" Por favor, use o Pedido de Venda Completo do grupo para xestionalo.\n" +"
\n" +" Grupo:" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "This repair order is not part of a group." +msgstr "Esta orde de reparación non pertence a ningún grupo." + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_order__state__to_approve +msgid "To Approve" +msgstr "Para aprobar" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__amount_total +msgid "Total" +msgstr "Total" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_stock_picking +msgid "Transfer" +msgstr "Transferencia" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_exception_decoration +msgid "Type of the exception activity on record." +msgstr "Tipo de actividade de excepción no rexistro." + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__under_repair +msgid "Under Repair" +msgstr "En reparación" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__user_id +msgid "User" +msgstr "Usuario" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Validate" +msgstr "Validar" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__website_message_ids +msgid "Website Messages" +msgstr "Mensaxes do sitio web" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__website_message_ids +msgid "Website communication history" +msgstr "Histórico de comunicación do sitio web" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__wizard_id +msgid "Wizard" +msgstr "Asistente" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_quotation_wizard_line +msgid "Wizard line for quotation selection" +msgstr "Liña do asistente para selección de orzamentos" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_quotation_wizard +msgid "Wizard to select quotations to merge" +msgstr "Asistente para seleccionar orzamentos a fusionar" diff --git a/repair_group/i18n/repair_group.pot b/repair_group/i18n/repair_group.pot new file mode 100755 index 00000000..3e1112ce --- /dev/null +++ b/repair_group/i18n/repair_group.pot @@ -0,0 +1,621 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * repair_group +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 18.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2026-03-16 19:36+0000\n" +"PO-Revision-Date: 2026-03-16 19:36+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_order__state +msgid "" +"* The 'New' status is used when a user is encoding a new and unconfirmed repair order.\n" +"* The 'Confirmed' status is used when a user confirms the repair order.\n" +"* The 'Under Repair' status is used when the repair is ongoing.\n" +"* The 'Repaired' status is set when repairing is completed.\n" +"* The 'Cancelled' status is used when user cancel repair order." +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +msgid "" +"
\n" +" Main Sale Order:" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_needaction +msgid "Action Needed" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_ids +msgid "Activities" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_exception_decoration +msgid "Activity Exception Decoration" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_state +msgid "Activity State" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_type_icon +msgid "Activity Type Icon" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Add internal notes." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_attachment_count +msgid "Attachment Count" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Cancel Complete Sale Order" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Cancel" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__cancel +msgid "Cancelled" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Cannot finish the group. The following repairs are not finished: {}" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__carrier_id +msgid "Carrier" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/wizards/wizard_repair_group_quotation.py:0 +msgid "Combined order from Repair Group: {}" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__company_id +msgid "Company" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_res_config_settings +msgid "Config Settings" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Confirm Complete Sale Order" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_res_config_settings__group_stock_tracking_owner +msgid "Consignment" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Create Complete Sale Order" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_picking_form_extended +msgid "Create repair group" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Create repairs" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__create_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__create_uid +msgid "Created by" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__create_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__create_date +msgid "Created on" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__currency_id +msgid "Currency" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__picking_ids +msgid "Delivery note" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Delivery notes" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__display_name +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__display_name +msgid "Display Name" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "End Repair Group" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__scheduled_datetime +msgid "Estimated date" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_follower_ids +msgid "Followers" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_partner_ids +msgid "Followers (Partners)" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_type_icon +msgid "Font awesome icon e.g. fa-tasks" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__complete_sale_order_id +msgid "General Sale Order" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__has_message +msgid "Has Message" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__id +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__id +msgid "ID" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_exception_icon +msgid "Icon" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_exception_icon +msgid "Icon to indicate an exception activity." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_needaction +msgid "If checked, new messages require your attention." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_has_error +msgid "If checked, some messages have a delivery error." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__selected +msgid "Include" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__internal_notes +msgid "Internal Notes" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Internal notes" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_is_follower +msgid "Is Follower" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__is_repair_main_order +msgid "Is Repair Main Order" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__write_uid +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__write_uid +msgid "Last Updated by" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_init__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__write_date +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__write_date +msgid "Last Updated on" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__main_repair_sale_id +msgid "Main Repair Sale Order" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_has_error +msgid "Message Delivery error" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_ids +msgid "Messages" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__my_activity_date_deadline +msgid "My Activity Deadline" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__draft +msgid "New" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_date_deadline +msgid "Next Activity Deadline" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_summary +msgid "Next Activity Summary" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_type_id +msgid "Next Activity Type" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "No sale orders found to merge." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_needaction_counter +msgid "Number of Actions" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__message_has_error_counter +msgid "Number of errors" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_needaction_counter +msgid "Number of messages requiring action" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__message_has_error_counter +msgid "Number of messages with delivery error" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "" +"Only the supervisor and the responsible can request approval for this " +"repair." +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Only the supervisor can approve this repair." +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Only the supervisor can reject this repair." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__partner_id +msgid "Partner" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/wizards/wizard_repair_group_quotation.py:0 +msgid "Please select at least one quotation to merge." +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Print" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__product_id +msgid "Product" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_stock_move_line +msgid "Product Moves (Stock Move Line)" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__sale_order_id +msgid "Quotation" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__quotation_requested +msgid "Quotation Requested" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Quotation requested for this repair order." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__line_ids +msgid "Quotations" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_group.py:0 +msgid "Quotations where requested." +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Reject" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_repair_order__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_sale_order__repair_group_id +#: model:ir.model.fields,field_description:repair_group.field_stock_picking__repair_group_id +#: model_terms:ir.ui.view,arch_db:repair_group.view_picking_form_extended +msgid "Repair Group" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_init +msgid "Repair Group Init" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__name +msgid "Repair Group Reference" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_order +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__repair_order_id +#: model:ir.model.fields,field_description:repair_group.field_stock_move_line__repair_order_id +msgid "Repair Order" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__repair_order_ids +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Repair Orders" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Repair group" +msgstr "" + +#. module: repair_group +#: model:ir.actions.act_window,name:repair_group.action_repair_group_tree +#: model:ir.ui.menu,name:repair_group.repair_group_menu +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form_filter +msgid "Repair groups" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "Repair rejected by supervisor. Please review it." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__done +msgid "Repaired" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__repairs_created +msgid "Repairs Created" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Request Approval" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Request Quotations" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__activity_user_id +msgid "Responsible User" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_form +msgid "Return to draft" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__sale_order_ids +msgid "Sale Orders" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_sale_order +msgid "Sales Order" +msgstr "" + +#. module: repair_group +#: model:ir.actions.act_window,name:repair_group.action_repair_group_quotation_wizard +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_group_quotation_wizard_form +msgid "Select Quotations to Merge" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_sale_order__source_sale_order_ids +msgid "Source Sale Orders" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__state +#: model:ir.model.fields,field_description:repair_group.field_repair_order__state +msgid "Status" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_state +msgid "" +"Status based on activities\n" +"Overdue: Due date is already passed\n" +"Today: Activity date is today\n" +"Planned: Future activities." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__supervisor_id +msgid "Supervisor" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_order_form_inherit_repair_group +msgid "" +"This quotation is part of a Repair Group.\n" +" Please use the Complete Sale Order of the group to manage it.\n" +"
\n" +" Group:" +msgstr "" + +#. module: repair_group +#. odoo-python +#: code:addons/repair_group/models/repair_order.py:0 +msgid "This repair order is not part of a group." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_order__state__to_approve +msgid "To Approve" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__amount_total +msgid "Total" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_stock_picking +msgid "Transfer" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__activity_exception_decoration +msgid "Type of the exception activity on record." +msgstr "" + +#. module: repair_group +#: model:ir.model.fields.selection,name:repair_group.selection__repair_group__state__under_repair +msgid "Under Repair" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__user_id +msgid "User" +msgstr "" + +#. module: repair_group +#: model_terms:ir.ui.view,arch_db:repair_group.view_repair_order_form_inherit_group_supervisor +msgid "Validate" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group__website_message_ids +msgid "Website Messages" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,help:repair_group.field_repair_group__website_message_ids +msgid "Website communication history" +msgstr "" + +#. module: repair_group +#: model:ir.model.fields,field_description:repair_group.field_repair_group_quotation_wizard_line__wizard_id +msgid "Wizard" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_quotation_wizard_line +msgid "Wizard line for quotation selection" +msgstr "" + +#. module: repair_group +#: model:ir.model,name:repair_group.model_repair_group_quotation_wizard +msgid "Wizard to select quotations to merge" +msgstr "" diff --git a/repair_group/models/__init__.py b/repair_group/models/__init__.py new file mode 100644 index 00000000..51020430 --- /dev/null +++ b/repair_group/models/__init__.py @@ -0,0 +1,5 @@ +from . import repair_group +from . import repair_order +from . import stock_picking +from . import res_config_settings +from . import sale_order diff --git a/repair_group/models/repair_group.py b/repair_group/models/repair_group.py new file mode 100644 index 00000000..62bc8cd8 --- /dev/null +++ b/repair_group/models/repair_group.py @@ -0,0 +1,204 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from odoo import _, api, fields, models +from odoo.exceptions import UserError + + +class RepairGroup(models.Model): + _name = "repair.group" + _description = "Repair Group" + _inherit = ["mail.thread", "mail.activity.mixin"] + + name = fields.Char( + "Repair Group Reference", + default="New", + copy=False, + required=True, + readonly=True, + tracking=True, + ) + partner_id = fields.Many2one( + "res.partner", + string="Partner", + required=True, + domain=[("type", "!=", "private")], + tracking=True, + ) + carrier_id = fields.Many2one( + "res.partner", + string="Carrier", + domain=[("type", "!=", "private")], + tracking=True, + ) + state = fields.Selection( + [ + ("draft", "New"), + ("repairs_created", "Repairs Created"), + ("quotation_requested", "Quotation Requested"), + ("under_repair", "Under Repair"), + ("done", "Repaired"), + ("cancel", "Cancelled"), + ], + string="Status", + copy=False, + default="draft", + readonly=True, + tracking=True, + index=True, + ) + user_id = fields.Many2one( + "res.users", + required=True, + default=lambda self: self.env.user, + tracking=True, + ) + supervisor_id = fields.Many2one( + "res.users", + required=True, + default=lambda self: self.env.user, + tracking=True, + ) + scheduled_datetime = fields.Datetime(string="Estimated date", tracking=True) + company_id = fields.Many2one( + "res.company", + required=True, + default=lambda self: self.env.company, + tracking=True, + ) + internal_notes = fields.Html(sanitize=True) + picking_ids = fields.One2many( + "stock.picking", + "repair_group_id", + string="Delivery note", + ) + repair_order_ids = fields.One2many( + "repair.order", + "repair_group_id", + string="Repair Orders", + ) + sale_order_ids = fields.Many2many( + "sale.order", string="Sale Orders", compute="_compute_sale_order_ids" + ) + complete_sale_order_id = fields.Many2one("sale.order", string="General Sale Order") + + @api.model_create_multi + def create(self, vals_list): + for vals in vals_list: + vals.setdefault( + "name", + self.env["ir.sequence"].sudo().next_by_code("repair.group") or "New", + ) + res = super().create(vals_list) + return res + + def action_create_all_repairs(self): + Repair = self.env["repair.order"] + + for group in self: + for picking in group.picking_ids: + for move_line in picking.move_line_ids: + vals = { + "partner_id": group.partner_id.id, + "product_id": move_line.product_id.id, + "repair_group_id": group.id, + } + repair = Repair.create(vals) + move_line.write({"repair_order_id": repair.id}) + + return self.write({"state": "repairs_created"}) + + def action_return_to_draft(self): + Repair = self.env["repair.order"] + for group in self: + for picking in group.picking_ids: + for move_line in picking.move_line_ids: + if move_line.repair_order_id: + repair = Repair.browse(move_line.repair_order_id.id) + if repair.exists(): + repair.action_repair_cancel() + move_line.write({"repair_order_id": False}) + + return self.write({"state": "draft"}) + + def action_request_quotations(self): + for group in self: + message = _("Quotations where requested.") + group.message_post(body=message, subtype_xmlid="mail.mt_note") + + for repair_order in group.repair_order_ids: + message = _("Quotation requested for this repair order.") + repair_order.message_post(body=message, subtype_xmlid="mail.mt_note") + + return self.write({"state": "quotation_requested"}) + + @api.depends("repair_order_ids.sale_order_id") + def _compute_sale_order_ids(self): + for group in self: + sale_orders = self.env["sale.order"] + for repair_order in group.repair_order_ids: + if repair_order.sale_order_id: + sale_orders |= repair_order.sale_order_id + group.sale_order_ids = sale_orders + + def action_create_complete_sale_order(self): + self.ensure_one() + if not self.sale_order_ids: + raise UserError(_("No sale orders found to merge.")) + + Wizard = self.env["repair.group.quotation.wizard"] + wizard = Wizard.create({"repair_group_id": self.id}) + + WizardLine = self.env["repair.group.quotation.wizard.line"] + for repair_order in self.repair_order_ids: + if repair_order.sale_order_id: + WizardLine.create( + { + "wizard_id": wizard.id, + "sale_order_id": repair_order.sale_order_id.id, + "repair_order_id": repair_order.id, + } + ) + + return { + "type": "ir.actions.act_window", + "res_model": "repair.group.quotation.wizard", + "res_id": wizard.id, + "view_mode": "form", + "target": "new", + } + + def action_confirm_complete_sale_order(self): + for group in self: + group.complete_sale_order_id.action_confirm() + return self.write({"state": "under_repair"}) + + def action_cancel_complete_sale_order(self): + for group in self: + group.complete_sale_order_id.action_cancel() + return self.write({"state": "cancel"}) + + def action_cancel_repair_group(self): + return self.write({"state": "cancel"}) + + def action_end_repair_group(self): + Repair = self.env["repair.order"] + for group in self: + not_finished = [] + for picking in group.picking_ids: + for move_line in picking.move_line_ids: + if move_line.repair_order_id: + repair = Repair.browse(move_line.repair_order_id.id) + if repair.exists() and repair.state != "done": + not_finished.append( + repair.display_name or repair.name or str(repair.id) + ) + if not_finished: + raise UserError( + _( + "Cannot finish the group. The following repairs are not " + "finished: {}" + ).format(", ".join(not_finished)) + ) + + return self.write({"state": "done"}) diff --git a/repair_group/models/repair_order.py b/repair_group/models/repair_order.py new file mode 100644 index 00000000..044e7568 --- /dev/null +++ b/repair_group/models/repair_order.py @@ -0,0 +1,64 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from odoo import _, fields, models +from odoo.exceptions import UserError + + +class RepairOrder(models.Model): + _inherit = "repair.order" + + repair_group_id = fields.Many2one( + "repair.group", + string="Repair Group", + ondelete="set null", + copy=False, + ) + state = fields.Selection( + selection_add=[ + ("to_approve", "To Approve"), + ], + ondelete={"to_approve": "set default"}, + ) + + def action_request_approval(self): + self.ensure_one() + if not self.repair_group_id: + raise UserError(_("This repair order is not part of a group.")) + + if ( + self.env.user != self.repair_group_id.supervisor_id + and self.env.user != self.user_id + ): + raise UserError( + _( + "Only the supervisor and the responsible can request approval " + "for this repair." + ) + ) + return self.write({"state": "to_approve"}) + + def action_supervisor_approve(self): + self.ensure_one() + if self.env.user != self.repair_group_id.supervisor_id: + raise UserError(_("Only the supervisor can approve this repair.")) + + # We temporarily set to draft to allow standard validation if it has constraints + self.write({"state": "draft"}) + return self.action_validate() + + def action_supervisor_reject(self): + self.ensure_one() + if self.env.user != self.repair_group_id.supervisor_id: + raise UserError(_("Only the supervisor can reject this repair.")) + + self.message_post( + body=_("Repair rejected by supervisor. Please review it."), + message_type="comment", + subtype_xmlid="mail.mt_note", + partner_ids=[self.user_id.partner_id.id] if self.user_id else [], + ) + return self.write({"state": "draft"}) + + def action_print_repair_order(self): + return self.env.ref("repair.action_report_repair_order").report_action(self) diff --git a/repair_group/models/res_config_settings.py b/repair_group/models/res_config_settings.py new file mode 100644 index 00000000..a3ca271e --- /dev/null +++ b/repair_group/models/res_config_settings.py @@ -0,0 +1,38 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from odoo import SUPERUSER_ID, api, fields, models + + +class ResConfigSettings(models.TransientModel): + _inherit = "res.config.settings" + + group_stock_tracking_owner = fields.Boolean( + "Consignment", + implied_group="stock.group_tracking_owner", + default=True, + ) + + +def post_init_hook(env): + env["repair.group.init"]._init_settings() + + +class RepairGroupInit(models.TransientModel): + _name = "repair.group.init" + _description = "Repair Group Init" + + @api.model + def _init_settings(self): + # with_user(SUPERUSER_ID) is required here because res.config.settings + # restricts access to administrators. This method is only called from + # the post_init_hook during module installation, so the elevated + # privilege is intentional and limited in scope. + settings = ( + self.env["res.config.settings"] + .with_user(SUPERUSER_ID) + .with_context(active_test=False, install_mode=True) + .create({"group_stock_tracking_owner": True}) + ) + settings.execute() + return True diff --git a/repair_group/models/sale_order.py b/repair_group/models/sale_order.py new file mode 100644 index 00000000..b6dd2045 --- /dev/null +++ b/repair_group/models/sale_order.py @@ -0,0 +1,101 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from odoo import api, fields, models + + +class SaleOrder(models.Model): + _inherit = "sale.order" + + repair_group_id = fields.Many2one( + "repair.group", + string="Repair Group", + compute="_compute_repair_group_id", + store=False, + ) + main_repair_sale_id = fields.Many2one( + "sale.order", + string="Main Repair Sale Order", + compute="_compute_repair_info", + ) + source_sale_order_ids = fields.Many2many( + "sale.order", + string="Source Sale Orders", + compute="_compute_source_sale_order_ids", + ) + is_repair_main_order = fields.Boolean(compute="_compute_is_repair_main_order") + + @api.depends() + def _compute_repair_group_id(self): + for order in self: + repair = self.env["repair.order"].search( + [("sale_order_id", "=", order.id), ("repair_group_id", "!=", False)], + limit=1, + ) + if repair: + order.repair_group_id = repair.repair_group_id + else: + group = self.env["repair.group"].search( + [("complete_sale_order_id", "=", order.id)], limit=1 + ) + order.repair_group_id = group + + @api.depends("repair_group_id") + def _compute_is_repair_main_order(self): + for order in self: + order.is_repair_main_order = ( + order.repair_group_id + and order.repair_group_id.complete_sale_order_id == order + ) + + @api.depends("repair_group_id", "is_repair_main_order") + def _compute_source_sale_order_ids(self): + for order in self: + if order.is_repair_main_order: + order.source_sale_order_ids = order.repair_group_id.sale_order_ids + else: + order.source_sale_order_ids = False + + @api.depends("repair_group_id") + def _compute_repair_info(self): + for order in self: + order.main_repair_sale_id = order.repair_group_id.complete_sale_order_id + + def action_confirm(self): + res = super().action_confirm() + for order in self: + groups = self.env["repair.group"].search( + [("complete_sale_order_id", "=", order.id)] + ) + for group in groups: + for repair_order in group.repair_order_ids: + if ( + repair_order.sale_order_id + and repair_order.sale_order_id.state in ["draft", "sent"] + ): + repair_order.sale_order_id.action_confirm() + + if repair_order.state == "draft": + if hasattr(repair_order, "_action_repair_confirm"): + repair_order._action_repair_confirm() + elif hasattr(repair_order, "action_validate"): + repair_order.action_validate() + + if group.state != "under_repair": + group.write({"state": "under_repair"}) + return res + + def _action_cancel(self): + res = super()._action_cancel() + for order in self: + groups = self.env["repair.group"].search( + [("complete_sale_order_id", "=", order.id)] + ) + for group in groups: + for repair_order in group.repair_order_ids: + if ( + repair_order.sale_order_id + and repair_order.sale_order_id.state != "cancel" + ): + repair_order.sale_order_id._action_cancel() + return res diff --git a/repair_group/models/stock_picking.py b/repair_group/models/stock_picking.py new file mode 100644 index 00000000..a97244a1 --- /dev/null +++ b/repair_group/models/stock_picking.py @@ -0,0 +1,51 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from odoo import fields, models + + +class StockPicking(models.Model): + _inherit = "stock.picking" + + repair_group_id = fields.Many2one( + "repair.group", + string="Repair Group", + ondelete="set null", + ) + + def create_repair_group(self): + self.ensure_one() + repair_group = self.env["repair.group"].create( + { + "partner_id": self.partner_id.id, + "picking_ids": [(4, self.id)], + } + ) + return { + "name": "Repair Group", + "type": "ir.actions.act_window", + "res_model": "repair.group", + "res_id": repair_group.id, + "view_mode": "form", + } + + def action_view_repair_group(self): + self.ensure_one() + return { + "name": "Repair Group", + "type": "ir.actions.act_window", + "res_model": "repair.group", + "res_id": self.repair_group_id.id, + "view_mode": "form", + } + + +class StockMoveLine(models.Model): + _inherit = "stock.move.line" + + repair_order_id = fields.Many2one( + "repair.order", + string="Repair Order", + copy=False, + readonly=True, + ) diff --git a/repair_group/pyproject.toml b/repair_group/pyproject.toml new file mode 100644 index 00000000..4231d0cc --- /dev/null +++ b/repair_group/pyproject.toml @@ -0,0 +1,3 @@ +[build-system] +requires = ["whool"] +build-backend = "whool.buildapi" diff --git a/repair_group/readme/CONFIGURE.md b/repair_group/readme/CONFIGURE.md new file mode 100644 index 00000000..e94e6acd --- /dev/null +++ b/repair_group/readme/CONFIGURE.md @@ -0,0 +1,14 @@ +## Consignment (Stock Ownership Tracking) + +This module requires the **Consignment** feature (*Inventory → Configuration → +Settings → Traceability → Consignment*) to be enabled so that stock pickings +track the owner of the goods. + +The feature is **enabled automatically** when this module is installed via a +post-install hook. No manual configuration is required. + +## Supervisors + +Each repair group has a **Supervisor** field. Assign the user responsible for +approving repair orders in the group. By default, the user who creates the group +is set as supervisor. diff --git a/repair_group/readme/CONTRIBUTORS.md b/repair_group/readme/CONTRIBUTORS.md new file mode 100644 index 00000000..b2ddc38b --- /dev/null +++ b/repair_group/readme/CONTRIBUTORS.md @@ -0,0 +1 @@ +- Abel Suárez > diff --git a/repair_group/readme/DESCRIPTION.md b/repair_group/readme/DESCRIPTION.md new file mode 100644 index 00000000..ae0dea32 --- /dev/null +++ b/repair_group/readme/DESCRIPTION.md @@ -0,0 +1,20 @@ +This module allows grouping multiple repair orders by selecting completed stock +pickings (delivery notes) from a specific customer. + +**Key features:** + +- Create a **Repair Group** from one or more done stock pickings belonging to the + same customer (consignment/owner-tracked pickings). +- Automatically generate individual repair orders for each product in the + selected pickings. +- Supervisor approval workflow: technicians request approval and supervisors + validate or reject each repair order. +- Merge individual repair quotations (sale orders) into a single consolidated + sale order per group using a selection wizard. +- Full traceability: each repair order is linked to its origin picking move line + and to the repair group. +- State tracking across the full lifecycle: *Draft → Repairs Created → Quotation + Requested → Under Repair → Repaired* (or *Cancelled*). +- Mail thread and activity tracking on repair groups. +- Smart button on stock pickings to navigate directly to the associated repair + group. diff --git a/repair_group/readme/USAGE.md b/repair_group/readme/USAGE.md new file mode 100644 index 00000000..d77175b4 --- /dev/null +++ b/repair_group/readme/USAGE.md @@ -0,0 +1,35 @@ +## Creating a Repair Group from a Stock Picking + +1. Open a **done** stock picking whose products are owned by a customer + (Consignment must be enabled — this module enables it automatically on + installation). +2. Click **Create Repair Group** in the picking form header. +3. The system creates a new repair group pre-filled with the customer and the + picking. You are redirected to the repair group form. + +## Repair Group Workflow + +1. **Draft**: Add additional pickings if needed. Click **Create Repairs** to + generate one repair order per move line. +2. **Repairs Created**: Review the generated repair orders in the *Repair Orders* + tab. Click **Request Quotations** to move forward, or **Return to Draft** to + undo. +3. **Quotation Requested**: Technicians fill in repair details and generate + quotations from each repair order. Once all quotations exist, click **Create + Complete Sale Order** to open the wizard. +4. **Wizard – Select Quotations**: Choose which quotations to merge and click + **Create Complete Sale Order**. The system consolidates the selected lines + into a single sale order. +5. **Quotation Requested (with main order)**: Review the consolidated order. + Click **Confirm Complete Sale Order** to confirm it and move to *Under Repair*, + or **Cancel Complete Sale Order** to discard it. +6. **Under Repair**: Repairs are being performed. Click **End Repair Group** + once all repair orders are in *Done* state. +7. **Repaired**: The group is complete. + +## Supervisor Approval + +On each repair order that belongs to a group, the responsible user can click +**Request Approval** to submit it for supervisor validation. The supervisor +(configured on the repair group) can then **Validate** or **Reject** the repair +directly from the repair order form. diff --git a/repair_group/security/ir.model.access.csv b/repair_group/security/ir.model.access.csv new file mode 100644 index 00000000..61b14bb7 --- /dev/null +++ b/repair_group/security/ir.model.access.csv @@ -0,0 +1,4 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_repair_group_user,repair.group,model_repair_group,base.group_user,1,1,1,1 +access_repair_group_quotation_wizard,repair.group.quotation.wizard,model_repair_group_quotation_wizard,base.group_user,1,1,1,1 +access_repair_group_quotation_wizard_line,repair.group.quotation.wizard.line,model_repair_group_quotation_wizard_line,base.group_user,1,1,1,1 diff --git a/repair_group/static/description/icon.png b/repair_group/static/description/icon.png new file mode 100644 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/repair_group/static/description/index.html b/repair_group/static/description/index.html new file mode 100644 index 00000000..507a328b --- /dev/null +++ b/repair_group/static/description/index.html @@ -0,0 +1,521 @@ + + + + + +Repair Group + + + +
+

Repair Group

+ + +

Alpha License: LGPL-3 OCA/repair Translate me on Weblate Try me on Runboat

+

This module allows grouping multiple repair orders by selecting +completed stock pickings (delivery notes) from a specific customer.

+

Key features:

+
    +
  • Create a Repair Group from one or more done stock pickings +belonging to the same customer (consignment/owner-tracked pickings).
  • +
  • Automatically generate individual repair orders for each product in +the selected pickings.
  • +
  • Supervisor approval workflow: technicians request approval and +supervisors validate or reject each repair order.
  • +
  • Merge individual repair quotations (sale orders) into a single +consolidated sale order per group using a selection wizard.
  • +
  • Full traceability: each repair order is linked to its origin picking +move line and to the repair group.
  • +
  • State tracking across the full lifecycle: Draft → Repairs Created → +Quotation Requested → Under Repair → Repaired (or Cancelled).
  • +
  • Mail thread and activity tracking on repair groups.
  • +
  • Smart button on stock pickings to navigate directly to the associated +repair group.
  • +
+
+

Important

+

This is an alpha version, the data model and design can change at any time without warning. +Only for development or testing purpose, do not use in production. +More details on development status

+
+

Table of contents

+ +
+

Configuration

+
+

Consignment (Stock Ownership Tracking)

+

This module requires the Consignment feature (Inventory → +Configuration → Settings → Traceability → Consignment) to be enabled so +that stock pickings track the owner of the goods.

+

The feature is enabled automatically when this module is installed +via a post-install hook. No manual configuration is required.

+
+
+

Supervisors

+

Each repair group has a Supervisor field. Assign the user +responsible for approving repair orders in the group. By default, the +user who creates the group is set as supervisor.

+
+
+
+

Usage

+
+

Creating a Repair Group from a Stock Picking

+
    +
  1. Open a done stock picking whose products are owned by a customer +(Consignment must be enabled — this module enables it automatically +on installation).
  2. +
  3. Click Create Repair Group in the picking form header.
  4. +
  5. The system creates a new repair group pre-filled with the customer +and the picking. You are redirected to the repair group form.
  6. +
+
+
+

Repair Group Workflow

+
    +
  1. Draft: Add additional pickings if needed. Click Create +Repairs to generate one repair order per move line.
  2. +
  3. Repairs Created: Review the generated repair orders in the +Repair Orders tab. Click Request Quotations to move forward, or +Return to Draft to undo.
  4. +
  5. Quotation Requested: Technicians fill in repair details and +generate quotations from each repair order. Once all quotations +exist, click Create Complete Sale Order to open the wizard.
  6. +
  7. Wizard – Select Quotations: Choose which quotations to merge and +click Create Complete Sale Order. The system consolidates the +selected lines into a single sale order.
  8. +
  9. Quotation Requested (with main order): Review the consolidated +order. Click Confirm Complete Sale Order to confirm it and move +to Under Repair, or Cancel Complete Sale Order to discard it.
  10. +
  11. Under Repair: Repairs are being performed. Click End Repair +Group once all repair orders are in Done state.
  12. +
  13. Repaired: The group is complete.
  14. +
+
+
+

Supervisor Approval

+

On each repair order that belongs to a group, the responsible user can +click Request Approval to submit it for supervisor validation. The +supervisor (configured on the repair group) can then Validate or +Reject the repair directly from the repair order form.

+
+
+
+

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

+
    +
  • Grupo Isonor
  • +
+
+
+

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.

+

Current maintainer:

+

abelsrzz

+

This module is part of the OCA/repair project on GitHub.

+

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

+
+
+
+ + diff --git a/repair_group/tests/__init__.py b/repair_group/tests/__init__.py new file mode 100644 index 00000000..6e4720f4 --- /dev/null +++ b/repair_group/tests/__init__.py @@ -0,0 +1,4 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from . import test_repair_group diff --git a/repair_group/tests/test_repair_group.py b/repair_group/tests/test_repair_group.py new file mode 100644 index 00000000..3400cbd7 --- /dev/null +++ b/repair_group/tests/test_repair_group.py @@ -0,0 +1,134 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from odoo.exceptions import UserError +from odoo.tests.common import TransactionCase +from odoo.tests import tagged + + +@tagged("post_install", "-at_install") +class TestRepairGroup(TransactionCase): + @classmethod + def setUpClass(cls): + super().setUpClass() + cls.partner = cls.env["res.partner"].create({"name": "Test Customer"}) + cls.product = cls.env["product.product"].create( + {"name": "Test Product", "type": "product"} + ) + cls.location = cls.env.ref("stock.stock_location_stock") + cls.location_dest = cls.env.ref("stock.stock_location_customers") + + def _create_done_picking(self): + """Helper: create and validate a stock picking.""" + picking_type = self.env.ref("stock.picking_type_out") + picking = self.env["stock.picking"].create( + { + "partner_id": self.partner.id, + "picking_type_id": picking_type.id, + "location_id": self.location.id, + "location_dest_id": self.location_dest.id, + "owner_id": self.partner.id, + "move_ids": [ + ( + 0, + 0, + { + "name": self.product.name, + "product_id": self.product.id, + "product_uom_qty": 1.0, + "product_uom": self.product.uom_id.id, + "location_id": self.location.id, + "location_dest_id": self.location_dest.id, + }, + ) + ], + } + ) + picking.action_confirm() + picking.action_assign() + for ml in picking.move_line_ids: + ml.qty_done = ml.reserved_qty + picking.button_validate() + return picking + + def test_repair_group_creation(self): + """Test that a repair group is created with a sequential name.""" + group = self.env["repair.group"].create({"partner_id": self.partner.id}) + self.assertTrue(group.name.startswith("RG/"), "Name should start with RG/") + self.assertEqual(group.state, "draft") + + def test_action_create_all_repairs(self): + """Test that repairs are created from picking move lines.""" + picking = self._create_done_picking() + group = self.env["repair.group"].create( + { + "partner_id": self.partner.id, + "picking_ids": [(4, picking.id)], + } + ) + group.action_create_all_repairs() + self.assertEqual(group.state, "repairs_created") + self.assertTrue(group.repair_order_ids, "Repair orders should have been created") + self.assertEqual(len(group.repair_order_ids), len(picking.move_line_ids)) + + def test_action_return_to_draft(self): + """Test that repairs are cancelled when returning to draft.""" + picking = self._create_done_picking() + group = self.env["repair.group"].create( + { + "partner_id": self.partner.id, + "picking_ids": [(4, picking.id)], + } + ) + group.action_create_all_repairs() + self.assertEqual(group.state, "repairs_created") + group.action_return_to_draft() + self.assertEqual(group.state, "draft") + + def test_action_request_quotations(self): + """Test state transition to quotation_requested.""" + picking = self._create_done_picking() + group = self.env["repair.group"].create( + { + "partner_id": self.partner.id, + "picking_ids": [(4, picking.id)], + } + ) + group.action_create_all_repairs() + group.action_request_quotations() + self.assertEqual(group.state, "quotation_requested") + + def test_action_end_repair_group_not_done_raises(self): + """Test that ending a group with unfinished repairs raises UserError.""" + picking = self._create_done_picking() + group = self.env["repair.group"].create( + { + "partner_id": self.partner.id, + "picking_ids": [(4, picking.id)], + } + ) + group.action_create_all_repairs() + group.write({"state": "under_repair"}) + with self.assertRaises(UserError): + group.action_end_repair_group() + + def test_stock_picking_create_repair_group(self): + """Test creating a repair group directly from a stock picking.""" + picking = self._create_done_picking() + action = picking.create_repair_group() + self.assertEqual(action["res_model"], "repair.group") + group = self.env["repair.group"].browse(action["res_id"]) + self.assertTrue(group.exists()) + self.assertIn(picking, group.picking_ids) + self.assertEqual(group.partner_id, self.partner) + + def test_repair_order_request_approval_no_group_raises(self): + """Test that requesting approval without a group raises UserError.""" + repair = self.env["repair.order"].create( + { + "product_id": self.product.id, + "partner_id": self.partner.id, + } + ) + with self.assertRaises(UserError): + repair.action_request_approval() diff --git a/repair_group/views/repair_groups.xml b/repair_group/views/repair_groups.xml new file mode 100644 index 00000000..19a3d27f --- /dev/null +++ b/repair_group/views/repair_groups.xml @@ -0,0 +1,168 @@ + + + + + repair.group.tree + repair.group + + + + + + + + + + + + repair.group.search + repair.group + + + + + + + + + + repair.group.form + repair.group + +
+
+
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
diff --git a/repair_group/views/wizard_repair_group_quotation.xml b/repair_group/views/wizard_repair_group_quotation.xml new file mode 100644 index 00000000..ecbfc200 --- /dev/null +++ b/repair_group/views/wizard_repair_group_quotation.xml @@ -0,0 +1,44 @@ + + + + + repair.group.quotation.wizard.form + repair.group.quotation.wizard + +
+ + + + + + + + + + + + + + + +
+
+
+
+
+ + + Select Quotations to Merge + repair.group.quotation.wizard + form + new + +
diff --git a/repair_group/wizards/__init__.py b/repair_group/wizards/__init__.py new file mode 100644 index 00000000..63719519 --- /dev/null +++ b/repair_group/wizards/__init__.py @@ -0,0 +1 @@ +from . import wizard_repair_group_quotation diff --git a/repair_group/wizards/wizard_repair_group_quotation.py b/repair_group/wizards/wizard_repair_group_quotation.py new file mode 100644 index 00000000..323f36ea --- /dev/null +++ b/repair_group/wizards/wizard_repair_group_quotation.py @@ -0,0 +1,105 @@ +# Copyright 2026 Grupo Isonor - Abel Suárez +# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). + +from odoo import _, fields, models +from odoo.exceptions import UserError + + +class RepairGroupQuotationWizard(models.TransientModel): + _name = "repair.group.quotation.wizard" + _description = "Wizard to select quotations to merge" + + repair_group_id = fields.Many2one( + "repair.group", + string="Repair Group", + required=True, + readonly=True, + ) + line_ids = fields.One2many( + "repair.group.quotation.wizard.line", + "wizard_id", + string="Quotations", + ) + + def action_create_complete_sale_order(self): + self.ensure_one() + selected_lines = self.line_ids.filtered("selected") + if not selected_lines: + raise UserError(_("Please select at least one quotation to merge.")) + + SaleOrder = self.env["sale.order"] + SaleOrderLine = self.env["sale.order.line"] + group = self.repair_group_id + + first_order = selected_lines[0].sale_order_id + complete_order_vals = { + "partner_id": group.partner_id.id, + "user_id": group.user_id.id, + "company_id": group.company_id.id, + "date_order": fields.Datetime.now(), + "note": _("Combined order from Repair Group: {}").format(group.name), + } + + if first_order.payment_term_id: + complete_order_vals["payment_term_id"] = first_order.payment_term_id.id + if first_order.pricelist_id: + complete_order_vals["pricelist_id"] = first_order.pricelist_id.id + + complete_order = SaleOrder.create(complete_order_vals) + + for line in selected_lines: + for order_line in line.sale_order_id.order_line: + line_vals = { + "order_id": complete_order.id, + "product_id": order_line.product_id.id, + "name": order_line.name, + "product_uom_qty": order_line.product_uom_qty, + "product_uom": order_line.product_uom.id, + "price_unit": order_line.price_unit, + "tax_id": [(6, 0, order_line.tax_id.ids)], + "discount": order_line.discount, + } + if ( + hasattr(order_line, "analytic_distribution") + and order_line.analytic_distribution + ): + line_vals["analytic_distribution"] = ( + order_line.analytic_distribution + ) + SaleOrderLine.create(line_vals) + + group.complete_sale_order_id = complete_order + return {"type": "ir.actions.act_window_close"} + + +class RepairGroupQuotationWizardLine(models.TransientModel): + _name = "repair.group.quotation.wizard.line" + _description = "Wizard line for quotation selection" + + wizard_id = fields.Many2one( + "repair.group.quotation.wizard", + string="Wizard", + required=True, + ondelete="cascade", + ) + sale_order_id = fields.Many2one("sale.order", string="Quotation", readonly=True) + repair_order_id = fields.Many2one( + "repair.order", string="Repair Order", readonly=True + ) + product_id = fields.Many2one( + "product.product", + string="Product", + related="repair_order_id.product_id", + readonly=True, + ) + amount_total = fields.Monetary( + string="Total", + related="sale_order_id.amount_total", + readonly=True, + ) + currency_id = fields.Many2one( + "res.currency", + related="sale_order_id.currency_id", + readonly=True, + ) + selected = fields.Boolean(string="Include", default=True) From fea0aff5b7cb7d94cfaedb02c5b36dbd5af25397 Mon Sep 17 00:00:00 2001 From: abelsrzz Date: Mon, 16 Mar 2026 21:30:16 +0100 Subject: [PATCH 2/4] Fix pre-commit and test issues --- repair_group/i18n/repair_group.pot | 0 repair_group/models/repair_order.py | 2 +- repair_group/security/ir.model.access.csv | 1 + repair_group/tests/test_repair_group.py | 8 +- repair_group/views/repair_groups.xml | 180 ++++++++++----------- repair_group/views/repair_groups_menu.xml | 2 +- repair_group/views/repair_order_views.xml | 16 +- repair_group/views/sale_order_views.xml | 13 +- repair_group/views/stock_picking_views.xml | 3 +- 9 files changed, 112 insertions(+), 113 deletions(-) mode change 100755 => 100644 repair_group/i18n/repair_group.pot diff --git a/repair_group/i18n/repair_group.pot b/repair_group/i18n/repair_group.pot old mode 100755 new mode 100644 diff --git a/repair_group/models/repair_order.py b/repair_group/models/repair_order.py index 044e7568..be56f28e 100644 --- a/repair_group/models/repair_order.py +++ b/repair_group/models/repair_order.py @@ -10,7 +10,7 @@ class RepairOrder(models.Model): repair_group_id = fields.Many2one( "repair.group", - string="Repair Group", + string="Repair Group (Picking)", ondelete="set null", copy=False, ) diff --git a/repair_group/security/ir.model.access.csv b/repair_group/security/ir.model.access.csv index 61b14bb7..2636a307 100644 --- a/repair_group/security/ir.model.access.csv +++ b/repair_group/security/ir.model.access.csv @@ -2,3 +2,4 @@ id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink access_repair_group_user,repair.group,model_repair_group,base.group_user,1,1,1,1 access_repair_group_quotation_wizard,repair.group.quotation.wizard,model_repair_group_quotation_wizard,base.group_user,1,1,1,1 access_repair_group_quotation_wizard_line,repair.group.quotation.wizard.line,model_repair_group_quotation_wizard_line,base.group_user,1,1,1,1 +access_repair_group_init,repair.group.init,model_repair_group_init,base.group_user,1,0,0,0 diff --git a/repair_group/tests/test_repair_group.py b/repair_group/tests/test_repair_group.py index 3400cbd7..223f6742 100644 --- a/repair_group/tests/test_repair_group.py +++ b/repair_group/tests/test_repair_group.py @@ -2,8 +2,8 @@ # License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). from odoo.exceptions import UserError -from odoo.tests.common import TransactionCase from odoo.tests import tagged +from odoo.tests.common import TransactionCase @tagged("post_install", "-at_install") @@ -13,7 +13,7 @@ def setUpClass(cls): super().setUpClass() cls.partner = cls.env["res.partner"].create({"name": "Test Customer"}) cls.product = cls.env["product.product"].create( - {"name": "Test Product", "type": "product"} + {"name": "Test Product", "type": "consu"} ) cls.location = cls.env.ref("stock.stock_location_stock") cls.location_dest = cls.env.ref("stock.stock_location_customers") @@ -68,7 +68,9 @@ def test_action_create_all_repairs(self): ) group.action_create_all_repairs() self.assertEqual(group.state, "repairs_created") - self.assertTrue(group.repair_order_ids, "Repair orders should have been created") + self.assertTrue( + group.repair_order_ids, "Repair orders should have been created" + ) self.assertEqual(len(group.repair_order_ids), len(picking.move_line_ids)) def test_action_return_to_draft(self): diff --git a/repair_group/views/repair_groups.xml b/repair_group/views/repair_groups.xml index 19a3d27f..15fc2185 100644 --- a/repair_group/views/repair_groups.xml +++ b/repair_group/views/repair_groups.xml @@ -3,166 +3,164 @@ License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0). --> - repair.group.tree - repair.group - - - - - - - - - + repair.group.tree + repair.group + + + + + + + + + - - repair.group.search - repair.group - - - - - - - + + repair.group.search + repair.group + + + + + + + - - repair.group.form - repair.group - -
-
-
- -
-
+ + +
+
+ + + + - - - - - - - - + + + - - - - + + + - - - - - - - - + + + + + + + - - - - - - - + />
From 558275ad5ef6be10ccd0af80ec48849bb910e987 Mon Sep 17 00:00:00 2001 From: abelsrzz Date: Mon, 16 Mar 2026 21:33:50 +0100 Subject: [PATCH 3/4] Fix pre-commit and test issues --- repair_group/views/sale_order_views.xml | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/repair_group/views/sale_order_views.xml b/repair_group/views/sale_order_views.xml index 9855453f..77f3cc63 100644 --- a/repair_group/views/sale_order_views.xml +++ b/repair_group/views/sale_order_views.xml @@ -20,15 +20,15 @@ Please use the Complete Sale Order of the group to manage it.
Group: + name="repair_group_id" + readonly="1" + options="{'no_open': False}" + />
Main Sale Order: + name="main_repair_sale_id" + readonly="1" + /> From 96ee40e4c91da01d8b8a76d8a578359869c22bcd Mon Sep 17 00:00:00 2001 From: abelsrzz Date: Mon, 16 Mar 2026 21:40:31 +0100 Subject: [PATCH 4/4] [FIX] repair_group: fix stock.move.line field --- repair_group/tests/test_repair_group.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/repair_group/tests/test_repair_group.py b/repair_group/tests/test_repair_group.py index 223f6742..4672d18a 100644 --- a/repair_group/tests/test_repair_group.py +++ b/repair_group/tests/test_repair_group.py @@ -47,7 +47,7 @@ def _create_done_picking(self): picking.action_confirm() picking.action_assign() for ml in picking.move_line_ids: - ml.qty_done = ml.reserved_qty + ml.quantity = ml.reserved_uom_qty picking.button_validate() return picking