sprint-1: Studio shell PySide6 + wizard 8 pasos
Sprint 1 entrega el shell del Studio operativo. Para correrlo:
uv run vms-studio
Componentes:
vmssailor/studio/theme.py
- Aplica design tokens del Sprint 0 (paleta Deep Ocean) a PySide6
- QSS global completo + QPalette + fuentes Inter/Space Grotesk/JetBrains Mono
vmssailor/studio/app.py
- StudioApp (QApplication) con tema, logo, version
- run_studio() entry point
vmssailor/studio/main_window.py
- Layout: topbar / sidebar / canvas central / statusbar
- Menus: Proyecto (Nuevo wizard, Abrir, Guardar, Guardar como, Salir),
Edicion/Vista (stubs Sprint 2), Ayuda
- Operaciones funcionales: New from wizard, Open .vmsproj, Save, Save As,
Validate (cross-entity), Compile (placeholder)
- Reloj live + statusbar con stats del proyecto
vmssailor/studio/widgets/system_sidebar.py
- Sidebar dinamico que muestra wizard steps + sistemas habilitados + disponibles
- Lee catalogo maestro y proyecto activo
- Senial systemActivated(SystemId) al doble-click
vmssailor/studio/widgets/vessel_canvas.py
- QGraphicsView central con grilla naval (1m por celda)
- Renderiza silueta del buque en planta + mamparos + equipos
- ship_to_scene() transformacion canonica naval -> escena
- Centerline + Pp axis marcados
- Ruler de eslora con marcas cada 5m
- Zoom con wheel + scroll-pan, label de zoom% en header
vmssailor/studio/wizard/ - QWizard 8 pasos
- step_01_vessel_type: tipo + subtipo + nombre proyecto + cliente
- step_02_template: selector con biblioteca curada (Sunseeker, Ferretti, blank)
- step_03_dimensions: LOA/manga/calado/mamparos con pre-fill de plantilla
- step_04_systems: checkboxes agrupados por categoria con pre-select por default_for
- step_57_placeholder: stubs visuales para Sprint 2 (pasos 5, 6, 7)
- step_08_confirm: resumen HTML completo del proyecto a crear
- VesselWizard.build_project() construye un Project valido
Tests (tests/studio/, 11 nuevos, total 110/110):
- pytest-qt offscreen
- Smoke tests del MainWindow, wizard, canvas, sidebar
- test_ship_to_scene_mapping (transformacion naval->escena)
Stack agregado:
- PySide6 6.11.1
- pytest-qt 4.5.0
Decisiones autonomas:
- QFont.setWeight requiere QFont.Weight enum en PySide6 6.11 (no int)
- QFrame.Shape.NoFrame (no QListWidget.NoFrame) para PySide6 6.11
- Pasos 5-7 quedan placeholders explicitos: Sprint 2 implementa rule engine
- Wizard crea Project sin equipment todavia (Sprint 2 los agrega)
Criterios de aceptacion Sprint 1:
- uv run vms-studio: abre ventana operativa
- 110/110 pytest verde
- ruff clean
- Smoke offscreen: MainWindow + Wizard + Canvas + Sidebar OK
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,105 @@
|
||||
"""Paso 3: dimensiones físicas del buque."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from PySide6.QtWidgets import (
|
||||
QDoubleSpinBox,
|
||||
QFormLayout,
|
||||
QLabel,
|
||||
QVBoxLayout,
|
||||
QWidget,
|
||||
QWizardPage,
|
||||
)
|
||||
|
||||
from vmssailor.studio.theme import C_FOG, ui_font
|
||||
|
||||
|
||||
class Step03Dimensions(QWizardPage):
|
||||
def __init__(self, parent: QWidget | None = None) -> None:
|
||||
super().__init__(parent)
|
||||
self.setTitle("Paso 3 · Dimensiones principales")
|
||||
self.setSubTitle(
|
||||
"Edita las dimensiones principales del buque. Los valores se pre-rellenan "
|
||||
"desde la plantilla del paso 2 si elegiste una."
|
||||
)
|
||||
|
||||
layout = QVBoxLayout(self)
|
||||
layout.setSpacing(18)
|
||||
|
||||
intro = QLabel(
|
||||
"Todas las medidas en metros (sistema internacional). "
|
||||
"Estos valores se usan en core para validar coordenadas de equipos y tarjetas."
|
||||
)
|
||||
intro.setWordWrap(True)
|
||||
intro.setStyleSheet(f"color: {C_FOG};")
|
||||
intro.setFont(ui_font(10))
|
||||
layout.addWidget(intro)
|
||||
|
||||
form = QFormLayout()
|
||||
form.setHorizontalSpacing(20)
|
||||
form.setVerticalSpacing(14)
|
||||
|
||||
self._loa = self._make_spin(1.0, 200.0, 24.0, " m", 1)
|
||||
form.addRow("Eslora total (LOA):", self._loa)
|
||||
|
||||
self._beam = self._make_spin(1.0, 30.0, 5.5, " m", 2)
|
||||
form.addRow("Manga máxima:", self._beam)
|
||||
|
||||
self._draft = self._make_spin(0.3, 10.0, 1.8, " m", 2)
|
||||
form.addRow("Calado:", self._draft)
|
||||
|
||||
self._bh_aft = self._make_spin(0.0, 200.0, 4.0, " m desde Pp", 1)
|
||||
form.addRow("Mamparo popa sala máquinas:", self._bh_aft)
|
||||
|
||||
self._bh_fwd = self._make_spin(0.0, 200.0, 8.0, " m desde Pp", 1)
|
||||
form.addRow("Mamparo proa sala máquinas:", self._bh_fwd)
|
||||
|
||||
layout.addLayout(form)
|
||||
layout.addStretch(1)
|
||||
|
||||
from vmssailor.studio.wizard.wizard import (
|
||||
F_BEAM,
|
||||
F_BULKHEAD_AFT,
|
||||
F_BULKHEAD_FWD,
|
||||
F_DRAFT,
|
||||
F_LOA,
|
||||
)
|
||||
|
||||
self.registerField(F_LOA, self._loa, "value", "valueChanged")
|
||||
self.registerField(F_BEAM, self._beam, "value", "valueChanged")
|
||||
self.registerField(F_DRAFT, self._draft, "value", "valueChanged")
|
||||
self.registerField(F_BULKHEAD_AFT, self._bh_aft, "value", "valueChanged")
|
||||
self.registerField(F_BULKHEAD_FWD, self._bh_fwd, "value", "valueChanged")
|
||||
|
||||
def _make_spin(self, lo: float, hi: float, default: float, suffix: str, decimals: int) -> QDoubleSpinBox:
|
||||
s = QDoubleSpinBox()
|
||||
s.setRange(lo, hi)
|
||||
s.setDecimals(decimals)
|
||||
s.setSingleStep(0.1 if decimals >= 1 else 1)
|
||||
s.setSuffix(suffix)
|
||||
s.setValue(default)
|
||||
s.setMinimumWidth(180)
|
||||
return s
|
||||
|
||||
def initializePage(self) -> None:
|
||||
# If the user picked a template in step 2, pre-fill from it.
|
||||
from vmssailor.studio.wizard.wizard import F_TEMPLATE_ID
|
||||
|
||||
wizard = self.wizard()
|
||||
tpl_id = self.field(F_TEMPLATE_ID)
|
||||
if not tpl_id or not wizard:
|
||||
return
|
||||
# Look up the step 2 page to get the template dict
|
||||
from vmssailor.studio.wizard.step_02_template import Step02Template
|
||||
|
||||
for pid in range(8):
|
||||
page = wizard.page(pid)
|
||||
if isinstance(page, Step02Template):
|
||||
tpl = page.get_template()
|
||||
if tpl:
|
||||
self._loa.setValue(tpl["length_overall_m"])
|
||||
self._beam.setValue(tpl["beam_max_m"])
|
||||
self._draft.setValue(tpl["draft_m"])
|
||||
self._bh_aft.setValue(max(2.0, tpl["length_overall_m"] * 0.15))
|
||||
self._bh_fwd.setValue(max(5.0, tpl["length_overall_m"] * 0.30))
|
||||
break
|
||||
Reference in New Issue
Block a user