- 성적서 인쇄뷰(print.blade.php) 추가: 동적 검사 테이블 렌더링
- DocumentController: print() 메서드, create/edit HTMX HX-Redirect 추가
- 기본필드 field_key: Str::slug→bf_{id} (한글 빈문자열 버그 수정)
- show.blade.php: 성적서 버튼 추가
- 양식 편집 UI 개선 + 복제 기능
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
1144 lines
59 KiB
PHP
1144 lines
59 KiB
PHP
@extends('layouts.app')
|
|
|
|
@section('title', $isCreate ? '문서양식 등록' : '문서양식 편집')
|
|
|
|
@section('content')
|
|
<div class="max-w-7xl mx-auto">
|
|
<!-- 헤더 -->
|
|
<div class="flex flex-col lg:flex-row lg:justify-between lg:items-center gap-4 mb-6">
|
|
<div>
|
|
<h1 class="text-2xl font-bold text-gray-800">
|
|
{{ $isCreate ? '문서양식 등록' : '문서양식 편집' }}
|
|
</h1>
|
|
<p class="text-sm text-gray-500 mt-1">
|
|
검사 성적서, 작업지시서 등의 문서 양식을 설정합니다.
|
|
</p>
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<button type="button" onclick="openPreviewModal()" class="bg-gray-600 hover:bg-gray-700 text-white px-4 py-2 rounded-lg transition flex items-center gap-2">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" />
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z" />
|
|
</svg>
|
|
미리보기
|
|
</button>
|
|
<a href="{{ route('document-templates.index') }}" class="bg-gray-200 hover:bg-gray-300 text-gray-700 px-4 py-2 rounded-lg transition">
|
|
목록
|
|
</a>
|
|
<button type="button" onclick="saveTemplate()" class="bg-blue-600 hover:bg-blue-700 text-white px-4 py-2 rounded-lg transition flex items-center gap-2">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7" />
|
|
</svg>
|
|
저장
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 탭 네비게이션 -->
|
|
<div class="mb-6 border-b border-gray-200 bg-white rounded-t-lg">
|
|
<nav class="-mb-px flex">
|
|
<button onclick="switchTab('basic')" id="tab-basic"
|
|
class="tab-btn px-6 py-4 text-sm font-medium border-b-2 border-blue-500 text-blue-600">
|
|
기본정보
|
|
</button>
|
|
<button onclick="switchTab('approval')" id="tab-approval"
|
|
class="tab-btn px-6 py-4 text-sm font-medium border-b-2 border-transparent text-gray-500 hover:text-gray-700">
|
|
결재라인
|
|
</button>
|
|
<button onclick="switchTab('basic-fields')" id="tab-basic-fields"
|
|
class="tab-btn px-6 py-4 text-sm font-medium border-b-2 border-transparent text-gray-500 hover:text-gray-700">
|
|
기본필드
|
|
</button>
|
|
<button onclick="switchTab('sections')" id="tab-sections"
|
|
class="tab-btn px-6 py-4 text-sm font-medium border-b-2 border-transparent text-gray-500 hover:text-gray-700">
|
|
검사 기준서
|
|
</button>
|
|
<button onclick="switchTab('columns')" id="tab-columns"
|
|
class="tab-btn px-6 py-4 text-sm font-medium border-b-2 border-transparent text-gray-500 hover:text-gray-700">
|
|
테이블 컬럼
|
|
</button>
|
|
</nav>
|
|
</div>
|
|
|
|
<!-- 기본정보 탭 -->
|
|
<div id="content-basic" class="tab-content bg-white rounded-lg shadow-sm p-6">
|
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-6">
|
|
<!-- 양식명 -->
|
|
<div>
|
|
<label class="block text-sm font-medium text-gray-700 mb-1">양식명 <span class="text-red-500">*</span></label>
|
|
<input type="text" id="name" placeholder="예: 최종검사 성적서"
|
|
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
</div>
|
|
<!-- 분류 -->
|
|
<div>
|
|
<label class="block text-sm font-medium text-gray-700 mb-1">분류</label>
|
|
<input type="text" id="category" list="category-list" placeholder="분류를 입력하거나 선택하세요"
|
|
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
<datalist id="category-list">
|
|
@foreach($categories as $category)
|
|
<option value="{{ $category }}">
|
|
@endforeach
|
|
</datalist>
|
|
</div>
|
|
<!-- 문서 제목 -->
|
|
<div class="md:col-span-2">
|
|
<label class="block text-sm font-medium text-gray-700 mb-1">문서 제목</label>
|
|
<input type="text" id="title" placeholder="예: 최종 검사 성적서"
|
|
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
</div>
|
|
<!-- 회사명 -->
|
|
<div>
|
|
<label class="block text-sm font-medium text-gray-700 mb-1">회사명</label>
|
|
<input type="text" id="company_name" placeholder="예: 케이디산업"
|
|
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
</div>
|
|
<!-- 활성 상태 -->
|
|
<div class="flex items-center gap-2 self-end pb-2">
|
|
<input type="checkbox" id="is_active" checked class="w-4 h-4 text-blue-600 rounded focus:ring-blue-500">
|
|
<label for="is_active" class="text-sm text-gray-700">활성화</label>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 구분선 -->
|
|
<hr class="my-6 border-gray-200">
|
|
|
|
<!-- Footer 설정 -->
|
|
<h3 class="text-sm font-semibold text-gray-700 mb-4">Footer 설정</h3>
|
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-6">
|
|
<!-- 비고 라벨 -->
|
|
<div>
|
|
<label class="block text-sm font-medium text-gray-700 mb-1">비고 라벨</label>
|
|
<input type="text" id="footer_remark_label" placeholder="예: 부적합 내용"
|
|
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
</div>
|
|
<!-- 종합판정 라벨 -->
|
|
<div>
|
|
<label class="block text-sm font-medium text-gray-700 mb-1">종합판정 라벨</label>
|
|
<input type="text" id="footer_judgement_label" placeholder="예: 종합판정"
|
|
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
</div>
|
|
<!-- 종합판정 옵션 -->
|
|
<div class="md:col-span-2">
|
|
<label class="block text-sm font-medium text-gray-700 mb-1">종합판정 옵션</label>
|
|
<div class="flex items-center gap-2">
|
|
<div id="judgement-options-container" class="flex flex-wrap gap-2 flex-1">
|
|
<!-- 동적으로 렌더링 -->
|
|
</div>
|
|
<button type="button" onclick="addJudgementOption()" class="bg-gray-100 hover:bg-gray-200 text-gray-600 px-3 py-2 rounded-lg text-sm flex-shrink-0">
|
|
+ 옵션 추가
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 결재라인 탭 -->
|
|
<div id="content-approval" class="tab-content bg-white rounded-lg shadow-sm p-6 hidden">
|
|
<div class="flex justify-between items-center mb-4">
|
|
<h3 class="text-lg font-medium text-gray-800">결재라인 설정</h3>
|
|
<button type="button" onclick="addApprovalLine()" class="bg-blue-600 hover:bg-blue-700 text-white px-3 py-1.5 rounded-lg text-sm flex items-center gap-1">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" />
|
|
</svg>
|
|
추가
|
|
</button>
|
|
</div>
|
|
<div id="approval-lines" class="space-y-3">
|
|
<!-- 동적으로 추가됨 -->
|
|
</div>
|
|
<p id="approval-empty" class="text-gray-400 text-center py-8 hidden">결재라인이 없습니다. 추가 버튼을 클릭하세요.</p>
|
|
</div>
|
|
|
|
<!-- 기본필드 탭 -->
|
|
<div id="content-basic-fields" class="tab-content bg-white rounded-lg shadow-sm p-6 hidden">
|
|
<div class="flex justify-between items-center mb-4">
|
|
<div>
|
|
<h3 class="text-lg font-medium text-gray-800">기본필드 설정</h3>
|
|
<p class="text-xs text-gray-500 mt-1">문서 상단에 표시되는 필드입니다 (예: 품명, LOT NO, 검사일자)</p>
|
|
</div>
|
|
<button type="button" onclick="addBasicField()" class="bg-blue-600 hover:bg-blue-700 text-white px-3 py-1.5 rounded-lg text-sm flex items-center gap-1">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" />
|
|
</svg>
|
|
추가
|
|
</button>
|
|
</div>
|
|
<div id="basic-fields-container" class="space-y-3">
|
|
<!-- 동적으로 추가됨 -->
|
|
</div>
|
|
<p id="basic-fields-empty" class="text-gray-400 text-center py-8 hidden">기본필드가 없습니다. 추가 버튼을 클릭하세요.</p>
|
|
</div>
|
|
|
|
<!-- 검사 기준서 탭 -->
|
|
<div id="content-sections" class="tab-content bg-white rounded-lg shadow-sm p-6 hidden">
|
|
<div class="flex justify-between items-center mb-4">
|
|
<h3 class="text-lg font-medium text-gray-800">검사 기준서 섹션</h3>
|
|
<button type="button" onclick="addSection()" class="bg-blue-600 hover:bg-blue-700 text-white px-3 py-1.5 rounded-lg text-sm flex items-center gap-1">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" />
|
|
</svg>
|
|
섹션 추가
|
|
</button>
|
|
</div>
|
|
<div id="sections-container" class="space-y-4">
|
|
<!-- 동적으로 추가됨 -->
|
|
</div>
|
|
<p id="sections-empty" class="text-gray-400 text-center py-8 hidden">검사 기준서 섹션이 없습니다. 섹션 추가 버튼을 클릭하세요.</p>
|
|
</div>
|
|
|
|
<!-- 테이블 컬럼 탭 -->
|
|
<div id="content-columns" class="tab-content bg-white rounded-lg shadow-sm p-6 hidden">
|
|
<div class="flex justify-between items-center mb-4">
|
|
<h3 class="text-lg font-medium text-gray-800">검사 데이터 테이블 컬럼</h3>
|
|
<button type="button" onclick="addColumn()" class="bg-blue-600 hover:bg-blue-700 text-white px-3 py-1.5 rounded-lg text-sm flex items-center gap-1">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" />
|
|
</svg>
|
|
컬럼 추가
|
|
</button>
|
|
</div>
|
|
<div id="columns-container" class="space-y-3">
|
|
<!-- 동적으로 추가됨 -->
|
|
</div>
|
|
<p id="columns-empty" class="text-gray-400 text-center py-8 hidden">테이블 컬럼이 없습니다. 컬럼 추가 버튼을 클릭하세요.</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 미리보기 모달 -->
|
|
<div id="preview-modal" class="fixed inset-0 z-50 hidden overflow-y-auto">
|
|
<div class="flex items-center justify-center min-h-screen p-4">
|
|
<div class="fixed inset-0 bg-black bg-opacity-50" onclick="closePreviewModal()"></div>
|
|
<div class="relative bg-white rounded-lg shadow-xl max-w-5xl w-full max-h-[90vh] overflow-hidden">
|
|
<div class="flex justify-between items-center p-4 border-b">
|
|
<h3 class="text-lg font-bold text-gray-800">문서 미리보기</h3>
|
|
<button onclick="closePreviewModal()" class="text-gray-400 hover:text-gray-600">
|
|
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
<div id="preview-content" class="p-4 overflow-auto max-h-[calc(90vh-120px)]">
|
|
<!-- 미리보기 내용 -->
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
@endsection
|
|
|
|
@push('scripts')
|
|
<script>
|
|
// ===== 상태 관리 =====
|
|
const templateState = {
|
|
id: {{ $template->id ?? 'null' }},
|
|
name: '',
|
|
category: '',
|
|
title: '',
|
|
company_name: '',
|
|
footer_remark_label: '부적합 내용',
|
|
footer_judgement_label: '종합판정',
|
|
footer_judgement_options: ['합격', '불합격', '조건부합격'],
|
|
is_active: true,
|
|
approval_lines: [],
|
|
basic_fields: [],
|
|
sections: [],
|
|
columns: []
|
|
};
|
|
|
|
// 고유 ID 생성
|
|
let uniqueId = 0;
|
|
function generateId() {
|
|
return `temp_${++uniqueId}`;
|
|
}
|
|
|
|
// ===== 초기화 =====
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
@if($template && isset($templateData))
|
|
// 기존 데이터 로드 (컨트롤러에서 준비된 데이터 사용)
|
|
const loadedData = @json($templateData);
|
|
templateState.name = loadedData.name || '';
|
|
templateState.category = loadedData.category || '';
|
|
templateState.title = loadedData.title || '';
|
|
templateState.company_name = loadedData.company_name || '';
|
|
templateState.footer_remark_label = loadedData.footer_remark_label || '';
|
|
templateState.footer_judgement_label = loadedData.footer_judgement_label || '';
|
|
templateState.footer_judgement_options = loadedData.footer_judgement_options || [];
|
|
templateState.is_active = loadedData.is_active || false;
|
|
templateState.approval_lines = loadedData.approval_lines || [];
|
|
templateState.basic_fields = loadedData.basic_fields || [];
|
|
templateState.sections = loadedData.sections || [];
|
|
templateState.columns = loadedData.columns || [];
|
|
@endif
|
|
|
|
// UI 초기화
|
|
initBasicInfo();
|
|
renderJudgementOptions();
|
|
renderApprovalLines();
|
|
renderBasicFields();
|
|
renderSections();
|
|
renderColumns();
|
|
});
|
|
|
|
// ===== 기본정보 =====
|
|
function initBasicInfo() {
|
|
document.getElementById('name').value = templateState.name || '';
|
|
document.getElementById('category').value = templateState.category || '';
|
|
document.getElementById('title').value = templateState.title || '';
|
|
document.getElementById('company_name').value = templateState.company_name || '';
|
|
document.getElementById('footer_remark_label').value = templateState.footer_remark_label || '';
|
|
document.getElementById('footer_judgement_label').value = templateState.footer_judgement_label || '';
|
|
document.getElementById('is_active').checked = templateState.is_active;
|
|
|
|
// 변경 이벤트 바인딩
|
|
['name', 'category', 'title', 'company_name', 'footer_remark_label', 'footer_judgement_label'].forEach(field => {
|
|
document.getElementById(field).addEventListener('input', function() {
|
|
templateState[field] = this.value;
|
|
});
|
|
});
|
|
document.getElementById('is_active').addEventListener('change', function() {
|
|
templateState.is_active = this.checked;
|
|
});
|
|
}
|
|
|
|
// ===== 종합판정 옵션 =====
|
|
function renderJudgementOptions() {
|
|
const container = document.getElementById('judgement-options-container');
|
|
container.innerHTML = templateState.footer_judgement_options.map((opt, idx) => `
|
|
<div class="flex items-center gap-1 bg-gray-100 rounded-lg px-2 py-1">
|
|
<input type="text" value="${escapeHtml(opt)}" placeholder="옵션"
|
|
onchange="updateJudgementOption(${idx}, this.value)"
|
|
class="w-24 px-2 py-1 border border-gray-200 rounded text-sm bg-white">
|
|
<button onclick="removeJudgementOption(${idx})" class="text-red-400 hover:text-red-600">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
`).join('');
|
|
}
|
|
|
|
function addJudgementOption() {
|
|
templateState.footer_judgement_options.push('');
|
|
renderJudgementOptions();
|
|
// 마지막 input에 포커스
|
|
const inputs = document.querySelectorAll('#judgement-options-container input[type="text"]');
|
|
if (inputs.length) inputs[inputs.length - 1].focus();
|
|
}
|
|
|
|
function updateJudgementOption(idx, value) {
|
|
templateState.footer_judgement_options[idx] = value;
|
|
}
|
|
|
|
function removeJudgementOption(idx) {
|
|
templateState.footer_judgement_options.splice(idx, 1);
|
|
renderJudgementOptions();
|
|
}
|
|
|
|
// ===== 기본필드 (basic_fields) =====
|
|
function addBasicField() {
|
|
templateState.basic_fields.push({
|
|
id: generateId(),
|
|
label: '',
|
|
field_type: 'text',
|
|
default_value: ''
|
|
});
|
|
renderBasicFields();
|
|
}
|
|
|
|
function removeBasicField(id) {
|
|
templateState.basic_fields = templateState.basic_fields.filter(f => f.id !== id);
|
|
renderBasicFields();
|
|
}
|
|
|
|
function updateBasicField(id, field, value) {
|
|
const bf = templateState.basic_fields.find(f => f.id === id);
|
|
if (bf) bf[field] = value;
|
|
}
|
|
|
|
function renderBasicFields() {
|
|
const container = document.getElementById('basic-fields-container');
|
|
const emptyMsg = document.getElementById('basic-fields-empty');
|
|
|
|
if (templateState.basic_fields.length === 0) {
|
|
container.innerHTML = '';
|
|
emptyMsg.classList.remove('hidden');
|
|
return;
|
|
}
|
|
|
|
emptyMsg.classList.add('hidden');
|
|
container.innerHTML = templateState.basic_fields.map((field, idx) => `
|
|
<div class="flex items-center gap-3 p-3 bg-gray-50 rounded-lg cursor-move" data-bf-id="${field.id}">
|
|
<span class="text-gray-400 font-bold" title="드래그하여 순서 변경">⋮⋮</span>
|
|
<span class="text-gray-400 font-mono text-sm w-6">${idx + 1}</span>
|
|
<input type="text" value="${escapeHtml(field.label)}" placeholder="필드명 (예: 품명, LOT NO)"
|
|
onchange="updateBasicField('${field.id}', 'label', this.value)"
|
|
class="flex-1 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<select onchange="updateBasicField('${field.id}', 'field_type', this.value)"
|
|
class="w-28 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<option value="text" ${field.field_type === 'text' ? 'selected' : ''}>텍스트</option>
|
|
<option value="date" ${field.field_type === 'date' ? 'selected' : ''}>날짜</option>
|
|
<option value="number" ${field.field_type === 'number' ? 'selected' : ''}>숫자</option>
|
|
<option value="select" ${field.field_type === 'select' ? 'selected' : ''}>선택</option>
|
|
</select>
|
|
<input type="text" value="${escapeHtml(field.default_value || '')}" placeholder="기본값"
|
|
onchange="updateBasicField('${field.id}', 'default_value', this.value)"
|
|
class="w-32 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<button onclick="removeBasicField('${field.id}')" class="text-red-500 hover:text-red-700 p-1">
|
|
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
`).join('');
|
|
}
|
|
|
|
// ===== 탭 전환 =====
|
|
function switchTab(tabId) {
|
|
// 모든 탭 버튼 비활성화
|
|
document.querySelectorAll('.tab-btn').forEach(btn => {
|
|
btn.classList.remove('border-blue-500', 'text-blue-600');
|
|
btn.classList.add('border-transparent', 'text-gray-500');
|
|
});
|
|
// 선택된 탭 버튼 활성화
|
|
const activeBtn = document.getElementById(`tab-${tabId}`);
|
|
activeBtn.classList.add('border-blue-500', 'text-blue-600');
|
|
activeBtn.classList.remove('border-transparent', 'text-gray-500');
|
|
|
|
// 모든 컨텐츠 숨김
|
|
document.querySelectorAll('.tab-content').forEach(content => {
|
|
content.classList.add('hidden');
|
|
});
|
|
// 선택된 컨텐츠 표시
|
|
document.getElementById(`content-${tabId}`).classList.remove('hidden');
|
|
}
|
|
|
|
// ===== 결재라인 =====
|
|
function addApprovalLine() {
|
|
const line = { id: generateId(), name: '', dept: '', role: '' };
|
|
templateState.approval_lines.push(line);
|
|
renderApprovalLines();
|
|
}
|
|
|
|
function removeApprovalLine(id) {
|
|
templateState.approval_lines = templateState.approval_lines.filter(l => l.id !== id);
|
|
renderApprovalLines();
|
|
}
|
|
|
|
function updateApprovalLine(id, field, value) {
|
|
const line = templateState.approval_lines.find(l => l.id === id);
|
|
if (line) line[field] = value;
|
|
}
|
|
|
|
function renderApprovalLines() {
|
|
const container = document.getElementById('approval-lines');
|
|
const emptyMsg = document.getElementById('approval-empty');
|
|
|
|
if (templateState.approval_lines.length === 0) {
|
|
container.innerHTML = '';
|
|
emptyMsg.classList.remove('hidden');
|
|
return;
|
|
}
|
|
|
|
emptyMsg.classList.add('hidden');
|
|
container.innerHTML = templateState.approval_lines.map((line, idx) => `
|
|
<div class="flex items-center gap-3 p-3 bg-gray-50 rounded-lg cursor-move" data-id="${line.id}">
|
|
<span class="text-gray-400 font-bold" title="드래그하여 순서 변경">⋮⋮</span>
|
|
<span class="text-gray-400 font-mono text-sm w-6">${idx + 1}</span>
|
|
<input type="text" value="${escapeHtml(line.name)}" placeholder="단계명 (작성, 검토, 승인)"
|
|
onchange="updateApprovalLine('${line.id}', 'name', this.value)"
|
|
class="flex-1 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<input type="text" value="${escapeHtml(line.dept)}" placeholder="부서"
|
|
onchange="updateApprovalLine('${line.id}', 'dept', this.value)"
|
|
class="w-32 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<input type="text" value="${escapeHtml(line.role)}" placeholder="직책/담당자"
|
|
onchange="updateApprovalLine('${line.id}', 'role', this.value)"
|
|
class="w-32 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<button onclick="removeApprovalLine('${line.id}')" class="text-red-500 hover:text-red-700 p-1">
|
|
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
`).join('');
|
|
}
|
|
|
|
// ===== 섹션 관리 =====
|
|
function addSection() {
|
|
const section = { id: generateId(), title: '', image_path: null, items: [] };
|
|
templateState.sections.push(section);
|
|
renderSections();
|
|
}
|
|
|
|
function removeSection(id) {
|
|
templateState.sections = templateState.sections.filter(s => s.id !== id);
|
|
renderSections();
|
|
}
|
|
|
|
function updateSection(id, field, value) {
|
|
const section = templateState.sections.find(s => s.id === id);
|
|
if (section) section[field] = value;
|
|
}
|
|
|
|
function addSectionItem(sectionId) {
|
|
const section = templateState.sections.find(s => s.id === sectionId);
|
|
if (section) {
|
|
section.items.push({
|
|
id: generateId(),
|
|
category: '',
|
|
item: '',
|
|
standard: '',
|
|
method: '',
|
|
frequency: '',
|
|
regulation: ''
|
|
});
|
|
renderSections();
|
|
}
|
|
}
|
|
|
|
function removeSectionItem(sectionId, itemId) {
|
|
const section = templateState.sections.find(s => s.id === sectionId);
|
|
if (section) {
|
|
section.items = section.items.filter(i => i.id !== itemId);
|
|
renderSections();
|
|
}
|
|
}
|
|
|
|
function updateSectionItem(sectionId, itemId, field, value) {
|
|
const section = templateState.sections.find(s => s.id === sectionId);
|
|
if (section) {
|
|
const item = section.items.find(i => i.id === itemId);
|
|
if (item) item[field] = value;
|
|
}
|
|
}
|
|
|
|
function renderSections() {
|
|
const container = document.getElementById('sections-container');
|
|
const emptyMsg = document.getElementById('sections-empty');
|
|
|
|
if (templateState.sections.length === 0) {
|
|
container.innerHTML = '';
|
|
emptyMsg.classList.remove('hidden');
|
|
return;
|
|
}
|
|
|
|
emptyMsg.classList.add('hidden');
|
|
container.innerHTML = templateState.sections.map((section, idx) => `
|
|
<div class="border border-gray-200 rounded-lg overflow-hidden" data-section-id="${section.id}">
|
|
<div class="bg-gray-50 p-4 flex items-center justify-between">
|
|
<div class="flex items-center gap-3 flex-1">
|
|
<span class="text-gray-400 font-bold cursor-move drag-handle" title="드래그하여 순서 변경">⋮⋮</span>
|
|
<input type="text" value="${escapeHtml(section.title)}" placeholder="섹션 제목 (예: 가이드레일)"
|
|
onchange="updateSection('${section.id}', 'title', this.value)"
|
|
class="flex-1 px-3 py-2 border border-gray-300 rounded-lg text-sm font-medium">
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<label class="text-gray-600 hover:text-blue-600 text-sm flex items-center gap-1 cursor-pointer">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z" />
|
|
</svg>
|
|
이미지
|
|
<input type="file" accept="image/*" class="hidden" onchange="uploadSectionImage('${section.id}', this)">
|
|
</label>
|
|
<button onclick="addSectionItem('${section.id}')" class="text-blue-600 hover:text-blue-800 text-sm flex items-center gap-1">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" />
|
|
</svg>
|
|
항목 추가
|
|
</button>
|
|
<button onclick="removeSection('${section.id}')" class="text-red-500 hover:text-red-700 p-1">
|
|
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
${section.image_path ? `
|
|
<div class="px-4 py-2 bg-blue-50 flex items-center justify-between">
|
|
<div class="flex items-center gap-2">
|
|
<img src="/storage/${section.image_path}" alt="섹션 이미지" class="h-16 rounded border border-gray-200">
|
|
<span class="text-xs text-gray-500">이미지 첨부됨</span>
|
|
</div>
|
|
<button onclick="removeSectionImage('${section.id}')" class="text-red-500 hover:text-red-700 text-xs">
|
|
이미지 삭제
|
|
</button>
|
|
</div>
|
|
` : ''}
|
|
<div class="p-4">
|
|
${section.items.length > 0 ? `
|
|
<div class="overflow-x-auto">
|
|
<table class="w-full text-sm">
|
|
<thead class="bg-gray-100">
|
|
<tr>
|
|
<th class="px-2 py-2 text-left text-xs font-medium text-gray-500 w-24">구분</th>
|
|
<th class="px-2 py-2 text-left text-xs font-medium text-gray-500">검사항목</th>
|
|
<th class="px-2 py-2 text-left text-xs font-medium text-gray-500">검사기준</th>
|
|
<th class="px-2 py-2 text-left text-xs font-medium text-gray-500 w-24">검사방법</th>
|
|
<th class="px-2 py-2 text-left text-xs font-medium text-gray-500 w-24">검사주기</th>
|
|
<th class="px-2 py-2 text-left text-xs font-medium text-gray-500 w-28">관련규정</th>
|
|
<th class="px-2 py-2 w-10"></th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
${section.items.map(item => `
|
|
<tr class="border-t" data-item-id="${item.id}">
|
|
<td class="px-1 py-1">
|
|
<input type="text" value="${escapeHtml(item.category)}" placeholder="구분"
|
|
onchange="updateSectionItem('${section.id}', '${item.id}', 'category', this.value)"
|
|
class="w-full px-2 py-1 border border-gray-200 rounded text-xs">
|
|
</td>
|
|
<td class="px-1 py-1">
|
|
<input type="text" value="${escapeHtml(item.item)}" placeholder="검사항목"
|
|
onchange="updateSectionItem('${section.id}', '${item.id}', 'item', this.value)"
|
|
class="w-full px-2 py-1 border border-gray-200 rounded text-xs">
|
|
</td>
|
|
<td class="px-1 py-1">
|
|
<input type="text" value="${escapeHtml(item.standard)}" placeholder="검사기준"
|
|
onchange="updateSectionItem('${section.id}', '${item.id}', 'standard', this.value)"
|
|
class="w-full px-2 py-1 border border-gray-200 rounded text-xs">
|
|
</td>
|
|
<td class="px-1 py-1">
|
|
<input type="text" value="${escapeHtml(item.method)}" placeholder="방법"
|
|
onchange="updateSectionItem('${section.id}', '${item.id}', 'method', this.value)"
|
|
class="w-full px-2 py-1 border border-gray-200 rounded text-xs">
|
|
</td>
|
|
<td class="px-1 py-1">
|
|
<input type="text" value="${escapeHtml(item.frequency)}" placeholder="주기"
|
|
onchange="updateSectionItem('${section.id}', '${item.id}', 'frequency', this.value)"
|
|
class="w-full px-2 py-1 border border-gray-200 rounded text-xs">
|
|
</td>
|
|
<td class="px-1 py-1">
|
|
<input type="text" value="${escapeHtml(item.regulation)}" placeholder="규정"
|
|
onchange="updateSectionItem('${section.id}', '${item.id}', 'regulation', this.value)"
|
|
class="w-full px-2 py-1 border border-gray-200 rounded text-xs">
|
|
</td>
|
|
<td class="px-1 py-1">
|
|
<button onclick="removeSectionItem('${section.id}', '${item.id}')" class="text-red-400 hover:text-red-600">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
</button>
|
|
</td>
|
|
</tr>
|
|
`).join('')}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
` : `
|
|
<p class="text-gray-400 text-center py-4 text-sm">항목이 없습니다. 항목 추가 버튼을 클릭하세요.</p>
|
|
`}
|
|
</div>
|
|
</div>
|
|
`).join('');
|
|
}
|
|
|
|
// ===== 컬럼 관리 =====
|
|
function addColumn() {
|
|
const column = { id: generateId(), label: '', width: '100px', column_type: 'text', group_name: '', sub_labels: null };
|
|
templateState.columns.push(column);
|
|
renderColumns();
|
|
}
|
|
|
|
function removeColumn(id) {
|
|
templateState.columns = templateState.columns.filter(c => c.id !== id);
|
|
renderColumns();
|
|
}
|
|
|
|
function updateColumn(id, field, value) {
|
|
const column = templateState.columns.find(c => c.id === id);
|
|
if (column) column[field] = value;
|
|
}
|
|
|
|
function renderColumns() {
|
|
const container = document.getElementById('columns-container');
|
|
const emptyMsg = document.getElementById('columns-empty');
|
|
|
|
if (templateState.columns.length === 0) {
|
|
container.innerHTML = '';
|
|
emptyMsg.classList.remove('hidden');
|
|
return;
|
|
}
|
|
|
|
emptyMsg.classList.add('hidden');
|
|
container.innerHTML = templateState.columns.map((col, idx) => `
|
|
<div class="p-3 bg-gray-50 rounded-lg cursor-move" data-column-id="${col.id}">
|
|
<div class="flex items-center gap-3">
|
|
<span class="text-gray-400 font-bold" title="드래그하여 순서 변경">⋮⋮</span>
|
|
<span class="text-gray-400 font-mono text-sm w-6">${idx + 1}</span>
|
|
<input type="text" value="${escapeHtml(col.label)}" placeholder="컬럼명"
|
|
onchange="updateColumn('${col.id}', 'label', this.value)"
|
|
class="flex-1 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<input type="text" value="${escapeHtml(col.width)}" placeholder="너비"
|
|
onchange="updateColumn('${col.id}', 'width', this.value)"
|
|
class="w-24 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<select onchange="handleColumnTypeChange('${col.id}', this.value)"
|
|
class="w-32 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<option value="text" ${col.column_type === 'text' ? 'selected' : ''}>텍스트</option>
|
|
<option value="check" ${col.column_type === 'check' ? 'selected' : ''}>체크</option>
|
|
<option value="measurement" ${col.column_type === 'measurement' ? 'selected' : ''}>측정값</option>
|
|
<option value="select" ${col.column_type === 'select' ? 'selected' : ''}>선택</option>
|
|
<option value="complex" ${col.column_type === 'complex' ? 'selected' : ''}>복합(하위컬럼)</option>
|
|
</select>
|
|
<input type="text" value="${escapeHtml(col.group_name || '')}" placeholder="그룹명"
|
|
onchange="updateColumn('${col.id}', 'group_name', this.value)"
|
|
class="w-28 px-3 py-2 border border-gray-300 rounded-lg text-sm">
|
|
<button onclick="removeColumn('${col.id}')" class="text-red-500 hover:text-red-700 p-1">
|
|
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
${col.column_type === 'complex' ? `
|
|
<div class="ml-10 mt-2 p-2 bg-white rounded border border-gray-200">
|
|
<div class="flex items-center gap-2 mb-2">
|
|
<span class="text-xs font-medium text-gray-600">하위 라벨:</span>
|
|
<button onclick="addSubLabel('${col.id}')" class="text-blue-600 hover:text-blue-800 text-xs">+ 추가</button>
|
|
</div>
|
|
<div class="flex flex-wrap gap-2">
|
|
${(col.sub_labels || []).map((label, si) => `
|
|
<div class="flex items-center gap-1 bg-blue-50 rounded px-2 py-1">
|
|
<input type="text" value="${escapeHtml(label)}" placeholder="${si + 1}"
|
|
onchange="updateSubLabel('${col.id}', ${si}, this.value)"
|
|
class="w-16 px-1 py-0.5 border border-blue-200 rounded text-xs bg-white text-center">
|
|
<button onclick="removeSubLabel('${col.id}', ${si})" class="text-red-400 hover:text-red-600">
|
|
<svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
`).join('')}
|
|
${(!col.sub_labels || col.sub_labels.length === 0) ? '<span class="text-xs text-gray-400">하위 라벨이 없습니다.</span>' : ''}
|
|
</div>
|
|
</div>
|
|
` : ''}
|
|
</div>
|
|
`).join('');
|
|
}
|
|
|
|
// ===== 컬럼 타입 변경 처리 =====
|
|
function handleColumnTypeChange(id, value) {
|
|
const column = templateState.columns.find(c => c.id === id);
|
|
if (column) {
|
|
column.column_type = value;
|
|
if (value === 'complex' && !column.sub_labels) {
|
|
column.sub_labels = ['1', '2', '3', '4', '5'];
|
|
}
|
|
if (value !== 'complex') {
|
|
column.sub_labels = null;
|
|
}
|
|
renderColumns();
|
|
}
|
|
}
|
|
|
|
function addSubLabel(colId) {
|
|
const column = templateState.columns.find(c => c.id === colId);
|
|
if (column) {
|
|
if (!column.sub_labels) column.sub_labels = [];
|
|
column.sub_labels.push('');
|
|
renderColumns();
|
|
// 마지막 input에 포커스
|
|
setTimeout(() => {
|
|
const colEl = document.querySelector(`[data-column-id="${colId}"]`);
|
|
if (colEl) {
|
|
const inputs = colEl.querySelectorAll('.bg-blue-50 input');
|
|
if (inputs.length) inputs[inputs.length - 1].focus();
|
|
}
|
|
}, 50);
|
|
}
|
|
}
|
|
|
|
function updateSubLabel(colId, idx, value) {
|
|
const column = templateState.columns.find(c => c.id === colId);
|
|
if (column && column.sub_labels) {
|
|
column.sub_labels[idx] = value;
|
|
}
|
|
}
|
|
|
|
function removeSubLabel(colId, idx) {
|
|
const column = templateState.columns.find(c => c.id === colId);
|
|
if (column && column.sub_labels) {
|
|
column.sub_labels.splice(idx, 1);
|
|
renderColumns();
|
|
}
|
|
}
|
|
|
|
// ===== 저장 =====
|
|
function saveTemplate() {
|
|
const name = document.getElementById('name').value.trim();
|
|
if (!name) {
|
|
showToast('양식명은 필수입니다.', 'warning');
|
|
switchTab('basic');
|
|
document.getElementById('name').focus();
|
|
return;
|
|
}
|
|
|
|
const data = {
|
|
name: name,
|
|
category: document.getElementById('category').value,
|
|
title: document.getElementById('title').value,
|
|
|
|
company_name: document.getElementById('company_name').value,
|
|
footer_remark_label: document.getElementById('footer_remark_label').value,
|
|
footer_judgement_label: document.getElementById('footer_judgement_label').value,
|
|
footer_judgement_options: templateState.footer_judgement_options.filter(o => o.trim() !== ''),
|
|
is_active: document.getElementById('is_active').checked,
|
|
approval_lines: templateState.approval_lines,
|
|
basic_fields: templateState.basic_fields,
|
|
sections: templateState.sections,
|
|
columns: templateState.columns
|
|
};
|
|
|
|
const url = templateState.id
|
|
? `/api/admin/document-templates/${templateState.id}`
|
|
: '/api/admin/document-templates';
|
|
const method = templateState.id ? 'PUT' : 'POST';
|
|
|
|
fetch(url, {
|
|
method: method,
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}',
|
|
'Accept': 'application/json'
|
|
},
|
|
body: JSON.stringify(data)
|
|
})
|
|
.then(response => response.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
showToast(result.message || '저장되었습니다.', 'success');
|
|
if (!templateState.id && result.data?.id) {
|
|
// 새로 생성된 경우 편집 페이지로 이동
|
|
window.location.href = `/document-templates/${result.data.id}/edit`;
|
|
}
|
|
} else {
|
|
showToast(result.message || '저장에 실패했습니다.', 'error');
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error('Save error:', error);
|
|
showToast('저장 중 오류가 발생했습니다.', 'error');
|
|
});
|
|
}
|
|
|
|
// ===== 미리보기 =====
|
|
function openPreviewModal() {
|
|
const content = document.getElementById('preview-content');
|
|
content.innerHTML = generatePreviewHtml();
|
|
document.getElementById('preview-modal').classList.remove('hidden');
|
|
}
|
|
|
|
function closePreviewModal() {
|
|
document.getElementById('preview-modal').classList.add('hidden');
|
|
}
|
|
|
|
function generatePreviewHtml() {
|
|
const title = document.getElementById('title').value || '문서 양식';
|
|
const companyName = document.getElementById('company_name').value || '회사명';
|
|
const remarkLabel = document.getElementById('footer_remark_label').value || '비고';
|
|
const judgementLabel = document.getElementById('footer_judgement_label').value || '종합판정';
|
|
const judgementOptions = templateState.footer_judgement_options.filter(o => o.trim() !== '');
|
|
|
|
// 결재란 생성
|
|
const renderApproval = () => {
|
|
if (templateState.approval_lines.length === 0) {
|
|
return '<span class="text-xs text-gray-400">결재라인 미설정</span>';
|
|
}
|
|
return `<table class="border-collapse text-xs">
|
|
<tr>${templateState.approval_lines.map(l => `<td class="border border-gray-400 px-3 py-1 bg-gray-100 text-center font-medium">${escapeHtml(l.name || '-')}</td>`).join('')}</tr>
|
|
<tr>${templateState.approval_lines.map(l => `<td class="border border-gray-400 px-3 py-1 text-center"><div class="text-gray-400 text-xs">${escapeHtml(l.dept || '')}</div><div class="h-6"></div></td>`).join('')}</tr>
|
|
</table>`;
|
|
};
|
|
|
|
// 기본필드 테이블 생성
|
|
const renderBasicInfo = () => {
|
|
if (templateState.basic_fields.length === 0) {
|
|
return '<p class="text-xs text-gray-400 mb-4">기본필드 미설정</p>';
|
|
}
|
|
const fields = templateState.basic_fields;
|
|
const rows = [];
|
|
for (let i = 0; i < fields.length; i += 2) {
|
|
const f1 = fields[i];
|
|
const f2 = fields[i + 1];
|
|
rows.push(`<tr>
|
|
<td class="border border-gray-400 px-2 py-1.5 bg-gray-100 font-medium" style="width:100px">${escapeHtml(f1.label || '-')}</td>
|
|
<td class="border border-gray-400 px-2 py-1.5 text-gray-400">${escapeHtml(f1.default_value || '(입력)')}</td>
|
|
${f2 ? `<td class="border border-gray-400 px-2 py-1.5 bg-gray-100 font-medium" style="width:100px">${escapeHtml(f2.label || '-')}</td>
|
|
<td class="border border-gray-400 px-2 py-1.5 text-gray-400">${escapeHtml(f2.default_value || '(입력)')}</td>` : '<td class="border border-gray-400 px-2 py-1.5" colspan="2"></td>'}
|
|
</tr>`);
|
|
}
|
|
return `<table class="w-full border-collapse text-xs mb-4">${rows.join('')}</table>`;
|
|
};
|
|
|
|
// 검사 데이터 컬럼 헤더 생성
|
|
const renderColumnHeaders = () => {
|
|
if (templateState.columns.length === 0) return '';
|
|
const hasComplex = templateState.columns.some(c => c.column_type === 'complex' && c.sub_labels && c.sub_labels.length > 0);
|
|
let headerRow1 = '';
|
|
let headerRow2 = '';
|
|
templateState.columns.forEach(col => {
|
|
if (col.column_type === 'complex' && col.sub_labels && col.sub_labels.length > 0) {
|
|
headerRow1 += `<th class="border border-gray-400 px-2 py-1.5 bg-gray-100" colspan="${col.sub_labels.length}">${escapeHtml(col.label)}</th>`;
|
|
col.sub_labels.forEach(sl => {
|
|
headerRow2 += `<th class="border border-gray-400 px-1 py-1 bg-gray-100" style="width:40px">${escapeHtml(sl)}</th>`;
|
|
});
|
|
} else {
|
|
headerRow1 += `<th class="border border-gray-400 px-2 py-1.5 bg-gray-100" ${hasComplex ? 'rowspan="2"' : ''} style="width:${col.width || '80px'}">${escapeHtml(col.label)}</th>`;
|
|
}
|
|
});
|
|
if (hasComplex) {
|
|
return `<tr>${headerRow1}</tr><tr>${headerRow2}</tr>`;
|
|
}
|
|
return `<tr>${headerRow1}</tr>`;
|
|
};
|
|
|
|
// 검사항목 행 생성
|
|
const renderInspectionRows = () => {
|
|
const allItems = [];
|
|
templateState.sections.forEach(section => {
|
|
if (section.items) {
|
|
section.items.forEach(item => allItems.push(item));
|
|
}
|
|
});
|
|
if (allItems.length === 0) {
|
|
const colSpan = templateState.columns.reduce((sum, c) => sum + (c.column_type === 'complex' && c.sub_labels ? c.sub_labels.length : 1), 0) || 1;
|
|
return `<tr><td colspan="${colSpan}" class="text-center py-4 text-gray-400 border border-gray-400">검사항목이 없습니다.</td></tr>`;
|
|
}
|
|
return allItems.map((item, idx) => {
|
|
const cells = templateState.columns.map(col => {
|
|
if (col.column_type === 'complex' && col.sub_labels && col.sub_labels.length > 0) {
|
|
return col.sub_labels.map(() => '<td class="border border-gray-400 px-1 py-1.5 text-center text-gray-300">-</td>').join('');
|
|
}
|
|
// 기본 컬럼: 검사항목 데이터 매칭 시도
|
|
const label = (col.label || '').trim();
|
|
let val = '-';
|
|
if (label === 'NO' || label === 'no') val = idx + 1;
|
|
else if (label.includes('검사항목') || label.includes('항목')) val = escapeHtml(item.item || '-');
|
|
else if (label.includes('검사기준') || label.includes('기준')) val = escapeHtml(item.standard || '-');
|
|
else if (label.includes('검사방')) val = escapeHtml(item.method || '-');
|
|
else if (label.includes('주기')) val = escapeHtml(item.frequency || '-');
|
|
else if (label.includes('판정')) val = '<span class="text-gray-400">-</span>';
|
|
return `<td class="border border-gray-400 px-2 py-1.5 text-center">${val}</td>`;
|
|
}).join('');
|
|
return `<tr>${cells}</tr>`;
|
|
}).join('');
|
|
};
|
|
|
|
return `
|
|
<div class="bg-white p-6" style="font-family: 'Malgun Gothic', sans-serif; font-size: 12px; max-width: 900px; margin: 0 auto;">
|
|
<!-- 헤더: 로고 + 제목 + 결재란 -->
|
|
<div class="flex justify-between items-start mb-4">
|
|
<div class="text-center" style="width: 80px;">
|
|
<div class="text-2xl font-bold">KD</div>
|
|
<div class="text-xs">${escapeHtml(companyName)}</div>
|
|
</div>
|
|
<div class="flex-1 text-center">
|
|
<h1 class="text-xl font-bold tracking-widest">${escapeHtml(title)}</h1>
|
|
</div>
|
|
<div>${renderApproval()}</div>
|
|
</div>
|
|
|
|
<!-- 기본필드 정보 -->
|
|
${renderBasicInfo()}
|
|
|
|
<!-- 검사 기준 이미지 -->
|
|
${templateState.sections.filter(s => s.image_path).map(s => `
|
|
<div class="mb-4 text-center">
|
|
<p class="text-xs font-medium text-gray-600 mb-1">${escapeHtml(s.title || '검사 기준')}</p>
|
|
<img src="/storage/${s.image_path}" alt="${escapeHtml(s.title)}" class="max-h-40 mx-auto border rounded">
|
|
</div>
|
|
`).join('')}
|
|
|
|
<!-- 검사 데이터 테이블 -->
|
|
${templateState.columns.length > 0 ? `
|
|
<table class="w-full border-collapse text-xs">
|
|
<thead>${renderColumnHeaders()}</thead>
|
|
<tbody>${renderInspectionRows()}</tbody>
|
|
</table>
|
|
` : '<p class="text-xs text-gray-400 text-center py-4">테이블 컬럼이 설정되지 않았습니다.</p>'}
|
|
|
|
<!-- Footer: 비고 + 종합판정 -->
|
|
<div class="mt-4 flex gap-4">
|
|
<div class="flex-1">
|
|
<table class="w-full border-collapse text-xs">
|
|
<tr>
|
|
<td class="border border-gray-400 px-3 py-2 bg-gray-100 font-medium" style="width:100px">${escapeHtml(remarkLabel)}</td>
|
|
<td class="border border-gray-400 px-3 py-2 text-gray-400">(내용 입력)</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
<div>
|
|
<table class="border-collapse text-xs">
|
|
<tr>
|
|
<td class="border border-gray-400 px-4 py-2 bg-gray-100 font-medium">${escapeHtml(judgementLabel)}</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="border border-gray-400 px-4 py-3 text-center">
|
|
${judgementOptions.length > 0 ? judgementOptions.map(opt => `<span class="inline-block mx-1 px-2 py-0.5 border border-gray-300 rounded text-gray-500">${escapeHtml(opt)}</span>`).join('') : '<span class="text-gray-400">옵션 미설정</span>'}
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
`;
|
|
}
|
|
|
|
// ===== 이미지 업로드 =====
|
|
function uploadSectionImage(sectionId, input) {
|
|
const file = input.files[0];
|
|
if (!file) return;
|
|
|
|
const formData = new FormData();
|
|
formData.append('image', file);
|
|
|
|
showToast('이미지 업로드 중...', 'info');
|
|
|
|
fetch('/api/admin/document-templates/upload-image', {
|
|
method: 'POST',
|
|
headers: {
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}',
|
|
'Accept': 'application/json'
|
|
},
|
|
body: formData
|
|
})
|
|
.then(response => response.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
const section = templateState.sections.find(s => s.id === sectionId);
|
|
if (section) {
|
|
section.image_path = result.path;
|
|
renderSections();
|
|
showToast('이미지가 업로드되었습니다.', 'success');
|
|
}
|
|
} else {
|
|
showToast(result.message || '이미지 업로드에 실패했습니다.', 'error');
|
|
}
|
|
})
|
|
.catch(error => {
|
|
console.error('Upload error:', error);
|
|
showToast('이미지 업로드 중 오류가 발생했습니다.', 'error');
|
|
});
|
|
|
|
input.value = '';
|
|
}
|
|
|
|
function removeSectionImage(sectionId) {
|
|
const section = templateState.sections.find(s => s.id === sectionId);
|
|
if (section) {
|
|
section.image_path = null;
|
|
renderSections();
|
|
}
|
|
}
|
|
|
|
// ===== SortableJS 초기화 =====
|
|
function initSortable() {
|
|
// 섹션 정렬
|
|
const sectionsContainer = document.getElementById('sections-container');
|
|
if (sectionsContainer && typeof Sortable !== 'undefined') {
|
|
new Sortable(sectionsContainer, {
|
|
animation: 150,
|
|
handle: '.drag-handle',
|
|
onEnd: function(evt) {
|
|
const newOrder = [];
|
|
sectionsContainer.querySelectorAll('[data-section-id]').forEach((el, idx) => {
|
|
const sectionId = el.dataset.sectionId;
|
|
const section = templateState.sections.find(s => s.id === sectionId);
|
|
if (section) newOrder.push(section);
|
|
});
|
|
templateState.sections = newOrder;
|
|
}
|
|
});
|
|
}
|
|
|
|
// 결재라인 정렬
|
|
const approvalContainer = document.getElementById('approval-lines');
|
|
if (approvalContainer && typeof Sortable !== 'undefined') {
|
|
new Sortable(approvalContainer, {
|
|
animation: 150,
|
|
onEnd: function(evt) {
|
|
const newOrder = [];
|
|
approvalContainer.querySelectorAll('[data-id]').forEach((el) => {
|
|
const id = el.dataset.id;
|
|
const line = templateState.approval_lines.find(l => l.id === id);
|
|
if (line) newOrder.push(line);
|
|
});
|
|
templateState.approval_lines = newOrder;
|
|
}
|
|
});
|
|
}
|
|
|
|
// 기본필드 정렬
|
|
const basicFieldsContainer = document.getElementById('basic-fields-container');
|
|
if (basicFieldsContainer && typeof Sortable !== 'undefined') {
|
|
new Sortable(basicFieldsContainer, {
|
|
animation: 150,
|
|
onEnd: function(evt) {
|
|
const newOrder = [];
|
|
basicFieldsContainer.querySelectorAll('[data-bf-id]').forEach((el) => {
|
|
const id = el.dataset.bfId;
|
|
const field = templateState.basic_fields.find(f => f.id === id);
|
|
if (field) newOrder.push(field);
|
|
});
|
|
templateState.basic_fields = newOrder;
|
|
}
|
|
});
|
|
}
|
|
|
|
// 컬럼 정렬
|
|
const columnsContainer = document.getElementById('columns-container');
|
|
if (columnsContainer && typeof Sortable !== 'undefined') {
|
|
new Sortable(columnsContainer, {
|
|
animation: 150,
|
|
onEnd: function(evt) {
|
|
const newOrder = [];
|
|
columnsContainer.querySelectorAll('[data-column-id]').forEach((el) => {
|
|
const id = el.dataset.columnId;
|
|
const col = templateState.columns.find(c => c.id === id);
|
|
if (col) newOrder.push(col);
|
|
});
|
|
templateState.columns = newOrder;
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
// 렌더링 후 SortableJS 재초기화
|
|
const originalRenderApprovalLines = renderApprovalLines;
|
|
renderApprovalLines = function() {
|
|
originalRenderApprovalLines();
|
|
setTimeout(initSortable, 100);
|
|
};
|
|
|
|
const originalRenderBasicFields = renderBasicFields;
|
|
renderBasicFields = function() {
|
|
originalRenderBasicFields();
|
|
setTimeout(initSortable, 100);
|
|
};
|
|
|
|
const originalRenderSections = renderSections;
|
|
renderSections = function() {
|
|
originalRenderSections();
|
|
setTimeout(initSortable, 100);
|
|
};
|
|
|
|
const originalRenderColumns = renderColumns;
|
|
renderColumns = function() {
|
|
originalRenderColumns();
|
|
setTimeout(initSortable, 100);
|
|
};
|
|
|
|
// 초기 SortableJS 로드
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
setTimeout(initSortable, 500);
|
|
});
|
|
|
|
// ===== 유틸리티 =====
|
|
function escapeHtml(text) {
|
|
if (!text) return '';
|
|
const div = document.createElement('div');
|
|
div.textContent = text;
|
|
return div.innerHTML;
|
|
}
|
|
</script>
|
|
|
|
<!-- SortableJS CDN -->
|
|
<script src="https://cdn.jsdelivr.net/npm/sortablejs@1.15.0/Sortable.min.js"></script>
|
|
@endpush |