2025-11-20 16:24:40 +09:00
|
|
|
<?php
|
|
|
|
|
|
2025-12-17 22:06:28 +09:00
|
|
|
use App\Http\Controllers\ApiLogController;
|
2025-11-26 22:23:37 +09:00
|
|
|
use App\Http\Controllers\ArchivedRecordController;
|
2026-01-26 12:41:00 +09:00
|
|
|
use App\Http\Controllers\AuditLogController;
|
2025-11-20 16:24:40 +09:00
|
|
|
use App\Http\Controllers\Auth\LoginController;
|
feat: [pm] 프로젝트 진행 관리 시스템 구현
- Models: AdminPmProject, AdminPmTask, AdminPmIssue
- Services: ProjectService, TaskService, IssueService, ImportService
- API Controllers: ProjectController, TaskController, IssueController, ImportController
- FormRequests: Store/Update/BulkAction 요청 검증
- Views: 대시보드, 프로젝트 CRUD, JSON Import 화면
- Routes: API 42개 + Web 6개 엔드포인트
주요 기능:
- 프로젝트/작업/이슈 계층 구조 관리
- 상태 변경, 우선순위, 마감일 추적
- 작업 순서 드래그앤드롭 (reorder API)
- JSON Import로 일괄 등록
- Soft Delete 및 복원
2025-11-28 08:49:30 +09:00
|
|
|
use App\Http\Controllers\BoardController;
|
2025-12-27 20:50:58 +09:00
|
|
|
use App\Http\Controllers\CustomerCenterController;
|
2025-12-01 23:44:56 +09:00
|
|
|
use App\Http\Controllers\DailyLogController;
|
2025-11-24 16:52:23 +09:00
|
|
|
use App\Http\Controllers\DepartmentController;
|
2025-12-17 22:06:28 +09:00
|
|
|
use App\Http\Controllers\DevTools\ApiExplorerController;
|
2025-11-27 19:02:18 +09:00
|
|
|
use App\Http\Controllers\DevTools\FlowTesterController;
|
2025-12-19 09:04:42 +09:00
|
|
|
use App\Http\Controllers\FcmController;
|
2025-12-17 22:06:28 +09:00
|
|
|
use App\Http\Controllers\ItemFieldController;
|
|
|
|
|
use App\Http\Controllers\Lab\AIController;
|
|
|
|
|
use App\Http\Controllers\Lab\ManagementController;
|
|
|
|
|
use App\Http\Controllers\Lab\StrategyController;
|
2025-11-24 22:02:09 +09:00
|
|
|
use App\Http\Controllers\MenuController;
|
2025-11-25 11:05:57 +09:00
|
|
|
use App\Http\Controllers\PermissionController;
|
2025-12-01 23:44:56 +09:00
|
|
|
use App\Http\Controllers\PostController;
|
|
|
|
|
use App\Http\Controllers\ProfileController;
|
feat: [pm] 프로젝트 진행 관리 시스템 구현
- Models: AdminPmProject, AdminPmTask, AdminPmIssue
- Services: ProjectService, TaskService, IssueService, ImportService
- API Controllers: ProjectController, TaskController, IssueController, ImportController
- FormRequests: Store/Update/BulkAction 요청 검증
- Views: 대시보드, 프로젝트 CRUD, JSON Import 화면
- Routes: API 42개 + Web 6개 엔드포인트
주요 기능:
- 프로젝트/작업/이슈 계층 구조 관리
- 상태 변경, 우선순위, 마감일 추적
- 작업 순서 드래그앤드롭 (reorder API)
- JSON Import로 일괄 등록
- Soft Delete 및 복원
2025-11-28 08:49:30 +09:00
|
|
|
use App\Http\Controllers\ProjectManagementController;
|
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개 라우트
2025-12-04 14:00:24 +09:00
|
|
|
use App\Http\Controllers\QuoteFormulaController;
|
2025-11-24 16:36:02 +09:00
|
|
|
use App\Http\Controllers\RoleController;
|
2025-11-25 15:21:48 +09:00
|
|
|
use App\Http\Controllers\RolePermissionController;
|
2025-11-21 09:18:19 +09:00
|
|
|
use App\Http\Controllers\TenantController;
|
2026-01-26 20:16:05 +09:00
|
|
|
use App\Http\Controllers\TenantSettingController;
|
|
|
|
|
use App\Http\Controllers\CommonCodeController;
|
2026-01-27 11:31:02 +09:00
|
|
|
use App\Http\Controllers\DocumentTemplateController;
|
2026-01-26 20:16:05 +09:00
|
|
|
use App\Http\Controllers\MenuSyncController;
|
2025-11-24 18:49:02 +09:00
|
|
|
use App\Http\Controllers\UserController;
|
2025-11-20 16:24:40 +09:00
|
|
|
use Illuminate\Support\Facades\Route;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Guest Routes (인증 불필요)
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
Route::middleware('guest')->group(function () {
|
|
|
|
|
Route::get('/login', [LoginController::class, 'showLoginForm'])->name('login');
|
|
|
|
|
Route::post('/login', [LoginController::class, 'login']);
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-20 13:42:28 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Session Refresh Route (인증 미들웨어 없음)
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| 세션 만료 시 Remember Token으로 재인증을 시도합니다.
|
|
|
|
|
| HTMX 401 에러 핸들러에서 호출됩니다.
|
|
|
|
|
*/
|
|
|
|
|
Route::post('/auth/refresh-session', [LoginController::class, 'refreshSession'])
|
|
|
|
|
->name('auth.refresh-session');
|
|
|
|
|
|
2026-01-26 20:16:05 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Menu Sync API (외부 서버 호출용, API Key 인증)
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::prefix('menu-sync')->group(function () {
|
|
|
|
|
Route::get('/export', [MenuSyncController::class, 'export']);
|
|
|
|
|
Route::post('/import', [MenuSyncController::class, 'import']);
|
|
|
|
|
});
|
|
|
|
|
|
2025-11-20 16:24:40 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Authenticated Routes (인증 필요)
|
|
|
|
|
|--------------------------------------------------------------------------
|
2025-11-30 21:05:13 +09:00
|
|
|
| - auth: 기본 인증 확인
|
|
|
|
|
| - hq.member: 본사(HQ) 테넌트 소속 확인
|
2025-11-20 16:24:40 +09:00
|
|
|
*/
|
|
|
|
|
|
2025-12-01 23:44:56 +09:00
|
|
|
Route::middleware(['auth', 'hq.member', 'password.changed'])->group(function () {
|
2025-11-20 16:24:40 +09:00
|
|
|
Route::post('/logout', [LoginController::class, 'logout'])->name('logout');
|
|
|
|
|
|
2025-11-21 09:18:19 +09:00
|
|
|
// 테넌트 전환
|
|
|
|
|
Route::post('/tenant/switch', [TenantController::class, 'switch'])->name('tenant.switch');
|
|
|
|
|
|
2025-12-01 23:12:59 +09:00
|
|
|
// 프로필 설정
|
|
|
|
|
Route::prefix('profile')->name('profile.')->group(function () {
|
|
|
|
|
Route::get('/', [ProfileController::class, 'index'])->name('index');
|
|
|
|
|
Route::post('/update', [ProfileController::class, 'update'])->name('update');
|
|
|
|
|
Route::post('/password', [ProfileController::class, 'changePassword'])->name('password');
|
|
|
|
|
});
|
|
|
|
|
|
2025-11-21 15:00:30 +09:00
|
|
|
// 테넌트 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('tenants')->name('tenants.')->group(function () {
|
|
|
|
|
Route::get('/', [TenantController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [TenantController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [TenantController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
2025-11-24 16:36:02 +09:00
|
|
|
// 역할 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('roles')->name('roles.')->group(function () {
|
|
|
|
|
Route::get('/', [RoleController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [RoleController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [RoleController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
2025-11-24 16:52:23 +09:00
|
|
|
// 부서 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('departments')->name('departments.')->group(function () {
|
|
|
|
|
Route::get('/', [DepartmentController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [DepartmentController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [DepartmentController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
2025-11-24 18:49:02 +09:00
|
|
|
// 사용자 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('users')->name('users.')->group(function () {
|
|
|
|
|
Route::get('/', [UserController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [UserController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [UserController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
2025-11-24 22:02:09 +09:00
|
|
|
// 메뉴 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('menus')->name('menus.')->group(function () {
|
|
|
|
|
Route::get('/', [MenuController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [MenuController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [MenuController::class, 'edit'])->name('edit');
|
2025-12-02 21:41:15 +09:00
|
|
|
|
|
|
|
|
// 글로벌 메뉴 관리 (슈퍼관리자 전용)
|
|
|
|
|
Route::get('/global', [MenuController::class, 'globalIndex'])->name('global.index');
|
|
|
|
|
Route::get('/global/create', [MenuController::class, 'globalCreate'])->name('global.create');
|
|
|
|
|
Route::get('/global/{id}/edit', [MenuController::class, 'globalEdit'])->name('global.edit');
|
2026-01-26 20:16:05 +09:00
|
|
|
|
|
|
|
|
// 메뉴 동기화
|
|
|
|
|
Route::prefix('sync')->name('sync.')->group(function () {
|
|
|
|
|
Route::get('/', [MenuSyncController::class, 'index'])->name('index');
|
|
|
|
|
Route::post('/settings', [MenuSyncController::class, 'saveSettings'])->name('settings');
|
|
|
|
|
Route::post('/test', [MenuSyncController::class, 'testConnection'])->name('test');
|
|
|
|
|
Route::post('/push', [MenuSyncController::class, 'push'])->name('push');
|
|
|
|
|
Route::post('/pull', [MenuSyncController::class, 'pull'])->name('pull');
|
|
|
|
|
});
|
2025-11-24 22:02:09 +09:00
|
|
|
});
|
|
|
|
|
|
2025-11-25 11:05:57 +09:00
|
|
|
// 권한 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('permissions')->name('permissions.')->group(function () {
|
|
|
|
|
Route::get('/', [PermissionController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [PermissionController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [PermissionController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
2025-12-27 20:50:58 +09:00
|
|
|
// 고객센터 (활성화된 시스템 게시판 목록)
|
|
|
|
|
Route::get('/customer-center', [CustomerCenterController::class, 'index'])->name('customer-center.index');
|
|
|
|
|
|
2025-12-28 01:30:50 +09:00
|
|
|
// 고객센터 게시판 리다이렉트 (board_code → boards/{code}/posts)
|
2025-12-27 21:12:55 +09:00
|
|
|
Route::get('/customer-center/{code}', function (string $code) {
|
|
|
|
|
$board = \App\Models\Boards\Board::where('board_code', $code)
|
|
|
|
|
->whereNull('tenant_id')
|
|
|
|
|
->where('is_active', true)
|
|
|
|
|
->firstOrFail();
|
|
|
|
|
|
2025-12-28 01:30:50 +09:00
|
|
|
return redirect()->route('boards.posts.index', $board->board_code);
|
2025-12-27 21:12:55 +09:00
|
|
|
})->name('customer-center.board');
|
|
|
|
|
|
feat: [pm] 프로젝트 진행 관리 시스템 구현
- Models: AdminPmProject, AdminPmTask, AdminPmIssue
- Services: ProjectService, TaskService, IssueService, ImportService
- API Controllers: ProjectController, TaskController, IssueController, ImportController
- FormRequests: Store/Update/BulkAction 요청 검증
- Views: 대시보드, 프로젝트 CRUD, JSON Import 화면
- Routes: API 42개 + Web 6개 엔드포인트
주요 기능:
- 프로젝트/작업/이슈 계층 구조 관리
- 상태 변경, 우선순위, 마감일 추적
- 작업 순서 드래그앤드롭 (reorder API)
- JSON Import로 일괄 등록
- Soft Delete 및 복원
2025-11-28 08:49:30 +09:00
|
|
|
// 시스템 게시판 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('boards')->name('boards.')->group(function () {
|
|
|
|
|
Route::get('/', [BoardController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [BoardController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [BoardController::class, 'edit'])->name('edit');
|
2025-12-01 23:44:56 +09:00
|
|
|
|
2025-12-28 01:30:50 +09:00
|
|
|
// 사이드바 메뉴에서 접근 시 리다이렉트 (board_code → boards/{boardCode}/posts?t=)
|
|
|
|
|
// 좌측 메뉴는 로그인 회원의 테넌트 게시판 사용
|
2025-12-27 23:40:33 +09:00
|
|
|
Route::get('/{code}', function (string $code) {
|
|
|
|
|
// 숫자만 있는 경우 (기존 라우트와 충돌 방지)
|
|
|
|
|
if (is_numeric($code)) {
|
|
|
|
|
abort(404);
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-28 01:30:50 +09:00
|
|
|
// 로그인 회원의 tenant_id 사용 (사이드바 메뉴용)
|
|
|
|
|
// SidebarMenuService와 동일하게 tenant_id가 null이면 1 사용
|
|
|
|
|
$userTenantId = auth()->user()?->tenant_id ?? 1;
|
2025-12-27 23:40:33 +09:00
|
|
|
|
2025-12-28 01:30:50 +09:00
|
|
|
// 시스템 게시판 우선, 그 다음 로그인 회원의 테넌트 게시판
|
2025-12-27 23:40:33 +09:00
|
|
|
$board = \App\Models\Boards\Board::where('board_code', $code)
|
2025-12-28 01:30:50 +09:00
|
|
|
->whereNull('tenant_id')
|
2025-12-27 23:40:33 +09:00
|
|
|
->where('is_active', true)
|
2025-12-28 01:30:50 +09:00
|
|
|
->first();
|
2025-12-27 23:40:33 +09:00
|
|
|
|
2025-12-28 01:30:50 +09:00
|
|
|
if (! $board) {
|
|
|
|
|
$board = \App\Models\Boards\Board::where('board_code', $code)
|
|
|
|
|
->where('tenant_id', $userTenantId)
|
|
|
|
|
->where('is_active', true)
|
|
|
|
|
->firstOrFail();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// t 파라미터 포함하여 리다이렉트
|
|
|
|
|
return redirect()->route('boards.posts.index', [
|
|
|
|
|
'boardCode' => $board->board_code,
|
|
|
|
|
't' => $board->tenant_id,
|
|
|
|
|
]);
|
2025-12-27 23:40:33 +09:00
|
|
|
})->name('tenant-board')->where('code', '[a-zA-Z][a-zA-Z0-9_-]*');
|
|
|
|
|
|
2025-12-01 23:44:56 +09:00
|
|
|
// 게시글 CRUD (board 하위 중첩 라우트)
|
2025-12-28 01:30:50 +09:00
|
|
|
// board_code + tenant_id(query param 't')로 게시판 조회
|
|
|
|
|
// tenant_id 없으면 로그인 회원의 tenant_id 사용
|
|
|
|
|
Route::prefix('{boardCode}/posts')->name('posts.')->group(function () {
|
2025-12-01 23:44:56 +09:00
|
|
|
Route::get('/', [PostController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [PostController::class, 'create'])->name('create');
|
|
|
|
|
Route::post('/', [PostController::class, 'store'])->name('store');
|
|
|
|
|
Route::get('/{post}', [PostController::class, 'show'])->name('show');
|
|
|
|
|
Route::get('/{post}/edit', [PostController::class, 'edit'])->name('edit');
|
|
|
|
|
Route::put('/{post}', [PostController::class, 'update'])->name('update');
|
|
|
|
|
Route::delete('/{post}', [PostController::class, 'destroy'])->name('destroy');
|
2025-12-02 00:53:25 +09:00
|
|
|
|
2025-12-29 09:11:21 +09:00
|
|
|
// 슈퍼관리자 전용: 복원/영구삭제
|
|
|
|
|
Route::post('/{post}/restore', [PostController::class, 'restore'])->name('restore');
|
|
|
|
|
Route::delete('/{post}/force', [PostController::class, 'forceDestroy'])->name('forceDestroy');
|
|
|
|
|
|
2025-12-02 00:53:25 +09:00
|
|
|
// 파일 관리
|
|
|
|
|
Route::prefix('{post}/files')->name('files.')->group(function () {
|
|
|
|
|
Route::get('/{fileId}/download', [PostController::class, 'downloadFile'])->name('download');
|
2025-12-27 17:08:53 +09:00
|
|
|
Route::get('/{fileId}/preview', [PostController::class, 'previewFile'])->name('preview');
|
2025-12-02 00:53:25 +09:00
|
|
|
Route::post('/', [PostController::class, 'uploadFiles'])->name('upload');
|
|
|
|
|
Route::delete('/{fileId}', [PostController::class, 'deleteFile'])->name('delete');
|
|
|
|
|
});
|
2025-12-29 12:58:06 +09:00
|
|
|
|
|
|
|
|
// 댓글 관리
|
|
|
|
|
Route::prefix('{post}/comments')->name('comments.')->group(function () {
|
|
|
|
|
Route::post('/', [PostController::class, 'storeComment'])->name('store');
|
|
|
|
|
Route::put('/{comment}', [PostController::class, 'updateComment'])->name('update');
|
|
|
|
|
Route::delete('/{comment}', [PostController::class, 'destroyComment'])->name('destroy');
|
|
|
|
|
});
|
2025-12-28 01:30:50 +09:00
|
|
|
})->where('boardCode', '[a-zA-Z][a-zA-Z0-9_-]*');
|
feat: [pm] 프로젝트 진행 관리 시스템 구현
- Models: AdminPmProject, AdminPmTask, AdminPmIssue
- Services: ProjectService, TaskService, IssueService, ImportService
- API Controllers: ProjectController, TaskController, IssueController, ImportController
- FormRequests: Store/Update/BulkAction 요청 검증
- Views: 대시보드, 프로젝트 CRUD, JSON Import 화면
- Routes: API 42개 + Web 6개 엔드포인트
주요 기능:
- 프로젝트/작업/이슈 계층 구조 관리
- 상태 변경, 우선순위, 마감일 추적
- 작업 순서 드래그앤드롭 (reorder API)
- JSON Import로 일괄 등록
- Soft Delete 및 복원
2025-11-28 08:49:30 +09:00
|
|
|
});
|
|
|
|
|
|
2025-11-25 15:21:48 +09:00
|
|
|
// 역할 권한 관리 (Blade 화면만)
|
|
|
|
|
Route::get('/role-permissions', [RolePermissionController::class, 'index'])->name('role-permissions.index');
|
|
|
|
|
|
2025-11-25 15:32:58 +09:00
|
|
|
// 부서 권한 관리 (Blade 화면만)
|
|
|
|
|
Route::get('/department-permissions', [\App\Http\Controllers\DepartmentPermissionController::class, 'index'])->name('department-permissions.index');
|
|
|
|
|
|
2025-11-26 20:40:54 +09:00
|
|
|
// 개인 권한 관리 (Blade 화면만)
|
|
|
|
|
Route::get('/user-permissions', [\App\Http\Controllers\UserPermissionController::class, 'index'])->name('user-permissions.index');
|
|
|
|
|
|
2025-11-26 21:42:51 +09:00
|
|
|
// 권한 분석 (Blade 화면만)
|
|
|
|
|
Route::get('/permission-analyze', [\App\Http\Controllers\PermissionAnalyzeController::class, 'index'])->name('permission-analyze.index');
|
|
|
|
|
|
2025-11-26 22:23:37 +09:00
|
|
|
// 삭제된 데이터 백업 (Blade 화면만)
|
|
|
|
|
Route::prefix('archived-records')->name('archived-records.')->group(function () {
|
|
|
|
|
Route::get('/', [ArchivedRecordController::class, 'index'])->name('index');
|
2025-11-26 23:16:39 +09:00
|
|
|
Route::get('/{batchId}', [ArchivedRecordController::class, 'show'])->name('show');
|
2025-12-01 00:43:58 +09:00
|
|
|
Route::get('/{batchId}/restore-check', [ArchivedRecordController::class, 'checkRestore'])->name('restore-check');
|
|
|
|
|
Route::post('/{batchId}/restore', [ArchivedRecordController::class, 'restore'])->name('restore');
|
2025-11-26 22:23:37 +09:00
|
|
|
});
|
|
|
|
|
|
2026-01-26 12:41:00 +09:00
|
|
|
// 감사 로그 (Blade 화면만)
|
|
|
|
|
Route::prefix('audit-logs')->name('audit-logs.')->group(function () {
|
|
|
|
|
Route::get('/', [AuditLogController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/{id}', [AuditLogController::class, 'show'])->name('show');
|
|
|
|
|
});
|
|
|
|
|
|
feat: [pm] 프로젝트 진행 관리 시스템 구현
- Models: AdminPmProject, AdminPmTask, AdminPmIssue
- Services: ProjectService, TaskService, IssueService, ImportService
- API Controllers: ProjectController, TaskController, IssueController, ImportController
- FormRequests: Store/Update/BulkAction 요청 검증
- Views: 대시보드, 프로젝트 CRUD, JSON Import 화면
- Routes: API 42개 + Web 6개 엔드포인트
주요 기능:
- 프로젝트/작업/이슈 계층 구조 관리
- 상태 변경, 우선순위, 마감일 추적
- 작업 순서 드래그앤드롭 (reorder API)
- JSON Import로 일괄 등록
- Soft Delete 및 복원
2025-11-28 08:49:30 +09:00
|
|
|
// 프로젝트 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('project-management')->name('pm.')->group(function () {
|
|
|
|
|
// 대시보드
|
|
|
|
|
Route::get('/', [ProjectManagementController::class, 'index'])->name('index');
|
|
|
|
|
|
|
|
|
|
// 프로젝트
|
|
|
|
|
Route::get('/projects', [ProjectManagementController::class, 'projects'])->name('projects.index');
|
|
|
|
|
Route::get('/projects/create', [ProjectManagementController::class, 'createProject'])->name('projects.create');
|
|
|
|
|
Route::get('/projects/{id}', [ProjectManagementController::class, 'showProject'])->name('projects.show');
|
|
|
|
|
Route::get('/projects/{id}/edit', [ProjectManagementController::class, 'editProject'])->name('projects.edit');
|
|
|
|
|
|
|
|
|
|
// JSON Import
|
|
|
|
|
Route::get('/import', [ProjectManagementController::class, 'import'])->name('import');
|
|
|
|
|
});
|
|
|
|
|
|
feat: [daily-logs] 일일 스크럼 기능 구현
주요 기능:
- 일일 로그 CRUD (생성, 조회, 수정, 삭제, 복원, 영구삭제)
- 로그 항목(Entry) 관리 (추가, 상태변경, 삭제, 순서변경)
- 주간 타임라인 (최근 7일 진행률 표시)
- 테이블 리스트 아코디언 상세보기
- 담당자 자동완성 (일반 사용자는 슈퍼관리자 목록 제외)
- HTMX 기반 동적 테이블 로딩 및 필터링
- Soft Delete 지원
파일 추가:
- Models: AdminPmDailyLog, AdminPmDailyLogEntry
- Controllers: DailyLogController (Web, API)
- Service: DailyLogService
- Requests: StoreDailyLogRequest, UpdateDailyLogRequest
- Views: index, show, table partial, modal-form partial
라우트 추가:
- Web: /daily-logs, /daily-logs/today, /daily-logs/{id}
- API: /api/admin/daily-logs/* (CRUD + 항목관리)
2025-12-01 14:07:55 +09:00
|
|
|
// 일일 스크럼 (Blade 화면만)
|
|
|
|
|
Route::prefix('daily-logs')->name('daily-logs.')->group(function () {
|
|
|
|
|
Route::get('/', [DailyLogController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/today', [DailyLogController::class, 'today'])->name('today');
|
|
|
|
|
Route::get('/{id}', [DailyLogController::class, 'show'])->name('show');
|
|
|
|
|
});
|
|
|
|
|
|
2025-12-09 23:13:27 +09:00
|
|
|
// 품목기준 필드 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('item-fields')->name('item-fields.')->group(function () {
|
|
|
|
|
Route::get('/', [ItemFieldController::class, 'index'])->name('index');
|
|
|
|
|
});
|
|
|
|
|
|
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개 라우트
2025-12-04 14:00:24 +09:00
|
|
|
// 견적수식 관리 (Blade 화면만)
|
|
|
|
|
Route::prefix('quote-formulas')->name('quote-formulas.')->group(function () {
|
|
|
|
|
// 수식 관리
|
|
|
|
|
Route::get('/', [QuoteFormulaController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [QuoteFormulaController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [QuoteFormulaController::class, 'edit'])->name('edit');
|
|
|
|
|
|
|
|
|
|
// 카테고리 관리
|
|
|
|
|
Route::get('/categories', [QuoteFormulaController::class, 'categories'])->name('categories.index');
|
|
|
|
|
Route::get('/categories/create', [QuoteFormulaController::class, 'createCategory'])->name('categories.create');
|
|
|
|
|
Route::get('/categories/{id}/edit', [QuoteFormulaController::class, 'editCategory'])->name('categories.edit');
|
|
|
|
|
|
|
|
|
|
// 시뮬레이터
|
|
|
|
|
Route::get('/simulator', [QuoteFormulaController::class, 'simulator'])->name('simulator');
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-26 20:16:05 +09:00
|
|
|
// 테넌트 설정 (재고 설정 등)
|
|
|
|
|
Route::prefix('tenant-settings')->name('tenant-settings.')->group(function () {
|
|
|
|
|
Route::get('/', [TenantSettingController::class, 'index'])->name('index');
|
|
|
|
|
Route::post('/', [TenantSettingController::class, 'store'])->name('store');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// 공통코드 관리
|
|
|
|
|
Route::prefix('common-codes')->name('common-codes.')->group(function () {
|
|
|
|
|
Route::get('/', [CommonCodeController::class, 'index'])->name('index');
|
|
|
|
|
Route::post('/', [CommonCodeController::class, 'store'])->name('store');
|
2026-01-26 20:52:44 +09:00
|
|
|
Route::post('/bulk-copy', [CommonCodeController::class, 'bulkCopy'])->name('bulk-copy');
|
2026-01-26 20:16:05 +09:00
|
|
|
Route::put('/{id}', [CommonCodeController::class, 'update'])->name('update');
|
|
|
|
|
Route::post('/{id}/toggle', [CommonCodeController::class, 'toggle'])->name('toggle');
|
|
|
|
|
Route::post('/{id}/copy', [CommonCodeController::class, 'copy'])->name('copy');
|
|
|
|
|
Route::delete('/{id}', [CommonCodeController::class, 'destroy'])->name('destroy');
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-27 11:31:02 +09:00
|
|
|
// 문서양식 관리
|
|
|
|
|
Route::prefix('document-templates')->name('document-templates.')->group(function () {
|
|
|
|
|
Route::get('/', [DocumentTemplateController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [DocumentTemplateController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}/edit', [DocumentTemplateController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-21 21:16:06 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| 바로빌 Routes
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::prefix('barobill')->name('barobill.')->group(function () {
|
2026-01-22 15:27:01 +09:00
|
|
|
Route::get('/settings', [\App\Http\Controllers\Barobill\BarobillController::class, 'settings'])->name('settings.index');
|
2026-01-21 21:16:06 +09:00
|
|
|
Route::get('/members', [\App\Http\Controllers\Barobill\BarobillController::class, 'members'])->name('members.index');
|
2026-01-22 15:27:01 +09:00
|
|
|
Route::get('/tax-invoice', [\App\Http\Controllers\Barobill\BarobillController::class, 'taxInvoice'])->name('tax-invoice.index');
|
|
|
|
|
Route::get('/bank-account', [\App\Http\Controllers\Barobill\BarobillController::class, 'bankAccount'])->name('bank-account.index');
|
|
|
|
|
Route::get('/card-usage', [\App\Http\Controllers\Barobill\BarobillController::class, 'cardUsage'])->name('card-usage.index');
|
2026-01-23 17:12:03 +09:00
|
|
|
// Route::get('/hometax', ...) - 아래 hometax 그룹으로 이동됨
|
2026-01-22 15:27:01 +09:00
|
|
|
Route::get('/usage', [\App\Http\Controllers\Barobill\BarobillController::class, 'usage'])->name('usage.index');
|
2026-01-27 15:03:44 +09:00
|
|
|
Route::get('/billing', [\App\Http\Controllers\Barobill\BarobillController::class, 'billing'])->name('billing.index');
|
2026-01-22 15:27:01 +09:00
|
|
|
// 기존 config 라우트 (호환성)
|
|
|
|
|
Route::get('/config', [\App\Http\Controllers\Barobill\BarobillController::class, 'config'])->name('config.index');
|
2026-01-22 21:24:56 +09:00
|
|
|
|
|
|
|
|
// 전자세금계산서 (React 페이지)
|
|
|
|
|
Route::prefix('etax')->name('etax.')->group(function () {
|
|
|
|
|
Route::get('/', [\App\Http\Controllers\Barobill\EtaxController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/invoices', [\App\Http\Controllers\Barobill\EtaxController::class, 'getInvoices'])->name('invoices');
|
|
|
|
|
Route::post('/issue', [\App\Http\Controllers\Barobill\EtaxController::class, 'issue'])->name('issue');
|
|
|
|
|
Route::post('/send-to-nts', [\App\Http\Controllers\Barobill\EtaxController::class, 'sendToNts'])->name('send-to-nts');
|
|
|
|
|
Route::post('/delete', [\App\Http\Controllers\Barobill\EtaxController::class, 'delete'])->name('delete');
|
|
|
|
|
});
|
2026-01-23 10:13:28 +09:00
|
|
|
|
|
|
|
|
// 계좌 입출금내역 (React 페이지)
|
|
|
|
|
Route::prefix('eaccount')->name('eaccount.')->group(function () {
|
|
|
|
|
Route::get('/', [\App\Http\Controllers\Barobill\EaccountController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/accounts', [\App\Http\Controllers\Barobill\EaccountController::class, 'accounts'])->name('accounts');
|
|
|
|
|
Route::get('/transactions', [\App\Http\Controllers\Barobill\EaccountController::class, 'transactions'])->name('transactions');
|
2026-01-23 11:09:36 +09:00
|
|
|
Route::get('/account-codes', [\App\Http\Controllers\Barobill\EaccountController::class, 'accountCodes'])->name('account-codes');
|
2026-01-23 12:31:10 +09:00
|
|
|
Route::get('/account-codes/all', [\App\Http\Controllers\Barobill\EaccountController::class, 'accountCodesAll'])->name('account-codes.all');
|
|
|
|
|
Route::post('/account-codes', [\App\Http\Controllers\Barobill\EaccountController::class, 'accountCodeStore'])->name('account-codes.store');
|
|
|
|
|
Route::put('/account-codes/{id}', [\App\Http\Controllers\Barobill\EaccountController::class, 'accountCodeUpdate'])->name('account-codes.update');
|
|
|
|
|
Route::delete('/account-codes/{id}', [\App\Http\Controllers\Barobill\EaccountController::class, 'accountCodeDestroy'])->name('account-codes.destroy');
|
2026-01-23 11:09:36 +09:00
|
|
|
Route::post('/save', [\App\Http\Controllers\Barobill\EaccountController::class, 'save'])->name('save');
|
|
|
|
|
Route::get('/export', [\App\Http\Controllers\Barobill\EaccountController::class, 'exportExcel'])->name('export');
|
2026-01-23 10:13:28 +09:00
|
|
|
});
|
2026-01-23 12:46:31 +09:00
|
|
|
|
|
|
|
|
// 카드 사용내역 (React 페이지)
|
|
|
|
|
Route::prefix('ecard')->name('ecard.')->group(function () {
|
|
|
|
|
Route::get('/', [\App\Http\Controllers\Barobill\EcardController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/cards', [\App\Http\Controllers\Barobill\EcardController::class, 'cards'])->name('cards');
|
|
|
|
|
Route::get('/transactions', [\App\Http\Controllers\Barobill\EcardController::class, 'transactions'])->name('transactions');
|
|
|
|
|
Route::get('/account-codes', [\App\Http\Controllers\Barobill\EcardController::class, 'accountCodes'])->name('account-codes');
|
|
|
|
|
Route::post('/save', [\App\Http\Controllers\Barobill\EcardController::class, 'save'])->name('save');
|
2026-01-23 16:13:16 +09:00
|
|
|
Route::post('/export', [\App\Http\Controllers\Barobill\EcardController::class, 'exportExcel'])->name('export');
|
2026-01-23 14:08:44 +09:00
|
|
|
// 분개 관련
|
|
|
|
|
Route::get('/splits', [\App\Http\Controllers\Barobill\EcardController::class, 'splits'])->name('splits');
|
|
|
|
|
Route::post('/splits', [\App\Http\Controllers\Barobill\EcardController::class, 'saveSplits'])->name('splits.save');
|
|
|
|
|
Route::delete('/splits', [\App\Http\Controllers\Barobill\EcardController::class, 'deleteSplits'])->name('splits.delete');
|
2026-01-23 12:46:31 +09:00
|
|
|
});
|
2026-01-23 17:12:03 +09:00
|
|
|
|
|
|
|
|
// 홈택스 매입/매출 (React 페이지)
|
|
|
|
|
Route::prefix('hometax')->name('hometax.')->group(function () {
|
|
|
|
|
Route::get('/', [\App\Http\Controllers\Barobill\HometaxController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/sales', [\App\Http\Controllers\Barobill\HometaxController::class, 'sales'])->name('sales');
|
|
|
|
|
Route::get('/purchases', [\App\Http\Controllers\Barobill\HometaxController::class, 'purchases'])->name('purchases');
|
|
|
|
|
Route::post('/request-collect', [\App\Http\Controllers\Barobill\HometaxController::class, 'requestCollect'])->name('request-collect');
|
|
|
|
|
Route::get('/collect-status', [\App\Http\Controllers\Barobill\HometaxController::class, 'collectStatus'])->name('collect-status');
|
|
|
|
|
Route::post('/export', [\App\Http\Controllers\Barobill\HometaxController::class, 'exportExcel'])->name('export');
|
2026-01-23 18:03:41 +09:00
|
|
|
// 홈택스 스크래핑 관련
|
|
|
|
|
Route::get('/scrap-url', [\App\Http\Controllers\Barobill\HometaxController::class, 'getScrapRequestUrl'])->name('scrap-url');
|
|
|
|
|
Route::post('/refresh-scrap', [\App\Http\Controllers\Barobill\HometaxController::class, 'refreshScrap'])->name('refresh-scrap');
|
|
|
|
|
Route::get('/diagnose', [\App\Http\Controllers\Barobill\HometaxController::class, 'diagnose'])->name('diagnose');
|
2026-01-23 17:12:03 +09:00
|
|
|
});
|
2026-01-21 21:16:06 +09:00
|
|
|
});
|
|
|
|
|
|
2026-01-22 19:11:45 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| 신용평가 Routes
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::prefix('credit')->name('credit.')->group(function () {
|
2026-01-22 20:17:23 +09:00
|
|
|
// 조회 이력 목록
|
2026-01-22 19:11:45 +09:00
|
|
|
Route::get('/inquiry', [\App\Http\Controllers\Credit\CreditController::class, 'inquiry'])->name('inquiry.index');
|
2026-01-22 19:33:51 +09:00
|
|
|
Route::post('/inquiry/search', [\App\Http\Controllers\Credit\CreditController::class, 'search'])->name('inquiry.search');
|
|
|
|
|
Route::post('/inquiry/test', [\App\Http\Controllers\Credit\CreditController::class, 'testConnection'])->name('inquiry.test');
|
|
|
|
|
|
2026-01-22 20:17:23 +09:00
|
|
|
// 조회 이력 상세 데이터
|
|
|
|
|
Route::get('/inquiry/{inquiryKey}/raw', [\App\Http\Controllers\Credit\CreditController::class, 'getRawData'])->name('inquiry.raw');
|
|
|
|
|
Route::get('/inquiry/{inquiryKey}/report', [\App\Http\Controllers\Credit\CreditController::class, 'getReportData'])->name('inquiry.report');
|
|
|
|
|
Route::delete('/inquiry/{id}', [\App\Http\Controllers\Credit\CreditController::class, 'deleteInquiry'])->name('inquiry.destroy');
|
|
|
|
|
|
2026-01-22 19:33:51 +09:00
|
|
|
// 설정 관리
|
|
|
|
|
Route::get('/settings', [\App\Http\Controllers\Credit\CreditController::class, 'settings'])->name('settings.index');
|
|
|
|
|
Route::get('/settings/create', [\App\Http\Controllers\Credit\CreditController::class, 'createConfig'])->name('settings.create');
|
|
|
|
|
Route::post('/settings', [\App\Http\Controllers\Credit\CreditController::class, 'storeConfig'])->name('settings.store');
|
|
|
|
|
Route::get('/settings/{id}/edit', [\App\Http\Controllers\Credit\CreditController::class, 'editConfig'])->name('settings.edit');
|
|
|
|
|
Route::put('/settings/{id}', [\App\Http\Controllers\Credit\CreditController::class, 'updateConfig'])->name('settings.update');
|
|
|
|
|
Route::delete('/settings/{id}', [\App\Http\Controllers\Credit\CreditController::class, 'deleteConfig'])->name('settings.destroy');
|
|
|
|
|
Route::post('/settings/{id}/toggle', [\App\Http\Controllers\Credit\CreditController::class, 'toggleConfig'])->name('settings.toggle');
|
2026-01-22 19:11:45 +09:00
|
|
|
});
|
|
|
|
|
|
2025-11-21 15:00:30 +09:00
|
|
|
// 대시보드
|
2025-11-20 16:24:40 +09:00
|
|
|
Route::get('/dashboard', function () {
|
|
|
|
|
return view('dashboard.index');
|
|
|
|
|
})->name('dashboard');
|
|
|
|
|
|
|
|
|
|
// 루트 리다이렉트
|
|
|
|
|
Route::get('/', function () {
|
|
|
|
|
return redirect()->route('dashboard');
|
|
|
|
|
});
|
2025-11-27 19:02:18 +09:00
|
|
|
|
2025-12-13 23:41:08 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
2025-12-14 00:31:34 +09:00
|
|
|
| R&D Labs Routes (5130 마이그레이션)
|
2025-12-13 23:41:08 +09:00
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::prefix('lab')->name('lab.')->group(function () {
|
|
|
|
|
// S. 전략 (Strategy)
|
|
|
|
|
Route::prefix('strategy')->name('strategy.')->group(function () {
|
|
|
|
|
Route::get('/tax', [StrategyController::class, 'tax'])->name('tax');
|
|
|
|
|
Route::get('/labor', [StrategyController::class, 'labor'])->name('labor');
|
|
|
|
|
Route::get('/debt', [StrategyController::class, 'debt'])->name('debt');
|
|
|
|
|
Route::get('/stablecoin', [StrategyController::class, 'stablecoin'])->name('stablecoin');
|
|
|
|
|
Route::get('/mrp-overseas', [StrategyController::class, 'mrpOverseas'])->name('mrp-overseas');
|
|
|
|
|
Route::get('/chatbot', [StrategyController::class, 'chatbot'])->name('chatbot');
|
|
|
|
|
Route::get('/kodata-vs-nice', [StrategyController::class, 'kodataVsNice'])->name('kodata-vs-nice');
|
|
|
|
|
Route::get('/barobill-vs-popbill', [StrategyController::class, 'barobillVsPopbill'])->name('barobill-vs-popbill');
|
|
|
|
|
Route::get('/knowledge-search', [StrategyController::class, 'knowledgeSearch'])->name('knowledge-search');
|
|
|
|
|
Route::get('/chatbot-compare', [StrategyController::class, 'chatbotCompare'])->name('chatbot-compare');
|
|
|
|
|
Route::get('/rag-startups', [StrategyController::class, 'ragStartups'])->name('rag-startups');
|
|
|
|
|
Route::get('/douzone', [StrategyController::class, 'douzone'])->name('douzone');
|
|
|
|
|
Route::get('/confluence-vs-notion', [StrategyController::class, 'confluenceVsNotion'])->name('confluence-vs-notion');
|
|
|
|
|
Route::get('/qa-solution', [StrategyController::class, 'qaSolution'])->name('qa-solution');
|
|
|
|
|
Route::get('/sales-strategy', [StrategyController::class, 'salesStrategy'])->name('sales-strategy');
|
|
|
|
|
});
|
feat(lab): A 메뉴(AI/자동화) 12개, M 메뉴(관리) 11개 마이그레이션
- AIController 생성 (12개 메서드)
- ManagementController 생성 (11개 메서드)
- A 메뉴 플레이스홀더 템플릿 12개 생성
- 사업자등록증 OCR, 웹 녹음 AI 요약, 회의록 AI 요약
- 업무협의록 AI 요약, 운영자용 챗봇, Vertex RAG 챗봇
- 테넌트 지식 업로드, 테넌트 챗봇, SAM AI 메뉴 이동
- SAM AI 알람음 제작, GPS 출퇴근 관리, 기업개황 조회
- M 메뉴 플레이스홀더 템플릿 11개 생성
- 바로빌 테넌트 관리, 전자세금계산서 전략, 전자세금계산서
- 사업자등록번호 진위 확인, 영업관리 & 매니저 미팅관리
- 카드 세무항목 매칭 전략, 한국 카드사 API 보고서
- 카드 사용내역 수집 후 매칭, 계좌입출금 내역 조회 API
- 영업관리 시나리오, 매니저 시나리오
- 라우트 23개 추가 (AI 12개, Management 11개)
- 사이드바 메뉴 라우트 연결 완료
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-13 23:55:18 +09:00
|
|
|
|
|
|
|
|
// A. AI/자동화 (AI/Automation)
|
|
|
|
|
Route::prefix('ai')->name('ai.')->group(function () {
|
|
|
|
|
Route::get('/business-ocr', [AIController::class, 'businessOcr'])->name('business-ocr');
|
|
|
|
|
Route::get('/web-recording', [AIController::class, 'webRecording'])->name('web-recording');
|
|
|
|
|
Route::get('/meeting-summary', [AIController::class, 'meetingSummary'])->name('meeting-summary');
|
|
|
|
|
Route::get('/work-memo-summary', [AIController::class, 'workMemoSummary'])->name('work-memo-summary');
|
|
|
|
|
Route::get('/operator-chatbot', [AIController::class, 'operatorChatbot'])->name('operator-chatbot');
|
|
|
|
|
Route::get('/vertex-rag', [AIController::class, 'vertexRag'])->name('vertex-rag');
|
|
|
|
|
Route::get('/tenant-knowledge', [AIController::class, 'tenantKnowledge'])->name('tenant-knowledge');
|
|
|
|
|
Route::get('/tenant-chatbot', [AIController::class, 'tenantChatbot'])->name('tenant-chatbot');
|
|
|
|
|
Route::get('/sam-ai-menu', [AIController::class, 'samAiMenu'])->name('sam-ai-menu');
|
|
|
|
|
Route::get('/sam-ai-alarm', [AIController::class, 'samAiAlarm'])->name('sam-ai-alarm');
|
|
|
|
|
Route::get('/gps-attendance', [AIController::class, 'gpsAttendance'])->name('gps-attendance');
|
|
|
|
|
Route::get('/company-overview', [AIController::class, 'companyOverview'])->name('company-overview');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// M. 관리 (Management)
|
|
|
|
|
Route::prefix('management')->name('management.')->group(function () {
|
|
|
|
|
Route::get('/barobill-tenant', [ManagementController::class, 'barobillTenant'])->name('barobill-tenant');
|
|
|
|
|
Route::get('/tax-invoice-strategy', [ManagementController::class, 'taxInvoiceStrategy'])->name('tax-invoice-strategy');
|
|
|
|
|
Route::get('/tax-invoice', [ManagementController::class, 'taxInvoice'])->name('tax-invoice');
|
|
|
|
|
Route::get('/business-verify', [ManagementController::class, 'businessVerify'])->name('business-verify');
|
|
|
|
|
Route::get('/sales-meeting', [ManagementController::class, 'salesMeeting'])->name('sales-meeting');
|
|
|
|
|
Route::get('/card-tax-matching', [ManagementController::class, 'cardTaxMatching'])->name('card-tax-matching');
|
|
|
|
|
Route::get('/card-api-report', [ManagementController::class, 'cardApiReport'])->name('card-api-report');
|
|
|
|
|
Route::get('/card-usage-matching', [ManagementController::class, 'cardUsageMatching'])->name('card-usage-matching');
|
|
|
|
|
Route::get('/account-api', [ManagementController::class, 'accountApi'])->name('account-api');
|
|
|
|
|
Route::get('/sales-scenario', [ManagementController::class, 'salesScenario'])->name('sales-scenario');
|
2025-12-16 23:36:00 +09:00
|
|
|
Route::get('/sales-scenario/checklist', [ManagementController::class, 'salesScenarioChecklist'])->name('sales-scenario.checklist');
|
|
|
|
|
Route::post('/sales-scenario/toggle', [ManagementController::class, 'salesScenarioToggle'])->name('sales-scenario.toggle');
|
feat(lab): A 메뉴(AI/자동화) 12개, M 메뉴(관리) 11개 마이그레이션
- AIController 생성 (12개 메서드)
- ManagementController 생성 (11개 메서드)
- A 메뉴 플레이스홀더 템플릿 12개 생성
- 사업자등록증 OCR, 웹 녹음 AI 요약, 회의록 AI 요약
- 업무협의록 AI 요약, 운영자용 챗봇, Vertex RAG 챗봇
- 테넌트 지식 업로드, 테넌트 챗봇, SAM AI 메뉴 이동
- SAM AI 알람음 제작, GPS 출퇴근 관리, 기업개황 조회
- M 메뉴 플레이스홀더 템플릿 11개 생성
- 바로빌 테넌트 관리, 전자세금계산서 전략, 전자세금계산서
- 사업자등록번호 진위 확인, 영업관리 & 매니저 미팅관리
- 카드 세무항목 매칭 전략, 한국 카드사 API 보고서
- 카드 사용내역 수집 후 매칭, 계좌입출금 내역 조회 API
- 영업관리 시나리오, 매니저 시나리오
- 라우트 23개 추가 (AI 12개, Management 11개)
- 사이드바 메뉴 라우트 연결 완료
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-13 23:55:18 +09:00
|
|
|
Route::get('/manager-scenario', [ManagementController::class, 'managerScenario'])->name('manager-scenario');
|
|
|
|
|
});
|
2025-12-13 23:41:08 +09:00
|
|
|
});
|
|
|
|
|
|
2025-12-19 09:04:42 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| FCM 관리 Routes
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::prefix('fcm')->name('fcm.')->group(function () {
|
|
|
|
|
// 토큰 관리
|
|
|
|
|
Route::get('/tokens', [FcmController::class, 'tokens'])->name('tokens');
|
|
|
|
|
Route::get('/tokens/list', [FcmController::class, 'tokenList'])->name('tokens.list');
|
|
|
|
|
Route::get('/tokens/stats', [FcmController::class, 'tokenStats'])->name('tokens.stats');
|
|
|
|
|
Route::post('/tokens/{id}/toggle', [FcmController::class, 'toggleToken'])->name('tokens.toggle');
|
|
|
|
|
Route::delete('/tokens/{id}', [FcmController::class, 'deleteToken'])->name('tokens.delete');
|
|
|
|
|
|
|
|
|
|
// 테스트 발송
|
|
|
|
|
Route::get('/send', [FcmController::class, 'send'])->name('send');
|
|
|
|
|
Route::post('/send', [FcmController::class, 'sendPush'])->name('send.push');
|
|
|
|
|
Route::get('/preview-count', [FcmController::class, 'previewCount'])->name('preview-count');
|
|
|
|
|
|
|
|
|
|
// 발송 이력
|
|
|
|
|
Route::get('/history', [FcmController::class, 'history'])->name('history');
|
|
|
|
|
Route::get('/history/list', [FcmController::class, 'historyList'])->name('history.list');
|
|
|
|
|
});
|
|
|
|
|
|
2025-11-27 19:02:18 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| 개발 도구 Routes
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::prefix('dev-tools')->name('dev-tools.')->group(function () {
|
2025-12-15 15:45:10 +09:00
|
|
|
// API 요청 로그
|
|
|
|
|
Route::prefix('api-logs')->name('api-logs.')->group(function () {
|
|
|
|
|
Route::get('/', [ApiLogController::class, 'index'])->name('index');
|
|
|
|
|
Route::post('/prune', [ApiLogController::class, 'prune'])->name('prune');
|
2025-12-15 21:03:46 +09:00
|
|
|
Route::post('/truncate', [ApiLogController::class, 'truncate'])->name('truncate');
|
2025-12-15 15:45:10 +09:00
|
|
|
Route::get('/{id}', [ApiLogController::class, 'show'])->name('show');
|
2025-12-17 18:56:40 +09:00
|
|
|
Route::post('/{id}/resend', [ApiLogController::class, 'resend'])->name('resend');
|
2025-12-15 15:45:10 +09:00
|
|
|
});
|
|
|
|
|
|
2025-11-27 19:02:18 +09:00
|
|
|
// API 플로우 테스터
|
|
|
|
|
Route::prefix('flow-tester')->name('flow-tester.')->group(function () {
|
2025-12-03 16:47:57 +09:00
|
|
|
// 고정 경로 먼저 (구체적인 경로)
|
2025-11-27 19:02:18 +09:00
|
|
|
Route::get('/', [FlowTesterController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [FlowTesterController::class, 'create'])->name('create');
|
|
|
|
|
Route::post('/', [FlowTesterController::class, 'store'])->name('store');
|
2025-12-03 16:47:57 +09:00
|
|
|
Route::post('/validate-json', [FlowTesterController::class, 'validateJson'])->name('validate-json');
|
|
|
|
|
|
2025-12-21 01:35:54 +09:00
|
|
|
// 토큰 및 인증 관리 라우트 (API Explorer와 공유)
|
|
|
|
|
Route::post('/login-to-api', [FlowTesterController::class, 'loginToApi'])->name('login-to-api');
|
2025-12-18 15:42:01 +09:00
|
|
|
Route::post('/token/save', [FlowTesterController::class, 'saveToken'])->name('token.save');
|
|
|
|
|
Route::post('/token/clear', [FlowTesterController::class, 'clearToken'])->name('token.clear');
|
|
|
|
|
Route::get('/token/status', [FlowTesterController::class, 'tokenStatus'])->name('token.status');
|
|
|
|
|
|
2025-12-03 16:47:57 +09:00
|
|
|
// /runs/* 관련 라우트 (고정 경로)
|
|
|
|
|
Route::get('/runs/{runId}/status', [FlowTesterController::class, 'runStatus'])->name('run-status');
|
|
|
|
|
Route::get('/runs/{runId}', [FlowTesterController::class, 'runDetail'])->name('run-detail');
|
|
|
|
|
|
|
|
|
|
// /{id}/* 관련 라우트 (와일드카드는 마지막에)
|
2025-11-27 19:02:18 +09:00
|
|
|
Route::get('/{id}', [FlowTesterController::class, 'edit'])->name('edit');
|
|
|
|
|
Route::put('/{id}', [FlowTesterController::class, 'update'])->name('update');
|
|
|
|
|
Route::delete('/{id}', [FlowTesterController::class, 'destroy'])->name('destroy');
|
|
|
|
|
Route::post('/{id}/clone', [FlowTesterController::class, 'clone'])->name('clone');
|
|
|
|
|
Route::post('/{id}/run', [FlowTesterController::class, 'run'])->name('run');
|
|
|
|
|
Route::get('/{id}/history', [FlowTesterController::class, 'history'])->name('history');
|
|
|
|
|
});
|
2025-12-17 22:06:28 +09:00
|
|
|
|
|
|
|
|
// API Explorer (OpenAPI 3.0 뷰어)
|
|
|
|
|
Route::prefix('api-explorer')->name('api-explorer.')->group(function () {
|
|
|
|
|
// 메인 뷰
|
|
|
|
|
Route::get('/', [ApiExplorerController::class, 'index'])->name('index');
|
|
|
|
|
|
|
|
|
|
// 엔드포인트 (HTMX)
|
|
|
|
|
Route::get('/endpoints', [ApiExplorerController::class, 'endpoints'])->name('endpoints');
|
|
|
|
|
Route::get('/endpoints/{operationId}', [ApiExplorerController::class, 'endpoint'])->name('endpoint');
|
|
|
|
|
|
|
|
|
|
// API 실행
|
|
|
|
|
Route::post('/execute', [ApiExplorerController::class, 'execute'])->name('execute');
|
|
|
|
|
|
|
|
|
|
// 즐겨찾기
|
|
|
|
|
Route::get('/bookmarks', [ApiExplorerController::class, 'bookmarks'])->name('bookmarks.index');
|
|
|
|
|
Route::post('/bookmarks', [ApiExplorerController::class, 'storeBookmark'])->name('bookmarks.store');
|
|
|
|
|
Route::put('/bookmarks/{id}', [ApiExplorerController::class, 'updateBookmark'])->name('bookmarks.update');
|
|
|
|
|
Route::delete('/bookmarks/{id}', [ApiExplorerController::class, 'deleteBookmark'])->name('bookmarks.destroy');
|
|
|
|
|
Route::post('/bookmarks/toggle', [ApiExplorerController::class, 'toggleBookmark'])->name('bookmarks.toggle');
|
|
|
|
|
Route::post('/bookmarks/reorder', [ApiExplorerController::class, 'reorderBookmarks'])->name('bookmarks.reorder');
|
|
|
|
|
|
|
|
|
|
// 템플릿
|
2025-12-18 20:26:17 +09:00
|
|
|
Route::get('/templates/{endpoint}', [ApiExplorerController::class, 'templatesForEndpoint'])->name('templates.index');
|
|
|
|
|
Route::post('/templates', [ApiExplorerController::class, 'saveTemplate'])->name('templates.store');
|
2025-12-17 22:06:28 +09:00
|
|
|
Route::delete('/templates/{id}', [ApiExplorerController::class, 'deleteTemplate'])->name('templates.destroy');
|
|
|
|
|
|
|
|
|
|
// 히스토리
|
|
|
|
|
Route::get('/history', [ApiExplorerController::class, 'history'])->name('history.index');
|
|
|
|
|
Route::post('/history/{id}/replay', [ApiExplorerController::class, 'replayHistory'])->name('history.replay');
|
|
|
|
|
Route::delete('/history', [ApiExplorerController::class, 'clearHistory'])->name('history.clear');
|
|
|
|
|
|
|
|
|
|
// 환경 설정
|
|
|
|
|
Route::get('/environments', [ApiExplorerController::class, 'environments'])->name('environments.index');
|
2025-12-18 20:26:17 +09:00
|
|
|
Route::post('/environments', [ApiExplorerController::class, 'saveEnvironment'])->name('environments.store');
|
2025-12-17 22:06:28 +09:00
|
|
|
Route::delete('/environments/{id}', [ApiExplorerController::class, 'deleteEnvironment'])->name('environments.destroy');
|
2025-12-18 15:42:01 +09:00
|
|
|
|
|
|
|
|
// 사용자 목록 (인증용)
|
|
|
|
|
Route::get('/users', [ApiExplorerController::class, 'users'])->name('users');
|
2026-01-05 15:10:09 +09:00
|
|
|
Route::post('/issue-token', [ApiExplorerController::class, 'issueToken'])->name('issue-token');
|
2025-12-18 20:26:17 +09:00
|
|
|
|
|
|
|
|
// API 사용 현황 및 폐기 관리
|
|
|
|
|
Route::get('/usage', [ApiExplorerController::class, 'usage'])->name('usage');
|
|
|
|
|
Route::get('/usage/stats', [ApiExplorerController::class, 'usageStats'])->name('usage.stats');
|
|
|
|
|
Route::get('/usage/trend', [ApiExplorerController::class, 'dailyTrend'])->name('usage.trend');
|
|
|
|
|
Route::get('/usage/popular', [ApiExplorerController::class, 'popularApis'])->name('usage.popular');
|
|
|
|
|
Route::get('/usage/stale', [ApiExplorerController::class, 'staleApis'])->name('usage.stale');
|
|
|
|
|
|
|
|
|
|
// 폐기 후보 관리
|
|
|
|
|
Route::get('/deprecations', [ApiExplorerController::class, 'deprecations'])->name('deprecations.index');
|
|
|
|
|
Route::post('/deprecations', [ApiExplorerController::class, 'addDeprecation'])->name('deprecations.store');
|
|
|
|
|
Route::post('/deprecations/bulk-unused', [ApiExplorerController::class, 'addAllUnusedAsDeprecation'])->name('deprecations.bulk-unused');
|
|
|
|
|
Route::put('/deprecations/{id}', [ApiExplorerController::class, 'updateDeprecation'])->name('deprecations.update');
|
|
|
|
|
Route::delete('/deprecations/{id}', [ApiExplorerController::class, 'removeDeprecation'])->name('deprecations.destroy');
|
2025-12-17 22:06:28 +09:00
|
|
|
});
|
2025-11-27 19:02:18 +09:00
|
|
|
});
|
2025-11-20 16:24:40 +09:00
|
|
|
});
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Demo Pages (데모 페이지)
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::get('/재무관리', function () {
|
|
|
|
|
return response()->file(public_path('재무관리.html'));
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/일일자금일보', function () {
|
|
|
|
|
return response()->file(public_path('일일자금일보.html'));
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-20 20:21:06 +09:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Finance Routes (재무 관리)
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::middleware('auth')->prefix('finance')->name('finance.')->group(function () {
|
|
|
|
|
// 대시보드
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/dashboard', [\App\Http\Controllers\Finance\FinanceDashboardController::class, 'index'])->name('dashboard');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 계좌관리 (실제 구현)
|
|
|
|
|
Route::prefix('accounts')->name('accounts.')->group(function () {
|
|
|
|
|
Route::get('/', [\App\Http\Controllers\Finance\BankAccountController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [\App\Http\Controllers\Finance\BankAccountController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}', [\App\Http\Controllers\Finance\BankAccountController::class, 'show'])->name('show');
|
|
|
|
|
Route::get('/{id}/edit', [\App\Http\Controllers\Finance\BankAccountController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-21 19:10:52 +09:00
|
|
|
// 계좌거래내역
|
|
|
|
|
Route::get('/account-transactions', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.account-transactions'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.account-transactions');
|
|
|
|
|
})->name('account-transactions');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 자금계획일정 (실제 구현)
|
|
|
|
|
Route::prefix('fund-schedules')->name('fund-schedules.')->group(function () {
|
|
|
|
|
Route::get('/', [\App\Http\Controllers\Finance\FundScheduleController::class, 'index'])->name('index');
|
|
|
|
|
Route::get('/create', [\App\Http\Controllers\Finance\FundScheduleController::class, 'create'])->name('create');
|
|
|
|
|
Route::get('/{id}', [\App\Http\Controllers\Finance\FundScheduleController::class, 'show'])->name('show');
|
|
|
|
|
Route::get('/{id}/edit', [\App\Http\Controllers\Finance\FundScheduleController::class, 'edit'])->name('edit');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// 기존 fund-schedule URL 리다이렉트 (호환성)
|
|
|
|
|
Route::get('/fund-schedule', fn() => redirect()->route('finance.fund-schedules.index'))->name('fund-schedule');
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/daily-fund', function () {
|
|
|
|
|
// HTMX 요청이면 전체 페이지로 리다이렉트 (React 스크립트 로딩을 위해)
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.daily-fund'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.daily-fund');
|
|
|
|
|
})->name('daily-fund');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 카드관리
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/corporate-cards', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.corporate-cards'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.corporate-cards');
|
|
|
|
|
})->name('corporate-cards');
|
|
|
|
|
Route::get('/card-transactions', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.card-transactions'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.card-transactions');
|
|
|
|
|
})->name('card-transactions');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 수입/지출
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/income', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.income'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.income');
|
|
|
|
|
})->name('income');
|
|
|
|
|
Route::get('/expense', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.expense'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.expense');
|
|
|
|
|
})->name('expense');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 매출/매입
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/sales', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.sales'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.sales');
|
|
|
|
|
})->name('sales');
|
|
|
|
|
Route::get('/purchase', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.purchase'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.purchase');
|
|
|
|
|
})->name('purchase');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 정산관리
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/sales-commission', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.sales-commission'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.sales-commission');
|
|
|
|
|
})->name('sales-commission');
|
|
|
|
|
Route::get('/consulting-fee', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.consulting-fee'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.consulting-fee');
|
|
|
|
|
})->name('consulting-fee');
|
|
|
|
|
Route::get('/customer-settlement', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.customer-settlement'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.customer-settlement');
|
|
|
|
|
})->name('customer-settlement');
|
|
|
|
|
Route::get('/subscription', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.subscription'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.subscription');
|
|
|
|
|
})->name('subscription');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 차량관리
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/corporate-vehicles', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.corporate-vehicles'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.corporate-vehicles');
|
|
|
|
|
})->name('corporate-vehicles');
|
|
|
|
|
Route::get('/vehicle-maintenance', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.vehicle-maintenance'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.vehicle-maintenance');
|
|
|
|
|
})->name('vehicle-maintenance');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 거래처관리
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/customers', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.customers'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.customers');
|
|
|
|
|
})->name('customers');
|
|
|
|
|
Route::get('/partners', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.partners'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.partners');
|
|
|
|
|
})->name('partners');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 채권/채무
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/receivables', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.receivables'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.receivables');
|
|
|
|
|
})->name('receivables');
|
|
|
|
|
Route::get('/payables', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.payables'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.payables');
|
|
|
|
|
})->name('payables');
|
2026-01-20 20:21:06 +09:00
|
|
|
|
|
|
|
|
// 기타
|
2026-01-21 19:10:52 +09:00
|
|
|
Route::get('/refunds', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.refunds'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.refunds');
|
|
|
|
|
})->name('refunds');
|
|
|
|
|
Route::get('/vat', function () {
|
|
|
|
|
if (request()->header('HX-Request')) {
|
|
|
|
|
return response('', 200)->header('HX-Redirect', route('finance.vat'));
|
|
|
|
|
}
|
|
|
|
|
return view('finance.vat');
|
|
|
|
|
})->name('vat');
|
2026-01-20 20:21:06 +09:00
|
|
|
});
|
2026-01-26 11:09:42 +09:00
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Sales Management Routes (영업관리)
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
Route::middleware(['auth', 'hq.member'])->prefix('sales')->name('sales.')->group(function () {
|
2026-01-27 18:59:45 +09:00
|
|
|
// 영업관리 대시보드
|
|
|
|
|
Route::get('salesmanagement/dashboard', [\App\Http\Controllers\Sales\SalesDashboardController::class, 'index'])->name('salesmanagement.dashboard');
|
2026-01-27 19:28:48 +09:00
|
|
|
Route::get('salesmanagement/dashboard/refresh', [\App\Http\Controllers\Sales\SalesDashboardController::class, 'refresh'])->name('salesmanagement.dashboard.refresh');
|
2026-01-27 18:59:45 +09:00
|
|
|
|
2026-01-26 11:09:42 +09:00
|
|
|
// 영업 담당자 관리
|
|
|
|
|
Route::resource('managers', \App\Http\Controllers\Sales\SalesManagerController::class);
|
|
|
|
|
|
|
|
|
|
// 가망고객 관리
|
|
|
|
|
Route::resource('prospects', \App\Http\Controllers\Sales\SalesProspectController::class);
|
|
|
|
|
|
|
|
|
|
// 영업 실적 관리
|
|
|
|
|
Route::resource('records', \App\Http\Controllers\Sales\SalesRecordController::class);
|
|
|
|
|
});
|