- EstimateService, ItemService 기능 추가 - OrderService 공정 연동 개선 - SalaryService, ReceivablesService 수정 - HandoverReportService, SiteBriefingService 추가 - Pricing 서비스 추가 Co-Authored-By: Claude <noreply@anthropic.com>
269 lines
9.7 KiB
PHP
269 lines
9.7 KiB
PHP
<?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,
|
|
];
|
|
}
|
|
}
|