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.

362 lines
15 KiB

12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
12 years ago
  1. # -*- coding: utf-8 -*-
  2. ###########################################################################
  3. # Module Writen to OpenERP, Open Source Management Solution
  4. # Copyright (C) OpenERP Venezuela (<http://openerp.com.ve>).
  5. # Credits######################################################
  6. # Coded by: Humberto Arocha humberto@openerp.com.ve
  7. # Angelica Barrios angelicaisabelb@gmail.com
  8. # Jordi Esteve <jesteve@zikzakmedia.com>
  9. # Javier Duran <javieredm@gmail.com>
  10. # Planified by: Humberto Arocha
  11. # Finance by: LUBCAN COL S.A.S http://www.lubcancol.com
  12. # Audited by: Humberto Arocha humberto@openerp.com.ve
  13. #############################################################################
  14. # This program is free software: you can redistribute it and/or modify
  15. # it under the terms of the GNU General Public License as published by
  16. # the Free Software Foundation, either version 3 of the License, or
  17. # (at your option) any later version.
  18. #
  19. # This program is distributed in the hope that it will be useful,
  20. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  21. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  22. # GNU General Public License for more details.
  23. #
  24. # You should have received a copy of the GNU General Public License
  25. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  26. ##############################################################################
  27. from openerp.osv import osv, fields
  28. import time
  29. from openerp.tools.translate import _
  30. class wizard_report(osv.osv_memory):
  31. _name = "wizard.report"
  32. _columns = {
  33. 'afr_id': fields.many2one(
  34. 'afr', 'Custom Report',
  35. help='If you have already set a Custom Report, Select it Here.'),
  36. 'company_id': fields.many2one('res.company', 'Company', required=True),
  37. 'currency_id': fields.many2one(
  38. 'res.currency', 'Currency',
  39. help="Currency at which this report will be expressed. If not \
  40. selected will be used the one set in the company"),
  41. 'inf_type': fields.selection([('BS', 'Balance Sheet'),
  42. ('IS', 'Income Statement')],
  43. 'Type',
  44. required=True),
  45. 'columns': fields.selection(
  46. [('one', 'End. Balance'),
  47. ('two', 'Debit | Credit'),
  48. ('four', 'Initial | Debit | Credit | YTD'),
  49. ('five', 'Initial | Debit | Credit | Period | YTD'),
  50. ('qtr', "4 QTR's | YTD"), ('thirteen', '12 Months | YTD')],
  51. 'Columns', required=True),
  52. 'display_account': fields.selection(
  53. [('all', 'All Accounts'),
  54. ('bal', 'With Balance'),
  55. ('mov', 'With movements'),
  56. ('bal_mov', 'With Balance / Movements')],
  57. 'Display accounts'),
  58. 'display_account_level': fields.integer(
  59. 'Up to level',
  60. help='Display accounts up to this level (0 to show all)'),
  61. 'account_list': fields.many2many('account.account',
  62. 'rel_wizard_account',
  63. 'account_list',
  64. 'account_id',
  65. 'Root accounts',
  66. required=True),
  67. 'fiscalyear': fields.many2one('account.fiscalyear', 'Fiscal year',
  68. help='Fiscal Year for this report',
  69. required=True),
  70. 'periods': fields.many2many(
  71. 'account.period', 'rel_wizard_period',
  72. 'wizard_id', 'period_id', 'Periods',
  73. help='All periods in the fiscal year if empty'),
  74. 'analytic_ledger': fields.boolean(
  75. 'Analytic Ledger',
  76. help="Allows to Generate an Analytic Ledger for accounts with \
  77. moves. Available when Balance Sheet and 'Initial | Debit | Credit \
  78. | YTD' are selected"),
  79. 'journal_ledger': fields.boolean(
  80. 'Journal Ledger',
  81. help="Allows to Generate an Journal Ledger for accounts with \
  82. moves. Available when Balance Sheet and 'Initial | Debit | Credit \
  83. | YTD' are selected"),
  84. 'partner_balance': fields.boolean(
  85. 'Partner Balance',
  86. help="Allows to Generate a Partner Balance for accounts with \
  87. moves. Available when Balance Sheet and 'Initial | Debit | Credit \
  88. | YTD' are selected"),
  89. 'tot_check': fields.boolean('Summarize?',
  90. help='Checking will add a new line at the \
  91. end of the Report which will Summarize \
  92. Columns in Report'),
  93. 'lab_str': fields.char('Description',
  94. help='Description for the Summary', size=128),
  95. 'target_move': fields.selection(
  96. [('posted', 'All Posted Entries'),
  97. ('all', 'All Entries'),
  98. ], 'Entries to Include',
  99. required=True,
  100. help='Print All Accounting Entries or just Posted Accounting \
  101. Entries'),
  102. # ~ Deprecated fields
  103. 'filter': fields.selection([('bydate', 'By Date'),
  104. ('byperiod', 'By Period'),
  105. ('all', 'By Date and Period'),
  106. ('none', 'No Filter')],
  107. 'Date/Period Filter'),
  108. 'date_to': fields.date('End date'),
  109. 'date_from': fields.date('Start date'),
  110. }
  111. _defaults = {
  112. 'date_from': lambda *a: time.strftime('%Y-%m-%d'),
  113. 'date_to': lambda *a: time.strftime('%Y-%m-%d'),
  114. 'filter': lambda *a: 'byperiod',
  115. 'display_account_level': lambda *a: 0,
  116. 'inf_type': lambda *a: 'BS',
  117. 'company_id': lambda self, cr, uid, c: self.pool['res.company'].
  118. _company_default_get(cr, uid, 'account.invoice', context=c),
  119. 'fiscalyear': lambda self, cr, uid, c: self.
  120. pool['account.fiscalyear'].find(cr, uid),
  121. 'display_account': lambda *a: 'bal_mov',
  122. 'columns': lambda *a: 'five',
  123. 'target_move': 'posted',
  124. }
  125. def onchange_inf_type(self, cr, uid, ids, inf_type, context=None):
  126. if context is None:
  127. context = {}
  128. res = {'value': {}}
  129. if inf_type != 'BS':
  130. res['value'].update({'analytic_ledger': False})
  131. return res
  132. def onchange_columns(self, cr, uid, ids, columns, fiscalyear, periods,
  133. context=None):
  134. if context is None:
  135. context = {}
  136. res = {'value': {}}
  137. p_obj = self.pool.get("account.period")
  138. all_periods = p_obj.search(cr, uid,
  139. [('fiscalyear_id', '=', fiscalyear),
  140. ('special', '=', False)], context=context)
  141. s = set(periods[0][2])
  142. t = set(all_periods)
  143. go = periods[0][2] and s.issubset(t) or False
  144. if columns != 'four':
  145. res['value'].update({'analytic_ledger': False})
  146. if columns in ('qtr', 'thirteen'):
  147. res['value'].update({'periods': all_periods})
  148. else:
  149. if go:
  150. res['value'].update({'periods': periods})
  151. else:
  152. res['value'].update({'periods': []})
  153. return res
  154. def onchange_analytic_ledger(self, cr, uid, ids, company_id,
  155. analytic_ledger, context=None):
  156. if context is None:
  157. context = {}
  158. context['company_id'] = company_id
  159. res = {'value': {}}
  160. cur_id = self.pool.get('res.company').browse(
  161. cr, uid, company_id, context=context).currency_id.id
  162. res['value'].update({'currency_id': cur_id})
  163. return res
  164. def onchange_company_id(self, cr, uid, ids, company_id, context=None):
  165. if context is None:
  166. context = {}
  167. context['company_id'] = company_id
  168. res = {'value': {}}
  169. if not company_id:
  170. return res
  171. cur_id = self.pool.get('res.company').browse(
  172. cr, uid, company_id, context=context).currency_id.id
  173. fy_id = self.pool.get('account.fiscalyear').find(
  174. cr, uid, context=context)
  175. res['value'].update({'fiscalyear': fy_id})
  176. res['value'].update({'currency_id': cur_id})
  177. res['value'].update({'account_list': []})
  178. res['value'].update({'periods': []})
  179. res['value'].update({'afr_id': None})
  180. return res
  181. def onchange_afr_id(self, cr, uid, ids, afr_id, context=None):
  182. if context is None:
  183. context = {}
  184. res = {'value': {}}
  185. if not afr_id:
  186. return res
  187. afr_brw = self.pool.get('afr').browse(cr, uid, afr_id, context=context)
  188. res['value']['currency_id'] = (
  189. afr_brw.currency_id and afr_brw.currency_id.id or
  190. afr_brw.company_id.currency_id.id)
  191. res['value']['inf_type'] = afr_brw.inf_type or 'BS'
  192. res['value']['columns'] = afr_brw.columns or 'five'
  193. res['value']['display_account'] = afr_brw.display_account or 'bal_mov'
  194. res['value']['display_account_level'] = (
  195. afr_brw.display_account_level or 0)
  196. res['value']['fiscalyear'] = (
  197. afr_brw.fiscalyear_id and afr_brw.fiscalyear_id.id)
  198. res['value']['account_list'] = [acc.id for acc in afr_brw.account_ids]
  199. res['value']['periods'] = [p.id for p in afr_brw.period_ids]
  200. res['value']['analytic_ledger'] = afr_brw.analytic_ledger or False
  201. res['value']['tot_check'] = afr_brw.tot_check or False
  202. res['value']['lab_str'] = afr_brw.lab_str or _(
  203. 'Write a Description for your Summary Total')
  204. return res
  205. def _get_defaults(self, cr, uid, data, context=None):
  206. if context is None:
  207. context = {}
  208. user = self.pool['res.users'].browse(cr, uid, uid, context=context)
  209. if user.company_id:
  210. company_id = user.company_id.id
  211. else:
  212. company_id = self.pool['res.company'].search(
  213. cr, uid, [('parent_id', '=', False)])[0]
  214. data['form']['company_id'] = company_id
  215. fiscalyear_obj = self.pool['account.fiscalyear']
  216. data['form']['fiscalyear'] = fiscalyear_obj.find(cr, uid)
  217. data['form']['context'] = context
  218. return data['form']
  219. def _check_state(self, cr, uid, data, context=None):
  220. if context is None:
  221. context = {}
  222. if data['form']['filter'] == 'bydate':
  223. self._check_date(cr, uid, data, context)
  224. return data['form']
  225. def _check_date(self, cr, uid, data, context=None):
  226. if context is None:
  227. context = {}
  228. if data['form']['date_from'] > data['form']['date_to']:
  229. raise osv.except_osv(_('Error !'), (
  230. 'La fecha final debe ser mayor a la inicial'))
  231. sql = """SELECT f.id, f.date_start, f.date_stop
  232. FROM account_fiscalyear f
  233. WHERE '%s' = f.id """ % (data['form']['fiscalyear'])
  234. cr.execute(sql)
  235. res = cr.dictfetchall()
  236. if res:
  237. if (data['form']['date_to'] > res[0]['date_stop'] or
  238. data['form']['date_from'] < res[0]['date_start']):
  239. raise osv.except_osv(_('UserError'),
  240. 'Las fechas deben estar entre %s y %s'
  241. % (res[0]['date_start'],
  242. res[0]['date_stop']))
  243. else:
  244. return 'report'
  245. else:
  246. raise osv.except_osv(_('UserError'), 'No existe periodo fiscal')
  247. def period_span(self, cr, uid, ids, fy_id, context=None):
  248. if context is None:
  249. context = {}
  250. ap_obj = self.pool.get('account.period')
  251. fy_id = fy_id and type(fy_id) in (list, tuple) and fy_id[0] or fy_id
  252. if not ids:
  253. # ~ No hay periodos
  254. return ap_obj.search(cr, uid, [('fiscalyear_id', '=', fy_id),
  255. ('special', '=', False)],
  256. order='date_start asc')
  257. ap_brws = ap_obj.browse(cr, uid, ids, context=context)
  258. date_start = min([period.date_start for period in ap_brws])
  259. date_stop = max([period.date_stop for period in ap_brws])
  260. return ap_obj.search(cr, uid, [('fiscalyear_id', '=', fy_id),
  261. ('special', '=', False),
  262. ('date_start', '>=', date_start),
  263. ('date_stop', '<=', date_stop)],
  264. order='date_start asc')
  265. def print_report(self, cr, uid, ids, data, context=None):
  266. if context is None:
  267. context = {}
  268. data = {}
  269. data['ids'] = context.get('active_ids', [])
  270. data['model'] = context.get('active_model', 'ir.ui.menu')
  271. data['form'] = self.read(cr, uid, ids[0])
  272. if data['form']['filter'] == 'byperiod':
  273. del data['form']['date_from']
  274. del data['form']['date_to']
  275. data['form']['periods'] = self.period_span(
  276. cr, uid,
  277. data['form']['periods'],
  278. data['form']['fiscalyear'])
  279. elif data['form']['filter'] == 'bydate':
  280. self._check_date(cr, uid, data)
  281. del data['form']['periods']
  282. elif data['form']['filter'] == 'none':
  283. del data['form']['date_from']
  284. del data['form']['date_to']
  285. del data['form']['periods']
  286. else:
  287. self._check_date(cr, uid, data)
  288. lis2 = str(data['form']['periods']).replace(
  289. "[", "(").replace("]", ")")
  290. sqlmm = """select min(p.date_start) as inicio,
  291. max(p.date_stop) as fin
  292. from account_period p
  293. where p.id in %s""" % lis2
  294. cr.execute(sqlmm)
  295. minmax = cr.dictfetchall()
  296. if minmax:
  297. if (data['form']['date_to'] < minmax[0]['inicio']) \
  298. or (data['form']['date_from'] > minmax[0]['fin']):
  299. raise osv.except_osv(_('Error !'), _(
  300. 'La interseccion entre el periodo y fecha es vacio'))
  301. if data['form']['columns'] == 'one':
  302. name = 'afr.1cols'
  303. if data['form']['columns'] == 'two':
  304. name = 'afr.2cols'
  305. if data['form']['columns'] == 'four':
  306. if data['form']['analytic_ledger'] \
  307. and data['form']['inf_type'] == 'BS':
  308. name = 'afr.analytic.ledger'
  309. elif data['form']['journal_ledger'] \
  310. and data['form']['inf_type'] == 'BS':
  311. name = 'afr.journal.ledger'
  312. elif data['form']['partner_balance'] \
  313. and data['form']['inf_type'] == 'BS':
  314. name = 'afr.partner.balance'
  315. else:
  316. name = 'afr.4cols'
  317. if data['form']['columns'] == 'five':
  318. name = 'afr.5cols'
  319. if data['form']['columns'] == 'qtr':
  320. name = 'afr.qtrcols'
  321. if data['form']['columns'] == 'thirteen':
  322. name = 'afr.13cols'
  323. return {'type': 'ir.actions.report.xml',
  324. 'report_name': name,
  325. 'datas': data}