[ENH] account_spread_cost_revenue: add calculate by days feature
This commit is contained in:
parent
fb9f29fc0c
commit
0fdbf6f02c
@ -3,6 +3,7 @@
|
|||||||
|
|
||||||
import calendar
|
import calendar
|
||||||
import time
|
import time
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
from dateutil.relativedelta import relativedelta
|
from dateutil.relativedelta import relativedelta
|
||||||
|
|
||||||
@ -44,6 +45,11 @@ class AccountSpread(models.Model):
|
|||||||
help="Period length for the entries",
|
help="Period length for the entries",
|
||||||
required=True,
|
required=True,
|
||||||
)
|
)
|
||||||
|
days_calc = fields.Boolean(
|
||||||
|
string="Calculate by days",
|
||||||
|
default=False,
|
||||||
|
help="Use number of days to calculate amount",
|
||||||
|
)
|
||||||
use_invoice_line_account = fields.Boolean()
|
use_invoice_line_account = fields.Boolean()
|
||||||
credit_account_id = fields.Many2one(
|
credit_account_id = fields.Many2one(
|
||||||
"account.account",
|
"account.account",
|
||||||
@ -242,6 +248,7 @@ class AccountSpread(models.Model):
|
|||||||
self.period_type = self.template_id.period_type
|
self.period_type = self.template_id.period_type
|
||||||
if self.template_id.start_date:
|
if self.template_id.start_date:
|
||||||
self.spread_date = self.template_id.start_date
|
self.spread_date = self.template_id.start_date
|
||||||
|
self.days_calc = self.template_id.days_calc
|
||||||
|
|
||||||
@api.depends("invoice_type", "company_id")
|
@api.depends("invoice_type", "company_id")
|
||||||
def _compute_journal_id(self):
|
def _compute_journal_id(self):
|
||||||
@ -368,8 +375,9 @@ class AccountSpread(models.Model):
|
|||||||
|
|
||||||
for x in range(len(posted_line_ids), number_of_periods):
|
for x in range(len(posted_line_ids), number_of_periods):
|
||||||
sequence = x + 1
|
sequence = x + 1
|
||||||
|
date = self._get_last_day_of_month(spread_date)
|
||||||
amount = self._compute_board_amount(
|
amount = self._compute_board_amount(
|
||||||
sequence, unposted_amount, number_of_periods
|
sequence, unposted_amount, number_of_periods, date
|
||||||
)
|
)
|
||||||
amount = self.currency_id.round(amount)
|
amount = self.currency_id.round(amount)
|
||||||
rounding = self.currency_id.rounding
|
rounding = self.currency_id.rounding
|
||||||
@ -380,7 +388,7 @@ class AccountSpread(models.Model):
|
|||||||
"amount": amount,
|
"amount": amount,
|
||||||
"spread_id": self.id,
|
"spread_id": self.id,
|
||||||
"name": self._get_spread_entry_name(sequence),
|
"name": self._get_spread_entry_name(sequence),
|
||||||
"date": self._get_last_day_of_month(spread_date),
|
"date": date,
|
||||||
}
|
}
|
||||||
commands.append((0, False, vals))
|
commands.append((0, False, vals))
|
||||||
|
|
||||||
@ -398,22 +406,78 @@ class AccountSpread(models.Model):
|
|||||||
self.ensure_one()
|
self.ensure_one()
|
||||||
return self.period_number + 1 if month_day != 1 else self.period_number
|
return self.period_number + 1 if month_day != 1 else self.period_number
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _get_first_day_of_month(spread_date):
|
||||||
|
return spread_date + relativedelta(day=1)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _get_last_day_of_month(spread_date):
|
def _get_last_day_of_month(spread_date):
|
||||||
return spread_date + relativedelta(day=31)
|
return spread_date + relativedelta(day=31)
|
||||||
|
|
||||||
def _compute_board_amount(self, sequence, amount, number_of_periods):
|
def _get_spread_start_date(self, period_type, spread_end_date):
|
||||||
|
self.ensure_one()
|
||||||
|
spread_start_date = spread_end_date + relativedelta(days=1)
|
||||||
|
if period_type == "month":
|
||||||
|
spread_start_date = spread_end_date + relativedelta(day=1)
|
||||||
|
elif period_type == "quarter":
|
||||||
|
spread_start_date = spread_start_date - relativedelta(months=3)
|
||||||
|
elif period_type == "year":
|
||||||
|
spread_start_date = spread_start_date - relativedelta(years=1)
|
||||||
|
spread_start_date = self._get_first_day_of_month(spread_start_date)
|
||||||
|
spread_start_date = max(spread_start_date, self.spread_date)
|
||||||
|
return spread_start_date
|
||||||
|
|
||||||
|
def _get_spread_end_date(self, period_type, period_number, spread_start_date):
|
||||||
|
self.ensure_one()
|
||||||
|
spread_end_date = spread_start_date
|
||||||
|
number_of_periods = (
|
||||||
|
period_number if spread_start_date.day != 1 else period_number - 1
|
||||||
|
)
|
||||||
|
if period_type == "month":
|
||||||
|
spread_end_date = spread_start_date + relativedelta(
|
||||||
|
months=number_of_periods
|
||||||
|
)
|
||||||
|
elif period_type == "quarter":
|
||||||
|
months = number_of_periods * 3
|
||||||
|
spread_end_date = spread_start_date + relativedelta(months=months)
|
||||||
|
elif period_type == "year":
|
||||||
|
spread_end_date = spread_start_date + relativedelta(years=number_of_periods)
|
||||||
|
# calculate by days and not first day of month should compute residual day only
|
||||||
|
if self.days_calc and spread_end_date.day != 1:
|
||||||
|
spread_end_date = spread_end_date - timedelta(days=1)
|
||||||
|
else:
|
||||||
|
spread_end_date = self._get_last_day_of_month(spread_end_date)
|
||||||
|
return spread_end_date
|
||||||
|
|
||||||
|
def _get_amount_per_day(self, amount):
|
||||||
|
self.ensure_one()
|
||||||
|
spread_start_date = self.spread_date
|
||||||
|
spread_end_date = self._get_spread_end_date(
|
||||||
|
self.period_type, self.period_number, spread_start_date
|
||||||
|
)
|
||||||
|
number_of_days = (spread_end_date - spread_start_date).days + 1
|
||||||
|
return amount / number_of_days
|
||||||
|
|
||||||
|
def _compute_board_amount(
|
||||||
|
self, sequence, amount, number_of_periods, spread_end_date
|
||||||
|
):
|
||||||
"""Calculates the amount for the spread lines."""
|
"""Calculates the amount for the spread lines."""
|
||||||
self.ensure_one()
|
self.ensure_one()
|
||||||
amount_to_spread = self.total_amount
|
amount_to_spread = self.total_amount
|
||||||
|
period = self.period_number
|
||||||
if sequence != number_of_periods:
|
if sequence != number_of_periods:
|
||||||
amount = amount_to_spread / self.period_number
|
amount = amount_to_spread / period
|
||||||
if sequence == 1:
|
if sequence == 1:
|
||||||
date = self.spread_date
|
date = self.spread_date
|
||||||
month_days = calendar.monthrange(date.year, date.month)[1]
|
month_days = calendar.monthrange(date.year, date.month)[1]
|
||||||
days = month_days - date.day + 1
|
days = month_days - date.day + 1
|
||||||
period = self.period_number
|
|
||||||
amount = (amount_to_spread / period) / month_days * days
|
amount = (amount_to_spread / period) / month_days * days
|
||||||
|
if self.days_calc:
|
||||||
|
spread_start_date = self._get_spread_start_date(
|
||||||
|
self.period_type, spread_end_date
|
||||||
|
)
|
||||||
|
days = (spread_end_date - spread_start_date).days + 1
|
||||||
|
amount = self._get_amount_per_day(amount_to_spread) * days
|
||||||
return amount
|
return amount
|
||||||
|
|
||||||
def compute_spread_board(self):
|
def compute_spread_board(self):
|
||||||
|
@ -50,6 +50,11 @@ class AccountSpreadTemplate(models.Model):
|
|||||||
help="Period length for the entries",
|
help="Period length for the entries",
|
||||||
)
|
)
|
||||||
start_date = fields.Date()
|
start_date = fields.Date()
|
||||||
|
days_calc = fields.Boolean(
|
||||||
|
string="Calculate by days",
|
||||||
|
default=False,
|
||||||
|
help="Use number of days to calculate amount",
|
||||||
|
)
|
||||||
auto_spread = fields.Boolean(
|
auto_spread = fields.Boolean(
|
||||||
string="Auto assign template on invoice validate",
|
string="Auto assign template on invoice validate",
|
||||||
help="If checked, provide option to auto create spread during "
|
help="If checked, provide option to auto create spread during "
|
||||||
@ -120,6 +125,7 @@ class AccountSpreadTemplate(models.Model):
|
|||||||
"template_id": self.id,
|
"template_id": self.id,
|
||||||
"journal_id": self.spread_journal_id.id,
|
"journal_id": self.spread_journal_id.id,
|
||||||
"use_invoice_line_account": self.use_invoice_line_account,
|
"use_invoice_line_account": self.use_invoice_line_account,
|
||||||
|
"days_calc": self.days_calc,
|
||||||
"company_id": company.id,
|
"company_id": company.id,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -249,6 +249,7 @@
|
|||||||
domain="[('company_id', '=', company_id)]"
|
domain="[('company_id', '=', company_id)]"
|
||||||
widget="selection"
|
widget="selection"
|
||||||
/>
|
/>
|
||||||
|
<field name="days_calc" />
|
||||||
</group>
|
</group>
|
||||||
</group>
|
</group>
|
||||||
<notebook name="notebook">
|
<notebook name="notebook">
|
||||||
|
@ -26,6 +26,7 @@
|
|||||||
<field name="period_number" />
|
<field name="period_number" />
|
||||||
<field name="period_type" />
|
<field name="period_type" />
|
||||||
<field name="start_date" />
|
<field name="start_date" />
|
||||||
|
<field name="days_calc" />
|
||||||
</group>
|
</group>
|
||||||
<group>
|
<group>
|
||||||
<field name="use_invoice_line_account" />
|
<field name="use_invoice_line_account" />
|
||||||
|
Loading…
x
Reference in New Issue
Block a user