Initial commit: OCA Report packages (45 packages)

This commit is contained in:
Ernad Husremovic 2025-08-29 15:43:05 +02:00
commit 2f4db400df
2543 changed files with 469120 additions and 0 deletions

View file

@ -0,0 +1,47 @@
# Report Substitute
Odoo addon: report_substitute
## Installation
```bash
pip install odoo-bringout-oca-reporting-engine-report_substitute
```
## Dependencies
This addon depends on:
- base
- mail
## Manifest Information
- **Name**: Report Substitute
- **Version**: 16.0.1.1.1
- **Category**: N/A
- **License**: AGPL-3
- **Installable**: False
## Source
Based on [OCA/reporting-engine](https://github.com/OCA/reporting-engine) branch 16.0, addon `report_substitute`.
## License
This package maintains the original AGPL-3 license from the upstream Odoo project.
## Documentation
- Overview: doc/OVERVIEW.md
- Architecture: doc/ARCHITECTURE.md
- Models: doc/MODELS.md
- Controllers: doc/CONTROLLERS.md
- Wizards: doc/WIZARDS.md
- Reports: doc/REPORTS.md
- Security: doc/SECURITY.md
- Install: doc/INSTALL.md
- Usage: doc/USAGE.md
- Configuration: doc/CONFIGURATION.md
- Dependencies: doc/DEPENDENCIES.md
- Troubleshooting: doc/TROUBLESHOOTING.md
- FAQ: doc/FAQ.md

View file

@ -0,0 +1,32 @@
# Architecture
```mermaid
flowchart TD
U[Users] -->|HTTP| V[Views and QWeb Templates]
V --> C[Controllers]
V --> W[Wizards Transient Models]
C --> M[Models and ORM]
W --> M
M --> R[Reports]
DX[Data XML] --> M
S[Security ACLs and Groups] -. enforces .-> M
subgraph Report_substitute Module - report_substitute
direction LR
M:::layer
W:::layer
C:::layer
V:::layer
R:::layer
S:::layer
DX:::layer
end
classDef layer fill:#eef8ff,stroke:#6ea8fe,stroke-width:1px
```
Notes
- Views include tree/form/kanban templates and report templates.
- Controllers provide website/portal routes when present.
- Wizards are UI flows implemented with `models.TransientModel`.
- Data XML loads data/demo records; Security defines groups and access.

View file

@ -0,0 +1,3 @@
# Configuration
Refer to Odoo settings for report_substitute. Configure related models, access rights, and options as needed.

View file

@ -0,0 +1,3 @@
# Controllers
This module does not define custom HTTP controllers.

View file

@ -0,0 +1,6 @@
# Dependencies
This addon depends on:
- base
- [mail](../../odoo-bringout-oca-ocb-mail)

View file

@ -0,0 +1,4 @@
# FAQ
- Q: Which Odoo version? A: 16.0 (OCA/OCB packaged).
- Q: How to enable? A: Start server with --addon report_substitute or install in UI.

View file

@ -0,0 +1,7 @@
# Install
```bash
pip install odoo-bringout-oca-reporting-engine-report_substitute"
# or
uv pip install odoo-bringout-oca-reporting-engine-report_substitute"
```

View file

@ -0,0 +1,14 @@
# Models
Detected core models and extensions in report_substitute.
```mermaid
classDiagram
class ir_actions_report_substitution_rule
class ir_actions_report
class mail_thread
```
Notes
- Classes show model technical names; fields omitted for brevity.
- Items listed under _inherit are extensions of existing models.

View file

@ -0,0 +1,6 @@
# Overview
Packaged Odoo addon: report_substitute. Provides features documented in upstream Odoo 16 under this addon.
- Source: OCA/OCB 16.0, addon report_substitute
- License: LGPL-3

View file

@ -0,0 +1,3 @@
# Reports
This module does not define custom reports.

View file

@ -0,0 +1,66 @@
# Security
Access control and security definitions in report_substitute.
## Access Control Lists (ACLs)
Model access permissions defined in:
- **[bosnian_translations.json](../bosnian_translations.json)**
- 50 model access rules
- **[bosnian_translations_output.json](../bosnian_translations_output.json)**
- 444 model access rules
- **[CHANGELOG.md](../CHANGELOG.md)**
- 132 model access rules
- **[doc](../doc)**
- **[docker](../docker)**
- **[input](../input)**
- **[nix](../nix)**
- **[odoo.conf](../odoo.conf)**
- 58 model access rules
- **[odoo_packages_bez_l10n.txt](../odoo_packages_bez_l10n.txt)**
- 1947 model access rules
- **[odoo_packages_bringout.txt](../odoo_packages_bringout.txt)**
- 1947 model access rules
- **[odoo_packages.txt](../odoo_packages.txt)**
- 2085 model access rules
- **[output](../output)**
- **[packages](../packages)**
- **[PACKAGES.md](../PACKAGES.md)**
- 298 model access rules
- **[README.md](../README.md)**
- 338 model access rules
- **[scripts](../scripts)**
- **[temp](../temp)**
- **[TRANSLATION_BS_SUMMARY.md](../TRANSLATION_BS_SUMMARY.md)**
- 146 model access rules
## Record Rules
Row-level security rules defined in:
- **[ir_actions_report_substitution_rule.xml](../report_substitute/security/ir_actions_report_substitution_rule.xml)**
## Security Groups & Configuration
Security groups and permissions defined in:
- **[ir_actions_report_substitution_rule.xml](../report_substitute/security/ir_actions_report_substitution_rule.xml)**
```mermaid
graph TB
subgraph "Security Layers"
A[Users] --> B[Groups]
B --> C[Access Control Lists]
C --> D[Models]
B --> E[Record Rules]
E --> F[Individual Records]
end
```
Security files overview:
- **[ir_actions_report_substitution_rule.xml](../report_substitute/security/ir_actions_report_substitution_rule.xml)**
- Security groups, categories, and XML-based rules
Notes
- Access Control Lists define which groups can access which models
- Record Rules provide row-level security (filter records by user/group)
- Security groups organize users and define permission sets
- All security is enforced at the ORM level by Odoo

View file

@ -0,0 +1,5 @@
# Troubleshooting
- Ensure Python and Odoo environment matches repo guidance.
- Check database connectivity and logs if startup fails.
- Validate that dependent addons listed in DEPENDENCIES.md are installed.

View file

@ -0,0 +1,7 @@
# Usage
Start Odoo including this addon (from repo root):
```bash
python3 scripts/nix_odoo_web_server.py --db-name mydb --addon report_substitute
```

View file

@ -0,0 +1,3 @@
# Wizards
This module does not include UI wizards.

View file

@ -0,0 +1,45 @@
[project]
name = "odoo-bringout-oca-reporting-engine-report_substitute"
version = "16.0.0"
description = "Report Substitute -
This module allows to create substitution rules for report actions.
"
authors = [
{ name = "Ernad Husremovic", email = "hernad@bring.out.ba" }
]
dependencies = [
"odoo-bringout-oca-ocb-base>=16.0.0",
"odoo-bringout-oca-ocb-mail>=16.0.0",
"requests>=2.25.1"
]
readme = "README.md"
requires-python = ">= 3.11"
classifiers = [
"Development Status :: 5 - Production/Stable",
"Intended Audience :: Developers",
"License :: OSI Approved :: GNU Lesser General Public License v3 (LGPLv3)",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3.11",
"Programming Language :: Python :: 3.12",
"Topic :: Office/Business",
]
[project.urls]
homepage = "https://github.com/bringout/0"
repository = "https://github.com/bringout/0"
[build-system]
requires = ["hatchling"]
build-backend = "hatchling.build"
[tool.hatch.metadata]
allow-direct-references = true
[tool.hatch.build.targets.wheel]
packages = ["report_substitute"]
[tool.rye]
managed = true
dev-dependencies = [
"pytest>=8.4.1",
]

View file

@ -0,0 +1,115 @@
.. image:: https://odoo-community.org/readme-banner-image
:target: https://odoo-community.org/get-involved?utm_source=readme
:alt: Odoo Community Association
=================
Report Substitute
=================
..
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! This file is generated by oca-gen-addon-readme !!
!! changes will be overwritten. !!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! source digest: sha256:6e713ec7c433154ddfbdc2a954a61a5df2fcdabf03d875d3b3dd5bd0ce4ff980
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png
:target: https://odoo-community.org/page/development-status
:alt: Beta
.. |badge2| image:: https://img.shields.io/badge/license-AGPL--3-blue.png
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Freporting--engine-lightgray.png?logo=github
:target: https://github.com/OCA/reporting-engine/tree/16.0/report_substitute
:alt: OCA/reporting-engine
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
:target: https://translation.odoo-community.org/projects/reporting-engine-16-0/reporting-engine-16-0-report_substitute
: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/reporting-engine&target_branch=16.0
:alt: Try me on Runboat
|badge1| |badge2| |badge3| |badge4| |badge5|
This module allows you to create substitution rules for report actions.
A typical use case is to replace a standard report by alternative reports
when some conditions are met. For instance, it allows to configure alternate
reports for different companies.
**Table of contents**
.. contents::
:local:
Usage
=====
To use this module, you need to:
#. Go to 'Actions' / 'Reports'
#. Select the desired report you want to 'Substitution Rules'
#. In the substitutions page add a new line
#. Select the substitution report action
#. Set a domain to specify when this substitution should happen
When a user calls a report action, the system tries to find the first
substitution in with a domain that matches all records.
Known issues / Roadmap
======================
- The document name result should take the name of the substitution report.
Bug Tracker
===========
Bugs are tracked on `GitHub Issues <https://github.com/OCA/reporting-engine/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 <https://github.com/OCA/reporting-engine/issues/new?body=module:%20report_substitute%0Aversion:%2016.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
Do not contact contributors directly about support or help with technical issues.
Credits
=======
Authors
~~~~~~~
* ACSONE SA/NV
Contributors
~~~~~~~~~~~~
* Bejaoui Souheil <souheil.bejaoui@acsone.eu>
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-sbejaoui| image:: https://github.com/sbejaoui.png?size=40px
:target: https://github.com/sbejaoui
:alt: sbejaoui
Current `maintainer <https://odoo-community.org/page/maintainer-role>`__:
|maintainer-sbejaoui|
This module is part of the `OCA/reporting-engine <https://github.com/OCA/reporting-engine/tree/16.0/report_substitute>`_ project on GitHub.
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

View file

@ -0,0 +1,2 @@
from . import models
from . import wizards

View file

@ -0,0 +1,25 @@
# Copyright 2019 ACSONE SA/NV
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "Report Substitute",
"summary": """
This module allows to create substitution rules for report actions.
""",
"version": "16.0.1.1.1",
"license": "AGPL-3",
"author": "ACSONE SA/NV," "Odoo Community Association (OCA)",
"website": "https://github.com/OCA/reporting-engine",
"depends": ["base", "mail"],
"data": [
"security/ir_actions_report_substitution_rule.xml",
"views/ir_actions_report.xml",
],
"demo": ["demo/action_report.xml"],
"assets": {
"web.assets_backend": [
"report_substitute/static/src/js/action_manager.esm.js",
],
},
"maintainers": ["sbejaoui"],
}

View file

@ -0,0 +1,32 @@
<?xml version="1.0" encoding="utf-8" ?>
<odoo>
<template id="substitution_report">
<div class="page">Substitution Report</div>
</template>
<record id="substitution_report_print" model="ir.actions.report">
<field name="name">Substitution For Technical guide</field>
<field name="model">ir.module.module</field>
<field name="report_type">qweb-pdf</field>
<field name="report_name">report_substitute.substitution_report</field>
<field name="report_file">report_substitute.substitution_report</field>
<field name="binding_type">report</field>
</record>
<record id="substitution_rule_demo_1" model="ir.actions.report.substitution.rule">
<field name="action_report_id" ref="base.ir_module_reference_print" />
<field
name="substitution_action_report_id"
ref="report_substitute.substitution_report_print"
/>
</record>
<template id="substitution_report_2">
<div class="page">Substitution Report 2</div>
</template>
<record id="substitution_report_print_2" model="ir.actions.report">
<field name="name">Substitution 2 For Technical guide</field>
<field name="model">ir.module.module</field>
<field name="report_type">qweb-pdf</field>
<field name="report_name">report_substitute.substitution_report_2</field>
<field name="report_file">report_substitute.substitution_report_2</field>
<field name="binding_type">report</field>
</record>
</odoo>

View file

@ -0,0 +1,123 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * report_substitute
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report_substitution_rule
msgid "Action Report Substitution Rule"
msgstr "Pravilo supstitucije za izvještaj o akciji"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_uid
msgid "Created by"
msgstr "Kreirao"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_date
msgid "Created on"
msgstr "Kreirano"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__display_name
msgid "Display Name"
msgstr "Prikazani naziv"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__domain
msgid "Domain"
msgstr "Domena"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_thread
msgid "Email Thread"
msgstr "Nit e-pošte"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_compose_message
msgid "Email composition wizard"
msgstr "Čarobnjak sastavljanja email-a"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__id
msgid "ID"
msgstr "ID"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule____last_update
msgid "Last Modified on"
msgstr "Zadnje mijenjano"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_uid
msgid "Last Updated by"
msgstr "Zadnji ažurirao"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_date
msgid "Last Updated on"
msgstr "Zadnje ažurirano"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__model
msgid "Model Name"
msgstr "Naziv modela"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__action_report_id
msgid "Report Action"
msgstr "Akcija izvještaja"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__sequence
msgid "Sequence"
msgstr "Sekvenca"
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print_2
msgid "Substitution 2 For Technical guide"
msgstr "Supstitucija 2 za tehnički vodič"
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print
msgid "Substitution For Technical guide"
msgstr "Supstitucija za tehnički vodič"
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report
msgid "Substitution Report"
msgstr "Izvještaj o supstituciji"
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report_2
msgid "Substitution Report 2"
msgstr "Izvještaj o supstituciji 2"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__substitution_action_report_id
msgid "Substitution Report Action"
msgstr "Akcija izvještaja o zamjeni"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report__action_report_substitution_rule_ids
#: model_terms:ir.ui.view,arch_db:report_substitute.ir_actions_report_form_view
msgid "Substitution Rules"
msgstr "Pravila zamjene"
#. module: report_substitute
#. odoo-python
#: code:addons/report_substitute/models/ir_actions_report_substitution_rule.py:0
#, python-format
msgid "Substitution infinite loop detected"
msgstr "Otkrivena je beskonačna petlja supstitucije"

View file

@ -0,0 +1,126 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * report_substitute
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2023-09-07 16:37+0000\n"
"Last-Translator: Ivorra78 <informatica@totmaterial.es>\n"
"Language-Team: none\n"
"Language: es\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: nplurals=2; plural=n != 1;\n"
"X-Generator: Weblate 4.17\n"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report_substitution_rule
msgid "Action Report Substitution Rule"
msgstr "Informe de acción Regla de sustitución"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_uid
msgid "Created by"
msgstr "Creado por"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_date
msgid "Created on"
msgstr "Creado el"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__display_name
msgid "Display Name"
msgstr "Mostrar Nombre"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__domain
msgid "Domain"
msgstr "Dominio"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_thread
msgid "Email Thread"
msgstr "Hilo de correo electrónico"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_compose_message
msgid "Email composition wizard"
msgstr "Asistente de redacción de correo electrónico"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__id
msgid "ID"
msgstr "ID(identificación)"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule____last_update
msgid "Last Modified on"
msgstr "Última Modificación el"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_uid
msgid "Last Updated by"
msgstr "Última actualización por"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_date
msgid "Last Updated on"
msgstr "Ultima Actualización el"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__model
msgid "Model Name"
msgstr "Nombre del modelo"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__action_report_id
msgid "Report Action"
msgstr "Informar Acción"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__sequence
msgid "Sequence"
msgstr "Secuencia"
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print_2
msgid "Substitution 2 For Technical guide"
msgstr "Sustitución 2 Para Guía técnica"
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print
msgid "Substitution For Technical guide"
msgstr "Sustitución Para Guía técnica"
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report
msgid "Substitution Report"
msgstr "Informe de sustitución"
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report_2
msgid "Substitution Report 2"
msgstr "Informe de sustitución 2"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__substitution_action_report_id
msgid "Substitution Report Action"
msgstr "Acción del informe de sustitución"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report__action_report_substitution_rule_ids
#: model_terms:ir.ui.view,arch_db:report_substitute.ir_actions_report_form_view
msgid "Substitution Rules"
msgstr "Reglas de sustitución"
#. module: report_substitute
#. odoo-python
#: code:addons/report_substitute/models/ir_actions_report_substitution_rule.py:0
#, python-format
msgid "Substitution infinite loop detected"
msgstr "Detectado bucle infinito de sustitución"

View file

@ -0,0 +1,126 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * report_substitute
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2024-01-18 09:34+0000\n"
"Last-Translator: mymage <stefano.consolaro@mymage.it>\n"
"Language-Team: none\n"
"Language: it\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: nplurals=2; plural=n != 1;\n"
"X-Generator: Weblate 4.17\n"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report_substitution_rule
msgid "Action Report Substitution Rule"
msgstr "Regola sostituzione azione resoconto"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_uid
msgid "Created by"
msgstr "Creato da"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_date
msgid "Created on"
msgstr "Creato il"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__display_name
msgid "Display Name"
msgstr "Nome visualizzato"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__domain
msgid "Domain"
msgstr "Dominio"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_thread
msgid "Email Thread"
msgstr "Discussione e-mail"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_compose_message
msgid "Email composition wizard"
msgstr "Procedura guidata creazione e-mail"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__id
msgid "ID"
msgstr "ID"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule____last_update
msgid "Last Modified on"
msgstr "Ultima modifica il"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_uid
msgid "Last Updated by"
msgstr "Ultimo aggiornamento di"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_date
msgid "Last Updated on"
msgstr "Ultimo aggiornamento il"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__model
msgid "Model Name"
msgstr "Nome modello"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__action_report_id
msgid "Report Action"
msgstr "Azione resoconto"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__sequence
msgid "Sequence"
msgstr "Sequenza"
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print_2
msgid "Substitution 2 For Technical guide"
msgstr "Sostituzione 2 per guida tecnica"
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print
msgid "Substitution For Technical guide"
msgstr "Sostituzione per guida tecnica"
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report
msgid "Substitution Report"
msgstr "Resoconto sostituzione"
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report_2
msgid "Substitution Report 2"
msgstr "Resoconto 2 sostituzione"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__substitution_action_report_id
msgid "Substitution Report Action"
msgstr "Azione resoconto sostituzione"
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report__action_report_substitution_rule_ids
#: model_terms:ir.ui.view,arch_db:report_substitute.ir_actions_report_form_view
msgid "Substitution Rules"
msgstr "Regole sostituzione"
#. module: report_substitute
#. odoo-python
#: code:addons/report_substitute/models/ir_actions_report_substitution_rule.py:0
#, python-format
msgid "Substitution infinite loop detected"
msgstr "Rilevato ciclo infinito sostituzione"

View file

@ -0,0 +1,123 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * report_substitute
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report_substitution_rule
msgid "Action Report Substitution Rule"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_uid
msgid "Created by"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__create_date
msgid "Created on"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__display_name
msgid "Display Name"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__domain
msgid "Domain"
msgstr ""
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_thread
msgid "Email Thread"
msgstr ""
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_mail_compose_message
msgid "Email composition wizard"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__id
msgid "ID"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule____last_update
msgid "Last Modified on"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_uid
msgid "Last Updated by"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__write_date
msgid "Last Updated on"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__model
msgid "Model Name"
msgstr ""
#. module: report_substitute
#: model:ir.model,name:report_substitute.model_ir_actions_report
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__action_report_id
msgid "Report Action"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__sequence
msgid "Sequence"
msgstr ""
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print_2
msgid "Substitution 2 For Technical guide"
msgstr ""
#. module: report_substitute
#: model:ir.actions.report,name:report_substitute.substitution_report_print
msgid "Substitution For Technical guide"
msgstr ""
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report
msgid "Substitution Report"
msgstr ""
#. module: report_substitute
#: model_terms:ir.ui.view,arch_db:report_substitute.substitution_report_2
msgid "Substitution Report 2"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report_substitution_rule__substitution_action_report_id
msgid "Substitution Report Action"
msgstr ""
#. module: report_substitute
#: model:ir.model.fields,field_description:report_substitute.field_ir_actions_report__action_report_substitution_rule_ids
#: model_terms:ir.ui.view,arch_db:report_substitute.ir_actions_report_form_view
msgid "Substitution Rules"
msgstr ""
#. module: report_substitute
#. odoo-python
#: code:addons/report_substitute/models/ir_actions_report_substitution_rule.py:0
#, python-format
msgid "Substitution infinite loop detected"
msgstr ""

View file

@ -0,0 +1,3 @@
from . import ir_actions_report
from . import ir_actions_report_substitution_rule
from . import mail_thread

View file

@ -0,0 +1,86 @@
# Copyright 2019 ACSONE SA/NV
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import api, fields, models
from odoo.tools.safe_eval import safe_eval
class IrActionReport(models.Model):
_inherit = "ir.actions.report"
action_report_substitution_rule_ids = fields.One2many(
"ir.actions.report.substitution.rule",
"action_report_id",
string="Substitution Rules",
)
def _get_substitution_report(self, model, active_ids):
self.ensure_one()
model = self.env[model]
for substitution_report_rule in self.action_report_substitution_rule_ids:
domain = safe_eval(substitution_report_rule.domain)
domain.append(("id", "in", active_ids))
if set(model.search(domain).ids) == set(active_ids):
return substitution_report_rule.substitution_action_report_id
return False
def get_substitution_report(self, active_ids):
self.ensure_one()
action_report = self
substitution_report = action_report
while substitution_report:
action_report = substitution_report
substitution_report = action_report._get_substitution_report(
action_report.model, active_ids
)
return action_report
@api.model
def get_substitution_report_action(self, action, active_ids):
if action.get("id"):
action_report = self.browse(action["id"])
substitution_report = action_report
while substitution_report:
action_report = substitution_report
substitution_report = action_report._get_substitution_report(
action_report.model, active_ids
)
action.update(action_report.read()[0])
return action
def _render(self, report_ref, res_ids, data=None):
report = self._get_report(report_ref)
substitution_report = report.get_substitution_report(res_ids)
return super(IrActionReport, self)._render(
substitution_report.report_name, res_ids, data=data
)
def _render_qweb_pdf(self, report_ref, res_ids=None, data=None):
report = self._get_report(report_ref)
substitution_report = report.get_substitution_report(res_ids)
if substitution_report.filtered(lambda r: r.report_type == "qweb-pdf"):
return super(IrActionReport, self)._render_qweb_pdf(
substitution_report, res_ids=res_ids, data=data
)
return super(IrActionReport, self)._render_qweb_pdf(
report_ref, res_ids=res_ids, data=data
)
def report_action(self, docids, data=None, config=True):
if docids:
if isinstance(docids, models.Model):
active_ids = docids.ids
elif isinstance(docids, int):
active_ids = [docids]
elif isinstance(docids, list):
active_ids = docids
substitution_report = self.get_substitution_report(active_ids)
return super(IrActionReport, substitution_report).report_action(
docids, data, config
)
return super().report_action(docids, data, config)
def get_action_report_substitution_rule_ids(self):
return self.action_report_substitution_rule_ids.ids

View file

@ -0,0 +1,47 @@
# Copyright 2019 ACSONE SA/NV
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import _, api, fields, models
from odoo.exceptions import ValidationError
class ActionsReportSubstitutionRule(models.Model):
_name = "ir.actions.report.substitution.rule"
_description = "Action Report Substitution Rule"
_order = "sequence ASC"
sequence = fields.Integer(default=10)
action_report_id = fields.Many2one(
comodel_name="ir.actions.report",
string="Report Action",
required=True,
ondelete="cascade",
)
model = fields.Char(related="action_report_id.model", store=True)
domain = fields.Char(required=True, default="[]")
substitution_action_report_id = fields.Many2one(
comodel_name="ir.actions.report",
string="Substitution Report Action",
required=True,
ondelete="cascade",
domain="[('model', '=', model)]",
)
@api.constrains("substitution_action_report_id", "action_report_id")
def _check_substitution_infinite_loop(self):
def _check_infinite_loop(original_report, substitution_report):
if original_report == substitution_report:
raise ValidationError(_("Substitution infinite loop detected"))
for (
substitution_rule
) in substitution_report.action_report_substitution_rule_ids:
_check_infinite_loop(
original_report,
substitution_rule.substitution_action_report_id,
)
for rec in self:
_check_infinite_loop(
rec.action_report_id, rec.substitution_action_report_id
)

View file

@ -0,0 +1,21 @@
# Copyright 2019 ACSONE SA/NV
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import models
class MailThread(models.AbstractModel):
_inherit = "mail.thread"
def message_post_with_template(self, template_id, **kwargs):
template = self.env["mail.template"].browse(template_id)
if template and template.report_template and self.ids:
active_ids = self.ids
report_template = template.report_template.get_substitution_report(
active_ids
)
return super(
MailThread, self.with_context(default_report_template=report_template)
).message_post_with_template(template_id, **kwargs)
return super().message_post_with_template(template_id, **kwargs)

View file

@ -0,0 +1 @@
* Bejaoui Souheil <souheil.bejaoui@acsone.eu>

View file

@ -0,0 +1,4 @@
This module allows you to create substitution rules for report actions.
A typical use case is to replace a standard report by alternative reports
when some conditions are met. For instance, it allows to configure alternate
reports for different companies.

View file

@ -0,0 +1 @@
- The document name result should take the name of the substitution report.

View file

@ -0,0 +1,15 @@
To use this module, you need to:
#. Go to 'Actions' / 'Reports'
#. Select the desired report you want to 'Substitution Rules'
#. In the substitutions page add a new line
#. Select the substitution report action
#. Set a domain to specify when this substitution should happen
When a user calls a report action, the system tries to find the first
substitution in with a domain that matches all records.

View file

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="utf-8" ?>
<!-- Copyright 2019 ACSONE SA/NV
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
<odoo>
<record model="ir.model.access" id="action_report_substitution_rule_user_access">
<field name="name">action.report.substitution.rule user access</field>
<field name="model_id" ref="model_ir_actions_report_substitution_rule" />
<field name="perm_read" eval="1" />
<field name="perm_create" eval="0" />
<field name="perm_write" eval="0" />
<field name="perm_unlink" eval="0" />
</record>
<record model="ir.model.access" id="action_report_substitution_rule_manager_access">
<field name="name">action.report.substitution.rule manager access</field>
<field name="model_id" ref="model_ir_actions_report_substitution_rule" />
<field name="group_id" ref="base.group_system" />
<field name="perm_read" eval="1" />
<field name="perm_create" eval="1" />
<field name="perm_write" eval="1" />
<field name="perm_unlink" eval="1" />
</record>
</odoo>

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.2 KiB

View file

@ -0,0 +1,455 @@
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<meta name="generator" content="Docutils: https://docutils.sourceforge.io/" />
<title>README.rst</title>
<style type="text/css">
/*
:Author: David Goodger (goodger@python.org)
:Id: $Id: html4css1.css 9511 2024-01-13 09:50:07Z milde $
:Copyright: This stylesheet has been placed in the public domain.
Default cascading style sheet for the HTML output of Docutils.
Despite the name, some widely supported CSS2 features are used.
See https://docutils.sourceforge.io/docs/howto/html-stylesheets.html for how to
customize this style sheet.
*/
/* used to remove borders from tables and images */
.borderless, table.borderless td, table.borderless th {
border: 0 }
table.borderless td, table.borderless th {
/* Override padding for "table.docutils td" with "! important".
The right padding separates the table cells. */
padding: 0 0.5em 0 0 ! important }
.first {
/* Override more specific margin styles with "! important". */
margin-top: 0 ! important }
.last, .with-subtitle {
margin-bottom: 0 ! important }
.hidden {
display: none }
.subscript {
vertical-align: sub;
font-size: smaller }
.superscript {
vertical-align: super;
font-size: smaller }
a.toc-backref {
text-decoration: none ;
color: black }
blockquote.epigraph {
margin: 2em 5em ; }
dl.docutils dd {
margin-bottom: 0.5em }
object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
overflow: hidden;
}
/* Uncomment (and remove this text!) to get bold-faced definition list terms
dl.docutils dt {
font-weight: bold }
*/
div.abstract {
margin: 2em 5em }
div.abstract p.topic-title {
font-weight: bold ;
text-align: center }
div.admonition, div.attention, div.caution, div.danger, div.error,
div.hint, div.important, div.note, div.tip, div.warning {
margin: 2em ;
border: medium outset ;
padding: 1em }
div.admonition p.admonition-title, div.hint p.admonition-title,
div.important p.admonition-title, div.note p.admonition-title,
div.tip p.admonition-title {
font-weight: bold ;
font-family: sans-serif }
div.attention p.admonition-title, div.caution p.admonition-title,
div.danger p.admonition-title, div.error p.admonition-title,
div.warning p.admonition-title, .code .error {
color: red ;
font-weight: bold ;
font-family: sans-serif }
/* Uncomment (and remove this text!) to get reduced vertical space in
compound paragraphs.
div.compound .compound-first, div.compound .compound-middle {
margin-bottom: 0.5em }
div.compound .compound-last, div.compound .compound-middle {
margin-top: 0.5em }
*/
div.dedication {
margin: 2em 5em ;
text-align: center ;
font-style: italic }
div.dedication p.topic-title {
font-weight: bold ;
font-style: normal }
div.figure {
margin-left: 2em ;
margin-right: 2em }
div.footer, div.header {
clear: both;
font-size: smaller }
div.line-block {
display: block ;
margin-top: 1em ;
margin-bottom: 1em }
div.line-block div.line-block {
margin-top: 0 ;
margin-bottom: 0 ;
margin-left: 1.5em }
div.sidebar {
margin: 0 0 0.5em 1em ;
border: medium outset ;
padding: 1em ;
background-color: #ffffee ;
width: 40% ;
float: right ;
clear: right }
div.sidebar p.rubric {
font-family: sans-serif ;
font-size: medium }
div.system-messages {
margin: 5em }
div.system-messages h1 {
color: red }
div.system-message {
border: medium outset ;
padding: 1em }
div.system-message p.system-message-title {
color: red ;
font-weight: bold }
div.topic {
margin: 2em }
h1.section-subtitle, h2.section-subtitle, h3.section-subtitle,
h4.section-subtitle, h5.section-subtitle, h6.section-subtitle {
margin-top: 0.4em }
h1.title {
text-align: center }
h2.subtitle {
text-align: center }
hr.docutils {
width: 75% }
img.align-left, .figure.align-left, object.align-left, table.align-left {
clear: left ;
float: left ;
margin-right: 1em }
img.align-right, .figure.align-right, object.align-right, table.align-right {
clear: right ;
float: right ;
margin-left: 1em }
img.align-center, .figure.align-center, object.align-center {
display: block;
margin-left: auto;
margin-right: auto;
}
table.align-center {
margin-left: auto;
margin-right: auto;
}
.align-left {
text-align: left }
.align-center {
clear: both ;
text-align: center }
.align-right {
text-align: right }
/* reset inner alignment in figures */
div.align-right {
text-align: inherit }
/* div.align-center * { */
/* text-align: left } */
.align-top {
vertical-align: top }
.align-middle {
vertical-align: middle }
.align-bottom {
vertical-align: bottom }
ol.simple, ul.simple {
margin-bottom: 1em }
ol.arabic {
list-style: decimal }
ol.loweralpha {
list-style: lower-alpha }
ol.upperalpha {
list-style: upper-alpha }
ol.lowerroman {
list-style: lower-roman }
ol.upperroman {
list-style: upper-roman }
p.attribution {
text-align: right ;
margin-left: 50% }
p.caption {
font-style: italic }
p.credits {
font-style: italic ;
font-size: smaller }
p.label {
white-space: nowrap }
p.rubric {
font-weight: bold ;
font-size: larger ;
color: maroon ;
text-align: center }
p.sidebar-title {
font-family: sans-serif ;
font-weight: bold ;
font-size: larger }
p.sidebar-subtitle {
font-family: sans-serif ;
font-weight: bold }
p.topic-title {
font-weight: bold }
pre.address {
margin-bottom: 0 ;
margin-top: 0 ;
font: inherit }
pre.literal-block, pre.doctest-block, pre.math, pre.code {
margin-left: 2em ;
margin-right: 2em }
pre.code .ln { color: gray; } /* line numbers */
pre.code, code { background-color: #eeeeee }
pre.code .comment, code .comment { color: #5C6576 }
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold }
pre.code .literal.string, code .literal.string { color: #0C5404 }
pre.code .name.builtin, code .name.builtin { color: #352B84 }
pre.code .deleted, code .deleted { background-color: #DEB0A1}
pre.code .inserted, code .inserted { background-color: #A3D289}
span.classifier {
font-family: sans-serif ;
font-style: oblique }
span.classifier-delimiter {
font-family: sans-serif ;
font-weight: bold }
span.interpreted {
font-family: sans-serif }
span.option {
white-space: nowrap }
span.pre {
white-space: pre }
span.problematic, pre.problematic {
color: red }
span.section-subtitle {
/* font-size relative to parent (h1..h6 element) */
font-size: 80% }
table.citation {
border-left: solid 1px gray;
margin-left: 1px }
table.docinfo {
margin: 2em 4em }
table.docutils {
margin-top: 0.5em ;
margin-bottom: 0.5em }
table.footnote {
border-left: solid 1px black;
margin-left: 1px }
table.docutils td, table.docutils th,
table.docinfo td, table.docinfo th {
padding-left: 0.5em ;
padding-right: 0.5em ;
vertical-align: top }
table.docutils th.field-name, table.docinfo th.docinfo-name {
font-weight: bold ;
text-align: left ;
white-space: nowrap ;
padding-left: 0 }
/* "booktabs" style (no vertical lines) */
table.docutils.booktabs {
border: 0px;
border-top: 2px solid;
border-bottom: 2px solid;
border-collapse: collapse;
}
table.docutils.booktabs * {
border: 0px;
}
table.docutils.booktabs th {
border-bottom: thin solid;
text-align: left;
}
h1 tt.docutils, h2 tt.docutils, h3 tt.docutils,
h4 tt.docutils, h5 tt.docutils, h6 tt.docutils {
font-size: 100% }
ul.auto-toc {
list-style-type: none }
</style>
</head>
<body>
<div class="document">
<a class="reference external image-reference" href="https://odoo-community.org/get-involved?utm_source=readme">
<img alt="Odoo Community Association" src="https://odoo-community.org/readme-banner-image" />
</a>
<div class="section" id="report-substitute">
<h1>Report Substitute</h1>
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! This file is generated by oca-gen-addon-readme !!
!! changes will be overwritten. !!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! source digest: sha256:6e713ec7c433154ddfbdc2a954a61a5df2fcdabf03d875d3b3dd5bd0ce4ff980
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
<p><a class="reference external image-reference" href="https://odoo-community.org/page/development-status"><img alt="Beta" src="https://img.shields.io/badge/maturity-Beta-yellow.png" /></a> <a class="reference external image-reference" href="http://www.gnu.org/licenses/agpl-3.0-standalone.html"><img alt="License: AGPL-3" src="https://img.shields.io/badge/license-AGPL--3-blue.png" /></a> <a class="reference external image-reference" href="https://github.com/OCA/reporting-engine/tree/16.0/report_substitute"><img alt="OCA/reporting-engine" src="https://img.shields.io/badge/github-OCA%2Freporting--engine-lightgray.png?logo=github" /></a> <a class="reference external image-reference" href="https://translation.odoo-community.org/projects/reporting-engine-16-0/reporting-engine-16-0-report_substitute"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external image-reference" href="https://runboat.odoo-community.org/builds?repo=OCA/reporting-engine&amp;target_branch=16.0"><img alt="Try me on Runboat" src="https://img.shields.io/badge/runboat-Try%20me-875A7B.png" /></a></p>
<p>This module allows you to create substitution rules for report actions.
A typical use case is to replace a standard report by alternative reports
when some conditions are met. For instance, it allows to configure alternate
reports for different companies.</p>
<p><strong>Table of contents</strong></p>
<div class="contents local topic" id="contents">
<ul class="simple">
<li><a class="reference internal" href="#usage" id="toc-entry-1">Usage</a></li>
<li><a class="reference internal" href="#known-issues-roadmap" id="toc-entry-2">Known issues / Roadmap</a></li>
<li><a class="reference internal" href="#bug-tracker" id="toc-entry-3">Bug Tracker</a></li>
<li><a class="reference internal" href="#credits" id="toc-entry-4">Credits</a><ul>
<li><a class="reference internal" href="#authors" id="toc-entry-5">Authors</a></li>
<li><a class="reference internal" href="#contributors" id="toc-entry-6">Contributors</a></li>
<li><a class="reference internal" href="#maintainers" id="toc-entry-7">Maintainers</a></li>
</ul>
</li>
</ul>
</div>
<div class="section" id="usage">
<h2><a class="toc-backref" href="#toc-entry-1">Usage</a></h2>
<p>To use this module, you need to:</p>
<ol class="arabic simple">
<li>Go to Actions / Reports</li>
<li>Select the desired report you want to Substitution Rules</li>
<li>In the substitutions page add a new line</li>
<li>Select the substitution report action</li>
<li>Set a domain to specify when this substitution should happen</li>
</ol>
<p>When a user calls a report action, the system tries to find the first
substitution in with a domain that matches all records.</p>
</div>
<div class="section" id="known-issues-roadmap">
<h2><a class="toc-backref" href="#toc-entry-2">Known issues / Roadmap</a></h2>
<ul class="simple">
<li>The document name result should take the name of the substitution report.</li>
</ul>
</div>
<div class="section" id="bug-tracker">
<h2><a class="toc-backref" href="#toc-entry-3">Bug Tracker</a></h2>
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/reporting-engine/issues">GitHub Issues</a>.
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
<a class="reference external" href="https://github.com/OCA/reporting-engine/issues/new?body=module:%20report_substitute%0Aversion:%2016.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**">feedback</a>.</p>
<p>Do not contact contributors directly about support or help with technical issues.</p>
</div>
<div class="section" id="credits">
<h2><a class="toc-backref" href="#toc-entry-4">Credits</a></h2>
<div class="section" id="authors">
<h3><a class="toc-backref" href="#toc-entry-5">Authors</a></h3>
<ul class="simple">
<li>ACSONE SA/NV</li>
</ul>
</div>
<div class="section" id="contributors">
<h3><a class="toc-backref" href="#toc-entry-6">Contributors</a></h3>
<ul class="simple">
<li>Bejaoui Souheil &lt;<a class="reference external" href="mailto:souheil.bejaoui&#64;acsone.eu">souheil.bejaoui&#64;acsone.eu</a>&gt;</li>
</ul>
</div>
<div class="section" id="maintainers">
<h3><a class="toc-backref" href="#toc-entry-7">Maintainers</a></h3>
<p>This module is maintained by the OCA.</p>
<a class="reference external image-reference" href="https://odoo-community.org">
<img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" />
</a>
<p>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.</p>
<p>Current <a class="reference external" href="https://odoo-community.org/page/maintainer-role">maintainer</a>:</p>
<p><a class="reference external image-reference" href="https://github.com/sbejaoui"><img alt="sbejaoui" src="https://github.com/sbejaoui.png?size=40px" /></a></p>
<p>This module is part of the <a class="reference external" href="https://github.com/OCA/reporting-engine/tree/16.0/report_substitute">OCA/reporting-engine</a> project on GitHub.</p>
<p>You are welcome to contribute. To learn how please visit <a class="reference external" href="https://odoo-community.org/page/Contribute">https://odoo-community.org/page/Contribute</a>.</p>
</div>
</div>
</div>
</div>
</body>
</html>

View file

@ -0,0 +1,29 @@
/** @odoo-module **/
import {registry} from "@web/core/registry";
registry
.category("ir.actions.report handlers")
.add("sustitution_handler", async function (action, options, env) {
const orm = env.services.orm;
const action_report_substitution_rule_ids = await orm.call(
"ir.actions.report",
"get_action_report_substitution_rule_ids",
[action.id]
);
if (
action.type === "ir.actions.report" &&
action.context.active_ids &&
action_report_substitution_rule_ids &&
action_report_substitution_rule_ids.length !== 0
) {
var active_ids = action.context.active_ids;
const substitution = await orm.call(
"ir.actions.report",
"get_substitution_report_action",
[action, active_ids]
);
Object.assign(action, substitution);
}
return Promise.resolve(false);
});

View file

@ -0,0 +1 @@
from . import test_report_substitute

View file

@ -0,0 +1,105 @@
# Copyright 2019 ACSONE SA/NV
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo.exceptions import ValidationError
from odoo.tests.common import TransactionCase
class TestReportSubstitute(TransactionCase):
def setUp(self):
# In the demo file we create a new report for ir.module.module model
# with a substation rule from the original report action
super(TestReportSubstitute, self).setUp()
self.action_report = self.env.ref("base.ir_module_reference_print")
self.res_ids = self.env.ref("base.module_base").ids
self.substitution_rule = self.env.ref(
"report_substitute.substitution_rule_demo_1"
)
self.env.company.external_report_layout_id = self.env.ref(
"web.external_layout_standard"
).id
def test_substitution(self):
res = str(
self.action_report._render(
self.action_report.report_name, res_ids=self.res_ids
)[0]
)
self.assertIn('<div class="page">Substitution Report</div>', res)
# remove the substation rule
self.substitution_rule.unlink()
res = str(
self.action_report._render(
self.action_report.report_name, res_ids=self.res_ids
)[0]
)
self.assertNotIn('<div class="page">Substitution Report</div>', res)
def test_recursive_substitution(self):
res = str(
self.action_report._render(
self.action_report.report_name, res_ids=self.res_ids
)[0]
)
self.assertNotIn('<div class="page">Substitution Report 2</div>', res)
self.env["ir.actions.report.substitution.rule"].create(
{
"substitution_action_report_id": self.env.ref(
"report_substitute.substitution_report_print_2"
).id,
"action_report_id": self.env.ref(
"report_substitute.substitution_report_print"
).id,
}
)
res = str(
self.action_report._render(
self.action_report.report_name, res_ids=self.res_ids
)[0]
)
self.assertIn('<div class="page">Substitution Report 2</div>', res)
def test_substitution_with_domain(self):
self.substitution_rule.write({"domain": "[('name', '=', 'base')]"})
res = str(
self.action_report._render(
self.action_report.report_name, res_ids=self.res_ids
)[0]
)
self.assertIn('<div class="page">Substitution Report</div>', res)
self.substitution_rule.write({"domain": "[('name', '!=', 'base')]"})
res = str(
self.action_report._render(
self.action_report.report_name, res_ids=self.res_ids
)[0]
)
self.assertNotIn('<div class="page">Substitution Report</div>', res)
def test_substitution_with_action_dict(self):
substitution_report_action = self.env[
"ir.actions.report"
].get_substitution_report_action(self.action_report.read()[0], self.res_ids)
self.assertEqual(
substitution_report_action["id"],
self.substitution_rule.substitution_action_report_id.id,
)
def test_substitution_with_report_action(self):
res = self.action_report.report_action(self.res_ids)
self.assertEqual(
res["report_name"],
self.substitution_rule.substitution_action_report_id.report_name,
)
def test_substitution_infinite_loop(self):
with self.assertRaises(ValidationError):
self.env["ir.actions.report.substitution.rule"].create(
{
"action_report_id": self.env.ref(
"report_substitute.substitution_report_print"
).id,
"substitution_action_report_id": self.env.ref(
"base.ir_module_reference_print"
).id,
}
)

View file

@ -0,0 +1,47 @@
<?xml version="1.0" encoding="utf-8" ?>
<!-- Copyright 2019 ACSONE SA/NV
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
<odoo>
<record model="ir.ui.view" id="ir_actions_report_form_view">
<field name="name">ir.actions.report.form (in report_substitute)</field>
<field name="model">ir.actions.report</field>
<field name="inherit_id" ref="base.act_report_xml_view" />
<field name="arch" type="xml">
<xpath expr="//page[@name='advanced']" position="after">
<page name="report_substitution_rule" string="Substitution Rules">
<field name="action_report_substitution_rule_ids">
<tree>
<field name="sequence" widget="handle" />
<field name="substitution_action_report_id" />
<field name="domain" />
</tree>
<form>
<sheet>
<group>
<field
name="action_report_id"
invisible="1"
readonly="1"
required="0"
/>
<field name="model" invisible="1" />
<field
name="substitution_action_report_id"
domain="[('model', '=', model), ('id', '!=', parent.id)]"
/>
</group>
<group>
<field
name="domain"
widget="domain"
options="{'model': 'model'}"
/>
</group>
</sheet>
</form>
</field>
</page>
</xpath>
</field>
</record>
</odoo>

View file

@ -0,0 +1 @@
from . import mail_compose_message

View file

@ -0,0 +1,32 @@
# Copyright 2019 ACSONE SA/NV
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import api, models
class MailComposeMessage(models.TransientModel):
_inherit = "mail.compose.message"
@api.model
def generate_email_for_composer(self, template_id, res_ids, fields):
if self.template_id:
report_template = self.template_id.report_template
active_ids = []
if self.env.context.get("active_ids"):
active_ids = self.env.context.get("active_ids")
elif self.env.context.get("default_res_id"):
active_ids = [self.env.context.get("default_res_id")]
if (
report_template
and report_template.action_report_substitution_rule_ids
and active_ids
):
report_template = (
self.template_id.report_template.get_substitution_report(active_ids)
)
return super(
MailComposeMessage,
self.with_context(default_report_template=report_template),
).generate_email_for_composer(template_id, res_ids, fields)
return super().generate_email_for_composer(template_id, res_ids, fields)