# -*- coding: utf-8 -*- # Part of Odoo. See LICENSE file for full copyright and licensing details. from collections import defaultdict from odoo import api, Command, fields, models, _ from odoo.osv import expression from odoo.exceptions import ValidationError class AccountAnalyticLine(models.Model): _inherit = 'account.analytic.line' helpdesk_ticket_id = fields.Many2one( 'helpdesk.ticket', 'Helpdesk Ticket', index='btree_not_null', compute='_compute_helpdesk_ticket_id', store=True, readonly=False, domain="[('company_id', '=', company_id), ('project_id', '=?', project_id)]") has_helpdesk_team = fields.Boolean(related='project_id.has_helpdesk_team') display_task = fields.Boolean(compute="_compute_display_task") @api.depends('has_helpdesk_team', 'project_id', 'task_id', 'helpdesk_ticket_id') def _compute_display_task(self): for line in self: line.display_task = line.task_id or not line.has_helpdesk_team @api.depends('project_id') def _compute_helpdesk_ticket_id(self): for line in self: if not line.project_id or line.project_id != line.helpdesk_ticket_id.project_id: line.helpdesk_ticket_id = False @api.depends('helpdesk_ticket_id', 'helpdesk_ticket_id.project_id') def _compute_project_id(self): timesheets_with_ticket = self.filtered('helpdesk_ticket_id') for timesheet in timesheets_with_ticket: if not timesheet.helpdesk_ticket_id.project_id or timesheet.helpdesk_ticket_id.project_id == timesheet.project_id: continue timesheet.project_id = timesheet.helpdesk_ticket_id.project_id super(AccountAnalyticLine, self - timesheets_with_ticket)._compute_project_id() @api.constrains('task_id', 'helpdesk_ticket_id') def _check_no_link_task_and_ticket(self): # Check if any timesheets are not linked to a ticket and a task at the same time if any(timesheet.task_id and timesheet.helpdesk_ticket_id for timesheet in self): raise ValidationError(_("You cannot link a timesheet entry to a task and a ticket at the same time.")) @api.depends('helpdesk_ticket_id.partner_id') def _compute_partner_id(self): super(AccountAnalyticLine, self)._compute_partner_id() for line in self: if line.helpdesk_ticket_id: line.partner_id = line.helpdesk_ticket_id.partner_id or line.partner_id def _timesheet_preprocess(self, vals_list): vals_list_per_ticket_id = defaultdict(list) for vals in vals_list: if vals.get('helpdesk_ticket_id'): vals_list_per_ticket_id[vals['helpdesk_ticket_id']].append(vals) if vals_list_per_ticket_id: tickets_per_id = { ticket['id']: ticket for ticket in self.env['helpdesk.ticket'].sudo().browse(list(vals_list_per_ticket_id)) } for ticket_id, ticket_vals_list in vals_list_per_ticket_id.items(): ticket = tickets_per_id[ticket_id] vals_update = {'account_id': ticket.analytic_account_id.id} if ticket.project_id: vals_update['project_id'] = ticket.project_id.id if 'company_id' not in ticket_vals_list: vals_update['company_id'] = ticket.project_id.company_id.id for vals in ticket_vals_list: vals.update(vals_update) return super(AccountAnalyticLine, self)._timesheet_preprocess(vals_list) def _get_timesheet_field_and_model_name(self): if self._context.get('default_helpdesk_ticket_id', False): return 'helpdesk_ticket_id', 'helpdesk.ticket' return super()._get_timesheet_field_and_model_name() def _get_timesheet_timer_data(self, timer=None): timesheet_timer_data = super()._get_timesheet_timer_data(timer) if timesheet_timer_data.get('readonly'): timesheet_timer_data['helpdesk_ticket_id'] = self.helpdesk_ticket_id.sudo().name else: timesheet_timer_data['helpdesk_ticket_id'] = self.helpdesk_ticket_id.id return timesheet_timer_data def _timesheet_get_portal_domain(self): domain = super(AccountAnalyticLine, self)._timesheet_get_portal_domain() if not self.env.user.has_group('hr_timesheet.group_hr_timesheet_user'): domain = expression.OR([domain, self._timesheet_in_helpdesk_get_portal_domain()]) return domain def _timesheet_in_helpdesk_get_portal_domain(self): return [ '&', '&', '&', ('task_id', '=', False), ('helpdesk_ticket_id', '!=', False), '|', ('project_id.message_partner_ids', 'child_of', [self.env.user.partner_id.commercial_partner_id.id]), ('helpdesk_ticket_id.message_partner_ids', 'child_of', [self.env.user.partner_id.commercial_partner_id.id]), ('project_id.privacy_visibility', '=', 'portal') ] def _get_timesheet_timer_data(self, timer=None): return {**super()._get_timesheet_timer_data(), 'helpdesk_ticket_id': self.helpdesk_ticket_id.id} @api.model def _add_time_to_timesheet_fields(self): return super()._add_time_to_timesheet_fields() + ['helpdesk_ticket_id'] def _get_new_timesheet_timer_vals(self): vals = super()._get_new_timesheet_timer_vals() vals.update({'helpdesk_ticket_id': self.helpdesk_ticket_id.id}) return vals def _get_last_timesheet_domain(self): return expression.AND([ super()._get_last_timesheet_domain(), [('helpdesk_ticket_id', '=', self.helpdesk_ticket_id.id)], ])