mirror of
https://github.com/OCA/contract.git
synced 2025-02-13 17:57:24 +02:00
[12.0][ADD] - Add new addon: contract_forecast
This commit is contained in:
4
contract_forecast/models/__init__.py
Normal file
4
contract_forecast/models/__init__.py
Normal file
@@ -0,0 +1,4 @@
|
||||
from . import contract
|
||||
from . import contract_line
|
||||
from . import contract_line_forecast_period
|
||||
from . import res_company
|
||||
21
contract_forecast/models/contract.py
Normal file
21
contract_forecast/models/contract.py
Normal file
@@ -0,0 +1,21 @@
|
||||
# Copyright 2019 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import _, api, models
|
||||
|
||||
|
||||
class AccountAnalyticAccount(models.Model):
|
||||
|
||||
_inherit = "account.analytic.account"
|
||||
|
||||
@api.multi
|
||||
def action_show_contract_forecast(self):
|
||||
self.ensure_one()
|
||||
return {
|
||||
"type": "ir.actions.act_window",
|
||||
"name": _("Contract Forecast"),
|
||||
"res_model": "contract.line.forecast.period",
|
||||
"domain": [("contract_id", "=", self.id)],
|
||||
"view_mode": "pivot,tree",
|
||||
"context": self.env.context,
|
||||
}
|
||||
145
contract_forecast/models/contract_line.py
Normal file
145
contract_forecast/models/contract_line.py
Normal file
@@ -0,0 +1,145 @@
|
||||
# Copyright 2019 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from dateutil.relativedelta import relativedelta
|
||||
from odoo import api, fields, models
|
||||
from odoo.addons.queue_job.job import job
|
||||
|
||||
QUEUE_CHANNEL = "root.CONTRACT_FORECAST"
|
||||
|
||||
|
||||
class AccountAnalyticInvoiceLine(models.Model):
|
||||
|
||||
_inherit = "account.analytic.invoice.line"
|
||||
|
||||
forecast_period_ids = fields.One2many(
|
||||
comodel_name="contract.line.forecast.period",
|
||||
inverse_name="contract_line_id",
|
||||
string="Forecast Periods",
|
||||
required=False,
|
||||
)
|
||||
|
||||
@api.multi
|
||||
def _prepare_contract_line_forecast_period(
|
||||
self, period_date_start, period_date_end, recurring_next_date
|
||||
):
|
||||
self.ensure_one()
|
||||
return {
|
||||
"name": self._insert_markers(period_date_start, period_date_end),
|
||||
"contract_id": self.contract_id.id,
|
||||
"contract_line_id": self.id,
|
||||
"product_id": self.product_id.id,
|
||||
"date_start": period_date_start,
|
||||
"date_end": period_date_end,
|
||||
"date_invoice": recurring_next_date,
|
||||
"discount": self.discount,
|
||||
"price_unit": self.price_unit,
|
||||
"quantity": self._get_quantity_to_invoice(
|
||||
period_date_start, period_date_end, recurring_next_date
|
||||
),
|
||||
}
|
||||
|
||||
@api.multi
|
||||
def _get_contract_forecast_end_date(self):
|
||||
self.ensure_one()
|
||||
today = fields.Date.context_today(self)
|
||||
return today + self.get_relative_delta(
|
||||
self.contract_id.company_id.contract_forecast_rule_type,
|
||||
self.contract_id.company_id.contract_forecast_interval,
|
||||
)
|
||||
|
||||
@api.multi
|
||||
def _get_generate_forecast_periods_criteria(self, period_date_end):
|
||||
self.ensure_one()
|
||||
if self.is_canceled or not self.active:
|
||||
return False
|
||||
contract_forecast_end_date = self._get_contract_forecast_end_date()
|
||||
if not self.date_end:
|
||||
return period_date_end <= contract_forecast_end_date
|
||||
return (
|
||||
period_date_end < self.date_end
|
||||
and period_date_end <= contract_forecast_end_date
|
||||
)
|
||||
|
||||
@api.multi
|
||||
@job(default_channel=QUEUE_CHANNEL)
|
||||
def _generate_forecast_periods(self):
|
||||
values = []
|
||||
for rec in self:
|
||||
if rec.recurring_next_date:
|
||||
last_date_invoiced = (
|
||||
rec.last_date_invoiced
|
||||
if rec.last_date_invoiced
|
||||
else rec.date_start
|
||||
)
|
||||
period_date_end = last_date_invoiced
|
||||
recurring_next_date = rec.recurring_next_date
|
||||
while rec._get_generate_forecast_periods_criteria(
|
||||
period_date_end
|
||||
):
|
||||
period_dates = rec._get_period_to_invoice(
|
||||
last_date_invoiced, recurring_next_date
|
||||
)
|
||||
period_date_start, period_date_end, recurring_next_date = (
|
||||
period_dates
|
||||
)
|
||||
values.append(
|
||||
rec._prepare_contract_line_forecast_period(
|
||||
period_date_start,
|
||||
period_date_end,
|
||||
recurring_next_date,
|
||||
)
|
||||
)
|
||||
last_date_invoiced = period_date_end
|
||||
recurring_next_date = (
|
||||
recurring_next_date
|
||||
+ self.get_relative_delta(
|
||||
rec.recurring_rule_type, rec.recurring_interval
|
||||
)
|
||||
)
|
||||
return self.env["contract.line.forecast.period"].create(values)
|
||||
|
||||
@api.multi
|
||||
@job(default_channel=QUEUE_CHANNEL)
|
||||
def _unlink_forecast_periods(self):
|
||||
return self.mapped("forecast_period_ids").unlink()
|
||||
|
||||
@api.model
|
||||
def create(self, values):
|
||||
contract_lines = super(AccountAnalyticInvoiceLine, self).create(values)
|
||||
for contract_line in contract_lines:
|
||||
contract_line._generate_forecast_periods()
|
||||
return contract_lines
|
||||
|
||||
@api.model
|
||||
def _get_forecast_update_trigger_fields(self):
|
||||
return [
|
||||
"name",
|
||||
"sequence",
|
||||
"product_id",
|
||||
"date_start",
|
||||
"date_end",
|
||||
"quantity",
|
||||
"price_unit",
|
||||
"discount",
|
||||
"recurring_invoicing_type",
|
||||
"recurring_next_date",
|
||||
"recurring_rule_type",
|
||||
"recurring_interval",
|
||||
"is_canceled",
|
||||
"active",
|
||||
]
|
||||
|
||||
@api.multi
|
||||
def write(self, values):
|
||||
res = super(AccountAnalyticInvoiceLine, self).write(values)
|
||||
if any(
|
||||
[
|
||||
field in values
|
||||
for field in self._get_forecast_update_trigger_fields()
|
||||
]
|
||||
):
|
||||
for rec in self:
|
||||
rec._unlink_forecast_periods()
|
||||
rec._generate_forecast_periods()
|
||||
return res
|
||||
83
contract_forecast/models/contract_line_forecast_period.py
Normal file
83
contract_forecast/models/contract_line_forecast_period.py
Normal file
@@ -0,0 +1,83 @@
|
||||
# Copyright 2019 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import api, fields, models
|
||||
from odoo.addons import decimal_precision as dp
|
||||
|
||||
|
||||
class ContractLineForecastPeriod(models.Model):
|
||||
|
||||
_name = "contract.line.forecast.period"
|
||||
_description = "Contract Line Forecast Period"
|
||||
_order = "date_invoice, sequence"
|
||||
|
||||
name = fields.Char(string="Name", required=True, readonly=True)
|
||||
sequence = fields.Integer(
|
||||
string="Sequence", related="contract_line_id.sequence", store=True
|
||||
)
|
||||
contract_id = fields.Many2one(
|
||||
comodel_name="account.analytic.account",
|
||||
string="Contract",
|
||||
required=True,
|
||||
readonly=True,
|
||||
ondelete="cascade",
|
||||
related="contract_line_id.contract_id",
|
||||
store=True,
|
||||
index=True,
|
||||
)
|
||||
contract_line_id = fields.Many2one(
|
||||
comodel_name="account.analytic.invoice.line",
|
||||
string="Contract Line",
|
||||
required=True,
|
||||
readonly=True,
|
||||
ondelete="cascade",
|
||||
index=True,
|
||||
)
|
||||
product_id = fields.Many2one(
|
||||
comodel_name="product.product",
|
||||
string="Product",
|
||||
required=True,
|
||||
readonly=True,
|
||||
related="contract_line_id.product_id",
|
||||
store=True,
|
||||
index=True,
|
||||
)
|
||||
date_start = fields.Date(string="Date Start", required=True, readonly=True)
|
||||
date_end = fields.Date(string="Date End", required=True, readonly=True)
|
||||
date_invoice = fields.Date(
|
||||
string="Invoice Date", required=True, readonly=True
|
||||
)
|
||||
quantity = fields.Float(default=1.0, required=True)
|
||||
price_unit = fields.Float(string='Unit Price')
|
||||
price_subtotal = fields.Float(
|
||||
digits=dp.get_precision("Account"),
|
||||
string="Amount Untaxed",
|
||||
compute='_compute_price_subtotal',
|
||||
store=True
|
||||
)
|
||||
discount = fields.Float(
|
||||
string='Discount (%)',
|
||||
digits=dp.get_precision('Discount'),
|
||||
help='Discount that is applied in generated invoices.'
|
||||
' It should be less or equal to 100',
|
||||
)
|
||||
active = fields.Boolean(
|
||||
string="Active",
|
||||
related="contract_line_id.active",
|
||||
store=True,
|
||||
readonly=True,
|
||||
default=True,
|
||||
)
|
||||
|
||||
@api.multi
|
||||
@api.depends('quantity', 'price_unit', 'discount')
|
||||
def _compute_price_subtotal(self):
|
||||
for line in self:
|
||||
subtotal = line.quantity * line.price_unit
|
||||
discount = line.discount / 100
|
||||
subtotal *= 1 - discount
|
||||
if line.contract_id.pricelist_id:
|
||||
cur = line.contract_id.pricelist_id.currency_id
|
||||
line.price_subtotal = cur.round(subtotal)
|
||||
else:
|
||||
line.price_subtotal = subtotal
|
||||
16
contract_forecast/models/res_company.py
Normal file
16
contract_forecast/models/res_company.py
Normal file
@@ -0,0 +1,16 @@
|
||||
# Copyright 2019 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class ResCompany(models.Model):
|
||||
|
||||
_inherit = "res.company"
|
||||
|
||||
contract_forecast_interval = fields.Integer(
|
||||
string="Number of contract forecast Periods", default=12
|
||||
)
|
||||
contract_forecast_rule_type = fields.Selection(
|
||||
[("monthly", "Month(s)"), ("yearly", "Year(s)")], default="monthly"
|
||||
)
|
||||
Reference in New Issue
Block a user