feat(plating): Sub 8 — split receiving vs inspection + box parity
fp.receiving simplifies to box-count-only (new primary state machine: draft → counted → staged → closed). Legacy inspecting/accepted/discrepancy/resolved states stay in the Selection so existing records load without error but are surfaced behind a manager-only toggle. New box_count_in field + banner that tells the receiver "count boxes only — parts are inspected by the racking crew." New fp.racking.inspection + fp.racking.inspection.line models — one record per MO, auto-created by mrp.production.create() with one line per contributing SO line (qty_expected seeded, qty_found + condition filled in by the racking crew when they open the boxes). State: draft → inspecting → done | discrepancy_flagged (flagged when any line has a non-ok condition or qty variance). Reopen restricted to Plating Manager. WO soft gate: first plating WO button_start raises a UserError when the MO's racking inspection is still Draft or Inspecting. Plating Manager bypasses; later WOs are not gated. fp.delivery gains x_fc_box_count_out. action_mark_delivered calls _fp_check_box_parity which posts a non-blocking chatter warning when boxes out ≠ boxes in (resolved via job_ref → MO.origin → SO → receiving). Warning only — never blocks shipping. Menu entry: Plating → Operations → Racking Inspection. Module version bumps: fusion_plating_receiving → 19.0.3.0.0 fusion_plating_logistics → 19.0.3.0.0 fusion_plating_bridge_mrp → 19.0.12.0.0 (+depends receiving) Smoke on entech: 12/12 assertions pass (one gate test skipped — MO had no WOs to test) including box-count state machine, inspection auto-create, lifecycle, discrepancy flag, and box-parity chatter. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,165 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!--
|
||||
Copyright 2026 Nexa Systems Inc.
|
||||
License OPL-1 (Odoo Proprietary License v1.0)
|
||||
Sub 8 — Racking-time inspection views.
|
||||
-->
|
||||
<odoo>
|
||||
|
||||
<record id="view_fp_racking_inspection_form" model="ir.ui.view">
|
||||
<field name="name">fp.racking.inspection.form</field>
|
||||
<field name="model">fp.racking.inspection</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="Racking Inspection">
|
||||
<header>
|
||||
<button name="action_start" type="object"
|
||||
string="Start Inspection"
|
||||
class="btn-primary"
|
||||
invisible="state != 'draft'"/>
|
||||
<button name="action_complete" type="object"
|
||||
string="Complete"
|
||||
class="btn-primary"
|
||||
invisible="state != 'inspecting'"/>
|
||||
<button name="action_reopen" type="object"
|
||||
string="Reopen"
|
||||
groups="fusion_plating.group_fusion_plating_manager"
|
||||
invisible="state not in ('done', 'discrepancy_flagged')"/>
|
||||
<field name="state" widget="statusbar"
|
||||
statusbar_visible="draft,inspecting,done,discrepancy_flagged"/>
|
||||
</header>
|
||||
<sheet>
|
||||
<div class="oe_title">
|
||||
<h1><field name="name" readonly="1"/></h1>
|
||||
</div>
|
||||
<div class="alert alert-info" role="alert"
|
||||
invisible="state != 'draft'">
|
||||
<i class="fa fa-info-circle me-2"/>
|
||||
<strong>Racking inspection.</strong> Open the customer's boxes,
|
||||
count each part type, and note any condition issues below.
|
||||
This is separate from receiving (which just counted boxes).
|
||||
</div>
|
||||
<group>
|
||||
<group>
|
||||
<field name="production_id" readonly="1"/>
|
||||
<field name="sale_order_id" readonly="1"/>
|
||||
<field name="receiving_id" readonly="1"/>
|
||||
</group>
|
||||
<group>
|
||||
<field name="partner_id" readonly="1"/>
|
||||
<field name="inspector_id" readonly="1"/>
|
||||
<field name="inspection_started" readonly="1"/>
|
||||
<field name="inspection_completed" readonly="1"/>
|
||||
</group>
|
||||
</group>
|
||||
<notebook>
|
||||
<page string="Inspection Lines" name="lines">
|
||||
<field name="line_ids" readonly="state in ('done','discrepancy_flagged')">
|
||||
<list editable="bottom"
|
||||
decoration-warning="condition == 'minor' or qty_variance != 0"
|
||||
decoration-danger="condition in ('major','reject')">
|
||||
<field name="sequence" widget="handle"/>
|
||||
<field name="part_catalog_id"/>
|
||||
<field name="part_number" readonly="1"/>
|
||||
<field name="part_revision" readonly="1"/>
|
||||
<field name="qty_expected"/>
|
||||
<field name="qty_found"/>
|
||||
<field name="qty_variance" readonly="1"/>
|
||||
<field name="condition"/>
|
||||
<field name="notes"/>
|
||||
</list>
|
||||
</field>
|
||||
</page>
|
||||
<page string="Notes" name="notes">
|
||||
<field name="notes" nolabel="1"
|
||||
readonly="state in ('done','discrepancy_flagged')"/>
|
||||
</page>
|
||||
<page string="Summary" name="summary">
|
||||
<group>
|
||||
<field name="line_count" readonly="1"/>
|
||||
<field name="ok_count" readonly="1"/>
|
||||
<field name="flagged_count" readonly="1"/>
|
||||
<field name="has_variance" readonly="1"/>
|
||||
</group>
|
||||
</page>
|
||||
</notebook>
|
||||
</sheet>
|
||||
<chatter/>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="view_fp_racking_inspection_list" model="ir.ui.view">
|
||||
<field name="name">fp.racking.inspection.list</field>
|
||||
<field name="model">fp.racking.inspection</field>
|
||||
<field name="arch" type="xml">
|
||||
<list string="Racking Inspections"
|
||||
decoration-info="state == 'inspecting'"
|
||||
decoration-warning="state == 'draft'"
|
||||
decoration-success="state == 'done'"
|
||||
decoration-danger="state == 'discrepancy_flagged'">
|
||||
<field name="name"/>
|
||||
<field name="production_id"/>
|
||||
<field name="sale_order_id"/>
|
||||
<field name="partner_id"/>
|
||||
<field name="line_count" string="Parts"/>
|
||||
<field name="flagged_count" string="Flagged"/>
|
||||
<field name="inspector_id"/>
|
||||
<field name="state" widget="badge"/>
|
||||
</list>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="view_fp_racking_inspection_search" model="ir.ui.view">
|
||||
<field name="name">fp.racking.inspection.search</field>
|
||||
<field name="model">fp.racking.inspection</field>
|
||||
<field name="arch" type="xml">
|
||||
<search>
|
||||
<field name="name"/>
|
||||
<field name="production_id"/>
|
||||
<field name="partner_id"/>
|
||||
<separator/>
|
||||
<filter name="filter_pending"
|
||||
string="Pending"
|
||||
domain="[('state', 'in', ['draft', 'inspecting'])]"/>
|
||||
<filter name="filter_flagged"
|
||||
string="Flagged"
|
||||
domain="[('state', '=', 'discrepancy_flagged')]"/>
|
||||
<filter name="filter_done"
|
||||
string="Done"
|
||||
domain="[('state', '=', 'done')]"/>
|
||||
<group>
|
||||
<filter name="group_state" string="State"
|
||||
context="{'group_by': 'state'}"/>
|
||||
<filter name="group_customer" string="Customer"
|
||||
context="{'group_by': 'partner_id'}"/>
|
||||
</group>
|
||||
</search>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="action_fp_racking_inspection" model="ir.actions.act_window">
|
||||
<field name="name">Racking Inspection</field>
|
||||
<field name="res_model">fp.racking.inspection</field>
|
||||
<field name="view_mode">list,form</field>
|
||||
<field name="search_view_id" ref="view_fp_racking_inspection_search"/>
|
||||
<field name="context">{'search_default_filter_pending': 1}</field>
|
||||
<field name="help" type="html">
|
||||
<p class="o_view_nocontent_smiling_face">
|
||||
No racking inspections yet.
|
||||
</p>
|
||||
<p>
|
||||
Racking inspections are auto-created when an MO is confirmed.
|
||||
The racking crew opens the customer's boxes, counts parts,
|
||||
and logs condition findings — the per-part quality check
|
||||
that used to live on receiving.
|
||||
</p>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<menuitem id="menu_fp_racking_inspection"
|
||||
name="Racking Inspection"
|
||||
parent="fusion_plating.menu_fp_operations"
|
||||
action="action_fp_racking_inspection"
|
||||
sequence="23"/>
|
||||
|
||||
</odoo>
|
||||
@@ -40,28 +40,50 @@
|
||||
<field name="arch" type="xml">
|
||||
<form string="Receiving">
|
||||
<header>
|
||||
<button name="action_start_inspection"
|
||||
string="Start Inspection"
|
||||
type="object"
|
||||
invisible="state != 'draft'"/>
|
||||
<button name="action_accept"
|
||||
string="Accept"
|
||||
<!-- Sub 8 — new primary flow: box count only -->
|
||||
<button name="action_mark_counted"
|
||||
string="Mark Counted"
|
||||
type="object"
|
||||
class="btn-primary"
|
||||
invisible="state not in ('inspecting', 'resolved')"/>
|
||||
invisible="state not in ('draft', 'inspecting')"/>
|
||||
<button name="action_mark_staged"
|
||||
string="Stage for Racking"
|
||||
type="object"
|
||||
class="btn-primary"
|
||||
invisible="state != 'counted'"/>
|
||||
<button name="action_close"
|
||||
string="Close"
|
||||
type="object"
|
||||
invisible="state not in ('staged', 'accepted', 'resolved')"/>
|
||||
<!-- Legacy actions (hidden by default; surfaces for old records) -->
|
||||
<button name="action_accept"
|
||||
string="Accept (legacy)"
|
||||
type="object"
|
||||
invisible="state != 'inspecting'"
|
||||
groups="fusion_plating.group_fusion_plating_manager"/>
|
||||
<button name="action_flag_discrepancy"
|
||||
string="Flag Discrepancy"
|
||||
string="Flag Discrepancy (legacy)"
|
||||
type="object"
|
||||
class="btn-danger"
|
||||
invisible="state != 'inspecting'"/>
|
||||
invisible="state != 'inspecting'"
|
||||
groups="fusion_plating.group_fusion_plating_manager"/>
|
||||
<button name="action_resolve"
|
||||
string="Resolve"
|
||||
string="Resolve (legacy)"
|
||||
type="object"
|
||||
invisible="state != 'discrepancy'"/>
|
||||
invisible="state != 'discrepancy'"
|
||||
groups="fusion_plating.group_fusion_plating_manager"/>
|
||||
<field name="state" widget="statusbar"
|
||||
statusbar_visible="draft,inspecting,accepted"/>
|
||||
statusbar_visible="draft,counted,staged,closed"/>
|
||||
</header>
|
||||
<sheet>
|
||||
<div class="alert alert-info" role="alert">
|
||||
<i class="fa fa-info-circle me-2"/>
|
||||
<strong>Receiving = box count only.</strong>
|
||||
Count the boxes the truck dropped off, set the number
|
||||
below, and stage them for racking. The racking crew
|
||||
opens the boxes and inspects each part — see
|
||||
<em>Plating → Operations → Racking Inspection</em>.
|
||||
</div>
|
||||
<div class="oe_title">
|
||||
<h1>
|
||||
<field name="name" readonly="1"/>
|
||||
@@ -73,6 +95,9 @@
|
||||
<field name="partner_id"/>
|
||||
<field name="po_number"/>
|
||||
</group>
|
||||
<group string="Box Count">
|
||||
<field name="box_count_in"/>
|
||||
</group>
|
||||
</group>
|
||||
<group>
|
||||
<group string="Reception">
|
||||
@@ -81,9 +106,9 @@
|
||||
<field name="carrier_name"/>
|
||||
<field name="carrier_tracking"/>
|
||||
</group>
|
||||
<group string="Quantities">
|
||||
<field name="expected_qty"/>
|
||||
<field name="received_qty"/>
|
||||
<group string="Quantities (populated by racking crew)">
|
||||
<field name="expected_qty" readonly="1"/>
|
||||
<field name="received_qty" readonly="1"/>
|
||||
<field name="qty_match" widget="boolean_toggle" readonly="1"/>
|
||||
</group>
|
||||
</group>
|
||||
|
||||
Reference in New Issue
Block a user