feat:영업파트너 승인 페이지 추가 (본사 관리자 전용)
- 승인 대기 목록 페이지 (/sales/managers/approvals) - isAdmin() 권한 체크: admin 또는 super_admin만 접근 가능 - 승인/반려 기능 (AJAX 지원) - 상세 보기 모달 - 승인 대기 통계 (오늘 승인/반려 수) - DB 메뉴 추가: 영업관리 > 영업파트너 승인 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -332,4 +332,81 @@ public function deleteDocument(int $id, int $documentId)
|
||||
return redirect()->back()
|
||||
->with('success', '서류가 삭제되었습니다.');
|
||||
}
|
||||
|
||||
/**
|
||||
* 영업파트너 승인 목록 (본사 관리자 전용)
|
||||
*/
|
||||
public function approvals(Request $request): View|Response
|
||||
{
|
||||
// 권한 체크: admin 역할만 접근 가능
|
||||
if (!auth()->user()->isAdmin()) {
|
||||
abort(403, '접근 권한이 없습니다.');
|
||||
}
|
||||
|
||||
if ($request->header('HX-Request')) {
|
||||
return response('', 200)->header('HX-Redirect', route('sales.managers.approvals'));
|
||||
}
|
||||
|
||||
$filters = [
|
||||
'search' => $request->get('search'),
|
||||
'approval_status' => 'pending', // 승인 대기만
|
||||
];
|
||||
|
||||
$partners = $this->service->getSalesPartners($filters)->paginate(20);
|
||||
$stats = $this->service->getApprovalStats();
|
||||
|
||||
return view('sales.managers.approvals', compact('partners', 'stats'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 승인 목록에서 승인 처리
|
||||
*/
|
||||
public function approveFromList(Request $request, int $id)
|
||||
{
|
||||
// 권한 체크
|
||||
if (!auth()->user()->isAdmin()) {
|
||||
abort(403, '접근 권한이 없습니다.');
|
||||
}
|
||||
|
||||
$partner = User::findOrFail($id);
|
||||
$this->service->approve($partner, auth()->id());
|
||||
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => "{$partner->name}님이 승인되었습니다.",
|
||||
]);
|
||||
}
|
||||
|
||||
return redirect()->route('sales.managers.approvals')
|
||||
->with('success', "{$partner->name}님이 승인되었습니다.");
|
||||
}
|
||||
|
||||
/**
|
||||
* 승인 목록에서 반려 처리
|
||||
*/
|
||||
public function rejectFromList(Request $request, int $id)
|
||||
{
|
||||
// 권한 체크
|
||||
if (!auth()->user()->isAdmin()) {
|
||||
abort(403, '접근 권한이 없습니다.');
|
||||
}
|
||||
|
||||
$validated = $request->validate([
|
||||
'rejection_reason' => 'required|string|max:1000',
|
||||
]);
|
||||
|
||||
$partner = User::findOrFail($id);
|
||||
$this->service->reject($partner, auth()->id(), $validated['rejection_reason']);
|
||||
|
||||
if ($request->header('HX-Request')) {
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'message' => "{$partner->name}님이 반려되었습니다.",
|
||||
]);
|
||||
}
|
||||
|
||||
return redirect()->route('sales.managers.approvals')
|
||||
->with('success', "{$partner->name}님이 반려되었습니다.");
|
||||
}
|
||||
}
|
||||
|
||||
@@ -491,6 +491,34 @@ public function getStats(?int $parentId = null): array
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* 승인 대기 통계 조회 (본사 관리자용)
|
||||
*/
|
||||
public function getApprovalStats(): array
|
||||
{
|
||||
$tenantId = session('selected_tenant_id', 1);
|
||||
|
||||
$baseQuery = User::query()
|
||||
->whereHas('userRoles', function ($q) use ($tenantId) {
|
||||
$q->where('tenant_id', $tenantId)
|
||||
->whereHas('role', function ($rq) {
|
||||
$rq->whereIn('name', self::SALES_ROLES);
|
||||
});
|
||||
});
|
||||
|
||||
return [
|
||||
'pending' => (clone $baseQuery)->where('approval_status', 'pending')->count(),
|
||||
'approved_today' => (clone $baseQuery)
|
||||
->where('approval_status', 'approved')
|
||||
->whereDate('approved_at', today())
|
||||
->count(),
|
||||
'rejected_today' => (clone $baseQuery)
|
||||
->where('approval_status', 'rejected')
|
||||
->whereDate('approved_at', today())
|
||||
->count(),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* 파트너의 계층 레벨 계산
|
||||
*/
|
||||
|
||||
335
resources/views/sales/managers/approvals.blade.php
Normal file
335
resources/views/sales/managers/approvals.blade.php
Normal file
@@ -0,0 +1,335 @@
|
||||
@extends('layouts.app')
|
||||
|
||||
@section('title', '영업파트너 승인')
|
||||
|
||||
@section('content')
|
||||
<div class="flex flex-col h-full">
|
||||
<!-- 페이지 헤더 -->
|
||||
<div class="flex flex-col sm:flex-row sm:justify-between sm:items-center gap-4 mb-6 flex-shrink-0">
|
||||
<div>
|
||||
<h1 class="text-2xl font-bold text-gray-800">영업파트너 승인</h1>
|
||||
<p class="text-sm text-gray-500 mt-1">영업파트너 가입 신청을 검토하고 승인/반려합니다</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 통계 카드 -->
|
||||
<div class="grid grid-cols-3 gap-4 mb-6 flex-shrink-0">
|
||||
<div class="bg-yellow-50 rounded-lg shadow-sm p-4">
|
||||
<div class="text-sm text-yellow-600">승인 대기</div>
|
||||
<div class="text-2xl font-bold text-yellow-800">{{ number_format($stats['pending']) }}명</div>
|
||||
</div>
|
||||
<div class="bg-emerald-50 rounded-lg shadow-sm p-4">
|
||||
<div class="text-sm text-emerald-600">오늘 승인</div>
|
||||
<div class="text-2xl font-bold text-emerald-800">{{ number_format($stats['approved_today']) }}명</div>
|
||||
</div>
|
||||
<div class="bg-red-50 rounded-lg shadow-sm p-4">
|
||||
<div class="text-sm text-red-600">오늘 반려</div>
|
||||
<div class="text-2xl font-bold text-red-800">{{ number_format($stats['rejected_today']) }}명</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 필터 영역 -->
|
||||
<div class="flex-shrink-0 mb-4">
|
||||
<form method="GET" class="flex flex-wrap gap-2 sm:gap-4 items-center bg-white p-4 rounded-lg shadow-sm">
|
||||
<div class="flex-1 min-w-0 w-full sm:w-auto">
|
||||
<input type="text"
|
||||
name="search"
|
||||
value="{{ request('search') }}"
|
||||
placeholder="이름, 아이디, 이메일, 전화번호로 검색..."
|
||||
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
||||
</div>
|
||||
<button type="submit" class="bg-gray-600 hover:bg-gray-700 text-white px-6 py-2 rounded-lg transition w-full sm:w-auto">
|
||||
검색
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<!-- 테이블 -->
|
||||
<div class="bg-white rounded-lg shadow-sm overflow-hidden flex-1 flex flex-col min-h-0">
|
||||
<div class="overflow-x-auto">
|
||||
<table class="min-w-full divide-y divide-gray-200">
|
||||
<thead class="bg-gray-50">
|
||||
<tr>
|
||||
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">신청자</th>
|
||||
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">아이디</th>
|
||||
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">역할</th>
|
||||
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">연락처</th>
|
||||
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">추천인(유치자)</th>
|
||||
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">신청일</th>
|
||||
<th class="px-6 py-3 text-center text-xs font-medium text-gray-500 uppercase tracking-wider">처리</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody class="bg-white divide-y divide-gray-200">
|
||||
@forelse($partners as $partner)
|
||||
<tr class="hover:bg-gray-50" id="partner-row-{{ $partner->id }}">
|
||||
<td class="px-6 py-4 whitespace-nowrap">
|
||||
<div class="font-medium text-gray-900">{{ $partner->name }}</div>
|
||||
@if($partner->email)
|
||||
<div class="text-sm text-gray-500">{{ $partner->email }}</div>
|
||||
@endif
|
||||
</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-900">
|
||||
{{ $partner->user_id ?? '-' }}
|
||||
</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap">
|
||||
<div class="flex flex-wrap gap-1">
|
||||
@foreach($partner->userRoles as $userRole)
|
||||
@php
|
||||
$roleColor = match($userRole->role->name ?? '') {
|
||||
'sales' => 'bg-blue-100 text-blue-800',
|
||||
'manager' => 'bg-purple-100 text-purple-800',
|
||||
'recruiter' => 'bg-green-100 text-green-800',
|
||||
default => 'bg-gray-100 text-gray-800',
|
||||
};
|
||||
$roleLabel = match($userRole->role->name ?? '') {
|
||||
'sales' => '영업',
|
||||
'manager' => '매니저',
|
||||
'recruiter' => '유치담당',
|
||||
default => $userRole->role->name ?? '-',
|
||||
};
|
||||
@endphp
|
||||
<span class="px-2 py-1 text-xs font-medium rounded-full {{ $roleColor }}">
|
||||
{{ $roleLabel }}
|
||||
</span>
|
||||
@endforeach
|
||||
</div>
|
||||
</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
|
||||
{{ $partner->phone ?? '-' }}
|
||||
</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
|
||||
@if($partner->parent)
|
||||
<span class="text-blue-600">{{ $partner->parent->name }}</span>
|
||||
@else
|
||||
<span class="text-gray-400">-</span>
|
||||
@endif
|
||||
</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
|
||||
{{ $partner->created_at->format('Y-m-d H:i') }}
|
||||
</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap text-center">
|
||||
<div class="flex items-center justify-center gap-2">
|
||||
<button type="button"
|
||||
onclick="approvePartner({{ $partner->id }}, '{{ $partner->name }}')"
|
||||
class="px-3 py-1.5 bg-emerald-500 hover:bg-emerald-600 text-white text-sm font-medium rounded-lg transition">
|
||||
승인
|
||||
</button>
|
||||
<button type="button"
|
||||
onclick="openRejectModal({{ $partner->id }}, '{{ $partner->name }}')"
|
||||
class="px-3 py-1.5 bg-red-500 hover:bg-red-600 text-white text-sm font-medium rounded-lg transition">
|
||||
반려
|
||||
</button>
|
||||
<button type="button"
|
||||
onclick="openDetailModal({{ $partner->id }})"
|
||||
class="px-3 py-1.5 bg-gray-500 hover:bg-gray-600 text-white text-sm font-medium rounded-lg transition">
|
||||
상세
|
||||
</button>
|
||||
</div>
|
||||
</td>
|
||||
</tr>
|
||||
@empty
|
||||
<tr>
|
||||
<td colspan="7" class="px-6 py-12 text-center text-gray-500">
|
||||
승인 대기 중인 영업파트너가 없습니다.
|
||||
</td>
|
||||
</tr>
|
||||
@endforelse
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
<!-- 페이지네이션 -->
|
||||
@if($partners->hasPages())
|
||||
<div class="px-6 py-4 border-t border-gray-200">
|
||||
{{ $partners->withQueryString()->links() }}
|
||||
</div>
|
||||
@endif
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 반려 사유 모달 -->
|
||||
<div id="rejectModal" class="hidden fixed inset-0 z-50 overflow-y-auto">
|
||||
<div class="fixed inset-0 bg-black bg-opacity-50 transition-opacity"></div>
|
||||
<div class="flex min-h-full items-center justify-center p-4">
|
||||
<div class="relative bg-white rounded-xl shadow-2xl w-full max-w-md p-6">
|
||||
<h3 class="text-lg font-semibold text-gray-900 mb-4">반려 사유 입력</h3>
|
||||
<p class="text-sm text-gray-600 mb-4">
|
||||
<span id="rejectPartnerName" class="font-medium text-red-600"></span>님의 가입 신청을 반려합니다.
|
||||
</p>
|
||||
<form id="rejectForm" onsubmit="submitReject(event)">
|
||||
<input type="hidden" id="rejectPartnerId" name="partner_id">
|
||||
<div class="mb-4">
|
||||
<label class="block text-sm font-medium text-gray-700 mb-1">반려 사유 <span class="text-red-500">*</span></label>
|
||||
<textarea name="rejection_reason" id="rejectionReason" rows="4" required
|
||||
class="w-full px-3 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-red-500"
|
||||
placeholder="반려 사유를 입력하세요..."></textarea>
|
||||
</div>
|
||||
<div class="flex justify-end gap-3">
|
||||
<button type="button" onclick="closeRejectModal()"
|
||||
class="px-4 py-2 border border-gray-300 rounded-lg text-gray-700 hover:bg-gray-50 transition">
|
||||
취소
|
||||
</button>
|
||||
<button type="submit"
|
||||
class="px-4 py-2 bg-red-600 hover:bg-red-700 text-white rounded-lg transition">
|
||||
반려 처리
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 상세 모달 -->
|
||||
<div id="detailModal" class="hidden fixed inset-0 z-50 overflow-y-auto">
|
||||
<div class="fixed inset-0 bg-black bg-opacity-50 transition-opacity" onclick="closeDetailModal()"></div>
|
||||
<div class="flex min-h-full items-center justify-center p-4">
|
||||
<div id="detailModalContent" class="relative bg-white rounded-xl shadow-2xl w-full max-w-2xl">
|
||||
<div class="p-6 text-center">
|
||||
<svg class="w-8 h-8 animate-spin text-blue-600 mx-auto" fill="none" viewBox="0 0 24 24">
|
||||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z"></path>
|
||||
</svg>
|
||||
<p class="mt-2 text-gray-500">로딩 중...</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@endsection
|
||||
|
||||
@push('scripts')
|
||||
<script>
|
||||
// 승인 처리
|
||||
function approvePartner(id, name) {
|
||||
showConfirm(
|
||||
`<strong>${name}</strong>님의 가입 신청을 승인하시겠습니까?`,
|
||||
() => {
|
||||
fetch(`/sales/managers/approvals/${id}/approve`, {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
'X-CSRF-TOKEN': '{{ csrf_token() }}',
|
||||
'Accept': 'application/json',
|
||||
'HX-Request': 'true'
|
||||
}
|
||||
})
|
||||
.then(response => response.json())
|
||||
.then(data => {
|
||||
if (data.success) {
|
||||
showToast(data.message, 'success');
|
||||
// 테이블에서 해당 행 제거
|
||||
document.getElementById(`partner-row-${id}`)?.remove();
|
||||
// 테이블이 비었으면 새로고침
|
||||
const tbody = document.querySelector('tbody');
|
||||
if (tbody && tbody.children.length === 0) {
|
||||
window.location.reload();
|
||||
}
|
||||
} else {
|
||||
showToast(data.message || '승인 처리에 실패했습니다.', 'error');
|
||||
}
|
||||
})
|
||||
.catch(error => {
|
||||
showToast('서버 오류가 발생했습니다.', 'error');
|
||||
console.error(error);
|
||||
});
|
||||
},
|
||||
{ title: '승인 확인', icon: 'question', confirmText: '승인' }
|
||||
);
|
||||
}
|
||||
|
||||
// 반려 모달 열기
|
||||
function openRejectModal(id, name) {
|
||||
document.getElementById('rejectPartnerId').value = id;
|
||||
document.getElementById('rejectPartnerName').textContent = name;
|
||||
document.getElementById('rejectionReason').value = '';
|
||||
document.getElementById('rejectModal').classList.remove('hidden');
|
||||
}
|
||||
|
||||
// 반려 모달 닫기
|
||||
function closeRejectModal() {
|
||||
document.getElementById('rejectModal').classList.add('hidden');
|
||||
}
|
||||
|
||||
// 반려 제출
|
||||
function submitReject(event) {
|
||||
event.preventDefault();
|
||||
|
||||
const id = document.getElementById('rejectPartnerId').value;
|
||||
const reason = document.getElementById('rejectionReason').value;
|
||||
|
||||
fetch(`/sales/managers/approvals/${id}/reject`, {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
'X-CSRF-TOKEN': '{{ csrf_token() }}',
|
||||
'Accept': 'application/json',
|
||||
'HX-Request': 'true'
|
||||
},
|
||||
body: JSON.stringify({ rejection_reason: reason })
|
||||
})
|
||||
.then(response => response.json())
|
||||
.then(data => {
|
||||
if (data.success) {
|
||||
closeRejectModal();
|
||||
showToast(data.message, 'success');
|
||||
document.getElementById(`partner-row-${id}`)?.remove();
|
||||
const tbody = document.querySelector('tbody');
|
||||
if (tbody && tbody.children.length === 0) {
|
||||
window.location.reload();
|
||||
}
|
||||
} else {
|
||||
showToast(data.message || '반려 처리에 실패했습니다.', 'error');
|
||||
}
|
||||
})
|
||||
.catch(error => {
|
||||
showToast('서버 오류가 발생했습니다.', 'error');
|
||||
console.error(error);
|
||||
});
|
||||
}
|
||||
|
||||
// 상세 모달 열기
|
||||
function openDetailModal(id) {
|
||||
document.getElementById('detailModal').classList.remove('hidden');
|
||||
document.getElementById('detailModalContent').innerHTML = `
|
||||
<div class="p-6 text-center">
|
||||
<svg class="w-8 h-8 animate-spin text-blue-600 mx-auto" fill="none" viewBox="0 0 24 24">
|
||||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z"></path>
|
||||
</svg>
|
||||
<p class="mt-2 text-gray-500">로딩 중...</p>
|
||||
</div>
|
||||
`;
|
||||
|
||||
fetch(`/sales/managers/${id}/modal-show`, {
|
||||
headers: {
|
||||
'X-Requested-With': 'XMLHttpRequest',
|
||||
'Accept': 'text/html'
|
||||
}
|
||||
})
|
||||
.then(response => response.text())
|
||||
.then(html => {
|
||||
document.getElementById('detailModalContent').innerHTML = html;
|
||||
})
|
||||
.catch(error => {
|
||||
document.getElementById('detailModalContent').innerHTML = `
|
||||
<div class="p-6 text-center">
|
||||
<p class="text-red-500">오류가 발생했습니다.</p>
|
||||
<button onclick="closeDetailModal()" class="mt-4 px-4 py-2 bg-gray-600 text-white rounded-lg">닫기</button>
|
||||
</div>
|
||||
`;
|
||||
});
|
||||
}
|
||||
|
||||
// 상세 모달 닫기
|
||||
function closeDetailModal() {
|
||||
document.getElementById('detailModal').classList.add('hidden');
|
||||
}
|
||||
|
||||
// ESC 키로 모달 닫기
|
||||
document.addEventListener('keydown', function(e) {
|
||||
if (e.key === 'Escape') {
|
||||
closeRejectModal();
|
||||
closeDetailModal();
|
||||
}
|
||||
});
|
||||
</script>
|
||||
@endpush
|
||||
@@ -868,6 +868,11 @@
|
||||
Route::get('salesmanagement/dashboard/refresh', [\App\Http\Controllers\Sales\SalesDashboardController::class, 'refresh'])->name('salesmanagement.dashboard.refresh');
|
||||
Route::get('salesmanagement/dashboard/tenants', [\App\Http\Controllers\Sales\SalesDashboardController::class, 'refreshTenantList'])->name('salesmanagement.dashboard.tenants');
|
||||
|
||||
// 영업파트너 승인 (본사 관리자 전용) - resource 전에 정의해야 함
|
||||
Route::get('managers/approvals', [\App\Http\Controllers\Sales\SalesManagerController::class, 'approvals'])->name('managers.approvals');
|
||||
Route::post('managers/approvals/{id}/approve', [\App\Http\Controllers\Sales\SalesManagerController::class, 'approveFromList'])->name('managers.approvals.approve');
|
||||
Route::post('managers/approvals/{id}/reject', [\App\Http\Controllers\Sales\SalesManagerController::class, 'rejectFromList'])->name('managers.approvals.reject');
|
||||
|
||||
// 영업 담당자 관리
|
||||
Route::resource('managers', \App\Http\Controllers\Sales\SalesManagerController::class);
|
||||
Route::get('managers/{id}/modal-show', [\App\Http\Controllers\Sales\SalesManagerController::class, 'modalShow'])->name('managers.modal-show');
|
||||
|
||||
Reference in New Issue
Block a user