- 월간 캘린더 뷰 (사원별 필터, 날짜 클릭 등록, HTMX 월 이동) - 일괄 등록 (다수 사원 체크박스 선택 후 일괄 등록, upsert 처리) - 사원별 월간 요약 (상태별 카운트 + 총 근무시간 집계 테이블) - 초과근무 알림 (주 48h 경고 / 52h 위험 배너) - 근태 승인 워크플로우 (신청→승인→근태 레코드 자동 생성) - 자동 결근 처리 (매일 23:50 스케줄러, 주말 제외) - 연차 관리 연동 (휴가 등록 시 leave_balances 자동 차감) - GPS 출퇴근 UI (테이블 GPS 아이콘 + 상세 모달) - 탭 네비게이션 (목록/캘린더/요약/승인) HTMX 기반 전환
142 lines
4.4 KiB
PHP
142 lines
4.4 KiB
PHP
<?php
|
|
|
|
namespace App\Http\Controllers\Api\Admin\HR;
|
|
|
|
use App\Http\Controllers\Controller;
|
|
use App\Services\HR\AttendanceRequestService;
|
|
use Illuminate\Http\JsonResponse;
|
|
use Illuminate\Http\Request;
|
|
use Illuminate\Http\Response;
|
|
|
|
class AttendanceRequestController extends Controller
|
|
{
|
|
public function __construct(
|
|
private AttendanceRequestService $requestService
|
|
) {}
|
|
|
|
/**
|
|
* 신청 목록 (HTMX → HTML / 일반 → JSON)
|
|
*/
|
|
public function index(Request $request): JsonResponse|Response
|
|
{
|
|
$requests = $this->requestService->getRequests(
|
|
$request->all(),
|
|
$request->integer('per_page', 20)
|
|
);
|
|
|
|
if ($request->header('HX-Request')) {
|
|
return response(view('hr.attendances.partials.requests', compact('requests')));
|
|
}
|
|
|
|
return response()->json([
|
|
'success' => true,
|
|
'data' => $requests->items(),
|
|
'meta' => [
|
|
'current_page' => $requests->currentPage(),
|
|
'last_page' => $requests->lastPage(),
|
|
'per_page' => $requests->perPage(),
|
|
'total' => $requests->total(),
|
|
],
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* 신청 등록
|
|
*/
|
|
public function store(Request $request): JsonResponse
|
|
{
|
|
$validated = $request->validate([
|
|
'user_id' => 'required|integer|exists:users,id',
|
|
'request_type' => 'required|string|in:vacation,businessTrip,remote,fieldWork',
|
|
'start_date' => 'required|date',
|
|
'end_date' => 'required|date|after_or_equal:start_date',
|
|
'reason' => 'nullable|string|max:1000',
|
|
'json_details' => 'nullable|array',
|
|
]);
|
|
|
|
try {
|
|
$attendanceRequest = $this->requestService->storeRequest($validated);
|
|
|
|
return response()->json([
|
|
'success' => true,
|
|
'message' => '근태 신청이 등록되었습니다.',
|
|
'data' => $attendanceRequest,
|
|
], 201);
|
|
} catch (\Throwable $e) {
|
|
report($e);
|
|
|
|
return response()->json([
|
|
'success' => false,
|
|
'message' => '신청 등록 중 오류가 발생했습니다.',
|
|
'error' => config('app.debug') ? $e->getMessage() : null,
|
|
], 500);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 승인
|
|
*/
|
|
public function approve(Request $request, int $id): JsonResponse
|
|
{
|
|
try {
|
|
$attendanceRequest = $this->requestService->approve($id);
|
|
|
|
if (! $attendanceRequest) {
|
|
return response()->json([
|
|
'success' => false,
|
|
'message' => '대기 중인 신청을 찾을 수 없습니다.',
|
|
], 404);
|
|
}
|
|
|
|
return response()->json([
|
|
'success' => true,
|
|
'message' => '승인 처리되었습니다. 근태 레코드가 자동 생성되었습니다.',
|
|
'data' => $attendanceRequest,
|
|
]);
|
|
} catch (\Throwable $e) {
|
|
report($e);
|
|
|
|
return response()->json([
|
|
'success' => false,
|
|
'message' => '승인 처리 중 오류가 발생했습니다.',
|
|
'error' => config('app.debug') ? $e->getMessage() : null,
|
|
], 500);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 반려
|
|
*/
|
|
public function reject(Request $request, int $id): JsonResponse
|
|
{
|
|
$validated = $request->validate([
|
|
'reject_reason' => 'nullable|string|max:1000',
|
|
]);
|
|
|
|
try {
|
|
$attendanceRequest = $this->requestService->reject($id, $validated['reject_reason'] ?? null);
|
|
|
|
if (! $attendanceRequest) {
|
|
return response()->json([
|
|
'success' => false,
|
|
'message' => '대기 중인 신청을 찾을 수 없습니다.',
|
|
], 404);
|
|
}
|
|
|
|
return response()->json([
|
|
'success' => true,
|
|
'message' => '반려 처리되었습니다.',
|
|
'data' => $attendanceRequest,
|
|
]);
|
|
} catch (\Throwable $e) {
|
|
report($e);
|
|
|
|
return response()->json([
|
|
'success' => false,
|
|
'message' => '반려 처리 중 오류가 발생했습니다.',
|
|
'error' => config('app.debug') ? $e->getMessage() : null,
|
|
], 500);
|
|
}
|
|
}
|
|
}
|