106 lines
4.7 KiB
Python
106 lines
4.7 KiB
Python
|
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||
|
from copy import deepcopy
|
||
|
|
||
|
from odoo import fields, models, api, Command
|
||
|
from odoo.tools import float_compare
|
||
|
|
||
|
|
||
|
class HrExpenseSplit(models.TransientModel):
|
||
|
_name = 'hr.expense.split'
|
||
|
_inherit = ['analytic.mixin']
|
||
|
_description = 'Expense Split'
|
||
|
_check_company_auto = True
|
||
|
|
||
|
def default_get(self, fields):
|
||
|
result = super().default_get(fields)
|
||
|
if 'expense_id' in result:
|
||
|
expense = self.env['hr.expense'].browse(result['expense_id'])
|
||
|
result['total_amount_currency'] = 0.0
|
||
|
result['name'] = expense.name
|
||
|
result['tax_ids'] = expense.tax_ids
|
||
|
result['product_id'] = expense.product_id
|
||
|
result['company_id'] = expense.company_id
|
||
|
result['analytic_distribution'] = deepcopy(expense.analytic_distribution) or {}
|
||
|
result['employee_id'] = expense.employee_id
|
||
|
result['currency_id'] = expense.currency_id
|
||
|
return result
|
||
|
|
||
|
name = fields.Char(string='Description', required=True)
|
||
|
wizard_id = fields.Many2one(comodel_name='hr.expense.split.wizard')
|
||
|
expense_id = fields.Many2one(comodel_name='hr.expense', string='Expense')
|
||
|
product_id = fields.Many2one(comodel_name='product.product', string='Product', required=True, check_company=True)
|
||
|
tax_ids = fields.Many2many(
|
||
|
comodel_name='account.tax',
|
||
|
check_company=True,
|
||
|
domain="[('type_tax_use', '=', 'purchase')]",
|
||
|
)
|
||
|
total_amount_currency = fields.Monetary(
|
||
|
string="Total In Currency",
|
||
|
required=True,
|
||
|
compute='_compute_from_product_id', store=True, readonly=False,
|
||
|
)
|
||
|
tax_amount_currency = fields.Monetary(string='Tax amount in Currency', compute='_compute_tax_amount_currency')
|
||
|
employee_id = fields.Many2one(comodel_name='hr.employee', string="Employee", required=True)
|
||
|
company_id = fields.Many2one(comodel_name='res.company')
|
||
|
currency_id = fields.Many2one(comodel_name='res.currency')
|
||
|
product_has_tax = fields.Boolean(
|
||
|
string="Whether tax is defined on a selected product",
|
||
|
compute='_compute_product_has_tax',
|
||
|
)
|
||
|
product_has_cost = fields.Boolean(
|
||
|
string="Is product with non zero cost selected",
|
||
|
compute='_compute_from_product_id', store=True,
|
||
|
)
|
||
|
|
||
|
@api.depends('total_amount_currency', 'tax_ids')
|
||
|
def _compute_tax_amount_currency(self):
|
||
|
for split in self:
|
||
|
taxes = split.tax_ids.with_context(force_price_include=True).compute_all(
|
||
|
price_unit=split.total_amount_currency,
|
||
|
currency=split.currency_id,
|
||
|
quantity=1,
|
||
|
product=split.product_id
|
||
|
)
|
||
|
split.tax_amount_currency = taxes['total_included'] - taxes['total_excluded']
|
||
|
|
||
|
@api.depends('product_id')
|
||
|
def _compute_from_product_id(self):
|
||
|
for split in self:
|
||
|
split.product_has_cost = split.product_id and (float_compare(split.product_id.standard_price, 0.0, precision_digits=2) != 0)
|
||
|
if split.product_has_cost:
|
||
|
split.total_amount_currency = split.product_id._price_compute('standard_price', currency=split.currency_id)[split.product_id.id]
|
||
|
|
||
|
@api.onchange('product_id')
|
||
|
def _onchange_product_id(self):
|
||
|
"""
|
||
|
In case we switch to the product without taxes defined on it, taxes should be removed.
|
||
|
Computed method won't be good for this purpose, as we don't want to recompute and reset taxes in case they are removed on purpose during splitting.
|
||
|
"""
|
||
|
if self.product_has_tax and self.tax_ids:
|
||
|
self.tax_ids = self.tax_ids
|
||
|
else:
|
||
|
self.tax_ids = self.product_id.supplier_taxes_id.filtered_domain(self.env['account.tax']._check_company_domain(self.company_id))
|
||
|
|
||
|
@api.depends('product_id')
|
||
|
def _compute_product_has_tax(self):
|
||
|
for split in self:
|
||
|
split.product_has_tax = split.product_id and split.product_id.supplier_taxes_id.filtered_domain(self.env['account.tax']._check_company_domain(split.company_id))
|
||
|
|
||
|
def _get_values(self):
|
||
|
self.ensure_one()
|
||
|
vals = {
|
||
|
'name': self.name,
|
||
|
'product_id': self.product_id.id,
|
||
|
'total_amount_currency': self.total_amount_currency,
|
||
|
'total_amount': self.expense_id.currency_id.round(self.expense_id.currency_rate * self.total_amount_currency),
|
||
|
'tax_ids': [Command.set(self.tax_ids.ids)],
|
||
|
'analytic_distribution': self.analytic_distribution,
|
||
|
'employee_id': self.employee_id.id,
|
||
|
'product_uom_id': self.product_id.uom_id.id,
|
||
|
}
|
||
|
|
||
|
account = self.product_id.product_tmpl_id._get_product_accounts()['expense']
|
||
|
if account:
|
||
|
vals['account_id'] = account.id
|
||
|
return vals
|