@ -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> |