Sylvain LE GAL
5 years ago
17 changed files with 235 additions and 231 deletions
-
15pos_invoicing/__manifest__.py
-
13pos_invoicing/demo/res_groups.xml
-
38pos_invoicing/i18n/fr.po
-
56pos_invoicing/i18n/pos_invoicing.pot
-
3pos_invoicing/models/__init__.py
-
14pos_invoicing/models/account_invoice.py
-
22pos_invoicing/models/account_payment.py
-
28pos_invoicing/models/account_voucher.py
-
13pos_invoicing/models/pos_order.py
-
59pos_invoicing/models/pos_session.py
-
43pos_invoicing/readme/DESCRIPTION.rst
-
4pos_invoicing/readme/ROADMAP.rst
-
BINpos_invoicing/static/description/account_invoice_form.png
-
BINpos_invoicing/static/description/icon.png
-
1pos_invoicing/tests/__init__.py
-
148pos_invoicing/tests/test_module.py
-
9pos_invoicing/views/view_account_invoice.xml
@ -1,13 +0,0 @@ |
|||||
<?xml version="1.0" encoding="UTF-8"?> |
|
||||
<!-- |
|
||||
Copyright (C) 2015 - Today: GRAP (http://www.grap.coop) |
|
||||
@author: Sylvain LE GAL (https://twitter.com/legalsylvain) |
|
||||
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|
||||
--> |
|
||||
<openerp><data> |
|
||||
|
|
||||
<record id="account.group_account_manager" model="res.groups"> |
|
||||
<field name="users" eval="[(4, ref('base.user_root'))]"/> |
|
||||
</record> |
|
||||
|
|
||||
</data></openerp> |
|
@ -0,0 +1,56 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * pos_invoicing |
||||
|
# |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 12.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2019-07-11 16:17+0000\n" |
||||
|
"PO-Revision-Date: 2019-07-11 16:17+0000\n" |
||||
|
"Last-Translator: <>\n" |
||||
|
"Language-Team: \n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: \n" |
||||
|
|
||||
|
#. module: pos_invoicing |
||||
|
#: model:ir.model.fields,help:pos_invoicing.field_account_invoice__pos_pending_payment |
||||
|
msgid "Indicates an invoice for which there are pending payments in the Point of Sale. \n" |
||||
|
"The invoice will be marked as paid when the session will be closed." |
||||
|
msgstr "" |
||||
|
|
||||
|
#. module: pos_invoicing |
||||
|
#: model:ir.model,name:pos_invoicing.model_account_invoice |
||||
|
msgid "Invoice" |
||||
|
msgstr "" |
||||
|
|
||||
|
#. module: pos_invoicing |
||||
|
#: model:ir.model,name:pos_invoicing.model_account_payment |
||||
|
msgid "Payments" |
||||
|
msgstr "" |
||||
|
|
||||
|
#. module: pos_invoicing |
||||
|
#: model:ir.model.fields,field_description:pos_invoicing.field_account_invoice__pos_pending_payment |
||||
|
msgid "PoS - Pending Payment" |
||||
|
msgstr "" |
||||
|
|
||||
|
#. module: pos_invoicing |
||||
|
#: model:ir.model,name:pos_invoicing.model_pos_session |
||||
|
msgid "Point of Sale Session" |
||||
|
msgstr "" |
||||
|
|
||||
|
#. module: pos_invoicing |
||||
|
#: code:addons/pos_invoicing/models/account_invoice.py:36 |
||||
|
#: code:addons/pos_invoicing/models/account_invoice.py:38 |
||||
|
#, python-format |
||||
|
msgid "You can not realize this action on the invoice(s) %s because there are pending payments in the Point of Sale." |
||||
|
msgstr "" |
||||
|
|
||||
|
#. module: pos_invoicing |
||||
|
#: code:addons/pos_invoicing/models/account_payment.py:17 |
||||
|
#, python-format |
||||
|
msgid "You can not realize this action on the payments(s) %s because there are pending payments in the Point of Sale." |
||||
|
msgstr "" |
||||
|
|
@ -1,5 +1,4 @@ |
|||||
# coding: utf-8 |
|
||||
from . import pos_order |
from . import pos_order |
||||
from . import pos_session |
from . import pos_session |
||||
from . import account_invoice |
from . import account_invoice |
||||
from . import account_voucher |
|
||||
|
from . import account_payment |
@ -0,0 +1,22 @@ |
|||||
|
# Copyright (C) 2019 - Today: GRAP (http://www.grap.coop) |
||||
|
# @author: Sylvain LE GAL (https://twitter.com/legalsylvain) |
||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
from odoo import _, api, models |
||||
|
from odoo.exceptions import Warning as UserError |
||||
|
|
||||
|
|
||||
|
class AccountPayment(models.Model): |
||||
|
_inherit = 'account.payment' |
||||
|
|
||||
|
@api.multi |
||||
|
def post(self): |
||||
|
payments = self.filtered( |
||||
|
lambda x: any(x.mapped('invoice_ids.pos_pending_payment'))) |
||||
|
if payments: |
||||
|
raise UserError(_( |
||||
|
"You can not realize this action on the payments(s) %s because" |
||||
|
" there are pending payments in the Point of Sale.") % ( |
||||
|
', '.join( |
||||
|
[x for x in payments.mapped('communication') if x]))) |
||||
|
return super().post() |
@ -1,28 +0,0 @@ |
|||||
# coding: utf-8 |
|
||||
# Copyright (C) 2013 - Today: GRAP (http://www.grap.coop) |
|
||||
# @author: Julien WESTE |
|
||||
# @author: Sylvain LE GAL (https://twitter.com/legalsylvain) |
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|
||||
|
|
||||
from openerp import api, models |
|
||||
|
|
||||
|
|
||||
class AccountVoucher(models.Model): |
|
||||
_inherit = 'account.voucher' |
|
||||
|
|
||||
# Override section |
|
||||
@api.multi |
|
||||
def recompute_voucher_lines( |
|
||||
self, partner_id, journal_id, price, currency_id, ttype, date): |
|
||||
|
|
||||
move_line_obj = self.env['account.move.line'] |
|
||||
|
|
||||
res = super(AccountVoucher, self).recompute_voucher_lines( |
|
||||
partner_id, journal_id, price, currency_id, ttype, date) |
|
||||
|
|
||||
for voucher_type in ['line_dr_ids', 'line_cr_ids']: |
|
||||
for voucher_line in res['value'][voucher_type]: |
|
||||
move_line = move_line_obj.browse(voucher_line['move_line_id']) |
|
||||
if move_line.invoice.pos_pending_payment: |
|
||||
res['value'][voucher_type].remove(voucher_line) |
|
||||
return res |
|
@ -1,18 +1,17 @@ |
|||||
# coding: utf-8 |
|
||||
# Copyright (C) 2013 - Today: GRAP (http://www.grap.coop) |
# Copyright (C) 2013 - Today: GRAP (http://www.grap.coop) |
||||
# @author: Julien WESTE |
# @author: Julien WESTE |
||||
# @author: Sylvain LE GAL (https://twitter.com/legalsylvain) |
# @author: Sylvain LE GAL (https://twitter.com/legalsylvain) |
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
from openerp import api, models |
|
||||
|
from odoo import models |
||||
|
|
||||
|
|
||||
class PosOrder(models.Model): |
class PosOrder(models.Model): |
||||
_inherit = 'pos.order' |
_inherit = 'pos.order' |
||||
|
|
||||
@api.multi |
|
||||
def action_invoice(self): |
|
||||
res = super(PosOrder, self).action_invoice() |
|
||||
self.mapped('invoice_id').write({'pos_pending_payment': True}) |
|
||||
self.mapped('invoice_id').signal_workflow('invoice_open') |
|
||||
|
def _prepare_invoice(self): |
||||
|
res = super()._prepare_invoice() |
||||
|
res.update({ |
||||
|
'pos_pending_payment': True, |
||||
|
}) |
||||
return res |
return res |
@ -1,25 +1,38 @@ |
|||||
When you pay a pos_order, and then create an invoice : |
|
||||
|
This module extend the Point of Sale Odoo module, regarding invoicing. |
||||
|
|
||||
* you mustn't register a payment against the invoice as the payment |
|
||||
already exists in POS |
|
||||
* The POS payment will be reconciled with the invoice when the session |
|
||||
is closed |
|
||||
* You mustn't modify the invoice because the amount could become |
|
||||
different from the one registered in POS. Thus we have to |
|
||||
automatically validate the created invoice |
|
||||
|
This module prevent to make some mistakes in Odoo Point of Sale |
||||
|
regarding invoices generated via Point of Sale. |
||||
|
|
||||
Functionality |
|
||||
------------- |
|
||||
About the invoices created from POS after payment: |
|
||||
|
Without this module |
||||
|
~~~~~~~~~~~~~~~~~~~ |
||||
|
|
||||
* automatically validate them and don't allow modifications |
|
||||
* Disable the Pay Button |
|
||||
* Don't display them in the Customer Payment tool |
|
||||
|
When an invoice generated from Point of Sale is confirmed |
||||
|
it is in a 'open' state, until the session is closed, and the entries are |
||||
|
generated. At this step, invoice will be marked as 'paid' and the related |
||||
|
accounting moves will be reconcilied. |
||||
|
So, as long as the session is not closed, any user can: |
||||
|
|
||||
|
* cancel the invoice; |
||||
|
* register a payment; |
||||
|
* reconcile the invoice with an existing payment; |
||||
|
|
||||
|
All that action should be prohibited. |
||||
|
|
||||
|
With that module |
||||
|
~~~~~~~~~~~~~~~~ |
||||
|
|
||||
|
All those actions will not be possible anymore. |
||||
|
|
||||
|
|
||||
|
Note that the changes only impact the opened invoice coming from point of sale, |
||||
|
before the session is closed. |
||||
|
|
||||
Technically |
Technically |
||||
----------- |
----------- |
||||
|
|
||||
add a ``pos_pending_payment`` field on the ``account.invoice`` to mark the |
|
||||
|
* add a ``pos_pending_payment`` field on the ``account.invoice`` to mark the |
||||
items that shouldn't be paid. |
items that shouldn't be paid. |
||||
|
This field is checked when the invoice is created from point of sale, |
||||
|
and is unchecked, when the session is closed. |
||||
|
|
||||
.. figure:: ../static/description/account_invoice_form.png |
.. figure:: ../static/description/account_invoice_form.png |
@ -1,4 +0,0 @@ |
|||||
* This module reconcile invoiced orders only if a customer has one invoice per |
|
||||
session. |
|
||||
|
|
||||
* It should be great to use the OCA module ``pos_autoreconcile``. |
|
Before Width: 852 | Height: 430 | Size: 32 KiB After Width: 1196 | Height: 385 | Size: 38 KiB |
Before Width: 64 | Height: 64 | Size: 4.0 KiB |
@ -1,2 +1 @@ |
|||||
# coding: utf-8 |
|
||||
from . import test_module |
from . import test_module |
@ -1,92 +1,102 @@ |
|||||
# coding: utf-8 |
|
||||
# Copyright (C) 2013 - Today: GRAP (http://www.grap.coop) |
# Copyright (C) 2013 - Today: GRAP (http://www.grap.coop) |
||||
# @author: Julien WESTE |
# @author: Julien WESTE |
||||
# @author: Sylvain LE GAL (https://twitter.com/legalsylvain) |
# @author: Sylvain LE GAL (https://twitter.com/legalsylvain) |
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
import time |
|
||||
from openerp.tests.common import TransactionCase |
|
||||
|
from odoo import fields |
||||
|
from odoo.tests.common import TransactionCase |
||||
|
from odoo.exceptions import Warning as UserError |
||||
|
|
||||
|
|
||||
class TestPosInvoicing(TransactionCase): |
|
||||
"""Tests for POS Invoicing Module""" |
|
||||
|
class TestModule(TransactionCase): |
||||
|
|
||||
def setUp(self): |
def setUp(self): |
||||
super(TestPosInvoicing, self).setUp() |
|
||||
|
super(TestModule, self).setUp() |
||||
|
|
||||
# Get Registry |
# Get Registry |
||||
self.session_obj = self.env['pos.session'] |
|
||||
self.order_obj = self.env['pos.order'] |
|
||||
|
self.PosOrder = self.env['pos.order'] |
||||
|
self.AccountPayment = self.env['account.payment'] |
||||
|
|
||||
# Get Object |
# Get Object |
||||
self.config = self.env.ref('point_of_sale.pos_config_main') |
|
||||
self.partner_A = self.env.ref('base.res_partner_2') |
|
||||
self.partner_B = self.env.ref('base.res_partner_12') |
|
||||
self.product = self.env.ref('product.product_product_48') |
|
||||
self.payment_journal = self.env.ref('account.cash_journal') |
|
||||
|
self.pos_product = self.env.ref('point_of_sale.whiteboard_pen') |
||||
|
self.pricelist = self.env.ref('product.list0') |
||||
|
self.partner = self.env.ref('base.res_partner_12') |
||||
|
|
||||
# Test Section |
|
||||
def test_01_invoice_with_payment(self): |
|
||||
"""Test the workflow: Draft Order -> Payment -> Invoice""" |
|
||||
# Opening Session |
|
||||
session = self.session_obj.create({'config_id': self.config.id}) |
|
||||
|
|
||||
# TODO FIXME, for the time being, reconciliation is not |
|
||||
# set if a customer make many invoices in the same pos session. |
|
||||
# self._create_order(session, self.partner_A, 100, True) |
|
||||
# self._create_order(session, self.partner_A, 200, True) |
|
||||
self._create_order(session, self.partner_B, 400, True) |
|
||||
|
# Create a new pos config and open it |
||||
|
self.pos_config = self.env.ref('point_of_sale.pos_config_main').copy() |
||||
|
self.pos_config.open_session_cb() |
||||
|
|
||||
# The Invoice must be unpayable but in 'open' state |
|
||||
# Invoice created by order should be in open state |
|
||||
invoiced_orders = session.mapped('order_ids').filtered( |
|
||||
lambda x: x.state == 'invoiced') |
|
||||
invoices = invoiced_orders.mapped('invoice_id') |
|
||||
|
|
||||
self.assertEquals( |
|
||||
[x for x in invoices.mapped('state') if x != 'open'], [], |
|
||||
"All invoices generated from PoS should be in the 'open' state" |
|
||||
" when session is opened") |
|
||||
|
# Test Section |
||||
|
def test_order_invoice(self): |
||||
|
order = self._create_order() |
||||
|
|
||||
self.assertEquals( |
|
||||
[x for x in invoices.mapped('pos_pending_payment') if not x], |
|
||||
[], |
|
||||
"All invoices generated from PoS should be marked as PoS Pending" |
|
||||
" Payment when session is opened") |
|
||||
|
# Check if invoice is correctly set |
||||
|
self.assertEquals(order.invoice_id.pos_pending_payment, True) |
||||
|
|
||||
# Close Session |
|
||||
session.signal_workflow('close') |
|
||||
|
# Try to register payment should fail on this invoice should fail |
||||
|
with self.assertRaises(UserError): |
||||
|
payment = self.register_payment(order.invoice_id) |
||||
|
payment.post() |
||||
|
|
||||
self.assertEquals( |
|
||||
[x for x in invoices.mapped('state') if x != 'paid'], [], |
|
||||
"All invoices generated from PoS should be in the 'paid' state" |
|
||||
" when session is closed") |
|
||||
|
# Try to register a payment not linked to this invoice should be ok |
||||
|
payment = self.register_payment() |
||||
|
payment.post() |
||||
|
|
||||
self.assertEquals( |
|
||||
[x for x in invoices.mapped('pos_pending_payment') if x], [], |
|
||||
"Invoices generated from PoS should not be marked as PoS Pending" |
|
||||
" Payment when session is closed") |
|
||||
|
# Once closed check if the invoice is correctly set |
||||
|
self.pos_config.current_session_id.action_pos_session_closing_control() |
||||
|
self.assertEquals(order.invoice_id.pos_pending_payment, False) |
||||
|
|
||||
# Private Section |
# Private Section |
||||
def _create_order(self, session, partner, amount, with_invoice): |
|
||||
# create Pos Order |
|
||||
order = self.order_obj.create({ |
|
||||
'session_id': session.id, |
|
||||
'partner_id': partner.id, |
|
||||
'lines': [[0, False, { |
|
||||
'product_id': self.product.id, |
|
||||
'qty': 1, |
|
||||
'price_unit': amount, |
|
||||
}]], |
|
||||
|
def register_payment(self, invoice_id=False): |
||||
|
journal = self.pos_config.journal_ids[0] |
||||
|
return self.AccountPayment.create({ |
||||
|
'invoice_ids': invoice_id and [(4, invoice_id.id, None)] or False, |
||||
|
'payment_type': 'inbound', |
||||
|
'partner_type': 'customer', |
||||
|
'payment_date': fields.Datetime.now(), |
||||
|
'partner_id': self.partner.id, |
||||
|
'amount': 0.9, |
||||
|
'journal_id': journal.id, |
||||
|
'payment_method_id': journal.inbound_payment_method_ids[0].id, |
||||
}) |
}) |
||||
# Finish Payment |
|
||||
self.order_obj.add_payment(order.id, { |
|
||||
'journal': self.payment_journal.id, |
|
||||
'payment_date': time.strftime('%Y-%m-%d'), |
|
||||
'amount': amount, |
|
||||
}) |
|
||||
# Mark as Paid |
|
||||
order.signal_workflow('paid') |
|
||||
if with_invoice: |
|
||||
order.action_invoice() |
|
||||
|
|
||||
|
def _create_order(self): |
||||
|
# Create order |
||||
|
order_data = { |
||||
|
'id': u'0006-001-0010', |
||||
|
'to_invoice': True, |
||||
|
'data': { |
||||
|
'pricelist_id': self.pricelist.id, |
||||
|
'user_id': 1, |
||||
|
'name': 'Order 0006-001-0010', |
||||
|
'partner_id': self.partner.id, |
||||
|
'amount_paid': 0.9, |
||||
|
'pos_session_id': self.pos_config.current_session_id.id, |
||||
|
'lines': [[0, 0, { |
||||
|
'product_id': self.pos_product.id, |
||||
|
'price_unit': 0.9, |
||||
|
'qty': 1, |
||||
|
'price_subtotal': 0.9, |
||||
|
'price_subtotal_incl': 0.9, |
||||
|
}]], |
||||
|
'statement_ids': [[0, 0, { |
||||
|
'journal_id': self.pos_config.journal_ids[0].id, |
||||
|
'amount': 0.9, |
||||
|
'name': fields.Datetime.now(), |
||||
|
'account_id': |
||||
|
self.env.user.partner_id.property_account_receivable_id.id, |
||||
|
'statement_id': |
||||
|
self.pos_config.current_session_id.statement_ids[0].id, |
||||
|
}]], |
||||
|
'creation_date': u'2018-09-27 15:51:03', |
||||
|
'amount_tax': 0, |
||||
|
'fiscal_position_id': False, |
||||
|
'uid': u'00001-001-0001', |
||||
|
'amount_return': 0, |
||||
|
'sequence_number': 1, |
||||
|
'amount_total': 0.9, |
||||
|
}} |
||||
|
|
||||
|
result = self.PosOrder.create_from_ui([order_data]) |
||||
|
order = self.PosOrder.browse(result[0]) |
||||
return order |
return order |
Write
Preview
Loading…
Cancel
Save
Reference in new issue