mirror of
https://github.com/OCA/account-financial-tools.git
synced 2025-02-02 12:47:26 +02:00
Create module account_tag_category
This commit is contained in:
62
account_tag_category/README.rst
Normal file
62
account_tag_category/README.rst
Normal file
@@ -0,0 +1,62 @@
|
||||
.. image:: https://img.shields.io/badge/licence-AGPL--3-blue.svg
|
||||
:target: http://www.gnu.org/licenses/agpl
|
||||
:alt: License: AGPL-3
|
||||
|
||||
====================
|
||||
Account Tag Category
|
||||
====================
|
||||
|
||||
This module extends the functionality of account module to support the grouping
|
||||
of accounts tags by category and to allow you to set a category as optional
|
||||
or required.
|
||||
|
||||
Usage
|
||||
=====
|
||||
|
||||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas
|
||||
:alt: Try me on Runbot
|
||||
:target: https://runbot.odoo-community.org/runbot/82/10.0
|
||||
|
||||
Known issues / Roadmap
|
||||
======================
|
||||
|
||||
* Color picker should be improved
|
||||
* Taxes applicable tags should be supported
|
||||
|
||||
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 smash it by providing detailed and welcomed feedback.
|
||||
|
||||
Credits
|
||||
=======
|
||||
|
||||
Images
|
||||
------
|
||||
|
||||
* Odoo Community Association: `Icon <https://github.com/OCA/maintainer-tools/blob/master/template/module/static/description/icon.svg>`_.
|
||||
|
||||
Contributors
|
||||
------------
|
||||
|
||||
* Akim Juillerat <akim.juillerat@camptocamp.com>
|
||||
|
||||
Do not contact contributors directly about support or help with technical issues.
|
||||
|
||||
Maintainer
|
||||
----------
|
||||
|
||||
.. image:: https://odoo-community.org/logo.png
|
||||
:alt: Odoo Community Association
|
||||
:target: https://odoo-community.org
|
||||
|
||||
This module is maintained by the OCA.
|
||||
|
||||
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.
|
||||
|
||||
To contribute to this module, please visit https://odoo-community.org.
|
||||
2
account_tag_category/__init__.py
Normal file
2
account_tag_category/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
||||
from . import models
|
||||
from . import wizard
|
||||
23
account_tag_category/__manifest__.py
Normal file
23
account_tag_category/__manifest__.py
Normal file
@@ -0,0 +1,23 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Camptocamp SA
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
{
|
||||
"name": "Account Tag Category",
|
||||
"summary": "Group account tags by categories",
|
||||
"version": "10.0.1.0.0",
|
||||
"category": "Accounting & Finance",
|
||||
"website": "https://www.camptocamp.com/",
|
||||
"author": "Camptocamp SA,Odoo Community Association (OCA)",
|
||||
"license": "AGPL-3",
|
||||
"depends": [
|
||||
"account",
|
||||
"web_m2x_options"
|
||||
],
|
||||
"data": [
|
||||
"security/ir.model.access.csv",
|
||||
"wizard/update_tags.xml",
|
||||
"views/account.xml",
|
||||
],
|
||||
"application": False,
|
||||
"installable": True,
|
||||
}
|
||||
1
account_tag_category/models/__init__.py
Normal file
1
account_tag_category/models/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from . import account
|
||||
127
account_tag_category/models/account.py
Normal file
127
account_tag_category/models/account.py
Normal file
@@ -0,0 +1,127 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Camptocamp SA
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import models, fields, api, _
|
||||
from odoo.exceptions import ValidationError
|
||||
|
||||
|
||||
class AccountAccountTag(models.Model):
|
||||
|
||||
_inherit = 'account.account.tag'
|
||||
|
||||
tag_category_id = fields.Many2one('account.account.tag.category',
|
||||
'tag_ids', ondelete='set null')
|
||||
|
||||
category_color = fields.Integer(related='tag_category_id.color')
|
||||
|
||||
@api.multi
|
||||
def read(self, fields=None, load='_classic_read'):
|
||||
# This function is used to read from category_color field instead of
|
||||
# color without having to change anything on the client side.
|
||||
if 'color' in fields:
|
||||
fields.append('category_color')
|
||||
if 'tag_category_id' not in fields:
|
||||
fields.append('tag_category_id')
|
||||
result = super(AccountAccountTag, self).read(fields, load)
|
||||
for rec in result:
|
||||
if 'category_color' in rec and rec.get('tag_category_id', False):
|
||||
color = rec.pop('category_color')
|
||||
rec['color'] = color
|
||||
return result
|
||||
|
||||
|
||||
class AccountAccountTagCategory(models.Model):
|
||||
|
||||
_name = 'account.account.tag.category'
|
||||
|
||||
_description = 'Account Tag Category'
|
||||
|
||||
name = fields.Char(required=True)
|
||||
|
||||
color = fields.Integer('Color Index', compute='_compute_color_index',
|
||||
store=True)
|
||||
|
||||
color_picker = fields.Selection([('0', 'Grey'),
|
||||
('1', 'Green'),
|
||||
('2', 'Yellow'),
|
||||
('3', 'Orange'),
|
||||
('4', 'Red'),
|
||||
('5', 'Purple'),
|
||||
('6', 'Blue'),
|
||||
('7', 'Cyan'),
|
||||
('8', 'Aquamarine'),
|
||||
('9', 'Pink')], string='Tags Color',
|
||||
required=True, default='0')
|
||||
|
||||
enforce_policy = fields.Selection(
|
||||
[('required', 'Required'), ('optional', 'Optional')],
|
||||
required=True, default='optional',
|
||||
help='If required, this option enforces the use of a tag from this '
|
||||
'category. If optional, the user is not required to use a tag '
|
||||
'from this category.')
|
||||
|
||||
tag_ids = fields.One2many('account.account.tag', 'tag_category_id',
|
||||
string='Tags',)
|
||||
|
||||
# TODO support applicability for taxes
|
||||
|
||||
applicability = fields.Selection([
|
||||
('accounts', 'Accounts'),
|
||||
# ('taxes', 'Taxes'),
|
||||
], default='accounts', required=True)
|
||||
|
||||
@api.depends('color_picker')
|
||||
def _compute_color_index(self):
|
||||
for category in self:
|
||||
category.color = int(category.color_picker)
|
||||
|
||||
@api.constrains('tag_ids', 'applicability')
|
||||
def _check_tags_applicability(self):
|
||||
self.ensure_one()
|
||||
for tag in self.tag_ids:
|
||||
if tag.applicability != self.applicability:
|
||||
raise ValidationError(_('Selected tag must be applicable on '
|
||||
'the same model as this category (%s)'
|
||||
) % self.applicability)
|
||||
|
||||
|
||||
class AccountAccount(models.Model):
|
||||
|
||||
_inherit = 'account.account'
|
||||
|
||||
@api.constrains('tag_ids')
|
||||
def _check_tags_categories(self):
|
||||
self.ensure_one()
|
||||
self._check_required_categories()
|
||||
self._check_unique_tag_per_category()
|
||||
|
||||
def _check_unique_tag_per_category(self):
|
||||
used_categories_ids = []
|
||||
for tag in self.tag_ids:
|
||||
if not tag.tag_category_id:
|
||||
continue
|
||||
tag_id = tag.tag_category_id.id
|
||||
if tag_id in used_categories_ids:
|
||||
raise ValidationError(_('There is more than one tag from the '
|
||||
'same category which is used'))
|
||||
else:
|
||||
used_categories_ids.append(tag_id)
|
||||
|
||||
def _check_required_categories(self):
|
||||
required_categories = self.env['account.account.tag.category'].search(
|
||||
[('applicability', '=', 'accounts'),
|
||||
('enforce_policy', '=', 'required')])
|
||||
errors = []
|
||||
for category in required_categories:
|
||||
found = False
|
||||
for tag in self.tag_ids:
|
||||
if tag in category.tag_ids:
|
||||
found = True
|
||||
if not found:
|
||||
errors.append(category.name)
|
||||
if errors:
|
||||
text_error = '\n'.join(errors)
|
||||
raise ValidationError(_('Following tag categories are set as '
|
||||
'required, but there is no tag from these '
|
||||
'categories : \n %s') % text_error)
|
||||
3
account_tag_category/security/ir.model.access.csv
Normal file
3
account_tag_category/security/ir.model.access.csv
Normal file
@@ -0,0 +1,3 @@
|
||||
"id","name","model_id:id","group_id:id","perm_read","perm_write","perm_create","perm_unlink"
|
||||
access_account_account_tag_category_user,access_account_account_tag_category_user,model_account_account_tag_category,account.group_account_user,1,0,0,0
|
||||
access_account_account_tag_category_manager,access_account_account_tag_category_manager,model_account_account_tag_category,account.group_account_manager,1,1,1,1
|
||||
|
0
account_tag_category/tests/__init__.py
Normal file
0
account_tag_category/tests/__init__.py
Normal file
110
account_tag_category/tests/test_account_tag_category.py
Normal file
110
account_tag_category/tests/test_account_tag_category.py
Normal file
@@ -0,0 +1,110 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Camptocamp SA
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo.tests.common import TransactionCase
|
||||
from odoo.exceptions import ValidationError
|
||||
|
||||
|
||||
class TestAccountTagCategory(TransactionCase):
|
||||
|
||||
def setUp(self):
|
||||
super(TestAccountTagCategory, self).setUp()
|
||||
|
||||
# Create a few tags
|
||||
self.tag_model = self.env['account.account.tag'].with_context(
|
||||
default_applicability='accounts')
|
||||
category_model = self.env[
|
||||
'account.account.tag.category'].with_context(
|
||||
default_applicability='accounts')
|
||||
|
||||
tags_to_create = ['123', '456', '789', 'ABC', 'DEF', 'GHI']
|
||||
|
||||
for tag_name in tags_to_create:
|
||||
self.tag_model.create({
|
||||
'name': tag_name,
|
||||
})
|
||||
|
||||
self.letters_category = category_model.create({
|
||||
'name': 'Letters',
|
||||
'enforce_policy': 'required',
|
||||
'color_picker': '1',
|
||||
})
|
||||
self.numbers_category = category_model.create({
|
||||
'name': 'Numbers',
|
||||
'enforce_policy': 'optional',
|
||||
'color_picker': '2',
|
||||
})
|
||||
|
||||
update_wizard = self.env['account.tag.category.update.tags']
|
||||
|
||||
update_wizard.create({
|
||||
'tag_category_id': self.letters_category.id,
|
||||
'tag_ids': [(6, False, self.tag_model.search(
|
||||
['|', '|', ('name', '=', 'ABC'), ('name', '=', 'DEF'),
|
||||
('name', '=', 'GHI')]).ids)],
|
||||
}).save_tags_to_category()
|
||||
|
||||
update_wizard.create({
|
||||
'tag_category_id': self.numbers_category.id,
|
||||
'tag_ids': [(6, False, self.tag_model.search(
|
||||
['|', '|', ('name', '=', '123'), ('name', '=', '456'),
|
||||
('name', '=', '789')]).ids)],
|
||||
}).save_tags_to_category()
|
||||
|
||||
|
||||
def test_categories(self):
|
||||
|
||||
self.assertEqual(self.letters_category.color,
|
||||
int(self.letters_category.color_picker))
|
||||
|
||||
self.tag_model.invalidate_cache()
|
||||
|
||||
self.assertEqual(self.tag_model.search(
|
||||
[('name', '=', 'ABC')]).read(['color'])[0]['color'],
|
||||
self.letters_category.color)
|
||||
|
||||
self.assertEqual(len(self.letters_category.tag_ids), 3)
|
||||
|
||||
# Missing required category
|
||||
with self.assertRaises(ValidationError):
|
||||
self.env['account.account'].create({
|
||||
'name': "Dummy account",
|
||||
'code': "DUMMY",
|
||||
'user_type_id': self.env.ref(
|
||||
'account.data_account_type_equity').id,
|
||||
})
|
||||
|
||||
with self.assertRaises(ValidationError):
|
||||
self.env['account.account'].create({
|
||||
'name': "Dummy account",
|
||||
'code': "DUMMY 2",
|
||||
'user_type_id': self.env.ref(
|
||||
'account.data_account_type_equity').id,
|
||||
'tag_ids': [(6, False, self.tag_model.search(
|
||||
[('name', '=', '123')]).ids)]
|
||||
})
|
||||
|
||||
# Two times same category
|
||||
with self.assertRaises(ValidationError):
|
||||
self.env['account.account'].create({
|
||||
'name': "Dummy account",
|
||||
'code': "DUMMY 3",
|
||||
'user_type_id': self.env.ref(
|
||||
'account.data_account_type_equity').id,
|
||||
'tag_ids': [(6, False,
|
||||
self.tag_model.search(
|
||||
['|', ('name', '=', 'ABC'),
|
||||
('name', '=', 'DEF')]).ids)]
|
||||
})
|
||||
|
||||
self.env['account.account'].create({
|
||||
'name': "Dummy account",
|
||||
'code': "DUMMY 4",
|
||||
'user_type_id': self.env.ref(
|
||||
'account.data_account_type_equity').id,
|
||||
'tag_ids': [(6, False,
|
||||
self.tag_model.search(
|
||||
['|', ('name', '=', '123'),
|
||||
('name', '=', 'DEF')]).ids)]
|
||||
})
|
||||
58
account_tag_category/views/account.xml
Normal file
58
account_tag_category/views/account.xml
Normal file
@@ -0,0 +1,58 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<odoo>
|
||||
|
||||
<record id="account_tag_category_form" model="ir.ui.view">
|
||||
<field name="name">account.account.tag.category.form</field>
|
||||
<field name="model">account.account.tag.category</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="Tag Category">
|
||||
<header class="oe_edit_only">
|
||||
<button name="%(action_update_tags_on_category)d" type="action" string="Update Tags" class="oe_highlight" groups="account.group_account_manager"/>
|
||||
</header>
|
||||
<group>
|
||||
<field name="name" />
|
||||
<field name="applicability" />
|
||||
<field name="enforce_policy" />
|
||||
<field name="color_picker" />
|
||||
</group>
|
||||
<group>
|
||||
<field name="tag_ids" readonly="True"/>
|
||||
</group>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
<record id="account_tag_category_tree" model="ir.ui.view">
|
||||
<field name="name">account.account.tag.category.tree</field>
|
||||
<field name="model">account.account.tag.category</field>
|
||||
<field name="arch" type="xml">
|
||||
<tree string="Tag Categories">
|
||||
<field name="name" />
|
||||
<field name="applicability" />
|
||||
<field name="enforce_policy" />
|
||||
</tree>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="account_tag_category_action" model="ir.actions.act_window">
|
||||
<field name="name">Account Tag Categories</field>
|
||||
<field name="res_model">account.account.tag.category</field>
|
||||
<field name="view_mode">tree,form</field>
|
||||
<field name="context">{'default_applicability': 'accounts'}</field>
|
||||
</record>
|
||||
|
||||
<menuitem id="account_tag_category_menu"
|
||||
action="account_tag_category_action"
|
||||
parent="account.account_account_menu"
|
||||
sequence="40" />
|
||||
|
||||
<record id="view_account_form" model="ir.ui.view">
|
||||
<field name="name">account.account.form.inherit</field>
|
||||
<field name="model">account.account</field>
|
||||
<field name="inherit_id" ref="account.view_account_form" />
|
||||
<field name="arch" type="xml">
|
||||
<field name="tag_ids" position="attributes">
|
||||
<attribute name="options">{'no_color_picker': True}</attribute>
|
||||
</field>
|
||||
</field>
|
||||
</record>
|
||||
</odoo>
|
||||
1
account_tag_category/wizard/__init__.py
Normal file
1
account_tag_category/wizard/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from . import update_tags_wizard
|
||||
31
account_tag_category/wizard/update_tags.xml
Normal file
31
account_tag_category/wizard/update_tags.xml
Normal file
@@ -0,0 +1,31 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<odoo>
|
||||
|
||||
<record id="action_update_tags_on_category" model="ir.actions.act_window">
|
||||
<field name="name">Update Tags</field>
|
||||
<field name="res_model">account.tag.category.update.tags</field>
|
||||
<field name="view_type">form</field>
|
||||
<field name="view_mode">form</field>
|
||||
<field name="groups_id" eval="[(4, ref('account.group_account_manager'))]"/>
|
||||
<field name="context">{'default_tag_category_id': active_id}</field>
|
||||
<field name="target">new</field>
|
||||
</record>
|
||||
|
||||
<record id="account_tag_category_update_tags_form" model="ir.ui.view">
|
||||
<field name="name">account.tag.category.update.tags.form</field>
|
||||
<field name="model">account.tag.category.update.tags</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="Update tags">
|
||||
<group>
|
||||
<field name="tag_category_id" />
|
||||
<field name="tag_ids" />
|
||||
</group>
|
||||
<footer>
|
||||
<button name='save_tags_to_category' string='Save tags' class='oe_highlight' type='object' />
|
||||
<button special="cancel" string="Close" type="object" class="oe_link oe_inline"/>
|
||||
</footer>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
</odoo>
|
||||
35
account_tag_category/wizard/update_tags_wizard.py
Normal file
35
account_tag_category/wizard/update_tags_wizard.py
Normal file
@@ -0,0 +1,35 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2017 Camptocamp SA
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||
|
||||
from odoo import models, fields, api
|
||||
|
||||
|
||||
class AccountTagCategoryUpdateTags(models.TransientModel):
|
||||
|
||||
_name = 'account.tag.category.update.tags'
|
||||
|
||||
_description = 'Update account tags on account tag category'
|
||||
|
||||
tag_ids = fields.Many2many('account.account.tag',
|
||||
domain=[('applicability', '=', 'accounts')])
|
||||
# TODO support applicability for taxes in domain
|
||||
|
||||
tag_category_id = fields.Many2one('account.account.tag.category')
|
||||
|
||||
@api.model
|
||||
def default_get(self, fields):
|
||||
res = super(AccountTagCategoryUpdateTags, self).default_get(fields)
|
||||
if 'tag_ids' in fields and not res.get('tag_ids'):
|
||||
res['tag_ids'] = self.env['account.account.tag'].search(
|
||||
[('tag_category_id', '=', res.get('tag_category_id'))]).ids
|
||||
return res
|
||||
|
||||
@api.multi
|
||||
def save_tags_to_category(self):
|
||||
|
||||
self.tag_category_id.write({
|
||||
'tag_ids': [(6, False, self.tag_ids.ids)]
|
||||
})
|
||||
|
||||
return
|
||||
Reference in New Issue
Block a user