feat(API): Service 로직 개선
- EstimateService, ItemService 기능 추가 - OrderService 공정 연동 개선 - SalaryService, ReceivablesService 수정 - HandoverReportService, SiteBriefingService 추가 - Pricing 서비스 추가 Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
268
app/Services/SiteBriefingService.php
Normal file
268
app/Services/SiteBriefingService.php
Normal file
@@ -0,0 +1,268 @@
|
||||
<?php
|
||||
|
||||
namespace App\Services;
|
||||
|
||||
use App\Models\Tenants\SiteBriefing;
|
||||
use App\Services\Quote\QuoteService;
|
||||
use Illuminate\Contracts\Pagination\LengthAwarePaginator;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
|
||||
class SiteBriefingService extends Service
|
||||
{
|
||||
public function __construct(
|
||||
private QuoteService $quoteService
|
||||
) {}
|
||||
|
||||
/**
|
||||
* 현장설명회 목록 조회
|
||||
*/
|
||||
public function index(array $params): LengthAwarePaginator
|
||||
{
|
||||
$tenantId = $this->tenantId();
|
||||
|
||||
$query = SiteBriefing::query()
|
||||
->with(['partner:id,name', 'site:id,name'])
|
||||
->where('tenant_id', $tenantId);
|
||||
|
||||
// 검색
|
||||
if (! empty($params['search'])) {
|
||||
$search = $params['search'];
|
||||
$query->where(function ($q) use ($search) {
|
||||
$q->where('title', 'like', "%{$search}%")
|
||||
->orWhere('briefing_code', 'like', "%{$search}%")
|
||||
->orWhere('location', 'like', "%{$search}%")
|
||||
->orWhereHas('partner', function ($pq) use ($search) {
|
||||
$pq->where('name', 'like', "%{$search}%");
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
// 상태 필터
|
||||
if (! empty($params['status']) && $params['status'] !== 'all') {
|
||||
$query->where('status', $params['status']);
|
||||
}
|
||||
|
||||
// 입찰상태 필터
|
||||
if (! empty($params['bid_status']) && $params['bid_status'] !== 'all') {
|
||||
$query->where('bid_status', $params['bid_status']);
|
||||
}
|
||||
|
||||
// 거래처 필터
|
||||
if (! empty($params['partner_id'])) {
|
||||
$query->where('partner_id', $params['partner_id']);
|
||||
}
|
||||
|
||||
// 현장 필터
|
||||
if (! empty($params['site_id'])) {
|
||||
$query->where('site_id', $params['site_id']);
|
||||
}
|
||||
|
||||
// 날짜 범위 필터
|
||||
if (! empty($params['start_date'])) {
|
||||
$query->where('briefing_date', '>=', $params['start_date']);
|
||||
}
|
||||
if (! empty($params['end_date'])) {
|
||||
$query->where('briefing_date', '<=', $params['end_date']);
|
||||
}
|
||||
|
||||
// 정렬
|
||||
$sortBy = $params['sort_by'] ?? 'created_at';
|
||||
$sortDir = $params['sort_dir'] ?? 'desc';
|
||||
|
||||
$sortMapping = [
|
||||
'created_at' => 'created_at',
|
||||
'briefing_date' => 'briefing_date',
|
||||
'title' => 'title',
|
||||
];
|
||||
|
||||
$sortColumn = $sortMapping[$sortBy] ?? 'created_at';
|
||||
$query->orderBy($sortColumn, $sortDir);
|
||||
|
||||
$perPage = min($params['per_page'] ?? 20, 100);
|
||||
|
||||
return $query->paginate($perPage);
|
||||
}
|
||||
|
||||
/**
|
||||
* 현장설명회 상세 조회
|
||||
*/
|
||||
public function show(int $id): SiteBriefing
|
||||
{
|
||||
return SiteBriefing::query()
|
||||
->with(['partner:id,name', 'site:id,name,address', 'creator:id,name'])
|
||||
->where('tenant_id', $this->tenantId())
|
||||
->findOrFail($id);
|
||||
}
|
||||
|
||||
/**
|
||||
* 현장설명회 등록
|
||||
*/
|
||||
public function store(array $data): SiteBriefing
|
||||
{
|
||||
$tenantId = $this->tenantId();
|
||||
$userId = $this->apiUserId();
|
||||
|
||||
return DB::transaction(function () use ($data, $tenantId, $userId) {
|
||||
// 현설번호 자동 생성
|
||||
$briefingCode = SiteBriefing::generateBriefingCode($tenantId);
|
||||
|
||||
// 참석자 배열 처리
|
||||
$attendees = $data['attendees'] ?? null;
|
||||
$attendeeCount = is_array($attendees) ? count($attendees) : 0;
|
||||
|
||||
$siteBriefing = SiteBriefing::create([
|
||||
'tenant_id' => $tenantId,
|
||||
'briefing_code' => $briefingCode,
|
||||
'title' => $data['title'],
|
||||
'description' => $data['description'] ?? null,
|
||||
'partner_id' => $data['partner_id'] ?? null,
|
||||
'site_id' => $data['site_id'] ?? null,
|
||||
'briefing_date' => $data['briefing_date'],
|
||||
'briefing_time' => $data['briefing_time'] ?? null,
|
||||
'briefing_type' => $data['briefing_type'] ?? SiteBriefing::TYPE_OFFLINE,
|
||||
'location' => $data['location'] ?? null,
|
||||
'address' => $data['address'] ?? null,
|
||||
'status' => $data['status'] ?? SiteBriefing::STATUS_SCHEDULED,
|
||||
'bid_status' => $data['bid_status'] ?? SiteBriefing::BID_STATUS_PENDING,
|
||||
'bid_date' => $data['bid_date'] ?? null,
|
||||
'attendees' => $attendees,
|
||||
'attendee_count' => $attendeeCount,
|
||||
'attendance_status' => $data['attendance_status'] ?? SiteBriefing::ATTENDANCE_SCHEDULED,
|
||||
'site_count' => $data['site_count'] ?? 0,
|
||||
'construction_start_date' => $data['construction_start_date'] ?? null,
|
||||
'construction_end_date' => $data['construction_end_date'] ?? null,
|
||||
'vat_type' => $data['vat_type'] ?? SiteBriefing::VAT_EXCLUDED,
|
||||
'created_by' => $userId,
|
||||
]);
|
||||
|
||||
return $siteBriefing->load(['partner:id,name', 'site:id,name']);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 현장설명회 수정
|
||||
*
|
||||
* attendance_status가 'attended'(참석완료) 상태면 견적을 upsert합니다.
|
||||
* - 견적이 없으면: 신규 생성
|
||||
* - 견적이 있으면: 거래처, 현장 정보 등 동기화
|
||||
*/
|
||||
public function update(int $id, array $data): SiteBriefing
|
||||
{
|
||||
$userId = $this->apiUserId();
|
||||
|
||||
return DB::transaction(function () use ($id, $data, $userId) {
|
||||
$siteBriefing = SiteBriefing::query()
|
||||
->where('tenant_id', $this->tenantId())
|
||||
->findOrFail($id);
|
||||
|
||||
// 저장 후 참석완료 상태인지 확인
|
||||
$newAttendanceStatus = $data['attendance_status'] ?? $siteBriefing->attendance_status;
|
||||
$isAttended = $newAttendanceStatus === SiteBriefing::ATTENDANCE_ATTENDED;
|
||||
|
||||
$updateData = array_filter([
|
||||
'title' => $data['title'] ?? null,
|
||||
'description' => $data['description'] ?? null,
|
||||
'partner_id' => $data['partner_id'] ?? null,
|
||||
'site_id' => $data['site_id'] ?? null,
|
||||
'briefing_date' => $data['briefing_date'] ?? null,
|
||||
'briefing_time' => $data['briefing_time'] ?? null,
|
||||
'briefing_type' => $data['briefing_type'] ?? null,
|
||||
'location' => $data['location'] ?? null,
|
||||
'address' => $data['address'] ?? null,
|
||||
'status' => $data['status'] ?? null,
|
||||
'bid_status' => $data['bid_status'] ?? null,
|
||||
'bid_date' => $data['bid_date'] ?? null,
|
||||
'attendance_status' => $data['attendance_status'] ?? null,
|
||||
'site_count' => $data['site_count'] ?? null,
|
||||
'construction_start_date' => $data['construction_start_date'] ?? null,
|
||||
'construction_end_date' => $data['construction_end_date'] ?? null,
|
||||
'vat_type' => $data['vat_type'] ?? null,
|
||||
], fn ($v) => $v !== null);
|
||||
|
||||
// 참석자 배열 처리 (명시적으로 전달된 경우에만 업데이트)
|
||||
if (array_key_exists('attendees', $data)) {
|
||||
$attendees = $data['attendees'];
|
||||
$updateData['attendees'] = $attendees;
|
||||
$updateData['attendee_count'] = is_array($attendees) ? count($attendees) : 0;
|
||||
}
|
||||
|
||||
$updateData['updated_by'] = $userId;
|
||||
|
||||
$siteBriefing->update($updateData);
|
||||
|
||||
// 참석완료 상태면 견적 upsert (신규 생성 또는 정보 동기화)
|
||||
if ($isAttended) {
|
||||
$siteBriefing->load('partner'); // partner 정보 로드 (견적에 필요)
|
||||
$this->quoteService->upsertFromSiteBriefing($siteBriefing);
|
||||
}
|
||||
|
||||
return $siteBriefing->load(['partner:id,name', 'site:id,name']);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 현장설명회 삭제
|
||||
*/
|
||||
public function destroy(int $id): bool
|
||||
{
|
||||
return DB::transaction(function () use ($id) {
|
||||
$siteBriefing = SiteBriefing::query()
|
||||
->where('tenant_id', $this->tenantId())
|
||||
->findOrFail($id);
|
||||
|
||||
$siteBriefing->update(['deleted_by' => $this->apiUserId()]);
|
||||
|
||||
return $siteBriefing->delete();
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 현장설명회 일괄 삭제
|
||||
*/
|
||||
public function bulkDestroy(array $ids): int
|
||||
{
|
||||
return DB::transaction(function () use ($ids) {
|
||||
$userId = $this->apiUserId();
|
||||
$tenantId = $this->tenantId();
|
||||
|
||||
$siteBriefings = SiteBriefing::query()
|
||||
->where('tenant_id', $tenantId)
|
||||
->whereIn('id', $ids)
|
||||
->get();
|
||||
|
||||
foreach ($siteBriefings as $siteBriefing) {
|
||||
$siteBriefing->update(['deleted_by' => $userId]);
|
||||
$siteBriefing->delete();
|
||||
}
|
||||
|
||||
return $siteBriefings->count();
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 현장설명회 통계 조회
|
||||
*/
|
||||
public function stats(): array
|
||||
{
|
||||
$tenantId = $this->tenantId();
|
||||
|
||||
$stats = SiteBriefing::query()
|
||||
->where('tenant_id', $tenantId)
|
||||
->selectRaw("
|
||||
COUNT(*) as total,
|
||||
SUM(CASE WHEN status = 'scheduled' THEN 1 ELSE 0 END) as scheduled,
|
||||
SUM(CASE WHEN status = 'ongoing' THEN 1 ELSE 0 END) as ongoing,
|
||||
SUM(CASE WHEN status = 'completed' THEN 1 ELSE 0 END) as completed,
|
||||
SUM(CASE WHEN status IN ('cancelled', 'postponed') THEN 1 ELSE 0 END) as cancelled
|
||||
")
|
||||
->first();
|
||||
|
||||
return [
|
||||
'total' => (int) $stats->total,
|
||||
'scheduled' => (int) $stats->scheduled,
|
||||
'ongoing' => (int) $stats->ongoing,
|
||||
'completed' => (int) $stats->completed,
|
||||
'cancelled' => (int) $stats->cancelled,
|
||||
];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user