diff --git a/project_forecast_line_priority/README.rst b/project_forecast_line_priority/README.rst new file mode 100644 index 0000000000..6457cf5c9d --- /dev/null +++ b/project_forecast_line_priority/README.rst @@ -0,0 +1,90 @@ +============================== +Project Forecast Line Priority +============================== + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:8dba9357cfaa876178cfa0b1d8a6723a256afe60f3110ecfa9a1eb05d033cd9a + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |badge1| image:: https://img.shields.io/badge/maturity-Alpha-red.png + :target: https://odoo-community.org/page/development-status + :alt: Alpha +.. |badge2| image:: https://img.shields.io/badge/licence-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/14.0/project_forecast_line_priority + :alt: OCA/project +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/project-14-0/project-14-0-project_forecast_line_priority + :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=14.0 + :alt: Try me on Runboat + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module recomputes forecast line end dates based on the priority of connected task(s) + +.. IMPORTANT:: + This is an alpha version, the data model and design can change at any time without warning. + Only for development or testing purpose, do not use in production. + `More details on development status `_ + +**Table of contents** + +.. contents:: + :local: + +Bug Tracker +=========== + +Bugs are tracked on `GitHub Issues `_. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us to smash it by providing a detailed and welcomed +`feedback `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +~~~~~~~ + +* Therp BV + +Contributors +~~~~~~~~~~~~ + +* Nikos Tsirintanis +* Gijs-Jan Otten + +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-ntsirintanis| image:: https://github.com/ntsirintanis.png?size=40px + :target: https://github.com/ntsirintanis + :alt: ntsirintanis + +Current `maintainer `__: + +|maintainer-ntsirintanis| + +This module is part of the `OCA/project `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/project_forecast_line_priority/__init__.py b/project_forecast_line_priority/__init__.py new file mode 100644 index 0000000000..0650744f6b --- /dev/null +++ b/project_forecast_line_priority/__init__.py @@ -0,0 +1 @@ +from . import models diff --git a/project_forecast_line_priority/__manifest__.py b/project_forecast_line_priority/__manifest__.py new file mode 100644 index 0000000000..0dbf27403d --- /dev/null +++ b/project_forecast_line_priority/__manifest__.py @@ -0,0 +1,19 @@ +# Copyright 2024 Therp BV +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). +{ + "name": "Project Forecast Line Priority", + "summary": "Project Forecast Line dates according to task priority", + "version": "14.0.1.0.0", + "author": "Therp BV, Odoo Community Association (OCA)", + "maintainers": ["ntsirintanis"], + "license": "AGPL-3", + "category": "Project", + "website": "https://github.com/OCA/project", + "depends": ["project_forecast_line_deadline", "project_task_add_very_high"], + "data": [ + "data/ir_actions_server.xml", + "views/res_config_settings.xml", + ], + "installable": True, + "development_status": "Alpha", +} diff --git a/project_forecast_line_priority/data/ir_actions_server.xml b/project_forecast_line_priority/data/ir_actions_server.xml new file mode 100644 index 0000000000..11cf65e053 --- /dev/null +++ b/project_forecast_line_priority/data/ir_actions_server.xml @@ -0,0 +1,13 @@ + + + + Project Task: Update forecast date end + + + action + code + + model._action_update_forecast_date_end(records) + + + diff --git a/project_forecast_line_priority/models/__init__.py b/project_forecast_line_priority/models/__init__.py new file mode 100644 index 0000000000..0ce264d79d --- /dev/null +++ b/project_forecast_line_priority/models/__init__.py @@ -0,0 +1,3 @@ +from . import project_task +from . import res_company +from . import res_config_settings diff --git a/project_forecast_line_priority/models/project_task.py b/project_forecast_line_priority/models/project_task.py new file mode 100644 index 0000000000..167b46ed40 --- /dev/null +++ b/project_forecast_line_priority/models/project_task.py @@ -0,0 +1,63 @@ +# Copyright 2024 Therp BV +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). +from datetime import timedelta + +from odoo import api, fields, models + + +class ProjectTask(models.Model): + _inherit = "project.task" + + def _forecast_date_planned_end_depends_list(self): + """Returns a list of fields to trigger recomputation""" + return super()._forecast_date_planned_end_depends_list() + ["priority"] + + @api.depends(_forecast_date_planned_end_depends_list) + def _compute_forecast_date_planned_end(self): + """Override method to use recompute based on priority""" + res = super()._compute_forecast_date_planned_end() + for task in self: + task.forecast_date_planned_end = ( + task._get_forecast_date_planned() or task.forecast_date_planned_end + ) + return res + + def _update_forecast_lines(self): + """Override cron method and inject forecast date recomputation""" + for task in self: + forecast_date_planned_end = task._get_forecast_date_planned() + if not forecast_date_planned_end: + continue + task.forecast_date_planned_end = forecast_date_planned_end + return super()._update_forecast_lines() + + @api.model + def _action_update_forecast_date_end(self, tasks): + for task in tasks: + new_forecast_date_planned_end = task._get_forecast_date_planned() + if not new_forecast_date_planned_end: + continue + task.write( + { + "forecast_date_planned_end": new_forecast_date_planned_end, + } + ) + + def _get_forecast_date_planned(self, priority=None): + """Update forecast date end based on priority""" + self.ensure_one() + if self.date_deadline: + return False + priority = priority or self.priority + if not priority: + # This may happen when a portal user + # is creating a task on portal + priority = "0" + selection = self.company_id["forecast_line_priority_%s_selection" % priority] + if selection == "delta": + return fields.Date.today() + timedelta( + days=int(self.company_id["forecast_line_priority_%s_delta" % priority]) + ) + if selection == "date": + return self.company_id["forecast_line_priority_%s_date" % priority] + return False diff --git a/project_forecast_line_priority/models/res_company.py b/project_forecast_line_priority/models/res_company.py new file mode 100644 index 0000000000..1e9ea186e9 --- /dev/null +++ b/project_forecast_line_priority/models/res_company.py @@ -0,0 +1,38 @@ +# Copyright 2024 Therp BV +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). +from odoo import fields, models + +SELECTION_FORECAST_ENDDATE = [ + ("none", "None"), + ("date", "Date"), + ("delta", "Delta (in days)"), +] + + +class ResCompany(models.Model): + _inherit = "res.company" + + forecast_line_priority_0_date = fields.Date() + forecast_line_priority_1_date = fields.Date() + forecast_line_priority_2_date = fields.Date() + forecast_line_priority_3_date = fields.Date() + forecast_line_priority_0_delta = fields.Integer() + forecast_line_priority_1_delta = fields.Integer() + forecast_line_priority_2_delta = fields.Integer() + forecast_line_priority_3_delta = fields.Integer() + forecast_line_priority_0_selection = fields.Selection( + SELECTION_FORECAST_ENDDATE, + default="none", + ) + forecast_line_priority_1_selection = fields.Selection( + SELECTION_FORECAST_ENDDATE, + default="none", + ) + forecast_line_priority_2_selection = fields.Selection( + SELECTION_FORECAST_ENDDATE, + default="none", + ) + forecast_line_priority_3_selection = fields.Selection( + SELECTION_FORECAST_ENDDATE, + default="none", + ) diff --git a/project_forecast_line_priority/models/res_config_settings.py b/project_forecast_line_priority/models/res_config_settings.py new file mode 100644 index 0000000000..5b1c327c67 --- /dev/null +++ b/project_forecast_line_priority/models/res_config_settings.py @@ -0,0 +1,53 @@ +# Copyright 2024 Therp BV +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from odoo import fields, models + + +class ResConfigSettings(models.TransientModel): + _inherit = "res.config.settings" + + forecast_line_priority_0_date = fields.Date( + related="company_id.forecast_line_priority_0_date", readonly=False + ) + forecast_line_priority_1_date = fields.Date( + related="company_id.forecast_line_priority_1_date", readonly=False + ) + forecast_line_priority_2_date = fields.Date( + related="company_id.forecast_line_priority_2_date", readonly=False + ) + forecast_line_priority_3_date = fields.Date( + related="company_id.forecast_line_priority_3_date", readonly=False + ) + forecast_line_priority_0_delta = fields.Integer( + related="company_id.forecast_line_priority_0_delta", readonly=False + ) + forecast_line_priority_1_delta = fields.Integer( + related="company_id.forecast_line_priority_1_delta", readonly=False + ) + forecast_line_priority_2_delta = fields.Integer( + related="company_id.forecast_line_priority_2_delta", readonly=False + ) + forecast_line_priority_3_delta = fields.Integer( + related="company_id.forecast_line_priority_3_delta", readonly=False + ) + forecast_line_priority_0_selection = fields.Selection( + related="company_id.forecast_line_priority_0_selection", + readonly=False, + required=True, + ) + forecast_line_priority_1_selection = fields.Selection( + related="company_id.forecast_line_priority_1_selection", + readonly=False, + required=True, + ) + forecast_line_priority_2_selection = fields.Selection( + related="company_id.forecast_line_priority_2_selection", + readonly=False, + required=True, + ) + forecast_line_priority_3_selection = fields.Selection( + related="company_id.forecast_line_priority_3_selection", + readonly=False, + required=True, + ) diff --git a/project_forecast_line_priority/readme/CONTRIBUTORS.rst b/project_forecast_line_priority/readme/CONTRIBUTORS.rst new file mode 100644 index 0000000000..0d46453290 --- /dev/null +++ b/project_forecast_line_priority/readme/CONTRIBUTORS.rst @@ -0,0 +1,2 @@ +* Nikos Tsirintanis +* Gijs-Jan Otten diff --git a/project_forecast_line_priority/readme/DESCRIPTION.rst b/project_forecast_line_priority/readme/DESCRIPTION.rst new file mode 100644 index 0000000000..92abc338ba --- /dev/null +++ b/project_forecast_line_priority/readme/DESCRIPTION.rst @@ -0,0 +1 @@ +This module recomputes forecast line end dates based on the priority of connected task(s) diff --git a/project_forecast_line_priority/static/description/icon.png b/project_forecast_line_priority/static/description/icon.png new file mode 100644 index 0000000000..3a0328b516 Binary files /dev/null and b/project_forecast_line_priority/static/description/icon.png differ diff --git a/project_forecast_line_priority/static/description/index.html b/project_forecast_line_priority/static/description/index.html new file mode 100644 index 0000000000..2e921b0f53 --- /dev/null +++ b/project_forecast_line_priority/static/description/index.html @@ -0,0 +1,430 @@ + + + + + + +Project Forecast Line Priority + + + +
+

Project Forecast Line Priority

+ + +

Alpha License: AGPL-3 OCA/project Translate me on Weblate Try me on Runboat

+

This module recomputes forecast line end dates based on the priority of connected task(s)

+
+

Important

+

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

+
+

Table of contents

+ +
+

Bug Tracker

+

Bugs are tracked on GitHub Issues. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us to smash it by providing a detailed and welcomed +feedback.

+

Do not contact contributors directly about support or help with technical issues.

+
+
+

Credits

+
+

Authors

+
    +
  • Therp BV
  • +
+
+
+

Contributors

+ +
+
+

Maintainers

+

This module is maintained by the OCA.

+Odoo Community Association +

OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use.

+

Current maintainer:

+

ntsirintanis

+

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

+

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

+
+
+
+ + diff --git a/project_forecast_line_priority/tests/__init__.py b/project_forecast_line_priority/tests/__init__.py new file mode 100644 index 0000000000..5d54e7e98a --- /dev/null +++ b/project_forecast_line_priority/tests/__init__.py @@ -0,0 +1 @@ +from . import test_forecast_line_priority diff --git a/project_forecast_line_priority/tests/test_forecast_line_priority.py b/project_forecast_line_priority/tests/test_forecast_line_priority.py new file mode 100644 index 0000000000..6534f9a193 --- /dev/null +++ b/project_forecast_line_priority/tests/test_forecast_line_priority.py @@ -0,0 +1,93 @@ +# Copyright 2024 Therp BV +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). +from freezegun import freeze_time + +import odoo.tests.common as common +from odoo import fields + + +class TestForecastLinePriotity(common.TransactionCase): + def setUp(self): + super().setUp() + self.project_task = self.env.ref("project.project_task_1") + self.company_1 = self.env.company + self.company_2 = self.env["res.company"].create({"name": "Another Company"}) + self.company_1.write( + { + "forecast_line_priority_0_selection": "date", + "forecast_line_priority_0_date": "2026-01-01", + "forecast_line_priority_2_selection": "date", + "forecast_line_priority_2_date": "2024-02-01", + "forecast_line_priority_3_selection": "delta", + "forecast_line_priority_3_delta": "15", + } + ) + self.company_2.write( + { + "forecast_line_priority_2_selection": "delta", + "forecast_line_priority_2_delta": "7", + "forecast_line_priority_3_selection": "date", + "forecast_line_priority_3_date": "2022-01-01", + } + ) + + @freeze_time("2024-01-01") + def test_forecast_line_priority(self): + """Test forecast_date_planned_end vs task priority""" + # See that relevant date fields are falsy + task = self.project_task + self.assertFalse(task.forecast_date_planned_end) + self.assertFalse(task.date_deadline) + self.assertEqual(task.priority, "0") + task.priority = "1" + # no shift in date due to priority + self.assertFalse(task.forecast_date_planned_end) + task.priority = "2" + # fixed date + self.assertEqual( + fields.Date.to_string(task.forecast_date_planned_end), "2024-02-01" + ) + task.priority = "3" + # +15 days + self.assertEqual( + fields.Date.to_string(task.forecast_date_planned_end), "2024-01-16" + ) + # set deadline to task + task.date_deadline = "2025-01-01" + self.assertEqual(task.forecast_date_planned_end, task.date_deadline) + # change priority, but nothing changes in forecast end date + task.priority = "2" + self.assertEqual(task.forecast_date_planned_end, task.date_deadline) + # reset date_deadline for original task + task.date_deadline = False + # new task, but for a different company + other_task = task.create( + { + "project_id": self.env["project.project"] + .create({"company_id": self.company_2.id, "name": "other project"}) + .id, + "priority": "2", + "name": "other task", + } + ) + # launch the server action + self.env["project.task"]._action_update_forecast_date_end(task + other_task) + # task is for company_1, and has priority 2, + # config selection is date, with value "2024-02-01" + self.assertEqual( + fields.Date.to_string(task.forecast_date_planned_end), "2024-02-01" + ) + # other_task is for company_2, and has priority 2, + # config selection is delta, 7 days, expected value "2024-01-08" + self.assertEqual( + fields.Date.to_string(other_task.forecast_date_planned_end), "2024-01-08" + ) + # set priority for task to False. + # (possible for portal users) + task.priority = False + # system sees priority False as 0, i.e, normal priority + self.env["project.task"]._action_update_forecast_date_end(task) + # config selection is date, with value "2026-01-01" + self.assertEqual( + fields.Date.to_string(task.forecast_date_planned_end), "2026-01-01" + ) diff --git a/project_forecast_line_priority/views/res_config_settings.xml b/project_forecast_line_priority/views/res_config_settings.xml new file mode 100644 index 0000000000..629059bddb --- /dev/null +++ b/project_forecast_line_priority/views/res_config_settings.xml @@ -0,0 +1,140 @@ + + + + res.config.settings.view.form.inherit.forecast.priority + res.config.settings + + + +
+

Forecast Task Priority Management

+
+
+
+ Use the below selection fields to set per-priority values for forecast planned end date +
+
+
+ +
+
+ + + +
+
+
+
+ +
+
+ + + +
+
+
+
+ + +
+
+ + + +
+
+
+
+ + + +
+
+ + + +
+
+
+
+
+
+
+
+
diff --git a/setup/project_forecast_line_priority/odoo/addons/project_forecast_line_priority b/setup/project_forecast_line_priority/odoo/addons/project_forecast_line_priority new file mode 120000 index 0000000000..c0300c2a1d --- /dev/null +++ b/setup/project_forecast_line_priority/odoo/addons/project_forecast_line_priority @@ -0,0 +1 @@ +../../../../project_forecast_line_priority \ No newline at end of file diff --git a/setup/project_forecast_line_priority/setup.py b/setup/project_forecast_line_priority/setup.py new file mode 100644 index 0000000000..28c57bb640 --- /dev/null +++ b/setup/project_forecast_line_priority/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +)