trip-planner/backend/app/Infrastructure/Http/Controllers/API/PlannableItem/PlannableItemController.php
myrmidex 935162ea70 Implement timeline scheduling feature with hour-based calendar slots
- Add CalendarSlotService methods for slot order calculation
  - Update PlannedItemController to create slots from datetime
  - Update CalendarSlotController with proper eager loading
  - Create timeline UI components (TripTimeline, DaySection, HourRow, ScheduleItemModal)
  - Add comprehensive PHPUnit tests (PlannedItemTest, CalendarSlotServiceTest)
  - Add comprehensive Selenium E2E tests (timeline-scheduling.test.js)
  - Update PlannablesList to support onItemsChange callback
2025-10-07 22:54:09 +02:00

88 lines
No EOL
2.8 KiB
PHP

<?php
namespace App\Infrastructure\Http\Controllers\API\PlannableItem;
use App\Domain\PlannableItem\Policies\PlannableItemPolicy;
use App\Infrastructure\Http\Controllers\Controller;
use App\Models\PlannableItem;
use App\Models\Trip;
use Illuminate\Http\Request;
use Illuminate\Http\JsonResponse;
class PlannableItemController extends Controller
{
public function __construct(
private PlannableItemPolicy $policy
) {}
public function index(Trip $trip): JsonResponse
{
if (!$this->policy->create(auth()->user(), $trip)) {
return response()->json(['message' => 'Forbidden'], 403);
}
$plannableItems = $trip->plannableItems()
->with(['calendarSlots'])
->get();
return response()->json(['data' => $plannableItems]);
}
public function store(Request $request, Trip $trip): JsonResponse
{
if (!$this->policy->create(auth()->user(), $trip)) {
return response()->json(['message' => 'Forbidden'], 403);
}
$validated = $request->validate([
'name' => 'required|string|max:255',
'type' => 'required|in:hotel,restaurant,attraction,transport,activity',
'address' => 'nullable|string|max:255',
'notes' => 'nullable|string',
'metadata' => 'nullable|array',
]);
$plannableItem = $trip->plannableItems()->create($validated);
return response()->json(['data' => $plannableItem], 201);
}
public function show(PlannableItem $plannableItem): JsonResponse
{
if (!$this->policy->view(auth()->user(), $plannableItem)) {
return response()->json(['message' => 'Forbidden'], 403);
}
$plannableItem->load(['calendarSlots', 'trip']);
return response()->json($plannableItem);
}
public function update(Request $request, PlannableItem $plannableItem): JsonResponse
{
if (!$this->policy->update(auth()->user(), $plannableItem)) {
return response()->json(['message' => 'Forbidden'], 403);
}
$validated = $request->validate([
'name' => 'sometimes|required|string|max:255',
'type' => 'sometimes|required|in:hotel,restaurant,attraction,transport,activity',
'address' => 'nullable|string|max:255',
'notes' => 'nullable|string',
'metadata' => 'nullable|array',
]);
$plannableItem->update($validated);
return response()->json(['data' => $plannableItem]);
}
public function destroy(PlannableItem $plannableItem): JsonResponse
{
if (!$this->policy->delete(auth()->user(), $plannableItem)) {
return response()->json(['message' => 'Forbidden'], 403);
}
$plannableItem->delete();
return response()->json(null, 204);
}
}