Files
Odoo-Modules/fusion_repairs/views/repair_order_views.xml
gsinghpal 48dd7718e2 feat(fusion_repairs): Bundle 10 - align pricing to Westin's printed rate card
User shared their actual published service-rate card. Bundle 9's seeded
numbers were placeholders that no longer match. Realigned the rate card,
added the LIFT & ELEVATING SERVICE class, added the in-shop labour
rate path, added the delivery / pickup charge model, added rush as a
proper tier (distinct from after-hours), and added 30-min increment
rounding on top of the existing 1-hour minimum.

EQUIPMENT CLASS

  fusion.repair.product.category gets a new x_fc_equipment_class
  selection: 'standard' vs 'lift_elevating'. The published card splits
  pricing into two service classes - lift_elevating has higher rates
  ($160 callout vs $95, $110/h vs $85).

  Categories marked lift_elevating in seed:
    stairlift, porch_lift, lift_chair (new)

  New 'Lift Chair' category seeded (power recliner / lift chair).

CALLOUT RATE CARD

  fusion.repair.callout.rate gets:
    - equipment_class field (standard / lift_elevating)
    - in_shop_labor_rate field (separate $75 vs $85 on-site)
    - 'rush' tier value (was missing - rush was implicit via emergency
       surcharge from Bundle 8; now a proper tier matching the printed
       rate card row 'Rush Service Calls $120')

  Re-seeded with the PUBLISHED Westin rate card (exact values):

    STANDARD SERVICE
      regular         $95  callout / $85/h on-site  / $75/h in-shop
      rush            $120 callout / $85/h          / $75/h
      after_hours     $140 callout / $85/h          / $75/h
      weekend         $180 callout / $85/h          / $75/h   (extension)
      holiday         $220 callout / $85/h          / $75/h   (extension)

    LIFT & ELEVATING SERVICE
      regular         $160 callout / $110/h on-site / $110/h in-shop
      rush            $200 callout / $110/h         / $110/h  (extension)
      after_hours     $240 callout / $110/h         / $110/h  (extension)
      weekend         $300 callout / $110/h         / $110/h  (extension)
      holiday         $360 callout / $110/h         / $110/h  (extension)

    Travel: $0.70 per km, BOTH WAYS, past 25 km, per technician
    (matches the per-card '$0.70 per km x 2-way' footnote).

  get_for_tier(tier, equipment_class) now resolves with a fallback:
  tries (tier, lift_elevating) first, falls back to (tier, standard)
  if no lift-specific row exists - so an admin can leave standard rows
  as the catch-all and only customise lift for the exceptions.

DELIVERY / PICKUP RATE CARD

  New fusion.repair.delivery.charge model + seed of all 7 items from
  the printed card:
    Local Service Area (within Brampton) ........ $35
    Outside Local Area .......................... $60
    Rush Pickups / Delivery ..................... $60 + $0.70/km x 2-way
    Lift Chair Delivery and Set-Up .............. $120
    Hospital Bed Delivery and Set-Up ............ $120
    Stairlift Delivery and Set-Up ............... $300
    Stairlift Removal ........................... $300

  quote_rush(distance_km) helper for the office's delivery scheduling.
  New menu: Configuration > Delivery / Pickup Charges.

PRICING ENGINE UPDATES (repair.order._compute_callout_quote)

  - Class-aware rate lookup (uses category.equipment_class).
  - In-shop mode (x_fc_in_shop=True): skips callout fee + extra-tech +
    travel; charges in_shop_labor_rate * hours * techs only. Per the
    rate-card footnote 'In-Shop Labour Rate'.
  - 30-min increment rounding ON TOP of the 1-hour floor:
    billable_h = max(ceil(actual * 2) / 2, min_hours)
    -> 20-min work bills 1.0 h
    -> 75-min work bills 1.5 h
    -> 95-min work bills 2.0 h
  - Improved breakdown text shows the rate-card row name + class +
    pro-ration math so the client can see how the total was computed.

NEW FIELDS

  repair.order:
    x_fc_in_shop  (Boolean) - flip to switch the quote engine to
                              in-shop mode.
    x_fc_callout_tier now includes 'rush' as a value (was missing).

  visit-report wizard:
    callout_in_shop related field - tech can flip the mode on-site if
    the work was actually done in-store after pickup.

MIGRATION SCRIPT

  migrations/19.0.2.1.0/post-migration.py runs once on existing
  installs:
    1. Updates stairlift / porch_lift / lift_chair categories
       equipment_class -> lift_elevating
    2. Wipes the 4 Bundle 9 rate-card xml_ids so the new noupdate=1
       seed creates them with the correct printed values.

  Fresh installs get the right values directly from the seed XML.
  Admin-created custom rate rows (no xml_id) are NEVER touched.

VERIFIED END-TO-END (0 bugs across 28 checks)

  Rate card matches printed values exactly:
    regular/standard      = $95/$85h/$75h          PASS
    rush/standard         = $120/$85h/$75h         PASS
    after_hours/standard  = $140/$85h/$75h         PASS
    regular/lift          = $160/$110h/$110h       PASS

  Six end-to-end quote scenarios:
    A. Standard 12km 20-min   -> $180  ($95 + 1h*$85)
    B. Lift     12km 20-min   -> $270  ($160 + 1h*$110)
    C. Rush     30km 1.2h     -> $254.50
       ($120 + ceil(2.4)/2=1.5h * $85 + 5km*2*$0.70 = $7)
    D. After-hours lift 2-tech 35km 2.6h -> $928.00
       ($240 + ceil(5.2)/2=3.0h * $110 * 2 + 10km*2*$0.70*2)
    E. In-shop  standard 2h   -> $150  (2h * $75 in-shop, no callout)
    F. In-shop  lift 1.5h     -> $165  (1.5h * $110 in-shop)

  Seven delivery rates loaded with correct amounts; rush 40km calc
  = $81 ($60 base + 15km*2*$0.70).

  Stairlift / Porch Lift / Lift Chair categories correctly marked
  lift_elevating; rest stay standard.

Bumped to 19.0.2.1.0.

Co-authored-by: Cursor <cursoragent@cursor.com>
2026-05-21 02:47:11 -04:00

420 lines
23 KiB
XML

<?xml version="1.0" encoding="utf-8"?>
<odoo>
<!-- ============================================================== -->
<!-- Form view extensions -->
<!-- ============================================================== -->
<record id="view_repair_order_form_inherit_fusion_repairs" model="ir.ui.view">
<field name="name">repair.order.form.inherit.fusion_repairs</field>
<field name="model">repair.order</field>
<field name="inherit_id" ref="repair.view_repair_order_form"/>
<field name="arch" type="xml">
<!-- Header action buttons (visit report + collect payment) -->
<xpath expr="//header" position="inside">
<button name="action_open_visit_report"
type="object"
string="Visit Report"
class="btn-primary"
invisible="state in ('draft', 'cancel') or x_fc_technician_task_count == 0"
groups="fusion_repairs.group_fusion_repairs_user"/>
<button name="action_collect_payment"
type="object"
string="Collect Payment"
class="btn-secondary"
invisible="state != 'done'"
groups="fusion_repairs.group_fusion_repairs_user"/>
<button name="action_offer_loaner"
type="object"
string="Offer Loaner"
class="btn-secondary"
icon="fa-handshake-o"
invisible="state in ('done', 'cancel')"
groups="fusion_repairs.group_fusion_repairs_user"/>
<!-- Bundle 8: squeeze a rush into today's tech route + acknowledge surcharge -->
<button name="action_squeeze_into_today"
type="object"
string="Squeeze into Today"
class="btn-warning"
icon="fa-flash"
invisible="state in ('done', 'cancel') or not x_fc_rush_requested"
groups="fusion_repairs.group_fusion_repairs_dispatcher"/>
<button name="action_acknowledge_rush"
type="object"
string="Client Agreed to Rush Price"
class="btn-warning"
icon="fa-check"
invisible="not x_fc_rush_requested or x_fc_rush_acknowledged_at"
groups="fusion_repairs.group_fusion_repairs_user"/>
<!-- Bundle 9: warranty + waive (waive group-gated server-side too) -->
<button name="action_check_labor_warranty"
type="object"
string="Check Labor Warranty"
class="btn-secondary"
icon="fa-shield"
invisible="state in ('done', 'cancel')"
groups="fusion_repairs.group_fusion_repairs_user"/>
<button name="action_waive_labor_fee"
type="object"
string="Waive Labor Fee"
class="btn-warning"
icon="fa-percent"
invisible="x_fc_labor_waived or state in ('done', 'cancel')"
groups="fusion_repairs.group_fusion_repairs_sales_rep"/>
</xpath>
<!-- Smart buttons: Technician Tasks + Intake Answers + Original SO. -->
<xpath expr="//div[hasclass('oe_button_box')]" position="inside">
<button name="action_view_technician_tasks"
type="object"
class="oe_stat_button"
icon="fa-wrench"
invisible="x_fc_technician_task_count == 0">
<field name="x_fc_technician_task_count" widget="statinfo" string="Tech Tasks"/>
</button>
<button name="action_view_intake_answers"
type="object"
class="oe_stat_button"
icon="fa-list-alt"
invisible="x_fc_intake_answer_count == 0">
<field name="x_fc_intake_answer_count" widget="statinfo" string="Answers"/>
</button>
<button name="action_view_original_sale_order"
type="object"
class="oe_stat_button"
icon="fa-dollar"
invisible="not x_fc_original_sale_order_id">
<field name="x_fc_original_sale_order_id" widget="statinfo" string="Original SO"/>
</button>
</xpath>
<!-- Add intake metadata under partner_id -->
<xpath expr="//field[@name='partner_id']" position="after">
<field name="x_fc_repair_category_id" options="{'no_create': True}"/>
<field name="x_fc_urgency" widget="badge"
decoration-success="x_fc_urgency == 'normal'"
decoration-warning="x_fc_urgency == 'urgent'"
decoration-danger="x_fc_urgency == 'safety'"/>
<field name="x_fc_third_party_equipment"/>
<field name="x_fc_is_quote_only"/>
<field name="x_fc_intake_source" readonly="1"/>
<field name="x_fc_intake_user_id" readonly="1" invisible="not x_fc_intake_user_id"/>
<field name="x_fc_intake_session_id" readonly="1" invisible="not x_fc_intake_session_id"/>
</xpath>
<!-- Add a Fusion Repairs notebook tab with intake + photos. -->
<xpath expr="//notebook" position="inside">
<page string="Intake" name="fusion_intake">
<group>
<group>
<field name="x_fc_intake_template_id" readonly="1"/>
<field name="x_fc_issue_category"/>
</group>
<group>
<field name="x_fc_warranty_override_reason"
placeholder="Reason if warranty status was overridden"/>
<field name="x_fc_estimated_duration" widget="float_time"/>
</group>
</group>
<separator string="Answers"/>
<field name="x_fc_intake_answer_ids" readonly="1">
<list>
<field name="sequence" column_invisible="True"/>
<field name="question_name"/>
<field name="value_display"/>
<field name="question_type" optional="hide"/>
</list>
</field>
<separator string="Photos &amp; Videos"/>
<field name="x_fc_photo_ids" widget="many2many_binary"/>
</page>
<page string="Pricing" name="fusion_pricing" invisible="not x_fc_estimated_cost and not x_fc_actual_cost">
<group>
<group>
<field name="x_fc_estimated_cost" widget="monetary"/>
<field name="x_fc_actual_cost" widget="monetary"/>
</group>
<group>
<field name="x_fc_cost_variance_pct" widget="float" digits="[16,2]"/>
<field name="x_fc_requires_requote"/>
<field name="company_currency_id" invisible="1"/>
</group>
</group>
</page>
<page string="AI Brief" name="fusion_ai" invisible="not x_fc_ai_summary">
<field name="x_fc_ai_summary" readonly="1"/>
</page>
<page string="Margin" name="fusion_margin"
groups="fusion_repairs.group_fusion_repairs_manager">
<group>
<group>
<field name="x_fc_revenue" widget="monetary"/>
<field name="x_fc_labour_cost" widget="monetary"/>
<field name="x_fc_parts_cost" widget="monetary"/>
</group>
<group>
<field name="x_fc_margin" widget="monetary"/>
<field name="x_fc_margin_pct" widget="float" digits="[12,1]"/>
<field name="company_currency_id" invisible="1"/>
</group>
</group>
</page>
<page string="Callout Pricing" name="fusion_callout">
<group>
<group string="Inputs">
<field name="x_fc_callout_tier"/>
<field name="x_fc_in_shop"/>
<field name="x_fc_callout_techs"
invisible="x_fc_in_shop"/>
<field name="x_fc_callout_distance_km"
invisible="x_fc_in_shop"/>
<field name="x_fc_callout_labor_hours"/>
</group>
<group string="Warranty / Waiver">
<field name="x_fc_labor_warranty_id" readonly="1"/>
<field name="x_fc_labor_warranty_status" widget="badge"
decoration-success="x_fc_labor_warranty_status == 'eligible'"
decoration-warning="x_fc_labor_warranty_status in ('expired', 'waived')"
decoration-danger="x_fc_labor_warranty_status == 'void_misuse'"/>
<field name="x_fc_labor_waived" readonly="1"/>
<field name="x_fc_labor_waived_by_id" readonly="1"
invisible="not x_fc_labor_waived"/>
<field name="x_fc_labor_waived_at" readonly="1"
invisible="not x_fc_labor_waived"/>
<field name="x_fc_labor_waived_reason"
invisible="not x_fc_labor_waived"
readonly="not x_fc_labor_waived"/>
</group>
</group>
<separator string="Quote Breakdown"/>
<group>
<group>
<field name="x_fc_quote_callout_base" widget="monetary" readonly="1"/>
<field name="x_fc_quote_extra_techs" widget="monetary" readonly="1"/>
<field name="x_fc_quote_labor" widget="monetary" readonly="1"/>
<field name="x_fc_quote_travel" widget="monetary" readonly="1"/>
</group>
<group>
<field name="x_fc_quote_waived" widget="monetary" readonly="1"/>
<field name="x_fc_quote_total" widget="monetary" readonly="1"
class="oe_subtotal_footer_separator"/>
</group>
</group>
</page>
<page string="Rush / Parts" name="fusion_rush_parts">
<group>
<group string="Rush Service">
<field name="x_fc_rush_requested"/>
<field name="x_fc_rush_tier"
invisible="not x_fc_rush_requested"/>
<field name="x_fc_rush_techs_required"
invisible="not x_fc_rush_requested"/>
<field name="x_fc_rush_surcharge"
widget="monetary"
readonly="1"
invisible="not x_fc_rush_requested"/>
<field name="x_fc_rush_acknowledged_at"
readonly="1"
invisible="not x_fc_rush_requested"/>
<field name="x_fc_rush_acknowledged_by_id"
readonly="1"
invisible="not x_fc_rush_requested"/>
</group>
<group string="Awaiting Parts">
<field name="x_fc_parts_awaiting" readonly="1"/>
<field name="x_fc_parts_eta_date" readonly="1"
invisible="not x_fc_parts_awaiting"/>
<field name="x_fc_part_order_count" readonly="1"/>
</group>
</group>
<field name="x_fc_part_order_ids" readonly="1">
<list>
<field name="name"/>
<field name="description"/>
<field name="oem_part_number"/>
<field name="quantity"/>
<field name="expected_date"/>
<field name="received_date"/>
<field name="state" widget="badge"/>
</list>
</field>
</page>
</xpath>
</field>
</record>
<!-- ============================================================== -->
<!-- Kanban: add urgency badge + intake source -->
<!-- ============================================================== -->
<record id="view_repair_order_kanban_inherit_fusion_repairs" model="ir.ui.view">
<field name="name">repair.order.kanban.inherit.fusion_repairs</field>
<field name="model">repair.order</field>
<field name="inherit_id" ref="repair.view_repair_kanban"/>
<field name="arch" type="xml">
<xpath expr="//field[@name='partner_id']" position="after">
<field name="x_fc_urgency"/>
<field name="x_fc_third_party_equipment"/>
</xpath>
</field>
</record>
<!-- ============================================================== -->
<!-- List: add urgency + source columns -->
<!-- ============================================================== -->
<record id="view_repair_order_list_inherit_fusion_repairs" model="ir.ui.view">
<field name="name">repair.order.list.inherit.fusion_repairs</field>
<field name="model">repair.order</field>
<field name="inherit_id" ref="repair.view_repair_order_tree"/>
<field name="arch" type="xml">
<xpath expr="//field[@name='partner_id']" position="after">
<field name="x_fc_urgency" widget="badge"
decoration-success="x_fc_urgency == 'normal'"
decoration-warning="x_fc_urgency == 'urgent'"
decoration-danger="x_fc_urgency == 'safety'"
optional="show"/>
<field name="x_fc_intake_source" optional="hide"/>
<field name="x_fc_third_party_equipment" optional="hide"/>
</xpath>
</field>
</record>
<!-- ============================================================== -->
<!-- New Service Call action - opens the wizard as a modal -->
<!-- ============================================================== -->
<record id="action_open_repair_intake_wizard" model="ir.actions.act_window">
<field name="name">New Service Call</field>
<field name="res_model">fusion.repair.intake.wizard</field>
<field name="view_mode">form</field>
<field name="target">new</field>
</record>
<!-- ============================================================== -->
<!-- Fusion Repairs Service Calls dashboard -->
<!-- Branded kanban / list of repair.order filtered to repairs that -->
<!-- came through one of the Fusion intake surfaces, with the -->
<!-- New Service Call wizard wired into the header. -->
<!-- ============================================================== -->
<record id="view_fusion_repair_dashboard_kanban" model="ir.ui.view">
<field name="name">repair.order.dashboard.fusion_repairs</field>
<field name="model">repair.order</field>
<field name="arch" type="xml">
<kanban default_group_by="state"
class="o_kanban_small_column o_kanban_repair_dashboard"
sample="1">
<field name="name"/>
<field name="partner_id"/>
<field name="state"/>
<field name="x_fc_urgency"/>
<field name="x_fc_third_party_equipment"/>
<field name="x_fc_repair_category_id"/>
<field name="x_fc_intake_source"/>
<field name="x_fc_estimated_cost"/>
<field name="company_currency_id"/>
<field name="schedule_date"/>
<templates>
<t t-name="card">
<div class="d-flex justify-content-between mb-2">
<strong class="o_kanban_record_title">
<field name="name"/>
</strong>
<span t-attf-class="badge {{ {'safety':'text-bg-danger','urgent':'text-bg-warning','normal':'text-bg-secondary'}[record.x_fc_urgency.raw_value] }}">
<field name="x_fc_urgency"/>
</span>
</div>
<div class="text-muted small mb-1">
<i class="fa fa-user me-1"/>
<field name="partner_id"/>
</div>
<div class="text-muted small mb-1" t-if="record.x_fc_repair_category_id.raw_value">
<i class="fa fa-wrench me-1"/>
<field name="x_fc_repair_category_id"/>
</div>
<div class="text-muted small mb-1" t-if="record.schedule_date.raw_value">
<i class="fa fa-calendar me-1"/>
<field name="schedule_date" widget="date"/>
</div>
<div class="d-flex justify-content-between mt-2">
<span class="small text-muted">
<field name="x_fc_intake_source"/>
</span>
<span t-if="record.x_fc_third_party_equipment.raw_value"
class="badge text-bg-warning small">3rd-party</span>
</div>
</t>
</templates>
</kanban>
</field>
</record>
<record id="view_fusion_repair_dashboard_search" model="ir.ui.view">
<field name="name">repair.order.search.fusion_repairs</field>
<field name="model">repair.order</field>
<field name="arch" type="xml">
<search string="Service Calls">
<field name="name"/>
<field name="partner_id"/>
<field name="x_fc_repair_category_id"/>
<filter string="Today" name="today"
domain="[('create_date', '&gt;=', datetime.datetime.combine(context_today(), datetime.time(0,0,0)))]"/>
<filter string="This Week" name="week"
domain="[('create_date', '&gt;=', datetime.datetime.combine(context_today() - datetime.timedelta(days=7), datetime.time(0,0,0)))]"/>
<separator/>
<filter string="Safety" name="safety"
domain="[('x_fc_urgency', '=', 'safety')]"/>
<filter string="Urgent" name="urgent"
domain="[('x_fc_urgency', '=', 'urgent')]"/>
<filter string="Third-Party" name="thirdparty"
domain="[('x_fc_third_party_equipment', '=', True)]"/>
<filter string="Quote Only" name="quote_only"
domain="[('x_fc_is_quote_only', '=', True)]"/>
<filter string="Rush / Emergency" name="rush"
domain="[('x_fc_rush_requested', '=', True)]"/>
<filter string="Awaiting Parts" name="awaiting_parts"
domain="[('x_fc_parts_awaiting', '=', True)]"/>
<separator/>
<filter string="From Backend Wizard" name="src_backend"
domain="[('x_fc_intake_source', '=', 'backend_wizard')]"/>
<filter string="From Sales Rep Portal" name="src_salesrep"
domain="[('x_fc_intake_source', '=', 'sales_rep_portal')]"/>
<filter string="From Client Portal" name="src_client"
domain="[('x_fc_intake_source', '=', 'client_portal')]"/>
<separator/>
<filter string="Open (not closed)" name="open"
domain="[('state', 'not in', ('done', 'cancel'))]"/>
<group>
<filter string="Status" name="group_state" context="{'group_by': 'state'}"/>
<filter string="Urgency" name="group_urgency" context="{'group_by': 'x_fc_urgency'}"/>
<filter string="Category" name="group_category" context="{'group_by': 'x_fc_repair_category_id'}"/>
<filter string="Intake Source" name="group_source" context="{'group_by': 'x_fc_intake_source'}"/>
</group>
</search>
</field>
</record>
<record id="action_fusion_repair_dashboard" model="ir.actions.act_window">
<field name="name">Service Calls</field>
<field name="res_model">repair.order</field>
<field name="view_mode">kanban,list,form</field>
<field name="search_view_id" ref="view_fusion_repair_dashboard_search"/>
<field name="context">{'search_default_open': 1}</field>
<field name="help" type="html">
<p class="o_view_nocontent_smiling_face">No service calls yet</p>
<p>
Click <strong>New</strong> in the top-left to open the guided
intake wizard. The form will walk you through caller info,
equipment selection, the issue, urgency and photos.
</p>
</field>
</record>
<record id="action_fusion_repair_dashboard_kanban" model="ir.actions.act_window.view">
<field name="sequence" eval="1"/>
<field name="view_mode">kanban</field>
<field name="view_id" ref="view_fusion_repair_dashboard_kanban"/>
<field name="act_window_id" ref="action_fusion_repair_dashboard"/>
</record>
</odoo>