@ -0,0 +1,51 @@ |
|||||
|
.. image:: https://img.shields.io/badge/license-AGPL--3-blue.svg |
||||
|
:target: https://www.gnu.org/licenses/agpl-3.0-standalone.html |
||||
|
:alt: License: AGPL-3 |
||||
|
|
||||
|
Global Search |
||||
|
============= |
||||
|
This module allows users to search the records in Customers, Products, Sale, Purchase, Inventory and Accounting Modules. |
||||
|
|
||||
|
Configuration |
||||
|
============= |
||||
|
* No additional configuration is needed. |
||||
|
|
||||
|
Company |
||||
|
------- |
||||
|
* `Cybrosys Techno Solutions <https://cybrosys.com/>`__ |
||||
|
|
||||
|
License |
||||
|
------- |
||||
|
Affero General Public License, v3.0 (AGPL v3). |
||||
|
(https://www.gnu.org/licenses/agpl-3.0-standalone.html) |
||||
|
|
||||
|
Credits |
||||
|
------- |
||||
|
* Developers: (v14) Shahil, |
||||
|
(v15) Athul, |
||||
|
(v16) Rosmy, |
||||
|
(v17) Anjhana A K, |
||||
|
(v18) Aysha Shalin |
||||
|
Contact: odoo@cybrosys.com |
||||
|
|
||||
|
Contacts |
||||
|
-------- |
||||
|
* Mail Contact : odoo@cybrosys.com |
||||
|
* Website : https://cybrosys.com |
||||
|
|
||||
|
Bug Tracker |
||||
|
----------- |
||||
|
Bugs are tracked on GitHub Issues. In case of trouble, please check there if your issue has already been reported. |
||||
|
|
||||
|
Maintainer |
||||
|
========== |
||||
|
.. image:: https://cybrosys.com/images/logo.png |
||||
|
:target: https://cybrosys.com |
||||
|
|
||||
|
This module is maintained by Cybrosys Technologies. |
||||
|
|
||||
|
For support and more information, please visit `Our Website <https://cybrosys.com/>`__ |
||||
|
|
||||
|
Further information |
||||
|
=================== |
||||
|
HTML Description: `<static/description/index.html>`__ |
@ -0,0 +1,22 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Aysha Shalin (odoo@cybrosys.com) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU AFFERO |
||||
|
# GENERAL PUBLIC LICENSE (AGPL 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 AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC |
||||
|
# LICENSE (AGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################## |
||||
|
from . import models |
@ -0,0 +1,51 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Aysha Shalin (odoo@cybrosys.com) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU AFFERO |
||||
|
# GENERAL PUBLIC LICENSE (AGPL 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 AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC |
||||
|
# LICENSE (AGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################## |
||||
|
{ |
||||
|
'name': 'Global Search', |
||||
|
'version': '18.0.1.0.0', |
||||
|
'category': 'Extra Tools', |
||||
|
'summary': """Easy Search in Customers, Products, Sale, Purchase, Inventory |
||||
|
and Accounting modules""", |
||||
|
'description': """This module allows users to search the records in |
||||
|
Customers, Products, Sale, Purchase, Inventory and Accounting Modules.""", |
||||
|
'author': 'Cybrosys Techno Solutions', |
||||
|
'company': 'Cybrosys Techno Solutions', |
||||
|
'maintainer': 'Cybrosys Techno Solutions', |
||||
|
'website': 'https://www.cybrosys.com', |
||||
|
'depends': ['base', 'stock', 'sale', 'purchase'], |
||||
|
'data': [ |
||||
|
'security/master_search_security.xml', |
||||
|
'security/ir.model.access.csv', |
||||
|
'views/master_search_view.xml' |
||||
|
], |
||||
|
'assets': { |
||||
|
'web.assets_backend': [ |
||||
|
'master_search/static/src/scss/master_search.scss', |
||||
|
'master_search/static/src/js/master_search.js', |
||||
|
], |
||||
|
}, |
||||
|
'images': ['static/description/banner.png'], |
||||
|
'license': 'AGPL-3', |
||||
|
'installable': True, |
||||
|
'auto_install': False, |
||||
|
'application': False, |
||||
|
} |
@ -0,0 +1,6 @@ |
|||||
|
## Module <master_search> |
||||
|
|
||||
|
#### 30.11.2024 |
||||
|
#### Version 18.0.1.0.0 |
||||
|
##### ADD |
||||
|
- Initial Commit for Global Search |
@ -0,0 +1,22 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Aysha Shalin (odoo@cybrosys.com) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU AFFERO |
||||
|
# GENERAL PUBLIC LICENSE (AGPL 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 AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC |
||||
|
# LICENSE (AGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################## |
||||
|
from . import master_search |
@ -0,0 +1,317 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Aysha Shalin (odoo@cybrosys.com) |
||||
|
# |
||||
|
# You can modify it under the terms of the GNU AFFERO |
||||
|
# GENERAL PUBLIC LICENSE (AGPL 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 AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU AFFERO GENERAL PUBLIC |
||||
|
# LICENSE (AGPL v3) along with this program. |
||||
|
# If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################## |
||||
|
from odoo import api, fields, models, _ |
||||
|
from odoo.exceptions import UserError |
||||
|
|
||||
|
|
||||
|
class MasterSearch(models.Model): |
||||
|
""" Master search model for easy search """ |
||||
|
_name = 'master.search' |
||||
|
_description = "Model for master search" |
||||
|
_rec_name = 'name' |
||||
|
_order = "create_date desc" |
||||
|
|
||||
|
name = fields.Char(string="Name", default=lambda self: _('Search'), |
||||
|
help='Searched names') |
||||
|
search_string = fields.Char(string="Search", help='Details to search') |
||||
|
search_mode = fields.Selection( |
||||
|
[('all', 'All'), ('active', 'Active'), |
||||
|
('inactive', 'Inactive')], |
||||
|
string="Search Mode", default="active", help='Search Details based on') |
||||
|
search_by = fields.Selection( |
||||
|
[('any', 'Any'), ('customer', 'Customer'), |
||||
|
('product', 'Product'), |
||||
|
('sale details', 'Sale'), |
||||
|
('purchase details', 'Purchase'), |
||||
|
('transaction details', 'Inventory'), |
||||
|
('account details', 'Accounting')], |
||||
|
string="Search By", default='any', help='Search in which model') |
||||
|
master_search_ids = fields.Many2many('master.search', |
||||
|
'master_search_self_rel', |
||||
|
'search_id', |
||||
|
'search_id1', |
||||
|
compute="_get_recent_searches", |
||||
|
limit=1, help='Recent search details') |
||||
|
history_count = fields.Integer(string="History Count", |
||||
|
compute="_get_history_count", |
||||
|
help='Recent search History Count') |
||||
|
customer_ids = fields.Many2many('res.partner', |
||||
|
'master_search_company_rel', |
||||
|
'search_id', 'company_id', |
||||
|
help='To fetch datas of customer search') |
||||
|
product_ids = fields.Many2many('product.template', |
||||
|
'master_search_product_rel', |
||||
|
'search_id', 'company_id', |
||||
|
help='To fetch datas of product search') |
||||
|
transaction_ids = fields.Many2many('stock.picking', |
||||
|
'master_search_transaction_rel', |
||||
|
'search_id', |
||||
|
'company_id', |
||||
|
string="Inventory", |
||||
|
help='To fetch datas of inventory' |
||||
|
' search') |
||||
|
customer_count = fields.Integer(string="Company Count", |
||||
|
compute="_get_operator_count", |
||||
|
help='To fetched customer search count') |
||||
|
product_count = fields.Integer(string="Product Count", |
||||
|
compute="_get_product_count", |
||||
|
help='To fetched product search count') |
||||
|
transaction_count = fields.Integer(string="Transaction Count", |
||||
|
compute="_get_transaction_count", |
||||
|
help='To fetched inventory search ' |
||||
|
'count') |
||||
|
sale_count = fields.Integer(string="Sale Count", compute="_get_sale_count", |
||||
|
help='To fetched sale search count') |
||||
|
purchase_count = fields.Integer(string="Sale Count", |
||||
|
compute="_get_purchase_count", |
||||
|
help='To fetched purchase search count') |
||||
|
account_count = fields.Integer(string="Account Count", |
||||
|
compute="_get_account_count", |
||||
|
help='To fetched account search count') |
||||
|
user_id = fields.Many2one('res.users', string="User", |
||||
|
default=lambda self: self.env.user) |
||||
|
match_entire = fields.Boolean(string="Match entire sentence", |
||||
|
help='Only matched datas to be viewed') |
||||
|
sale_ids = fields.Many2many('sale.order', |
||||
|
'master_search_sale_details_rel', |
||||
|
'search_id', 'company_id', |
||||
|
string="Sale", help='To fetch datas of sale ' |
||||
|
'search') |
||||
|
purchase_ids = fields.Many2many('purchase.order', |
||||
|
'master_search_purchase_details_rel', |
||||
|
'search_id', 'company_id', |
||||
|
string="Sale", help='To fetch datas of' |
||||
|
' purchase search') |
||||
|
account_ids = fields.Many2many('account.move', |
||||
|
'master_search_account_details_rel', |
||||
|
'search_id', 'company_id', |
||||
|
string="Account", help='To fetch datas of' |
||||
|
'account search') |
||||
|
|
||||
|
@api.depends('search_string') |
||||
|
def _get_recent_searches(self): |
||||
|
""" Get recent searches """ |
||||
|
try: |
||||
|
current_id = self.id if isinstance(self.id, int) \ |
||||
|
else self._origin.id |
||||
|
except: |
||||
|
current_id = False |
||||
|
pass |
||||
|
empty_search = self.env['master.search'].search( |
||||
|
[('search_string', 'in', ['', False]), |
||||
|
('id', 'not in', [current_id, False] |
||||
|
if current_id else [False])]) |
||||
|
if empty_search: |
||||
|
empty_search.unlink() |
||||
|
recent_searches = self.env['master.search'].search([ |
||||
|
('search_string', 'not in', ['', False])]) |
||||
|
self.master_search_ids = recent_searches |
||||
|
|
||||
|
def action_unlink_search(self): |
||||
|
""" Unlink search """ |
||||
|
self.unlink() |
||||
|
action = self.env.ref('master_search.master_search_action').read()[0] |
||||
|
return action |
||||
|
|
||||
|
@api.depends('master_search_ids') |
||||
|
def _get_history_count(self): |
||||
|
""" Get history count """ |
||||
|
self.history_count = len(self.master_search_ids) |
||||
|
|
||||
|
@api.depends('product_ids') |
||||
|
def _get_product_count(self): |
||||
|
""" Get product count """ |
||||
|
self.product_count = len(self.product_ids) |
||||
|
|
||||
|
@api.depends('customer_ids') |
||||
|
def _get_operator_count(self): |
||||
|
""" Get customer count """ |
||||
|
self.customer_count = len(self.customer_ids) |
||||
|
|
||||
|
@api.depends('transaction_count') |
||||
|
def _get_transaction_count(self): |
||||
|
""" Get transaction details count """ |
||||
|
self.transaction_count = len(self.transaction_ids) |
||||
|
|
||||
|
@api.depends('sale_count') |
||||
|
def _get_sale_count(self): |
||||
|
""" Get sale details count """ |
||||
|
self.sale_count = len(self.sale_ids) |
||||
|
|
||||
|
@api.depends('purchase_count') |
||||
|
def _get_purchase_count(self): |
||||
|
""" Get purchase details count """ |
||||
|
self.purchase_count = len(self.purchase_ids) |
||||
|
|
||||
|
@api.depends('account_count') |
||||
|
def _get_account_count(self): |
||||
|
""" Get account details count """ |
||||
|
self.account_count = len(self.account_ids) |
||||
|
|
||||
|
def action_clear_search(self): |
||||
|
""" clear search input """ |
||||
|
self.search_string = "" |
||||
|
self.name = "Search" |
||||
|
|
||||
|
@api.model_create_multi |
||||
|
def create(self, vals_list): |
||||
|
""" Function for unlink first result and raise error if no string """ |
||||
|
res = super(MasterSearch, self).create(vals_list) |
||||
|
search_index = self.env['master.search'].search_count( |
||||
|
[('user_id', '=', self.env.user.id)]) |
||||
|
# unlink old search result if count greater than 10 |
||||
|
if search_index > 10: |
||||
|
last_search = self.env['master.search'].search( |
||||
|
[('id', '!=', res.id), ('user_id', '=', self.env.user.id)], |
||||
|
order="create_date asc", limit=1) |
||||
|
last_search.unlink() if last_search else False |
||||
|
return res |
||||
|
|
||||
|
def action_search(self): |
||||
|
""" search for the string and store search data """ |
||||
|
if self.search_string and "*" in self.search_string: |
||||
|
return |
||||
|
if not self.search_string: |
||||
|
raise UserError(_("Please provide a search string!")) |
||||
|
search_keys = self.search_string.split(" ") |
||||
|
self.customer_ids = self.product_ids = self.transaction_ids = False |
||||
|
if self.match_entire: |
||||
|
return self._search_query(self.search_string) |
||||
|
for key in search_keys: |
||||
|
self._search_query(key) |
||||
|
self.name = self.search_string |
||||
|
|
||||
|
def _search_query(self, key): |
||||
|
""" search for the model with given key and update result """ |
||||
|
company_id = self.env.user.company_id.id |
||||
|
if self.search_mode == 'all': |
||||
|
active_qry = """ and obj.active in ({},{}) |
||||
|
""".format("'FALSE'", "'TRUE'") |
||||
|
elif self.search_mode == 'active': |
||||
|
active_qry = """ and obj.active in ({})""".format("'TRUE'") |
||||
|
else: |
||||
|
active_qry = """ and obj.active in ({})""".format("'FALSE'") |
||||
|
self._search_customer(key, active_qry) \ |
||||
|
if self.search_by in ['any', 'customer'] else False |
||||
|
self._search_products(key, active_qry, company_id) \ |
||||
|
if self.search_by in ['any', 'product'] else False |
||||
|
self._search_inventory_transactions(key, active_qry, company_id) \ |
||||
|
if self.search_by in ['any', 'transaction details'] else False |
||||
|
self._search_sale_transactions(key, active_qry, company_id) \ |
||||
|
if self.search_by in ['any', 'sale details'] else False |
||||
|
self._search_purchase_transactions(key, active_qry, company_id) \ |
||||
|
if self.search_by in ['any', 'purchase details'] else False |
||||
|
self._search_account_transactions(key, active_qry, company_id) \ |
||||
|
if self.search_by in ['any', 'account details'] else False |
||||
|
|
||||
|
def _search_account_transactions(self, key, active_qry, company_id): |
||||
|
""" Search for all account transactions """ |
||||
|
sp_query = """ SELECT am.id from account_move am |
||||
|
LEFT JOIN res_partner p on p.id = am.partner_id |
||||
|
WHERE am.company_id = {op_id} AND (am.name ILIKE '%{key}%' OR |
||||
|
p.name ILIKE '%{key}%' OR am.state ILIKE '%{key}%') |
||||
|
GROUP BY am.id,p.name |
||||
|
""" |
||||
|
self._cr.execute( |
||||
|
sp_query.format(op_id=company_id, key=key, active=active_qry)) |
||||
|
moves = self._cr.dictfetchall() |
||||
|
move_ids = self.env['account.move'].browse([i['id'] for i in moves]) |
||||
|
self.account_ids += move_ids |
||||
|
|
||||
|
def _search_purchase_transactions(self, key, active_qry, company_id): |
||||
|
""" Search for all purchase transactions """ |
||||
|
sp_query = """ SELECT po.id from purchase_order po |
||||
|
LEFT JOIN res_partner p on p.id = po.partner_id |
||||
|
WHERE po.company_id = {op_id} AND (po.name ILIKE '%{key}%' OR |
||||
|
p.name ILIKE '%{key}%' OR po.state ILIKE '%{key}%') |
||||
|
GROUP BY po.id,p.name |
||||
|
""" |
||||
|
self._cr.execute( |
||||
|
sp_query.format(op_id=company_id, key=key, active=active_qry)) |
||||
|
purchases = self._cr.dictfetchall() |
||||
|
purchase_ids = self.env['purchase.order'].browse( |
||||
|
[i['id'] for i in purchases]) |
||||
|
self.purchase_ids += purchase_ids |
||||
|
|
||||
|
def _search_sale_transactions(self, key, active_qry, company_id): |
||||
|
""" Search for all sale transactions """ |
||||
|
sp_query = """ SELECT sl.id from sale_order sl |
||||
|
LEFT JOIN res_partner p on p.id = sl.partner_id |
||||
|
LEFT JOIN product_pricelist pl ON pl.id = sl.pricelist_id |
||||
|
LEFT JOIN account_payment_term pt ON pt.id = sl.payment_term_id |
||||
|
WHERE sl.company_id = {op_id} AND (sl.name ILIKE '%{key}%' OR |
||||
|
p.name ILIKE '%{key}%' OR sl.state ILIKE '%{key}%' OR |
||||
|
pl.name::text ILIKE '%{key}%') |
||||
|
GROUP BY sl.id,p.name,pl.name,pt.name |
||||
|
""" |
||||
|
self._cr.execute( |
||||
|
sp_query.format(op_id=company_id, key=key, active=active_qry)) |
||||
|
sales = self._cr.dictfetchall() |
||||
|
sale_ids = self.env['sale.order'].browse([i['id'] for i in sales]) |
||||
|
self.sale_ids += sale_ids |
||||
|
|
||||
|
def _search_inventory_transactions(self, key, active_qry, company_id): |
||||
|
""" Search for all inventory transactions """ |
||||
|
sp_query = """ SELECT sp.id from stock_picking sp |
||||
|
LEFT JOIN res_partner p on p.id = sp.partner_id |
||||
|
LEFT JOIN stock_picking_type t ON t.id = sp.picking_type_id |
||||
|
WHERE sp.company_id = {op_id} AND (sp.name ILIKE '%{key}%' OR |
||||
|
p.name ILIKE '%{key}%' OR sp.state ILIKE '%{key}%' OR |
||||
|
t.name::text ILIKE '%{key}%') GROUP BY sp.id,p.name,t.name |
||||
|
""" |
||||
|
self._cr.execute( |
||||
|
sp_query.format(op_id=company_id, key=key, active=active_qry)) |
||||
|
transactions = self._cr.dictfetchall() |
||||
|
transaction_ids = self.env['stock.picking'].browse( |
||||
|
[i['id'] for i in transactions]) |
||||
|
self.transaction_ids += transaction_ids |
||||
|
|
||||
|
def _search_products(self, key, active_qry, company_id): |
||||
|
""" search for products """ |
||||
|
pt_query = """ SELECT pt.id FROM product_template pt |
||||
|
LEFT JOIN product_category pc ON pc.id = pt.categ_id |
||||
|
WHERE (pt.name::text ILIKE '%{key}%' OR |
||||
|
pt.default_code ILIKE '%{key}%' OR pt.type ILIKE '%{key}%' OR |
||||
|
pt.description::text ILIKE '%{key}%' OR pc.name ILIKE '%{key}%') |
||||
|
""" |
||||
|
self._cr.execute(pt_query.format(op_id=company_id, key=key, |
||||
|
active=active_qry).replace( |
||||
|
'obj', 'pt')) |
||||
|
template_ids = self._cr.dictfetchall() |
||||
|
product_template_ids = self.env['product.template'].browse( |
||||
|
[i['id'] for i in template_ids]) |
||||
|
self.product_ids += product_template_ids |
||||
|
|
||||
|
def _search_customer(self, key, active_qry): |
||||
|
""" search for customer """ |
||||
|
query = """ SELECT r.id from res_partner r WHERE |
||||
|
(r.parent_id is NULL ) AND r.type = 'contact' {active} AND |
||||
|
(r.name ILIKE '%{key}%' OR r.street ILIKE '%{key}%' OR |
||||
|
r.street2 ILIKE '%{key}%' OR r.city ILIKE '%{key}%' OR |
||||
|
r.zip ILIKE '%{key}%' OR r.email ILIKE '%{key}%') """ |
||||
|
query_params = query.format(key=key, active=active_qry).replace( |
||||
|
'obj', 'r') |
||||
|
self._cr.execute(query_params) |
||||
|
customers = self._cr.dictfetchall() |
||||
|
customer_ids = self.env['res.partner'].browse( |
||||
|
[i['id'] for i in customers]) |
||||
|
self.customer_ids += customer_ids |
|
@ -0,0 +1,18 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Record rule for search history --> |
||||
|
<record model="ir.rule" id="company_search_rule"> |
||||
|
<field name="name">Company Search Rule</field> |
||||
|
<field name="model_id" ref="model_master_search"/> |
||||
|
<field name="domain_force">[('user_id', '=', user.id)]</field> |
||||
|
</record> |
||||
|
<!-- Record rules for restrict search access --> |
||||
|
<record model="ir.module.category" id="module_master_search"> |
||||
|
<field name="name">Search Rights</field> |
||||
|
<field name="sequence">11</field> |
||||
|
</record> |
||||
|
<record id="master_search_read" model="res.groups"> |
||||
|
<field name="name">Global Search</field> |
||||
|
<field name="comment">Global Search Group</field> |
||||
|
</record> |
||||
|
</odoo> |
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 28 KiB |
After Width: | Height: | Size: 628 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 210 KiB |
After Width: | Height: | Size: 209 KiB |
After Width: | Height: | Size: 109 KiB |
After Width: | Height: | Size: 495 B |
After Width: | Height: | Size: 1.0 KiB |
After Width: | Height: | Size: 624 B |
After Width: | Height: | Size: 136 KiB |
After Width: | Height: | Size: 214 KiB |
After Width: | Height: | Size: 36 KiB |
After Width: | Height: | Size: 3.6 KiB |
After Width: | Height: | Size: 310 B |
After Width: | Height: | Size: 929 B |
After Width: | Height: | Size: 1.3 KiB |
After Width: | Height: | Size: 3.3 KiB |
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 17 KiB |
After Width: | Height: | Size: 542 B |
After Width: | Height: | Size: 576 B |
After Width: | Height: | Size: 733 B |
After Width: | Height: | Size: 4.3 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 4.0 KiB |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 738 KiB |
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 911 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 600 B |
After Width: | Height: | Size: 673 B |
After Width: | Height: | Size: 2.0 KiB |
After Width: | Height: | Size: 462 B |
After Width: | Height: | Size: 2.1 KiB |
After Width: | Height: | Size: 926 B |
After Width: | Height: | Size: 9.0 KiB |
After Width: | Height: | Size: 23 KiB |
After Width: | Height: | Size: 7.0 KiB |
After Width: | Height: | Size: 878 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 653 B |
After Width: | Height: | Size: 800 B |
After Width: | Height: | Size: 905 B |
After Width: | Height: | Size: 189 KiB |
After Width: | Height: | Size: 4.3 KiB |
After Width: | Height: | Size: 839 B |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 5.9 KiB |
After Width: | Height: | Size: 1.6 KiB |
After Width: | Height: | Size: 34 KiB |
After Width: | Height: | Size: 26 KiB |
After Width: | Height: | Size: 3.8 KiB |
After Width: | Height: | Size: 23 KiB |
After Width: | Height: | Size: 1.9 KiB |
After Width: | Height: | Size: 2.3 KiB |
After Width: | Height: | Size: 427 B |
After Width: | Height: | Size: 627 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 988 B |
After Width: | Height: | Size: 3.7 KiB |
After Width: | Height: | Size: 5.0 KiB |
After Width: | Height: | Size: 875 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 757 KiB |
After Width: | Height: | Size: 95 KiB |
After Width: | Height: | Size: 85 KiB |
After Width: | Height: | Size: 126 KiB |
After Width: | Height: | Size: 146 KiB |
After Width: | Height: | Size: 83 KiB |
After Width: | Height: | Size: 108 KiB |
After Width: | Height: | Size: 357 KiB |
After Width: | Height: | Size: 48 KiB |
After Width: | Height: | Size: 45 KiB |
After Width: | Height: | Size: 55 KiB |
After Width: | Height: | Size: 40 KiB |
After Width: | Height: | Size: 38 KiB |
After Width: | Height: | Size: 57 KiB |
After Width: | Height: | Size: 45 KiB |
After Width: | Height: | Size: 237 KiB |
After Width: | Height: | Size: 880 KiB |
After Width: | Height: | Size: 88 KiB |
After Width: | Height: | Size: 14 KiB |
@ -0,0 +1,119 @@ |
|||||
|
/** @odoo-module **/ |
||||
|
import { formView } from '@web/views/form/form_view'; |
||||
|
import { registry } from "@web/core/registry"; |
||||
|
import { FormRenderer } from "@web/views/form/form_renderer"; |
||||
|
import { FormController } from "@web/views/form/form_controller"; |
||||
|
import { useEffect, useExternalListener } from "@odoo/owl"; |
||||
|
export class MasterSearch extends FormController { |
||||
|
setup() { |
||||
|
super.setup(); |
||||
|
useEffect(() => { |
||||
|
const expandElements = document.querySelectorAll('.expand_tile') |
||||
|
const background = document.querySelector('.oe_search_bgnd') |
||||
|
const RadioBtns = document.querySelectorAll('.o_radio_input') |
||||
|
if (background.length > 0) { |
||||
|
document.title = 'Search'; |
||||
|
} |
||||
|
expandElements.forEach((element) => { |
||||
|
element.addEventListener("click", this.clickedExpandTile) |
||||
|
}) |
||||
|
RadioBtns.forEach((element) => { |
||||
|
element.addEventListener("change", this.onChangeRadioBtn) |
||||
|
}) |
||||
|
return () => { |
||||
|
expandElements.forEach((element) => { |
||||
|
element.removeEventListener("click", this.clickedExpandTile) |
||||
|
}) |
||||
|
RadioBtns.forEach((element) => { |
||||
|
element.removeEventListener("change", this.onChangeRadioBtn) |
||||
|
}) |
||||
|
} |
||||
|
}) |
||||
|
} |
||||
|
clickedExpandTile(e) { |
||||
|
var resultDiv = this.closest('.oe_result_div') |
||||
|
var element = resultDiv.querySelector('.oe_details_list') |
||||
|
if (element.style.display === "none" || element.style.display === "") { |
||||
|
console.log("block") |
||||
|
element.style.display = "block"; |
||||
|
} else { |
||||
|
console.log("none") |
||||
|
element.style.display = "none"; |
||||
|
} |
||||
|
} |
||||
|
onChangeRadioBtn(e) { |
||||
|
let targetEl = this.dataset.value; |
||||
|
const treeEls = document.querySelectorAll('.oe_details_list') |
||||
|
treeEls.forEach((element) => { |
||||
|
if (element.style.display === "none" || element.style.display === "") { |
||||
|
element.classList.remove('oe_details_list--show'); |
||||
|
} else { |
||||
|
element.classList.add('d-none'); |
||||
|
} |
||||
|
}); |
||||
|
const searchTabs = document.querySelectorAll('.oe_search_tab') |
||||
|
switch(targetEl){ |
||||
|
case 'customer': |
||||
|
searchTabs.forEach((el) => { |
||||
|
el.classList.add('d-none'); |
||||
|
}) |
||||
|
const customerResult = document.querySelector('#customer_search_results') |
||||
|
customerResult.classList.add('d-block'); |
||||
|
customerResult.classList.remove('d-none'); |
||||
|
break; |
||||
|
case 'product': |
||||
|
searchTabs.forEach((el) => { |
||||
|
el.classList.add('d-none'); |
||||
|
}) |
||||
|
const productResult = document.querySelector('#product_search_results') |
||||
|
productResult.classList.add('d-block'); |
||||
|
productResult.classList.remove('d-none'); |
||||
|
break; |
||||
|
case 'transaction details': |
||||
|
searchTabs.forEach((el) => { |
||||
|
el.classList.add('d-none'); |
||||
|
}) |
||||
|
const TransactionResult = document.querySelector('#inventory_search_results') |
||||
|
TransactionResult.classList.add('d-block'); |
||||
|
TransactionResult.classList.remove('d-none'); |
||||
|
break; |
||||
|
case 'sale details': |
||||
|
searchTabs.forEach((el) => { |
||||
|
el.classList.add('d-none'); |
||||
|
}) |
||||
|
const SaleResult = document.querySelector('#sale_search_results') |
||||
|
SaleResult.classList.add('d-block'); |
||||
|
SaleResult.classList.remove('d-none'); |
||||
|
break; |
||||
|
case 'purchase details': |
||||
|
searchTabs.forEach((el) => { |
||||
|
el.classList.add('d-none'); |
||||
|
}) |
||||
|
const PurchaseResult = document.querySelector('#purchase_search_results') |
||||
|
PurchaseResult.classList.add('d-block'); |
||||
|
PurchaseResult.classList.remove('d-none'); |
||||
|
break; |
||||
|
case 'account details': |
||||
|
searchTabs.forEach((el) => { |
||||
|
el.classList.add('d-none'); |
||||
|
}) |
||||
|
const AccountResult = document.querySelector('#accounting_search_results') |
||||
|
AccountResult.classList.add('d-block'); |
||||
|
AccountResult.classList.remove('d-none'); |
||||
|
break; |
||||
|
case 'any': |
||||
|
searchTabs.forEach((el) => { |
||||
|
el.classList.remove('d-none'); |
||||
|
}) |
||||
|
break; |
||||
|
} |
||||
|
const RecentSearch = document.querySelector('#recent_searches') |
||||
|
RecentSearch.classList.add('d-block'); |
||||
|
RecentSearch.classList.remove('d-none'); |
||||
|
} |
||||
|
} |
||||
|
export const MasterSearchClass = { |
||||
|
...formView, |
||||
|
Controller: MasterSearch, |
||||
|
}; |
||||
|
registry.category("views").add("master_search", MasterSearchClass); |
@ -0,0 +1,36 @@ |
|||||
|
.oe_search_tab{ |
||||
|
cursor: pointer; |
||||
|
transition: .2s; |
||||
|
&:hover { |
||||
|
background-color: #7c7bad !important; |
||||
|
} |
||||
|
} |
||||
|
.oe_details_list{ |
||||
|
display: none; |
||||
|
} |
||||
|
|
||||
|
.oe_details_tree--show{ |
||||
|
display: block !important; |
||||
|
} |
||||
|
|
||||
|
.btn_master_search{ |
||||
|
display: inline-block; |
||||
|
background-color:#5f5e97; |
||||
|
color:#fff !important; |
||||
|
color: #fff;margin-left: 1px; |
||||
|
margin-top: -1px; |
||||
|
} |
||||
|
.btn_master_search:hover{ |
||||
|
background-color: rgb(255, 255, 255) !important; |
||||
|
color: rgb(61, 155, 187) !important; |
||||
|
} |
||||
|
.not-allowed{ |
||||
|
color: #d94242; |
||||
|
border: 1px solid #ea9292; |
||||
|
border-radius: 5px; |
||||
|
background-color: #f9e7e7; |
||||
|
padding: 1px 55px; |
||||
|
width: 370px; |
||||
|
margin-bottom: 5px; |
||||
|
display: none; |
||||
|
} |
@ -0,0 +1,298 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- form view --> |
||||
|
<record model="ir.ui.view" id="master_search_form"> |
||||
|
<field name="name">Search</field> |
||||
|
<field name="model">master.search</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<form string="Search" js_class="master_search"> |
||||
|
<style> |
||||
|
.o_control_panel{display: none;} |
||||
|
</style> |
||||
|
<sheet> |
||||
|
<div class="row oe_search_bgnd" |
||||
|
style="padding: 24px 32px;background-color:#e5e4f9;border-radius: 5px;"> |
||||
|
<div class="col-12 col-md-8 col-xl-8"> |
||||
|
<label for="search_string" string="Search " |
||||
|
style="display: inline-block;" |
||||
|
class="oe_read_only"/> |
||||
|
<field name="search_string" default_focus="1" |
||||
|
class="search_input" |
||||
|
style="width: 80%; display: inline-block;margin-right: 0px;block-size: 32px;" |
||||
|
placeholder="Type here to search.."/> |
||||
|
<button name="action_clear_search" |
||||
|
type="object" title="Search" |
||||
|
class="fa fa-times oe_edit_only btn_master_search"/> |
||||
|
<button name="action_search" |
||||
|
type="object" |
||||
|
title="Search" |
||||
|
class="fa fa-search oe_search_btn btn_master_search oe_edit_only"/> |
||||
|
<div class="not-allowed">Regular expressions are |
||||
|
not allowed in search! |
||||
|
</div> |
||||
|
<div> |
||||
|
<label for="match_entire" |
||||
|
string="Match Phrase :" |
||||
|
style="display: inline-block;"/> |
||||
|
<field name="match_entire" |
||||
|
style="display: inline-block; margin-left: 10px;"/> |
||||
|
</div> |
||||
|
<div style="display: block;"> |
||||
|
<label for="search_by" string="Search For :" |
||||
|
class="oe_read_only"/> |
||||
|
<field name="search_by" widget="radio" |
||||
|
options="{'horizontal': true}"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="col-12 col-md-4 col-xl-4"> |
||||
|
<div style="display: block;"> |
||||
|
<label for="search_mode" string="Mode :" |
||||
|
class="oe_read_only"/> |
||||
|
<field name="search_mode" widget="radio" |
||||
|
nolabel="1" |
||||
|
options="{'horizontal': false}"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="oe_result_div"> |
||||
|
<div id="recent_searches" style="background-color: #5f5e97; color: #fff; width: 100%; padding: 5px 10px; |
||||
|
margin-top: 10px; border-top-right-radius: 5px;border-top-left-radius: 5px;" |
||||
|
class="oe_search_tab expand_tile"> |
||||
|
Recent Searches |
||||
|
<div class="history" style="float: right;"> |
||||
|
<field name="history_count" |
||||
|
class="oe_tab_count" readonly="1"/> |
||||
|
Records Found |
||||
|
<span class="oe_drop_down"> |
||||
|
<i style="color: #fff !important; font-size: 16px;" |
||||
|
title="Search" |
||||
|
class="fa fa-caret-down dropdown" |
||||
|
/> |
||||
|
</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div style="height: 200px; overflow-y: scroll;width: 100%;" |
||||
|
class="oe_details_list"> |
||||
|
<field name="master_search_ids"> |
||||
|
<list create="0" class="disable_open"> |
||||
|
<field name="search_string" |
||||
|
class="re_search oe_key"/> |
||||
|
<field name="search_mode" string="Mode" |
||||
|
class="re_search"/> |
||||
|
<field name="search_by" string="Type" |
||||
|
class="re_search"/> |
||||
|
<button style="display: inline-block;" |
||||
|
name="action_unlink_search" |
||||
|
title="Search" type="object" |
||||
|
class="fa fa-trash"/> |
||||
|
</list> |
||||
|
</field> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="oe_result_div"> |
||||
|
<div id="customer_search_results" |
||||
|
style="background-color: #5f5e97; color: #fff; width: 100%; padding: 5px 10px; |
||||
|
margin-top: 10px; border-top-right-radius: 5px;border-top-left-radius: 5px;" |
||||
|
class="oe_search_tab accordion_tab expand_tile"> |
||||
|
Customer Search Results |
||||
|
<div class="history" style="float: right;"> |
||||
|
<field name="customer_count" |
||||
|
class="oe_tab_count" readonly="1"/> |
||||
|
Records Found |
||||
|
<span class="oe_drop_down"> |
||||
|
<i style="color: #fff !important; font-size: 16px;" |
||||
|
title="Search" |
||||
|
class="fa fa-caret-down dropdown" |
||||
|
/> |
||||
|
</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div style="height: 200px; overflow-y: scroll;width: 100%;" |
||||
|
class="oe_details_list"> |
||||
|
<field name="customer_ids"> |
||||
|
<list create="0" delete="0"> |
||||
|
<field name="name" string="Name"/> |
||||
|
<field name="email" string="Email"/> |
||||
|
</list> |
||||
|
</field> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="oe_result_div"> |
||||
|
<div id="product_search_results" style="background-color: #5f5e97; color: #fff; width: 100%; padding: 5px 10px; |
||||
|
margin-top: 10px; border-top-right-radius: 5px;border-top-left-radius: 5px;" |
||||
|
class="oe_search_tab accordion_tab expand_tile"> |
||||
|
Product Search Results |
||||
|
<div class="history" style="float: right;"> |
||||
|
<field name="product_count" |
||||
|
class="oe_tab_count" readonly="1"/> |
||||
|
Records Found |
||||
|
<span class="oe_drop_down"> |
||||
|
<i style="color: #fff !important; font-size: 16px;" |
||||
|
title="Search" |
||||
|
class="fa fa-caret-down dropdown"/> |
||||
|
</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div style="height: 200px; overflow-y: scroll;width: 100%;" |
||||
|
class="oe_details_list" |
||||
|
name="products"> |
||||
|
<field name="product_ids"> |
||||
|
<list create="0" delete="0"> |
||||
|
<field name="name" string="Name"/> |
||||
|
<field name="default_code" |
||||
|
string="Code"/> |
||||
|
<field name="description" |
||||
|
string="Description"/> |
||||
|
<field name="type" |
||||
|
string="Product Type"/> |
||||
|
<field name="categ_id" |
||||
|
string="Category"/> |
||||
|
<field name="uom_id" string="Base UoM" |
||||
|
optional="hide"/> |
||||
|
<field name="qty_available" |
||||
|
string="Stock"/> |
||||
|
</list> |
||||
|
</field> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="oe_result_div"> |
||||
|
<div id="inventory_search_results" style="background-color: #5f5e97; color: #fff; width: 100%; padding: 5px 10px; |
||||
|
margin-top: 10px; border-top-right-radius: 5px;border-top-left-radius: 5px;" |
||||
|
class="oe_search_tab accordion_tab expand_tile"> |
||||
|
Inventory Search Results |
||||
|
<div class="history" style="float: right;"> |
||||
|
<field name="transaction_count" |
||||
|
class="oe_tab_count" readonly="1"/> |
||||
|
Records Found |
||||
|
<span class="oe_drop_down"> |
||||
|
<i style="color: #fff !important; font-size: 16px;" |
||||
|
title="Search" |
||||
|
class="fa fa-caret-down dropdown"/> |
||||
|
</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div style="height: 200px; overflow-y: scroll;width: 100%;" |
||||
|
class="oe_details_list"> |
||||
|
<field name="transaction_ids"> |
||||
|
<list create="0" delete="0"> |
||||
|
<field name="name" string="Trans. # "/> |
||||
|
<field name="date_done" |
||||
|
string="Transaction Date"/> |
||||
|
<field name="picking_type_id" |
||||
|
string="Transaction Type"/> |
||||
|
<field name="partner_id" |
||||
|
string="Reference Name"/> |
||||
|
<field name="state" |
||||
|
string="Transaction Status"/> |
||||
|
</list> |
||||
|
</field> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="oe_result_div"> |
||||
|
<div id="sale_search_results" style="background-color: #5f5e97; color: #fff; width: 100%; padding: 5px 10px; |
||||
|
margin-top: 10px; border-top-right-radius: 5px;border-top-left-radius: 5px;" |
||||
|
class="oe_search_tab accordion_tab expand_tile"> |
||||
|
Sale Search Results |
||||
|
<div class="history" style="float: right;"> |
||||
|
<field name="sale_count" |
||||
|
class="oe_tab_count" readonly="1"/> |
||||
|
Records Found |
||||
|
<span class="oe_drop_down"> |
||||
|
<i style="color: #fff !important; font-size: 16px;" |
||||
|
title="Search" |
||||
|
class="fa fa-caret-down dropdown"/> |
||||
|
</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div style="height: 200px; overflow-y: scroll;width: 100%;" |
||||
|
class="oe_details_list"> |
||||
|
<field name="sale_ids"> |
||||
|
<list create="0" delete="0"> |
||||
|
<field name="name" string="Trans. # "/> |
||||
|
<field name="partner_id" |
||||
|
string="Customer"/> |
||||
|
<field name="pricelist_id" |
||||
|
string="Pricelist"/> |
||||
|
<field name="payment_term_id" |
||||
|
string="Payment Term"/> |
||||
|
<field name="state" string="Status"/> |
||||
|
</list> |
||||
|
</field> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="oe_result_div"> |
||||
|
<div id="purchase_search_results" style="background-color: #5f5e97; color: #fff; width: 100%; padding: 5px 10px; |
||||
|
margin-top: 10px; border-top-right-radius: 5px;border-top-left-radius: 5px;" |
||||
|
class="oe_search_tab accordion_tab expand_tile"> |
||||
|
Purchase Search Results |
||||
|
<div class="history" style="float: right;"> |
||||
|
<field name="purchase_count" |
||||
|
class="oe_tab_count" readonly="1"/> |
||||
|
Records Found |
||||
|
<span class="oe_drop_down"> |
||||
|
<i style="color: #fff !important; font-size: 16px;" |
||||
|
title="Search" |
||||
|
class="fa fa-caret-down dropdown"/> |
||||
|
</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div style="height: 200px; overflow-y: scroll;width: 100%;" |
||||
|
class="oe_details_list"> |
||||
|
<field name="purchase_ids"> |
||||
|
<list create="0" delete="0"> |
||||
|
<field name="name" string="Trans. # "/> |
||||
|
<field name="partner_id" |
||||
|
string="Customer"/> |
||||
|
<field name="state" string="Status"/> |
||||
|
</list> |
||||
|
</field> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="oe_result_div"> |
||||
|
<div id="accounting_search_results" style="background-color: #5f5e97; color: #fff; width: 100%; padding: 5px 10px; |
||||
|
margin-top: 10px; border-top-right-radius: 5px;border-top-left-radius: 5px;" |
||||
|
class="oe_search_tab accordion_tab expand_tile"> |
||||
|
Accounting Search Results |
||||
|
<div class="history" style="float: right;"> |
||||
|
<field name="account_count" |
||||
|
class="oe_tab_count" readonly="1"/> |
||||
|
Records Found |
||||
|
<span class="oe_drop_down"> |
||||
|
<i style="color: #fff !important; font-size: 16px;" |
||||
|
title="Search" |
||||
|
class="fa fa-caret-down dropdown"/> |
||||
|
</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div style="height: 200px; overflow-y: scroll;width: 100%;" |
||||
|
class="oe_details_list"> |
||||
|
<field name="account_ids"> |
||||
|
<list create="false" delete="false" |
||||
|
edit="false"> |
||||
|
<field name="name" string="Trans. # "/> |
||||
|
<field name="partner_id" |
||||
|
string="Customer"/> |
||||
|
<field name="state" string="Status"/> |
||||
|
</list> |
||||
|
</field> |
||||
|
</div> |
||||
|
</div> |
||||
|
</sheet> |
||||
|
</form> |
||||
|
</field> |
||||
|
</record> |
||||
|
<!-- actions opening views on models --> |
||||
|
<record model="ir.actions.act_window" id="master_search_action"> |
||||
|
<field name="name">Search</field> |
||||
|
<field name="res_model">master.search</field> |
||||
|
<field name="view_mode">form</field> |
||||
|
<field name="context">{'active_test': False, |
||||
|
'search_default_filter_active': 1} |
||||
|
</field> |
||||
|
</record> |
||||
|
<!-- Search root menu --> |
||||
|
<menuitem id="master_search_root" name="Search" |
||||
|
action="master_search_action" |
||||
|
web_icon="master_search,static/description/icon.png" |
||||
|
sequence="2" groups="master_search.master_search_read"/> |
||||
|
</odoo> |