feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
namespace App\Http\Controllers;
|
|
|
|
|
|
2026-03-04 20:29:25 +09:00
|
|
|
use App\Models\Finance\BankAccount;
|
|
|
|
|
use App\Models\Finance\CorporateCard;
|
2026-03-06 20:48:01 +09:00
|
|
|
use App\Models\Tenants\Tenant;
|
|
|
|
|
use App\Models\Tenants\TenantSetting;
|
feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
use App\Services\ApprovalService;
|
2026-03-05 15:57:36 +09:00
|
|
|
use App\Services\HR\LeaveService;
|
feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
use Illuminate\Http\Request;
|
|
|
|
|
use Illuminate\Http\Response;
|
|
|
|
|
use Illuminate\View\View;
|
|
|
|
|
|
|
|
|
|
class ApprovalController extends Controller
|
|
|
|
|
{
|
|
|
|
|
public function __construct(
|
|
|
|
|
private readonly ApprovalService $service
|
|
|
|
|
) {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 기안함
|
|
|
|
|
*/
|
|
|
|
|
public function drafts(Request $request): View|Response
|
|
|
|
|
{
|
|
|
|
|
if ($request->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('approvals.drafts'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return view('approvals.drafts');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 기안 작성
|
|
|
|
|
*/
|
|
|
|
|
public function create(Request $request): View|Response
|
|
|
|
|
{
|
|
|
|
|
if ($request->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('approvals.create'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$forms = $this->service->getApprovalForms();
|
|
|
|
|
$lines = $this->service->getApprovalLines();
|
2026-03-04 20:29:25 +09:00
|
|
|
[$cards, $accounts] = $this->getCardAndAccountData();
|
2026-03-05 15:57:36 +09:00
|
|
|
$employees = app(LeaveService::class)->getActiveEmployees();
|
2026-03-06 20:48:01 +09:00
|
|
|
$tenantInfo = $this->getTenantInfo();
|
feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
|
2026-03-06 20:48:01 +09:00
|
|
|
return view('approvals.create', compact('forms', 'lines', 'cards', 'accounts', 'employees', 'tenantInfo'));
|
feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 기안 수정
|
|
|
|
|
*/
|
|
|
|
|
public function edit(Request $request, int $id): View|Response
|
|
|
|
|
{
|
|
|
|
|
if ($request->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('approvals.edit', $id));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$approval = $this->service->getApproval($id);
|
|
|
|
|
|
|
|
|
|
if (! $approval->isEditable() || $approval->drafter_id !== auth()->id()) {
|
|
|
|
|
abort(403, '수정할 수 없습니다.');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$forms = $this->service->getApprovalForms();
|
|
|
|
|
$lines = $this->service->getApprovalLines();
|
2026-03-04 20:29:25 +09:00
|
|
|
[$cards, $accounts] = $this->getCardAndAccountData();
|
2026-03-05 18:53:42 +09:00
|
|
|
$employees = app(LeaveService::class)->getActiveEmployees();
|
2026-03-06 20:48:01 +09:00
|
|
|
$tenantInfo = $this->getTenantInfo();
|
feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
|
2026-03-06 20:48:01 +09:00
|
|
|
return view('approvals.edit', compact('approval', 'forms', 'lines', 'cards', 'accounts', 'employees', 'tenantInfo'));
|
feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 결재 상세
|
|
|
|
|
*/
|
|
|
|
|
public function show(Request $request, int $id): View|Response
|
|
|
|
|
{
|
|
|
|
|
if ($request->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('approvals.show', $id));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$approval = $this->service->getApproval($id);
|
|
|
|
|
|
|
|
|
|
return view('approvals.show', compact('approval'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 결재 대기함
|
|
|
|
|
*/
|
|
|
|
|
public function pending(Request $request): View|Response
|
|
|
|
|
{
|
|
|
|
|
if ($request->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('approvals.pending'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return view('approvals.pending');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 참조함
|
|
|
|
|
*/
|
|
|
|
|
public function references(Request $request): View|Response
|
|
|
|
|
{
|
|
|
|
|
if ($request->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('approvals.references'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return view('approvals.references');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 완료함
|
|
|
|
|
*/
|
|
|
|
|
public function completed(Request $request): View|Response
|
|
|
|
|
{
|
|
|
|
|
if ($request->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('approvals.completed'));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return view('approvals.completed');
|
|
|
|
|
}
|
2026-03-04 20:29:25 +09:00
|
|
|
|
2026-03-06 20:48:01 +09:00
|
|
|
private function getTenantInfo(): array
|
|
|
|
|
{
|
|
|
|
|
$tenantId = session('selected_tenant_id');
|
|
|
|
|
$tenant = Tenant::find($tenantId);
|
|
|
|
|
|
|
|
|
|
if (! $tenant) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$displaySetting = TenantSetting::withoutGlobalScopes()
|
|
|
|
|
->where('tenant_id', $tenantId)
|
|
|
|
|
->where('setting_group', 'company')
|
|
|
|
|
->where('setting_key', 'display_company_name')
|
|
|
|
|
->first();
|
|
|
|
|
$displayName = $displaySetting?->setting_value ?? '';
|
|
|
|
|
|
|
|
|
|
return [
|
|
|
|
|
'company_name' => ! empty($displayName) ? $displayName : ($tenant->company_name ?? ''),
|
|
|
|
|
'business_num' => $tenant->business_num ?? '',
|
|
|
|
|
'ceo_name' => $tenant->ceo_name ?? '',
|
|
|
|
|
'address' => $tenant->address ?? '',
|
|
|
|
|
'phone' => $tenant->phone ?? '',
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-04 20:29:25 +09:00
|
|
|
private function getCardAndAccountData(): array
|
|
|
|
|
{
|
2026-03-04 21:10:56 +09:00
|
|
|
$tenantId = session('selected_tenant_id');
|
2026-03-04 20:29:25 +09:00
|
|
|
|
|
|
|
|
$cards = CorporateCard::forTenant($tenantId)
|
|
|
|
|
->active()
|
2026-03-04 21:10:56 +09:00
|
|
|
->where('card_name', 'not like', '%하이패스%')
|
2026-03-04 20:29:25 +09:00
|
|
|
->select('id', 'card_name', 'card_company', 'card_number', 'card_holder_name')
|
|
|
|
|
->get();
|
|
|
|
|
|
2026-03-05 20:28:55 +09:00
|
|
|
$accounts = BankAccount::where('tenant_id', $tenantId)
|
|
|
|
|
->active()
|
2026-03-04 20:29:25 +09:00
|
|
|
->ordered()
|
|
|
|
|
->select('id', 'bank_name', 'account_number', 'account_holder', 'is_primary')
|
|
|
|
|
->get();
|
|
|
|
|
|
|
|
|
|
return [$cards, $accounts];
|
|
|
|
|
}
|
feat: [approval] 결재관리 Phase 1 MVP 구현
- 모델 4개: Approval, ApprovalStep, ApprovalForm, ApprovalLine
- ApprovalService: 목록/CRUD/워크플로우(상신/승인/반려/회수) 비즈니스 로직
- ApprovalApiController: JSON API 엔드포인트 (기안함/결재함/완료함/참조함)
- ApprovalController: Blade 뷰 컨트롤러 (HX-Redirect 처리)
- 뷰 8개: drafts, pending, completed, references, create, edit, show
- partials: _status-badge, _step-progress, _approval-line-editor
- api.php/web.php 라우트 등록
2026-02-27 23:17:17 +09:00
|
|
|
}
|