feat(quote-formulas): 견적수식 관리 기능 구현

## 구현 내용

### 모델 (5개)
- QuoteFormulaCategory: 수식 카테고리
- QuoteFormula: 수식 정의 (input/calculation/range/mapping)
- QuoteFormulaRange: 범위별 값 정의
- QuoteFormulaMapping: 매핑 테이블
- QuoteFormulaItem: 수식-품목 연결

### 서비스 (3개)
- QuoteFormulaCategoryService: 카테고리 CRUD
- QuoteFormulaService: 수식 CRUD, 복제, 재정렬
- FormulaEvaluatorService: 수식 계산 엔진
  - 지원 함수: SUM, ROUND, CEIL, FLOOR, ABS, MIN, MAX, IF, AND, OR, NOT

### API Controller (2개)
- QuoteFormulaCategoryController: 카테고리 API (11개 엔드포인트)
- QuoteFormulaController: 수식 API (16개 엔드포인트)

### FormRequest (4개)
- Store/Update QuoteFormulaCategoryRequest
- Store/Update QuoteFormulaRequest

### Blade Views (8개)
- 수식 목록/추가/수정/시뮬레이터
- 카테고리 목록/추가/수정
- HTMX 테이블 partial

### 라우트
- API: 27개 엔드포인트
- Web: 7개 라우트
This commit is contained in:
2025-12-04 14:00:24 +09:00
parent 477779f3ac
commit dac02f120b
28 changed files with 6489 additions and 0 deletions

View File

@@ -10,6 +10,8 @@
use App\Http\Controllers\Api\Admin\ProjectManagement\IssueController as PmIssueController;
use App\Http\Controllers\Api\Admin\ProjectManagement\ProjectController as PmProjectController;
use App\Http\Controllers\Api\Admin\ProjectManagement\TaskController as PmTaskController;
use App\Http\Controllers\Api\Admin\Quote\QuoteFormulaCategoryController;
use App\Http\Controllers\Api\Admin\Quote\QuoteFormulaController;
use App\Http\Controllers\Api\Admin\RoleController;
use App\Http\Controllers\Api\Admin\RolePermissionController;
use App\Http\Controllers\Api\Admin\TenantController;
@@ -389,4 +391,71 @@
Route::put('/{entryId}/status', [DailyLogController::class, 'updateEntryStatus'])->name('updateStatus');
Route::delete('/{entryId}', [DailyLogController::class, 'deleteEntry'])->name('delete');
});
/*
|--------------------------------------------------------------------------
| 견적수식 관리 API
|--------------------------------------------------------------------------
*/
Route::prefix('quote-formulas')->name('quote-formulas.')->group(function () {
// 카테고리 관리 API
Route::prefix('categories')->name('categories.')->group(function () {
// 고정 경로
Route::get('/stats', [QuoteFormulaCategoryController::class, 'stats'])->name('stats');
Route::get('/dropdown', [QuoteFormulaCategoryController::class, 'dropdown'])->name('dropdown');
Route::post('/reorder', [QuoteFormulaCategoryController::class, 'reorder'])->name('reorder');
// 기본 CRUD
Route::get('/', [QuoteFormulaCategoryController::class, 'index'])->name('index');
Route::post('/', [QuoteFormulaCategoryController::class, 'store'])->name('store');
Route::get('/{id}', [QuoteFormulaCategoryController::class, 'show'])->name('show');
Route::put('/{id}', [QuoteFormulaCategoryController::class, 'update'])->name('update');
Route::delete('/{id}', [QuoteFormulaCategoryController::class, 'destroy'])->name('destroy');
// 복원 (일반관리자 가능)
Route::post('/{id}/restore', [QuoteFormulaCategoryController::class, 'restore'])->name('restore');
// 슈퍼관리자 전용 액션 (영구삭제)
Route::middleware('super.admin')->group(function () {
Route::delete('/{id}/force', [QuoteFormulaCategoryController::class, 'forceDestroy'])->name('forceDestroy');
});
// 추가 액션
Route::post('/{id}/toggle-active', [QuoteFormulaCategoryController::class, 'toggleActive'])->name('toggleActive');
});
// 수식 관리 API
Route::prefix('formulas')->name('formulas.')->group(function () {
// 고정 경로
Route::get('/stats', [QuoteFormulaController::class, 'stats'])->name('stats');
Route::get('/variables', [QuoteFormulaController::class, 'variables'])->name('variables');
Route::post('/reorder', [QuoteFormulaController::class, 'reorder'])->name('reorder');
Route::post('/validate', [QuoteFormulaController::class, 'validate'])->name('validate');
Route::post('/test', [QuoteFormulaController::class, 'test'])->name('test');
Route::post('/simulate', [QuoteFormulaController::class, 'simulate'])->name('simulate');
// 기본 CRUD
Route::get('/', [QuoteFormulaController::class, 'index'])->name('index');
Route::post('/', [QuoteFormulaController::class, 'store'])->name('store');
Route::get('/{id}', [QuoteFormulaController::class, 'show'])->name('show');
Route::put('/{id}', [QuoteFormulaController::class, 'update'])->name('update');
Route::delete('/{id}', [QuoteFormulaController::class, 'destroy'])->name('destroy');
// 복원 (일반관리자 가능)
Route::post('/{id}/restore', [QuoteFormulaController::class, 'restore'])->name('restore');
// 슈퍼관리자 전용 액션 (영구삭제)
Route::middleware('super.admin')->group(function () {
Route::delete('/{id}/force', [QuoteFormulaController::class, 'forceDestroy'])->name('forceDestroy');
});
// 추가 액션
Route::post('/{id}/toggle-active', [QuoteFormulaController::class, 'toggleActive'])->name('toggleActive');
Route::post('/{id}/duplicate', [QuoteFormulaController::class, 'duplicate'])->name('duplicate');
// 카테고리별
Route::get('/category/{categoryId}', [QuoteFormulaController::class, 'byCategory'])->name('byCategory');
});
});
});