6ad76a89fa
Wizard pasos 5-7 ahora funcionales (1-4 ya estaban en Sprint 1).
vmssailor/studio/designer/rule_engine.py
- RuleContext, RuleEngine, EquipmentProposal
- Lee library/rules/*.yaml y aplica reglas heuristicas
- Filtra por vessel_type, vessel_subtype, length_overall_m range
- Selecciona candidato segun condiciones 'when' (loa min/max)
- Genera tag_prefix con sustitucion {side}/{idx}
vmssailor/studio/designer/port_auto_assigner.py
- auto_assign() greedy: 1 bus Modbus RTU + tarjetas dedicadas para motores/gensets
- Tarjeta auxiliar compartida para resto de equipos
- Mapea SignalType -> ChannelType (AI/DI/DO/RPM)
- Genera TagBindings con scaling apropiado por tipo de senal
- Respeta capacidades 10/5/4/1 de AR-NMEA-IO-v1.0
- AssignmentReport con cards + tags + warnings
vmssailor/studio/wizard/step_05_equipment.py
- Tabla con propuestas del rule engine
- Checkboxes accept/reject + edicion inline de columnas
- Boton 'Regenerar' para re-aplicar reglas
vmssailor/studio/wizard/step_06_refinement.py
- Vista resumen de equipos aceptados
vmssailor/studio/wizard/step_07_topology.py
- Llama auto_assign sobre los equipos materializados
- Muestra tabla de tarjetas con uso por canal (DO/DI/AI/RPM)
- Lista warnings de capacidad
vmssailor/studio/editors/equipment_editor.py
- CRUD de Equipment del proyecto activo
- Tabla editable inline (tag_prefix, name, model_ref, system_id, coords, deck)
- Dialog modal para agregar equipos
- Senal projectMutated para refrescar canvas + sidebar
vmssailor/studio/main_window.py
- Layout actualizado: splitter vertical en panel derecho
(canvas arriba + equipment editor abajo)
- _on_project_mutated() re-distribuye al sidebar y canvas
Biblioteca expandida (Sprint 2 brief: 5-7 yates, 10+ motores, gensets, bombas):
- vessels: + azimut_grande_32m, princess_y85, trawler_32m_offshore, patrol_coastal_30m (total: 6)
- engines: + cat_c32_acert, mtu_16v_2000_m96, yanmar_8lv_370 (total: 5)
- gensets: + kohler_28efkozd, onan_qd13500 (total: 3)
- pumps: + jabsco_36800, grundfos_cm10 (NUEVO categoria pumps)
Tests (tests/studio/test_designer.py, 10 nuevos, total 120/120):
- Rule engine: load default, propose engines, candidate picking por LOA
- auto_assign builds topology compatible with Project (Pydantic validation)
- Equipment editor smoke
VesselWizard.build_project() ahora materializa equipment + topology + tags
desde las propuestas y la asignacion automatica del paso 7.
Criterios Sprint 2:
- uv run vms-studio crea proyecto completo desde wizard con equipos + tags + topologia
- vms-validate-library: OK 6 vessels, 10 equipment, 1 rules
- 120/120 pytest verde, ruff clean
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
63 lines
2.0 KiB
Python
63 lines
2.0 KiB
Python
"""Paso 6: refinamiento manual (Sprint 2).
|
|
|
|
En Sprint 1 era placeholder. En Sprint 2 permite reasignar ubicaciones de
|
|
equipos y editar tag_prefix con una vista preview de la silueta.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
from PySide6.QtWidgets import (
|
|
QLabel,
|
|
QListWidget,
|
|
QVBoxLayout,
|
|
QWidget,
|
|
QWizardPage,
|
|
)
|
|
|
|
from vmssailor.studio.theme import C_FOG, mono_font, ui_font
|
|
|
|
|
|
class Step06Refinement(QWizardPage):
|
|
"""Muestra resumen de equipos aceptados con preview textual."""
|
|
|
|
def __init__(self, parent: QWidget | None = None) -> None:
|
|
super().__init__(parent)
|
|
self.setTitle("Paso 6 · Refinamiento")
|
|
self.setSubTitle(
|
|
"Revisa los equipos finales antes de generar topología. "
|
|
"El editor visual completo viene en Sprint 3."
|
|
)
|
|
|
|
layout = QVBoxLayout(self)
|
|
layout.setSpacing(12)
|
|
|
|
self._list = QListWidget()
|
|
self._list.setFont(mono_font(10))
|
|
layout.addWidget(self._list, 1)
|
|
|
|
# Stats
|
|
self._stats = QLabel("Cargando…")
|
|
self._stats.setStyleSheet(f"color: {C_FOG};")
|
|
self._stats.setFont(ui_font(10))
|
|
layout.addWidget(self._stats)
|
|
|
|
def initializePage(self) -> None:
|
|
from vmssailor.studio.wizard.wizard import F_PROPOSALS
|
|
|
|
proposals = self.field(F_PROPOSALS) or []
|
|
self._list.clear()
|
|
by_system: dict[str, int] = {}
|
|
for p in proposals:
|
|
self._list.addItem(
|
|
f"{p.system_id.value:20s} {p.tag_prefix:14s} "
|
|
f"{p.model_ref:30s} @ x_pp={p.location_x_pp:.2f}m y_cl={p.location_y_cl:+.2f}m"
|
|
)
|
|
by_system[p.system_id.value] = by_system.get(p.system_id.value, 0) + 1
|
|
n = len(proposals)
|
|
groups = ", ".join(f"{k}: {v}" for k, v in sorted(by_system.items()))
|
|
self._stats.setText(
|
|
f"{n} equipos aceptados · {groups}"
|
|
if n
|
|
else "Sin equipos aceptados. Vuelve atrás y selecciona al menos uno."
|
|
)
|