c378ab5884
- PhaseList pasa a contenedor: botón "Agregar fase" + modal crear/editar con todos los parámetros (nombre, descripción, orden, color, progreso, fechas previstas y reales) y validación. Antes "Agregar fase" creaba directamente 'Nueva fase'. - PhaseTable (Rappasoft): orden, nombre+descripción, barra de progreso, fechas, color y acciones (editar abre el modal vía evento, actualizar progreso, eliminar); búsqueda y ordenación. Gateado por 'manage phases' + acceso al proyecto. Tests: PhaseManagementTest (4). Suite 65 passing (solo 2 pre-existentes sqlite). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
116 lines
7.5 KiB
PHP
116 lines
7.5 KiB
PHP
<div>
|
|
<div class="flex items-center justify-between gap-3 mb-4">
|
|
<div>
|
|
<h3 class="font-bold">{{ __('Phases') }}</h3>
|
|
<p class="text-sm text-base-content/60">Fases del proyecto y su progreso</p>
|
|
</div>
|
|
@can('manage phases')
|
|
<button wire:click="openForm()" class="btn btn-primary btn-sm gap-2">
|
|
<x-heroicon-o-plus class="w-4 h-4" /> {{ __('Add Phase') }}
|
|
</button>
|
|
@endcan
|
|
</div>
|
|
|
|
{{-- Tabla Rappasoft de fases --}}
|
|
<livewire:phase-table :project-id="$project->id" :key="'phase-table-'.$project->id" />
|
|
|
|
{{-- ================================================================
|
|
MODAL crear / editar fase
|
|
================================================================ --}}
|
|
@if($showForm)
|
|
<div class="fixed inset-0 z-40 bg-black/50" wire:click="closeForm"></div>
|
|
<div class="fixed inset-0 z-50 flex items-center justify-center p-4">
|
|
<div class="bg-base-100 rounded-box shadow-2xl w-full max-w-2xl max-h-[90vh] overflow-y-auto">
|
|
<div class="flex items-center justify-between p-5 border-b border-base-300">
|
|
<h3 class="text-lg font-bold">{{ $editingId ? 'Editar fase' : 'Nueva fase' }}</h3>
|
|
<button wire:click="closeForm" class="btn btn-sm btn-ghost btn-circle">
|
|
<x-heroicon-o-x-mark class="w-4 h-4" />
|
|
</button>
|
|
</div>
|
|
|
|
<form wire:submit.prevent="save" class="p-5 space-y-4">
|
|
{{-- Nombre --}}
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Nombre <span class="text-error">*</span></span></label>
|
|
<input type="text" wire:model="name" autofocus
|
|
class="input input-bordered w-full @error('name') input-error @enderror"
|
|
placeholder="Ej.: Cimentación" />
|
|
@error('name')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
|
|
{{-- Descripción --}}
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Descripción</span></label>
|
|
<textarea wire:model="description"
|
|
class="textarea textarea-bordered w-full h-20 resize-y"
|
|
placeholder="Detalle de la fase..."></textarea>
|
|
</div>
|
|
|
|
{{-- Orden + Color + Progreso --}}
|
|
<div class="grid grid-cols-1 sm:grid-cols-3 gap-4">
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Orden <span class="text-error">*</span></span></label>
|
|
<input type="number" min="0" wire:model="order"
|
|
class="input input-bordered w-full @error('order') input-error @enderror" />
|
|
@error('order')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Color</span></label>
|
|
<input type="color" wire:model="color"
|
|
class="input input-bordered w-full h-12 p-1 @error('color') input-error @enderror" />
|
|
@error('color')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Progreso (%)</span></label>
|
|
<input type="number" min="0" max="100" wire:model="progressPercent"
|
|
class="input input-bordered w-full @error('progressPercent') input-error @enderror" />
|
|
@error('progressPercent')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
</div>
|
|
|
|
{{-- Fechas previstas --}}
|
|
<div class="grid grid-cols-1 sm:grid-cols-2 gap-4">
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Inicio previsto</span></label>
|
|
<input type="date" wire:model="plannedStart"
|
|
class="input input-bordered w-full @error('plannedStart') input-error @enderror" />
|
|
@error('plannedStart')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Fin previsto</span></label>
|
|
<input type="date" wire:model="plannedEnd"
|
|
class="input input-bordered w-full @error('plannedEnd') input-error @enderror" />
|
|
@error('plannedEnd')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
</div>
|
|
|
|
{{-- Fechas reales --}}
|
|
<div class="grid grid-cols-1 sm:grid-cols-2 gap-4">
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Inicio real</span></label>
|
|
<input type="date" wire:model="actualStart"
|
|
class="input input-bordered w-full @error('actualStart') input-error @enderror" />
|
|
@error('actualStart')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
<div class="form-control">
|
|
<label class="label"><span class="label-text font-medium">Fin real</span></label>
|
|
<input type="date" wire:model="actualEnd"
|
|
class="input input-bordered w-full @error('actualEnd') input-error @enderror" />
|
|
@error('actualEnd')<label class="label"><span class="label-text-alt text-error">{{ $message }}</span></label>@enderror
|
|
</div>
|
|
</div>
|
|
|
|
<div class="flex items-center justify-end gap-3 pt-2 border-t border-base-300">
|
|
<button type="button" wire:click="closeForm" class="btn btn-ghost">Cancelar</button>
|
|
<button type="submit" class="btn btn-primary gap-2" wire:loading.attr="disabled" wire:target="save">
|
|
<span wire:loading.remove wire:target="save"><x-heroicon-o-check class="w-4 h-4" /></span>
|
|
<span wire:loading wire:target="save" class="loading loading-spinner loading-sm"></span>
|
|
{{ $editingId ? 'Actualizar fase' : 'Crear fase' }}
|
|
</button>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
</div>
|
|
@endif
|
|
</div>
|