Initial commit: Sale packages

This commit is contained in:
Ernad Husremovic 2025-08-29 15:20:49 +02:00
commit 14e3d26998
6469 changed files with 2479670 additions and 0 deletions

View file

@ -0,0 +1,4 @@
# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from . import models

View file

@ -0,0 +1,16 @@
# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
{
'name': 'Service Margins in Sales Orders',
'version': '1.0',
'summary': 'Bridge module between Sales Margin and Sales Timesheet',
'description': """
Allows to compute accurate margin for Service sales.
======================================================
""",
'category': 'Hidden',
'depends': ['sale_margin', 'sale_timesheet'],
'auto_install': True,
'license': 'LGPL-3',
}

View file

@ -0,0 +1,21 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * sale_timesheet_margin
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2024-02-06 13:32+0000\n"
"PO-Revision-Date: 2024-02-06 13:32+0000\n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: sale_timesheet_margin
#: model:ir.model,name:sale_timesheet_margin.model_sale_order_line
msgid "Sales Order Line"
msgstr "Stavka prodajne narudžbe"

View file

@ -0,0 +1,21 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * sale_timesheet_margin
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2024-02-06 13:32+0000\n"
"PO-Revision-Date: 2024-02-06 13:32+0000\n"
"Last-Translator: \n"
"Language-Team: \n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: \n"
#. module: sale_timesheet_margin
#: model:ir.model,name:sale_timesheet_margin.model_sale_order_line
msgid "Sales Order Line"
msgstr ""

View file

@ -0,0 +1,4 @@
# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from . import sale_order_line

View file

@ -0,0 +1,40 @@
# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo import api, models
class SaleOrderLine(models.Model):
_inherit = "sale.order.line"
@api.depends('analytic_line_ids.amount', 'qty_delivered_method')
def _compute_purchase_price(self):
timesheet_sols = self.filtered(
lambda sol: sol.qty_delivered_method == 'timesheet' and not sol.product_id.standard_price
)
# filter out the sale.order.lines called by this override of _compute_purchase_price for which
# we don't want the purchase price to be recomputed. Without filtring out the sale.order.lines
# for which the recomputation was triggered by a depency from another override of _compute_purchase_price
service_non_timesheet_sols = self.filtered(
lambda sol: not sol.is_expense and sol.is_service and
sol.product_id.service_policy == 'ordered_prepaid' and sol.state == 'sale'
)
super(SaleOrderLine, self - timesheet_sols - service_non_timesheet_sols)._compute_purchase_price()
if timesheet_sols:
group_amount = self.env['account.analytic.line'].read_group(
[('so_line', 'in', timesheet_sols.ids), ('project_id', '!=', False)],
['so_line', 'amount:sum', 'unit_amount:sum'],
['so_line'])
mapped_sol_timesheet_amount = {
amount['so_line'][0]: -amount['amount'] / amount['unit_amount'] if amount['unit_amount'] else 0.0
for amount in group_amount
}
for line in timesheet_sols:
line = line.with_company(line.company_id)
product_cost = mapped_sol_timesheet_amount.get(line.id, line.product_id.standard_price)
product_uom = line.product_uom or line.product_id.uom_id
if product_uom != line.company_id.project_time_mode_id and\
product_uom.category_id.id == line.company_id.project_time_mode_id.category_id.id:
product_cost = product_uom._compute_quantity(
product_cost,
line.company_id.project_time_mode_id
)
line.purchase_price = line._convert_price(product_cost, product_uom)

View file

@ -0,0 +1,4 @@
# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from . import test_sale_timesheet_margin

View file

@ -0,0 +1,109 @@
# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo.addons.sale_timesheet.tests.common import TestCommonSaleTimesheet
from odoo import Command
from odoo.tests import tagged
@tagged('-at_install', 'post_install')
class TestSaleTimesheetMargin(TestCommonSaleTimesheet):
def setUp(self):
super(TestSaleTimesheetMargin, self).setUp()
uom_day_id = self.ref('uom.product_uom_day')
self.uom_day = self.env['uom.uom'].browse(uom_day_id)
self.product_1 = self.env['product.product'].create({
'name': "Service Ordered, create no task, uom day",
'list_price': 1.0,
'type': 'service',
'invoice_policy': 'order',
'uom_id': uom_day_id,
'uom_po_id': uom_day_id,
'default_code': 'SERV-ORDERED-DAY',
'service_type': 'timesheet',
'service_tracking': 'task_in_project',
'project_id': False,
'taxes_id': False,
})
self.employee_manager.hourly_cost = 10
def test_sale_timesheet_margin(self):
""" Test the timesheet cost is reported correctly in sale order line. """
sale_order = self.env['sale.order'].create({
'name': 'Test_SO0001',
'order_line': [
Command.create({
'product_id': self.product_1.id,
'price_unit': 1.0,
'product_uom': self.uom_day.id,
'product_uom_qty': 1.0,
})],
'partner_id': self.partner_b.id,
'partner_invoice_id': self.partner_b.id,
'partner_shipping_id': self.partner_b.id,
})
# Confirm the sales order, create project and task.
sale_order.action_confirm()
# Add timesheet line
self.env['account.analytic.line'].create({
'name': 'Test Line',
'unit_amount': 2,
'employee_id': self.employee_manager.id,
'project_id': sale_order.project_ids.id,
'task_id': sale_order.order_line.task_id.id,
'account_id': self.analytic_account_sale.id,
'so_line': sale_order.order_line.id,
})
sale_order.order_line._compute_purchase_price()
# Cost is expressed in SO line uom
expected_cost = self.uom_day._compute_quantity(
self.employee_manager.hourly_cost,
self.env.company.project_time_mode_id
)
self.assertEqual(sale_order.order_line.purchase_price, expected_cost, "Sale order line cost should be number of working hours on one day * timesheet cost of the employee set on the timesheet linked to the SOL.")
def test_no_recompute_purchase_price_not_timesheet(self):
project = self.env['project.project'].create({
'name': "Test",
})
self.product_1.write({
'uom_id': self.ref('uom.product_uom_unit'),
'uom_po_id': self.ref('uom.product_uom_unit'),
'service_type': 'timesheet',
'service_policy': 'ordered_prepaid',
'service_tracking': 'task_global_project',
'project_id': project.id,
'standard_price': 2,
})
sale_order = self.env['sale.order'].create({
'name': 'Test_SO0002',
'order_line': [
Command.create({
'product_id': self.product_1.id,
'price_unit': 1.0,
'product_uom': self.ref('uom.product_uom_unit'),
'product_uom_qty': 1.0,
})],
'partner_id': self.partner_b.id,
'partner_invoice_id': self.partner_b.id,
'partner_shipping_id': self.partner_b.id,
})
sale_order.order_line.purchase_price = 3
# Confirm the sales order, create project and task.
sale_order.action_confirm()
# Add timesheet line
self.env['account.analytic.line'].create({
'name': 'Test Line 222',
'unit_amount': 2,
'amount': 1,
'employee_id': self.employee_manager.id,
'project_id': project.id,
'task_id': sale_order.order_line.task_id.id,
'account_id': self.analytic_account_sale.id,
'so_line': sale_order.order_line.id,
})
self.env.flush_all()
self.assertEqual(sale_order.order_line.purchase_price, 3)