feat:경동기업 견적/수주 전환 로직 개선

- KyungdongFormulaHandler: 수식 계산 로직 리팩토링 및 확장
- OrderService: 수주 전환 시 BOM 품목 매핑 로직 추가
- QuoteService: 견적 상태 처리 개선
- FormulaEvaluatorService: 디버그 로깅 추가
- Quote 모델: 캐스팅 타입 수정

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-02-05 21:58:49 +09:00
parent 9f2b1cf44a
commit f640a837e9
7 changed files with 386 additions and 85 deletions

View File

@@ -3,6 +3,7 @@
namespace App\Services;
use App\Models\Orders\Order;
use App\Models\Orders\OrderHistory;
use App\Models\Production\WorkOrder;
use App\Models\Quote\Quote;
use App\Models\Tenants\Sale;
@@ -470,19 +471,45 @@ public function createFromQuote(int $quoteId, array $data = [])
$order->save();
// calculation_inputs에서 제품 정보 추출 (floor, code)
// 단일 제품인 경우 모든 BOM 품목에 동일한 floor_code/symbol_code 적용
$calculationInputs = $quote->calculation_inputs ?? [];
$productItems = $calculationInputs['items'] ?? [];
// 견적 품목을 수주 품목으로 변환
foreach ($quote->items as $index => $quoteItem) {
// floor_code/symbol_code 추출:
// 1순위: quoteItem->note에서 파싱 (형식: "4F DS-01" → floor=4F, symbol=DS-01)
// 2순위: NULL
// 1순위: calculation_inputs.items[].floor, code (제품 정보)
// 2순위: quoteItem->note에서 파싱 (형식: "4F DS-01" → floor=4F, symbol=DS-01)
// 3순위: NULL
$floorCode = null;
$symbolCode = null;
$note = trim($quoteItem->note ?? '');
if ($note !== '') {
$parts = preg_split('/\s+/', $note, 2);
$floorCode = $parts[0] ?? null;
$symbolCode = $parts[1] ?? null;
// formula_source에서 제품 인덱스 추출 시도 (예: "product_0" → 0)
$productIndex = 0;
$formulaSource = $quoteItem->formula_source ?? '';
if (preg_match('/product_(\d+)/', $formulaSource, $matches)) {
$productIndex = (int) $matches[1];
}
// calculation_inputs에서 floor/code 가져오기
if (isset($productItems[$productIndex])) {
$floorCode = $productItems[$productIndex]['floor'] ?? null;
$symbolCode = $productItems[$productIndex]['code'] ?? null;
} elseif (count($productItems) === 1) {
// 단일 제품인 경우 첫 번째 제품 사용
$floorCode = $productItems[0]['floor'] ?? null;
$symbolCode = $productItems[0]['code'] ?? null;
}
// calculation_inputs에서 못 찾은 경우 note에서 파싱 시도
if (empty($floorCode) && empty($symbolCode)) {
$note = trim($quoteItem->note ?? '');
if ($note !== '') {
$parts = preg_split('/\s+/', $note, 2);
$floorCode = $parts[0] ?? null;
$symbolCode = $parts[1] ?? null;
}
}
$order->items()->create([
@@ -520,6 +547,149 @@ public function createFromQuote(int $quoteId, array $data = [])
});
}
/**
* 견적 변경사항을 수주에 동기화
*
* 견적이 수정되면 연결된 수주도 함께 업데이트하고 히스토리를 생성합니다.
*
* @param Quote $quote 수정된 견적
* @param int $revision 견적 수정 차수
* @return Order|null 업데이트된 수주 또는 null (연결된 수주가 없는 경우)
*/
public function syncFromQuote(Quote $quote, int $revision): ?Order
{
$tenantId = $this->tenantId();
$userId = $this->apiUserId();
// 연결된 수주 확인
$order = Order::where('tenant_id', $tenantId)
->where('quote_id', $quote->id)
->first();
if (! $order) {
return null;
}
// 생산 진행 이상의 상태면 동기화 불가 (DRAFT, CONFIRMED, IN_PROGRESS만 허용)
$allowedStatuses = [
Order::STATUS_DRAFT,
Order::STATUS_CONFIRMED,
Order::STATUS_IN_PROGRESS,
];
if (! in_array($order->status_code, $allowedStatuses)) {
throw new BadRequestHttpException(__('error.order.cannot_sync_after_production'));
}
return DB::transaction(function () use ($order, $quote, $tenantId, $userId, $revision) {
// 변경 전 데이터 스냅샷 (히스토리용)
$beforeData = [
'site_name' => $order->site_name,
'client_name' => $order->client_name,
'total_amount' => $order->total_amount,
'items_count' => $order->items()->count(),
];
// 수주 기본 정보 업데이트
$order->update([
'site_name' => $quote->site_name,
'client_id' => $quote->client_id,
'client_name' => $quote->client_name,
'discount_rate' => $quote->discount_rate ?? 0,
'discount_amount' => $quote->discount_amount ?? 0,
'total_amount' => $quote->total_amount,
'updated_by' => $userId,
]);
// 기존 품목 삭제 후 새로 생성
$order->items()->delete();
// calculation_inputs에서 제품 정보 추출
$calculationInputs = $quote->calculation_inputs ?? [];
$productItems = $calculationInputs['items'] ?? [];
// 견적 품목을 수주 품목으로 변환
foreach ($quote->items as $index => $quoteItem) {
$floorCode = null;
$symbolCode = null;
// formula_source에서 제품 인덱스 추출
$productIndex = 0;
$formulaSource = $quoteItem->formula_source ?? '';
if (preg_match('/product_(\d+)/', $formulaSource, $matches)) {
$productIndex = (int) $matches[1];
}
// calculation_inputs에서 floor/code 가져오기
if (isset($productItems[$productIndex])) {
$floorCode = $productItems[$productIndex]['floor'] ?? null;
$symbolCode = $productItems[$productIndex]['code'] ?? null;
} elseif (count($productItems) === 1) {
$floorCode = $productItems[0]['floor'] ?? null;
$symbolCode = $productItems[0]['code'] ?? null;
}
// note에서 파싱 시도
if (empty($floorCode) && empty($symbolCode)) {
$note = trim($quoteItem->note ?? '');
if ($note !== '') {
$parts = preg_split('/\s+/', $note, 2);
$floorCode = $parts[0] ?? null;
$symbolCode = $parts[1] ?? null;
}
}
$order->items()->create([
'tenant_id' => $tenantId,
'serial_no' => $index + 1,
'item_id' => $quoteItem->item_id,
'item_code' => $quoteItem->item_code,
'item_name' => $quoteItem->item_name,
'specification' => $quoteItem->specification,
'floor_code' => $floorCode,
'symbol_code' => $symbolCode,
'quantity' => $quoteItem->calculated_quantity,
'unit' => $quoteItem->unit,
'unit_price' => $quoteItem->unit_price,
'supply_amount' => $quoteItem->total_price,
'tax_amount' => round($quoteItem->total_price * 0.1, 2),
'total_amount' => round($quoteItem->total_price * 1.1, 2),
'note' => $quoteItem->formula_category,
'sort_order' => $index,
]);
}
// 합계 재계산
$order->refresh();
$order->recalculateTotals()->save();
// 변경 후 데이터 스냅샷
$afterData = [
'site_name' => $order->site_name,
'client_name' => $order->client_name,
'total_amount' => $order->total_amount,
'items_count' => $order->items()->count(),
];
// 히스토리 생성
OrderHistory::create([
'tenant_id' => $tenantId,
'order_id' => $order->id,
'history_type' => 'quote_updated',
'content' => json_encode([
'message' => "견적 {$revision}차 수정으로 수주 정보가 업데이트되었습니다.",
'quote_id' => $quote->id,
'quote_number' => $quote->quote_number,
'revision' => $revision,
'before' => $beforeData,
'after' => $afterData,
], JSON_UNESCAPED_UNICODE),
'created_by' => $userId,
]);
return $order->load(['client:id,name', 'items', 'quote:id,quote_number']);
});
}
/**
* 생산지시 생성 (공정별 작업지시 다중 생성)
*/