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.
 
 
 
 
 

130 lines
6.3 KiB

# -*- coding: utf-8 -*-
#############################################################################
#
# Cybrosys Technologies Pvt. Ltd.
#
# Copyright (C) 2019-TODAY Cybrosys Technologies(<https://www.cybrosys.com>)
# Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>)
#
# You can modify it under the terms of the GNU LESSER
# GENERAL PUBLIC LICENSE (LGPL v3), Version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details.
#
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE
# (LGPL v3) along with this program.
# If not, see <http://www.gnu.org/licenses/>.
#
#############################################################################
from odoo import models, fields, _
from odoo.exceptions import UserError, ValidationError
class AccountPayment(models.Model):
_inherit = "account.payment"
def _check_is_approver(self):
approval = self.env['ir.config_parameter'].sudo().get_param(
'account_payment_approval.payment_approval')
approver_id = int(self.env['ir.config_parameter'].sudo().get_param(
'account_payment_approval.approval_user_id'))
self.is_approver = True if self.env.user.id == approver_id and approval else False
state = fields.Selection(selection_add=[('waiting_approval', 'Waiting For Approval'),
('approved', 'Approved'),
('rejected', 'Rejected')])
is_approver = fields.Boolean(compute=_check_is_approver, readonly=True)
def post(self):
"""Overwrites the post() to validate the payment in the 'approved' stage too.
Currently Odoo allows payment posting only in draft stage.
"""
AccountMove = self.env['account.move'].with_context(default_type='entry')
for rec in self:
validation = rec._check_payment_approval()
if validation:
if rec.state not in ('draft', 'approved'):
raise UserError(_("Only a draft or approved payment can be posted."))
if any(inv.state != 'posted' for inv in rec.invoice_ids):
raise ValidationError(_("The payment cannot be processed because the invoice is not open!"))
# keep the name in case of a payment reset to draft
if not rec.name:
# Use the right sequence to set the name
if rec.payment_type == 'transfer':
sequence_code = 'account.payment.transfer'
else:
if rec.partner_type == 'customer':
if rec.payment_type == 'inbound':
sequence_code = 'account.payment.customer.invoice'
if rec.payment_type == 'outbound':
sequence_code = 'account.payment.customer.refund'
if rec.partner_type == 'supplier':
if rec.payment_type == 'inbound':
sequence_code = 'account.payment.supplier.refund'
if rec.payment_type == 'outbound':
sequence_code = 'account.payment.supplier.invoice'
rec.name = self.env['ir.sequence'].next_by_code(sequence_code, sequence_date=rec.payment_date)
if not rec.name and rec.payment_type != 'transfer':
raise UserError(_("You have to define a sequence for %s in your company.") % (sequence_code,))
moves = AccountMove.create(rec._prepare_payment_moves())
moves.filtered(lambda move: move.journal_id.post_at != 'bank_rec').post()
# Update the state / move before performing any reconciliation.
move_name = self._get_move_name_transfer_separator().join(moves.mapped('name'))
rec.write({'state': 'posted', 'move_name': move_name})
if rec.payment_type in ('inbound', 'outbound'):
# ==== 'inbound' / 'outbound' ====
if rec.invoice_ids:
(moves[0] + rec.invoice_ids).line_ids \
.filtered(
lambda line: not line.reconciled and line.account_id == rec.destination_account_id) \
.reconcile()
elif rec.payment_type == 'transfer':
# ==== 'transfer' ====
moves.mapped('line_ids') \
.filtered(lambda line: line.account_id == rec.company_id.transfer_account_id) \
.reconcile()
return True
def _check_payment_approval(self):
if self.state == "draft":
first_approval = self.env['ir.config_parameter'].sudo().get_param(
'account_payment_approval.payment_approval')
if first_approval:
amount = float(self.env['ir.config_parameter'].sudo().get_param(
'account_payment_approval.approval_amount'))
payment_currency_id = int(self.env['ir.config_parameter'].sudo().get_param(
'account_payment_approval.approval_currency_id'))
payment_amount = self.amount
if payment_currency_id:
if self.currency_id and self.currency_id.id != payment_currency_id:
currency_id = self.env['res.currency'].browse(payment_currency_id)
payment_amount = self.currency_id._convert(
self.amount, currency_id, self.company_id,
self.payment_date or fields.Date.today(), round=True)
if payment_amount > amount:
self.write({
'state': 'waiting_approval'
})
return False
return True
def approve_transfer(self):
if self.is_approver:
self.write({
'state': 'approved'
})
def reject_transfer(self):
self.write({
'state': 'rejected'
})