first commit

This commit is contained in:
Suherdy Yacob 2026-03-03 18:09:01 +07:00
commit cf8bb6ced1
13 changed files with 383 additions and 0 deletions

24
.gitignore vendored Normal file
View File

@ -0,0 +1,24 @@
# Python
*.py[cod]
*$py.class
__pycache__/
# Odoo
*.log
*.pot
# OS generated files
.DS_Store
.DS_Store?
._*
.Spotlight-V100
.Trashes
ehthumbs.db
Thumbs.db
# IDEs
.idea/
.vscode/
*.swp
*.swo
*~

17
README.md Normal file
View File

@ -0,0 +1,17 @@
# MRP Components Report
## Overview
This custom Odoo 19 module adds a new "Components Required" report under the **Manufacturing > Reporting** menu. It allows users to view a consolidated list of components needed for Manufacturing Orders on a specific date (or within a selected date range).
## Key Features
- **Wizard Entry Point**: Allows users to filter required components by a custom date range.
- **Reporting Views**: Provides robust List and Pivot views for analyzing component demand data.
- **Multi-Level Grouping**: Components are pre-grouped by **Work Center** and then by **Parent Product** for clear visibility into production station needs.
- **Excel Export**: Native Odoo list views support immediate export to `.xlsx`.
- **Custom PDF Generation**: Includes a dedicated "Print PDF" action within the wizard for a neatly formatted printable report.
## Technical Details
This module utilizes a PostgreSQL `VIEW` (`mrp.components.report`) to performantly aggregate data from `mrp_production` (Manufacturing Orders) and `stock_move` (Component requirements). Work Center mapping is derived from standard Odoo routing and workorder relationships, gracefully falling back to the Manufacturing Order's primary Work Center if specific operations are undefined.
## Author
Kipas System Implementation

2
__init__.py Normal file
View File

@ -0,0 +1,2 @@
from . import report
from . import wizard

17
__manifest__.py Normal file
View File

@ -0,0 +1,17 @@
# -*- coding: utf-8 -*-
{
'name': 'MRP Components Report',
'version': '19.0.1.0.0',
'category': 'Manufacturing/Manufacturing',
'summary': 'Report showing needed components for a certain date grouped by work center and parent product',
'depends': ['mrp'],
'data': [
'security/ir.model.access.csv',
'report/mrp_components_report_views.xml',
'wizard/mrp_components_wizard_views.xml',
'report/mrp_components_qweb_report.xml',
],
'installable': True,
'application': False,
'license': 'LGPL-3',
}

2
report/__init__.py Normal file
View File

@ -0,0 +1,2 @@
from . import mrp_components_report
from . import mrp_components_pdf_report

View File

@ -0,0 +1,69 @@
# -*- coding: utf-8 -*-
from odoo import models, api
from odoo.exceptions import UserError
class ReportMrpComponents(models.AbstractModel):
_name = 'report.mrp_components_report.report_mrp_components_template'
_description = 'MRP Components PDF Report'
@api.model
def _get_report_values(self, docids, data=None):
if not data or not data.get('form'):
raise UserError("Form content is missing, this report cannot be printed.")
date_start = data['form'].get('date_start')
date_end = data['form'].get('date_end')
domain = [
('date', '>=', date_start),
('date', '<=', date_end)
]
report_records = self.env['mrp.components.report'].search(domain)
grouped_data = {}
for record in report_records:
wc_name = record.workcenter_id.name or 'No Work Center'
prod_name = record.product_id.display_name or 'No Parent Product'
if wc_name not in grouped_data:
grouped_data[wc_name] = {}
if prod_name not in grouped_data[wc_name]:
grouped_data[wc_name][prod_name] = []
grouped_data[wc_name][prod_name].append({
'component_name': record.component_id.display_name,
'qty': record.product_qty,
'uom_name': record.product_uom_id.name,
'date': record.date,
'mo': record.production_id.name,
})
# Optional: consolidate same components if needed.
# But showing individual MOs might be better. Let's consolidate if same component in same MO?
# Standard reports usually aggregate quantities. Let's aggregate by component within the product.
aggregated_data = {}
for wc, products in grouped_data.items():
aggregated_data[wc] = {}
for prod, components in products.items():
agg_comps = {}
for comp in components:
key = comp['component_name']
if key not in agg_comps:
agg_comps[key] = {
'component_name': key,
'qty': 0.0,
'uom_name': comp['uom_name']
}
agg_comps[key]['qty'] += comp['qty']
aggregated_data[wc][prod] = list(agg_comps.values())
return {
'doc_ids': data.get('ids'),
'doc_model': 'mrp.components.wizard',
'date_start': date_start,
'date_end': date_end,
'grouped_data': aggregated_data,
}

View File

@ -0,0 +1,58 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<record id="action_report_mrp_components" model="ir.actions.report">
<field name="name">Components Required Report</field>
<field name="model">mrp.components.wizard</field>
<field name="report_type">qweb-pdf</field>
<field name="report_name">mrp_components_report.report_mrp_components_template</field>
<field name="report_file">mrp_components_report.report_mrp_components_template</field>
<field name="print_report_name">'Components Required - %s to %s' % (object.date_start, object.date_end)</field>
<field name="binding_model_id" ref="model_mrp_components_wizard"/>
<field name="binding_type">report</field>
</record>
<template id="report_mrp_components_template">
<t t-call="web.html_container">
<t t-call="web.external_layout">
<div class="page">
<div class="text-center mt-4 mb-4">
<h2>Components Required Report</h2>
<p><strong>From:</strong> <span t-esc="date_start"/> <strong>To:</strong> <span t-esc="date_end"/></p>
</div>
<t t-if="not grouped_data">
<div class="alert alert-info mt-4">No components needed for the selected period.</div>
</t>
<t t-foreach="grouped_data.keys()" t-as="workcenter">
<h4 class="mt-4 pb-2 border-bottom text-primary"><span t-esc="workcenter"/></h4>
<t t-foreach="grouped_data[workcenter].keys()" t-as="product">
<h6 class="mt-3 ml-3 text-muted"><strong>Parent Product:</strong> <span t-esc="product"/></h6>
<table class="table table-sm table-bordered mt-2 ml-4">
<thead class="thead-light">
<tr>
<th class="w-50">Component</th>
<th class="w-25 text-right">Quantity Needed</th>
<th class="w-25">Unit of Measure</th>
</tr>
</thead>
<tbody>
<t t-foreach="grouped_data[workcenter][product]" t-as="comp">
<tr>
<td><span t-esc="comp['component_name']"/></td>
<td class="text-right"><span t-esc="comp['qty']" t-options='{"widget": "float", "precision": 3}'/></td>
<td><span t-esc="comp['uom_name']"/></td>
</tr>
</t>
</tbody>
</table>
</t>
<br/>
</t>
</div>
</t>
</t>
</template>
</odoo>

View File

@ -0,0 +1,64 @@
# -*- coding: utf-8 -*-
from odoo import tools
from odoo import models, fields
class MrpComponentsReport(models.Model):
_name = 'mrp.components.report'
_description = 'MRP Components Report'
_auto = False
id = fields.Id(readonly=True)
date = fields.Date('Date Needed', readonly=True)
production_id = fields.Many2one('mrp.production', 'Manufacturing Order', readonly=True)
product_id = fields.Many2one('product.product', 'Parent Product', readonly=True)
component_id = fields.Many2one('product.product', 'Component', readonly=True)
workcenter_id = fields.Many2one('mrp.workcenter', 'Work Center', readonly=True)
product_qty = fields.Float('Quantity Needed', readonly=True)
product_uom_id = fields.Many2one('uom.uom', 'Unit of Measure', readonly=True)
state = fields.Selection([
('draft', 'Draft'),
('confirmed', 'Confirmed'),
('progress', 'In Progress'),
('to_close', 'To Close'),
('done', 'Done'),
('cancel', 'Cancelled')], string='MO State', readonly=True)
def _select(self):
return """
SELECT
sm.id AS id,
CAST(mp.date_start AS DATE) AS date,
mp.id AS production_id,
mp.product_id AS product_id,
sm.product_id AS component_id,
COALESCE(mw.id, (SELECT workcenter_id FROM mrp_workorder WHERE production_id = mp.id ORDER BY id ASC LIMIT 1)) AS workcenter_id,
sm.product_uom_qty AS product_qty,
sm.product_uom AS product_uom_id,
mp.state AS state
"""
def _from(self):
return """
FROM stock_move sm
JOIN mrp_production mp ON sm.raw_material_production_id = mp.id
LEFT JOIN mrp_bom_line mbl ON sm.bom_line_id = mbl.id
LEFT JOIN mrp_routing_workcenter mrw ON sm.operation_id = mrw.id
LEFT JOIN mrp_routing_workcenter mrw_bom ON mbl.operation_id = mrw_bom.id
LEFT JOIN mrp_workorder mwo ON sm.workorder_id = mwo.id
LEFT JOIN mrp_workcenter mw ON COALESCE(mrw.workcenter_id, mwo.workcenter_id, mrw_bom.workcenter_id) = mw.id
"""
def _where(self):
return """
WHERE sm.raw_material_production_id IS NOT NULL
AND sm.state != 'cancel'
AND mp.state != 'cancel'
"""
def init(self):
tools.drop_view_if_exists(self.env.cr, self._table)
self.env.cr.execute("""CREATE or REPLACE VIEW %s as (
%s
%s
%s
)""" % (self._table, self._select(), self._from(), self._where()))

View File

@ -0,0 +1,51 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<record id="view_mrp_components_report_tree" model="ir.ui.view">
<field name="name">mrp.components.report.tree</field>
<field name="model">mrp.components.report</field>
<field name="arch" type="xml">
<list string="Components Needed" create="false" edit="false" delete="false">
<field name="date" optional="show"/>
<field name="workcenter_id" optional="show"/>
<field name="product_id" optional="show"/>
<field name="production_id" optional="show"/>
<field name="component_id"/>
<field name="product_qty" sum="Total Quantity"/>
<field name="product_uom_id"/>
<field name="state" optional="hide"/>
</list>
</field>
</record>
<record id="view_mrp_components_report_pivot" model="ir.ui.view">
<field name="name">mrp.components.report.pivot</field>
<field name="model">mrp.components.report</field>
<field name="arch" type="xml">
<pivot string="Components Needed" disable_linking="True">
<field name="workcenter_id" type="row"/>
<field name="product_id" type="row"/>
<field name="component_id" type="row"/>
<field name="product_qty" type="measure"/>
</pivot>
</field>
</record>
<record id="view_mrp_components_report_search" model="ir.ui.view">
<field name="name">mrp.components.report.search</field>
<field name="model">mrp.components.report</field>
<field name="arch" type="xml">
<search string="Search Components Needed">
<field name="component_id"/>
<field name="product_id"/>
<field name="workcenter_id"/>
<field name="production_id"/>
<group>
<filter string="Work Center" name="group_by_workcenter" context="{'group_by':'workcenter_id'}"/>
<filter string="Parent Product" name="group_by_product" context="{'group_by':'product_id'}"/>
<filter string="Component" name="group_by_component" context="{'group_by':'component_id'}"/>
<filter string="Date" name="group_by_date" context="{'group_by':'date'}"/>
</group>
</search>
</field>
</record>
</odoo>

View File

@ -0,0 +1,3 @@
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
access_mrp_components_report,mrp.components.report,model_mrp_components_report,mrp.group_mrp_user,1,0,0,0
access_mrp_components_wizard,mrp.components.wizard,model_mrp_components_wizard,mrp.group_mrp_user,1,1,1,1
1 id name model_id:id group_id:id perm_read perm_write perm_create perm_unlink
2 access_mrp_components_report mrp.components.report model_mrp_components_report mrp.group_mrp_user 1 0 0 0
3 access_mrp_components_wizard mrp.components.wizard model_mrp_components_wizard mrp.group_mrp_user 1 1 1 1

1
wizard/__init__.py Normal file
View File

@ -0,0 +1 @@
from . import mrp_components_wizard

View File

@ -0,0 +1,40 @@
# -*- coding: utf-8 -*-
from odoo import models, fields
class MrpComponentsWizard(models.TransientModel):
_name = 'mrp.components.wizard'
_description = 'MRP Components Report Wizard'
date_start = fields.Date(string='Start Date', required=True, default=fields.Date.context_today)
date_end = fields.Date(string='End Date', required=True, default=fields.Date.context_today)
def action_view_report(self):
self.ensure_one()
domain = [
('date', '>=', self.date_start),
('date', '<=', self.date_end)
]
return {
'name': 'Components Needed',
'type': 'ir.actions.act_window',
'res_model': 'mrp.components.report',
'view_mode': 'list,pivot',
'domain': domain,
'context': {
'search_default_group_by_workcenter': 1,
'search_default_group_by_product': 1,
}
}
def action_print_pdf(self):
self.ensure_one()
data = {
'ids': self.ids,
'model': self._name,
'form': {
'date_start': self.date_start,
'date_end': self.date_end,
},
}
return self.env.ref('mrp_components_report.action_report_mrp_components').report_action(self, data=data)

View File

@ -0,0 +1,35 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<record id="view_mrp_components_wizard_form" model="ir.ui.view">
<field name="name">mrp.components.wizard.form</field>
<field name="model">mrp.components.wizard</field>
<field name="arch" type="xml">
<form string="Components Needed Report">
<group>
<group>
<field name="date_start"/>
<field name="date_end"/>
</group>
</group>
<footer>
<button name="action_view_report" string="View Report" type="object" class="btn-primary"/>
<button name="action_print_pdf" string="Print PDF" type="object" class="btn-secondary"/>
<button string="Cancel" class="btn-secondary" special="cancel"/>
</footer>
</form>
</field>
</record>
<record id="action_mrp_components_wizard" model="ir.actions.act_window">
<field name="name">Components Required</field>
<field name="res_model">mrp.components.wizard</field>
<field name="view_mode">form</field>
<field name="target">new</field>
</record>
<menuitem id="menu_mrp_components_report"
name="Components Required"
parent="mrp.menu_mrp_reporting"
action="action_mrp_components_wizard"
sequence="20"/>
</odoo>