142 lines
5.1 KiB
Python
142 lines
5.1 KiB
Python
|
# -*- coding: utf-8 -*-
|
||
|
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||
|
from lxml.builder import E
|
||
|
|
||
|
from odoo import api, fields, models
|
||
|
from odoo.osv.expression import OR
|
||
|
|
||
|
|
||
|
class AccountAnalyticLine(models.Model):
|
||
|
_name = 'account.analytic.line'
|
||
|
_description = 'Analytic Line'
|
||
|
_order = 'date desc, id desc'
|
||
|
_check_company_auto = True
|
||
|
|
||
|
name = fields.Char(
|
||
|
'Description',
|
||
|
required=True,
|
||
|
)
|
||
|
date = fields.Date(
|
||
|
'Date',
|
||
|
required=True,
|
||
|
index=True,
|
||
|
default=fields.Date.context_today,
|
||
|
)
|
||
|
amount = fields.Monetary(
|
||
|
'Amount',
|
||
|
required=True,
|
||
|
default=0.0,
|
||
|
)
|
||
|
unit_amount = fields.Float(
|
||
|
'Quantity',
|
||
|
default=0.0,
|
||
|
)
|
||
|
product_uom_id = fields.Many2one(
|
||
|
'uom.uom',
|
||
|
string='Unit of Measure',
|
||
|
domain="[('category_id', '=', product_uom_category_id)]",
|
||
|
)
|
||
|
product_uom_category_id = fields.Many2one(
|
||
|
related='product_uom_id.category_id',
|
||
|
string='UoM Category',
|
||
|
readonly=True,
|
||
|
)
|
||
|
account_id = fields.Many2one(
|
||
|
'account.analytic.account',
|
||
|
'Project Account',
|
||
|
ondelete='restrict',
|
||
|
index=True,
|
||
|
check_company=True,
|
||
|
)
|
||
|
# Magic column that represents all the plans at the same time, except for the compute
|
||
|
# where it is context dependent, and needs the id of the desired plan.
|
||
|
# Used as a syntactic sugar for search views, and magic field for one2many relation
|
||
|
auto_account_id = fields.Many2one(
|
||
|
comodel_name='account.analytic.account',
|
||
|
string='Analytic Account',
|
||
|
compute='_compute_auto_account',
|
||
|
inverse='_inverse_auto_account',
|
||
|
search='_search_auto_account',
|
||
|
)
|
||
|
partner_id = fields.Many2one(
|
||
|
'res.partner',
|
||
|
string='Partner',
|
||
|
check_company=True,
|
||
|
)
|
||
|
user_id = fields.Many2one(
|
||
|
'res.users',
|
||
|
string='User',
|
||
|
default=lambda self: self.env.context.get('user_id', self.env.user.id),
|
||
|
index=True,
|
||
|
)
|
||
|
company_id = fields.Many2one(
|
||
|
'res.company',
|
||
|
string='Company',
|
||
|
required=True,
|
||
|
readonly=True,
|
||
|
default=lambda self: self.env.company,
|
||
|
)
|
||
|
currency_id = fields.Many2one(
|
||
|
related="company_id.currency_id",
|
||
|
string="Currency",
|
||
|
readonly=True,
|
||
|
store=True,
|
||
|
compute_sudo=True,
|
||
|
)
|
||
|
category = fields.Selection(
|
||
|
[('other', 'Other')],
|
||
|
default='other',
|
||
|
)
|
||
|
|
||
|
@api.depends_context('analytic_plan_id')
|
||
|
def _compute_auto_account(self):
|
||
|
plan = self.env['account.analytic.plan'].browse(self.env.context.get('analytic_plan_id'))
|
||
|
for line in self:
|
||
|
line.auto_account_id = bool(plan) and line[plan._column_name()]
|
||
|
|
||
|
def _inverse_auto_account(self):
|
||
|
for line in self:
|
||
|
line[line.auto_account_id.plan_id._column_name()] = line.auto_account_id
|
||
|
|
||
|
def _search_auto_account(self, operator, value):
|
||
|
project_plan, other_plans = self.env['account.analytic.plan']._get_all_plans()
|
||
|
return OR([
|
||
|
[(plan._column_name(), operator, value)]
|
||
|
for plan in project_plan + other_plans
|
||
|
])
|
||
|
|
||
|
def _get_view(self, view_id=None, view_type='form', **options):
|
||
|
arch, view = super()._get_view(view_id, view_type, **options)
|
||
|
if self.env['account.analytic.plan'].check_access_rights('read', raise_exception=False):
|
||
|
project_plan, other_plans = self.env['account.analytic.plan']._get_all_plans()
|
||
|
|
||
|
# Find main account nodes
|
||
|
account_node = next(iter(arch.xpath('//field[@name="account_id"]')), None)
|
||
|
account_filter_node = next(iter(arch.xpath('//filter[@name="account_id"]')), None)
|
||
|
|
||
|
# Force domain on main account node as the fields_get doesn't do the trick
|
||
|
if account_node is not None and view_type == 'search':
|
||
|
account_node.attrib['domain'] = f"[('plan_id', 'child_of', {project_plan.id})]"
|
||
|
|
||
|
# If there is a main node, append the ones for other plans
|
||
|
if account_node is not None or account_filter_node is not None:
|
||
|
for plan in other_plans[::-1]:
|
||
|
fname = plan._column_name()
|
||
|
if account_node is not None:
|
||
|
account_node.addnext(E.field(name=fname, domain=f"[('plan_id', 'child_of', {plan.id})]", optional="show"))
|
||
|
if account_filter_node is not None:
|
||
|
account_filter_node.addnext(E.filter(name=fname, context=f"{{'group_by': '{fname}'}}"))
|
||
|
return arch, view
|
||
|
|
||
|
@api.model
|
||
|
def fields_get(self, allfields=None, attributes=None):
|
||
|
fields = super().fields_get(allfields, attributes)
|
||
|
if self.env['account.analytic.plan'].check_access_rights('read', raise_exception=False):
|
||
|
project_plan, other_plans = self.env['account.analytic.plan']._get_all_plans()
|
||
|
for plan in project_plan + other_plans:
|
||
|
fname = plan._column_name()
|
||
|
if fname in fields:
|
||
|
fields[fname]['string'] = plan.name
|
||
|
fields[fname]['domain'] = f"[('plan_id', 'child_of', {plan.id})]"
|
||
|
return fields
|