feat: [roadmap] 중장기 계획 메뉴 및 전용 페이지 개발

- 모델: AdminRoadmapPlan, AdminRoadmapMilestone
- 서비스: RoadmapPlanService, RoadmapMilestoneService
- FormRequest: Store/Update Plan/Milestone 4개
- 컨트롤러: Blade(RoadmapController), API(Plan/Milestone) 3개
- 라우트: web.php, api.php에 roadmap 라우트 추가
- Blade 뷰: 대시보드, 목록, 생성, 수정, 상세, 파셜 테이블 6개
- HTMX 기반 필터링/페이지네이션, 마일스톤 인라인 추가/토글
This commit is contained in:
김보곤
2026-03-02 15:50:20 +09:00
parent 458e5f890a
commit f3f1416004
19 changed files with 2161 additions and 0 deletions

View File

@@ -0,0 +1,68 @@
<?php
namespace App\Http\Controllers\Api\Admin\Roadmap;
use App\Http\Controllers\Controller;
use App\Http\Requests\Roadmap\StoreMilestoneRequest;
use App\Http\Requests\Roadmap\UpdateMilestoneRequest;
use App\Services\Roadmap\RoadmapMilestoneService;
use Illuminate\Http\JsonResponse;
class RoadmapMilestoneController extends Controller
{
public function __construct(
private readonly RoadmapMilestoneService $milestoneService
) {}
public function byPlan(int $planId): JsonResponse
{
$milestones = $this->milestoneService->getMilestonesByPlan($planId);
return response()->json([
'success' => true,
'data' => $milestones,
]);
}
public function store(StoreMilestoneRequest $request): JsonResponse
{
$milestone = $this->milestoneService->createMilestone($request->validated());
return response()->json([
'success' => true,
'message' => '마일스톤이 추가되었습니다.',
'data' => $milestone,
]);
}
public function update(UpdateMilestoneRequest $request, int $id): JsonResponse
{
$this->milestoneService->updateMilestone($id, $request->validated());
return response()->json([
'success' => true,
'message' => '마일스톤이 수정되었습니다.',
]);
}
public function destroy(int $id): JsonResponse
{
$this->milestoneService->deleteMilestone($id);
return response()->json([
'success' => true,
'message' => '마일스톤이 삭제되었습니다.',
]);
}
public function toggle(int $id): JsonResponse
{
$milestone = $this->milestoneService->toggleStatus($id);
return response()->json([
'success' => true,
'message' => $milestone->status === 'completed' ? '마일스톤이 완료 처리되었습니다.' : '마일스톤이 미완료로 변경되었습니다.',
'data' => $milestone,
]);
}
}

View File

@@ -0,0 +1,130 @@
<?php
namespace App\Http\Controllers\Api\Admin\Roadmap;
use App\Http\Controllers\Controller;
use App\Http\Requests\Roadmap\StorePlanRequest;
use App\Http\Requests\Roadmap\UpdatePlanRequest;
use App\Services\Roadmap\RoadmapPlanService;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
use Illuminate\View\View;
class RoadmapPlanController extends Controller
{
public function __construct(
private readonly RoadmapPlanService $planService
) {}
public function index(Request $request): View|JsonResponse
{
$filters = $request->only([
'search', 'status', 'category', 'priority', 'phase',
'trashed', 'sort_by', 'sort_direction',
]);
$plans = $this->planService->getPlans($filters, 15);
if ($request->header('HX-Request')) {
return view('roadmap.plans.partials.table', compact('plans'));
}
return response()->json([
'success' => true,
'data' => $plans,
]);
}
public function stats(): JsonResponse
{
$stats = $this->planService->getStats();
return response()->json([
'success' => true,
'data' => $stats,
]);
}
public function timeline(Request $request): JsonResponse
{
$phase = $request->input('phase');
$timeline = $this->planService->getTimelineData($phase);
return response()->json([
'success' => true,
'data' => $timeline,
]);
}
public function show(int $id): JsonResponse
{
$plan = $this->planService->getPlanById($id, true);
if (! $plan) {
return response()->json([
'success' => false,
'message' => '계획을 찾을 수 없습니다.',
], 404);
}
return response()->json([
'success' => true,
'data' => $plan,
]);
}
public function store(StorePlanRequest $request): JsonResponse
{
$plan = $this->planService->createPlan($request->validated());
return response()->json([
'success' => true,
'message' => '계획이 생성되었습니다.',
'data' => $plan,
]);
}
public function update(UpdatePlanRequest $request, int $id): JsonResponse
{
$this->planService->updatePlan($id, $request->validated());
return response()->json([
'success' => true,
'message' => '계획이 수정되었습니다.',
]);
}
public function destroy(int $id): JsonResponse
{
$this->planService->deletePlan($id);
return response()->json([
'success' => true,
'message' => '계획이 삭제되었습니다.',
]);
}
public function restore(int $id): JsonResponse
{
$this->planService->restorePlan($id);
return response()->json([
'success' => true,
'message' => '계획이 복원되었습니다.',
]);
}
public function changeStatus(Request $request, int $id): JsonResponse
{
$validated = $request->validate([
'status' => 'required|in:planned,in_progress,completed,delayed,cancelled',
]);
$plan = $this->planService->changeStatus($id, $validated['status']);
return response()->json([
'success' => true,
'message' => '상태가 변경되었습니다.',
'data' => $plan,
]);
}
}

View File

@@ -0,0 +1,79 @@
<?php
namespace App\Http\Controllers;
use App\Models\Admin\AdminRoadmapPlan;
use App\Services\Roadmap\RoadmapPlanService;
use Illuminate\View\View;
class RoadmapController extends Controller
{
public function __construct(
private readonly RoadmapPlanService $planService
) {}
public function index(): View
{
$summary = $this->planService->getDashboardSummary();
$statuses = AdminRoadmapPlan::getStatuses();
$categories = AdminRoadmapPlan::getCategories();
$priorities = AdminRoadmapPlan::getPriorities();
$phases = AdminRoadmapPlan::getPhases();
return view('roadmap.index', compact(
'summary', 'statuses', 'categories', 'priorities', 'phases'
));
}
public function plans(): View
{
$statuses = AdminRoadmapPlan::getStatuses();
$categories = AdminRoadmapPlan::getCategories();
$priorities = AdminRoadmapPlan::getPriorities();
$phases = AdminRoadmapPlan::getPhases();
return view('roadmap.plans.index', compact('statuses', 'categories', 'priorities', 'phases'));
}
public function createPlan(): View
{
$statuses = AdminRoadmapPlan::getStatuses();
$categories = AdminRoadmapPlan::getCategories();
$priorities = AdminRoadmapPlan::getPriorities();
$phases = AdminRoadmapPlan::getPhases();
return view('roadmap.plans.create', compact('statuses', 'categories', 'priorities', 'phases'));
}
public function showPlan(int $id): View
{
$plan = $this->planService->getPlanById($id, true);
if (! $plan) {
abort(404, '계획을 찾을 수 없습니다.');
}
$statuses = AdminRoadmapPlan::getStatuses();
$categories = AdminRoadmapPlan::getCategories();
$priorities = AdminRoadmapPlan::getPriorities();
$phases = AdminRoadmapPlan::getPhases();
return view('roadmap.plans.show', compact(
'plan', 'statuses', 'categories', 'priorities', 'phases'
));
}
public function editPlan(int $id): View
{
$plan = $this->planService->getPlanById($id, true);
if (! $plan) {
abort(404, '계획을 찾을 수 없습니다.');
}
$statuses = AdminRoadmapPlan::getStatuses();
$categories = AdminRoadmapPlan::getCategories();
$priorities = AdminRoadmapPlan::getPriorities();
$phases = AdminRoadmapPlan::getPhases();
return view('roadmap.plans.edit', compact(
'plan', 'statuses', 'categories', 'priorities', 'phases'
));
}
}