1
0
forked from Mapan/odoo17e
odoo17e-kedaikipas58/addons/l10n_be_hr_payroll_fleet/models/fleet.py
2024-12-10 09:04:09 +07:00

256 lines
12 KiB
Python

# -*- coding:utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
import babel.dates
from odoo import api, fields, models
from odoo.tools.misc import format_date
from odoo.tools.float_utils import float_round
from odoo.fields import Datetime, Date
class FleetVehicle(models.Model):
_inherit = 'fleet.vehicle'
co2_fee = fields.Float(compute='_compute_co2_fee', string="CO2 Fee")
total_depreciated_cost = fields.Float(compute='_compute_total_depreciated_cost',
string="Total Cost (Depreciated)", tracking=True, compute_sudo=True,
help="This includes all the depreciated costs and the CO2 fee")
total_cost = fields.Float(compute='_compute_total_cost', string="Total Cost", help="This include all the costs and the CO2 fee")
fuel_type = fields.Selection(required=True, default='diesel')
atn = fields.Float(compute='_compute_car_atn', string="BIK")
acquisition_date = fields.Date(required=True)
tax_deduction = fields.Float(compute='_compute_tax_deduction')
def _from_be(self):
if self:
return self.company_id.country_id.code == "BE"
else:
return self.env.company.country_id.code == "BE"
@api.depends('co2_fee', 'log_contracts', 'log_contracts.state', 'log_contracts.recurring_cost_amount_depreciated')
def _compute_total_depreciated_cost(self):
for car in self:
if car.log_contracts:
car.total_depreciated_cost = car.co2_fee + \
sum(car.log_contracts.filtered(
lambda contract: contract.state == 'open'
).mapped('recurring_cost_amount_depreciated'))
else:
car.total_depreciated_cost = car.model_id.with_company(car.company_id).default_total_depreciated_cost
@api.depends('co2_fee', 'log_contracts', 'log_contracts.state', 'log_contracts.cost_generated')
def _compute_total_cost(self):
for car in self:
car.total_cost = car.co2_fee
contracts = car.log_contracts.filtered(
lambda contract: contract.state == 'open' and contract.cost_frequency != 'no'
)
for contract in contracts:
if contract.cost_frequency == "daily":
car.total_cost += contract.cost_generated * 30.0
elif contract.cost_frequency == "weekly":
car.total_cost += contract.cost_generated * 4.0
elif contract.cost_frequency == "monthly":
car.total_cost += contract.cost_generated
elif contract.cost_frequency == "yearly":
car.total_cost += contract.cost_generated / 12.0
def _get_tax_deduction(self, co2, fuel, coefficients, horsepower):
if fuel == 'electric':
return 1
if co2 >= 200:
return 0.4
if coefficients and fuel in coefficients:
coeff = coefficients[fuel]
# Special case for cng which has a different coeff depending on horweposer
if fuel == 'cng' and horsepower < coefficients.get('cng_hp_lower_bound', 12):
coeff = coefficients.get('cng_low', coeff)
return min(max(1.2 - (0.005 * coeff * co2), 0.5), 1)
return 0
@api.depends('fuel_type', 'co2', 'horsepower')
def _compute_tax_deduction(self):
be_vehicles = self.filtered(lambda vehicle: vehicle.company_id.country_id.code == "BE")
(self - be_vehicles).tax_deduction = 0
coefficients = self.env['hr.rule.parameter'].sudo()._get_parameter_from_code('tax_deduction_fuel_coefficients', raise_if_not_found=False) if be_vehicles else None
for vehicle in be_vehicles:
if vehicle.vehicle_type == "bike":
vehicle.tax_deduction = 1
else:
vehicle.tax_deduction = vehicle._get_tax_deduction(vehicle.co2, vehicle.fuel_type, coefficients, vehicle.horsepower)
def _get_co2_fee(self, co2, fuel_type):
# Reference: https://www.socialsecurity.be/employer/instructions/dmfa/fr/latest/instructions/special_contributions/companycar.html
if not self._from_be() or self.vehicle_type == 'bike':
return 0
date = self.env.context.get('co2_fee_date', fields.Date.today())
fuel_coefficient = self.env['hr.rule.parameter']._get_parameter_from_code('fuel_coefficient', date)
co2_fee_min = self.env['hr.rule.parameter']._get_parameter_from_code('co2_fee_min', date)
co2_fee = co2_fee_min
if fuel_type and fuel_type in ['gasoline', 'diesel', 'lpg']:
health_indice = self.env['hr.rule.parameter']._get_parameter_from_code('health_indice', date)
health_indice_reference = self.env['hr.rule.parameter']._get_parameter_from_code('health_indice_reference', date)
co2_fee = ((co2 * 9.0) - fuel_coefficient.get(fuel_type)) * health_indice / health_indice_reference / 12.0
return max(co2_fee, co2_fee_min)
@api.depends('co2', 'fuel_type', 'company_id.country_id')
def _compute_co2_fee(self):
for car in self:
car.co2_fee = car._get_co2_fee(car.co2, car.fuel_type)
@api.depends('fuel_type', 'car_value', 'acquisition_date', 'co2', 'company_id.country_id')
def _compute_car_atn(self):
for car in self:
car.atn = car._get_car_atn()
@api.depends('model_id', 'license_plate', 'log_contracts', 'acquisition_date',
'co2_fee', 'log_contracts', 'log_contracts.state', 'log_contracts.recurring_cost_amount_depreciated')
def _compute_vehicle_name(self):
super(FleetVehicle, self)._compute_vehicle_name()
for vehicle in self:
acquisition_date = vehicle._get_acquisition_date()
vehicle.name += u" \u2022 " + acquisition_date
def _get_analytic_name(self):
# This function is used in fleet_account
name = super(FleetVehicle, self)._get_analytic_name()
if self.tax_deduction:
return '%s, %s%s' % (name, round(self.tax_deduction * 100, 2), "%")
else:
return name
def _get_acquisition_date(self):
self.ensure_one()
return format_date(self.env, self.acquisition_date, date_format='MMMM y')
def _get_car_atn(self, date=None):
return self._get_car_atn_from_values(self.acquisition_date, self.car_value, self.fuel_type, self.co2, date)
@api.model
def _get_car_atn_from_values(self, acquisition_date, car_value, fuel_type, co2, date=None):
if not self._from_be():
return 0
# Compute the correction coefficient from the age of the car
date = date or Date.today()
if acquisition_date:
number_of_month = ((date.year - acquisition_date.year) * 12.0 + date.month -
acquisition_date.month +
int(bool(date.day - acquisition_date.day + 1)))
if number_of_month <= 12:
age_coefficient = 1.00
elif number_of_month <= 24:
age_coefficient = 0.94
elif number_of_month <= 36:
age_coefficient = 0.88
elif number_of_month <= 48:
age_coefficient = 0.82
elif number_of_month <= 60:
age_coefficient = 0.76
else:
age_coefficient = 0.70
car_value = car_value * age_coefficient
# Compute atn value from corrected car_value
magic_coeff = 6.0 / 7.0 # Don't ask me why
if fuel_type in ['electric', 'hydrogen']:
atn = car_value * 0.04 * magic_coeff
else:
if fuel_type in ['diesel', 'full_hybrid', 'plug_in_hybrid_diesel']:
reference = self.env['hr.rule.parameter']._get_parameter_from_code('co2_reference_diesel', date)
else:
reference = self.env['hr.rule.parameter']._get_parameter_from_code('co2_reference_petrol_lpg', date)
if co2 <= reference:
atn = car_value * max(0.04, (0.055 - 0.001 * (reference - co2))) * magic_coeff
else:
atn = car_value * min(0.18, (0.055 + 0.001 * (co2 - reference))) * magic_coeff
return max(self.env['hr.rule.parameter']._get_parameter_from_code('min_car_atn', date), atn) / 12.0
return 0.0
@api.onchange('model_id')
def _onchange_model_id(self):
self.car_value = self.model_id.default_car_value
self.co2 = self.model_id.default_co2
self.fuel_type = self.model_id.default_fuel_type
class FleetVehicleLogContract(models.Model):
_inherit = 'fleet.vehicle.log.contract'
recurring_cost_amount_depreciated = fields.Float(
"Depreciated Cost Amount", tracking=True,
compute='_compute_recurring_cost_amount_depreciated', store=True, readonly=False)
@api.depends('vehicle_id')
def _compute_recurring_cost_amount_depreciated(self):
for log_contract in self:
default_cost = log_contract.vehicle_id.model_id.default_recurring_cost_amount_depreciated
if not default_cost:
continue
log_contract.recurring_cost_amount_depreciated = default_cost
class FleetVehicleModel(models.Model):
_inherit = 'fleet.vehicle.model'
default_recurring_cost_amount_depreciated = fields.Float(string="Cost (Depreciated)",
help="Default recurring cost amount that should be applied to a new vehicle from this model")
default_car_value = fields.Float(string="Catalog Value (VAT Incl.)")
can_be_requested = fields.Boolean(
string="Can be requested", company_dependent=True,
help="Can be requested on a contract as a new vehicle")
default_atn = fields.Float(compute='_compute_atn', string="BIK")
default_total_depreciated_cost = fields.Float(compute='_compute_default_total_depreciated_cost', compute_sudo=True, string="Total Cost (Depreciated)")
default_co2 = fields.Float(compute='_compute_default_co2', readonly=False, store=True)
co2_fee = fields.Float(compute='_compute_co2_fee', string="CO2 fee")
tax_deduction = fields.Float(compute='_compute_tax_deduction')
current_country_code = fields.Char(compute='_compute_current_country_code')
@api.depends('default_car_value', 'default_co2', 'default_fuel_type')
def _compute_atn(self):
now = Datetime.now()
for model in self:
model.default_atn = self.env['fleet.vehicle']._get_car_atn_from_values(now, model.default_car_value, model.default_fuel_type, model.default_co2)
@api.depends('co2_fee', 'default_recurring_cost_amount_depreciated')
def _compute_default_total_depreciated_cost(self):
for model in self:
model.default_total_depreciated_cost = model.co2_fee + model.default_recurring_cost_amount_depreciated
@api.depends('default_co2', 'default_fuel_type')
def _compute_co2_fee(self):
for model in self:
if model.vehicle_type == 'bike':
model.co2_fee = 0
else:
model.co2_fee = self.env['fleet.vehicle']._get_co2_fee(model.default_co2, model.default_fuel_type)
@api.depends('default_fuel_type')
def _compute_default_co2(self):
# Fill in a default co2 depending on the fuel type only if it is zero
default_co2_map = {
'diesel': 195,
'electric': 0,
}
for model in self:
if not model.default_co2:
model.default_co2 = default_co2_map.get(model.default_fuel_type, 205)
@api.depends('default_co2', 'default_fuel_type', 'horsepower')
def _compute_tax_deduction(self):
coefficients = self.env['hr.rule.parameter'].sudo()._get_parameter_from_code('tax_deduction_fuel_coefficients', raise_if_not_found=False)
for model in self:
if model.vehicle_type == "bike":
model.tax_deduction = 1
else:
model.tax_deduction = self.env['fleet.vehicle']._get_tax_deduction(
model.default_co2, model.default_fuel_type, coefficients, model.horsepower)
@api.depends_context('uid')
def _compute_current_country_code(self):
for model in self:
model.current_country_code = self.env.company.country_id.code