You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

191 lines
8.0 KiB

  1. # Copyright 2019 Coop IT Easy SCRL fs
  2. # Houssine BAKKALI <houssine@coopiteasy.be>
  3. # License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
  4. import logging
  5. from odoo import api, fields, models, _
  6. _logger = logging.getLogger(__name__)
  7. class LoanIssue(models.Model):
  8. _name = 'loan.issue'
  9. _description = 'Loan Issue'
  10. @api.multi
  11. def _compute_subscribed_amount(self):
  12. for issue in self:
  13. susbscribed_amount = 0.0
  14. for line in issue.loan_issue_lines.filtered(
  15. lambda record: record.state != 'cancelled'):
  16. susbscribed_amount += line.amount
  17. issue.subscribed_amount = susbscribed_amount
  18. name = fields.Char(string="Name",
  19. translate=True)
  20. default_issue = fields.Boolean(string="Default issue")
  21. subscription_start_date = fields.Date(string="Start date subscription period")
  22. subscription_end_date = fields.Date(string="End date subscription period")
  23. user_id = fields.Many2one('res.users',
  24. string="Responsible")
  25. loan_start_date = fields.Date(string="Loan start date")
  26. term_date = fields.Date(string="Term date")
  27. loan_term = fields.Float(string="Duration of the loan in month")
  28. rate = fields.Float(string="Interest rate")
  29. face_value = fields.Monetary(string="Facial value",
  30. currency_field='company_currency_id',
  31. required=True)
  32. minimum_amount = fields.Monetary(string="Minimum amount of issue",
  33. currency_field='company_currency_id')
  34. maximum_amount = fields.Monetary(string="Maximum amount of issue",
  35. currency_field='company_currency_id')
  36. min_amount_company = fields.Monetary(string="Minimum amount for a company",
  37. currency_field='company_currency_id')
  38. max_amount_company = fields.Monetary(string="Maximum amount for a company",
  39. currency_field='company_currency_id')
  40. min_amount_person = fields.Monetary(string="Minimum amount for a person",
  41. currency_field='company_currency_id')
  42. max_amount_person = fields.Monetary(string="Maximum amount for a person",
  43. currency_field='company_currency_id')
  44. subscribed_amount = fields.Monetary(string="Subscribed amount",
  45. compute="_compute_subscribed_amount",
  46. currency_field='company_currency_id')
  47. interest_payment = fields.Selection([('end', 'End'),
  48. ('yearly', 'Yearly')],
  49. string="Interest payment")
  50. interest_payment_info = fields.Char(string="Yearly payment on")
  51. loan_issue_lines = fields.One2many('loan.issue.line',
  52. 'loan_issue_id',
  53. string="Loan issue lines")
  54. state = fields.Selection([('draft', 'Draft'),
  55. ('confirmed', 'Confirmed'),
  56. ('cancelled', 'Cancelled'),
  57. ('ongoing', 'Ongoing'),
  58. ('closed', 'Closed')],
  59. string="State",
  60. default='draft')
  61. company_currency_id = fields.Many2one('res.currency',
  62. related='company_id.currency_id',
  63. string="Company Currency",
  64. readonly=True)
  65. company_id = fields.Many2one('res.company',
  66. string='Company',
  67. required=True,
  68. readonly=True,
  69. default=lambda self: self.env['res.company']._company_default_get()) # noqa
  70. by_company = fields.Boolean(string="By company")
  71. by_individual = fields.Boolean(string='By individuals')
  72. display_on_website = fields.Boolean(sting='Display on website')
  73. taxes_rate = fields.Float(string="Taxes on interest",
  74. required=True)
  75. @api.multi
  76. def get_max_amount(self, partner):
  77. self.ensure_one()
  78. lines = self.loan_issue_lines.filtered(
  79. lambda r: r.partner_id == partner and r.state != 'cancelled')
  80. already_subscribed = sum(line.amount for line in lines)
  81. if partner.is_company:
  82. max_amount = self.max_amount_company - already_subscribed
  83. else:
  84. max_amount = self.max_amount_person - already_subscribed
  85. return max_amount
  86. @api.multi
  87. def get_min_amount(self, partner):
  88. self.ensure_one()
  89. lines = self.loan_issue_lines.filtered(
  90. lambda r: r.partner_id == partner and r.state != 'cancelled')
  91. amount_subscribed = sum(line.amount for line in lines)
  92. if partner.is_company:
  93. min_amount = self.min_amount_company - amount_subscribed
  94. else:
  95. min_amount = self.min_amount_person - amount_subscribed
  96. return max(0, min_amount)
  97. @api.multi
  98. def get_web_issues(self, is_company):
  99. bond_issues = self.search([
  100. ('display_on_website', '=', True),
  101. ('state', '=', 'ongoing')
  102. ])
  103. if is_company is True:
  104. return bond_issues.filtered('by_company')
  105. else:
  106. return bond_issues.filtered('by_company')
  107. @api.multi
  108. def action_confirm(self):
  109. self.ensure_one()
  110. self.write({'state': 'confirmed'})
  111. @api.multi
  112. def action_open(self):
  113. self.ensure_one()
  114. self.write({'state': 'ongoing'})
  115. @api.multi
  116. def action_draft(self):
  117. self.ensure_one()
  118. self.write({'state': 'draft'})
  119. @api.multi
  120. def action_cancel(self):
  121. self.ensure_one()
  122. self.write({'state': 'cancelled'})
  123. @api.multi
  124. def action_close(self):
  125. self.ensure_one()
  126. self.write({'state': 'closed'})
  127. def get_interest_vals(self, line, vals):
  128. interest_obj = self.env['loan.interest.line']
  129. accrued_amount = line.amount
  130. accrued_interest = 0
  131. accrued_net_interest = 0
  132. accrued_taxes = 0
  133. for year in range(1, int(self.loan_term) + 1):
  134. interest = accrued_amount * (line.loan_issue_id.rate / 100)
  135. accrued_amount += interest
  136. taxes_amount = interest * (self.taxes_rate / 100)
  137. net_interest = interest - taxes_amount
  138. accrued_interest += interest
  139. accrued_net_interest += net_interest
  140. accrued_taxes += taxes_amount
  141. vals['interest'] = interest
  142. vals['net_interest'] = net_interest
  143. vals['taxes_amount'] = taxes_amount
  144. vals['accrued_amount'] = accrued_amount
  145. vals['accrued_interest'] = accrued_interest
  146. vals['accrued_net_interest'] = accrued_net_interest
  147. vals['accrued_taxes'] = accrued_taxes
  148. vals['name'] = year
  149. interest_obj.create(vals)
  150. @api.multi
  151. def compute_loan_interest(self):
  152. self.ensure_one()
  153. if self.interest_payment == 'end':
  154. due_date = self.term_date
  155. else:
  156. raise NotImplementedError(_("Interest payment by year hasn't been "
  157. "implemented yet"))
  158. for line in self.loan_issue_lines:
  159. # TODO remove this line
  160. line.interest_lines.unlink()
  161. # Please Do not Forget
  162. vals = {
  163. 'issue_line': line.id,
  164. 'due_date': due_date,
  165. 'taxes_rate': self.taxes_rate
  166. }
  167. self.get_interest_vals(line, vals)
  168. rounded_term = int(self.loan_term)
  169. if self.loan_term - rounded_term > 0:
  170. # TODO Handle this case
  171. raise NotImplementedError(_("Calculation on non entire year "
  172. "hasn't been implemented yet"))