diff --git a/hr_employee_stats_sheet/.gitignore b/hr_employee_stats_sheet/.gitignore new file mode 100644 index 0000000..df8a896 --- /dev/null +++ b/hr_employee_stats_sheet/.gitignore @@ -0,0 +1,2 @@ +*.*~ +*pyc \ No newline at end of file diff --git a/hr_employee_stats_sheet/README.rst b/hr_employee_stats_sheet/README.rst new file mode 100644 index 0000000..c4b2416 --- /dev/null +++ b/hr_employee_stats_sheet/README.rst @@ -0,0 +1,55 @@ +======================= +hr_employee_stats_sheet +======================= + +Summary +======= + +Add global sheet for employee stats + +Description +=========== + +This module add a global sheet stats in timesheet sheet +It compare planified working hours (from calendar ressouce) and worked hours form timesheets + +Installation +============ + +Use Odoo normal module installation procedure to install +``hr_employee_stats_sheet``. + +Known issues / Roadmap +====================== + +None yet. + +Bug Tracker +=========== + +Bugs are tracked on `our issues website `_. In case of +trouble, please check there if your issue has already been +reported. If you spotted it first, help us smashing it by providing a +detailed and welcomed feedback. + +Credits +======= + +Contributors +------------ + +* `Alusage : Nicolas JEUDY` +* `Elabore ` + +Funders +------- + +The development of this module has been financially supported by: +* Alusage (https://github.com/Alusage) +* Elabore (https://elabore.coop) + + +Maintainer +---------- + +This module is maintained by Elabore. \ No newline at end of file diff --git a/hr_employee_stats_sheet/__init__.py b/hr_employee_stats_sheet/__init__.py new file mode 100755 index 0000000..0650744 --- /dev/null +++ b/hr_employee_stats_sheet/__init__.py @@ -0,0 +1 @@ +from . import models diff --git a/hr_employee_stats_sheet/__manifest__.py b/hr_employee_stats_sheet/__manifest__.py new file mode 100755 index 0000000..3d4aaf5 --- /dev/null +++ b/hr_employee_stats_sheet/__manifest__.py @@ -0,0 +1,27 @@ +{ + "name": "hr_employee_stats_sheet", + "version": "16.0.2.0.0", + "description": "Add global sheet for employee stats", + "summary": "Add global sheet for employee stats", + "author": "Nicolas JEUDY", + "website": "https://github.com/Alusage/odoo-hr-addons", + "license": "LGPL-3", + "category": "Human Resources", + "depends": [ + "allow_negative_leave_and_allocation", + "base", + "hr", + "hr_holidays", + "hr_timesheet", + "hr_timesheet_sheet", + "resource", + ], + "data": [ + "security/ir.model.access.csv", + "views/hr_employee_stats.xml", + "views/hr_timesheet_sheet.xml", + "views/res_config_settings_views.xml", + ], + "installable": True, + "application": False, +} diff --git a/hr_employee_stats_sheet/i18n/fr.po b/hr_employee_stats_sheet/i18n/fr.po new file mode 100644 index 0000000..609a751 --- /dev/null +++ b/hr_employee_stats_sheet/i18n/fr.po @@ -0,0 +1,446 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * hr_employee_stats_sheet +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 16.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2025-06-19 14:31+0000\n" +"PO-Revision-Date: 2025-06-19 14:31+0000\n" +"Last-Translator: \n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.res_config_settings_view_form +msgid "" +"" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_needaction +msgid "Action Needed" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_ids +msgid "Activities" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_exception_decoration +msgid "Activity Exception Decoration" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_state +msgid "Activity State" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_type_icon +msgid "Activity Type Icon" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_attachment_count +msgid "Attachment Count" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_res_company__auto_validate_recovery_allocation +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_res_config_settings__auto_validate_recovery_allocation +msgid "Auto validate recovery allocation" +msgstr "Auto-validation des allocations de récupération" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.res_config_settings_view_form +msgid "Choose the coef to apply to recovery hours." +msgstr "Indiquer le coefficient multiplicateur à appliquer sur les heures supplémentaires (en %) pour générer les heures de récupération. Par défaut : 25,00%" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.res_config_settings_view_form +msgid "Choose the recovery type." +msgstr "" +"Sélectionner le type de congé utilisé pour générer des allocations de " +"récupération. Le type de congé doit accepter les allocations négatives et être pris à l'heure." + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.res_config_settings_view_form +msgid "Choose to auto-validate the recovery allocation or not" +msgstr "Validation automatique des allocations de récupération" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_res_company__coef +msgid "Coef" +msgstr "Coefficient" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_res_config_settings__coef +msgid "Coef (in %)" +msgstr "Coefficient (en %)" + +#. module: hr_employee_stats_sheet +#: model:ir.model,name:hr_employee_stats_sheet.model_res_company +msgid "Companies" +msgstr "Sociétés" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__company_id +msgid "Company" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model,name:hr_employee_stats_sheet.model_res_config_settings +msgid "Config Settings" +msgstr "Paramètres de configuration" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.view_hr_timesheet_sheet_form_inherit +msgid "Create recovery allocation" +msgstr "Générer une récupération" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__create_uid +msgid "Created by" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__create_date +msgid "Created on" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__date +msgid "Date" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__dayofweek +msgid "Day of Week" +msgstr "Jour de la semaine" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__department_id +msgid "Department" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__display_name +msgid "Display Name" +msgstr "" + +#. module: hr_employee_stats_sheet +#. odoo-python +#: code:addons/hr_employee_stats_sheet/models/hr_timesheet_sheet.py:0 +#, python-format +msgid "" +"Employe not defined for the timesheet sheet or recovery type not defined in " +"settings" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__employee_id +msgid "Employee" +msgstr "Employé·e" + +#. module: hr_employee_stats_sheet +#: model:ir.model,name:hr_employee_stats_sheet.model_hr_employee_stats +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_timesheet_sheet__employee_stats_ids +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.view_hr_timesheet_sheet_form_inherit +msgid "Employee Stats" +msgstr "Statistique des employé·e·s" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.hr_timesheet_sheet_stats_overview_pivot_view +msgid "Employee time stats pivot" +msgstr "Statistique de l'employé·e" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__failed_message_ids +msgid "Failed Messages" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_follower_ids +msgid "Followers" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_partner_ids +msgid "Followers (Partners)" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__activity_type_icon +msgid "Font awesome icon e.g. fa-tasks" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__gap_hours +msgid "Gap Hours" +msgstr "Heures d'écart" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.view_hr_timesheet_sheet_form_inherit +msgid "Generated recovery allocations" +msgstr "Allocation(s) de récupération générée(s)" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__has_message +msgid "Has Message" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__id +msgid "ID" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_exception_icon +msgid "Icon" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__activity_exception_icon +msgid "Icon to indicate an exception activity." +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__message_needaction +msgid "If checked, new messages require your attention." +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__message_has_error +msgid "If checked, some messages have a delivery error." +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_is_follower +msgid "Is Follower" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats____last_update +msgid "Last Modified on" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__write_uid +msgid "Last Updated by" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__write_date +msgid "Last Updated on" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_res_company__recovery_type_id +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_res_config_settings__recovery_type_id +msgid "Leave recovery type" +msgstr "Type de congé pour les récupérations" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_main_attachment_id +msgid "Main Attachment" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_has_error +msgid "Message Delivery error" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_ids +msgid "Messages" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__my_activity_date_deadline +msgid "My Activity Deadline" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__name +msgid "Name" +msgstr "Nom" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_calendar_event_id +msgid "Next Activity Calendar Event" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_date_deadline +msgid "Next Activity Deadline" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_summary +msgid "Next Activity Summary" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_type_id +msgid "Next Activity Type" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_needaction_counter +msgid "Number of Actions" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__message_has_error_counter +msgid "Number of errors" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__message_needaction_counter +msgid "Number of messages requiring action" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__message_has_error_counter +msgid "Number of messages with delivery error" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__is_public_holiday +msgid "Public Holiday" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_timesheet_sheet__recovery_allocation_ids +msgid "Recovery Allocations" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__activity_user_id +msgid "Responsible User" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__activity_state +msgid "" +"Status based on activities\n" +"Overdue: Due date is already passed\n" +"Today: Activity date is today\n" +"Planned: Future activities." +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_res_config_settings__coef +msgid "" +"The coef is applied to recovery hoursExample : an employe make 1h overtime, " +"if coef is set to 25%, the recovery hours allocated will be 1,25h" +msgstr "" + +#. module: hr_employee_stats_sheet +#. odoo-python +#: code:addons/hr_employee_stats_sheet/models/hr_timesheet_sheet.py:0 +#, python-format +msgid "" +"The number of recovery hours exceeds the authorized limit (%s h) by %s hours" +msgstr "" + +#. module: hr_employee_stats_sheet +#. odoo-python +#: code:addons/hr_employee_stats_sheet/models/hr_timesheet_sheet.py:0 +#, python-format +msgid "" +"There is a contract starting during the timesheet sheet time period for the " +"employee %sPlease create a new timesheet sheet starting from the new " +"contract start date" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model,name:hr_employee_stats_sheet.model_hr_leave_allocation +msgid "Time Off Allocation" +msgstr "Allocation de congés" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__sheet_id +msgid "Timesheet" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model,name:hr_employee_stats_sheet.model_hr_timesheet_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_leave_allocation__timesheet_sheet_id +msgid "Timesheet Sheet" +msgstr "Feuille de temps" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__timesheet_line_ids +msgid "Timesheet lines" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_timesheet_sheet__timesheet_sheet_gap_hours +msgid "Timesheet sheet gap hours" +msgstr "Heures d'écart sur la période de la feuille de temps" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_timesheet_sheet__timesheet_sheet_recovery_hours +msgid "Timesheet sheet recovery hours" +msgstr "Heures de récupération sur la période de la feuille de temps" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.view_hr_employee_stats_tree +msgid "Total" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__total_hours +msgid "Total Hours" +msgstr "Heures comptabilisées" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__total_leave_hours +msgid "Total Leave Hours" +msgstr "Heures de congés" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__total_planned_hours +msgid "Total Planning Hours" +msgstr "Heures planifiées" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__total_recovery_hours +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_timesheet_sheet__total_recovery_hours +msgid "Total Recovery Hours" +msgstr "Heures de récupération" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.view_hr_timesheet_sheet_form_inherit +msgid "Total gap hours in this timesheet sheet time range" +msgstr "Total heures d'écart sur cette période de temps" + +#. module: hr_employee_stats_sheet +#: model_terms:ir.ui.view,arch_db:hr_employee_stats_sheet.view_hr_timesheet_sheet_form_inherit +msgid "Total recovery hours in this timesheet sheet time range" +msgstr "" +"Total heures de repos compensateur accumulé sur cette période de temps" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__activity_exception_decoration +msgid "Type of the exception activity on record." +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,field_description:hr_employee_stats_sheet.field_hr_employee_stats__website_message_ids +msgid "Website Messages" +msgstr "" + +#. module: hr_employee_stats_sheet +#: model:ir.model.fields,help:hr_employee_stats_sheet.field_hr_employee_stats__website_message_ids +msgid "Website communication history" +msgstr "" \ No newline at end of file diff --git a/hr_employee_stats_sheet/models/__init__.py b/hr_employee_stats_sheet/models/__init__.py new file mode 100644 index 0000000..b2348e6 --- /dev/null +++ b/hr_employee_stats_sheet/models/__init__.py @@ -0,0 +1,5 @@ +from . import hr_employee_stats +from . import hr_timesheet_sheet +from . import res_config +from . import res_company +from . import hr_leave_allocation \ No newline at end of file diff --git a/hr_employee_stats_sheet/models/hr_employee_stats.py b/hr_employee_stats_sheet/models/hr_employee_stats.py new file mode 100644 index 0000000..64d2211 --- /dev/null +++ b/hr_employee_stats_sheet/models/hr_employee_stats.py @@ -0,0 +1,189 @@ +import logging + +from odoo import api, fields, models +from datetime import timedelta + +_logger = logging.getLogger(__name__) + + +class HrEmployeeStats(models.Model): + _name = "hr.employee.stats" + _description = "Employee Stats" + _order = "date desc" + _inherit = ["mail.thread", "mail.activity.mixin"] + + name = fields.Char("Name", compute="_compute_name", store=True) + dayofweek = fields.Integer("Day of Week", compute="_compute_dayofweek") + is_public_holiday = fields.Boolean("Public Holiday", compute="_compute_dayofweek") + employee_id = fields.Many2one("hr.employee", "Employee", required=True) + department_id = fields.Many2one("hr.department", "Department") + timesheet_line_ids = fields.One2many( + "account.analytic.line", + "employee_id", + "Timesheet lines", + compute="_compute_timesheet_line_ids", + ) + date = fields.Date("Date", required=True) + company_id = fields.Many2one( + "res.company", + "Company", + default=lambda self: self.env.company, + required=True, + ) + sheet_id = fields.Many2one("hr_timesheet.sheet", "Timesheet") + total_hours = fields.Float("Total Hours", compute="_compute_hours") + total_planned_hours = fields.Float("Total Planning Hours", compute="_compute_hours") + total_leave_hours = fields.Float("Total Leave Hours", compute="_compute_hours") + total_recovery_hours = fields.Float( + "Total Recovery Hours", compute="_compute_hours" + ) + gap_hours = fields.Float("Gap Hours", compute="_compute_hours") + + def _get_holiday_status_id(self): + recovery_type_id = self.env.company.recovery_type_id + if recovery_type_id: + return recovery_type_id.id + else: + return False + + def _compute_timesheet_line_ids(self): + for stat in self: + stat.timesheet_line_ids = self.env["account.analytic.line"].search( + [ + ("employee_id", "=", stat.employee_id.id), + ("date", "=", stat.date), + ] + ) + + def _get_intersects( + self, datetime1_start, datetime1_end, datetime2_start, datetime2_end + ): + latest_start = max(datetime1_start, datetime2_start) + earliest_end = min(datetime1_end, datetime2_end) + delta = (earliest_end - latest_start).total_seconds() / 3600 + return max(0, delta) + + def get_total_hours_domain(self): + return [ + ("employee_id", "=", self.employee_id.id), + ("date", "=", self.date), + ] + + @api.depends("timesheet_line_ids") + def _get_total_hours(self): + timesheet_line = self.env["account.analytic.line"] + for stat in self: + if stat.date and stat.employee_id: + timesheet_line_ids = timesheet_line.search( + stat.get_total_hours_domain() + ) + total_hours = sum(timesheet_line_ids.mapped("unit_amount")) + else: + total_hours = 0 + return total_hours + + def _get_total_planned_hours(self): + for stat in self: + if stat.employee_id and stat.date and not stat.is_public_holiday: + dayofweek = int(stat.date.strftime("%u")) - 1 + calendar_id = stat.employee_id.resource_calendar_id + week_number = stat.date.isocalendar()[1] % 2 + if calendar_id.two_weeks_calendar: + hours = calendar_id.attendance_ids.search( + [ + ("dayofweek", "=", dayofweek), + ("calendar_id", "=", calendar_id.id), + ("week_type", "=", week_number), + ] + ) + else: + hours = calendar_id.attendance_ids.search( + [ + ("dayofweek", "=", dayofweek), + ("calendar_id", "=", calendar_id.id), + ] + ) + total_planned_hours = sum( + hours.mapped(lambda r: r.hour_to - r.hour_from) + ) + else: + total_planned_hours = 0 + return total_planned_hours + + def _get_total_recovery_hours(self): + recovery = self.env["hr.leave"] + for stat in self: + if stat.date and stat.employee_id and stat._get_holiday_status_id(): + recovery_ids = recovery.search( + [ + ("employee_id", "=", stat.employee_id.id), + ("request_date_from", ">=", stat.date), + ("request_date_from", "<=", stat.date), + ("holiday_status_id", "=", stat._get_holiday_status_id()), + ] + ) + total_recovery_hours = sum( + recovery_ids.mapped("number_of_hours_display") + ) + else: + total_recovery_hours = 0 + return total_recovery_hours + + def _get_total_leave_hours(self): + total_leave_hours = 0 + for stat in self: + if stat.date and stat.employee_id: + leave_ids = self.env["hr.leave"].search( + [ + ("employee_id", "=", stat.employee_id.id), + ("holiday_status_id", "!=", stat._get_holiday_status_id()), + ("request_date_from", ">=", stat.date), + ("request_date_to", "<=", stat.date), + ] + ) + total_leave_hours = sum(leave_ids.mapped("number_of_hours_display")) + return total_leave_hours + + @api.depends("employee_id", "date") + def _compute_name(self): + for stat in self: + stat.name = "%s - %s" % (stat.employee_id.name, stat.date) + + @api.depends("date","employee_id") + def _compute_dayofweek(self): + for stat in self: + if not stat.date: + stat.dayofweek = None + stat.is_public_holiday = False + continue + stat.dayofweek = int(stat.date.strftime("%u")) - 1 + stat.is_public_holiday = bool(stat.sheet_id.employee_id._get_public_holidays(stat.date, stat.date - timedelta(days=1))) + + def _get_gap_hours(self, total_hours, total_recovery_hours, total_leave_hours, total_planned_hours): + self.ensure_one() + balance = ( + total_hours + + total_recovery_hours + + total_leave_hours + - total_planned_hours + ) + return balance + + @api.depends( + "employee_id", + "date", + "total_hours", + "total_planned_hours", + "timesheet_line_ids", + ) + def _compute_hours(self): + for stat in self: + total_hours = stat._get_total_hours() + total_recovery_hours = stat._get_total_recovery_hours() + total_planned_hours = stat._get_total_planned_hours() + total_leave_hours = stat._get_total_leave_hours() + stat.total_hours = total_hours + stat.total_planned_hours = total_planned_hours + stat.gap_hours = stat._get_gap_hours(total_hours, total_recovery_hours, total_leave_hours, total_planned_hours) + stat.total_recovery_hours = total_recovery_hours + stat.total_leave_hours = total_leave_hours diff --git a/hr_employee_stats_sheet/models/hr_leave_allocation.py b/hr_employee_stats_sheet/models/hr_leave_allocation.py new file mode 100644 index 0000000..068fe15 --- /dev/null +++ b/hr_employee_stats_sheet/models/hr_leave_allocation.py @@ -0,0 +1,9 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. + +from odoo import models, fields + +class HrLeaveAllocation(models.Model): + _inherit = "hr.leave.allocation" + + timesheet_sheet_id = fields.Many2one("hr_timesheet.sheet", string="Timesheet Sheet", readonly=True) diff --git a/hr_employee_stats_sheet/models/hr_timesheet_sheet.py b/hr_employee_stats_sheet/models/hr_timesheet_sheet.py new file mode 100644 index 0000000..681862c --- /dev/null +++ b/hr_employee_stats_sheet/models/hr_timesheet_sheet.py @@ -0,0 +1,265 @@ +from datetime import timedelta + +from odoo import api, fields, models, _ +from odoo.exceptions import UserError +from odoo.addons.resource.models.resource import HOURS_PER_DAY + + +class HrTimesheetSheet(models.Model): + _inherit = "hr_timesheet.sheet" + + employee_stats_ids = fields.One2many( + "hr.employee.stats", "sheet_id", "Employee Stats" + ) + timesheet_sheet_gap_hours = fields.Float( + "Timesheet sheet gap hours", compute="_compute_hours" + ) + timesheet_sheet_recovery_hours = fields.Float( + "Timesheet sheet recovery hours", compute="_compute_hours" + ) + + recovery_allocation_ids = fields.One2many( + 'hr.leave.allocation', + 'timesheet_sheet_id', + string='Recovery Allocations', + ) + + @api.depends("employee_stats_ids.gap_hours", "employee_id") + def _compute_hours(self): + for sheet in self: + sheet.timesheet_sheet_gap_hours = sheet._get_timesheet_sheet_gap_hours() + sheet.timesheet_sheet_recovery_hours = sheet._get_timesheet_sheet_recovery_hours() + + def _get_timesheet_sheet_gap_hours(self): + for sheet in self: + timesheet_sheet_gap_hours = sum( + sheet.employee_stats_ids.filtered( + lambda stat: stat.date <= fields.Date.today() + ).mapped("gap_hours") + ) + return timesheet_sheet_gap_hours + + def _get_timesheet_sheet_recovery_hours(self): + self.ensure_one() + coef = self.env.company.coef + # apply coef only if hours to recovery are positive and coef is existing + if self.timesheet_sheet_gap_hours > 0 and coef > 0: + timesheet_sheet_recovery_hours = self.timesheet_sheet_gap_hours + ( + self.timesheet_sheet_gap_hours * coef / 100 + ) + else: + timesheet_sheet_recovery_hours = self.timesheet_sheet_gap_hours + return timesheet_sheet_recovery_hours + + def search_and_create_employee_stats(self): + for sheet in self: + if sheet.employee_id: + for day in range((sheet.date_end - sheet.date_start).days + 1): + date = sheet.date_start + timedelta(days=day) + stats = self.env["hr.employee.stats"].search( + [ + ("employee_id", "=", sheet.employee_id.id), + ("date", "=", date), + ] + ) + if stats and not stats.sheet_id: + stats.write({"sheet_id": sheet.id}) + if not stats: + self.env["hr.employee.stats"].create( + { + "employee_id": sheet.employee_id.id, + "date": date, + "sheet_id": sheet.id, + "company_id": sheet.company_id.id, + } + ) + return True + + @api.model + def create(self, vals): + res = super().create(vals) + res.search_and_create_employee_stats() + return res + + def write(self, vals): + res = super().write(vals) + if "date_end" in vals or "date_start" in vals or "employee_id" in vals: + self.search_and_create_employee_stats() + return res + + def unlink(self): + for sheet in self: + sheet.employee_stats_ids.unlink() + return super().unlink() + + def action_timesheet_draft(self): + res = super().action_timesheet_draft() + for sheet in self: + # set the state of the recovery allocation to refuse if the timesheet sheet is set to draft + if sheet.state == "draft": + sheet.recovery_allocation_ids.write({"state": "refuse"}) + return res + + def _get_contracts_in_progress_during_timesheet_sheet_time_period(self): + """ + get the contracts which was in progress during the timesheet sheet range time + """ + contracts = self.env["hr.contract"].search( + [ + ("employee_id", "=", self.employee_id.id), + ("state", "in", ("open", "close")), + ("date_start", "<=", self.date_end), + "|", + ("date_end", "=", False), # pas de date de fin OU + ("date_end", ">=", self.date_start), # date de fin après le début + ], + ) + return contracts + + def _get_calendar_in_progress_during_timesheet_time_period(self): + """ + get the ressource calendar which was used during the timesheet sheet time period + """ + #checks if only one contract runs over the duration of the timesheet + contracts = self._get_contracts_in_progress_during_timesheet_sheet_time_period() + if len(contracts) > 1: + # check if a new contract start during timesheet sheet time period. If yes, raise an error + raise UserError( + _("There is a contract starting during the timesheet sheet time period for the employee %s" + "Please create a new timesheet sheet starting from the new contract start date") + % self.employee_id.display_name + ) + # get the ressource calendar id according to the work contract + elif self.employee_id.resource_calendar_id: + return self.employee_id.resource_calendar_id + elif self.env.company.resource_calendar_id: + return self.env.company.resource_calendar_id + return None + + def _get_working_hours_per_week(self): + """ + Get the weekly working hours for the employee, which is defined by: + - the employee's work contract, + - or their resource calendar, + - or the company's resource calendar, + - or the default value of 40 hours per week. + :return: limit recovery hours + """ + # get ressource calendar id used during the timesheet sheet time period + ressource_calendar_id = self._get_calendar_in_progress_during_timesheet_time_period() + if ressource_calendar_id: + resource_calendar_attendance_ids = self.env[ + "resource.calendar.attendance" + ].search([("calendar_id", "=", ressource_calendar_id.id)]) + # calculate working hours per week according to the employee's resource calendar + weekly_working_hours = 0 + for day in resource_calendar_attendance_ids: + weekly_working_hours += day.hour_to - day.hour_from + return weekly_working_hours + return HOURS_PER_DAY * 5 + + def _get_working_hours_per_day(self): + """ + Get the hours per day for the employee according to: + - the employee's work contract, + - or their resource calendar, + - or the company's resource calendar, + - or the default value of 8 hours per day. + :return: hours per day + """ + # get ressource calendar id used during the timesheet sheet time period + ressource_calendar_id = self._get_calendar_in_progress_during_timesheet_time_period() + if ressource_calendar_id: + return ressource_calendar_id.hours_per_day + return HOURS_PER_DAY + + def _get_max_allowed_recovery_hours(self): + """ + Get the maximum number of hours beyond which new recovery allowances cannot be created + """ + return self._get_working_hours_per_week() + + def action_generate_recovery_allocation(self): + # check if the user has the right to review the timesheet sheet + self.ensure_one() + self._check_can_review() + recovery_type_id = self.env.company.recovery_type_id + employee_id = self.employee_id + + if not employee_id or not recovery_type_id: + raise UserError( + _("Employe not defined for the timesheet sheet or recovery type not defined in settings") + ) + if recovery_type_id.request_unit != 'hour' or not recovery_type_id.allows_negative: + raise UserError( + _("The recovery type must be set to 'Hours' and allow negative leaves in the settings") + ) + # check if allocation already exists for this timesheet sheet, if yes, refuse it + allocations = ( + self.env["hr.leave.allocation"] + .sudo() + .search( + [ + ("timesheet_sheet_id.id", "=", self.id), + ("state", "!=", "refuse"), + ] + ) + ) + if allocations: + allocations.write({"state": "refuse"}) + + # get recovery hours from total gap hours of the timesheet sheet + recovery_hours = self._get_timesheet_sheet_recovery_hours() + + # get recovery hours cap + max_allowed_recovery_hours = self._get_max_allowed_recovery_hours() + if max_allowed_recovery_hours: + # find recovery remaining leaves for the employee + recovery_type_id = self.env.company.recovery_type_id + # get virtual remaining leaves for the employee and the recovery leaves type + data_days = recovery_type_id.get_employees_days([employee_id.id])[employee_id.id] + total_recovery_type_leaves = data_days.get(recovery_type_id.id,{}) + total_virtual_remaining_recovery_type_leaves = total_recovery_type_leaves.get('virtual_remaining_leaves', 0) + # add the recovery hours to the total remaining leaves recovery type, and check if the limit of recovery hours is exceeded + exceeded_hours = total_virtual_remaining_recovery_type_leaves + recovery_hours - max_allowed_recovery_hours + # if limit recovery hours is exceeded, don't create a new allocation + if exceeded_hours > 0: + raise UserError( + _( + "The number of recovery hours exceeds the authorized limit (%s h) by %s hours" + ) + % (max_allowed_recovery_hours, exceeded_hours) + ) + # convert recovery hours into days + recovery_days = recovery_hours / self._get_working_hours_per_day() + + # create an allocation (positive or negative) according to the total gap hours for the timesheet sheet range time + new_alloc = self.env["hr.leave.allocation"].create( + { + "private_name": "Allocation : %s - %s" % (recovery_type_id.name, employee_id.display_name), + "holiday_status_id": recovery_type_id.id, + "employee_id": employee_id.id, + "number_of_days": recovery_days, + "timesheet_sheet_id": self.id, + "allocation_type": 'accrual', + } + ) + + # set the allocation to validate or not + if self.env.company.auto_validate_recovery_allocation: + new_alloc.write({"state": "validate"}) + + return { + "type": "ir.actions.act_window", + "res_model": "hr.leave.allocation", + "res_id": new_alloc.id, + "view_mode": "form", + "target": "current", + "views": [ + ( + self.env.ref("hr_holidays.hr_leave_allocation_view_form").id, + "form", + ) + ], + } + diff --git a/hr_employee_stats_sheet/models/res_company.py b/hr_employee_stats_sheet/models/res_company.py new file mode 100644 index 0000000..b7efb9b --- /dev/null +++ b/hr_employee_stats_sheet/models/res_company.py @@ -0,0 +1,12 @@ +from odoo import fields, models + + +class ResCompany(models.Model): + _inherit = "res.company" + + recovery_type_id = fields.Many2one( + "hr.leave.type", string="Leave recovery type" + ) + coef = fields.Float("Coef", default=25) + + auto_validate_recovery_allocation = fields.Boolean("Auto validate recovery allocation", default=True) \ No newline at end of file diff --git a/hr_employee_stats_sheet/models/res_config.py b/hr_employee_stats_sheet/models/res_config.py new file mode 100644 index 0000000..da3e403 --- /dev/null +++ b/hr_employee_stats_sheet/models/res_config.py @@ -0,0 +1,29 @@ +from odoo import fields, models + + +class ResConfigSettings(models.TransientModel): + _inherit = "res.config.settings" + + recovery_type_id = fields.Many2one( + "hr.leave.type", + related="company_id.recovery_type_id", + string="Leave recovery type", + readonly=False, + domain="[('request_unit', '=', 'hour'), ('allows_negative', '=', True)]", + ) + + coef = fields.Float( + related="company_id.coef", + required=True, + string="Coef (in %)", + domain="[('company_id', '=', company_id)]", + readonly=False, + help="The coef is applied to recovery hours" + "Example : an employe make 1h overtime, if coef is set to 25%, the recovery hours allocated will be 1,25h", + ) + + auto_validate_recovery_allocation = fields.Boolean( + related="company_id.auto_validate_recovery_allocation", + readonly=False, + domain="[('company_id', '=', company_id)]", + ) diff --git a/hr_employee_stats_sheet/security/ir.model.access.csv b/hr_employee_stats_sheet/security/ir.model.access.csv new file mode 100644 index 0000000..a0597ca --- /dev/null +++ b/hr_employee_stats_sheet/security/ir.model.access.csv @@ -0,0 +1,3 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_hr_employee_stats_manager,hr.employee.stats,model_hr_employee_stats,hr_timesheet.group_timesheet_manager,1,1,1,1 +access_hr_employee_stats_user,hr.employee.stats,model_hr_employee_stats,hr_timesheet.group_hr_timesheet_user,1,0,1,0 \ No newline at end of file diff --git a/hr_employee_stats_sheet/tests/__init__.py b/hr_employee_stats_sheet/tests/__init__.py new file mode 100644 index 0000000..ddd6ef5 --- /dev/null +++ b/hr_employee_stats_sheet/tests/__init__.py @@ -0,0 +1 @@ +from . import test_employee_stats diff --git a/hr_employee_stats_sheet/tests/test_employee_stats.py b/hr_employee_stats_sheet/tests/test_employee_stats.py new file mode 100644 index 0000000..2594b3b --- /dev/null +++ b/hr_employee_stats_sheet/tests/test_employee_stats.py @@ -0,0 +1,254 @@ +from odoo.tests.common import TransactionCase +from odoo.tests import tagged +from datetime import date, timedelta +from odoo.exceptions import UserError + +@tagged("post_install", "-at_install") +class TestHrEmployeeStatsRecovery(TransactionCase): + def setUp(self): + super().setUp() + self.user = self.env['res.users'].create({ + 'name': 'Camille', + 'login': 'camille', + }) + self.recovery_type = self.env['hr.leave.type'].create({ + 'name': 'Recovery', + 'request_unit': 'hour', + 'allows_negative': True, + }) + self.employee = self.env['hr.employee'].create({ + 'name': 'Camille', + 'user_id': self.user.id, + }) + self.base_calendar = self.env['resource.calendar'].create({ + 'name': 'Default Calendar', + 'attendance_ids': [ + (0, 0, {'name': 'Monday Morning', 'dayofweek': '0', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Monday Afternoon', 'dayofweek': '0', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Tuesday Morning', 'dayofweek': '1', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Tuesday Afternoon', 'dayofweek': '1', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Wednesday Morning', 'dayofweek': '2', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Wednesday Afternoon', 'dayofweek': '2', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Thursday Morning', 'dayofweek': '3', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Thursday Afternoon', 'dayofweek': '3', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Friday Morning', 'dayofweek': '4', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Friday Afternoon', 'dayofweek': '4', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + ], + }) + self.employee.resource_calendar_id = self.base_calendar + self.env.company.coef = 25 + + def _create_timesheet_sheet(self, start_date): + #Crée une feuille de temps pour la semaine du lundi au dimanche + return self.env['hr_timesheet.sheet'].create({ + 'employee_id': self.employee.id, + 'date_start': start_date, + 'date_end': start_date + timedelta(days=6), + }) + + def _create_stats(self, start_date, nb_days, unit_amount): + # Crée des temps du lundi au vendredi (ou nb_days) + for i in range(nb_days): + self.env['account.analytic.line'].create({ + 'employee_id': self.employee.id, + 'date': start_date + timedelta(days=i), + 'unit_amount': unit_amount, + 'account_id': 1, + 'name': 'Work Entry', + }) + # Génère les hr_employee_stats pour chaque jour de la période + stat = self.env['hr.employee.stats'].create({ + 'employee_id': self.employee.id, + 'date': start_date + timedelta(days=i), + }) + stat._compute_hours() + yield stat + + def test_invalide_recovery_type(self): + start_date = date.today() - timedelta(days=date.today().weekday() + 7) # lundi de la semaine dernière + self.recovery_type.request_unit = 'day' + self.recovery_type.allows_negative = False + timesheet_sheet = self._create_timesheet_sheet(start_date) + self.env.company.recovery_type_id = self.recovery_type + with self.assertRaises(UserError): + timesheet_sheet.action_generate_recovery_allocation() + + def test_no_recovery_hours(self): + start_date = date.today() - timedelta(days=date.today().weekday() + 7) # lundi de la semaine dernière + timesheet_sheet = self._create_timesheet_sheet(start_date) + for stat in self._create_stats(start_date, 5, 7): #créer 5 stats de 7h chacune + # Compare les heures de récupération calculées et le calendrier qui prévoit 7h par jour + self.assertEqual(stat.total_hours, 7, "total_hours should be 7",) # l'employé a travaillé 7h chaque jour + self.assertEqual(stat.total_planned_hours, 7, "total_planned_hours should be 7",) # le calendrier prévoit 7h chaque jour + self.assertEqual(stat.total_leave_hours, 0, "total_leave_hours should be 0",) # l'employé n'a pas de congé sur ce jour + self.assertEqual(stat.total_recovery_hours, 0, "total_recovery_hours should be 0",) # l'employé n'a pas posé de récupération sur ce jour + self.assertEqual(stat.gap_hours, 0, "gap_hours should be 0",) # pas de différence entre les heures travaillées et les heures planifiées + # La feuille de temps sur cette période doit compter 0h de déficit et 0h de récupération + self.assertEqual(timesheet_sheet.timesheet_sheet_gap_hours, 0, "timesheet_sheet_gap_hours should be 0",) + self.assertEqual(timesheet_sheet.timesheet_sheet_recovery_hours, 0, "timesheet_sheet_recovery_hours should be 0",) + + def test_positive_recovery_hours(self): + start_date = date.today() - timedelta(days=date.today().weekday() + 7) # lundi de la semaine dernière + timesheet_sheet = self._create_timesheet_sheet(start_date) + for stat in self._create_stats(start_date, 5, 8): #créer 5 stats de 8h chacune + # Compare les heures de récupération calculées et le calendrier qui prévoit 7h par jour + self.assertEqual(stat.total_hours, 8, "total_hours should be 8",) # l'employé a travaillé 8h chaque jour + self.assertEqual(stat.total_planned_hours, 7, "total_planned_hours should be 7",) # le calendrier prévoit 7h chaque jour + self.assertEqual(stat.total_leave_hours, 0, "total_leave_hours should be 0",) # l'employé n'a pas de congé sur ce jour + self.assertEqual(stat.total_recovery_hours, 0, "total_recovery_hours should be 0",) # l'employé n'a pas posé de récupération sur ce jour + self.assertEqual(stat.gap_hours, 1, "gap_hours should be 1",) # l'employée a travaillé une heure de plus que prévu + # La feuille de temps doit compter 5h d'heure sup soit 6,25h de récupération avec la majoration de 25% + self.assertEqual(timesheet_sheet.timesheet_sheet_gap_hours, 5, "timesheet_sheet_gap_hours should be 5",) + self.assertEqual(timesheet_sheet.timesheet_sheet_recovery_hours, 6.25, "timesheet_sheet_recovery_hours should be 6,25",) + # générer l'allocation de récupération depuis la feuille de temps + timesheet_sheet.action_generate_recovery_allocation() + recovery_allocation = self.env["hr.leave.allocation"].search([("timesheet_sheet_id","=",timesheet_sheet.id)]) + self.assertEqual(len(recovery_allocation), 1, "There should be one recovery") + self.assertEqual(recovery_allocation.number_of_days,0.78125, "The recovery allocation should be for 6.25h/8h = 0.78125 day") + + def test_negative_recovery_hours(self): + start_date = date.today() - timedelta(days=date.today().weekday() + 7) # lundi de la semaine dernière + timesheet_sheet = self._create_timesheet_sheet(start_date) + for stat in self._create_stats(start_date, 5, 6): #créer 5 stats de 6h chacune + # Compare les heures de récupération calculées et le calendrier qui prévoit 7h par jour + self.assertEqual(stat.total_hours, 6, "total_hours should be 6",) # l'employé a travaillé 6h chaque jour + self.assertEqual(stat.total_planned_hours, 7, "total_planned_hours should be 7",) # le calendrier prévoit 7h chaque jour + self.assertEqual(stat.total_leave_hours, 0, "total_leave_hours should be 0",) # l'employé n'a pas de congé sur ce jour + self.assertEqual(stat.total_recovery_hours, 0, "total_recovery_hours should be 0",) # l'employé n'a pas posé de récupération sur ce jour + self.assertEqual(stat.gap_hours, -1, "gap_hours should be -1",) # l'employée a travaillé une heure de moins que prévu + # La feuille de temps doit compter -5h de déficit et -5h de récupération + self.assertEqual(timesheet_sheet.timesheet_sheet_gap_hours, -5, "timesheet_sheet_gap_hours should be -5",) # l'employé a travaillé -5h au total sur la semaine + self.assertEqual(timesheet_sheet.timesheet_sheet_recovery_hours, -5, "timesheet_sheet_recovery_hours should be -5",) # -5h sera le montant de l'allocation de récupération (pas de coef appliqué pour les déficites d'heures) + # générer l'allocation de récupération depuis la feuille de temps + timesheet_sheet.action_generate_recovery_allocation() + recovery_allocation = self.env["hr.leave.allocation"].search([("timesheet_sheet_id","=",timesheet_sheet.id)]) + self.assertEqual(len(recovery_allocation), 1, "There should be one recovery") + self.assertEqual(recovery_allocation.number_of_days, -0.625, "The recovery allocation should be for -5/8 hours = ") + + def test_recovery_hours_part_time_employee(self): + part_time_calendar = self.env['resource.calendar'].create({ + 'name': 'Part Time Calendar', + 'attendance_ids': [ + (0, 0, {'name': 'Monday Morning', 'dayofweek': '0', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Monday Afternoon', 'dayofweek': '0', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Tuesday Morning', 'dayofweek': '1', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Tuesday Afternoon', 'dayofweek': '1', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Wednesday Morning', 'dayofweek': '2', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Wednesday Afternoon', 'dayofweek': '2', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Thursday Morning', 'dayofweek': '3', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Thursday Afternoon', 'dayofweek': '3', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + ], + }) + self.employee.resource_calendar_id = part_time_calendar.id + + start_date = date.today() - timedelta(days=date.today().weekday() + 7) # lundi de la semaine dernière + timesheet_sheet = self._create_timesheet_sheet(start_date) + for stat in self._create_stats(start_date, 4, 8): #créer 4 stats de 8h chacune + # Compare les heures de récupération calculées et le calendrier qui prévoit 7h par jour pendant 4 jours + self.assertEqual(stat.total_hours, 8, "total_hours should be 8",) # l'employé a travaillé 6h chaque jour + self.assertEqual(stat.total_planned_hours, 7, "total_planned_hours should be 7",) # le calendrier prévoit 7h chaque jour + self.assertEqual(stat.total_leave_hours, 0, "total_leave_hours should be 0",) # l'employé n'a pas de congé sur ce jour + self.assertEqual(stat.total_recovery_hours, 0, "total_recovery_hours should be 0",) # l'employé n'a pas posé de récupération sur ce jour + self.assertEqual(stat.gap_hours, 1, "gap_hours should be 1",) # l'employée a travaillé une heure de moins que prévu + # La feuille de temps doit compter 4h d'heure sup soit 5h de récupération avec la majoration de 25% + self.assertEqual(timesheet_sheet.timesheet_sheet_gap_hours, 4, "timesheet_sheet_gap_hours should be 4",) # l'employé a travaillé supplémentaire 4h au total sur la semaine + self.assertEqual(timesheet_sheet.timesheet_sheet_recovery_hours, 5, "timesheet_sheet_recovery_hours should be 5",) # 5h sera le montant de l'allocation de récupération (coef de 25% de majoration) + + def test_recovery_hours_change_contract(self): + part_time_calendar = self.env['resource.calendar'].create({ + 'name': 'Part Time Calendar', + 'attendance_ids': [ + (0, 0, {'name': 'Monday Morning', 'dayofweek': '0', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Monday Afternoon', 'dayofweek': '0', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Tuesday Morning', 'dayofweek': '1', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Tuesday Afternoon', 'dayofweek': '1', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Wednesday Morning', 'dayofweek': '2', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Wednesday Afternoon', 'dayofweek': '2', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Thursday Morning', 'dayofweek': '3', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Thursday Afternoon', 'dayofweek': '3', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + ], + }) + #create one contract ending on wednesday and one other starting on thursday + self.env['hr.contract'].create({ + 'name': 'Contract 1', + 'employee_id': self.employee.id, + 'date_start': date.today() - timedelta(days=300), # date de début factice + 'date_end': date.today() - timedelta(days= date.today().weekday() + 5), # date de fin le mercredi de la semaine dernière + 'resource_calendar_id': self.base_calendar.id, + 'wage': 2000, + 'state': 'close', + }) + self.env['hr.contract'].create({ + 'name': 'Contract 2', + 'employee_id': self.employee.id, + 'state': 'open', + 'date_start': date.today() - timedelta(days= date.today().weekday() + 4), # date de début le jeudi de la semaine dernière + 'date_end': False, + 'resource_calendar_id': self.base_calendar.id, + 'wage': 1500, + }) + self.employee.resource_calendar_id = part_time_calendar.id + start_date = date.today() - timedelta(days=date.today().weekday() + 7) # lundi de la semaine dernière + #create a timesheet with period including the change of contract + timesheet_sheet = self._create_timesheet_sheet(start_date) + #the create of recovery allocation should raise an error + with self.assertRaises(UserError): + timesheet_sheet.action_generate_recovery_allocation() + + def test_recovery_hours_change_contract_sucess(self): + part_time_calendar = self.env['resource.calendar'].create({ + 'name': 'Part Time Calendar', + 'attendance_ids': [ + (0, 0, {'name': 'Monday Morning', 'dayofweek': '0', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Monday Afternoon', 'dayofweek': '0', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Tuesday Morning', 'dayofweek': '1', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Tuesday Afternoon', 'dayofweek': '1', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Wednesday Morning', 'dayofweek': '2', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Wednesday Afternoon', 'dayofweek': '2', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + (0, 0, {'name': 'Thursday Morning', 'dayofweek': '3', 'hour_from': 9, 'hour_to': 12, 'day_period': 'morning'}), + (0, 0, {'name': 'Thursday Afternoon', 'dayofweek': '3', 'hour_from': 13, 'hour_to': 17, 'day_period': 'afternoon'}), + ], + }) + #create one contract ending on wednesday and one other starting on thursday + self.env['hr.contract'].create({ + 'name': 'Contract 1', + 'employee_id': self.employee.id, + 'date_start': date(2025,8,18), + 'date_end': date(2025,8,24), + 'resource_calendar_id': self.base_calendar.id, + 'wage': 2000, + 'state': 'close', + }) + self.env['hr.contract'].create({ + 'name': 'Contract 2', + 'employee_id': self.employee.id, + 'state': 'open', + 'date_start': date(2025,8,25), + 'date_end': date(2025,8,31), + 'resource_calendar_id': part_time_calendar.id, + 'wage': 1500, + }) + self.employee.resource_calendar_id = part_time_calendar.id + #create a timesheet with period including the change of contract + timesheet_sheet_1 = self._create_timesheet_sheet(date(2025,8,18)) + timesheet_sheet_2 = self._create_timesheet_sheet(date(2025,8,25)) + + timesheet_sheet_1.action_generate_recovery_allocation() + timesheet_sheet_2.action_generate_recovery_allocation() + + recovery_allocation = self.env["hr.leave.allocation"].search([("timesheet_sheet_id","=",timesheet_sheet_1.id)]) + self.assertEqual(len(recovery_allocation), 1, "There should be one recovery") + + recovery_allocation = self.env["hr.leave.allocation"].search([("timesheet_sheet_id","=",timesheet_sheet_2.id)]) + self.assertEqual(len(recovery_allocation), 1, "There should be one recovery") + + + + + + + + + + + diff --git a/hr_employee_stats_sheet/views/hr_employee_stats.xml b/hr_employee_stats_sheet/views/hr_employee_stats.xml new file mode 100644 index 0000000..08db1ed --- /dev/null +++ b/hr_employee_stats_sheet/views/hr_employee_stats.xml @@ -0,0 +1,20 @@ + + + hr.employee.stats.tree + hr.employee.stats + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/hr_employee_stats_sheet/views/hr_timesheet_sheet.xml b/hr_employee_stats_sheet/views/hr_timesheet_sheet.xml new file mode 100644 index 0000000..cfd6a75 --- /dev/null +++ b/hr_employee_stats_sheet/views/hr_timesheet_sheet.xml @@ -0,0 +1,42 @@ + + + + hr.timesheet.sheet.form.inherit + hr_timesheet.sheet + + + + + +