mirror of
https://github.com/OCA/account-financial-tools.git
synced 2025-02-02 12:47:26 +02:00
97
account_clearance_plan/README.rst
Normal file
97
account_clearance_plan/README.rst
Normal file
@@ -0,0 +1,97 @@
|
||||
======================
|
||||
Account Clearance Plan
|
||||
======================
|
||||
|
||||
.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
!! This file is generated by oca-gen-addon-readme !!
|
||||
!! changes will be overwritten. !!
|
||||
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
|
||||
.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png
|
||||
:target: https://odoo-community.org/page/development-status
|
||||
:alt: Beta
|
||||
.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png
|
||||
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
|
||||
:alt: License: AGPL-3
|
||||
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Faccount--financial--tools-lightgray.png?logo=github
|
||||
:target: https://github.com/OCA/account-financial-tools/tree/12.0/account_clearance_plan
|
||||
:alt: OCA/account-financial-tools
|
||||
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
|
||||
:target: https://translation.odoo-community.org/projects/account-financial-tools-12-0/account-financial-tools-12-0-account_clearance_plan
|
||||
:alt: Translate me on Weblate
|
||||
.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png
|
||||
:target: https://runbot.odoo-community.org/runbot/92/12.0
|
||||
:alt: Try me on Runbot
|
||||
|
||||
|badge1| |badge2| |badge3| |badge4| |badge5|
|
||||
|
||||
This module allows to reorganize customers debts in case of difficult financial conditions in order to spread the debts over a longer period of time.
|
||||
The module can also be used in a similar way for the user's company in case their suppliers offer them the opportunity to discuss a clearance plan.
|
||||
|
||||
One can decide the debts to be part of the clearance plan (all partner's debts or some of them only) and split this amount into new financial commitments at new dates.
|
||||
Please notice that one clearance plan = one payable/receivable account.
|
||||
|
||||
**Table of contents**
|
||||
|
||||
.. contents::
|
||||
:local:
|
||||
|
||||
Usage
|
||||
=====
|
||||
|
||||
To use this module, you need to:
|
||||
|
||||
#. Go to *Invoicing > Accounting > Journal Items*, or *Invoicing > Vendor Bills*, or *Invoicing > Customer Invoices*
|
||||
|
||||
#. Then select the journal items or invoices and go to *Action > Clearance Plan*
|
||||
|
||||
#. Choose a journal for the new entry that will be created. Default journal can be configured in *Invoicing > Configuration > Settings > Clearance Plan*
|
||||
|
||||
#. Add one line for each new payment amounts and dates you would like to set.
|
||||
|
||||
#. Click *Confirm*, the journal items or invoices you selected are now reconciled and new journal items are open according to your choices.
|
||||
|
||||
Known issues / Roadmap
|
||||
======================
|
||||
|
||||
* Support VAT on collection (Taxes due upon payment, e.g. in France)
|
||||
|
||||
Bug Tracker
|
||||
===========
|
||||
|
||||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/account-financial-tools/issues>`_.
|
||||
In case of trouble, please check there if your issue has already been reported.
|
||||
If you spotted it first, help us smashing it by providing a detailed and welcomed
|
||||
`feedback <https://github.com/OCA/account-financial-tools/issues/new?body=module:%20account_clearance_plan%0Aversion:%2012.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
|
||||
|
||||
Do not contact contributors directly about support or help with technical issues.
|
||||
|
||||
Credits
|
||||
=======
|
||||
|
||||
Authors
|
||||
~~~~~~~
|
||||
|
||||
* ACSONE SA/NV
|
||||
|
||||
Contributors
|
||||
~~~~~~~~~~~~
|
||||
|
||||
* Quentin Groulard <quentin.groulard@acsone.eu>
|
||||
|
||||
Maintainers
|
||||
~~~~~~~~~~~
|
||||
|
||||
This module is maintained by the OCA.
|
||||
|
||||
.. image:: https://odoo-community.org/logo.png
|
||||
:alt: Odoo Community Association
|
||||
:target: https://odoo-community.org
|
||||
|
||||
OCA, or the Odoo Community Association, is a nonprofit organization whose
|
||||
mission is to support the collaborative development of Odoo features and
|
||||
promote its widespread use.
|
||||
|
||||
This module is part of the `OCA/account-financial-tools <https://github.com/OCA/account-financial-tools/tree/12.0/account_clearance_plan>`_ project on GitHub.
|
||||
|
||||
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.
|
||||
2
account_clearance_plan/__init__.py
Normal file
2
account_clearance_plan/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
||||
from . import wizard
|
||||
from . import models
|
||||
19
account_clearance_plan/__manifest__.py
Normal file
19
account_clearance_plan/__manifest__.py
Normal file
@@ -0,0 +1,19 @@
|
||||
# Copyright 2020 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
{
|
||||
"name": "Account Clearance Plan",
|
||||
"summary": """
|
||||
This addon allows to define clearance plans
|
||||
in order to reorganize debts (own and customers' ones).""",
|
||||
"version": "12.0.1.0.0",
|
||||
"license": "AGPL-3",
|
||||
"author": "ACSONE SA/NV, Odoo Community Association (OCA)",
|
||||
"website": "https://github.com/OCA/account-financial-tools",
|
||||
"depends": ["account"],
|
||||
"data": [
|
||||
"views/res_config_settings.xml",
|
||||
"wizard/account_clearance_plan_wizard.xml",
|
||||
],
|
||||
"demo": [],
|
||||
}
|
||||
3
account_clearance_plan/models/__init__.py
Normal file
3
account_clearance_plan/models/__init__.py
Normal file
@@ -0,0 +1,3 @@
|
||||
from . import res_company
|
||||
from . import res_config_settings
|
||||
from . import account_invoice
|
||||
15
account_clearance_plan/models/account_invoice.py
Normal file
15
account_clearance_plan/models/account_invoice.py
Normal file
@@ -0,0 +1,15 @@
|
||||
# Copyright 2020 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import models
|
||||
|
||||
|
||||
class AccountInvoice(models.Model):
|
||||
|
||||
_inherit = "account.invoice"
|
||||
|
||||
def _get_open_move_lines_ids(self):
|
||||
self.ensure_one()
|
||||
return self.mapped("move_id.line_ids").filtered(
|
||||
lambda l: l.account_id == self.account_id and not l.reconciled
|
||||
).ids
|
||||
20
account_clearance_plan/models/res_company.py
Normal file
20
account_clearance_plan/models/res_company.py
Normal file
@@ -0,0 +1,20 @@
|
||||
# Copyright 2020 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class ResCompany(models.Model):
|
||||
_inherit = "res.company"
|
||||
|
||||
clearance_plan_journal_id = fields.Many2one(
|
||||
comodel_name="account.journal",
|
||||
string="Default Clearance Plan Journal",
|
||||
help="The journal used by default on clearance plans.",
|
||||
)
|
||||
clearance_plan_move_line_name = fields.Char(
|
||||
string="Default Clearance Plan Move Line Name",
|
||||
help="Default name that will be given to new open "
|
||||
"move lines created by clearance plans",
|
||||
default="Clearance Plan",
|
||||
)
|
||||
23
account_clearance_plan/models/res_config_settings.py
Normal file
23
account_clearance_plan/models/res_config_settings.py
Normal file
@@ -0,0 +1,23 @@
|
||||
# Copyright 2020 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class ResConfigSettings(models.TransientModel):
|
||||
_inherit = "res.config.settings"
|
||||
|
||||
clearance_plan_journal_id = fields.Many2one(
|
||||
comodel_name="account.journal",
|
||||
related="company_id.clearance_plan_journal_id",
|
||||
readonly=False,
|
||||
string="Default Clearance Plan Journal",
|
||||
help="The journal used by default on clearance plans.",
|
||||
)
|
||||
clearance_plan_move_line_name = fields.Char(
|
||||
string="Default Clearance Plan Move Line Name",
|
||||
help="Default name that will be given to new open "
|
||||
"move lines created by clearance plans",
|
||||
related="company_id.clearance_plan_move_line_name",
|
||||
readonly=False,
|
||||
)
|
||||
1
account_clearance_plan/readme/CONTRIBUTORS.rst
Normal file
1
account_clearance_plan/readme/CONTRIBUTORS.rst
Normal file
@@ -0,0 +1 @@
|
||||
* Quentin Groulard <quentin.groulard@acsone.eu>
|
||||
5
account_clearance_plan/readme/DESCRIPTION.rst
Normal file
5
account_clearance_plan/readme/DESCRIPTION.rst
Normal file
@@ -0,0 +1,5 @@
|
||||
This module allows to reorganize customers debts in case of difficult financial conditions in order to spread the debts over a longer period of time.
|
||||
The module can also be used in a similar way for the user's company in case their suppliers offer them the opportunity to discuss a clearance plan.
|
||||
|
||||
One can decide the debts to be part of the clearance plan (all partner's debts or some of them only) and split this amount into new financial commitments at new dates.
|
||||
Please notice that one clearance plan = one payable/receivable account.
|
||||
1
account_clearance_plan/readme/ROADMAP.rst
Normal file
1
account_clearance_plan/readme/ROADMAP.rst
Normal file
@@ -0,0 +1 @@
|
||||
* Support VAT on collection (Taxes due upon payment, e.g. in France)
|
||||
11
account_clearance_plan/readme/USAGE.rst
Normal file
11
account_clearance_plan/readme/USAGE.rst
Normal file
@@ -0,0 +1,11 @@
|
||||
To use this module, you need to:
|
||||
|
||||
#. Go to *Invoicing > Accounting > Journal Items*, or *Invoicing > Vendor Bills*, or *Invoicing > Customer Invoices*
|
||||
|
||||
#. Then select the journal items or invoices and go to *Action > Clearance Plan*
|
||||
|
||||
#. Choose a journal for the new entry that will be created. Default journal can be configured in *Invoicing > Configuration > Settings > Clearance Plan*
|
||||
|
||||
#. Add one line for each new payment amounts and dates you would like to set.
|
||||
|
||||
#. Click *Confirm*, the journal items or invoices you selected are now reconciled and new journal items are open according to your choices.
|
||||
BIN
account_clearance_plan/static/description/icon.png
Normal file
BIN
account_clearance_plan/static/description/icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 9.2 KiB |
441
account_clearance_plan/static/description/index.html
Normal file
441
account_clearance_plan/static/description/index.html
Normal file
@@ -0,0 +1,441 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
|
||||
<head>
|
||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||
<meta name="generator" content="Docutils 0.16: http://docutils.sourceforge.net/" />
|
||||
<title>Account Clearance Plan</title>
|
||||
<style type="text/css">
|
||||
|
||||
/*
|
||||
:Author: David Goodger (goodger@python.org)
|
||||
:Id: $Id: html4css1.css 7952 2016-07-26 18:15:59Z milde $
|
||||
:Copyright: This stylesheet has been placed in the public domain.
|
||||
|
||||
Default cascading style sheet for the HTML output of Docutils.
|
||||
|
||||
See http://docutils.sf.net/docs/howto/html-stylesheets.html for how to
|
||||
customize this style sheet.
|
||||
*/
|
||||
|
||||
/* used to remove borders from tables and images */
|
||||
.borderless, table.borderless td, table.borderless th {
|
||||
border: 0 }
|
||||
|
||||
table.borderless td, table.borderless th {
|
||||
/* Override padding for "table.docutils td" with "! important".
|
||||
The right padding separates the table cells. */
|
||||
padding: 0 0.5em 0 0 ! important }
|
||||
|
||||
.first {
|
||||
/* Override more specific margin styles with "! important". */
|
||||
margin-top: 0 ! important }
|
||||
|
||||
.last, .with-subtitle {
|
||||
margin-bottom: 0 ! important }
|
||||
|
||||
.hidden {
|
||||
display: none }
|
||||
|
||||
.subscript {
|
||||
vertical-align: sub;
|
||||
font-size: smaller }
|
||||
|
||||
.superscript {
|
||||
vertical-align: super;
|
||||
font-size: smaller }
|
||||
|
||||
a.toc-backref {
|
||||
text-decoration: none ;
|
||||
color: black }
|
||||
|
||||
blockquote.epigraph {
|
||||
margin: 2em 5em ; }
|
||||
|
||||
dl.docutils dd {
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
/* Uncomment (and remove this text!) to get bold-faced definition list terms
|
||||
dl.docutils dt {
|
||||
font-weight: bold }
|
||||
*/
|
||||
|
||||
div.abstract {
|
||||
margin: 2em 5em }
|
||||
|
||||
div.abstract p.topic-title {
|
||||
font-weight: bold ;
|
||||
text-align: center }
|
||||
|
||||
div.admonition, div.attention, div.caution, div.danger, div.error,
|
||||
div.hint, div.important, div.note, div.tip, div.warning {
|
||||
margin: 2em ;
|
||||
border: medium outset ;
|
||||
padding: 1em }
|
||||
|
||||
div.admonition p.admonition-title, div.hint p.admonition-title,
|
||||
div.important p.admonition-title, div.note p.admonition-title,
|
||||
div.tip p.admonition-title {
|
||||
font-weight: bold ;
|
||||
font-family: sans-serif }
|
||||
|
||||
div.attention p.admonition-title, div.caution p.admonition-title,
|
||||
div.danger p.admonition-title, div.error p.admonition-title,
|
||||
div.warning p.admonition-title, .code .error {
|
||||
color: red ;
|
||||
font-weight: bold ;
|
||||
font-family: sans-serif }
|
||||
|
||||
/* Uncomment (and remove this text!) to get reduced vertical space in
|
||||
compound paragraphs.
|
||||
div.compound .compound-first, div.compound .compound-middle {
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
div.compound .compound-last, div.compound .compound-middle {
|
||||
margin-top: 0.5em }
|
||||
*/
|
||||
|
||||
div.dedication {
|
||||
margin: 2em 5em ;
|
||||
text-align: center ;
|
||||
font-style: italic }
|
||||
|
||||
div.dedication p.topic-title {
|
||||
font-weight: bold ;
|
||||
font-style: normal }
|
||||
|
||||
div.figure {
|
||||
margin-left: 2em ;
|
||||
margin-right: 2em }
|
||||
|
||||
div.footer, div.header {
|
||||
clear: both;
|
||||
font-size: smaller }
|
||||
|
||||
div.line-block {
|
||||
display: block ;
|
||||
margin-top: 1em ;
|
||||
margin-bottom: 1em }
|
||||
|
||||
div.line-block div.line-block {
|
||||
margin-top: 0 ;
|
||||
margin-bottom: 0 ;
|
||||
margin-left: 1.5em }
|
||||
|
||||
div.sidebar {
|
||||
margin: 0 0 0.5em 1em ;
|
||||
border: medium outset ;
|
||||
padding: 1em ;
|
||||
background-color: #ffffee ;
|
||||
width: 40% ;
|
||||
float: right ;
|
||||
clear: right }
|
||||
|
||||
div.sidebar p.rubric {
|
||||
font-family: sans-serif ;
|
||||
font-size: medium }
|
||||
|
||||
div.system-messages {
|
||||
margin: 5em }
|
||||
|
||||
div.system-messages h1 {
|
||||
color: red }
|
||||
|
||||
div.system-message {
|
||||
border: medium outset ;
|
||||
padding: 1em }
|
||||
|
||||
div.system-message p.system-message-title {
|
||||
color: red ;
|
||||
font-weight: bold }
|
||||
|
||||
div.topic {
|
||||
margin: 2em }
|
||||
|
||||
h1.section-subtitle, h2.section-subtitle, h3.section-subtitle,
|
||||
h4.section-subtitle, h5.section-subtitle, h6.section-subtitle {
|
||||
margin-top: 0.4em }
|
||||
|
||||
h1.title {
|
||||
text-align: center }
|
||||
|
||||
h2.subtitle {
|
||||
text-align: center }
|
||||
|
||||
hr.docutils {
|
||||
width: 75% }
|
||||
|
||||
img.align-left, .figure.align-left, object.align-left, table.align-left {
|
||||
clear: left ;
|
||||
float: left ;
|
||||
margin-right: 1em }
|
||||
|
||||
img.align-right, .figure.align-right, object.align-right, table.align-right {
|
||||
clear: right ;
|
||||
float: right ;
|
||||
margin-left: 1em }
|
||||
|
||||
img.align-center, .figure.align-center, object.align-center {
|
||||
display: block;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
table.align-center {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
.align-left {
|
||||
text-align: left }
|
||||
|
||||
.align-center {
|
||||
clear: both ;
|
||||
text-align: center }
|
||||
|
||||
.align-right {
|
||||
text-align: right }
|
||||
|
||||
/* reset inner alignment in figures */
|
||||
div.align-right {
|
||||
text-align: inherit }
|
||||
|
||||
/* div.align-center * { */
|
||||
/* text-align: left } */
|
||||
|
||||
.align-top {
|
||||
vertical-align: top }
|
||||
|
||||
.align-middle {
|
||||
vertical-align: middle }
|
||||
|
||||
.align-bottom {
|
||||
vertical-align: bottom }
|
||||
|
||||
ol.simple, ul.simple {
|
||||
margin-bottom: 1em }
|
||||
|
||||
ol.arabic {
|
||||
list-style: decimal }
|
||||
|
||||
ol.loweralpha {
|
||||
list-style: lower-alpha }
|
||||
|
||||
ol.upperalpha {
|
||||
list-style: upper-alpha }
|
||||
|
||||
ol.lowerroman {
|
||||
list-style: lower-roman }
|
||||
|
||||
ol.upperroman {
|
||||
list-style: upper-roman }
|
||||
|
||||
p.attribution {
|
||||
text-align: right ;
|
||||
margin-left: 50% }
|
||||
|
||||
p.caption {
|
||||
font-style: italic }
|
||||
|
||||
p.credits {
|
||||
font-style: italic ;
|
||||
font-size: smaller }
|
||||
|
||||
p.label {
|
||||
white-space: nowrap }
|
||||
|
||||
p.rubric {
|
||||
font-weight: bold ;
|
||||
font-size: larger ;
|
||||
color: maroon ;
|
||||
text-align: center }
|
||||
|
||||
p.sidebar-title {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold ;
|
||||
font-size: larger }
|
||||
|
||||
p.sidebar-subtitle {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold }
|
||||
|
||||
p.topic-title {
|
||||
font-weight: bold }
|
||||
|
||||
pre.address {
|
||||
margin-bottom: 0 ;
|
||||
margin-top: 0 ;
|
||||
font: inherit }
|
||||
|
||||
pre.literal-block, pre.doctest-block, pre.math, pre.code {
|
||||
margin-left: 2em ;
|
||||
margin-right: 2em }
|
||||
|
||||
pre.code .ln { color: grey; } /* line numbers */
|
||||
pre.code, code { background-color: #eeeeee }
|
||||
pre.code .comment, code .comment { color: #5C6576 }
|
||||
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold }
|
||||
pre.code .literal.string, code .literal.string { color: #0C5404 }
|
||||
pre.code .name.builtin, code .name.builtin { color: #352B84 }
|
||||
pre.code .deleted, code .deleted { background-color: #DEB0A1}
|
||||
pre.code .inserted, code .inserted { background-color: #A3D289}
|
||||
|
||||
span.classifier {
|
||||
font-family: sans-serif ;
|
||||
font-style: oblique }
|
||||
|
||||
span.classifier-delimiter {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold }
|
||||
|
||||
span.interpreted {
|
||||
font-family: sans-serif }
|
||||
|
||||
span.option {
|
||||
white-space: nowrap }
|
||||
|
||||
span.pre {
|
||||
white-space: pre }
|
||||
|
||||
span.problematic {
|
||||
color: red }
|
||||
|
||||
span.section-subtitle {
|
||||
/* font-size relative to parent (h1..h6 element) */
|
||||
font-size: 80% }
|
||||
|
||||
table.citation {
|
||||
border-left: solid 1px gray;
|
||||
margin-left: 1px }
|
||||
|
||||
table.docinfo {
|
||||
margin: 2em 4em }
|
||||
|
||||
table.docutils {
|
||||
margin-top: 0.5em ;
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
table.footnote {
|
||||
border-left: solid 1px black;
|
||||
margin-left: 1px }
|
||||
|
||||
table.docutils td, table.docutils th,
|
||||
table.docinfo td, table.docinfo th {
|
||||
padding-left: 0.5em ;
|
||||
padding-right: 0.5em ;
|
||||
vertical-align: top }
|
||||
|
||||
table.docutils th.field-name, table.docinfo th.docinfo-name {
|
||||
font-weight: bold ;
|
||||
text-align: left ;
|
||||
white-space: nowrap ;
|
||||
padding-left: 0 }
|
||||
|
||||
/* "booktabs" style (no vertical lines) */
|
||||
table.docutils.booktabs {
|
||||
border: 0px;
|
||||
border-top: 2px solid;
|
||||
border-bottom: 2px solid;
|
||||
border-collapse: collapse;
|
||||
}
|
||||
table.docutils.booktabs * {
|
||||
border: 0px;
|
||||
}
|
||||
table.docutils.booktabs th {
|
||||
border-bottom: thin solid;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
h1 tt.docutils, h2 tt.docutils, h3 tt.docutils,
|
||||
h4 tt.docutils, h5 tt.docutils, h6 tt.docutils {
|
||||
font-size: 100% }
|
||||
|
||||
ul.auto-toc {
|
||||
list-style-type: none }
|
||||
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="document" id="account-clearance-plan">
|
||||
<h1 class="title">Account Clearance Plan</h1>
|
||||
|
||||
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
!! This file is generated by oca-gen-addon-readme !!
|
||||
!! changes will be overwritten. !!
|
||||
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
|
||||
<p><a class="reference external" href="https://odoo-community.org/page/development-status"><img alt="Beta" src="https://img.shields.io/badge/maturity-Beta-yellow.png" /></a> <a class="reference external" href="http://www.gnu.org/licenses/agpl-3.0-standalone.html"><img alt="License: AGPL-3" src="https://img.shields.io/badge/licence-AGPL--3-blue.png" /></a> <a class="reference external" href="https://github.com/OCA/account-financial-tools/tree/12.0/account_clearance_plan"><img alt="OCA/account-financial-tools" src="https://img.shields.io/badge/github-OCA%2Faccount--financial--tools-lightgray.png?logo=github" /></a> <a class="reference external" href="https://translation.odoo-community.org/projects/account-financial-tools-12-0/account-financial-tools-12-0-account_clearance_plan"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external" href="https://runbot.odoo-community.org/runbot/92/12.0"><img alt="Try me on Runbot" src="https://img.shields.io/badge/runbot-Try%20me-875A7B.png" /></a></p>
|
||||
<p>This module allows to reorganize customers debts in case of difficult financial conditions in order to spread the debts over a longer period of time.
|
||||
The module can also be used in a similar way for the user’s company in case their suppliers offer them the opportunity to discuss a clearance plan.</p>
|
||||
<p>One can decide the debts to be part of the clearance plan (all partner’s debts or some of them only) and split this amount into new financial commitments at new dates.
|
||||
Please notice that one clearance plan = one payable/receivable account.</p>
|
||||
<p><strong>Table of contents</strong></p>
|
||||
<div class="contents local topic" id="contents">
|
||||
<ul class="simple">
|
||||
<li><a class="reference internal" href="#usage" id="id1">Usage</a></li>
|
||||
<li><a class="reference internal" href="#known-issues-roadmap" id="id2">Known issues / Roadmap</a></li>
|
||||
<li><a class="reference internal" href="#bug-tracker" id="id3">Bug Tracker</a></li>
|
||||
<li><a class="reference internal" href="#credits" id="id4">Credits</a><ul>
|
||||
<li><a class="reference internal" href="#authors" id="id5">Authors</a></li>
|
||||
<li><a class="reference internal" href="#contributors" id="id6">Contributors</a></li>
|
||||
<li><a class="reference internal" href="#maintainers" id="id7">Maintainers</a></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="usage">
|
||||
<h1><a class="toc-backref" href="#id1">Usage</a></h1>
|
||||
<p>To use this module, you need to:</p>
|
||||
<ol class="arabic simple">
|
||||
<li>Go to <em>Invoicing > Accounting > Journal Items</em>, or <em>Invoicing > Vendor Bills</em>, or <em>Invoicing > Customer Invoices</em></li>
|
||||
<li>Then select the journal items or invoices and go to <em>Action > Clearance Plan</em></li>
|
||||
<li>Choose a journal for the new entry that will be created. Default journal can be configured in <em>Invoicing > Configuration > Settings > Clearance Plan</em></li>
|
||||
<li>Add one line for each new payment amounts and dates you would like to set.</li>
|
||||
<li>Click <em>Confirm</em>, the journal items or invoices you selected are now reconciled and new journal items are open according to your choices.</li>
|
||||
</ol>
|
||||
</div>
|
||||
<div class="section" id="known-issues-roadmap">
|
||||
<h1><a class="toc-backref" href="#id2">Known issues / Roadmap</a></h1>
|
||||
<ul class="simple">
|
||||
<li>Support VAT on collection (Taxes due upon payment, e.g. in France)</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="bug-tracker">
|
||||
<h1><a class="toc-backref" href="#id3">Bug Tracker</a></h1>
|
||||
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/account-financial-tools/issues">GitHub Issues</a>.
|
||||
In case of trouble, please check there if your issue has already been reported.
|
||||
If you spotted it first, help us smashing it by providing a detailed and welcomed
|
||||
<a class="reference external" href="https://github.com/OCA/account-financial-tools/issues/new?body=module:%20account_clearance_plan%0Aversion:%2012.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**">feedback</a>.</p>
|
||||
<p>Do not contact contributors directly about support or help with technical issues.</p>
|
||||
</div>
|
||||
<div class="section" id="credits">
|
||||
<h1><a class="toc-backref" href="#id4">Credits</a></h1>
|
||||
<div class="section" id="authors">
|
||||
<h2><a class="toc-backref" href="#id5">Authors</a></h2>
|
||||
<ul class="simple">
|
||||
<li>ACSONE SA/NV</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="contributors">
|
||||
<h2><a class="toc-backref" href="#id6">Contributors</a></h2>
|
||||
<ul class="simple">
|
||||
<li>Quentin Groulard <<a class="reference external" href="mailto:quentin.groulard@acsone.eu">quentin.groulard@acsone.eu</a>></li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="maintainers">
|
||||
<h2><a class="toc-backref" href="#id7">Maintainers</a></h2>
|
||||
<p>This module is maintained by the OCA.</p>
|
||||
<a class="reference external image-reference" href="https://odoo-community.org"><img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" /></a>
|
||||
<p>OCA, or the Odoo Community Association, is a nonprofit organization whose
|
||||
mission is to support the collaborative development of Odoo features and
|
||||
promote its widespread use.</p>
|
||||
<p>This module is part of the <a class="reference external" href="https://github.com/OCA/account-financial-tools/tree/12.0/account_clearance_plan">OCA/account-financial-tools</a> project on GitHub.</p>
|
||||
<p>You are welcome to contribute. To learn how please visit <a class="reference external" href="https://odoo-community.org/page/Contribute">https://odoo-community.org/page/Contribute</a>.</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
1
account_clearance_plan/tests/__init__.py
Normal file
1
account_clearance_plan/tests/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from . import test_account_clearance_plan
|
||||
133
account_clearance_plan/tests/test_account_clearance_plan.py
Normal file
133
account_clearance_plan/tests/test_account_clearance_plan.py
Normal file
@@ -0,0 +1,133 @@
|
||||
# Copyright 2020 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo.tests.common import TransactionCase, Form
|
||||
from odoo.exceptions import ValidationError
|
||||
|
||||
from datetime import datetime, timedelta
|
||||
|
||||
|
||||
class TestAccountClearancePlan(TransactionCase):
|
||||
def setUp(self):
|
||||
super(TestAccountClearancePlan, self).setUp()
|
||||
self.company = self.env.ref("base.main_company")
|
||||
self.partner = self.env["res.partner"].create({"name": "Test"})
|
||||
self.account_type_receivable = self.env["account.account.type"].create(
|
||||
{"name": "Test Receivable", "type": "receivable"}
|
||||
)
|
||||
self.account_type_regular = self.env["account.account.type"].create(
|
||||
{"name": "Test Regular", "type": "other"}
|
||||
)
|
||||
self.account_receivable = self.env["account.account"].create(
|
||||
{
|
||||
"name": "Test Receivable",
|
||||
"code": "TEST_AR",
|
||||
"user_type_id": self.account_type_receivable.id,
|
||||
"reconcile": True,
|
||||
}
|
||||
)
|
||||
self.account_income = self.env["account.account"].create(
|
||||
{
|
||||
"name": "Test Income",
|
||||
"code": "TEST_IN",
|
||||
"user_type_id": self.account_type_regular.id,
|
||||
"reconcile": False,
|
||||
}
|
||||
)
|
||||
self.sale_journal = self.env["account.journal"].search(
|
||||
[("type", "=", "sale"), ("company_id", "=", self.company.id)]
|
||||
)[0]
|
||||
self.cash_journal = self.env["account.journal"].search(
|
||||
[("type", "=", "cash"), ("company_id", "=", self.company.id)]
|
||||
)[0]
|
||||
self.general_journal = self.env["account.journal"].search(
|
||||
[("type", "=", "general"), ("company_id", "=", self.company.id)]
|
||||
)[0]
|
||||
self.company.clearance_plan_journal_id = self.general_journal
|
||||
self.payment_method_manual_in = self.env.ref(
|
||||
"account.account_payment_method_manual_in"
|
||||
)
|
||||
self.invoice_line = self.env["account.invoice.line"].create(
|
||||
{
|
||||
"name": "Line",
|
||||
"price_unit": 1000.0,
|
||||
"account_id": self.account_income.id,
|
||||
"quantity": 1,
|
||||
}
|
||||
)
|
||||
self.invoice = self.env["account.invoice"].create(
|
||||
{
|
||||
"name": "Test Customer Invoice",
|
||||
"journal_id": self.sale_journal.id,
|
||||
"partner_id": self.partner.id,
|
||||
"account_id": self.account_receivable.id,
|
||||
"invoice_line_ids": [(4, self.invoice_line.id)],
|
||||
}
|
||||
)
|
||||
self.invoice.action_invoice_open()
|
||||
self.invoice_ctx = {
|
||||
"active_model": "account.invoice",
|
||||
"active_ids": [self.invoice.id],
|
||||
}
|
||||
self.register_payments = (
|
||||
self.env["account.register.payments"]
|
||||
.with_context(self.invoice_ctx)
|
||||
.create(
|
||||
{
|
||||
"payment_date": datetime.now().strftime("%Y-%m-%d"),
|
||||
"payment_method_id": self.payment_method_manual_in.id,
|
||||
"journal_id": self.cash_journal.id,
|
||||
"amount": 200.0,
|
||||
}
|
||||
)
|
||||
)
|
||||
self.register_payments.create_payments()
|
||||
|
||||
def create_and_fill_wizard(self):
|
||||
clearance_plan_wizard = Form(
|
||||
self.env["account.clearance.plan"].with_context(self.invoice_ctx)
|
||||
)
|
||||
i = 1
|
||||
while i <= 4:
|
||||
with clearance_plan_wizard.clearance_plan_line_ids.new() as line:
|
||||
line.amount = 200.0
|
||||
line.date_maturity = (datetime.now() + timedelta(days=30 * i)).strftime(
|
||||
"%Y-%m-%d"
|
||||
)
|
||||
i += 1
|
||||
return clearance_plan_wizard
|
||||
|
||||
def test_wizard_values(self):
|
||||
clearance_plan = self.create_and_fill_wizard().save()
|
||||
self.assertEqual(clearance_plan.journal_id.id, self.general_journal.id)
|
||||
self.assertEqual(clearance_plan.amount_to_allocate, 800.0)
|
||||
self.assertEqual(clearance_plan.amount_unallocated, 0.0)
|
||||
|
||||
def test_wizard_negative_amount(self):
|
||||
clearance_plan_wizard = self.create_and_fill_wizard()
|
||||
with clearance_plan_wizard.clearance_plan_line_ids.new() as line:
|
||||
line.amount = -200.0
|
||||
line.date_maturity = datetime.now().strftime("%Y-%m-%d")
|
||||
with self.assertRaises(ValidationError):
|
||||
clearance_plan_wizard.save()
|
||||
|
||||
def test_confirm_clearance_plan(self):
|
||||
clearance_plan = self.create_and_fill_wizard().save()
|
||||
res = clearance_plan.confirm_plan()
|
||||
move = self.env["account.move"].browse(res["res_id"])
|
||||
self.assertEqual(move.journal_id, clearance_plan.journal_id)
|
||||
for line in clearance_plan.clearance_plan_line_ids:
|
||||
self.assertTrue(
|
||||
move.line_ids.filtered(
|
||||
lambda l: l.debit == line.amount
|
||||
and l.date_maturity == line.date_maturity
|
||||
)
|
||||
)
|
||||
for line in self.invoice.move_id.line_ids.filtered(
|
||||
lambda l: l.account_id == self.invoice.account_id
|
||||
):
|
||||
self.assertTrue(line.reconciled)
|
||||
for reconciled_line in line.full_reconcile_id.reconciled_line_ids.filtered(
|
||||
lambda l: l.credit == line.debit
|
||||
):
|
||||
self.assertEqual(reconciled_line.move_id.id, move.id)
|
||||
47
account_clearance_plan/views/res_config_settings.xml
Normal file
47
account_clearance_plan/views/res_config_settings.xml
Normal file
@@ -0,0 +1,47 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Copyright 2020 ACSONE SA/NV
|
||||
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||
|
||||
<odoo>
|
||||
|
||||
|
||||
<record model="ir.ui.view" id="res_config_settings_form_view">
|
||||
<field name="name">res.config.settings.form (in account_clearance_plan)</field>
|
||||
<field name="model">res.config.settings</field>
|
||||
<field name="inherit_id" ref="account.res_config_settings_view_form"/>
|
||||
<field name="arch" type="xml">
|
||||
<xpath expr="//div[@id='invoicing_settings']" position="after">
|
||||
<h2>Clearance Plan</h2>
|
||||
<div class="row mt16 o_settings_container">
|
||||
<div class="row col-md-6 o_setting_box" id="clearance_plan_journal">
|
||||
<div class="col-xs-12 col-md-12 o_setting_box">
|
||||
<div class="o_setting_left_pane"/>
|
||||
<div class="o_setting_right_pane">
|
||||
<div class="row">
|
||||
<label for="clearance_plan_journal_id" class="col-md-5 o_light_label"/>
|
||||
<field name="clearance_plan_journal_id"/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row o_settings_container">
|
||||
<div class="row col-md-6 o_setting_box" id="clearance_plan_move_line">
|
||||
<div class="col-xs-12 col-md-12 o_setting_box">
|
||||
<div class="o_setting_left_pane"/>
|
||||
<div class="o_setting_right_pane">
|
||||
<div class="row">
|
||||
<label for="clearance_plan_move_line_name" class="col-md-5 o_light_label"/>
|
||||
<field name="clearance_plan_move_line_name"/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
|
||||
|
||||
</odoo>
|
||||
1
account_clearance_plan/wizard/__init__.py
Normal file
1
account_clearance_plan/wizard/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from . import account_clearance_plan
|
||||
177
account_clearance_plan/wizard/account_clearance_plan.py
Normal file
177
account_clearance_plan/wizard/account_clearance_plan.py
Normal file
@@ -0,0 +1,177 @@
|
||||
# Copyright 2020 ACSONE SA/NV
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import api, fields, models, _
|
||||
from odoo.exceptions import UserError
|
||||
|
||||
|
||||
class AccountClearancePlanLine(models.TransientModel):
|
||||
_name = "account.clearance.plan.line"
|
||||
_description = "Clearance Plan Line"
|
||||
|
||||
name = fields.Char(
|
||||
string="Label",
|
||||
required=True,
|
||||
default=lambda self:
|
||||
self.env.user.company_id.clearance_plan_move_line_name,
|
||||
)
|
||||
clearance_plan_id = fields.Many2one(
|
||||
comodel_name="account.clearance.plan", required=True
|
||||
)
|
||||
amount = fields.Float(required=True)
|
||||
date_maturity = fields.Date(string="Due Date", required=True)
|
||||
|
||||
@api.constrains("amount")
|
||||
def _check_positive_amount(self):
|
||||
for rec in self:
|
||||
if rec.amount < 0:
|
||||
raise Warning(_("Amounts should all be positive."))
|
||||
|
||||
|
||||
class AccountClearancePlan(models.TransientModel):
|
||||
_name = "account.clearance.plan"
|
||||
_description = "Clearance Plan"
|
||||
|
||||
move_line_ids = fields.Many2many("account.move.line", readonly=True)
|
||||
journal_id = fields.Many2one(
|
||||
string="Journal",
|
||||
comodel_name="account.journal",
|
||||
required=True,
|
||||
help="Journal of the new entry.",
|
||||
)
|
||||
move_ref = fields.Char(
|
||||
string="Journal Entry Reference",
|
||||
help="Reference of the new journal entry that will be generated.",
|
||||
)
|
||||
move_narration = fields.Text(
|
||||
string="Journal Entry Internal Note",
|
||||
help="Internal note of the new journal entry that will be generated.",
|
||||
)
|
||||
amount_to_allocate = fields.Float(string="Total Amount to Allocate", readonly=True)
|
||||
amount_unallocated = fields.Float(
|
||||
string="Amount Unallocated", compute="_compute_amount_unallocated"
|
||||
)
|
||||
clearance_plan_line_ids = fields.One2many(
|
||||
comodel_name="account.clearance.plan.line", inverse_name="clearance_plan_id"
|
||||
)
|
||||
|
||||
@api.onchange("clearance_plan_line_ids")
|
||||
def _compute_amount_unallocated(self):
|
||||
for rec in self:
|
||||
rec.amount_unallocated = rec.amount_to_allocate - sum(
|
||||
rec.clearance_plan_line_ids.mapped("amount")
|
||||
)
|
||||
|
||||
def _get_move_lines_from_context(self):
|
||||
active_model = self._context.get("active_model")
|
||||
if active_model == "account.invoice":
|
||||
move_line_ids = []
|
||||
for invoice in self.env["account.invoice"].browse(
|
||||
self._context.get("active_ids")
|
||||
):
|
||||
move_line_ids += invoice._get_open_move_lines_ids()
|
||||
elif not self._context.get("active_model") == "account.move.line":
|
||||
raise UserError(
|
||||
_(
|
||||
"Programming error: wizard action executed with 'active_model' "
|
||||
"different from 'account.move.line' in context."
|
||||
)
|
||||
)
|
||||
else:
|
||||
move_line_ids = self._context.get("active_ids")
|
||||
|
||||
return self.env["account.move.line"].browse(move_line_ids)
|
||||
|
||||
@api.model
|
||||
def default_get(self, fields):
|
||||
rec = super().default_get(fields)
|
||||
|
||||
move_lines = self._get_move_lines_from_context()
|
||||
account_id = move_lines.mapped("account_id")
|
||||
|
||||
# Check all move lines are from same partner
|
||||
if len(move_lines.mapped("partner_id").ids) != 1:
|
||||
raise UserError(_("Please select items from exactly one partner."))
|
||||
# Check all move lines are from same account
|
||||
if len(account_id.ids) != 1:
|
||||
raise UserError(_("Please select items from exactly one account."))
|
||||
# Check account is of type type is 'receivable' or 'payable'
|
||||
if account_id.user_type_id.type not in ("receivable", "payable"):
|
||||
raise UserError(
|
||||
_(
|
||||
"Please select items from an account "
|
||||
"of type 'receivable' or 'payable'."
|
||||
)
|
||||
)
|
||||
|
||||
rec.update(
|
||||
{
|
||||
"journal_id": self.env.user.company_id.clearance_plan_journal_id.id,
|
||||
"amount_to_allocate": abs(sum(move_lines.mapped("amount_residual"))),
|
||||
"move_line_ids": move_lines.ids,
|
||||
}
|
||||
)
|
||||
|
||||
return rec
|
||||
|
||||
def _create_reverse_amount_residual_lines(self, move):
|
||||
new_lines = self.env["account.move.line"]
|
||||
for line in self.move_line_ids:
|
||||
new_line = line.with_context(check_move_validity=False).copy(
|
||||
default={
|
||||
"move_id": move.id,
|
||||
"debit": abs(line.amount_residual) if line.credit > 0 else 0,
|
||||
"credit": abs(line.amount_residual) if line.debit > 0 else 0,
|
||||
"invoice_id": False,
|
||||
}
|
||||
)
|
||||
new_line.write({"name": (_("Clearance Plan: ") + new_line.name)})
|
||||
new_lines |= new_line
|
||||
return new_lines
|
||||
|
||||
def _create_clearance_move_lines(self, move):
|
||||
account_id = self.move_line_ids.mapped("account_id")
|
||||
partner_id = self.move_line_ids.mapped("partner_id")
|
||||
negative_amount_residual = sum(move.line_ids.mapped("amount_residual")) < 0
|
||||
for line in self.clearance_plan_line_ids:
|
||||
self.env["account.move.line"].with_context(
|
||||
check_move_validity=False
|
||||
).create(
|
||||
{
|
||||
"move_id": move.id,
|
||||
"debit": line.amount if negative_amount_residual else 0,
|
||||
"credit": line.amount if not negative_amount_residual else 0,
|
||||
"date_maturity": line.date_maturity,
|
||||
"name": line.name,
|
||||
"account_id": account_id.id,
|
||||
"partner_id": partner_id.id,
|
||||
}
|
||||
)
|
||||
|
||||
def confirm_plan(self):
|
||||
self.ensure_one()
|
||||
if self.amount_unallocated != 0:
|
||||
raise UserError(_("%s still to allocate.") % self.amount_unallocated)
|
||||
|
||||
move = self.env["account.move"].create(
|
||||
{
|
||||
"journal_id": self.journal_id.id,
|
||||
"ref": self.move_ref,
|
||||
"narration": self.move_narration,
|
||||
}
|
||||
)
|
||||
reversed_lines = self._create_reverse_amount_residual_lines(move)
|
||||
self._create_clearance_move_lines(move)
|
||||
|
||||
# Assert balance once all mv_line created
|
||||
move.assert_balanced()
|
||||
move.action_post()
|
||||
(self.move_line_ids | reversed_lines).reconcile()
|
||||
|
||||
return {
|
||||
"type": "ir.actions.act_window",
|
||||
"res_model": "account.move",
|
||||
"res_id": move.id,
|
||||
"view_mode": "form",
|
||||
"context": self.env.context,
|
||||
}
|
||||
@@ -0,0 +1,56 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<odoo>
|
||||
|
||||
<record id="clearance_plan_wizard" model="ir.ui.view">
|
||||
<field name="name">Clearance Plan</field>
|
||||
<field name="model">account.clearance.plan</field>
|
||||
<field name="arch" type="xml">
|
||||
<form>
|
||||
<group>
|
||||
<field name="journal_id"/>
|
||||
<field name="move_ref"/>
|
||||
<field name="move_narration"/>
|
||||
<field name="amount_to_allocate"/>
|
||||
<field name="amount_unallocated"/>
|
||||
</group>
|
||||
<notebook>
|
||||
<page string="Clearance Plan Board">
|
||||
<field name="clearance_plan_line_ids">
|
||||
<tree editable="bottom">
|
||||
<field name="name"/>
|
||||
<field name="date_maturity"/>
|
||||
<field name="amount"/>
|
||||
</tree>
|
||||
</field>
|
||||
</page>
|
||||
</notebook>
|
||||
<footer>
|
||||
<button name="confirm_plan" string="Confirm" type="object" class="btn-primary"/>
|
||||
<button string="Cancel" class="btn-default" special="cancel"/>
|
||||
</footer>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<!--Add to account.move.line action-->
|
||||
<act_window id="act__move_line_clearance_plan_wizard"
|
||||
name="Clearance Plan"
|
||||
res_model="account.clearance.plan"
|
||||
src_model="account.move.line"
|
||||
view_mode="form"
|
||||
groups="account.group_account_manager"
|
||||
key2="client_action_multi"
|
||||
target="new" />
|
||||
|
||||
<!--Add to account.invoice action-->
|
||||
<act_window id="act_invoice_clearance_plan_wizard"
|
||||
name="Clearance Plan"
|
||||
res_model="account.clearance.plan"
|
||||
src_model="account.invoice"
|
||||
view_mode="form"
|
||||
groups="account.group_account_manager"
|
||||
key2="client_action_multi"
|
||||
target="new" />
|
||||
|
||||
|
||||
</odoo>
|
||||
1
setup/account_clearance_plan/odoo/addons/account_clearance_plan
Symbolic link
1
setup/account_clearance_plan/odoo/addons/account_clearance_plan
Symbolic link
@@ -0,0 +1 @@
|
||||
../../../../account_clearance_plan
|
||||
6
setup/account_clearance_plan/setup.py
Normal file
6
setup/account_clearance_plan/setup.py
Normal file
@@ -0,0 +1,6 @@
|
||||
import setuptools
|
||||
|
||||
setuptools.setup(
|
||||
setup_requires=['setuptools-odoo'],
|
||||
odoo_addon=True,
|
||||
)
|
||||
Reference in New Issue
Block a user