feat(map): listas de elementos e inspecciones con tablas Rappasoft
- FeatureTable e InspectionTable (DataTableComponent) sustituyen las tablas HTML de las pestañas "Elementos" e "Inspecciones" del mapa: búsqueda, orden, filtro (progreso) y acciones. - Selección de elemento e "ver inspección" se comunican al ProjectMap por eventos (map-select-feature / map-view-inspection, vía #[On]); seleccionar abre el panel de edición y centra el mapa igual que antes. - Las relaciones requieren sus FKs en additionalSelects (layer_id / feature_id, template_id, user_id) porque Rappasoft no selecciona '*'. Tests: MapTablesTest (3). Suite 74 passing. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,92 @@
|
||||
<?php
|
||||
|
||||
namespace App\Livewire\Projects;
|
||||
|
||||
use App\Models\Feature;
|
||||
use App\Models\Project;
|
||||
use Illuminate\Database\Eloquent\Builder;
|
||||
use Illuminate\Support\Facades\Auth;
|
||||
use Rappasoft\LaravelLivewireTables\DataTableComponent;
|
||||
use Rappasoft\LaravelLivewireTables\Views\Column;
|
||||
use Rappasoft\LaravelLivewireTables\Views\Filters\SelectFilter;
|
||||
|
||||
class FeatureTable extends DataTableComponent
|
||||
{
|
||||
protected $model = Feature::class;
|
||||
|
||||
public int $projectId;
|
||||
|
||||
public function configure(): void
|
||||
{
|
||||
$this->setPrimaryKey('id')
|
||||
->setDefaultSort('name', 'asc')
|
||||
->setSortingPillsEnabled(false)
|
||||
->setAdditionalSelects(['features.id as id', 'features.layer_id as layer_id']);
|
||||
}
|
||||
|
||||
public function builder(): Builder
|
||||
{
|
||||
$user = Auth::user();
|
||||
abort_unless(
|
||||
$user->can('manage all') ||
|
||||
Project::whereKey($this->projectId)->whereHas('users', fn ($q) => $q->where('user_id', $user->id))->exists(),
|
||||
403
|
||||
);
|
||||
|
||||
return Feature::query()
|
||||
->whereHas('layer.phase', fn ($q) => $q->where('project_id', $this->projectId))
|
||||
->with(['layer.phase']);
|
||||
}
|
||||
|
||||
public function columns(): array
|
||||
{
|
||||
return [
|
||||
Column::make('Elemento', 'name')
|
||||
->sortable()
|
||||
->searchable()
|
||||
->format(fn ($value) => '<span class="font-medium">' . e($value) . '</span>')
|
||||
->html(),
|
||||
|
||||
Column::make('Capa')
|
||||
->label(fn ($row) => e($row->layer?->name ?? '—')),
|
||||
|
||||
Column::make('Fase')
|
||||
->label(fn ($row) => e($row->layer?->phase?->name ?? '—')),
|
||||
|
||||
Column::make('Progreso', 'progress')
|
||||
->sortable()
|
||||
->format(function ($value) {
|
||||
$cls = $value >= 100 ? 'badge-success' : ($value > 0 ? 'badge-warning' : 'badge-ghost');
|
||||
return '<span class="badge badge-sm ' . $cls . '">' . (int) $value . '%</span>';
|
||||
})
|
||||
->html(),
|
||||
|
||||
Column::make('Acciones')
|
||||
->label(fn ($row) =>
|
||||
'<div class="flex justify-end">
|
||||
<button wire:click="$dispatch(\'map-select-feature\', { id: ' . $row->id . ' })"
|
||||
class="btn btn-xs btn-primary gap-1" title="Editar elemento">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" class="w-3.5 h-3.5" fill="none" viewBox="0 0 24 24" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z"/></svg>
|
||||
Abrir
|
||||
</button>
|
||||
</div>')
|
||||
->html(),
|
||||
];
|
||||
}
|
||||
|
||||
public function filters(): array
|
||||
{
|
||||
return [
|
||||
SelectFilter::make('Progreso', 'progress')
|
||||
->options(['' => 'Progreso: todos', 'pending' => 'Sin iniciar', 'in_progress' => 'En curso', 'done' => 'Completado'])
|
||||
->filter(function (Builder $query, string $value) {
|
||||
match ($value) {
|
||||
'pending' => $query->where('features.progress', '=', 0),
|
||||
'in_progress' => $query->where('features.progress', '>', 0)->where('features.progress', '<', 100),
|
||||
'done' => $query->where('features.progress', '>=', 100),
|
||||
default => null,
|
||||
};
|
||||
}),
|
||||
];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user