Files
Odoo-Modules/fusion_repairs/models/res_partner.py
gsinghpal 429084e0bf feat(fusion_repairs): Phase 1 MVP - backend intake wizard + core models
Scaffolds the fusion_repairs module that extends Odoo 19 repair.order with
a guided medical-equipment intake workflow.

Models
- fusion.repair.product.category (8 medical equipment categories seeded)
- fusion.repair.intake.template / .question / .answer (7 templates,
  32 questions seeded across hospital bed, stairlift, porch lift,
  wheelchair, walker/rollator, mattress)
- fusion.repair.intake.service (AbstractModel) - single entry point used
  by backend wizard, sales rep portal, and public client portal so all
  three surfaces produce identical outcomes
- repair.order extensions (x_fc_intake_*, x_fc_third_party_equipment,
  x_fc_photo_ids, x_fc_urgency, x_fc_estimated/actual_cost, AI summary)
- fusion.technician.task back-link (x_fc_repair_order_id)
- res.partner service preferences (preferred tech, time window, access notes)
- res.users repair extensions (skills, cost rate, on-call rotation fields)
- res.config.settings for variance thresholds, portal URL, rate limit

UI
- Backend intake wizard with multi-equipment loop, third-party flag, photos
- repair.order form: Intake tab, Photos, Pricing tab, AI tab, smart buttons
  (technician tasks, intake answers, original SO)
- Kanban + list view urgency badges
- Fusion Repairs app menu (New Service Call, Repair Orders, Config)

Activities & Email
- 4 follow-up activity types (CS callback, tech dispatch, visit follow-up,
  manager review) with urgency-tiered deadlines
- 2 mail templates (client confirmation + office notification) with the
  same dark/light-safe styling as fusion_claims ADP templates

Security
- New res.groups.privilege + 3 groups (User, Dispatcher, Manager)
- Reuses fusion_tasks.group_field_technician (do NOT recreate)
- Reuses fusion_authorizer_portal.group_sales_rep_portal
- Multi-company global rule + technician scoping rule on repair.order

Verified end-to-end on local westin-v19 dev DB via odoo-shell - creates
multiple repairs in one session, auto-creates dispatch task for urgent,
attaches 4 activity types correctly per urgency tier and third-party flag.

Co-authored-by: Cursor <cursoragent@cursor.com>
2026-05-20 21:35:52 -04:00

65 lines
2.2 KiB
Python

# -*- coding: utf-8 -*-
# Copyright 2024-2026 Nexa Systems Inc.
# License OPL-1 (Odoo Proprietary License v1.0)
from odoo import fields, models
PREFERRED_WINDOW = [
('morning', 'Morning (9 AM - 12 PM)'),
('afternoon', 'Afternoon (12 PM - 5 PM)'),
('evening', 'Evening (after 5 PM)'),
('any', 'Any Time'),
]
class ResPartner(models.Model):
_inherit = 'res.partner'
# ------------------------------------------------------------------
# SERVICE PREFERENCES (P1 - shown in client history sidebar)
# ------------------------------------------------------------------
x_fc_preferred_tech_id = fields.Many2one(
'res.users',
string='Preferred Technician',
domain="[('x_fc_is_field_staff', '=', True)]",
help='If set, this technician is suggested first on dispatch.',
)
x_fc_preferred_window = fields.Selection(
PREFERRED_WINDOW,
string='Preferred Visit Window',
default='any',
)
x_fc_access_notes = fields.Text(
string='Access Notes',
help='Free-form notes for technicians arriving at this address: '
'gate code, dog warning, where to park, side door entry, etc.',
)
# ------------------------------------------------------------------
# CLIENT HISTORY SIDEBAR (C2 - pulled lazily on demand)
# ------------------------------------------------------------------
x_fc_repair_count = fields.Integer(
compute='_compute_x_fc_repair_count',
string='Repairs Count',
compute_sudo=True,
help='Lightweight count of repair orders for this partner. Heavier history '
'data is fetched lazily by the wizard / portal sidebar via RPC.',
)
def _compute_x_fc_repair_count(self):
# Non-stored compute - safe to omit @api.depends.
if not self.ids:
for partner in self:
partner.x_fc_repair_count = 0
return
Repair = self.env['repair.order'].sudo()
data = Repair._read_group(
[('partner_id', 'in', self.ids)],
['partner_id'],
['__count'],
)
counts = {row[0].id: row[1] for row in data}
for partner in self:
partner.x_fc_repair_count = counts.get(partner.id, 0)