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.

272 lines
11 KiB

  1. # © 2016 Julien Coux (Camptocamp)
  2. # Copyright 2020 ForgeFlow S.L. (https://www.forgeflow.com)
  3. # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
  4. import operator
  5. from datetime import date, datetime
  6. from odoo import api, models
  7. from odoo.tools import float_is_zero
  8. class OpenItemsReport(models.AbstractModel):
  9. _name = "report.account_financial_report.open_items"
  10. _description = "Open Items Report"
  11. _inherit = "report.account_financial_report.abstract_report"
  12. def _get_account_partial_reconciled(self, company_id, date_at_object):
  13. domain = [("max_date", ">", date_at_object), ("company_id", "=", company_id)]
  14. fields = ["debit_move_id", "credit_move_id", "amount"]
  15. accounts_partial_reconcile = self.env["account.partial.reconcile"].search_read(
  16. domain=domain, fields=fields
  17. )
  18. debit_amount = {}
  19. credit_amount = {}
  20. for account_partial_reconcile_data in accounts_partial_reconcile:
  21. debit_move_id = account_partial_reconcile_data["debit_move_id"][0]
  22. credit_move_id = account_partial_reconcile_data["credit_move_id"][0]
  23. if debit_move_id not in debit_amount.keys():
  24. debit_amount[debit_move_id] = 0.0
  25. debit_amount[debit_move_id] += account_partial_reconcile_data["amount"]
  26. if credit_move_id not in credit_amount.keys():
  27. credit_amount[credit_move_id] = 0.0
  28. credit_amount[credit_move_id] += account_partial_reconcile_data["amount"]
  29. account_partial_reconcile_data.update(
  30. {"debit_move_id": debit_move_id, "credit_move_id": credit_move_id}
  31. )
  32. return accounts_partial_reconcile, debit_amount, credit_amount
  33. def _get_data(
  34. self,
  35. account_ids,
  36. partner_ids,
  37. date_at_object,
  38. only_posted_moves,
  39. company_id,
  40. date_from,
  41. ):
  42. domain = self._get_move_lines_domain_not_reconciled(
  43. company_id, account_ids, partner_ids, only_posted_moves, date_from
  44. )
  45. ml_fields = [
  46. "id",
  47. "name",
  48. "date",
  49. "move_id",
  50. "journal_id",
  51. "account_id",
  52. "partner_id",
  53. "amount_residual",
  54. "date_maturity",
  55. "ref",
  56. "debit",
  57. "credit",
  58. "reconciled",
  59. "currency_id",
  60. "amount_currency",
  61. "amount_residual_currency",
  62. ]
  63. move_lines = self.env["account.move.line"].search_read(
  64. domain=domain, fields=ml_fields
  65. )
  66. journals_ids = set()
  67. partners_ids = set()
  68. partners_data = {}
  69. if date_at_object < date.today():
  70. (
  71. acc_partial_rec,
  72. debit_amount,
  73. credit_amount,
  74. ) = self._get_account_partial_reconciled(company_id, date_at_object)
  75. if acc_partial_rec:
  76. ml_ids = list(map(operator.itemgetter("id"), move_lines))
  77. debit_ids = list(
  78. map(operator.itemgetter("debit_move_id"), acc_partial_rec)
  79. )
  80. credit_ids = list(
  81. map(operator.itemgetter("credit_move_id"), acc_partial_rec)
  82. )
  83. move_lines = self._recalculate_move_lines(
  84. move_lines,
  85. debit_ids,
  86. credit_ids,
  87. debit_amount,
  88. credit_amount,
  89. ml_ids,
  90. account_ids,
  91. company_id,
  92. partner_ids,
  93. only_posted_moves,
  94. )
  95. move_lines = [
  96. move_line
  97. for move_line in move_lines
  98. if move_line["date"] <= date_at_object
  99. and not float_is_zero(move_line["amount_residual"], precision_digits=2)
  100. ]
  101. open_items_move_lines_data = {}
  102. for move_line in move_lines:
  103. journals_ids.add(move_line["journal_id"][0])
  104. acc_id = move_line["account_id"][0]
  105. # Partners data
  106. if move_line["partner_id"]:
  107. prt_id = move_line["partner_id"][0]
  108. prt_name = move_line["partner_id"][1]
  109. else:
  110. prt_id = 0
  111. prt_name = "Missing Partner"
  112. if prt_id not in partners_ids:
  113. partners_data.update({prt_id: {"id": prt_id, "name": prt_name}})
  114. partners_ids.add(prt_id)
  115. # Move line update
  116. original = 0
  117. if not float_is_zero(move_line["credit"], precision_digits=2):
  118. original = move_line["credit"] * (-1)
  119. if not float_is_zero(move_line["debit"], precision_digits=2):
  120. original = move_line["debit"]
  121. if move_line["ref"] == move_line["name"]:
  122. if move_line["ref"]:
  123. ref_label = move_line["ref"]
  124. else:
  125. ref_label = ""
  126. elif not move_line["ref"]:
  127. ref_label = move_line["name"]
  128. elif not move_line["name"]:
  129. ref_label = move_line["ref"]
  130. else:
  131. ref_label = move_line["ref"] + str(" - ") + move_line["name"]
  132. move_line.update(
  133. {
  134. "date": move_line["date"],
  135. "date_maturity": move_line["date_maturity"]
  136. and move_line["date_maturity"].strftime("%d/%m/%Y"),
  137. "original": original,
  138. "partner_id": prt_id,
  139. "partner_name": prt_name,
  140. "ref_label": ref_label,
  141. "journal_id": move_line["journal_id"][0],
  142. "move_name": move_line["move_id"][1],
  143. "entry_id": move_line["move_id"][0],
  144. "currency_id": move_line["currency_id"][0]
  145. if move_line["currency_id"]
  146. else False,
  147. "currency_name": move_line["currency_id"][1]
  148. if move_line["currency_id"]
  149. else False,
  150. }
  151. )
  152. # Open Items Move Lines Data
  153. if acc_id not in open_items_move_lines_data.keys():
  154. open_items_move_lines_data[acc_id] = {prt_id: [move_line]}
  155. else:
  156. if prt_id not in open_items_move_lines_data[acc_id].keys():
  157. open_items_move_lines_data[acc_id][prt_id] = [move_line]
  158. else:
  159. open_items_move_lines_data[acc_id][prt_id].append(move_line)
  160. journals_data = self._get_journals_data(list(journals_ids))
  161. accounts_data = self._get_accounts_data(open_items_move_lines_data.keys())
  162. return (
  163. move_lines,
  164. partners_data,
  165. journals_data,
  166. accounts_data,
  167. open_items_move_lines_data,
  168. )
  169. @api.model
  170. def _calculate_amounts(self, open_items_move_lines_data):
  171. total_amount = {}
  172. for account_id in open_items_move_lines_data.keys():
  173. total_amount[account_id] = {}
  174. total_amount[account_id]["residual"] = 0.0
  175. for partner_id in open_items_move_lines_data[account_id].keys():
  176. total_amount[account_id][partner_id] = {}
  177. total_amount[account_id][partner_id]["residual"] = 0.0
  178. for move_line in open_items_move_lines_data[account_id][partner_id]:
  179. total_amount[account_id][partner_id]["residual"] += move_line[
  180. "amount_residual"
  181. ]
  182. total_amount[account_id]["residual"] += move_line["amount_residual"]
  183. return total_amount
  184. @api.model
  185. def _order_open_items_by_date(
  186. self, open_items_move_lines_data, show_partner_details
  187. ):
  188. new_open_items = {}
  189. if not show_partner_details:
  190. for acc_id in open_items_move_lines_data.keys():
  191. new_open_items[acc_id] = {}
  192. move_lines = []
  193. for prt_id in open_items_move_lines_data[acc_id]:
  194. for move_line in open_items_move_lines_data[acc_id][prt_id]:
  195. move_lines += [move_line]
  196. move_lines = sorted(move_lines, key=lambda k: (k["date"]))
  197. new_open_items[acc_id] = move_lines
  198. else:
  199. for acc_id in open_items_move_lines_data.keys():
  200. new_open_items[acc_id] = {}
  201. for prt_id in open_items_move_lines_data[acc_id]:
  202. new_open_items[acc_id][prt_id] = {}
  203. move_lines = []
  204. for move_line in open_items_move_lines_data[acc_id][prt_id]:
  205. move_lines += [move_line]
  206. move_lines = sorted(move_lines, key=lambda k: (k["date"]))
  207. new_open_items[acc_id][prt_id] = move_lines
  208. return new_open_items
  209. def _get_report_values(self, docids, data):
  210. wizard_id = data["wizard_id"]
  211. company = self.env["res.company"].browse(data["company_id"])
  212. company_id = data["company_id"]
  213. account_ids = data["account_ids"]
  214. partner_ids = data["partner_ids"]
  215. date_at = data["date_at"]
  216. date_at_object = datetime.strptime(date_at, "%Y-%m-%d").date()
  217. date_from = data["date_from"]
  218. only_posted_moves = data["only_posted_moves"]
  219. show_partner_details = data["show_partner_details"]
  220. (
  221. move_lines_data,
  222. partners_data,
  223. journals_data,
  224. accounts_data,
  225. open_items_move_lines_data,
  226. ) = self._get_data(
  227. account_ids,
  228. partner_ids,
  229. date_at_object,
  230. only_posted_moves,
  231. company_id,
  232. date_from,
  233. )
  234. total_amount = self._calculate_amounts(open_items_move_lines_data)
  235. open_items_move_lines_data = self._order_open_items_by_date(
  236. open_items_move_lines_data, show_partner_details
  237. )
  238. return {
  239. "doc_ids": [wizard_id],
  240. "doc_model": "open.items.report.wizard",
  241. "docs": self.env["open.items.report.wizard"].browse(wizard_id),
  242. "foreign_currency": data["foreign_currency"],
  243. "show_partner_details": data["show_partner_details"],
  244. "company_name": company.display_name,
  245. "company_currency": company.currency_id,
  246. "currency_name": company.currency_id.name,
  247. "date_at": date_at_object.strftime("%d/%m/%Y"),
  248. "hide_account_at_0": data["hide_account_at_0"],
  249. "target_move": data["target_move"],
  250. "journals_data": journals_data,
  251. "partners_data": partners_data,
  252. "accounts_data": accounts_data,
  253. "total_amount": total_amount,
  254. "Open_Items": open_items_move_lines_data,
  255. }