GoodERP
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

451 行
21KB

  1. # Copyright 2016 上海开阖软件有限公司 (http://www.osbzr.com)
  2. # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
  3. from odoo.exceptions import UserError
  4. from odoo import fields, models, api
  5. from odoo.tools import float_compare, float_is_zero
  6. class ReconcileOrder(models.Model):
  7. _name = 'reconcile.order'
  8. _description = '核销单'
  9. _inherit = ['mail.thread']
  10. TYPE_SELECTION = [
  11. ('adv_pay_to_get', '预收冲应收'),
  12. ('adv_get_to_pay', '预付冲应付'),
  13. ('get_to_pay', '应收冲应付'),
  14. ('get_to_get', '应收转应收'),
  15. ('pay_to_pay', '应付转应付'),
  16. ]
  17. state = fields.Selection([
  18. ('draft', '草稿'),
  19. ('done', '已确认'),
  20. ('cancel', '已作废'),
  21. ], string='状态', readonly=True,
  22. default='draft', copy=False, index=True,
  23. tracking=True,
  24. help='核销单状态标识,新建时状态为草稿;确认后状态为已确认')
  25. partner_id = fields.Many2one('partner', string='往来单位', required=True,
  26. ondelete='restrict',
  27. help='该单据对应的业务伙伴,与业务类型一起带出待核销的明细行')
  28. to_partner_id = fields.Many2one('partner', string='转入往来单位',
  29. ondelete='restrict',
  30. help='应收转应收、应付转应付时对应的转入业务伙伴,'
  31. '订单确认时会影响该业务伙伴的应收/应付')
  32. advance_payment_ids = fields.One2many(
  33. 'advance.payment', 'pay_reconcile_id',
  34. string='预收/付款单行',
  35. help='业务伙伴有预收/付款单,自动带出,用来与应收/应付款单核销')
  36. receivable_source_ids = fields.One2many(
  37. 'source.order.line', 'receivable_reconcile_id',
  38. string='应收结算单行',
  39. help='业务伙伴有应收结算单,自动带出,待与预收款单核销')
  40. payable_source_ids = fields.One2many(
  41. 'source.order.line', 'payable_reconcile_id',
  42. string='应付结算单行',
  43. help='业务伙伴有应付结算单,自动带出,待与预付款单核销')
  44. business_type = fields.Selection(TYPE_SELECTION, string='业务类型',
  45. help='类型:预收冲应收,预付冲应付,应收冲应付,应收转应收,应付转应付'
  46. )
  47. name = fields.Char(string='单据编号', copy=False, readonly=True,
  48. help='单据编号,创建时会自动生成')
  49. date = fields.Date(string='单据日期',
  50. default=lambda self: fields.Date.context_today(self),
  51. help='单据创建日期')
  52. note = fields.Text(string='备注',
  53. help='可以为该单据添加一些需要的标识信息')
  54. company_id = fields.Many2one(
  55. 'res.company',
  56. string='公司',
  57. change_default=True,
  58. default=lambda self: self.env.company)
  59. @api.model_create_multi
  60. def create(self, vals_list):
  61. # 创建时查找该业务伙伴是否存在 未审核 状态下的核销单
  62. for values in vals_list:
  63. if values.get('partner_id'):
  64. orders = self.env['reconcile.order'].search([
  65. ('partner_id', '=', values.get('partner_id')),
  66. ('state', '=', 'draft'),
  67. ('id', '!=', self.id),
  68. ('business_type', '=', values.get('business_type'))])
  69. if orders:
  70. raise UserError(
  71. '业务伙伴(%s)、业务类型(%s)存在未审核的核销单,请先审核' % (
  72. orders.partner_id.name,
  73. dict(self.fields_get(
  74. allfields=['business_type']
  75. )['business_type']['selection']
  76. )[orders.business_type]))
  77. return super(ReconcileOrder, self).create(vals_list)
  78. def write(self, values):
  79. # 写入时查找该业务伙伴是否存在 未审核 状态下的核销单
  80. orders = self.env['reconcile.order'].search([
  81. ('partner_id', '=',
  82. (values.get('partner_id') or self.partner_id.id)),
  83. ('state', '=', 'draft'),
  84. ('id', '!=', self.id),
  85. ('business_type', '=',
  86. (values.get('business_type') or self.business_type))])
  87. if orders:
  88. raise UserError(
  89. '业务伙伴(%s)、业务类型(%s)存在未审核的核销单,请先审核' % (
  90. orders.partner_id.name,
  91. dict(self.fields_get(
  92. allfields=['business_type'])['business_type']
  93. ['selection'])[orders.business_type]))
  94. return super(ReconcileOrder, self).write(values)
  95. def _get_money_order(self, way='get'):
  96. """
  97. 搜索到满足条件的预收/付款单,为one2many字段赋值构造列表
  98. :param way: 收/付款单的type
  99. :return: list
  100. """
  101. money_orders = self.env['money.order'].search(
  102. [('partner_id', '=', self.partner_id.id),
  103. ('type', '=', way),
  104. ('state', '=', 'done'),
  105. ('to_reconcile', '!=', 0)])
  106. result = []
  107. for order in money_orders:
  108. result.append((0, 0, {
  109. 'name': order.id,
  110. 'amount': order.amount,
  111. 'date': order.date,
  112. 'reconciled': order.reconciled,
  113. 'to_reconcile': order.to_reconcile,
  114. 'this_reconcile': order.to_reconcile,
  115. }))
  116. return result
  117. def _get_money_invoice(self, way='income'):
  118. """
  119. 搜索到满足条件的money.invoice记录并且取出invoice对象 构造出one2many的
  120. :param way: money.invoice 中的category_id 的type
  121. :return:
  122. """
  123. MoneyInvoice = self.env['money.invoice'].search([
  124. ('category_id.type', '=', way),
  125. ('partner_id', '=', self.partner_id.id),
  126. ('state', '=', 'done'),
  127. ('to_reconcile', '!=', 0)])
  128. result = []
  129. for invoice in MoneyInvoice:
  130. result.append((0, 0, {
  131. 'name': invoice.id,
  132. 'category_id': invoice.category_id.id,
  133. 'amount': invoice.amount,
  134. 'date': invoice.date,
  135. 'reconciled': invoice.reconciled,
  136. 'to_reconcile': invoice.to_reconcile,
  137. 'date_due': invoice.date_due,
  138. 'this_reconcile': invoice.to_reconcile,
  139. }))
  140. return result
  141. @api.onchange('partner_id', 'to_partner_id', 'business_type')
  142. def onchange_partner_id(self):
  143. """
  144. onchange 类型字段 当改变 客户或者转入往来单位 业务类型 自动生成 对应的
  145. 核销单各种明细。
  146. :return:
  147. """
  148. if not self.partner_id or not self.business_type:
  149. return {}
  150. # 先清空之前填充的数据
  151. self.advance_payment_ids = None
  152. self.receivable_source_ids = None
  153. self.payable_source_ids = None
  154. if self.business_type == 'adv_pay_to_get': # 预收冲应收
  155. self.advance_payment_ids = self._get_money_order('get')
  156. self.receivable_source_ids = self._get_money_invoice('income')
  157. if self.business_type == 'adv_get_to_pay': # 预付冲应付
  158. self.advance_payment_ids = self._get_money_order('pay')
  159. self.payable_source_ids = self._get_money_invoice('expense')
  160. if self.business_type == 'get_to_pay': # 应收冲应付
  161. self.receivable_source_ids = self._get_money_invoice('income')
  162. self.payable_source_ids = self._get_money_invoice('expense')
  163. if self.business_type == 'get_to_get': # 应收转应收
  164. self.receivable_source_ids = self._get_money_invoice('income')
  165. return {'domain':
  166. {'to_partner_id': [('c_category_id', '!=', False)]}}
  167. if self.business_type == 'pay_to_pay': # 应付转应付
  168. self.payable_source_ids = self._get_money_invoice('expense')
  169. return {'domain':
  170. {'to_partner_id': [('s_category_id', '!=', False)]}}
  171. def _get_or_pay(self, line, business_type,
  172. partner_id, to_partner_id, name):
  173. """
  174. 核销单 核销时 对具体核销单行进行的操作
  175. :param line:
  176. :param business_type:
  177. :param partner_id:
  178. :param to_partner_id:
  179. :param name:
  180. :return:
  181. """
  182. decimal_amount = self.env.ref('core.decimal_amount')
  183. if float_compare(
  184. line.this_reconcile,
  185. line.to_reconcile,
  186. precision_digits=decimal_amount.digits) == 1:
  187. raise UserError('核销金额不能大于未核销金额。\n核销金额:%s 未核销金额:%s' %
  188. (line.this_reconcile, line.to_reconcile))
  189. # 更新每一行的已核销余额、未核销余额
  190. line.name.to_reconcile -= line.this_reconcile
  191. line.name.reconciled += line.this_reconcile
  192. # 应收转应收、应付转应付
  193. if business_type in ['get_to_get', 'pay_to_pay']:
  194. if not float_is_zero(line.this_reconcile, 2):
  195. # 转入业务伙伴往来增加
  196. self.env['money.invoice'].create({
  197. 'name': name,
  198. 'category_id': line.category_id.id,
  199. 'amount': line.this_reconcile,
  200. 'date': self.date,
  201. 'reconciled': 0, # 已核销金额
  202. 'to_reconcile': line.this_reconcile, # 未核销金额
  203. 'date_due': line.date_due,
  204. 'partner_id': to_partner_id.id,
  205. })
  206. # 转出业务伙伴往来减少
  207. to_invoice_id = self.env['money.invoice'].create({
  208. 'name': name,
  209. 'category_id': line.category_id.id,
  210. 'amount': -line.this_reconcile,
  211. 'date': self.date,
  212. 'date_due': line.date_due,
  213. 'partner_id': partner_id.id,
  214. })
  215. # 核销 转出业务伙伴 的转出金额
  216. to_invoice_id.to_reconcile = 0
  217. to_invoice_id.reconciled = -line.this_reconcile
  218. # 应收冲应付,应收行、应付行分别生成负的结算单,并且核销
  219. if business_type in ['get_to_pay']:
  220. if not float_is_zero(line.this_reconcile, 2):
  221. invoice_id = self.env['money.invoice'].create({
  222. 'name': name,
  223. 'category_id': line.category_id.id,
  224. 'amount': -line.this_reconcile,
  225. 'date': self.date,
  226. 'date_due': line.date_due,
  227. 'partner_id': partner_id.id,
  228. })
  229. # 核销 业务伙伴 的本次核销金额
  230. invoice_id.to_reconcile = 0
  231. invoice_id.reconciled = -line.this_reconcile
  232. return True
  233. def reconcile_order_done(self):
  234. '''核销单的审核按钮'''
  235. # 核销金额不能大于未核销金额
  236. for order in self:
  237. if order.state == 'done':
  238. raise UserError('核销单%s已确认,不能再次确认。' % order.name)
  239. order_reconcile, invoice_reconcile = 0, 0
  240. if order.business_type in ['get_to_get', 'pay_to_pay'] \
  241. and order.partner_id == order.to_partner_id:
  242. raise UserError(
  243. '业务伙伴和转入往来单位不能相同。\n业务伙伴:%s 转入往来单位:%s'
  244. % (order.partner_id.name, order.to_partner_id.name))
  245. # 核销预收预付
  246. for line in order.advance_payment_ids:
  247. order_reconcile += line.this_reconcile
  248. decimal_amount = self.env.ref('core.decimal_amount')
  249. if float_compare(
  250. line.this_reconcile,
  251. line.to_reconcile,
  252. precision_digits=decimal_amount.digits) == 1:
  253. raise UserError('核销金额不能大于未核销金额。\n核销金额:%s 未核销金额:%s' % (
  254. line.this_reconcile, line.to_reconcile))
  255. # 更新每一行的已核销余额、未核销余额
  256. line.name.to_reconcile -= line.this_reconcile
  257. line.name.reconciled += line.this_reconcile
  258. for line in order.receivable_source_ids:
  259. invoice_reconcile += line.this_reconcile
  260. self._get_or_pay(line, order.business_type,
  261. order.partner_id,
  262. order.to_partner_id, order.name)
  263. for line in order.payable_source_ids:
  264. if self.business_type == 'adv_get_to_pay':
  265. invoice_reconcile += line.this_reconcile
  266. else:
  267. order_reconcile += line.this_reconcile
  268. self._get_or_pay(line, order.business_type,
  269. order.partner_id,
  270. order.to_partner_id, order.name)
  271. # 核销金额必须相同
  272. if order.business_type in ['adv_pay_to_get',
  273. 'adv_get_to_pay', 'get_to_pay']:
  274. decimal_amount = self.env.ref('core.decimal_amount')
  275. if float_compare(
  276. order_reconcile,
  277. invoice_reconcile,
  278. precision_digits=decimal_amount.digits) != 0:
  279. raise UserError('核销金额必须相同, %s 不等于 %s'
  280. % (order_reconcile, invoice_reconcile))
  281. order.state = 'done'
  282. return True
  283. def _get_or_pay_cancel(self, line, business_type, name):
  284. """
  285. 反核销时 对具体核销单行进行的操作
  286. """
  287. # 每一行的已核销金额减少、未核销金额增加
  288. line.name.to_reconcile += line.this_reconcile
  289. line.name.reconciled -= line.this_reconcile
  290. # 应收转应收、应付转应付、应收冲应付,找到生成的结算单反审核并删除
  291. if business_type in ['get_to_get', 'pay_to_pay', 'get_to_pay']:
  292. invoices = self.env['money.invoice'].search([('name', '=', name)])
  293. for inv in invoices:
  294. if inv.state == 'done':
  295. inv.reconciled = 0.0
  296. inv.money_invoice_draft()
  297. inv.unlink()
  298. return True
  299. def reconcile_order_draft(self):
  300. ''' 核销单的反审核按钮 '''
  301. for order in self:
  302. if order.state == 'draft':
  303. raise UserError('核销单%s已撤销,不能再次撤销。' % order.name)
  304. order_reconcile, invoice_reconcile = 0, 0
  305. if order.business_type in ['get_to_get', 'pay_to_pay'] \
  306. and order.partner_id == order.to_partner_id:
  307. raise UserError(
  308. '业务伙伴和转入往来单位不能相同。\n业务伙伴:%s 转入往来单位:%s'
  309. % (order.partner_id.name, order.to_partner_id.name))
  310. # 反核销预收预付
  311. for line in order.advance_payment_ids:
  312. order_reconcile += line.this_reconcile
  313. # 每一行的已核销余额减少、未核销余额增加
  314. line.name.to_reconcile += line.this_reconcile
  315. line.name.reconciled -= line.this_reconcile
  316. # 反核销应收行
  317. for line in order.receivable_source_ids:
  318. invoice_reconcile += line.this_reconcile
  319. self._get_or_pay_cancel(line, order.business_type, order.name)
  320. # 反核销应付行
  321. for line in order.payable_source_ids:
  322. if order.business_type == 'adv_get_to_pay':
  323. invoice_reconcile += line.this_reconcile
  324. else:
  325. order_reconcile += line.this_reconcile
  326. self._get_or_pay_cancel(line, order.business_type, order.name)
  327. # 反核销时,金额必须相同
  328. if self.business_type in [
  329. 'adv_pay_to_get', 'adv_get_to_pay', 'get_to_pay']:
  330. decimal_amount = self.env.ref('core.decimal_amount')
  331. if float_compare(
  332. order_reconcile,
  333. invoice_reconcile,
  334. precision_digits=decimal_amount.digits) != 0:
  335. raise UserError('反核销时,金额必须相同, %s 不等于 %s'
  336. % (order_reconcile, invoice_reconcile))
  337. order.state = 'draft'
  338. return True
  339. class SourceOrderLine(models.Model):
  340. _name = 'source.order.line'
  341. _description = '待核销行'
  342. money_id = fields.Many2one('money.order', string='收付款单',
  343. ondelete='cascade',
  344. help='待核销行对应的收付款单') # 收付款单上的待核销行
  345. receivable_reconcile_id = fields.Many2one(
  346. 'reconcile.order',
  347. string='应收核销单', ondelete='cascade',
  348. help='核销单上的应收结算单明细') # 核销单上的应收结算单明细
  349. payable_reconcile_id = fields.Many2one('reconcile.order',
  350. string='应付核销单', ondelete='cascade',
  351. help='核销单上的应付结算单明细') # 核销单上的应付结算单明细
  352. name = fields.Many2one('money.invoice', string='发票号',
  353. copy=False, required=True,
  354. ondelete='cascade',
  355. help='待核销行对应的结算单')
  356. category_id = fields.Many2one('core.category', string='类别',
  357. required=True, ondelete='restrict',
  358. help='待核销行类别:采购 或者 销售等')
  359. date = fields.Date(string='单据日期',
  360. help='单据创建日期')
  361. amount = fields.Float(string='单据金额',
  362. digits='Amount',
  363. help='待核销行对应金额')
  364. reconciled = fields.Float(string='已核销金额',
  365. digits='Amount',
  366. help='待核销行已核销掉的金额')
  367. to_reconcile = fields.Float(string='未核销金额',
  368. digits='Amount',
  369. help='待核销行未核销掉的金额')
  370. this_reconcile = fields.Float(string='本次核销金额',
  371. digits='Amount',
  372. help='本次要核销掉的金额')
  373. invoice_date = fields.Date(string='开票日期',
  374. help='待核销行开票日期',
  375. related='name.invoice_date')
  376. date_due = fields.Date(string='到期日',
  377. help='待核销行的到期日')
  378. company_id = fields.Many2one(
  379. 'res.company',
  380. string='公司',
  381. change_default=True,
  382. default=lambda self: self.env.company)
  383. class AdvancePayment(models.Model):
  384. _name = 'advance.payment'
  385. _description = '核销单预收付款行'
  386. pay_reconcile_id = fields.Many2one('reconcile.order',
  387. string='核销单', ondelete='cascade',
  388. help='核销单预收付款行对应的核销单')
  389. name = fields.Many2one('money.order', string='预收/付款单',
  390. copy=False, required=True, ondelete='cascade',
  391. help='核销单预收/付款行对应的预收/付款单')
  392. note = fields.Text('备注', related='name.note')
  393. date = fields.Date(string='单据日期',
  394. help='单据创建日期')
  395. amount = fields.Float(string='单据金额',
  396. digits='Amount',
  397. help='预收/付款单的预收/付金额')
  398. reconciled = fields.Float(string='已核销金额',
  399. digits='Amount',
  400. help='已核销的预收/预付款金额')
  401. to_reconcile = fields.Float(string='未核销金额',
  402. digits='Amount',
  403. help='未核销的预收/预付款金额')
  404. this_reconcile = fields.Float(string='本次核销金额',
  405. digits='Amount',
  406. help='本次核销的预收/预付款金额')
  407. company_id = fields.Many2one(
  408. 'res.company',
  409. string='公司',
  410. change_default=True,
  411. default=lambda self: self.env.company)
上海开阖软件有限公司 沪ICP备12045867号-1