[12.0][ADD] - Add new addon: contract_forecast

This commit is contained in:
sbejaoui
2019-03-07 12:32:06 +01:00
parent 3d8144d413
commit ea2e793c59
20 changed files with 1049 additions and 0 deletions

View File

@@ -0,0 +1,4 @@
from . import contract
from . import contract_line
from . import contract_line_forecast_period
from . import res_company

View 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,
}

View 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

View 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

View 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"
)