@ -0,0 +1,47 @@ |
|||
.. image:: https://img.shields.io/badge/license-LGPL--3-green.svg |
|||
:target: https://www.gnu.org/licenses/lgpl-3.0-standalone.html |
|||
:alt: License: LGPL-3 |
|||
|
|||
Inventory Dashboard Odoo 18 |
|||
=========================== |
|||
Dashboard for Inventory module. |
|||
|
|||
Configuration |
|||
============= |
|||
No additional configurations needed. |
|||
|
|||
Company |
|||
------- |
|||
* `Cybrosys Techno Solutions <https://cybrosys.com/>`__ |
|||
|
|||
License |
|||
------- |
|||
Lesser General Public License, Version 3 (LGPL v3) |
|||
(https://www.gnu.org/licenses/lgpl-3.0-standalone.html) |
|||
|
|||
Credits |
|||
------- |
|||
* Developer : (V18) Renu |
|||
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) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################# |
|||
from . import models |
@ -0,0 +1,53 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################### |
|||
# |
|||
# Cybrosys Technologies Pvt. Ltd. |
|||
# |
|||
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################### |
|||
{ |
|||
'name': 'Inventory Dashboard Odoo 18', |
|||
'version': '18.0.1.0.0', |
|||
'category': 'Warehouse', |
|||
'summary': 'Detailed dashboard view for Inventory module.', |
|||
'description': """This module presents a detailed dashboard view for the |
|||
Inventory module, delivering a compr ehensive and concise overview that |
|||
serves as a valuable tool for both inventory users and administrators.""", |
|||
'author': 'Cybrosys Techno Solutions', |
|||
'company': 'Cybrosys Techno Solutions', |
|||
'maintainer': 'Cybrosys Techno Solutions', |
|||
'website': "https://www.cybrosys.com", |
|||
'depends': ['stock', 'base'], |
|||
'data': [ |
|||
'views/res_config_settings_views.xml', |
|||
'views/dashboard_menu.xml', |
|||
], |
|||
'assets': { |
|||
'web.assets_backend': [ |
|||
'https://fonts.googleapis.com/css2?family=Poppins:wght@400;600&display=swap', |
|||
'inventory_stock_dashboard_odoo/static/src/css/dashboard.css', |
|||
'inventory_stock_dashboard_odoo/static/src/js/dashboard.js', |
|||
'inventory_stock_dashboard_odoo/static/src/xml/inventory_dashboard_template.xml', |
|||
'https://cdn.jsdelivr.net/npm/chart.js', |
|||
], |
|||
}, |
|||
'images': ['static/description/banner.png'], |
|||
'license': 'LGPL-3', |
|||
'installable': True, |
|||
'auto_install': False, |
|||
'application': False, |
|||
} |
@ -0,0 +1,6 @@ |
|||
## Module <inventory_stock_dashboard_odoo> |
|||
|
|||
#### 29.01.2025 |
|||
#### Version 18.0.1.0.0 |
|||
##### ADD |
|||
- Initial commit for Inventory Dashboard Odoo 18 |
@ -0,0 +1,26 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################### |
|||
# |
|||
# Cybrosys Technologies Pvt. Ltd. |
|||
# |
|||
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################### |
|||
from . import res_config_settings |
|||
from . import stock_move |
|||
from . import stock_move_line |
|||
from . import stock_picking |
|||
from . import stock_quant |
@ -0,0 +1,56 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################### |
|||
# |
|||
# Cybrosys Technologies Pvt. Ltd. |
|||
# |
|||
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################### |
|||
from odoo import fields, models |
|||
|
|||
|
|||
class ResConfigSettings(models.TransientModel): |
|||
""" Inheriting res.config.settings model for adding custom fields in |
|||
settings.""" |
|||
_inherit = "res.config.settings" |
|||
|
|||
out_of_stock = fields.Boolean( |
|||
string='Out of stock', |
|||
config_parameter='inventory_stock_dashboard_odoo.out_of_stock', |
|||
help="Enable if out of stock") |
|||
out_of_stock_quantity = fields.Integer( |
|||
string='Quantity', |
|||
config_parameter='inventory_stock_dashboard_odoo.out_of_stock_quantity', |
|||
required=True, |
|||
help='Set the minimum quantity for considering a product as out of' |
|||
'stock.') |
|||
dead_stock_bol = fields.Boolean( |
|||
string='Enable dead stock', |
|||
config_parameter='inventory_stock_dashboard_odoo.dead_stock_bol', |
|||
help='Enable if you want to consider dead stock.') |
|||
dead_stock = fields.Integer( |
|||
string='Dead stock', |
|||
config_parameter='inventory_stock_dashboard_odoo.dead_stock', |
|||
required=True, |
|||
help='Set the threshold quantity for considering a product as dead ' |
|||
'stock.') |
|||
dead_stock_type = fields.Selection( |
|||
[('day', 'Day'), ('week', 'Week'), ('month', 'Month')], |
|||
string='Type', default='day', |
|||
config_parameter='inventory_stock_dashboard_odoo.dead_stock_type', |
|||
required=True, |
|||
help='Select the time period to determine dead stock based on product' |
|||
'sales.') |
@ -0,0 +1,277 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################### |
|||
# |
|||
# Cybrosys Technologies Pvt. Ltd. |
|||
# |
|||
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################### |
|||
from odoo import api, models |
|||
|
|||
|
|||
class StockMove(models.Model): |
|||
""" Extends 'stock.move' and provides methods for retrieving data for |
|||
dashboard.""" |
|||
_inherit = "stock.move" |
|||
|
|||
@api.model |
|||
def get_the_top_products(self): |
|||
""" Returns top ten products and done quantity.""" |
|||
company_id = self.env.company.id |
|||
query = '''select product_template.name,sum(product_uom_qty) from stock_move |
|||
inner join stock_picking on stock_move.picking_id = stock_picking.id |
|||
inner join stock_picking_type on stock_picking.picking_type_id = stock_picking_type.id |
|||
inner join product_product on stock_move.product_id = product_product.id |
|||
inner join product_template on product_template.id = product_product.product_tmpl_id |
|||
where stock_move.state = 'done' and stock_move.company_id=%s and stock_picking_type.code = 'outgoing' and |
|||
stock_move.create_date between (now() - interval '10 day') and now() |
|||
group by product_template.name ORDER BY sum DESC''' % company_id |
|||
self._cr.execute(query) |
|||
top_product = self._cr.dictfetchall() |
|||
total_quantity = [] |
|||
product_name = [] |
|||
for record in top_product[:10]: |
|||
total_quantity.append(record.get('sum')) |
|||
product_name.append(record.get('name')['en_US']) |
|||
value = {'products': product_name, 'count': total_quantity} |
|||
return value |
|||
|
|||
@api.model |
|||
def top_products_last_ten(self): |
|||
""" Returns top ten products and done quantity for last 10 days.""" |
|||
company_id = self.env.company.id |
|||
query = '''select product_template.name,sum(product_uom_qty) from stock_move |
|||
inner join stock_picking on stock_move.picking_id = stock_picking.id |
|||
inner join stock_picking_type on stock_picking.picking_type_id = stock_picking_type.id |
|||
inner join product_product on stock_move.product_id = product_product.id |
|||
inner join product_template on product_template.id = product_product.product_tmpl_id |
|||
where stock_move.state = 'done' and stock_move.company_id=%s and stock_picking_type.code = 'outgoing' and |
|||
stock_move.create_date between (now() - interval '10 day') and now() |
|||
group by product_template.name ORDER BY sum DESC''' % company_id |
|||
self._cr.execute(query) |
|||
top_product = self._cr.dictfetchall() |
|||
total_quantity = [] |
|||
product_name = [] |
|||
for record in top_product[:10]: |
|||
total_quantity.append(record.get('sum')) |
|||
product_name.append(record.get('name')['en_US']) |
|||
value = {'products': product_name, 'count': total_quantity} |
|||
return value |
|||
|
|||
@api.model |
|||
def top_products_last_thirty(self): |
|||
""" Returns top ten products and done quantity for last 30 days.""" |
|||
company_id = self.env.company.id |
|||
query = '''select product_template.name,sum(product_uom_qty) from stock_move |
|||
inner join stock_picking on stock_move.picking_id = stock_picking.id |
|||
inner join stock_picking_type on stock_picking.picking_type_id = stock_picking_type.id |
|||
inner join product_product on stock_move.product_id = product_product.id |
|||
inner join product_template on product_template.id = product_product.product_tmpl_id |
|||
where stock_move.state = 'done' and stock_move.company_id=%s and stock_picking_type.code = 'outgoing' |
|||
and stock_move.create_date between (now() - interval '30 day') and now() |
|||
group by product_template.name ORDER BY sum DESC''' % company_id |
|||
self._cr.execute(query) |
|||
top_product = self._cr.dictfetchall() |
|||
total_quantity = [] |
|||
product_name = [] |
|||
for record in top_product[:10]: |
|||
total_quantity.append(record.get('sum')) |
|||
product_name.append(record.get('name')['en_US']) |
|||
value = {'products': product_name, 'count': total_quantity} |
|||
return value |
|||
|
|||
@api.model |
|||
def top_products_last_three_months(self): |
|||
""" Returns top ten products and done quantity for last 3 months.""" |
|||
company_id = self.env.company.id |
|||
query = '''select product_template.name,sum(product_uom_qty) from stock_move |
|||
inner join stock_picking on stock_move.picking_id = stock_picking.id |
|||
inner join stock_picking_type on stock_picking.picking_type_id = stock_picking_type.id |
|||
inner join product_product on stock_move.product_id = product_product.id |
|||
inner join product_template on product_template.id = product_product.product_tmpl_id |
|||
where stock_move.state = 'done' and stock_move.company_id=%s and stock_picking_type.code ='outgoing' |
|||
and stock_move.create_date between (now() - interval '3 month') and now() |
|||
group by product_template.name ORDER BY sum DESC''' % company_id |
|||
self._cr.execute(query) |
|||
top_product = self._cr.dictfetchall() |
|||
total_quantity = [] |
|||
product_name = [] |
|||
for record in top_product[:10]: |
|||
total_quantity.append(record.get('sum')) |
|||
product_name.append(record.get('name')['en_US']) |
|||
value = {'products': product_name, 'count': total_quantity} |
|||
return value |
|||
|
|||
@api.model |
|||
def top_products_last_year(self): |
|||
""" Returns top ten products and done quantity for last year.""" |
|||
company_id = self.env.company.id |
|||
query = '''select product_template.name,sum(product_uom_qty) from stock_move |
|||
inner join stock_picking on stock_move.picking_id = stock_picking.id |
|||
inner join stock_picking_type on stock_picking.picking_type_id = stock_picking_type.id |
|||
inner join product_product on stock_move.product_id = product_product.id |
|||
inner join product_template on product_template.id = product_product.product_tmpl_id |
|||
where stock_move.state = 'done' and stock_move.company_id=%s and |
|||
stock_picking_type.code = 'outgoing' and |
|||
stock_move.create_date between (now() - interval '1 year') and now() |
|||
group by product_template.name ORDER BY sum DESC''' % company_id |
|||
self._cr.execute(query) |
|||
top_product = self._cr.dictfetchall() |
|||
total_quantity = [] |
|||
product_name = [] |
|||
for record in top_product[:10]: |
|||
total_quantity.append(record.get('sum')) |
|||
product_name.append(record.get('name')['en_US']) |
|||
value = {'products': product_name, 'count': total_quantity} |
|||
return value |
|||
|
|||
@api.model |
|||
def get_stock_moves(self): |
|||
""" Returns location name and quantity_done of stock moves graph""" |
|||
company_id = self.env.company.id |
|||
query = ('''select stock_location.complete_name, count(stock_move.id) from stock_move |
|||
inner join stock_location on stock_move.location_id = stock_location.id where stock_move.state = 'done' |
|||
and stock_move.company_id = %s group by stock_location.complete_name''' % company_id) |
|||
self._cr.execute(query) |
|||
stock_move = self._cr.dictfetchall() |
|||
count = [] |
|||
complete_name = [] |
|||
for record in stock_move: |
|||
count.append(record.get('count')) |
|||
complete_name.append(record.get('complete_name')) |
|||
value = {'name': complete_name, 'count': count} |
|||
return value |
|||
|
|||
@api.model |
|||
def stock_move_last_ten_days(self, post): |
|||
""" Returns location name and quantity_done of stock moves graph last |
|||
ten days.""" |
|||
company_id = self.env.company.id |
|||
query = ('''select stock_location.name,sum(stock_move_line.quantity) from stock_move_line |
|||
inner join stock_location on stock_move_line.location_id = stock_location.id |
|||
where stock_move_line.state = 'done' and stock_move_line.company_id = %s |
|||
and stock_move_line.create_date between (now() - interval '10 day') and now() |
|||
group by stock_location.name''' % company_id) |
|||
self._cr.execute(query) |
|||
location_quantity = self._cr.dictfetchall() |
|||
quantity_done = [] |
|||
name = [] |
|||
for record in location_quantity: |
|||
quantity_done.append(record.get('sum')) |
|||
name.append(record.get('name')) |
|||
value = {'name': name, 'count': quantity_done} |
|||
return value |
|||
|
|||
@api.model |
|||
def this_month(self, post): |
|||
""" Returns location name and quantity_done of stock moves graph for |
|||
current month.""" |
|||
company_id = self.env.company.id |
|||
query = ('''select stock_location.name,sum(stock_move_line.quantity) from stock_move_line |
|||
inner join stock_location on stock_move_line.location_id = stock_location.id |
|||
where stock_move_line.state = 'done' and stock_move_line.company_id = %s |
|||
and stock_move_line.create_date between (now() - interval '1 months') and now() |
|||
group by stock_location.name''' % company_id) |
|||
self._cr.execute(query) |
|||
location_quantity = self._cr.dictfetchall() |
|||
quantity_done = [] |
|||
name = [] |
|||
for record in location_quantity: |
|||
quantity_done.append(record.get('sum')) |
|||
name.append(record.get('name')) |
|||
value = {'name': name, 'count': quantity_done} |
|||
return value |
|||
|
|||
@api.model |
|||
def last_three_month(self, post): |
|||
""" Returns location name and quantity_done of stock moves graph for |
|||
last three months.""" |
|||
company_id = self.env.company.id |
|||
query = ('''select stock_location.name,sum(stock_move_line.quantity) from stock_move_line |
|||
inner join stock_location on stock_move_line.location_id = stock_location.id |
|||
where stock_move_line.state = 'done' and stock_move_line.company_id = %s |
|||
and stock_move_line.create_date between (now() - interval '3 months') and now() |
|||
group by stock_location.name''' % company_id) |
|||
self._cr.execute(query) |
|||
location_quantity = self._cr.dictfetchall() |
|||
quantity_done = [] |
|||
name = [] |
|||
for record in location_quantity: |
|||
quantity_done.append(record.get('sum')) |
|||
name.append(record.get('name')) |
|||
value = {'name': name, 'count': quantity_done} |
|||
return value |
|||
|
|||
@api.model |
|||
def last_year(self, post): |
|||
""" Returns location name and quantity_done of stock moves graph for |
|||
last year.""" |
|||
company_id = self.env.company.id |
|||
query = ('''select stock_location.name,sum(stock_move_line.quantity) from stock_move_line |
|||
inner join stock_location on stock_move_line.location_id = stock_location.id |
|||
where stock_move_line.state = 'done' and stock_move_line.company_id = %s |
|||
and stock_move_line.create_date between (now() - interval '12 months') and now() |
|||
group by stock_location.name''' % company_id) |
|||
self._cr.execute(query) |
|||
location_quantity = self._cr.dictfetchall() |
|||
quantity_done = [] |
|||
name = [] |
|||
for record in location_quantity: |
|||
quantity_done.append(record.get('sum')) |
|||
name.append(record.get('name')) |
|||
value = {'name': name, 'count': quantity_done} |
|||
return value |
|||
|
|||
@api.model |
|||
def get_dead_of_stock(self): |
|||
""" Returns product name and dead quantity of dead stock graph.""" |
|||
company_id = self.env.company.id |
|||
sett_dead_stock_bool = self.env['ir.config_parameter'].sudo(). \ |
|||
get_param("inventory_stock_dashboard_odoo.dead_stock_bol", default="") |
|||
sett_dead_stock_quantity = self.env['ir.config_parameter'].sudo().get_param( |
|||
"inventory_stock_dashboard_odoo.dead_stock", |
|||
default="") |
|||
sett_dead_stock_type = self.env['ir.config_parameter'].sudo().get_param( |
|||
"inventory_stock_dashboard_odoo.dead_stock_type", |
|||
default="") |
|||
if sett_dead_stock_bool == "True": |
|||
if sett_dead_stock_quantity: |
|||
out_stock_value = int(sett_dead_stock_quantity) |
|||
query = '''select product_product.id,stock_quant.quantity from product_product |
|||
inner join stock_quant on product_product.id = stock_quant.product_id |
|||
where stock_quant.company_id = %s and product_product.create_date not between (now() - interval '%s %s') |
|||
and now() and product_product.id NOT IN (select product_id from stock_move |
|||
inner join stock_picking on stock_move.picking_id = stock_picking.id |
|||
inner join stock_picking_type on stock_picking.picking_type_id = stock_picking_type.id |
|||
where stock_move.company_id = %s and stock_picking_type.code = 'outgoing' and |
|||
stock_move.state = 'done' and stock_move.create_date between (now() - interval '%s %s') and now() |
|||
group by product_id)''' % \ |
|||
(company_id, out_stock_value, sett_dead_stock_type, company_id, out_stock_value, |
|||
sett_dead_stock_type) |
|||
self._cr.execute(query) |
|||
result = self._cr.fetchall() |
|||
total_quantity = [] |
|||
product_name = [] |
|||
for record in result: |
|||
if record[1] > 0: |
|||
complete_name = self.env['product.product'].browse(record[0]).display_name |
|||
product_name.append(complete_name) |
|||
total_quantity.append(record[1]) |
|||
value = { |
|||
'product_name': product_name, |
|||
'total_quantity': total_quantity |
|||
} |
|||
return value |
@ -0,0 +1,86 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################### |
|||
# |
|||
# Cybrosys Technologies Pvt. Ltd. |
|||
# |
|||
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################### |
|||
from odoo import api, models |
|||
|
|||
|
|||
class StockMoveLine(models.Model): |
|||
""" Extends 'stock.move.line' and provides methods for retrieving data for |
|||
dashboard.""" |
|||
_inherit = "stock.move.line" |
|||
|
|||
@api.model |
|||
def get_product_moves(self): |
|||
""" Returns product move and quantity_done.""" |
|||
company_id = self.env.company.id |
|||
query = ('''select product_template.name,sum(stock_move_line.quantity) from stock_move_line |
|||
inner join product_product on stock_move_line.product_id = product_product.id |
|||
inner join product_template on product_product.product_tmpl_id = product_template.id |
|||
where stock_move_line.company_id = %s group by product_template.name''' % company_id) |
|||
self._cr.execute(query) |
|||
products_quantity = self._cr.dictfetchall() |
|||
quantity_done = [] |
|||
name = [] |
|||
for record in products_quantity: |
|||
quantity_done.append(record.get('sum')) |
|||
name.append(record.get('name')) |
|||
value = {'name': name, 'count': quantity_done} |
|||
category_query = '''select product_category.id,product_category.name from stock_move_line |
|||
inner join product_product on stock_move_line.product_id = product_product.id |
|||
inner join product_template on product_product.product_tmpl_id = product_template.id |
|||
inner join product_category on product_template.categ_id = product_category.id |
|||
where stock_move_line.company_id = %s and stock_move_line.state = 'done' |
|||
group by product_category.id''' % company_id |
|||
self._cr.execute(category_query) |
|||
category = self._cr.dictfetchall() |
|||
category_id = [] |
|||
category_name = [] |
|||
for record in category: |
|||
category_id.append(record.get('id')) |
|||
category_name.append(record.get('name')) |
|||
value1 = {'category_id': category_id, 'category_name': category_name} |
|||
return value, value1 |
|||
|
|||
@api.model |
|||
def product_move_by_category(self, option): |
|||
""" Returns category name and quantity_done.""" |
|||
if option is None: |
|||
return {} |
|||
category_id = int(option) |
|||
company_id = self.env.company.id |
|||
query = ('''select product_template.name,sum(stock_move_line.quantity) from stock_move_line |
|||
inner join product_product on stock_move_line.product_id = product_product.id |
|||
inner join product_template on product_product.product_tmpl_id = product_template.id |
|||
inner join product_category on product_template.categ_id = product_category.id |
|||
where stock_move_line.company_id = %s and product_category.id = %s group by product_template.name''' % |
|||
(company_id, category_id)) |
|||
self._cr.execute(query) |
|||
product_move = self._cr.dictfetchall() |
|||
quantity_done = [] |
|||
name = [] |
|||
for record in product_move: |
|||
quantity_done.append(record.get('sum')) |
|||
name.append(record.get('name')['en_US']) |
|||
value = { |
|||
'name': name, |
|||
'count': quantity_done, |
|||
} |
|||
return value |
@ -0,0 +1,101 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################### |
|||
# |
|||
# Cybrosys Technologies Pvt. Ltd. |
|||
# |
|||
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################### |
|||
from odoo import api, models |
|||
|
|||
|
|||
class StockPicking(models.Model): |
|||
""" Extends 'stock.picking' for retrieving stock picking data for |
|||
dashboard.""" |
|||
_inherit = "stock.picking" |
|||
|
|||
@api.model |
|||
def get_operation_types(self): |
|||
""" Method for retrieving operation type tiles, operation type graph and |
|||
returns operation type details. |
|||
no_transfer - count of transfers for each operation type, |
|||
late - count of late occurrences for each operation type, |
|||
waiting - count of waiting operations for each operation type, |
|||
operation_type_name - retrieve all operation type names, |
|||
backorder - count the number of backorders for each operation type. |
|||
""" |
|||
no_transfer = {} |
|||
stock_picking_type = self.env['stock.picking.type'].search([]) |
|||
stock_picking = self.env['stock.picking'].search([]) |
|||
stock = [] |
|||
length = [] |
|||
names = [] |
|||
late = {} |
|||
query = '''select stock_picking.picking_type_id, count(stock_picking.picking_type_id) from stock_picking |
|||
inner join stock_picking_type on stock_picking.picking_type_id = stock_picking_type.id |
|||
where stock_picking.company_id = %s and |
|||
stock_picking.state in ('assigned', 'waiting', 'confirmed') and (has_deadline_issue = true or |
|||
date_deadline <= now() or scheduled_date <= now()) |
|||
group by stock_picking.picking_type_id''' % self.env.company.id |
|||
self._cr.execute(query) |
|||
lates = self._cr.dictfetchall() |
|||
for rec in lates: |
|||
late.update({rec.get('picking_type_id'): rec.get('count')}) |
|||
waiting = {} |
|||
backorder = {} |
|||
operation_type_name = {} |
|||
for i in stock_picking_type: |
|||
names.append(i.name) |
|||
orders = stock_picking.filtered(lambda r: r.picking_type_id.id == i.id) |
|||
stock.append(len(orders)) |
|||
length_stock_picking = len(orders) |
|||
length.append(len(stock_picking.filtered(lambda r: r.picking_type_id.id == i.id))) |
|||
no_transfer.update({i.id: length_stock_picking}) |
|||
operation_type_name.update({i.id: i.name}) |
|||
if len(orders) > 0: |
|||
if len(orders.filtered(lambda r: r.state == 'confirmed')) > 0: |
|||
waiting.update({i.id: len(orders.filtered(lambda r: r.state == 'confirmed'))}) |
|||
if len(orders.mapped('backorder_id')) > 0: |
|||
backorder.update({i.id: len(orders.mapped('backorder_id'))}) |
|||
return no_transfer, late, waiting, operation_type_name, backorder |
|||
|
|||
@api.model |
|||
def get_product_category(self): |
|||
""" Return product categories and categories that have on-hand product |
|||
quantities.""" |
|||
category_ids = self.env['product.category'].search([]) |
|||
category_name = [] |
|||
product_count = [] |
|||
for rec in category_ids: |
|||
name = rec.name |
|||
category_name.append(name) |
|||
count = rec.product_count |
|||
product_count.append(count) |
|||
value = {'name': category_name, 'count': product_count} |
|||
return value |
|||
|
|||
@api.model |
|||
def get_locations(self): |
|||
""" Returns locations and locations that have on-hand product |
|||
quantities.""" |
|||
stock_quant_ids = self.env['stock.quant'].search([]) |
|||
locations = stock_quant_ids.mapped('location_id') |
|||
value = {} |
|||
for rec in locations: |
|||
loc_stock_quant = stock_quant_ids.filtered(lambda x: x.location_id == rec) |
|||
on_hand_quantity = sum(loc_stock_quant.mapped('inventory_quantity_auto_apply')) |
|||
value[rec.name] = on_hand_quantity |
|||
return value |
@ -0,0 +1,59 @@ |
|||
# -*- coding: utf-8 -*- |
|||
############################################################################### |
|||
# |
|||
# Cybrosys Technologies Pvt. Ltd. |
|||
# |
|||
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
|||
# Author: Aysha Shalin (odoo@cybrosys.com) |
|||
# |
|||
# You can modify it under the terms of the GNU LESSER |
|||
# GENERAL PUBLIC LICENSE (LGPL 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 LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. |
|||
# |
|||
# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE |
|||
# (LGPL v3) along with this program. |
|||
# If not, see <http://www.gnu.org/licenses/>. |
|||
# |
|||
############################################################################### |
|||
from odoo import api, models |
|||
|
|||
|
|||
class StockQuant(models.Model): |
|||
""" Extends 'stock.quant' for retrieving data of products that are out of |
|||
stock.""" |
|||
_inherit = "stock.quant" |
|||
|
|||
@api.model |
|||
def get_out_of_stock(self): |
|||
""" Returns products and quantities that are out of stock.""" |
|||
company_id = self.env.company.id |
|||
sett_out_stock_bool = self.env['ir.config_parameter'].sudo(). \ |
|||
get_param("inventory_stock_dashboard_odoo.out_of_stock", default="") |
|||
sett_out_stock_quantity = self.env['ir.config_parameter'].sudo().\ |
|||
get_param("inventory_stock_dashboard_odoo.out_of_stock_quantity", default="") |
|||
if sett_out_stock_bool == "True": |
|||
if sett_out_stock_quantity: |
|||
out_stock_value = int(sett_out_stock_quantity) |
|||
query = '''select product_product.id,sum(stock_quant.quantity) from product_product |
|||
inner join stock_quant on product_product.id = stock_quant.product_id |
|||
where stock_quant.quantity < %s and stock_quant.company_id = %s group by product_product.id''' \ |
|||
% (out_stock_value, company_id) |
|||
self._cr.execute(query) |
|||
result = self._cr.fetchall() |
|||
total_quantity = [] |
|||
product_name = [] |
|||
for record in result: |
|||
total_quantity.append(record[1]) |
|||
for record in result: |
|||
complete_name = self.env['product.product'].browse( |
|||
record[0]).display_name |
|||
product_name.append(complete_name) |
|||
value = { |
|||
'product_name': product_name, |
|||
'total_quantity': total_quantity |
|||
} |
|||
return value |
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: 767 KiB |
After Width: | Height: | Size: 138 KiB |
After Width: | Height: | Size: 760 KiB |
After Width: | Height: | Size: 92 KiB |
After Width: | Height: | Size: 697 KiB |
After Width: | Height: | Size: 1.1 MiB |
After Width: | Height: | Size: 130 KiB |
After Width: | Height: | Size: 124 KiB |
After Width: | Height: | Size: 162 KiB |
After Width: | Height: | Size: 122 KiB |
After Width: | Height: | Size: 170 KiB |
After Width: | Height: | Size: 25 KiB |
After Width: | Height: | Size: 136 KiB |
After Width: | Height: | Size: 108 KiB |
After Width: | Height: | Size: 51 KiB |
After Width: | Height: | Size: 45 KiB |
After Width: | Height: | Size: 62 KiB |
After Width: | Height: | Size: 110 KiB |
After Width: | Height: | Size: 45 KiB |
After Width: | Height: | Size: 23 KiB |
After Width: | Height: | Size: 398 KiB |
After Width: | Height: | Size: 880 KiB |