123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146 |
- # -*- coding: utf-8 -*-
- # Part of Odoo. See LICENSE file for full copyright and licensing details.
- import threading
- from odoo import api, fields, models, _
- from odoo.exceptions import ValidationError
- class ProductTemplate(models.Model):
- _inherit = 'product.template'
- def _selection_service_policy(self):
- service_policies = super()._selection_service_policy()
- service_policies.insert(1, ('delivered_timesheet', _('Based on Timesheets')))
- return service_policies
- service_type = fields.Selection(selection_add=[
- ('timesheet', 'Timesheets on project (one fare per SO/Project)'),
- ], ondelete={'timesheet': 'set manual'})
- # override domain
- project_id = fields.Many2one(domain="[('company_id', '=', current_company_id), ('allow_billable', '=', True), ('pricing_type', '=', 'task_rate'), ('allow_timesheets', 'in', [service_policy == 'delivered_timesheet', True])]")
- project_template_id = fields.Many2one(domain="[('company_id', '=', current_company_id), ('allow_billable', '=', True), ('allow_timesheets', 'in', [service_policy == 'delivered_timesheet', True])]")
- service_upsell_threshold = fields.Float('Threshold', default=1, help="Percentage of time delivered compared to the prepaid amount that must be reached for the upselling opportunity activity to be triggered.")
- service_upsell_threshold_ratio = fields.Char(compute='_compute_service_upsell_threshold_ratio')
- @api.depends('uom_id')
- def _compute_service_upsell_threshold_ratio(self):
- product_uom_hour = self.env.ref('uom.product_uom_hour')
- for record in self:
- if not record.uom_id or product_uom_hour.factor == record.uom_id.factor:
- record.service_upsell_threshold_ratio = False
- continue
- if product_uom_hour.factor != record.uom_id.factor:
- record.service_upsell_threshold_ratio = f"(1 {record.uom_id.name} = {product_uom_hour.factor / record.uom_id.factor:.2f} Hours)"
- def _compute_visible_expense_policy(self):
- visibility = self.user_has_groups('project.group_project_user')
- for product_template in self:
- if not product_template.visible_expense_policy:
- product_template.visible_expense_policy = visibility
- return super(ProductTemplate, self)._compute_visible_expense_policy()
- @api.depends('service_tracking', 'service_policy', 'type')
- def _compute_product_tooltip(self):
- super()._compute_product_tooltip()
- for record in self.filtered(lambda record: record.type == 'service'):
- if record.service_policy == 'delivered_timesheet':
- if record.service_tracking == 'no':
- record.product_tooltip = _(
- "Invoice based on timesheets (delivered quantity) on projects or tasks "
- "you'll create later on."
- )
- elif record.service_tracking == 'task_global_project':
- record.product_tooltip = _(
- "Invoice based on timesheets (delivered quantity), and create a task in "
- "an existing project to track the time spent."
- )
- elif record.service_tracking == 'task_in_project':
- record.product_tooltip = _(
- "Invoice based on timesheets (delivered quantity), and create a project "
- "for the order with a task for each sales order line to track the time "
- "spent."
- )
- elif record.service_tracking == 'project_only':
- record.product_tooltip = _(
- "Invoice based on timesheets (delivered quantity), and create an empty "
- "project for the order to track the time spent."
- )
- def _get_service_to_general_map(self):
- return {
- **super()._get_service_to_general_map(),
- 'delivered_timesheet': ('delivery', 'timesheet'),
- 'ordered_prepaid': ('order', 'timesheet'),
- }
- @api.model
- def _get_onchange_service_policy_updates(self, service_tracking, service_policy, project_id, project_template_id):
- vals = {}
- if service_tracking != 'no' and service_policy == 'delivered_timesheet':
- if project_id and not project_id.allow_timesheets:
- vals['project_id'] = False
- elif project_template_id and not project_template_id.allow_timesheets:
- vals['project_template_id'] = False
- return vals
- @api.onchange('service_policy')
- def _onchange_service_policy(self):
- self._inverse_service_policy()
- vals = self._get_onchange_service_policy_updates(self.service_tracking,
- self.service_policy,
- self.project_id,
- self.project_template_id)
- if vals:
- self.update(vals)
- @api.ondelete(at_uninstall=False)
- def _unlink_except_master_data(self):
- time_product = self.env.ref('sale_timesheet.time_product')
- if time_product.product_tmpl_id in self:
- raise ValidationError(_('The %s product is required by the Timesheets app and cannot be archived nor deleted.') % time_product.name)
- def write(self, vals):
- # timesheet product can't be archived
- test_mode = getattr(threading.current_thread(), 'testing', False) or self.env.registry.in_test_mode()
- if not test_mode and 'active' in vals and not vals['active']:
- time_product = self.env.ref('sale_timesheet.time_product')
- if time_product.product_tmpl_id in self:
- raise ValidationError(_('The %s product is required by the Timesheets app and cannot be archived nor deleted.') % time_product.name)
- return super(ProductTemplate, self).write(vals)
- class ProductProduct(models.Model):
- _inherit = 'product.product'
- def _is_delivered_timesheet(self):
- """ Check if the product is a delivered timesheet """
- self.ensure_one()
- return self.type == 'service' and self.service_policy == 'delivered_timesheet'
- @api.onchange('service_policy')
- def _onchange_service_policy(self):
- self._inverse_service_policy()
- vals = self.product_tmpl_id._get_onchange_service_policy_updates(self.service_tracking,
- self.service_policy,
- self.project_id,
- self.project_template_id)
- if vals:
- self.update(vals)
- @api.ondelete(at_uninstall=False)
- def _unlink_except_master_data(self):
- time_product = self.env.ref('sale_timesheet.time_product')
- if time_product in self:
- raise ValidationError(_('The %s product is required by the Timesheets app and cannot be archived nor deleted.') % time_product.name)
- def write(self, vals):
- # timesheet product can't be archived
- test_mode = getattr(threading.current_thread(), 'testing', False) or self.env.registry.in_test_mode()
- if not test_mode and 'active' in vals and not vals['active']:
- time_product = self.env.ref('sale_timesheet.time_product')
- if time_product in self:
- raise ValidationError(_('The %s product is required by the Timesheets app and cannot be archived nor deleted.') % time_product.name)
- return super(ProductProduct, self).write(vals)
|