Skip to content

Commit 5d2f151

Browse files
arpr-odootivisse
authored andcommitted
[IMP] l10n_in_hr_holidays: add sandwich leaves to l10n_in
This commit, adds the Sandwich Leave Rule in Indian Localization. The sandwich leave rule will be applied if an Employee takes leave like in the scenario below. Scene 1: Friday (leave) -> Saturday -> Sunday -> Monday(Leave) 4 Leaves will be deducted from the Employee's leave balance. Scene 2: Tuesday(Leave) -> Wednesday(Public Holiday) -> Thursday (Leave) 3 Leaves will be deducted from the Employee's leave balance. task-3950973 Part-of: odoo#174933 Signed-off-by: Yannick Tivisse (yti) <[email protected]>
1 parent 56f1e48 commit 5d2f151

9 files changed

+206
-0
lines changed
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# Part of Odoo. See LICENSE file for full copyright and licensing details.
2+
3+
from . import models
+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
# Part of Odoo. See LICENSE file for full copyright and licensing details.
2+
3+
{
4+
'name': 'India - Time Off',
5+
'version': '1.0',
6+
'category': 'Human Resources/Time Off',
7+
'summary': 'Leave Management of Indian Localization',
8+
'countries': ['in'],
9+
'depends': ['hr_holidays'],
10+
'auto_install': ['hr_holidays'],
11+
'data': [
12+
'views/hr_leave_views.xml',
13+
'views/hr_leave_type_views.xml',
14+
],
15+
'license': 'LGPL-3',
16+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
# Part of Odoo. See LICENSE file for full copyright and licensing details.
2+
3+
from . import hr_leave
4+
from . import hr_leave_type
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
# Part of Odoo. See LICENSE file for full copyright and licensing details.
2+
3+
from datetime import datetime, timedelta
4+
5+
from odoo import models, fields
6+
7+
8+
class HolidaysRequest(models.Model):
9+
_inherit = "hr.leave"
10+
11+
l10n_in_contains_sandwich_leaves = fields.Boolean()
12+
13+
def _l10n_in_apply_sandwich_rule(self, public_holidays, employee_leaves):
14+
self.ensure_one()
15+
if not self.request_date_from or not self.request_date_to:
16+
return
17+
date_from = self.request_date_from
18+
date_to = self.request_date_to
19+
total_leaves = (self.request_date_to - self.request_date_from).days + 1
20+
21+
def is_non_working_day(calendar, date):
22+
return not calendar._works_on_date(date) or any(
23+
datetime.date(holiday['date_from']) <= date <= datetime.date(holiday['date_to']) for holiday in public_holidays
24+
)
25+
26+
def count_sandwich_days(calendar, date, direction):
27+
current_date = date + timedelta(days=direction)
28+
days_count = 0
29+
while is_non_working_day(calendar, current_date):
30+
days_count += 1
31+
current_date += timedelta(days=direction)
32+
for leave in employee_leaves:
33+
if leave['request_date_from'] <= current_date <= leave['request_date_to']:
34+
return days_count
35+
return 0
36+
37+
calendar = self.resource_calendar_id
38+
total_leaves += count_sandwich_days(calendar, date_from, -1) + count_sandwich_days(calendar, date_to, 1)
39+
if is_non_working_day(calendar, date_from):
40+
total_leaves -= 1
41+
if is_non_working_day(calendar, date_from + timedelta(days=+1)):
42+
total_leaves -= 1
43+
if is_non_working_day(calendar, date_to):
44+
total_leaves -= 1
45+
if is_non_working_day(calendar, date_to + timedelta(days=-1)):
46+
total_leaves -= 1
47+
return total_leaves
48+
49+
def _get_durations(self, check_leave_type=True, resource_calendar=None):
50+
result = super()._get_durations(check_leave_type, resource_calendar)
51+
indian_leaves = self.filtered(lambda c: c.company_id.country_id.code == 'IN')
52+
if not indian_leaves:
53+
return result
54+
55+
public_holidays = self.env['resource.calendar.leaves'].search([
56+
('resource_id', '=', False),
57+
('company_id', 'in', indian_leaves.company_id.ids),
58+
])
59+
leaves_by_employee = dict(self._read_group(
60+
domain=[
61+
('id', 'not in', self.ids),
62+
('employee_id', 'in', self.employee_id.ids),
63+
('state', 'not in', ['cancel', 'refuse']),
64+
('leave_type_request_unit', '=', 'day'),
65+
],
66+
groupby=['employee_id'],
67+
aggregates=['id:recordset'],
68+
))
69+
for leave in indian_leaves:
70+
if leave.holiday_status_id.l10n_in_is_sandwich_leave:
71+
days, hours = result[leave.id]
72+
updated_days = leave._l10n_in_apply_sandwich_rule(public_holidays, leaves_by_employee.get(leave.employee_id, []))
73+
result[leave.id] = (updated_days, hours)
74+
if updated_days:
75+
leave.l10n_in_contains_sandwich_leaves = updated_days != days
76+
else:
77+
leave.l10n_in_contains_sandwich_leaves = False
78+
return result
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
# Part of Odoo. See LICENSE file for full copyright and licensing details.
2+
3+
from odoo import fields, models
4+
5+
6+
class HolidaysType(models.Model):
7+
_inherit = "hr.leave.type"
8+
9+
l10n_in_is_sandwich_leave = fields.Boolean(
10+
help="""If a leave is covering holidays, the holiday period will be included in the requested time.
11+
The time took in addition will have the same treatment (allocation, pay, reports) as the initial request.
12+
Holidays includes public holidays, national days, paid holidays and week-ends.""")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# Part of Odoo. See LICENSE file for full copyright and licensing details.
2+
3+
from . import test_sandwich_leave
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
# Part of Odoo. See LICENSE file for full copyright and licensing details.
2+
3+
from freezegun import freeze_time
4+
from odoo.tests import tagged, TransactionCase
5+
6+
7+
@tagged('post_install_l10n', 'post_install', '-at_install')
8+
class TestSandwichLeave(TransactionCase):
9+
10+
def setUp(self):
11+
super().setUp()
12+
self.leave_type = self.env['hr.leave.type'].create({
13+
'name': 'Test Leave Type',
14+
'request_unit': 'day',
15+
'l10n_in_is_sandwich_leave': True,
16+
})
17+
18+
self.rahul_emp = self.env['hr.employee'].create({
19+
'name': 'Rahul',
20+
'country_id': self.env.ref('base.in').id,
21+
})
22+
23+
def test_sandwich_leave(self):
24+
with freeze_time('2023-08-15'):
25+
public_holiday = self.env['resource.calendar.leaves'].create({
26+
'name': 'Independence Day',
27+
'date_from': '2023-08-15',
28+
'date_to': '2023-08-15',
29+
'resource_id': False,
30+
'company_id': self.env.company.id,
31+
})
32+
before_holiday_leave = self.env['hr.leave'].create({
33+
'name': 'Test Leave',
34+
'employee_id': self.rahul_emp.id,
35+
'holiday_status_id': self.leave_type.id,
36+
'request_date_from': "2023-08-14",
37+
'request_date_to': "2023-08-14",
38+
})
39+
employee_leaves = self.env['hr.leave'].search([
40+
('employee_id', '=', self.rahul_emp.id),
41+
('state', 'not in', ['cancel', 'refuse']),
42+
('leave_type_request_unit', '=', 'day'),
43+
])
44+
after_holiday_leave = self.env['hr.leave'].create({
45+
'name': 'Test Leave',
46+
'employee_id': self.rahul_emp.id,
47+
'holiday_status_id': self.leave_type.id,
48+
'request_date_from': "2023-08-16",
49+
'request_date_to': "2023-08-16",
50+
})
51+
52+
leave = before_holiday_leave._l10n_in_apply_sandwich_rule(public_holiday, employee_leaves)
53+
self.assertEqual(leave, 1, "The total leaves should be 1")
54+
sandwiched_leave = after_holiday_leave._l10n_in_apply_sandwich_rule(public_holiday, employee_leaves)
55+
self.assertEqual(sandwiched_leave, 2, "The total leaves should be 2 including sandwich leave")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
<?xml version="1.0" encoding="utf-8"?>
2+
<odoo>
3+
4+
<record id="hr_leave_type_view_form_inherit" model="ir.ui.view">
5+
<field name="name">hr.leave.type.form.inherit</field>
6+
<field name="model">hr.leave.type</field>
7+
<field name="inherit_id" ref="hr_holidays.edit_holiday_status_form"/>
8+
<field name="arch" type="xml">
9+
<field name="show_on_dashboard" position="after">
10+
<field name="l10n_in_is_sandwich_leave" string="Sandwich Leaves" invisible="country_code != 'IN'" />
11+
</field>
12+
</field>
13+
</record>
14+
15+
</odoo>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
<?xml version="1.0" encoding="utf-8"?>
2+
<odoo>
3+
4+
<record id="hr_leave_view_form_inherit" model="ir.ui.view">
5+
<field name="name">hr.leave.view.form.inherit</field>
6+
<field name="model">hr.leave</field>
7+
<field name="inherit_id" ref="hr_holidays.hr_leave_view_form"/>
8+
<field name="arch" type="xml">
9+
<xpath expr="//field[@name='leave_type_request_unit']" position="after">
10+
<field name="l10n_in_contains_sandwich_leaves" invisible="1" />
11+
</xpath>
12+
<xpath expr="//field[@name='duration_display']" position="after">
13+
<div name="l10n_in_contains_sandwich_leaves" invisible="not l10n_in_contains_sandwich_leaves" class="w-md-75 float-end ps-3 alert alert-warning" role="alert">
14+
The requested time considers the sandwich leave policy.
15+
</div>
16+
</xpath>
17+
</field>
18+
</record>
19+
20+
</odoo>

0 commit comments

Comments
 (0)