Move 124 sale modules to oca-sale, create oca-project with 56 project modules from oca-workflow-process

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
Ernad Husremovic 2025-08-30 18:04:10 +02:00
parent 9eb7ae5807
commit 6094c218b2
2332 changed files with 125826 additions and 0 deletions

View file

@ -0,0 +1,46 @@
# Project Task Recurring Activity
Odoo addon: project_task_recurring_activity
## Installation
```bash
pip install odoo-bringout-oca-project-project_task_recurring_activity
```
## Dependencies
This addon depends on:
- project
## Manifest Information
- **Name**: Project Task Recurring Activity
- **Version**: 16.0.1.0.0
- **Category**: Project Management
- **License**: AGPL-3
- **Installable**: False
## Source
Based on [OCA/project](https://github.com/OCA/project) branch 16.0, addon `project_task_recurring_activity`.
## 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 Project_task_recurring_activity Module - project_task_recurring_activity
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 project_task_recurring_activity. 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,5 @@
# Dependencies
This addon depends on:
- [project](https://github.com/bringout/oca-ocb-project/tree/5bbf7d0517a5706a48472bdf6a077a4467d11869/odoo-bringout-oca-ocb-project)

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 project_task_recurring_activity or install in UI.

View file

@ -0,0 +1,7 @@
# Install
```bash
pip install odoo-bringout-oca-project-project_task_recurring_activity"
# or
uv pip install odoo-bringout-oca-project-project_task_recurring_activity"
```

View file

@ -0,0 +1,14 @@
# Models
Detected core models and extensions in project_task_recurring_activity.
```mermaid
classDiagram
class recurring_activity
class project_task
class project_task_recurrence
```
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: project_task_recurring_activity. Provides features documented in upstream Odoo 16 under this addon.
- Source: OCA/OCB 16.0, addon project_task_recurring_activity
- License: LGPL-3

View file

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

View file

@ -0,0 +1,34 @@
# Security
Access control and security definitions in project_task_recurring_activity.
## Access Control Lists (ACLs)
Model access permissions defined in:
- **[ir.model.access.csv](../project_task_recurring_activity/security/ir.model.access.csv)**
- 1 model access rules
## Record Rules
Row-level security rules defined in:
```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.model.access.csv](../project_task_recurring_activity/security/ir.model.access.csv)**
- Model access permissions (CRUD rights)
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 project_task_recurring_activity
```

View file

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

View file

@ -0,0 +1,101 @@
===============================
Project Task Recurring Activity
===============================
..
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! This file is generated by oca-gen-addon-readme !!
!! changes will be overwritten. !!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! source digest: sha256:e2aeb979b785a759d9358928d0bd8639c06511dc3a86447092ab50cc86709acd
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png
:target: https://odoo-community.org/page/development-status
:alt: Beta
.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
:alt: License: AGPL-3
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fproject-lightgray.png?logo=github
:target: https://github.com/OCA/project/tree/16.0/project_task_recurring_activity
:alt: OCA/project
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
:target: https://translation.odoo-community.org/projects/project-16-0/project-16-0-project_task_recurring_activity
: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/project&target_branch=16.0
:alt: Try me on Runboat
|badge1| |badge2| |badge3| |badge4| |badge5|
This module allow users to add activities to the recurring tasks and have them automatically duplicated within recurring tasks at set intervals.
**Table of contents**
.. contents::
:local:
Configuration
=============
#.Enable recurring task in config settings
Usage
=====
Open recurring task.
#. In tab "Recurrence", user can set the activites that should be created each time the task is duplicated.
#. For testing purposes, a button "Create next recurring task" is visible in developer mode to check how next task will be created.
Bug Tracker
===========
Bugs are tracked on `GitHub Issues <https://github.com/OCA/project/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/project/issues/new?body=module:%20project_task_recurring_activity%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
~~~~~~~
* Cetmix
Contributors
~~~~~~~~~~~~
* Cetmix <https://cetmix.com>
* Dessan Hemrayev <dessanhemrayev@gmail.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-dessanhemrayev| image:: https://github.com/dessanhemrayev.png?size=40px
:target: https://github.com/dessanhemrayev
:alt: dessanhemrayev
.. |maintainer-CetmixGitDrone| image:: https://github.com/CetmixGitDrone.png?size=40px
:target: https://github.com/CetmixGitDrone
:alt: CetmixGitDrone
Current `maintainers <https://odoo-community.org/page/maintainer-role>`__:
|maintainer-dessanhemrayev| |maintainer-CetmixGitDrone|
This module is part of the `OCA/project <https://github.com/OCA/project/tree/16.0/project_task_recurring_activity>`_ project on GitHub.
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

View file

@ -0,0 +1,3 @@
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from . import models

View file

@ -0,0 +1,18 @@
{
"name": "Project Task Recurring Activity",
"summary": """Project Task Recurring Activity""",
"author": "Cetmix, Odoo Community Association (OCA)",
"website": "https://github.com/OCA/project",
"category": "Project Management",
"version": "16.0.1.0.0",
"license": "AGPL-3",
"depends": ["project"],
"data": [
"security/ir.model.access.csv",
"views/recurring_activity.xml",
"views/project_task.xml",
"data/recurring_activity.xml",
],
"application": False,
"maintainers": ["dessanhemrayev", "CetmixGitDrone"],
}

View file

@ -0,0 +1,15 @@
<?xml version="1.0" encoding="utf-8" ?>
<odoo noupdate="1">
<record id="ir_cron_recurring_activities" model="ir.cron">
<field name="name">Project: Create Recurring Activities</field>
<field name="model_id" ref="model_recurring_activity" />
<field name="state">code</field>
<field name="code">model._cron_create_activities()</field>
<field name="interval_type">days</field>
<field name="numbercall">-1</field>
<field
name="nextcall"
eval="(DateTime.now().replace(hour=3, minute=0) + timedelta(days=1)).strftime('%Y-%m-%d %H:%M:%S')"
/>
</record>
</odoo>

View file

@ -0,0 +1,130 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_task_recurring_activity
#
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: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Activities"
msgstr "Aktivnosti"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__activity_type_id
msgid "Activity Type"
msgstr "Tip aktivnosti"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__user_id
msgid "Assigned to"
msgstr "Dodjeljeno"
#. module: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Create Next Recurring Task"
msgstr "Zadaci u toku"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_uid
msgid "Created by"
msgstr "Kreirao"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_date
msgid "Created on"
msgstr "Kreirano"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__custom_activity_ids
msgid "Custom Activity"
msgstr "Zahtev sa portala"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__days_after_task_creation_date
msgid "Days After Task Creation Date"
msgstr "Days After Zadatak Creation Datum"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__description
msgid "Description"
msgstr "Opis"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__display_name
msgid "Display Name"
msgstr "Prikazani naziv"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__id
msgid "ID"
msgstr "ID"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity____last_update
msgid "Last Modified on"
msgstr "Zadnje mijenjano"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_uid
msgid "Last Updated by"
msgstr "Zadnji ažurirao"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_date
msgid "Last Updated on"
msgstr "Zadnje ažurirano"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__old_date_recurring_task
msgid "Old Date Recurring Task"
msgstr "Planiranje i budžet"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__project_task_id
msgid "Project Task"
msgstr "Projektni zadatak"
#. module: project_task_recurring_activity
#: model:ir.actions.server,name:project_task_recurring_activity.ir_cron_recurring_activities_ir_actions_server
#: model:ir.cron,cron_name:project_task_recurring_activity.ir_cron_recurring_activities
msgid "Project: Create Recurring Activities"
msgstr "Projekat: Kreiraj Recurring Activities"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_recurring_activity
msgid "Recurring activity"
msgstr "Recurring activity"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__summary
msgid "Summary"
msgstr "Sažetak"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task
msgid "Task"
msgstr "Zadatak"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task_recurrence
msgid "Task Recurrence"
msgstr "Zadatak Recurrence"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__recurring_activity_ids
msgid "activity"
msgstr "Zahtev za uslugom"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__next_recurrence_date
msgid "next_date"
msgstr "Planiranje bašte"

View file

@ -0,0 +1,140 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_task_recurring_activity
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 14.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2024-03-24 23:53+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: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Activities"
msgstr "actividades"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__activity_type_id
msgid "Activity Type"
msgstr "Tipo de actividad"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__user_id
msgid "Assigned to"
msgstr "Asignado/a a"
#. module: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Create Next Recurring Task"
msgstr "Crear la Siguiente Tarea Recurrente"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_uid
msgid "Created by"
msgstr "Creado por"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_date
msgid "Created on"
msgstr "Creado el"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__custom_activity_ids
msgid "Custom Activity"
msgstr "Actividad personalizada"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__days_after_task_creation_date
msgid "Days After Task Creation Date"
msgstr "Días después de la fecha de creación de la tarea"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__description
msgid "Description"
msgstr "Descripción"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__display_name
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__display_name
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__display_name
msgid "Display Name"
msgstr "Mostrar Nombre"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__id
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__id
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__id
msgid "ID"
msgstr "ID (identificación)"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task____last_update
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence____last_update
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity____last_update
msgid "Last Modified on"
msgstr "Última Modificación el"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_uid
msgid "Last Updated by"
msgstr "Última Actualización realizada por"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_date
msgid "Last Updated on"
msgstr "Última Actualización el"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__old_date_recurring_task
msgid "Old Date Recurring Task"
msgstr "Tarea recurrente con fecha anterior"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__project_task_id
msgid "Project Task"
msgstr "Tarea de proyecto"
#. module: project_task_recurring_activity
#: model:ir.actions.server,name:project_task_recurring_activity.ir_cron_recurring_activities_ir_actions_server
#: model:ir.cron,cron_name:project_task_recurring_activity.ir_cron_recurring_activities
#: model:ir.cron,name:project_task_recurring_activity.ir_cron_recurring_activities
msgid "Project: Create Recurring Activities"
msgstr "Proyecto: Crear actividades recurrentes"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_recurring_activity
msgid "Recurring activity"
msgstr "Actividad recurrente"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__summary
msgid "Summary"
msgstr "Resumen"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task
msgid "Task"
msgstr "Tarea"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task_recurrence
msgid "Task Recurrence"
msgstr "Recurrencia de tareas"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__recurring_activity_ids
msgid "activity"
msgstr "Actividad"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__next_recurrence_date
msgid "next_date"
msgstr "next_date"

View file

@ -0,0 +1,140 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_task_recurring_activity
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 14.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2023-04-21 11:11+0000\n"
"Last-Translator: Francesco Foresti <francesco.foresti@ooops404.com>\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.14.1\n"
#. module: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Activities"
msgstr "Attività"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__activity_type_id
msgid "Activity Type"
msgstr "Tipo attività"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__user_id
msgid "Assigned to"
msgstr "Assegnato a"
#. module: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Create Next Recurring Task"
msgstr "Crea prossimo lavoro ricorrente"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_uid
msgid "Created by"
msgstr "Creato da"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_date
msgid "Created on"
msgstr "Creato il"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__custom_activity_ids
msgid "Custom Activity"
msgstr "Attività personalizzata"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__days_after_task_creation_date
msgid "Days After Task Creation Date"
msgstr "Giorni dopo la data creazione del lavoro"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__description
msgid "Description"
msgstr "Descrizione"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__display_name
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__display_name
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__display_name
msgid "Display Name"
msgstr "Nome visualizzato"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__id
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__id
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__id
msgid "ID"
msgstr "ID"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task____last_update
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence____last_update
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity____last_update
msgid "Last Modified on"
msgstr "Ultima modifica il"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_uid
msgid "Last Updated by"
msgstr "Ultimo aggiornamento di"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_date
msgid "Last Updated on"
msgstr "Ultimo aggiornamento il"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__old_date_recurring_task
msgid "Old Date Recurring Task"
msgstr "Vecchia data lavoro ricorrente"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__project_task_id
msgid "Project Task"
msgstr "Lavoro del progetto"
#. module: project_task_recurring_activity
#: model:ir.actions.server,name:project_task_recurring_activity.ir_cron_recurring_activities_ir_actions_server
#: model:ir.cron,cron_name:project_task_recurring_activity.ir_cron_recurring_activities
#: model:ir.cron,name:project_task_recurring_activity.ir_cron_recurring_activities
msgid "Project: Create Recurring Activities"
msgstr "Progetto: crea attività ricorrenti"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_recurring_activity
msgid "Recurring activity"
msgstr "Attività ricorrente"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__summary
msgid "Summary"
msgstr "Riepilogo"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task
msgid "Task"
msgstr "Lavoro"
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task_recurrence
msgid "Task Recurrence"
msgstr "Ricorrenza lavoro"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__recurring_activity_ids
msgid "activity"
msgstr "attività"
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__next_recurrence_date
msgid "next_date"
msgstr "next_date"

View file

@ -0,0 +1,130 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * project_task_recurring_activity
#
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: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Activities"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__activity_type_id
msgid "Activity Type"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__user_id
msgid "Assigned to"
msgstr ""
#. module: project_task_recurring_activity
#: model_terms:ir.ui.view,arch_db:project_task_recurring_activity.view_recurring_activity_form
msgid "Create Next Recurring Task"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_uid
msgid "Created by"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__create_date
msgid "Created on"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__custom_activity_ids
msgid "Custom Activity"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__days_after_task_creation_date
msgid "Days After Task Creation Date"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__description
msgid "Description"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__display_name
msgid "Display Name"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__id
msgid "ID"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity____last_update
msgid "Last Modified on"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_uid
msgid "Last Updated by"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__write_date
msgid "Last Updated on"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task_recurrence__old_date_recurring_task
msgid "Old Date Recurring Task"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__project_task_id
msgid "Project Task"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.actions.server,name:project_task_recurring_activity.ir_cron_recurring_activities_ir_actions_server
#: model:ir.cron,cron_name:project_task_recurring_activity.ir_cron_recurring_activities
msgid "Project: Create Recurring Activities"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_recurring_activity
msgid "Recurring activity"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__summary
msgid "Summary"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task
msgid "Task"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model,name:project_task_recurring_activity.model_project_task_recurrence
msgid "Task Recurrence"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_project_task__recurring_activity_ids
msgid "activity"
msgstr ""
#. module: project_task_recurring_activity
#: model:ir.model.fields,field_description:project_task_recurring_activity.field_recurring_activity__next_recurrence_date
msgid "next_date"
msgstr ""

View file

@ -0,0 +1,5 @@
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from . import project_task
from . import recurring_activity
from . import project_task_recurrence

View file

@ -0,0 +1,102 @@
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from datetime import timedelta
from odoo import api, fields, models
WEEKS = {
"first": 1,
"second": 2,
"third": 3,
"last": 4,
}
class ProjectTask(models.Model):
_inherit = "project.task"
recurring_activity_ids = fields.One2many(
"recurring.activity", "project_task_id", string="activity", copy=True
)
custom_activity_ids = fields.Many2many(
"recurring.activity", compute="_compute_activity_ids", store=True, copy=True
)
@api.depends("recurring_activity_ids")
def _compute_activity_ids(self):
for item in self:
item.custom_activity_ids = item.recurring_activity_ids.ids
@api.model
def _get_recurring_fields(self):
return ["custom_activity_ids"] + super()._get_recurring_fields()
def call_create_recurring_tasks(self):
self.recurrence_id.create_recurring_tasks()
def _get_new_next_date_recurring_task(self):
date = self.recurrence_id.next_recurrence_date
delta = self.repeat_interval if self.repeat_unit == "day" else 1
dates = self.recurrence_id._get_next_recurring_dates(
date + timedelta(days=delta),
self.repeat_interval,
self.repeat_unit,
self.repeat_type,
self.repeat_until,
self.repeat_on_month,
self.repeat_on_year,
self._get_weekdays(WEEKS.get(self.repeat_week)),
self.repeat_day,
self.repeat_week,
self.repeat_month,
count=1,
)
return dates[0]
def _get_recurrence_start_date(self):
if self.env.user.has_group("base.group_no_one"):
return self.recurrence_id.next_recurrence_date or fields.Datetime.now()
return super()._get_recurrence_start_date()
@api.model
def _forming_activity_data(self, task, custom_activity_ids):
"""Returns prepared data for creating records"""
result = []
for item in custom_activity_ids:
result.append(
(
0,
0,
{
"project_task_id": task.id,
"activity_type_id": item.activity_type_id.id,
"user_id": item.user_id.id,
"summary": item.summary,
"description": item.description,
"days_after_task_creation_date": item.days_after_task_creation_date,
},
)
)
return result
@api.model_create_multi
def create(self, vals_list):
results = super().create(vals_list)
for item in results:
item.create_date = item.recurrence_id.next_recurrence_date
if item.recurring_task and item.custom_activity_ids:
item.message_subscribe(
partner_ids=list(
set(
item.custom_activity_ids.mapped("user_id")
.mapped("partner_id")
.ids
)
- set(item.message_follower_ids.mapped("partner_id").ids)
)
)
item.recurring_activity_ids = self._forming_activity_data(
item, item.custom_activity_ids
)
return results

View file

@ -0,0 +1,31 @@
from odoo import api, fields, models
class ProjectTaskRecurrence(models.Model):
_inherit = "project.task.recurrence"
old_date_recurring_task = fields.Date(
default=fields.Date.today(),
)
@api.model
def _get_recurring_fields(self):
return ["custom_activity_ids"] + super()._get_recurring_fields()
def create_recurring_tasks(self):
"""Create recurring tasks"""
if not self.env.user.has_group("project.group_project_recurring_tasks"):
return
self._create_next_task()
for recurrence in self.filtered(lambda r: r.repeat_type == "after"):
recurrence.recurrence_left -= 1
task = self.task_ids[-1]
new_date = task._get_new_next_date_recurring_task()
self.write(
{
"old_date_recurring_task": self.next_recurrence_date,
"next_recurrence_date": new_date,
}
)
task.write({"create_date": new_date})

View file

@ -0,0 +1,118 @@
from datetime import timedelta
from odoo import _, api, fields, models
from odoo.exceptions import UserError
class RecurringActivity(models.Model):
_name = "recurring.activity"
_description = "Recurring activity"
project_task_id = fields.Many2one("project.task")
activity_type_id = fields.Many2one(
"mail.activity.type", string="Activity Type", ondelete="restrict"
)
user_id = fields.Many2one(
"res.users",
string="Assigned to",
index=True,
required=True,
compute="_compute_on_activity_type_id",
store=True,
readonly=False,
)
summary = fields.Char(
compute="_compute_on_activity_type_id",
store=True,
readonly=False,
)
description = fields.Html(
sanitize_style=True,
compute="_compute_on_activity_type_id",
store=True,
readonly=False,
)
days_after_task_creation_date = fields.Integer()
next_recurrence_date = fields.Date(
string="next_date", compute="_compute_next_recurrence_date", store=True
)
@api.depends("days_after_task_creation_date")
def _compute_next_recurrence_date(self):
for record in self:
record.next_recurrence_date = record._get_next_date()
def _get_next_date(self):
return fields.Date.today() + timedelta(
days=self.days_after_task_creation_date + 1
if self.days_after_task_creation_date == 0
else self.days_after_task_creation_date
)
@api.model
def _cron_create_activities(self):
today = fields.Date.today()
recurring_activities = self.search([("next_recurrence_date", "<=", today)])
for activity in recurring_activities:
activity.project_task_id.activity_schedule(
activity_type_id=activity.activity_type_id.id,
user_id=activity.user_id.id,
note=activity.description,
summary=activity.summary,
date_deadline=fields.Date.today(),
)
activity.write({"next_recurrence_date": activity._get_next_date()})
@api.constrains("user_id")
def _check_user_id(self):
for record in self:
task = record.project_task_id
if not (
record.user_id.partner_id.id
in task.message_follower_ids.mapped("partner_id").ids
):
raise UserError(
_(
f"Assigned user {record.user_id.name} has no access"
" to the document and is not able to handle this activity."
)
)
@api.depends("activity_type_id")
def _compute_on_activity_type_id(self):
for activity in self:
if (not activity.description) or activity.description == "<p><br></p>":
activity.description = activity.activity_type_id.default_note
if not activity.summary:
activity.summary = activity.activity_type_id.summary
@api.model
def delta_time(self, old, new):
return (new - old).days
@api.model_create_multi
def create(self, vals_list):
results = super().create(vals_list)
for item in results:
next_recurrence_date = (
item.project_task_id.recurrence_id.next_recurrence_date
)
item.write({"create_date": next_recurrence_date})
task = item.project_task_id
old_date = item.project_task_id.recurrence_id.old_date_recurring_task
delfa = 0
if len(item.project_task_id.recurrence_id.task_ids) == 1:
old_date = fields.Date.today()
delfa = self.delta_time(old_date, next_recurrence_date)
task.activity_schedule(
activity_type_id=item.activity_type_id.id,
user_id=item.user_id.id,
note=item.description,
summary=item.summary,
date_deadline=fields.Date.today()
+ timedelta(days=item.days_after_task_creation_date)
if not self.env.user.has_group("base.group_no_one")
else (next_recurrence_date - timedelta(days=delfa))
+ timedelta(days=item.days_after_task_creation_date),
)
return results

View file

@ -0,0 +1 @@
#.Enable recurring task in config settings

View file

@ -0,0 +1,2 @@
* Cetmix <https://cetmix.com>
* Dessan Hemrayev <dessanhemrayev@gmail.com>

View file

@ -0,0 +1 @@
This module allow users to add activities to the recurring tasks and have them automatically duplicated within recurring tasks at set intervals.

View file

@ -0,0 +1,4 @@
Open recurring task.
#. In tab "Recurrence", user can set the activites that should be created each time the task is duplicated.
#. For testing purposes, a button "Create next recurring task" is visible in developer mode to check how next task will be created.

View file

@ -0,0 +1,2 @@
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
project_recurring_activity,recurring.activity,model_recurring_activity,,1,1,1,1
1 id name model_id:id group_id:id perm_read perm_write perm_create perm_unlink
2 project_recurring_activity recurring.activity model_recurring_activity 1 1 1 1

View file

@ -0,0 +1,442 @@
<!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>Project Task Recurring Activity</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" id="project-task-recurring-activity">
<h1 class="title">Project Task Recurring Activity</h1>
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! This file is generated by oca-gen-addon-readme !!
!! changes will be overwritten. !!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! source digest: sha256:e2aeb979b785a759d9358928d0bd8639c06511dc3a86447092ab50cc86709acd
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
<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/licence-AGPL--3-blue.png" /></a> <a class="reference external image-reference" href="https://github.com/OCA/project/tree/16.0/project_task_recurring_activity"><img alt="OCA/project" src="https://img.shields.io/badge/github-OCA%2Fproject-lightgray.png?logo=github" /></a> <a class="reference external image-reference" href="https://translation.odoo-community.org/projects/project-16-0/project-16-0-project_task_recurring_activity"><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/project&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 allow users to add activities to the recurring tasks and have them automatically duplicated within recurring tasks at set intervals.</p>
<p><strong>Table of contents</strong></p>
<div class="contents local topic" id="contents">
<ul class="simple">
<li><a class="reference internal" href="#configuration" id="toc-entry-1">Configuration</a></li>
<li><a class="reference internal" href="#usage" id="toc-entry-2">Usage</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="configuration">
<h1><a class="toc-backref" href="#toc-entry-1">Configuration</a></h1>
<p>#.Enable recurring task in config settings</p>
</div>
<div class="section" id="usage">
<h1><a class="toc-backref" href="#toc-entry-2">Usage</a></h1>
<p>Open recurring task.</p>
<ol class="arabic simple">
<li>In tab “Recurrence”, user can set the activites that should be created each time the task is duplicated.</li>
<li>For testing purposes, a button “Create next recurring task” is visible in developer mode to check how next task will be created.</li>
</ol>
</div>
<div class="section" id="bug-tracker">
<h1><a class="toc-backref" href="#toc-entry-3">Bug Tracker</a></h1>
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/project/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/project/issues/new?body=module:%20project_task_recurring_activity%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">
<h1><a class="toc-backref" href="#toc-entry-4">Credits</a></h1>
<div class="section" id="authors">
<h2><a class="toc-backref" href="#toc-entry-5">Authors</a></h2>
<ul class="simple">
<li>Cetmix</li>
</ul>
</div>
<div class="section" id="contributors">
<h2><a class="toc-backref" href="#toc-entry-6">Contributors</a></h2>
<blockquote>
<ul class="simple">
<li>Cetmix &lt;<a class="reference external" href="https://cetmix.com">https://cetmix.com</a>&gt;</li>
<li>Dessan Hemrayev &lt;<a class="reference external" href="mailto:dessanhemrayev&#64;gmail.com">dessanhemrayev&#64;gmail.com</a>&gt;</li>
</ul>
</blockquote>
</div>
<div class="section" id="maintainers">
<h2><a class="toc-backref" href="#toc-entry-7">Maintainers</a></h2>
<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">maintainers</a>:</p>
<p><a class="reference external image-reference" href="https://github.com/dessanhemrayev"><img alt="dessanhemrayev" src="https://github.com/dessanhemrayev.png?size=40px" /></a> <a class="reference external image-reference" href="https://github.com/CetmixGitDrone"><img alt="CetmixGitDrone" src="https://github.com/CetmixGitDrone.png?size=40px" /></a></p>
<p>This module is part of the <a class="reference external" href="https://github.com/OCA/project/tree/16.0/project_task_recurring_activity">OCA/project</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>
</body>
</html>

View file

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

View file

@ -0,0 +1,319 @@
from datetime import date, datetime, timedelta
from freezegun import freeze_time
from odoo import _, fields
from odoo.exceptions import UserError
from odoo.tests.common import Form, TransactionCase
class TestProjectrecurrence(TransactionCase):
@classmethod
def setUpClass(cls):
super(TestProjectrecurrence, cls).setUpClass()
cls.env.user.groups_id += cls.env.ref("project.group_project_recurring_tasks")
cls.recurring_activity = cls.env["recurring.activity"]
cls.stage_a = cls.env["project.task.type"].create({"name": "a"})
cls.stage_b = cls.env["project.task.type"].create({"name": "b"})
cls.demo_user = (
cls.env["res.users"]
.with_context(no_reset_password=True)
.create(
{
"name": "demo",
"login": "demo_user",
"email": "dess@yourcompany.com",
"groups_id": [
(
6,
0,
[cls.env.ref("project.group_project_recurring_tasks").id],
)
],
}
)
)
cls.demo_user2 = (
cls.env["res.users"]
.with_context(no_reset_password=True)
.create(
{
"name": "demo2",
"login": "demo_user2",
"email": "dess2@yourcompany.com",
"groups_id": [
(
6,
0,
[cls.env.ref("project.group_project_recurring_tasks").id],
)
],
}
)
)
cls.mail_activity_a = cls.env["mail.activity.type"].create(
{
"name": "activity_a",
"default_user_id": cls.demo_user.id,
"summary": "summary",
}
)
cls.mail_activity_b = cls.env["mail.activity.type"].create(
{
"name": "activity_b",
"default_user_id": cls.demo_user.id,
}
)
cls.project_recurring = (
cls.env["project.project"]
.with_context(mail_create_nolog=True)
.create(
{
"name": "Recurring",
"allow_recurring_tasks": True,
"type_ids": [
(4, cls.stage_a.id),
(4, cls.stage_b.id),
],
}
)
)
cls.project_recurring2 = (
cls.env["project.project"]
.with_context(mail_create_nolog=True)
.create(
{
"name": "Recurring",
"allow_recurring_tasks": True,
"type_ids": [
(4, cls.stage_a.id),
(4, cls.stage_b.id),
],
}
)
)
def test_check_activity_fields(self):
with freeze_time("2020-01-01"):
form = Form(self.env["project.task"])
form.name = "test recurring task"
form.description = "my super recurring task"
form.project_id = self.project_recurring
form.date_deadline = datetime(2020, 2, 1)
form.recurring_task = True
form.repeat_interval = 1
form.repeat_unit = "month"
form.repeat_type = "forever"
form.repeat_on_month = "date"
form.repeat_day = "15"
task = form.save()
with self.assertRaisesRegex(
UserError,
(
_(
f"Assigned user {self.demo_user2.name} has no access"
" to the document and is not able to handle this activity."
)
),
):
self.recurring_activity.create(
{
"project_task_id": task.id,
"activity_type_id": self.mail_activity_a.id,
"user_id": self.demo_user2.id,
"days_after_task_creation_date": 1,
},
)
def test_recurrence_cron_repeat_forever(self):
domain = [("project_id", "=", self.project_recurring.id)]
with freeze_time("2020-01-01"):
form = Form(self.env["project.task"])
form.name = "test recurring task"
form.description = "my super recurring task"
form.project_id = self.project_recurring
form.date_deadline = datetime(2020, 2, 1)
form.recurring_task = True
form.repeat_interval = 1
form.repeat_unit = "month"
form.repeat_type = "forever"
form.repeat_on_month = "date"
form.repeat_day = "15"
task = form.save()
task.planned_hours = 2
self.assertEqual(len(task.recurring_activity_ids), 0, "Must be equal to 0")
self.assertEqual(len(task.activity_ids), 0, "Must be equal to 0")
task.message_subscribe(partner_ids=[self.demo_user.partner_id.id])
task.write(
{
"recurring_activity_ids": [
(
0,
0,
{
"activity_type_id": self.mail_activity_a.id,
"days_after_task_creation_date": 0,
"user_id": self.demo_user.id,
"summary": "summary",
"description": "description",
},
)
]
}
)
self.assertEqual(len(task.activity_ids), 1, "Must be equal to 1")
self.assertEqual(
task.activity_ids.summary, "summary", "Must be equal to 'summary'"
)
self.assertEqual(task.recurrence_id.next_recurrence_date, date(2020, 1, 15))
self.assertEqual(
self.env["project.task"].search_count(domain), 1, "Must be equal to 1"
)
self.env["project.task.recurrence"]._cron_create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain),
1,
"no extra task should be created",
)
with freeze_time("2020-01-15"):
self.assertEqual(
self.env["project.task"].search_count(domain), 1, "Must be equal to 1"
)
self.env["project.task.recurrence"]._cron_create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain), 2, "Must be equal to 2"
)
with freeze_time("2020-02-15"):
self.env["project.task.recurrence"]._cron_create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain), 3, "Must be equal to 3"
)
with freeze_time("2020-02-16"):
self.env["project.task.recurrence"]._cron_create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain), 3, "Must be equal to 3"
)
with freeze_time("2020-02-17"):
self.env["project.task.recurrence"]._cron_create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain), 3, "Must be equal to 3"
)
with freeze_time("2020-03-15"):
self.env["project.task.recurrence"]._cron_create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain), 4, "Must be equal to 4"
)
tasks = self.env["project.task"].search(domain)
self.assertEqual(len(tasks), 4, "Must be equal to 4")
self.assertEqual(len(tasks.mapped("activity_ids")), 4, "Must be equal to 4")
self.assertEqual(
len(tasks.mapped("recurring_activity_ids")), 4, "Must be equal to 4"
)
project_task = tasks[0]
activity = self.recurring_activity.search(
[("project_task_id", "=", project_task.id)]
)
self.assertEqual(activity.user_id, self.mail_activity_a.default_user_id)
self.assertEqual(activity.summary, project_task.activity_summary)
activity.write(
{
"description": "<p><br></p>",
"summary": None,
"activity_type_id": self.mail_activity_b.id,
}
)
activity._compute_on_activity_type_id()
self.assertEqual(activity.user_id, self.mail_activity_a.default_user_id)
self.assertEqual(activity.description, self.mail_activity_b.default_note)
self.assertEqual(activity.summary, self.mail_activity_b.summary)
def test_create_recurring_tasks(self):
"""Check custom method dev"""
domain = [("project_id", "=", self.project_recurring2.id)]
with freeze_time("2020-01-01"):
form = Form(self.env["project.task"])
form.name = "test recurring task"
form.description = "my super recurring task"
form.project_id = self.project_recurring2
form.date_deadline = datetime(2020, 2, 1)
form.recurring_task = True
form.repeat_interval = 1
form.repeat_unit = "month"
form.repeat_type = "forever"
form.repeat_on_month = "date"
form.repeat_day = "15"
task = form.save()
task.planned_hours = 2
task.message_subscribe(partner_ids=[self.demo_user.partner_id.id])
task.write(
{
"recurring_activity_ids": [
(
0,
0,
{
"activity_type_id": self.mail_activity_a.id,
"days_after_task_creation_date": 0,
"user_id": self.demo_user.id,
"summary": "summary",
"description": "description",
},
)
]
}
)
with freeze_time("2020-01-15"):
self.assertEqual(
self.env["project.task"].search_count(domain), 1, "Must be equal to 1"
)
task.recurrence_id.create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain), 2, "Must be equal to 2"
)
self.assertEqual(
task.recurring_activity_ids[0]._get_next_date(),
fields.Date.today() + timedelta(days=1),
"Must be equal to `2020-01-16`",
)
self.env["recurring.activity"]._cron_create_activities()
with freeze_time("2020-01-16"):
activity = self.env["recurring.activity"].search(
[("project_task_id", "=", task.id)]
)
self.assertEqual(
len(activity),
1,
"Must be equal to 1",
)
self.assertEqual(
activity.next_recurrence_date,
fields.Date.today(),
"Must be equal to `2020-01-16`",
)
with freeze_time("2020-02-15"):
task.recurrence_id.create_recurring_tasks()
self.assertEqual(
self.env["project.task"].search_count(domain), 3, "Must be equal to 3"
)
today = fields.Date.today()
self.assertEqual(
self.env["recurring.activity"].delta_time(
today, today + timedelta(days=1)
),
1,
"Must be equal to 1",
)

View file

@ -0,0 +1,35 @@
<odoo>
<record model="ir.ui.view" id="view_recurring_activity_form">
<field name="name">project_task_recurring_activity_form</field>
<field name="model">project.task</field>
<field name="inherit_id" ref="project.view_task_form2" />
<field name="arch" type="xml">
<xpath expr="//header" position="inside">
<div groups="project.group_project_recurring_tasks">
<button
string="Create Next Recurring Task"
class="oe_highlight"
groups="base.group_no_one"
name="call_create_recurring_tasks"
type="object"
attrs="{'invisible': [('recurring_task','=',False)] }"
/>
</div>
</xpath>
<xpath expr="//notebook/page[@name='recurrence']" position="inside">
<group name="activities" string="Activities" />
<field name="recurring_activity_ids">
<tree>
<field name="user_id" />
<field name="project_task_id" invisible='1' />
<field name="activity_type_id" />
<field name="summary" />
<field name="description" />
<field name="days_after_task_creation_date" />
</tree>
</field>
</xpath>
</field>
</record>
</odoo>

View file

@ -0,0 +1,27 @@
<odoo>
<record id="recurring_activity_view_form" model="ir.ui.view">
<field name="name">recurring_activity_view_form</field>
<field name="model">recurring.activity</field>
<field name="arch" type="xml">
<form string="">
<sheet>
<group name="recurring_activity">
<group name="main">
<field name="user_id" readonly="0" />
<field name="project_task_id" invisible='1' />
</group>
<group name="extra">
<field name="activity_type_id" required="1" />
<field name="days_after_task_creation_date" />
</group>
</group>
<group name="summary">
<field name="summary" readonly="0" />
</group>
<field name="description" readonly="0" />
</sheet>
</form>
</field>
</record>
</odoo>

View file

@ -0,0 +1,42 @@
[project]
name = "odoo-bringout-oca-project-project_task_recurring_activity"
version = "16.0.0"
description = "Project Task Recurring Activity - Project Task Recurring Activity"
authors = [
{ name = "Ernad Husremovic", email = "hernad@bring.out.ba" }
]
dependencies = [
"odoo-bringout-oca-ocb-project>=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 = ["project_task_recurring_activity"]
[tool.rye]
managed = true
dev-dependencies = [
"pytest>=8.4.1",
]