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,134 @@
|
||||
"""Paso 8: confirmación final con resumen del proyecto."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from PySide6.QtCore import Qt
|
||||
from PySide6.QtWidgets import QLabel, QVBoxLayout, QWidget, QWizardPage
|
||||
|
||||
from vmssailor.studio.theme import (
|
||||
C_FOAM,
|
||||
C_FOG,
|
||||
C_MIDNIGHT,
|
||||
C_OK,
|
||||
C_SAND,
|
||||
C_STEEL,
|
||||
ui_font,
|
||||
)
|
||||
|
||||
|
||||
class Step08Confirm(QWizardPage):
|
||||
def __init__(self, parent: QWidget | None = None) -> None:
|
||||
super().__init__(parent)
|
||||
self.setTitle("Paso 8 · Confirmación")
|
||||
self.setSubTitle(
|
||||
"Revisa el resumen del proyecto. Pulsa «Crear proyecto» para abrirlo en el editor."
|
||||
)
|
||||
|
||||
layout = QVBoxLayout(self)
|
||||
layout.setSpacing(16)
|
||||
|
||||
self._summary_card = QLabel()
|
||||
self._summary_card.setWordWrap(True)
|
||||
self._summary_card.setTextFormat(Qt.RichText)
|
||||
self._summary_card.setFont(ui_font(11))
|
||||
self._summary_card.setStyleSheet(
|
||||
f"""
|
||||
background: {C_MIDNIGHT};
|
||||
border: 1px solid {C_STEEL};
|
||||
border-radius: 12px;
|
||||
padding: 20px;
|
||||
color: {C_SAND};
|
||||
"""
|
||||
)
|
||||
layout.addWidget(self._summary_card, 1)
|
||||
|
||||
footer = QLabel(
|
||||
"Una vez creado el proyecto se abre en el editor principal del Studio. "
|
||||
"Podrás agregar equipos, tags y tarjetas, y guardar el .vmsproj cuando quieras.\n\n"
|
||||
"Sprint 2 traerá: editor de equipos, motor de reglas, asignación de I/O."
|
||||
)
|
||||
footer.setWordWrap(True)
|
||||
footer.setStyleSheet(f"color: {C_FOG};")
|
||||
footer.setFont(ui_font(9))
|
||||
layout.addWidget(footer)
|
||||
|
||||
def initializePage(self) -> None:
|
||||
from vmssailor.studio.wizard.wizard import (
|
||||
F_BEAM,
|
||||
F_BULKHEAD_AFT,
|
||||
F_BULKHEAD_FWD,
|
||||
F_CUSTOMER,
|
||||
F_DRAFT,
|
||||
F_LOA,
|
||||
F_PROJECT_NAME,
|
||||
F_SYSTEMS,
|
||||
F_TEMPLATE_ID,
|
||||
F_VESSEL_NAME,
|
||||
F_VESSEL_SUBTYPE,
|
||||
F_VESSEL_TYPE,
|
||||
)
|
||||
|
||||
project_name = self.field(F_PROJECT_NAME) or "(sin nombre)"
|
||||
customer = self.field(F_CUSTOMER) or "(sin cliente)"
|
||||
v_type = self.field(F_VESSEL_TYPE) or "—"
|
||||
v_sub = self.field(F_VESSEL_SUBTYPE) or "—"
|
||||
template = self.field(F_TEMPLATE_ID) or "—"
|
||||
v_name = self.field(F_VESSEL_NAME) or "—"
|
||||
loa = self.field(F_LOA) or 0.0
|
||||
beam = self.field(F_BEAM) or 0.0
|
||||
draft = self.field(F_DRAFT) or 0.0
|
||||
bh_aft = self.field(F_BULKHEAD_AFT) or 0.0
|
||||
bh_fwd = self.field(F_BULKHEAD_FWD) or 0.0
|
||||
systems = self.field(F_SYSTEMS) or []
|
||||
|
||||
n_sys = len(systems)
|
||||
systems_str = ", ".join(systems[:10])
|
||||
if n_sys > 10:
|
||||
systems_str += f", … (+{n_sys - 10})"
|
||||
if not systems_str:
|
||||
systems_str = "(ninguno seleccionado)"
|
||||
|
||||
html = f"""
|
||||
<h2 style="color:{C_FOAM}; font-family: sans-serif; margin: 0 0 16px 0;">
|
||||
{project_name}
|
||||
</h2>
|
||||
<p style="color:{C_FOG}; margin: 0 0 18px 0;">
|
||||
Cliente: <span style="color:{C_SAND};">{customer}</span>
|
||||
</p>
|
||||
|
||||
<div style="display: flex; gap: 28px; margin-bottom: 16px;">
|
||||
<div>
|
||||
<div style="color:{C_FOG}; font-size: 11px; text-transform: uppercase;
|
||||
letter-spacing: 2px;">Buque</div>
|
||||
<div style="color:{C_FOAM}; font-size: 14px; margin-top: 6px;">{v_name}</div>
|
||||
<div style="color:{C_SAND}; font-size: 12px; margin-top: 2px;">
|
||||
{v_type} · {v_sub}
|
||||
</div>
|
||||
<div style="color:{C_FOG}; font-size: 11px; margin-top: 2px;">
|
||||
Plantilla: {template}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<table cellpadding="6" style="color:{C_SAND}; font-family: monospace;
|
||||
border-collapse: collapse; margin-top: 12px;">
|
||||
<tr><td style="color:{C_FOG};">Eslora</td><td>{loa:.1f} m</td></tr>
|
||||
<tr><td style="color:{C_FOG};">Manga</td><td>{beam:.1f} m</td></tr>
|
||||
<tr><td style="color:{C_FOG};">Calado</td><td>{draft:.1f} m</td></tr>
|
||||
<tr><td style="color:{C_FOG};">Mamparo popa SM</td><td>x_pp {bh_aft:.1f} m</td></tr>
|
||||
<tr><td style="color:{C_FOG};">Mamparo proa SM</td><td>x_pp {bh_fwd:.1f} m</td></tr>
|
||||
</table>
|
||||
|
||||
<hr style="border: none; border-top: 1px solid {C_STEEL}; margin: 16px 0;">
|
||||
|
||||
<div style="color:{C_FOG}; font-size: 11px; text-transform: uppercase;
|
||||
letter-spacing: 2px;">Sistemas habilitados ({n_sys})</div>
|
||||
<div style="color:{C_SAND}; font-family: monospace; font-size: 11px;
|
||||
margin-top: 8px;">{systems_str}</div>
|
||||
|
||||
<p style="color:{C_OK}; font-size: 12px; margin-top: 20px;">
|
||||
✓ El proyecto se creará sin equipos. Agrega equipos manualmente o
|
||||
espera Sprint 2 (motor de reglas).
|
||||
</p>
|
||||
"""
|
||||
self._summary_card.setText(html)
|
||||
Reference in New Issue
Block a user