feat:경동기업 견적/수주 전환 로직 개선
- KyungdongFormulaHandler: 수식 계산 로직 리팩토링 및 확장 - OrderService: 수주 전환 시 BOM 품목 매핑 로직 추가 - QuoteService: 견적 상태 처리 개선 - FormulaEvaluatorService: 디버그 로깅 추가 - Quote 모델: 캐스팅 타입 수정 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -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']);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 생산지시 생성 (공정별 작업지시 다중 생성)
|
||||
*/
|
||||
|
||||
Reference in New Issue
Block a user