[MIG] hr_luncheon_voucher: migrate to 18.0
This commit is contained in:
170
hr_luncheon_voucher/README.md
Normal file
170
hr_luncheon_voucher/README.md
Normal file
@@ -0,0 +1,170 @@
|
||||
# HR Luncheon Voucher
|
||||
|
||||
[](https://odoo-community.org/page/development-status)
|
||||
[](http://www.gnu.org/licenses/agpl-3.0-standalone.html)
|
||||
[](https://github.com/elabore-coop/hr-tools)
|
||||
|
||||
Manage luncheon vouchers credit and distribution for employees.
|
||||
|
||||
## Description
|
||||
|
||||
This module allows the management of luncheon vouchers attribution and
|
||||
distribution. Employees can indicate which days are not concerned by luncheon
|
||||
vouchers. HR managers can adjust the number of luncheon vouchers to distribute
|
||||
and follow each employee's credit.
|
||||
|
||||
### Key Models
|
||||
|
||||
- **Luncheon Voucher Allocation** (`hr.lv.allocation`): tracks each distribution
|
||||
campaign per employee, with a state workflow (Draft → Confirmed → Distributed)
|
||||
and computed counters (acquired, due, distributed, balance).
|
||||
- **Employee extensions** (`hr.employee`): additional fields store running totals
|
||||
of acquired, distributed, and remaining vouchers, plus a default monthly
|
||||
distribution value.
|
||||
- **Meeting type extension** (`calendar.event.type`): a boolean field
|
||||
`remove_luncheon_voucher` lets you mark event categories that cancel the daily
|
||||
voucher (e.g. off-site meetings, free lunch).
|
||||
|
||||
### Attribution Rules
|
||||
|
||||
A luncheon voucher is acquired for a working day if:
|
||||
|
||||
- The employee worked on one or all the attendances of the day (depending on
|
||||
whether the option **Half working days cancel luncheon vouchers** is enabled).
|
||||
- There is no meeting that cancels the voucher during that day (e.g. off-site
|
||||
or free lunch meetings).
|
||||
- An attendance is considered worked as long as there is no leave covering the
|
||||
whole attendance time slot.
|
||||
|
||||
The calculation is fully automated when an allocation request is created: the
|
||||
system iterates through each day of the period, checks the working calendar,
|
||||
validates attendance coverage, verifies leaves, and cross-references calendar
|
||||
events.
|
||||
|
||||
### Allocation Lifecycle
|
||||
|
||||
| State | Description |
|
||||
|-------|-------------|
|
||||
| **Draft** | Allocation request created by the wizard. HR can adjust the number of vouchers to distribute. |
|
||||
| **Confirmed** | HR manager validates the figures. Employee counters are updated. |
|
||||
| **Distributed** | Vouchers have been effectively handed out. Final counters are set. |
|
||||
|
||||
### Employee Dashboard
|
||||
|
||||
In the employee form view, three stat buttons display the current running
|
||||
totals: Acquired, Distributed, and Remaining (Due) vouchers. HR can also
|
||||
refresh these values on demand.
|
||||
|
||||
## Installation
|
||||
|
||||
Use Odoo normal module installation procedure to install `hr_luncheon_voucher`.
|
||||
|
||||
### Dependencies
|
||||
|
||||
- `base` (Odoo core)
|
||||
- `calendar` (Odoo core)
|
||||
- `hr` (Odoo core)
|
||||
- `hr_holidays` (Odoo core)
|
||||
- `resource` (Odoo core)
|
||||
- `hr_effective_attendance_period` (this repository)
|
||||
|
||||
### Security / Access Rights
|
||||
|
||||
| Model | HR Users | HR Managers |
|
||||
|-------|----------|-------------|
|
||||
| `hr.lv.allocation` | Read only | Full CRUD |
|
||||
| `generate.lv.allocation.requests` | Read only | Full CRUD |
|
||||
|
||||
## Configuration
|
||||
|
||||
1. **Define cancelling meeting categories.**
|
||||
Go to *Configuration > Technical > Calendar > Meeting Types* and enable the
|
||||
*Remove luncheon voucher* flag on the relevant categories (e.g. Off-site,
|
||||
Free lunch). Two default categories are pre-loaded by the module.
|
||||
|
||||
2. **Toggle the half-day rule.**
|
||||
Go to *Configuration > General Settings > Employees* and check/uncheck
|
||||
*Half working days cancel luncheon vouchers* depending on whether partial
|
||||
attendance should still grant a voucher.
|
||||
|
||||
3. **Set default monthly distribution per employee.**
|
||||
Open each employee's form, go to the *Luncheon Vouchers* section under HR
|
||||
settings, and enter the *Default monthly distribution* value. This value is
|
||||
pre-filled when a new allocation is created.
|
||||
|
||||
4. **Configure effective attendance periods.**
|
||||
Go to *Configuration > Technical > Resource > Working Times* and create
|
||||
Working Time entries for each attendance combination used in your company.
|
||||
Make sure to indicate which periods are effectively attended — the module
|
||||
relies on `hr_effective_attendance_period` to determine daily presence.
|
||||
|
||||
## Usage
|
||||
|
||||
### Calendar Events
|
||||
|
||||
When creating a calendar event that cancels luncheon voucher distribution
|
||||
(off-site meeting, free lunch, etc.), simply assign the corresponding meeting
|
||||
category to the event.
|
||||
|
||||
### Generating Allocations
|
||||
|
||||
For each distribution period, the HR manager should:
|
||||
|
||||
1. Go to the **Employees** list view.
|
||||
2. Select all employees concerned by luncheon vouchers distribution.
|
||||
3. Click the header button **Generate Luncheon Vouchers Allocations**.
|
||||
4. Fill in the wizard: set a campaign name (e.g. "January 2026"), a start
|
||||
date, and an end date.
|
||||
5. Click **Create allocations requests**.
|
||||
|
||||
A voucher allocation request is created for each selected employee in **Draft**
|
||||
state. The number of acquired vouchers is computed automatically based on the
|
||||
attribution rules.
|
||||
|
||||
### Allocation Workflow
|
||||
|
||||
1. **Confirm** an allocation request when the figures are verified. Employee
|
||||
running counters are updated.
|
||||
2. **Distribute** the allocation when the vouchers have been effectively handed
|
||||
out. Click *Distribute Vouchers* to mark the request as distributed.
|
||||
3. **Adjust distribution** at any time in Draft state to override the
|
||||
pre-filled distributed quantity.
|
||||
4. **Back to draft** allows correcting an already confirmed or distributed
|
||||
allocation.
|
||||
5. The **Vouchers balance** column shows the difference between due and
|
||||
distributed vouchers for each allocation.
|
||||
|
||||
### Monitoring
|
||||
|
||||
Each employee form displays three stat buttons — *Acquired*, *Distributed*,
|
||||
and *Due* — giving a real-time summary of the employee's luncheon voucher
|
||||
status. Use the **Refresh Luncheon Vouchers** button in the form header to
|
||||
recompute these values if needed.
|
||||
|
||||
## Known Issues / Roadmap
|
||||
|
||||
None yet.
|
||||
|
||||
## Bug Tracker
|
||||
|
||||
Bugs are tracked on [Gitea Issues](https://git.elabore.coop/Elabore/hr-tools/issues).
|
||||
In case of trouble, please check there if your issue has already been reported.
|
||||
If you spotted it first, help us smash it by providing a detailed and welcomed
|
||||
feedback.
|
||||
|
||||
## Credits
|
||||
|
||||
### Contributors
|
||||
|
||||
- Stéphan Sainléger <https://github.com/stephansainleger>
|
||||
|
||||
### Funders
|
||||
|
||||
The development of this module has been financially supported by:
|
||||
|
||||
- Elabore (https://elabore.coop)
|
||||
- Amaco (https://amaco.org)
|
||||
|
||||
### Maintainer
|
||||
|
||||
This module is maintained by Elabore.
|
||||
2
hr_luncheon_voucher/__init__.py
Normal file
2
hr_luncheon_voucher/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
||||
from . import models
|
||||
from . import wizard
|
||||
30
hr_luncheon_voucher/__manifest__.py
Normal file
30
hr_luncheon_voucher/__manifest__.py
Normal file
@@ -0,0 +1,30 @@
|
||||
{
|
||||
"name": "HR Luncheon Voucher",
|
||||
"category": "Human Resources",
|
||||
"version": "18.0.1.0.0",
|
||||
"summary": "Manage luncheon vouchers credit and distribution",
|
||||
"author": "Elabore",
|
||||
"website": "https://git.elabore.coop/elabore/hr-tools",
|
||||
"license": "AGPL-3",
|
||||
"installable": True,
|
||||
"application": True,
|
||||
"auto_install": False,
|
||||
"depends": [
|
||||
"base",
|
||||
"calendar",
|
||||
"hr",
|
||||
"hr_effective_attendance_period",
|
||||
"hr_holidays",
|
||||
"resource",
|
||||
],
|
||||
"data": [
|
||||
"security/ir.model.access.csv",
|
||||
"views/event_type.xml",
|
||||
"views/hr_employee_views.xml",
|
||||
"views/hr_lv_allocation_views.xml",
|
||||
"views/res_config_settings_views.xml",
|
||||
"views/menus.xml",
|
||||
"wizard/generate_lv_allocations_wizard.xml",
|
||||
"data/event_type_data.xml",
|
||||
],
|
||||
}
|
||||
14
hr_luncheon_voucher/data/event_type_data.xml
Normal file
14
hr_luncheon_voucher/data/event_type_data.xml
Normal file
@@ -0,0 +1,14 @@
|
||||
<?xml version="1.0" ?>
|
||||
<odoo noupdate="1">
|
||||
<record id="categ_meet_free_lunch" model="calendar.event.type">
|
||||
<field name="name">Free lunch</field>
|
||||
<field name="ref">categ_meet_free_lunch</field>
|
||||
<field name="remove_luncheon_voucher">1</field>
|
||||
</record>
|
||||
|
||||
<record id="categ_meet_offsite" model="calendar.event.type">
|
||||
<field name="name">Off-site</field>
|
||||
<field name="ref">categ_meet_offsite</field>
|
||||
<field name="remove_luncheon_voucher">1</field>
|
||||
</record>
|
||||
</odoo>
|
||||
347
hr_luncheon_voucher/i18n/fr.po
Normal file
347
hr_luncheon_voucher/i18n/fr.po
Normal file
@@ -0,0 +1,347 @@
|
||||
# Translation of Odoo Server.
|
||||
# This file contains the translation of the following modules:
|
||||
# * hr_luncheon_voucher
|
||||
#
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Odoo Server 14.0\n"
|
||||
"Report-Msgid-Bugs-To: \n"
|
||||
"POT-Creation-Date: 2023-04-28 12:49+0000\n"
|
||||
"PO-Revision-Date: 2023-04-28 12:49+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_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.view_employee_form_lv
|
||||
msgid "Acquired"
|
||||
msgstr "Acquis"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__number_acquired_lv
|
||||
msgid "Acquired Vouchers"
|
||||
msgstr "Tickets Acquis"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_form
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_tree
|
||||
msgid "Adjust distribution"
|
||||
msgstr "Ajuster la distribution"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_form
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_tree
|
||||
msgid "Back to draft"
|
||||
msgstr "Brouillon"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model,name:hr_luncheon_voucher.model_hr_employee_base
|
||||
msgid "Basic Employee"
|
||||
msgstr "Employé basique"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.lv_allocations_requests_wizard
|
||||
msgid "Cancel"
|
||||
msgstr "Annuler"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model,name:hr_luncheon_voucher.model_res_company
|
||||
msgid "Companies"
|
||||
msgstr "Sociétés"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model,name:hr_luncheon_voucher.model_res_config_settings
|
||||
msgid "Config Settings"
|
||||
msgstr "Paramètres de config"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_form
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_tree
|
||||
msgid "Confirm"
|
||||
msgstr "Confirmer"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields.selection,name:hr_luncheon_voucher.selection__hr_lv_allocation__state__confirmed
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_search
|
||||
msgid "Confirmed"
|
||||
msgstr "Confirmé"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.actions.act_window,name:hr_luncheon_voucher.lv_allocations_requests_wizard_action
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.lv_allocations_requests_wizard
|
||||
msgid "Create Luncheon Vouchers allocations requests"
|
||||
msgstr "Créer demandes d'allocation de Tickets Restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.lv_allocations_requests_wizard
|
||||
msgid "Create allocations requests"
|
||||
msgstr "Créer les demandes d'allocation"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__create_uid
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__create_uid
|
||||
msgid "Created by"
|
||||
msgstr "Créé par"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__create_date
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__create_date
|
||||
msgid "Created on"
|
||||
msgstr "Créé le"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee__default_monthly_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base__default_monthly_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_public__default_monthly_lv
|
||||
msgid "Default monthly distribution"
|
||||
msgstr ""
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_calendar_event_type__display_name
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__display_name
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base__display_name
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__display_name
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_company__display_name
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_config_settings__display_name
|
||||
msgid "Display Name"
|
||||
msgstr "Nom affiché"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_form
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_tree
|
||||
msgid "Distribute Vouchers"
|
||||
msgstr "Tickets distribués"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields.selection,name:hr_luncheon_voucher.selection__hr_lv_allocation__state__distributed
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_search
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.view_employee_form_lv
|
||||
msgid "Distributed"
|
||||
msgstr "Distribué"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__number_distributed_lv
|
||||
msgid "Distributed Vouchers"
|
||||
msgstr "Tickets distribués"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee__distributed_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base__distributed_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_public__distributed_lv
|
||||
msgid "Distributed luncheon vouchers"
|
||||
msgstr "Tickets resto distribués"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__distrib_campaign_name
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__distrib_campaign_name
|
||||
msgid "Distribution campaign"
|
||||
msgstr "Campagne de distribution"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields.selection,name:hr_luncheon_voucher.selection__hr_lv_allocation__state__draft
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_search
|
||||
msgid "Draft"
|
||||
msgstr "Brouillon"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.view_employee_form_lv
|
||||
msgid "Dued"
|
||||
msgstr "Dû"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__number_dued_lv
|
||||
msgid "Dued Vouchers"
|
||||
msgstr "Tickets dûs"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__employee_id
|
||||
msgid "Employee"
|
||||
msgstr "Employé"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.res_config_settings_lv_form
|
||||
msgid "Employee Luncheon Vouchers"
|
||||
msgstr "Tickets Restaurant de l'employé"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__date_to
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__date_to
|
||||
msgid "End Date"
|
||||
msgstr "Date de fin"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model,name:hr_luncheon_voucher.model_calendar_event_type
|
||||
msgid "Event Meeting Type"
|
||||
msgstr "Type d'événement réunion"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.view_employee_tree_lv
|
||||
msgid "Generate Luncheon Vouchers Allocations"
|
||||
msgstr "Générer les allocations de Tickets Restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model,name:hr_luncheon_voucher.model_generate_lv_allocation_requests
|
||||
msgid "Generate Luncheon Vouchers Allocations Requests"
|
||||
msgstr "Générer les demandes d'allocations de Tickets Restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_company__hr_half_day_cancels_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_config_settings__hr_half_day_cancels_voucher
|
||||
msgid "Half working days cancel luncheon vouchers"
|
||||
msgstr "Une demi-journée travaillée annule les tickets restaurants."
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_calendar_event_type__id
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__id
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base__id
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__id
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_company__id
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_config_settings__id
|
||||
msgid "ID"
|
||||
msgstr ""
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_calendar_event_type____last_update
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests____last_update
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base____last_update
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation____last_update
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_company____last_update
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_res_config_settings____last_update
|
||||
msgid "Last Modified on"
|
||||
msgstr "Dernière modification le"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__write_uid
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__write_uid
|
||||
msgid "Last Updated by"
|
||||
msgstr "Dernière mise à jour par"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__write_date
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__write_date
|
||||
msgid "Last Updated on"
|
||||
msgstr "Dernière mise à jour le"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.view_employee_form_lv
|
||||
msgid "Luncheon Vouchers"
|
||||
msgstr "Tickets Restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model,name:hr_luncheon_voucher.model_hr_lv_allocation
|
||||
msgid "Luncheon Vouchers Allocation"
|
||||
msgstr "Allocation de Tickets Restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.res_config_settings_lv_form
|
||||
msgid "Luncheon Vouchers Half-day Cancel"
|
||||
msgstr "Annulation Tickets Restaurants sur une demi-journée"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.ui.menu,name:hr_luncheon_voucher.menu_hr_lv_allocations
|
||||
msgid "Luncheon vouchers"
|
||||
msgstr "Ticket restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.actions.act_window,name:hr_luncheon_voucher.act_lv_allocations
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_search
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_tree
|
||||
msgid "Luncheon vouchers allocations"
|
||||
msgstr "Allocations de Tickets restaurant"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_form
|
||||
msgid "Luncheon vouchers calculation"
|
||||
msgstr "Calculs des tickets restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee__lv_allocations_ids
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base__lv_allocations_ids
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_public__lv_allocations_ids
|
||||
msgid "Lv Allocations"
|
||||
msgstr "Allocations TR"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__name
|
||||
msgid "Name"
|
||||
msgstr "Nom"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.lv_allocations_requests_wizard
|
||||
msgid "Period to consider"
|
||||
msgstr "Période du calcul"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_calendar_event_type__ref
|
||||
msgid "Reference"
|
||||
msgstr "Référence"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.view_employee_form_lv
|
||||
msgid "Refresh Luncheon Vouchers"
|
||||
msgstr "Rafraîchir les Tickets Restaurants"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee__dued_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base__dued_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_public__dued_lv
|
||||
msgid "Remaining luncheon vouchers"
|
||||
msgstr ""
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_calendar_event_type__remove_luncheon_voucher
|
||||
msgid "Remove luncheon voucher"
|
||||
msgstr "Supprimer le ticket"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.hr_lv_allocation_form
|
||||
msgid "Request context"
|
||||
msgstr "Contexte de la requête"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_generate_lv_allocation_requests__date_from
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__date_from
|
||||
msgid "Start Date"
|
||||
msgstr "Date de début"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__state
|
||||
msgid "Status"
|
||||
msgstr "État"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,help:hr_luncheon_voucher.field_hr_lv_allocation__state
|
||||
msgid ""
|
||||
"The status is set to 'Draft', when an allocation request is created. The "
|
||||
"status is 'Confirmed', when an allocation request is confirmed by HR "
|
||||
"manager. The status is 'Distributed', when the luncheon vouchers have been "
|
||||
"distributed."
|
||||
msgstr ""
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee__total_acquired_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_base__total_acquired_lv
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_employee_public__total_acquired_lv
|
||||
msgid "Total allocated luncheon vouchers"
|
||||
msgstr ""
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model_terms:ir.ui.view,arch_db:hr_luncheon_voucher.res_config_settings_lv_form
|
||||
msgid ""
|
||||
"Voucher is acquired only if the employee worked during all his\n"
|
||||
" attendance."
|
||||
msgstr ""
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,help:hr_luncheon_voucher.field_hr_lv_allocation__lv_balance
|
||||
msgid ""
|
||||
"Vouchers available after distribution. Dued vouchers - Distributed vouchers"
|
||||
msgstr "Tickets disponibles après distribution. = Tickets dûs - tickets distribués"
|
||||
|
||||
#. module: hr_luncheon_voucher
|
||||
#: model:ir.model.fields,field_description:hr_luncheon_voucher.field_hr_lv_allocation__lv_balance
|
||||
msgid "Vouchers balance"
|
||||
msgstr "Solde TR"
|
||||
5
hr_luncheon_voucher/models/__init__.py
Normal file
5
hr_luncheon_voucher/models/__init__.py
Normal file
@@ -0,0 +1,5 @@
|
||||
from . import calendar_event_type
|
||||
from . import hr_employee
|
||||
from . import hr_lv_allocation
|
||||
from . import res_company
|
||||
from . import res_config_settings
|
||||
16
hr_luncheon_voucher/models/calendar_event_type.py
Normal file
16
hr_luncheon_voucher/models/calendar_event_type.py
Normal file
@@ -0,0 +1,16 @@
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class MeetingType(models.Model):
|
||||
_inherit = "calendar.event.type"
|
||||
|
||||
ref = fields.Char(
|
||||
string="Reference",
|
||||
copy=False,
|
||||
store=True,
|
||||
)
|
||||
remove_luncheon_voucher = fields.Boolean(
|
||||
string="Remove luncheon voucher",
|
||||
copy=True,
|
||||
store=True,
|
||||
)
|
||||
88
hr_luncheon_voucher/models/hr_employee.py
Normal file
88
hr_luncheon_voucher/models/hr_employee.py
Normal file
@@ -0,0 +1,88 @@
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class HrEmployeeBase(models.AbstractModel):
|
||||
_inherit = "hr.employee.base"
|
||||
|
||||
lv_allocations_ids = fields.One2many("hr.lv.allocation", "employee_id")
|
||||
|
||||
total_acquired_lv = fields.Integer(
|
||||
string="Total allocated luncheon vouchers", store=True, copy=False
|
||||
)
|
||||
distributed_lv = fields.Integer(
|
||||
string="Distributed luncheon vouchers", store=True, copy=False
|
||||
)
|
||||
dued_lv = fields.Integer(
|
||||
string="Remaining luncheon vouchers", store=True, copy=False
|
||||
)
|
||||
|
||||
default_monthly_lv = fields.Integer(
|
||||
string="Default monthly distribution", store=True, copy=True
|
||||
)
|
||||
|
||||
def refresh_lv_values(self):
|
||||
for record in self:
|
||||
record._compute_total_acquired_lv()
|
||||
record._compute_distributed_lv()
|
||||
record._compute_dued_lv()
|
||||
|
||||
def _compute_total_acquired_lv(self):
|
||||
for record in self:
|
||||
allocations = self.env["hr.lv.allocation"].search(
|
||||
[
|
||||
("employee_id", "=", record.id),
|
||||
("state", "in", ["confirmed", "distributed"]),
|
||||
]
|
||||
)
|
||||
record.total_acquired_lv = sum(allocations.mapped("number_acquired_lv"))
|
||||
|
||||
def _compute_distributed_lv(self):
|
||||
for record in self:
|
||||
allocations = self.env["hr.lv.allocation"].search(
|
||||
[("employee_id", "=", record.id), ("state", "=", "distributed")]
|
||||
)
|
||||
record.distributed_lv = sum(allocations.mapped("number_distributed_lv"))
|
||||
|
||||
def _compute_dued_lv(self):
|
||||
for record in self:
|
||||
record.dued_lv = record.total_acquired_lv - record.distributed_lv
|
||||
|
||||
def generate_mass_lv_allocation(self, values):
|
||||
for record in self:
|
||||
record.generate_lv_allocation(values)
|
||||
|
||||
def generate_lv_allocation(self, values):
|
||||
self.ensure_one()
|
||||
values["employee_id"] = self.id
|
||||
values["name"] = values["distrib_campaign_name"] + " - " + self.name
|
||||
self.env["hr.lv.allocation"].create(values)
|
||||
|
||||
def action_lv_allocations(self):
|
||||
action = self.env["ir.actions.act_window"]._for_xml_id(
|
||||
"hr_luncheon_voucher.act_lv_allocations"
|
||||
)
|
||||
action["context"] = {
|
||||
"search_default_employee_id": self.id,
|
||||
"default_employee_id": self.id,
|
||||
}
|
||||
action["domain"] = [("employee_id", "=", self.id)]
|
||||
return action
|
||||
|
||||
def action_lv_allocations_requests_wizard(self):
|
||||
action = self.env["ir.actions.act_window"]._for_xml_id(
|
||||
"hr_luncheon_voucher.lv_allocations_requests_wizard_action"
|
||||
)
|
||||
ctx = dict(self.env.context)
|
||||
ctx["active_ids"] = self.ids
|
||||
action["context"] = ctx
|
||||
return action
|
||||
|
||||
def _search_display_name(self, operator, value):
|
||||
"""
|
||||
On lv allocation, allow employee search on all companies.
|
||||
"""
|
||||
if self.env.context.get("search_all_campanies"):
|
||||
return super(HrEmployeeBase, self.sudo())._search_display_name(
|
||||
operator, value
|
||||
)
|
||||
return super()._search_display_name(operator, value)
|
||||
216
hr_luncheon_voucher/models/hr_lv_allocation.py
Normal file
216
hr_luncheon_voucher/models/hr_lv_allocation.py
Normal file
@@ -0,0 +1,216 @@
|
||||
from datetime import datetime, time, timedelta
|
||||
|
||||
from dateutil.rrule import DAILY, rrule
|
||||
from pytz import UTC
|
||||
|
||||
from odoo import api, fields, models
|
||||
|
||||
|
||||
class LuncheonVouchersAllocation(models.Model):
|
||||
_name = "hr.lv.allocation"
|
||||
_description = "Luncheon Vouchers Allocation"
|
||||
|
||||
name = fields.Char()
|
||||
distrib_campaign_name = fields.Char("Distribution campaign")
|
||||
state = fields.Selection(
|
||||
[
|
||||
("draft", "Draft"),
|
||||
("confirmed", "Confirmed"),
|
||||
("distributed", "Distributed"),
|
||||
],
|
||||
string="Status",
|
||||
readonly=True,
|
||||
copy=False,
|
||||
default="draft",
|
||||
help=(
|
||||
"The status is set to 'Draft', when an allocation request is created. "
|
||||
"The status is 'Confirmed', when an allocation request is confirmed by "
|
||||
"HR manager. The status is 'Distributed', when the luncheon vouchers "
|
||||
"have been distributed."
|
||||
),
|
||||
)
|
||||
date_from = fields.Datetime(
|
||||
string="Start Date",
|
||||
store=True,
|
||||
readonly=False,
|
||||
copy=False,
|
||||
states={
|
||||
"confirmed": [("readonly", True)],
|
||||
"distributed": [("readonly", True)],
|
||||
},
|
||||
)
|
||||
date_to = fields.Datetime(
|
||||
string="End Date",
|
||||
store=True,
|
||||
readonly=False,
|
||||
copy=False,
|
||||
states={
|
||||
"confirmed": [("readonly", True)],
|
||||
"distributed": [("readonly", True)],
|
||||
},
|
||||
)
|
||||
employee_id = fields.Many2one(
|
||||
"hr.employee",
|
||||
store=True,
|
||||
string="Employee",
|
||||
index=True,
|
||||
readonly=False,
|
||||
ondelete="restrict",
|
||||
states={
|
||||
"confirmed": [("readonly", True)],
|
||||
"distributed": [("readonly", True)],
|
||||
},
|
||||
)
|
||||
number_acquired_lv = fields.Integer(
|
||||
string="Acquired Vouchers",
|
||||
store=True,
|
||||
readonly=False,
|
||||
states={
|
||||
"confirmed": [("readonly", True)],
|
||||
"distributed": [("readonly", True)],
|
||||
},
|
||||
)
|
||||
number_dued_lv = fields.Integer(
|
||||
string="Dued Vouchers",
|
||||
store=True,
|
||||
readonly=False,
|
||||
states={
|
||||
"confirmed": [("readonly", True)],
|
||||
"distributed": [("readonly", True)],
|
||||
},
|
||||
)
|
||||
number_distributed_lv = fields.Integer(
|
||||
string="Distributed Vouchers",
|
||||
store=True,
|
||||
readonly=False,
|
||||
states={
|
||||
"confirmed": [("readonly", False)],
|
||||
"distributed": [("readonly", True)],
|
||||
},
|
||||
)
|
||||
|
||||
lv_balance = fields.Integer(
|
||||
"Vouchers balance",
|
||||
compute="_compute_lv_balance",
|
||||
help="Vouchers available after distribution. Dued vouchers - Distributed.",
|
||||
)
|
||||
|
||||
@api.depends("number_dued_lv", "number_distributed_lv")
|
||||
def _compute_lv_balance(self):
|
||||
for allocation in self:
|
||||
allocation.lv_balance = (
|
||||
allocation.number_dued_lv - allocation.number_distributed_lv
|
||||
)
|
||||
|
||||
@api.model_create_multi
|
||||
def create(self, values):
|
||||
res = super().create(values)
|
||||
res._calculate_number_acquired_lv()
|
||||
res._calculate_number_dued_lv()
|
||||
res._default_number_distributed_lv()
|
||||
return res
|
||||
|
||||
@api.depends("employee_id")
|
||||
def _default_number_distributed_lv(self):
|
||||
for record in self:
|
||||
record.number_distributed_lv = record.employee_id.default_monthly_lv
|
||||
|
||||
def _has_cancelling_voucher_event(self, day):
|
||||
category_no_voucher_ids = self.env["calendar.event.type"].search(
|
||||
[("remove_luncheon_voucher", "=", True)]
|
||||
)
|
||||
events = self.env["calendar.event"].search(
|
||||
[
|
||||
("categ_ids", "in", category_no_voucher_ids.ids),
|
||||
("partner_ids", "in", self.employee_id.user_id.partner_id.id),
|
||||
]
|
||||
)
|
||||
day_start = fields.Datetime.to_datetime(day.date())
|
||||
day_end = fields.Datetime.to_datetime(day.date()) + timedelta(hours=24)
|
||||
cancelling_events = events.filtered(
|
||||
lambda x: not ((x.start < day_start) and (x.stop <= day_start))
|
||||
and not ((x.start >= day_end) and (x.stop > day_end))
|
||||
)
|
||||
if len(cancelling_events) > 0:
|
||||
return True
|
||||
else:
|
||||
return False
|
||||
|
||||
def _calculate_number_acquired_lv(self):
|
||||
nb_eligible_days = 0
|
||||
dfrom = datetime.combine(
|
||||
fields.Date.from_string(self.date_from), time.min
|
||||
).replace(tzinfo=UTC)
|
||||
dto = datetime.combine(fields.Date.from_string(self.date_to), time.max).replace(
|
||||
tzinfo=UTC
|
||||
)
|
||||
period_days = rrule(DAILY, dfrom, until=dto)
|
||||
calendar_resource = self.employee_id.resource_calendar_id
|
||||
for day in period_days:
|
||||
# Check if this days is a working day
|
||||
if not calendar_resource.is_working_day(day):
|
||||
continue
|
||||
# The employee should work this day but...
|
||||
if (
|
||||
self.env.company.hr_half_day_cancels_voucher
|
||||
and not calendar_resource.is_full_working_day(day)
|
||||
):
|
||||
# Voucher requires full-day attendance
|
||||
continue
|
||||
# Check leaves
|
||||
if not calendar_resource.is_worked_day(self.employee_id, day):
|
||||
continue
|
||||
# The employee has worked this day but...
|
||||
if (
|
||||
self.env.company.hr_half_day_cancels_voucher
|
||||
and not calendar_resource.all_attendances_worked(
|
||||
self.employee_id.resource_id, day
|
||||
)
|
||||
):
|
||||
# Voucher requires all attendances to be worked
|
||||
continue
|
||||
# Check there is no event cancelling the voucher
|
||||
if self._has_cancelling_voucher_event(day):
|
||||
continue
|
||||
# All checks passed, the days is eligible for a voucher
|
||||
nb_eligible_days += 1
|
||||
self.number_acquired_lv = nb_eligible_days
|
||||
|
||||
def _calculate_number_dued_lv(self):
|
||||
for record in self:
|
||||
if record.state == "distributed":
|
||||
record.number_dued_lv = record.employee_id.dued_lv
|
||||
else:
|
||||
record.number_dued_lv = (
|
||||
record.employee_id.dued_lv + record.number_acquired_lv
|
||||
)
|
||||
|
||||
def confirm_allocation(self):
|
||||
for record in self:
|
||||
if record.state == "draft":
|
||||
record.state = "confirmed"
|
||||
record.employee_id._compute_total_acquired_lv()
|
||||
record.employee_id._compute_dued_lv()
|
||||
|
||||
def back_to_draft(self):
|
||||
for record in self:
|
||||
if record.state in ["confirmed", "distributed"]:
|
||||
record.state = "draft"
|
||||
record.employee_id._compute_total_acquired_lv()
|
||||
record.employee_id._compute_distributed_lv()
|
||||
record.employee_id._compute_dued_lv()
|
||||
|
||||
def distribute_allocation(self):
|
||||
for record in self:
|
||||
if record.state == "confirmed":
|
||||
record.state = "distributed"
|
||||
record.employee_id._compute_distributed_lv()
|
||||
record.employee_id._compute_dued_lv()
|
||||
|
||||
def adjust_distribution(self):
|
||||
for record in self:
|
||||
for record in self:
|
||||
if record.state == "draft":
|
||||
record.number_distributed_lv = (
|
||||
record.employee_id.dued_lv + record.number_acquired_lv
|
||||
)
|
||||
9
hr_luncheon_voucher/models/res_company.py
Normal file
9
hr_luncheon_voucher/models/res_company.py
Normal file
@@ -0,0 +1,9 @@
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class Company(models.Model):
|
||||
_inherit = "res.company"
|
||||
|
||||
hr_half_day_cancels_voucher = fields.Boolean(
|
||||
string="Half working days cancel luncheon vouchers"
|
||||
)
|
||||
11
hr_luncheon_voucher/models/res_config_settings.py
Normal file
11
hr_luncheon_voucher/models/res_config_settings.py
Normal file
@@ -0,0 +1,11 @@
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class ResConfigSettings(models.TransientModel):
|
||||
_inherit = "res.config.settings"
|
||||
|
||||
hr_half_day_cancels_voucher = fields.Boolean(
|
||||
string="Half working days cancel luncheon vouchers",
|
||||
related="company_id.hr_half_day_cancels_voucher",
|
||||
readonly=False,
|
||||
)
|
||||
5
hr_luncheon_voucher/security/ir.model.access.csv
Normal file
5
hr_luncheon_voucher/security/ir.model.access.csv
Normal file
@@ -0,0 +1,5 @@
|
||||
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
||||
access_lv_allocation_user,access_lv_allocation_user,model_hr_lv_allocation,hr.group_hr_user,1,0,0,0
|
||||
access_lv_allocation_manager,access_lv_allocation_manager,model_hr_lv_allocation,hr.group_hr_manager,1,1,1,1
|
||||
access_lv_allocation_wizard_user,access_lv_allocation_wizard_user,model_generate_lv_allocation_requests,hr.group_hr_user,1,0,0,0
|
||||
access_lv_allocation_wizard_manager,access_lv_allocation_wizard_manager,model_generate_lv_allocation_requests,hr.group_hr_manager,1,1,1,1
|
||||
|
3
hr_luncheon_voucher/tests/__init__.py
Normal file
3
hr_luncheon_voucher/tests/__init__.py
Normal file
@@ -0,0 +1,3 @@
|
||||
from . import test_hr_employee
|
||||
from . import test_hr_lv_allocation
|
||||
from . import test_generate_lv_allocations_wizard
|
||||
112
hr_luncheon_voucher/tests/test_generate_lv_allocations_wizard.py
Normal file
112
hr_luncheon_voucher/tests/test_generate_lv_allocations_wizard.py
Normal file
@@ -0,0 +1,112 @@
|
||||
from odoo.tests.common import TransactionCase
|
||||
|
||||
|
||||
class TestGenerateLvAllocationsWizard(TransactionCase):
|
||||
"""Tests for generate.lv.allocation.requests wizard."""
|
||||
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
super().setUpClass()
|
||||
cls.employee_admin = cls.env.ref("hr.employee_admin")
|
||||
cls.employee_other = cls.env["hr.employee"].create(
|
||||
{
|
||||
"name": "Other Employee",
|
||||
"resource_calendar_id": cls.employee_admin.resource_calendar_id.id,
|
||||
}
|
||||
)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# generate_lv_allocations
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_generate_for_single_employee(self):
|
||||
"""Wizard creates one allocation for the selected employee."""
|
||||
wizard = self._create_wizard()
|
||||
# Simulate single employee selection
|
||||
ctx = {"active_ids": [self.employee_admin.id]}
|
||||
wizard = wizard.with_context(**ctx)
|
||||
wizard.generate_lv_allocations()
|
||||
|
||||
allocations = self.env["hr.lv.allocation"].search(
|
||||
[("distrib_campaign_name", "=", "March 2026")]
|
||||
)
|
||||
self.assertEqual(len(allocations), 1)
|
||||
self.assertEqual(allocations.employee_id, self.employee_admin)
|
||||
|
||||
def test_generate_for_multiple_employees(self):
|
||||
"""Wizard creates one allocation per selected employee."""
|
||||
wizard = self._create_wizard()
|
||||
ctx = {
|
||||
"active_ids": [
|
||||
self.employee_admin.id,
|
||||
self.employee_other.id,
|
||||
]
|
||||
}
|
||||
wizard = wizard.with_context(**ctx)
|
||||
wizard.generate_lv_allocations()
|
||||
|
||||
allocations = self.env["hr.lv.allocation"].search(
|
||||
[("distrib_campaign_name", "=", "March 2026")]
|
||||
)
|
||||
self.assertEqual(len(allocations), 2)
|
||||
|
||||
def test_generate_allocation_name_format(self):
|
||||
"""Allocation name follows 'campaign - employee' pattern."""
|
||||
wizard = self._create_wizard()
|
||||
ctx = {"active_ids": [self.employee_admin.id]}
|
||||
wizard = wizard.with_context(**ctx)
|
||||
wizard.generate_lv_allocations()
|
||||
|
||||
alloc = self.env["hr.lv.allocation"].search(
|
||||
[("distrib_campaign_name", "=", "March 2026")]
|
||||
)
|
||||
expected_name = f"March 2026 - {self.employee_admin.name}"
|
||||
self.assertEqual(alloc.name, expected_name)
|
||||
|
||||
def test_generate_allocations_have_dates(self):
|
||||
"""Allocations carry the wizard's date range."""
|
||||
wizard = self._create_wizard()
|
||||
ctx = {"active_ids": [self.employee_admin.id]}
|
||||
wizard = wizard.with_context(**ctx)
|
||||
wizard.generate_lv_allocations()
|
||||
|
||||
alloc = self.env["hr.lv.allocation"].search(
|
||||
[("distrib_campaign_name", "=", "March 2026")]
|
||||
)
|
||||
self.assertEqual(alloc.date_from, wizard.date_from)
|
||||
self.assertEqual(alloc.date_to, wizard.date_to)
|
||||
|
||||
def test_generate_allocations_start_in_draft(self):
|
||||
"""New allocations start in 'draft' state."""
|
||||
wizard = self._create_wizard()
|
||||
ctx = {"active_ids": [self.employee_admin.id]}
|
||||
wizard = wizard.with_context(**ctx)
|
||||
wizard.generate_lv_allocations()
|
||||
|
||||
alloc = self.env["hr.lv.allocation"].search(
|
||||
[("distrib_campaign_name", "=", "March 2026")]
|
||||
)
|
||||
self.assertEqual(alloc.state, "draft")
|
||||
|
||||
def test_generate_returns_action(self):
|
||||
"""The wizard returns an action dict (act_window)."""
|
||||
wizard = self._create_wizard()
|
||||
ctx = {"active_ids": [self.employee_admin.id]}
|
||||
wizard = wizard.with_context(**ctx)
|
||||
action = wizard.generate_lv_allocations()
|
||||
self.assertIn("type", action)
|
||||
self.assertEqual(action["type"], "ir.actions.act_window")
|
||||
self.assertEqual(action["res_model"], "hr.lv.allocation")
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# helpers
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def _create_wizard(self):
|
||||
return self.env["generate.lv.allocation.requests"].create(
|
||||
{
|
||||
"distrib_campaign_name": "March 2026",
|
||||
"date_from": "2026-03-01",
|
||||
"date_to": "2026-03-31",
|
||||
}
|
||||
)
|
||||
169
hr_luncheon_voucher/tests/test_hr_employee.py
Normal file
169
hr_luncheon_voucher/tests/test_hr_employee.py
Normal file
@@ -0,0 +1,169 @@
|
||||
from odoo.tests.common import TransactionCase
|
||||
|
||||
|
||||
class TestHrEmployee(TransactionCase):
|
||||
"""Tests for hr.employee extensions (_compute*, generate_lv_allocation)."""
|
||||
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
super().setUpClass()
|
||||
# Use a demo employee (created by hr module)
|
||||
cls.employee = cls.env.ref("hr.employee_admin")
|
||||
# Ensure a clean slate
|
||||
cls.employee.write(
|
||||
{
|
||||
"total_acquired_lv": 0,
|
||||
"distributed_lv": 0,
|
||||
"dued_lv": 0,
|
||||
}
|
||||
)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# _compute_total_acquired_lv
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_compute_total_acquired_lv_no_allocations(self):
|
||||
"""No allocations → total_acquired_lv is 0."""
|
||||
self.employee.refresh_lv_values()
|
||||
self.assertEqual(self.employee.total_acquired_lv, 0)
|
||||
|
||||
def test_compute_total_acquired_lv_with_allocations(self):
|
||||
"""Confirmed + distributed allocations are summed."""
|
||||
self._create_allocation(state="confirmed", number_acquired_lv=5)
|
||||
self._create_allocation(state="distributed", number_acquired_lv=3)
|
||||
self.employee.refresh_lv_values()
|
||||
self.assertEqual(self.employee.total_acquired_lv, 8)
|
||||
|
||||
def test_compute_total_acquired_lv_excludes_draft(self):
|
||||
"""Draft allocations are NOT counted."""
|
||||
self._create_allocation(state="draft", number_acquired_lv=10)
|
||||
self.employee.refresh_lv_values()
|
||||
self.assertEqual(self.employee.total_acquired_lv, 0)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# _compute_distributed_lv
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_compute_distributed_lv_no_allocations(self):
|
||||
"""No allocations → distributed_lv is 0."""
|
||||
self.employee.refresh_lv_values()
|
||||
self.assertEqual(self.employee.distributed_lv, 0)
|
||||
|
||||
def test_compute_distributed_lv_only_distributed(self):
|
||||
"""Only 'distributed' state allocations are counted."""
|
||||
self._create_allocation(state="distributed", number_distributed_lv=4)
|
||||
self._create_allocation(state="confirmed", number_distributed_lv=2)
|
||||
self._create_allocation(state="draft", number_distributed_lv=1)
|
||||
self.employee.refresh_lv_values()
|
||||
self.assertEqual(self.employee.distributed_lv, 4)
|
||||
|
||||
def test_compute_distributed_lv_multiple_distributed(self):
|
||||
"""Multiple distributed allocations are summed."""
|
||||
self._create_allocation(state="distributed", number_distributed_lv=3)
|
||||
self._create_allocation(state="distributed", number_distributed_lv=7)
|
||||
self.employee.refresh_lv_values()
|
||||
self.assertEqual(self.employee.distributed_lv, 10)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# _compute_dued_lv
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_compute_dued_lv(self):
|
||||
"""dued_lv = total_acquired_lv - distributed_lv."""
|
||||
self.employee.write(
|
||||
{"total_acquired_lv": 20, "distributed_lv": 6}
|
||||
)
|
||||
self.employee._compute_dued_lv()
|
||||
self.assertEqual(self.employee.dued_lv, 14)
|
||||
|
||||
def test_compute_dued_lv_zero_when_equal(self):
|
||||
"""dued_lv is 0 when acquired equals distributed."""
|
||||
self.employee.write(
|
||||
{"total_acquired_lv": 10, "distributed_lv": 10}
|
||||
)
|
||||
self.employee._compute_dued_lv()
|
||||
self.assertEqual(self.employee.dued_lv, 0)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# generate_lv_allocation
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_generate_lv_allocation_creates_record(self):
|
||||
"""generate_lv_allocation creates an hr.lv.allocation with correct values."""
|
||||
campaign_name = "Test Campaign"
|
||||
values = {
|
||||
"distrib_campaign_name": campaign_name,
|
||||
"date_from": "2026-01-01",
|
||||
"date_to": "2026-01-31",
|
||||
}
|
||||
self.employee.generate_lv_allocation(values)
|
||||
alloc = self.env["hr.lv.allocation"].search(
|
||||
[
|
||||
("employee_id", "=", self.employee.id),
|
||||
("distrib_campaign_name", "=", campaign_name),
|
||||
]
|
||||
)
|
||||
self.assertTrue(alloc)
|
||||
self.assertEqual(len(alloc), 1)
|
||||
self.assertEqual(alloc.distrib_campaign_name, campaign_name)
|
||||
self.assertEqual(alloc.employee_id, self.employee)
|
||||
self.assertIn(self.employee.name, alloc.name)
|
||||
|
||||
def test_generate_lv_allocation_sets_employee_id(self):
|
||||
"""The passed employee_id in values is overridden by the record's own id."""
|
||||
other = self.env["hr.employee"].create(
|
||||
{
|
||||
"name": "Other Employee",
|
||||
"resource_calendar_id": self.employee.resource_calendar_id.id,
|
||||
}
|
||||
)
|
||||
values = {
|
||||
"distrib_campaign_name": "Campaign Override",
|
||||
"employee_id": other.id,
|
||||
"date_from": "2026-02-01",
|
||||
"date_to": "2026-02-28",
|
||||
}
|
||||
self.employee.generate_lv_allocation(values)
|
||||
alloc = self.env["hr.lv.allocation"].search(
|
||||
[("distrib_campaign_name", "=", "Campaign Override")]
|
||||
)
|
||||
self.assertEqual(alloc.employee_id, self.employee)
|
||||
|
||||
def test_generate_lv_allocation_updates_counters(self):
|
||||
"""After creation, acquired and due are computed (triggers create logic)."""
|
||||
values = {
|
||||
"distrib_campaign_name": "Counters Check",
|
||||
"date_from": "2026-03-01",
|
||||
"date_to": "2026-03-31",
|
||||
}
|
||||
self.employee.generate_lv_allocation(values)
|
||||
alloc = self.env["hr.lv.allocation"].search(
|
||||
[("distrib_campaign_name", "=", "Counters Check")]
|
||||
)
|
||||
# number_acquired_lv should be computed (may be 0 on non-working days)
|
||||
self.assertIsNotNone(alloc.number_acquired_lv)
|
||||
self.assertIsNotNone(alloc.number_dued_lv)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# helpers
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def _create_allocation(self, **kwargs):
|
||||
# Extract numeric fields that create() overrides, to set them after creation
|
||||
numeric_fields = {
|
||||
k: kwargs.pop(k)
|
||||
for k in ["number_acquired_lv", "number_dued_lv", "number_distributed_lv"]
|
||||
if k in kwargs
|
||||
}
|
||||
defaults = {
|
||||
"employee_id": self.employee.id,
|
||||
"distrib_campaign_name": "Test",
|
||||
"date_from": "2026-01-01",
|
||||
"date_to": "2026-01-31",
|
||||
"name": "Test - Employee",
|
||||
}
|
||||
defaults.update(kwargs)
|
||||
alloc = self.env["hr.lv.allocation"].create(defaults)
|
||||
if numeric_fields:
|
||||
alloc.write(numeric_fields)
|
||||
return alloc
|
||||
317
hr_luncheon_voucher/tests/test_hr_lv_allocation.py
Normal file
317
hr_luncheon_voucher/tests/test_hr_lv_allocation.py
Normal file
@@ -0,0 +1,317 @@
|
||||
from odoo.tests.common import TransactionCase
|
||||
|
||||
|
||||
class TestHrLvAllocation(TransactionCase):
|
||||
"""Tests for hr.lv.allocation model methods."""
|
||||
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
super().setUpClass()
|
||||
cls.employee = cls.env.ref("hr.employee_admin")
|
||||
cls.calendar = cls.env.ref("resource.resource_calendar_std")
|
||||
cls.employee.resource_calendar_id = cls.calendar
|
||||
|
||||
# Ensure the calendar has attendances marked as effective periods
|
||||
# (required by hr_effective_attendance_period for is_working_day etc.)
|
||||
Attendance = cls.env["resource.calendar.attendance"]
|
||||
existing = Attendance.search([("calendar_id", "=", cls.calendar.id)])
|
||||
if existing:
|
||||
existing.write({"effective_attendance_period": True})
|
||||
else:
|
||||
for day_idx, day_name in enumerate(
|
||||
["Monday", "Tuesday", "Wednesday", "Thursday", "Friday"]
|
||||
):
|
||||
Attendance.create(
|
||||
{
|
||||
"calendar_id": cls.calendar.id,
|
||||
"name": f"{day_name} Morning",
|
||||
"dayofweek": str(day_idx),
|
||||
"day_period": "morning",
|
||||
"hour_from": 8,
|
||||
"hour_to": 12,
|
||||
"effective_attendance_period": True,
|
||||
}
|
||||
)
|
||||
Attendance.create(
|
||||
{
|
||||
"calendar_id": cls.calendar.id,
|
||||
"name": f"{day_name} Afternoon",
|
||||
"dayofweek": str(day_idx),
|
||||
"day_period": "afternoon",
|
||||
"hour_from": 13,
|
||||
"hour_to": 17,
|
||||
"effective_attendance_period": True,
|
||||
}
|
||||
)
|
||||
|
||||
cls.allocation = cls._build_allocation(cls)
|
||||
|
||||
# Ensure a cancelling event type exists
|
||||
cls.cancelling_categ = cls.env.ref(
|
||||
"hr_luncheon_voucher.categ_meet_free_lunch",
|
||||
raise_if_not_found=False,
|
||||
)
|
||||
if not cls.cancelling_categ:
|
||||
cls.cancelling_categ = cls.env["calendar.event.type"].create(
|
||||
{"name": "Free Lunch Test", "remove_luncheon_voucher": True}
|
||||
)
|
||||
else:
|
||||
cls.cancelling_categ.remove_luncheon_voucher = True
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# Helpers
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def _build_allocation(self, **overrides):
|
||||
vals = {
|
||||
"employee_id": self.employee.id,
|
||||
"distrib_campaign_name": "Test Campaign",
|
||||
"date_from": "2026-01-01",
|
||||
"date_to": "2026-01-31",
|
||||
"name": "Test Campaign - Admin",
|
||||
}
|
||||
vals.update(overrides)
|
||||
return self.env["hr.lv.allocation"].create(vals)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# _compute_lv_balance
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_lv_balance_positive(self):
|
||||
"""Balance = number_dued_lv - number_distributed_lv (positive)."""
|
||||
alloc = self._build_allocation()
|
||||
alloc.write(
|
||||
{"number_dued_lv": 10, "number_distributed_lv": 3}
|
||||
)
|
||||
alloc._compute_lv_balance()
|
||||
self.assertEqual(alloc.lv_balance, 7)
|
||||
|
||||
def test_lv_balance_zero(self):
|
||||
"""Balance is 0 when due equals distributed."""
|
||||
alloc = self._build_allocation()
|
||||
alloc.write(
|
||||
{"number_dued_lv": 5, "number_distributed_lv": 5}
|
||||
)
|
||||
alloc._compute_lv_balance()
|
||||
self.assertEqual(alloc.lv_balance, 0)
|
||||
|
||||
def test_lv_balance_negative(self):
|
||||
"""Balance can be negative (over-distribution)."""
|
||||
alloc = self._build_allocation()
|
||||
alloc.write(
|
||||
{"number_dued_lv": 2, "number_distributed_lv": 5}
|
||||
)
|
||||
alloc._compute_lv_balance()
|
||||
self.assertEqual(alloc.lv_balance, -3)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# _has_cancelling_voucher_event
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_has_cancelling_event_true(self):
|
||||
"""Returns True when a cancelling event exists for the employee."""
|
||||
partner = self.employee.user_id.partner_id
|
||||
self.env["calendar.event"].create(
|
||||
{
|
||||
"name": "Team Lunch",
|
||||
"start": "2026-01-15 12:00:00",
|
||||
"stop": "2026-01-15 14:00:00",
|
||||
"categ_ids": [(6, 0, self.cancelling_categ.ids)],
|
||||
"partner_ids": [(4, partner.id)],
|
||||
}
|
||||
)
|
||||
day = self.env.cr.now().replace(day=15, month=1, year=2026)
|
||||
self.assertTrue(self.allocation._has_cancelling_voucher_event(day))
|
||||
|
||||
def test_has_cancelling_event_false_no_event(self):
|
||||
"""Returns False when no cancelling event exists."""
|
||||
day = self.env.cr.now().replace(day=10, month=1, year=2026)
|
||||
self.assertFalse(self.allocation._has_cancelling_voucher_event(day))
|
||||
|
||||
def test_has_cancelling_event_false_non_cancelling_category(self):
|
||||
"""Returns False when events exist but without the cancelling flag."""
|
||||
categ = self.env["calendar.event.type"].create(
|
||||
{"name": "Regular Meeting", "remove_luncheon_voucher": False}
|
||||
)
|
||||
partner = self.employee.user_id.partner_id
|
||||
self.env["calendar.event"].create(
|
||||
{
|
||||
"name": "Stand-up",
|
||||
"start": "2026-01-15 09:00:00",
|
||||
"stop": "2026-01-15 09:30:00",
|
||||
"categ_ids": [(6, 0, categ.ids)],
|
||||
"partner_ids": [(4, partner.id)],
|
||||
}
|
||||
)
|
||||
day = self.env.cr.now().replace(day=15, month=1, year=2026)
|
||||
self.assertFalse(self.allocation._has_cancelling_voucher_event(day))
|
||||
|
||||
def test_has_cancelling_event_outside_day(self):
|
||||
"""Returns False when the event is entirely outside the target day."""
|
||||
partner = self.employee.user_id.partner_id
|
||||
self.env["calendar.event"].create(
|
||||
{
|
||||
"name": "Late Dinner",
|
||||
"start": "2026-01-15 22:00:00",
|
||||
"stop": "2026-01-16 00:30:00",
|
||||
"categ_ids": [(6, 0, self.cancelling_categ.ids)],
|
||||
"partner_ids": [(4, partner.id)],
|
||||
}
|
||||
)
|
||||
# Check a different day
|
||||
day = self.env.cr.now().replace(day=14, month=1, year=2026)
|
||||
self.assertFalse(self.allocation._has_cancelling_voucher_event(day))
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# _calculate_number_acquired_lv
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_calculate_acquired_on_non_working_day(self):
|
||||
"""Acquired is 0 for a period of non-working days (e.g. weekend-only)."""
|
||||
alloc = self._build_allocation(
|
||||
date_from="2026-01-03", # Saturday
|
||||
date_to="2026-01-04", # Sunday
|
||||
)
|
||||
alloc._calculate_number_acquired_lv()
|
||||
self.assertEqual(alloc.number_acquired_lv, 0)
|
||||
|
||||
def test_calculate_acquired_basic(self):
|
||||
"""Acquired is > 0 for a normal working period (employee works)."""
|
||||
alloc = self._build_allocation(
|
||||
date_from="2026-01-05", # Monday
|
||||
date_to="2026-01-09", # Friday
|
||||
)
|
||||
alloc._calculate_number_acquired_lv()
|
||||
# Standard 5-day calendar: 5 working days, no leaves, no events
|
||||
self.assertEqual(alloc.number_acquired_lv, 5)
|
||||
|
||||
def test_calculate_acquired_with_leave(self):
|
||||
"""Acquired excludes days covered by leave."""
|
||||
# Create a leave covering Wednesday 2026-01-07
|
||||
self.env["resource.calendar.leaves"].create(
|
||||
{
|
||||
"name": "Sick leave",
|
||||
"date_from": "2026-01-07 00:00:00",
|
||||
"date_to": "2026-01-07 23:59:59",
|
||||
"calendar_id": self.calendar.id,
|
||||
"resource_id": self.employee.resource_id.id,
|
||||
}
|
||||
)
|
||||
alloc = self._build_allocation(
|
||||
date_from="2026-01-05", # Monday
|
||||
date_to="2026-01-09", # Friday
|
||||
)
|
||||
alloc._calculate_number_acquired_lv()
|
||||
# 5 working days - 1 leave = 4
|
||||
self.assertEqual(alloc.number_acquired_lv, 4)
|
||||
|
||||
def test_calculate_acquired_with_half_day_cancel(self):
|
||||
"""When hr_half_day_cancels_voucher is enabled, partial days are excluded."""
|
||||
self.env.company.hr_half_day_cancels_voucher = True
|
||||
# Modify calendar to have only morning attendance on Wednesday
|
||||
wed_attendance = self.env["resource.calendar.attendance"].search(
|
||||
[
|
||||
("calendar_id", "=", self.calendar.id),
|
||||
("dayofweek", "=", "2"), # Wednesday
|
||||
("day_period", "=", "afternoon"),
|
||||
]
|
||||
)
|
||||
wed_attendance.unlink()
|
||||
|
||||
alloc = self._build_allocation(
|
||||
date_from="2026-01-05", # Monday
|
||||
date_to="2026-01-09", # Friday
|
||||
)
|
||||
alloc._calculate_number_acquired_lv()
|
||||
# 4 full days + 1 half-day excluded = 4
|
||||
self.assertEqual(alloc.number_acquired_lv, 4)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# _calculate_number_dued_lv
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_calculate_dued_draft(self):
|
||||
"""In non-distributed state: dued = employee.dued_lv + number_acquired_lv."""
|
||||
self.employee.write({"dued_lv": 5})
|
||||
alloc = self._build_allocation(state="draft")
|
||||
alloc.number_acquired_lv = 10
|
||||
alloc._calculate_number_dued_lv()
|
||||
self.assertEqual(alloc.number_dued_lv, 15)
|
||||
|
||||
def test_calculate_dued_distributed(self):
|
||||
"""In distributed: dued = employee.dued_lv only (no addition)."""
|
||||
self.employee.write({"dued_lv": 8})
|
||||
alloc = self._build_allocation(state="distributed")
|
||||
alloc.number_acquired_lv = 20
|
||||
alloc._calculate_number_dued_lv()
|
||||
self.assertEqual(alloc.number_dued_lv, 8)
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# confirm_allocation
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_confirm_draft(self):
|
||||
"""confirm_allocation transitions draft → confirmed and updates counters."""
|
||||
self.employee.write({"total_acquired_lv": 0, "dued_lv": 0})
|
||||
alloc = self._build_allocation(state="draft")
|
||||
alloc.number_acquired_lv = 6
|
||||
alloc.confirm_allocation()
|
||||
self.assertEqual(alloc.state, "confirmed")
|
||||
self.employee.refresh_lv_values()
|
||||
self.assertEqual(self.employee.total_acquired_lv, 6)
|
||||
|
||||
def test_confirm_already_confirmed_does_not_change(self):
|
||||
"""confirm_allocation on a confirmed record is a no-op."""
|
||||
alloc = self._build_allocation(state="confirmed")
|
||||
alloc.confirm_allocation()
|
||||
self.assertEqual(alloc.state, "confirmed")
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# back_to_draft
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_back_to_draft_from_confirmed(self):
|
||||
"""back_to_draft transitions confirmed → draft."""
|
||||
alloc = self._build_allocation(state="confirmed")
|
||||
alloc.back_to_draft()
|
||||
self.assertEqual(alloc.state, "draft")
|
||||
|
||||
def test_back_to_draft_from_distributed(self):
|
||||
"""back_to_draft transitions distributed → draft."""
|
||||
alloc = self._build_allocation(state="distributed")
|
||||
alloc.back_to_draft()
|
||||
self.assertEqual(alloc.state, "draft")
|
||||
|
||||
def test_back_to_draft_draft_is_noop(self):
|
||||
"""back_to_draft on a draft record is a no-op."""
|
||||
alloc = self._build_allocation(state="draft")
|
||||
alloc.back_to_draft()
|
||||
self.assertEqual(alloc.state, "draft")
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# distribute_allocation
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_distribute_confirmed(self):
|
||||
"""distribute_allocation transitions confirmed → distributed."""
|
||||
alloc = self._build_allocation(state="confirmed")
|
||||
alloc.distribute_allocation()
|
||||
self.assertEqual(alloc.state, "distributed")
|
||||
|
||||
def test_distribute_draft_is_noop(self):
|
||||
"""distribute_allocation on draft is a no-op."""
|
||||
alloc = self._build_allocation(state="draft")
|
||||
alloc.distribute_allocation()
|
||||
self.assertEqual(alloc.state, "draft")
|
||||
|
||||
# ------------------------------------------------------------------
|
||||
# adjust_distribution
|
||||
# ------------------------------------------------------------------
|
||||
|
||||
def test_adjust_distribution(self):
|
||||
"""Adjust distribution sets number_distributed_lv = dued + acquired."""
|
||||
alloc = self._build_allocation(state="draft")
|
||||
alloc.number_acquired_lv = 5
|
||||
alloc.employee_id.dued_lv = 3
|
||||
alloc.adjust_distribution()
|
||||
self.assertEqual(alloc.number_distributed_lv, 8)
|
||||
13
hr_luncheon_voucher/views/event_type.xml
Normal file
13
hr_luncheon_voucher/views/event_type.xml
Normal file
@@ -0,0 +1,13 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
<record id="view_event_type_luncheonvoucher_tree" model="ir.ui.view">
|
||||
<field name="name">event.type.luncheonvoucher</field>
|
||||
<field name="model">calendar.event.type</field>
|
||||
<field name="inherit_id" ref="calendar.view_calendar_event_type_tree" />
|
||||
<field name="arch" type="xml">
|
||||
<xpath expr="//field[@name='name']" position="after">
|
||||
<field name="remove_luncheon_voucher" />
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
</odoo>
|
||||
78
hr_luncheon_voucher/views/hr_employee_views.xml
Normal file
78
hr_luncheon_voucher/views/hr_employee_views.xml
Normal file
@@ -0,0 +1,78 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
|
||||
<record id="view_employee_form_lv" model="ir.ui.view">
|
||||
<field name="name">hr.employee.form.lv</field>
|
||||
<field name="model">hr.employee</field>
|
||||
<field name="inherit_id" ref="hr.view_employee_form" />
|
||||
<field name="arch" type="xml">
|
||||
<header position="inside">
|
||||
<button
|
||||
type="object"
|
||||
name="refresh_lv_values"
|
||||
string="Refresh Luncheon Vouchers"
|
||||
class="btn-secundary"
|
||||
/>
|
||||
</header>
|
||||
<xpath expr="//div[@name='button_box']" position="inside">
|
||||
<button
|
||||
type="object"
|
||||
name="action_lv_allocations"
|
||||
class="oe_stat_button"
|
||||
icon="fa-ticket"
|
||||
>
|
||||
<field
|
||||
name="total_acquired_lv"
|
||||
widget="statinfo"
|
||||
string="Acquired"
|
||||
/>
|
||||
</button>
|
||||
<button
|
||||
type="object"
|
||||
name="action_lv_allocations"
|
||||
class="oe_stat_button"
|
||||
icon="fa-ticket"
|
||||
>
|
||||
<field
|
||||
name="distributed_lv"
|
||||
widget="statinfo"
|
||||
string="Distributed"
|
||||
/>
|
||||
</button>
|
||||
<button
|
||||
type="object"
|
||||
name="action_lv_allocations"
|
||||
class="oe_stat_button"
|
||||
icon="fa-ticket"
|
||||
>
|
||||
<field name="dued_lv" widget="statinfo" string="Dued" />
|
||||
</button>
|
||||
</xpath>
|
||||
<xpath expr="//page[@name='hr_settings']/group" position="inside">
|
||||
<group name="luncheon_vouchers" string="Luncheon Vouchers">
|
||||
<field name="default_monthly_lv" />
|
||||
<field name="lv_allocations_ids" />
|
||||
</group>
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="view_employee_tree_lv" model="ir.ui.view">
|
||||
<field name="name">view_employee_tree_lv</field>
|
||||
<field name="model">hr.employee</field>
|
||||
<field name="inherit_id" ref="hr.view_employee_tree" />
|
||||
<field name="arch" type="xml">
|
||||
<xpath expr="//list" position="inside">
|
||||
<header>
|
||||
<button
|
||||
type="object"
|
||||
name="action_lv_allocations_requests_wizard"
|
||||
string="Generate Luncheon Vouchers Allocations"
|
||||
class="btn-primary"
|
||||
/>
|
||||
</header>
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
</odoo>
|
||||
136
hr_luncheon_voucher/views/hr_lv_allocation_views.xml
Normal file
136
hr_luncheon_voucher/views/hr_lv_allocation_views.xml
Normal file
@@ -0,0 +1,136 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
|
||||
<record id="hr_lv_allocation_search" model="ir.ui.view">
|
||||
<field name="name">hr.lv.allocation.search</field>
|
||||
<field name="model">hr.lv.allocation</field>
|
||||
<field name="arch" type="xml">
|
||||
<search string="Luncheon vouchers allocations">
|
||||
<field name="state" />
|
||||
<field name="employee_id" context="{'search_all_campanies':True}" />
|
||||
<field name="distrib_campaign_name" />
|
||||
<filter
|
||||
name="draft"
|
||||
string="Draft"
|
||||
domain="[('state', '=', 'draft')]"
|
||||
/>
|
||||
<filter
|
||||
name="confirmed"
|
||||
string="Confirmed"
|
||||
domain="[('state', '=', 'confirmed')]"
|
||||
/>
|
||||
<filter
|
||||
name="distributed"
|
||||
string="Distributed"
|
||||
domain="[('state', '=', 'distributed')]"
|
||||
/>
|
||||
</search>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="hr_lv_allocation_tree" model="ir.ui.view">
|
||||
<field name="name">hr.lv.allocation.list</field>
|
||||
<field name="model">hr.lv.allocation</field>
|
||||
<field name="arch" type="xml">
|
||||
<list>
|
||||
<header>
|
||||
<button
|
||||
type="object"
|
||||
name="confirm_allocation"
|
||||
string="Confirm"
|
||||
class="btn-primary"
|
||||
/>
|
||||
<button
|
||||
type="object"
|
||||
name="distribute_allocation"
|
||||
string="Distribute Vouchers"
|
||||
class="btn-primary"
|
||||
/>
|
||||
<button
|
||||
type="object"
|
||||
name="back_to_draft"
|
||||
string="Back to draft"
|
||||
class="btn-secundary"
|
||||
/>
|
||||
<button
|
||||
type="object"
|
||||
name="adjust_distribution"
|
||||
string="Adjust distribution"
|
||||
class="btn-secundary"
|
||||
/>
|
||||
</header>
|
||||
<field name="distrib_campaign_name" />
|
||||
<field name="employee_id" />
|
||||
<field name="state" />
|
||||
<field name="date_from" widget="date" />
|
||||
<field name="date_to" widget="date" />
|
||||
<field name="number_acquired_lv" />
|
||||
<field name="number_dued_lv" />
|
||||
<field name="number_distributed_lv" />
|
||||
<field name="lv_balance" />
|
||||
</list>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="hr_lv_allocation_form" model="ir.ui.view">
|
||||
<field name="name">hr.lv.allocation.form</field>
|
||||
<field name="model">hr.lv.allocation</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="">
|
||||
<header>
|
||||
<button
|
||||
type="object"
|
||||
name="confirm_allocation"
|
||||
string="Confirm"
|
||||
class="btn-primary"
|
||||
invisible="state != 'draft'"
|
||||
/>
|
||||
<button
|
||||
type="object"
|
||||
name="distribute_allocation"
|
||||
string="Distribute Vouchers"
|
||||
class="btn-primary"
|
||||
invisible="state != 'confirmed'"
|
||||
/>
|
||||
<button
|
||||
type="object"
|
||||
name="back_to_draft"
|
||||
string="Back to draft"
|
||||
class="btn-secundary"
|
||||
invisible="state == 'draft'"
|
||||
/>
|
||||
<button
|
||||
type="object"
|
||||
name="adjust_distribution"
|
||||
string="Adjust distribution"
|
||||
class="btn-secundary"
|
||||
invisible="state != 'draft'"
|
||||
/>
|
||||
</header>
|
||||
<sheet>
|
||||
<h1>
|
||||
<field name="name" readonly="1" />
|
||||
</h1>
|
||||
<group string="Request context">
|
||||
<field name="distrib_campaign_name" />
|
||||
<field name="employee_id" />
|
||||
<field name="state" />
|
||||
<field name="date_from" widget="date" />
|
||||
<field name="date_to" widget="date" />
|
||||
</group>
|
||||
<group string="Luncheon vouchers calculation">
|
||||
<field name="number_acquired_lv" />
|
||||
<field name="number_dued_lv" />
|
||||
<field name="number_distributed_lv" />
|
||||
</group>
|
||||
</sheet>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="act_lv_allocations" model="ir.actions.act_window">
|
||||
<field name="name">Luncheon vouchers allocations</field>
|
||||
<field name="res_model">hr.lv.allocation</field>
|
||||
<field name="view_mode">list,form</field>
|
||||
</record>
|
||||
</odoo>
|
||||
12
hr_luncheon_voucher/views/menus.xml
Normal file
12
hr_luncheon_voucher/views/menus.xml
Normal file
@@ -0,0 +1,12 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
|
||||
<menuitem
|
||||
id="menu_hr_lv_allocations"
|
||||
action="act_lv_allocations"
|
||||
parent="hr.menu_hr_employee_payroll"
|
||||
sequence="1"
|
||||
name="Luncheon vouchers"
|
||||
/>
|
||||
|
||||
</odoo>
|
||||
35
hr_luncheon_voucher/views/res_config_settings_views.xml
Normal file
35
hr_luncheon_voucher/views/res_config_settings_views.xml
Normal file
@@ -0,0 +1,35 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
<record id="res_config_settings_lv_form" model="ir.ui.view">
|
||||
<field name="name">res.config.settings.view.form.inherit.lv</field>
|
||||
<field name="model">res.config.settings</field>
|
||||
<field name="priority" eval="99" />
|
||||
<field name="inherit_id" ref="hr.res_config_settings_view_form" />
|
||||
<field name="arch" type="xml">
|
||||
<xpath
|
||||
expr="//block[@name='employee_rights_setting_container']/.."
|
||||
position="inside"
|
||||
>
|
||||
<h2>Employee Luncheon Vouchers</h2>
|
||||
<div class="row mt16 o_settings_container" name="employee_lv_container">
|
||||
<div
|
||||
class="col-12 col-lg-6 o_setting_box"
|
||||
id="employee_lv_halfday_cancel"
|
||||
title="Luncheon Vouchers Half-day Cancel"
|
||||
>
|
||||
<div class="o_setting_left_pane">
|
||||
<field name="hr_half_day_cancels_voucher" />
|
||||
</div>
|
||||
<div class="o_setting_right_pane">
|
||||
<label for="hr_half_day_cancels_voucher" />
|
||||
<div class="text-muted" name="hr_presence_options_advanced">
|
||||
Voucher is acquired only if the employee worked during all his
|
||||
attendance.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
</odoo>
|
||||
1
hr_luncheon_voucher/wizard/__init__.py
Normal file
1
hr_luncheon_voucher/wizard/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from . import generate_lv_allocations_wizard
|
||||
26
hr_luncheon_voucher/wizard/generate_lv_allocations_wizard.py
Normal file
26
hr_luncheon_voucher/wizard/generate_lv_allocations_wizard.py
Normal file
@@ -0,0 +1,26 @@
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class GenerateLVAllocationRequests(models.TransientModel):
|
||||
_name = "generate.lv.allocation.requests"
|
||||
_description = "Generate Luncheon Vouchers Allocations Requests"
|
||||
|
||||
distrib_campaign_name = fields.Char("Distribution campaign", required=True)
|
||||
date_from = fields.Datetime(string="Start Date", required=True)
|
||||
date_to = fields.Datetime(string="End Date", required=True)
|
||||
|
||||
def generate_lv_allocations(self):
|
||||
values = {}
|
||||
values["distrib_campaign_name"] = self.distrib_campaign_name
|
||||
values["date_from"] = self.date_from
|
||||
values["date_to"] = self.date_to
|
||||
employees = self.env["hr.employee"].search(
|
||||
[
|
||||
("id", "in", self.env.context.get("active_ids")),
|
||||
]
|
||||
)
|
||||
employees.generate_mass_lv_allocation(values)
|
||||
# Open lv allocation tree view
|
||||
return self.env["ir.actions.act_window"]._for_xml_id(
|
||||
"hr_luncheon_voucher.act_lv_allocations"
|
||||
)
|
||||
@@ -0,0 +1,42 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
<record id="lv_allocations_requests_wizard" model="ir.ui.view">
|
||||
<field name="name">lv.allocations.requests.wizard</field>
|
||||
<field name="model">generate.lv.allocation.requests</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="Create Luncheon Vouchers allocations requests">
|
||||
<group name="dates" string="Period to consider">
|
||||
<group>
|
||||
<field name="distrib_campaign_name" />
|
||||
<field name="date_from" widget="date" />
|
||||
<field name="date_to" widget="date" />
|
||||
</group>
|
||||
</group>
|
||||
<footer>
|
||||
<button
|
||||
string="Create allocations requests"
|
||||
name="generate_lv_allocations"
|
||||
type="object"
|
||||
class="btn-primary"
|
||||
/>
|
||||
<button
|
||||
string="Cancel"
|
||||
class="btn-secondary"
|
||||
special="cancel"
|
||||
/>
|
||||
</footer>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record
|
||||
id="lv_allocations_requests_wizard_action"
|
||||
model="ir.actions.act_window"
|
||||
>
|
||||
<field name="name">Create Luncheon Vouchers allocations requests</field>
|
||||
<field name="res_model">generate.lv.allocation.requests</field>
|
||||
<field name="view_mode">form</field>
|
||||
<field name="view_id" ref="lv_allocations_requests_wizard" />
|
||||
<field name="target">new</field>
|
||||
</record>
|
||||
</odoo>
|
||||
Reference in New Issue
Block a user