- layouts/app.blade.php에 SweetAlert2 CDN 및 전역 헬퍼 함수 추가 - showToast(): 토스트 알림 (success, error, warning, info) - showConfirm(): 확인 대화상자 - showDeleteConfirm(): 삭제 확인 (경고 아이콘) - showPermanentDeleteConfirm(): 영구 삭제 확인 (빨간색 경고) - showSuccess(), showError(): 성공/에러 알림 - 변환된 파일 목록 (48개 Blade 파일): - menus/* (6개), boards/* (2개), posts/* (3개) - daily-logs/* (3개), project-management/* (6개) - dev-tools/flow-tester/* (6개) - quote-formulas/* (4개), permission-analyze/* (1개) - archived-records/* (1개), profile/* (1개) - roles/* (3개), permissions/* (3개) - departments/* (3개), tenants/* (3개), users/* (3개) - 주요 개선사항: - Tailwind CSS 테마와 일관된 디자인 - 비동기 콜백 패턴으로 리팩토링 - 삭제/복원/영구삭제 각각 다른 스타일 적용
1226 lines
57 KiB
PHP
1226 lines
57 KiB
PHP
@extends('layouts.app')
|
|
|
|
@section('title', '일일 스크럼')
|
|
|
|
@section('content')
|
|
<!-- 페이지 헤더 -->
|
|
<div class="flex justify-between items-center mb-6">
|
|
<h1 class="text-2xl font-bold text-gray-800">📅 일일 스크럼</h1>
|
|
<div class="flex gap-2">
|
|
<a href="{{ route('daily-logs.today') }}" class="bg-green-600 hover:bg-green-700 text-white px-4 py-2 rounded-lg transition">
|
|
오늘 작성
|
|
</a>
|
|
<button type="button"
|
|
onclick="openCreateModal()"
|
|
class="bg-blue-600 hover:bg-blue-700 text-white px-4 py-2 rounded-lg transition">
|
|
+ 새 로그
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 주간 타임라인 -->
|
|
<div class="bg-white rounded-lg shadow-sm p-4 mb-6">
|
|
<h2 class="text-sm font-medium text-gray-500 mb-3">최근 7일</h2>
|
|
<div class="grid grid-cols-7 gap-2">
|
|
@foreach($weeklyTimeline as $index => $day)
|
|
<div class="relative group">
|
|
<button type="button"
|
|
onclick="{{ $day['log'] ? 'scrollToCard(' . $day['log']['id'] . ')' : 'openCreateModalWithDate(\'' . $day['date'] . '\')' }}"
|
|
data-log-id="{{ $day['log']['id'] ?? '' }}"
|
|
data-date="{{ $day['date'] }}"
|
|
class="day-card w-full text-left p-3 rounded-lg border-2 transition-all hover:shadow-md
|
|
{{ $day['is_today'] ? 'border-blue-500 bg-blue-50' : 'border-gray-200' }}
|
|
{{ $day['is_weekend'] && !$day['is_today'] ? 'bg-gray-50' : '' }}
|
|
{{ $day['log'] ? 'cursor-pointer' : 'cursor-pointer hover:border-blue-300' }}">
|
|
<!-- 요일 -->
|
|
<div class="text-xs font-medium {{ $day['is_weekend'] ? 'text-red-500' : 'text-gray-500' }} {{ $day['is_today'] ? 'text-blue-600' : '' }}">
|
|
{{ $day['day_name'] }}
|
|
</div>
|
|
<!-- 날짜 -->
|
|
<div class="text-lg font-bold {{ $day['is_today'] ? 'text-blue-600' : 'text-gray-800' }}">
|
|
{{ $day['day_number'] }}
|
|
</div>
|
|
|
|
@if($day['log'])
|
|
<!-- 로그 있음: 진행률 표시 -->
|
|
<div class="mt-2">
|
|
<div class="w-full bg-gray-200 rounded-full h-1.5">
|
|
<div class="bg-green-500 h-1.5 rounded-full transition-all" style="width: {{ $day['log']['completion_rate'] }}%"></div>
|
|
</div>
|
|
<div class="flex justify-between items-center mt-1">
|
|
<span class="text-xs text-gray-500">{{ $day['log']['entry_stats']['total'] }}건</span>
|
|
<span class="text-xs font-medium {{ $day['log']['completion_rate'] >= 100 ? 'text-green-600' : 'text-blue-600' }}">
|
|
{{ $day['log']['completion_rate'] }}%
|
|
</span>
|
|
</div>
|
|
</div>
|
|
@else
|
|
<!-- 로그 없음 -->
|
|
<div class="mt-2 text-center">
|
|
<span class="text-xs text-gray-400">미작성</span>
|
|
</div>
|
|
@endif
|
|
|
|
<!-- 오늘 표시 -->
|
|
@if($day['is_today'])
|
|
<div class="absolute -top-1 -right-1 w-3 h-3 bg-blue-500 rounded-full animate-pulse"></div>
|
|
@endif
|
|
</button>
|
|
|
|
<!-- 툴팁 (호버 시 표시) -->
|
|
@if($day['log'] && $day['log']['entry_stats']['total'] > 0)
|
|
<div class="hidden group-hover:block absolute z-10 left-1/2 -translate-x-1/2 top-full mt-2 w-36 bg-gray-800 text-white text-xs rounded-lg p-2 shadow-lg pointer-events-none">
|
|
<div class="flex justify-between mb-1">
|
|
<span>예정</span>
|
|
<span>{{ $day['log']['entry_stats']['todo'] }}</span>
|
|
</div>
|
|
<div class="flex justify-between mb-1">
|
|
<span>진행중</span>
|
|
<span>{{ $day['log']['entry_stats']['in_progress'] }}</span>
|
|
</div>
|
|
<div class="flex justify-between">
|
|
<span>완료</span>
|
|
<span>{{ $day['log']['entry_stats']['done'] }}</span>
|
|
</div>
|
|
<div class="absolute -top-1 left-1/2 -translate-x-1/2 w-2 h-2 bg-gray-800 rotate-45"></div>
|
|
</div>
|
|
@endif
|
|
</div>
|
|
@endforeach
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 통계 카드 -->
|
|
<div class="grid grid-cols-1 md:grid-cols-4 gap-4 mb-6">
|
|
<div class="bg-white rounded-lg shadow-sm p-4">
|
|
<div class="text-sm text-gray-500">전체 로그</div>
|
|
<div class="text-2xl font-bold text-gray-800">{{ $stats['total_logs'] ?? 0 }}</div>
|
|
</div>
|
|
<div class="bg-white rounded-lg shadow-sm p-4">
|
|
<div class="text-sm text-gray-500">최근 7일</div>
|
|
<div class="text-2xl font-bold text-blue-600">{{ $stats['recent_logs'] ?? 0 }}</div>
|
|
</div>
|
|
<div class="bg-white rounded-lg shadow-sm p-4">
|
|
<div class="text-sm text-gray-500">진행중 항목</div>
|
|
<div class="text-2xl font-bold text-yellow-600">{{ $stats['entries']['in_progress'] ?? 0 }}</div>
|
|
</div>
|
|
<div class="bg-white rounded-lg shadow-sm p-4">
|
|
<div class="text-sm text-gray-500">완료 항목</div>
|
|
<div class="text-2xl font-bold text-green-600">{{ $stats['entries']['done'] ?? 0 }}</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 미완료 항목 (예정/진행중) - 담당자별 그룹핑 -->
|
|
@if(count($pendingEntries) > 0)
|
|
<div class="bg-white rounded-lg shadow-sm p-4 mb-6">
|
|
<div class="flex items-center justify-between mb-3">
|
|
<h2 class="text-sm font-medium text-gray-500 flex items-center gap-2">
|
|
<svg class="w-4 h-4 text-indigo-500" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2" />
|
|
</svg>
|
|
미완료 항목
|
|
</h2>
|
|
@php $totalPending = collect($pendingEntries)->sum('total_count'); @endphp
|
|
<span class="text-xs text-gray-400">{{ count($pendingEntries) }}명 / {{ $totalPending }}건</span>
|
|
</div>
|
|
<div class="flex flex-wrap gap-4" id="pendingEntriesGrid">
|
|
@foreach($pendingEntries as $group)
|
|
@php
|
|
$entriesJson = collect($group['entries'])->map(fn($e) => [
|
|
'id' => $e['id'],
|
|
'content' => $e['content'],
|
|
'status' => $e['status'],
|
|
'daily_log_id' => $e['daily_log_id'] ?? null,
|
|
])->toJson();
|
|
@endphp
|
|
<div class="pending-assignee-card bg-white rounded-lg p-3 border border-gray-200 hover:border-blue-300 transition-all shrink-0"
|
|
style="width: 300px;"
|
|
data-assignee="{{ $group['assignee_name'] }}">
|
|
<!-- 담당자 이름 + 수정 버튼 -->
|
|
<div class="flex items-center justify-between mb-2 pb-2 border-b border-gray-100">
|
|
<div class="flex items-center gap-2">
|
|
<span class="text-sm font-semibold text-gray-900">{{ $group['assignee_name'] }}</span>
|
|
<span class="text-xs text-gray-400">({{ $group['total_count'] }})</span>
|
|
</div>
|
|
<button onclick='openPendingEditModal({!! htmlspecialchars($entriesJson, ENT_QUOTES) !!}, "{{ addslashes($group['assignee_name']) }}")'
|
|
class="p-1 text-gray-400 hover:text-indigo-600 hover:bg-indigo-50 rounded transition-colors" title="수정">
|
|
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z"/>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
<!-- 항목 목록 -->
|
|
<div class="space-y-1.5">
|
|
@foreach($group['entries'] as $entry)
|
|
<div class="group" data-entry-id="{{ $entry['id'] }}">
|
|
<div class="flex items-center gap-2">
|
|
<!-- 상태 + 날짜 -->
|
|
<span class="px-1.5 py-0.5 text-[10px] rounded shrink-0 {{ $entry['status'] === 'in_progress' ? 'bg-yellow-100 text-yellow-700' : 'bg-gray-100 text-gray-600' }}">
|
|
{{ $entry['status'] === 'in_progress' ? '진행' : '예정' }}
|
|
</span>
|
|
<span class="text-[10px] text-gray-400 shrink-0">{{ \Carbon\Carbon::parse($entry['log_date'])->format('m/d') }}</span>
|
|
<!-- 내용 (클릭하면 펼침) -->
|
|
<span class="pending-entry-text flex-1 text-sm text-gray-700 truncate cursor-pointer hover:text-blue-600"
|
|
onclick="togglePendingEntry(this)"
|
|
title="{{ $entry['content'] }}">{{ $entry['content'] }}</span>
|
|
<!-- 상태변경 버튼 -->
|
|
<div class="flex gap-0.5 opacity-30 group-hover:opacity-100 transition-opacity shrink-0">
|
|
@if($entry['status'] !== 'todo')
|
|
<button onclick="event.stopPropagation(); updatePendingStatus({{ $entry['id'] }}, 'todo')"
|
|
class="p-0.5 text-gray-400 hover:bg-gray-100 rounded" title="예정">
|
|
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9" stroke-width="2"/></svg>
|
|
</button>
|
|
@endif
|
|
@if($entry['status'] !== 'in_progress')
|
|
<button onclick="event.stopPropagation(); updatePendingStatus({{ $entry['id'] }}, 'in_progress')"
|
|
class="p-0.5 text-yellow-500 hover:bg-yellow-50 rounded" title="진행중">
|
|
<svg class="w-3.5 h-3.5" fill="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9"/></svg>
|
|
</button>
|
|
@endif
|
|
<button onclick="event.stopPropagation(); updatePendingStatus({{ $entry['id'] }}, 'done')"
|
|
class="p-0.5 text-green-500 hover:bg-green-50 rounded" title="완료">
|
|
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<!-- 펼쳐지는 전체 내용 (전체 너비) -->
|
|
<div class="pending-entry-full hidden mt-1.5 text-sm text-gray-600 bg-gray-50 rounded p-2 whitespace-pre-wrap">{{ $entry['content'] }}</div>
|
|
</div>
|
|
@endforeach
|
|
</div>
|
|
</div>
|
|
@endforeach
|
|
</div>
|
|
</div>
|
|
@endif
|
|
|
|
<!-- 필터 영역 -->
|
|
<div class="bg-white rounded-lg shadow-sm p-4 mb-6">
|
|
<form id="filterForm" class="flex gap-4 flex-wrap">
|
|
<!-- 검색 -->
|
|
<div class="flex-1 min-w-[200px]">
|
|
<input type="text"
|
|
name="search"
|
|
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="w-48">
|
|
<select name="project_id" class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
<option value="">전체 프로젝트</option>
|
|
@foreach($projects as $project)
|
|
<option value="{{ $project->id }}">{{ $project->name }}</option>
|
|
@endforeach
|
|
</select>
|
|
</div>
|
|
|
|
<!-- 날짜 범위 -->
|
|
<div class="w-40">
|
|
<input type="date"
|
|
name="start_date"
|
|
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
</div>
|
|
<span class="self-center text-gray-500">~</span>
|
|
<div class="w-40">
|
|
<input type="date"
|
|
name="end_date"
|
|
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="w-36">
|
|
<select name="trashed" class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500">
|
|
<option value="">활성만</option>
|
|
<option value="with">삭제 포함</option>
|
|
<option value="only">삭제만</option>
|
|
</select>
|
|
</div>
|
|
|
|
<!-- 검색 버튼 -->
|
|
<button type="submit" class="bg-gray-600 hover:bg-gray-700 text-white px-6 py-2 rounded-lg transition">
|
|
검색
|
|
</button>
|
|
</form>
|
|
</div>
|
|
|
|
<!-- 테이블 영역 (HTMX로 로드) -->
|
|
<div id="log-table"
|
|
hx-get="/api/admin/daily-logs"
|
|
hx-trigger="load, filterSubmit from:body"
|
|
hx-include="#filterForm"
|
|
hx-headers='{"X-CSRF-TOKEN": "{{ csrf_token() }}"}'
|
|
class="bg-white rounded-lg shadow-sm overflow-hidden">
|
|
<!-- 로딩 스피너 -->
|
|
<div class="flex justify-center items-center p-12">
|
|
<div class="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-600"></div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 생성/수정 모달 -->
|
|
@include('daily-logs.partials.modal-form', [
|
|
'projects' => $projects,
|
|
'assigneeTypes' => $assigneeTypes,
|
|
'entryStatuses' => $entryStatuses,
|
|
'assignees' => $assignees
|
|
])
|
|
|
|
<!-- 미완료 항목 수정 모달 -->
|
|
<div id="pendingEditModal" class="fixed inset-0 z-50 hidden overflow-y-auto" aria-labelledby="pending-edit-modal-title" role="dialog" aria-modal="true">
|
|
<div class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
|
|
<div class="fixed inset-0 bg-gray-500 bg-opacity-75 transition-opacity" onclick="closePendingEditModal()"></div>
|
|
<span class="hidden sm:inline-block sm:align-middle sm:h-screen">​</span>
|
|
<div class="inline-block align-bottom bg-white rounded-lg px-4 pt-5 pb-4 text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-2xl sm:w-full sm:p-6">
|
|
<div>
|
|
<div class="flex items-center justify-between mb-4">
|
|
<h3 class="text-lg font-medium text-gray-900 flex items-center gap-2" id="pending-edit-modal-title">
|
|
<svg class="w-5 h-5 text-indigo-500" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
|
|
</svg>
|
|
<span>미완료 항목 수정</span>
|
|
</h3>
|
|
<button type="button" onclick="closePendingEditModal()" class="text-gray-400 hover:text-gray-500">
|
|
<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 class="mb-4 p-3 bg-gray-50 rounded-lg">
|
|
<label class="text-xs text-gray-500 block mb-1">담당자</label>
|
|
<!-- 기존 항목 수정 시: 텍스트 표시 -->
|
|
<p id="pendingEditAssigneeNameText" class="font-medium text-gray-900 hidden"></p>
|
|
<!-- 새 항목 추가 시: 입력 필드 (datalist로 자동완성) -->
|
|
<input type="text" id="pendingEditAssigneeNameInput"
|
|
list="assigneeDatalist"
|
|
class="w-full px-3 py-2 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-indigo-500 hidden"
|
|
placeholder="담당자 이름 입력 또는 선택" autocomplete="off" required>
|
|
<datalist id="assigneeDatalist">
|
|
@foreach($assignees['teams'] ?? [] as $team)
|
|
<option value="{{ $team['name'] }}"></option>
|
|
@endforeach
|
|
@foreach($assignees['users'] ?? [] as $user)
|
|
<option value="{{ $user['name'] }}"></option>
|
|
@endforeach
|
|
</datalist>
|
|
</div>
|
|
|
|
<form id="pendingEditForm" onsubmit="submitPendingEditEntries(event)">
|
|
<!-- 업무 항목 목록 -->
|
|
<div class="mb-4">
|
|
<div class="flex items-center justify-between mb-2">
|
|
<label class="block text-sm font-medium text-gray-700">업무 내용</label>
|
|
<button type="button" onclick="addPendingEntryRow()" class="text-xs text-indigo-600 hover:text-indigo-800 flex items-center gap-1">
|
|
<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="M12 4v16m8-8H4" />
|
|
</svg>
|
|
항목 추가
|
|
</button>
|
|
</div>
|
|
<div id="pendingEditEntriesContainer" class="space-y-2">
|
|
<!-- 동적으로 항목 행들이 추가됨 -->
|
|
</div>
|
|
</div>
|
|
|
|
<div class="flex gap-3 pt-4 border-t border-gray-200">
|
|
<button type="button" onclick="closePendingEditModal()"
|
|
class="flex-1 px-4 py-2 border border-gray-300 rounded-lg text-gray-700 hover:bg-gray-50">
|
|
취소
|
|
</button>
|
|
<button type="submit" id="pendingEditSubmitBtn"
|
|
class="flex-1 px-4 py-2 bg-indigo-600 text-white rounded-lg hover:bg-indigo-700 disabled:opacity-50">
|
|
저장
|
|
</button>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
@endsection
|
|
|
|
@push('scripts')
|
|
<script src="https://unpkg.com/htmx.org@1.9.10"></script>
|
|
<script>
|
|
// 담당자 데이터
|
|
const assignees = @json($assignees);
|
|
|
|
// HTML 이스케이프 및 줄바꿈 처리 헬퍼
|
|
function escapeHtml(text) {
|
|
if (!text) return '';
|
|
const div = document.createElement('div');
|
|
div.textContent = text;
|
|
return div.innerHTML;
|
|
}
|
|
|
|
function nl2br(text) {
|
|
if (!text) return '';
|
|
return escapeHtml(text).replace(/\n/g, '<br>');
|
|
}
|
|
|
|
// ISO 날짜를 YYYY-MM-DD 형식으로 변환
|
|
function formatDateForInput(dateString) {
|
|
if (!dateString) return '';
|
|
// ISO 형식 또는 일반 날짜 문자열에서 YYYY-MM-DD 추출
|
|
return dateString.substring(0, 10);
|
|
}
|
|
|
|
// 폼 제출 시 HTMX 이벤트 트리거
|
|
document.getElementById('filterForm').addEventListener('submit', function(e) {
|
|
e.preventDefault();
|
|
htmx.trigger('#log-table', 'filterSubmit');
|
|
});
|
|
|
|
// 모달 열기
|
|
function openCreateModal() {
|
|
document.getElementById('modalTitle').textContent = '새 일일 로그';
|
|
document.getElementById('logForm').reset();
|
|
document.getElementById('logId').value = '';
|
|
document.getElementById('logDate').value = new Date().toISOString().split('T')[0];
|
|
clearEntries();
|
|
document.getElementById('logModal').classList.remove('hidden');
|
|
}
|
|
|
|
// 특정 날짜로 모달 열기 (주간 타임라인에서 사용)
|
|
function openCreateModalWithDate(date) {
|
|
document.getElementById('modalTitle').textContent = '새 일일 로그';
|
|
document.getElementById('logForm').reset();
|
|
document.getElementById('logId').value = '';
|
|
document.getElementById('logDate').value = date;
|
|
clearEntries();
|
|
document.getElementById('logModal').classList.remove('hidden');
|
|
}
|
|
|
|
// 모달 닫기
|
|
function closeModal() {
|
|
document.getElementById('logModal').classList.add('hidden');
|
|
}
|
|
|
|
// 로그 수정 모달 열기
|
|
function editLog(id) {
|
|
fetch(`/api/admin/daily-logs/${id}`, {
|
|
headers: {
|
|
'Accept': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
}
|
|
})
|
|
.then(res => res.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
document.getElementById('modalTitle').textContent = '일일 로그 수정';
|
|
document.getElementById('logId').value = data.data.id;
|
|
document.getElementById('logDate').value = formatDateForInput(data.data.log_date);
|
|
document.getElementById('projectId').value = data.data.project_id || '';
|
|
document.getElementById('summary').value = data.data.summary || '';
|
|
|
|
// 항목들 로드
|
|
clearEntries();
|
|
if (data.data.entries && data.data.entries.length > 0) {
|
|
data.data.entries.forEach(entry => addEntry(entry));
|
|
}
|
|
|
|
document.getElementById('logModal').classList.remove('hidden');
|
|
}
|
|
});
|
|
}
|
|
|
|
// 항목 비우기
|
|
function clearEntries() {
|
|
document.getElementById('entriesContainer').innerHTML = '';
|
|
}
|
|
|
|
// 항목 추가
|
|
function addEntry(entry = null) {
|
|
const container = document.getElementById('entriesContainer');
|
|
const index = container.children.length;
|
|
|
|
const html = `
|
|
<div class="entry-item border rounded-lg p-4 bg-gray-50" data-index="${index}">
|
|
<div class="flex gap-4 mb-2">
|
|
<input type="hidden" name="entries[${index}][id]" value="${entry?.id || ''}">
|
|
<div class="w-24">
|
|
<select name="entries[${index}][assignee_type]" class="w-full px-2 py-1 border rounded text-sm" onchange="updateAssigneeOptions(this, ${index})">
|
|
<option value="user" ${(!entry || entry.assignee_type === 'user') ? 'selected' : ''}>개인</option>
|
|
<option value="team" ${entry?.assignee_type === 'team' ? 'selected' : ''}>팀</option>
|
|
</select>
|
|
</div>
|
|
<div class="flex-1">
|
|
<input type="text"
|
|
name="entries[${index}][assignee_name]"
|
|
value="${entry?.assignee_name || ''}"
|
|
placeholder="담당자 (직접입력 또는 선택)"
|
|
list="assigneeList${index}"
|
|
class="w-full px-2 py-1 border rounded text-sm"
|
|
required>
|
|
<datalist id="assigneeList${index}">
|
|
${getAssigneeOptions(entry?.assignee_type || 'user')}
|
|
</datalist>
|
|
<input type="hidden" name="entries[${index}][assignee_id]" value="${entry?.assignee_id || ''}">
|
|
</div>
|
|
<div class="w-24">
|
|
<select name="entries[${index}][status]" class="w-full px-2 py-1 border rounded text-sm">
|
|
<option value="todo" ${(!entry || entry.status === 'todo') ? 'selected' : ''}>예정</option>
|
|
<option value="in_progress" ${entry?.status === 'in_progress' ? 'selected' : ''}>진행중</option>
|
|
<option value="done" ${entry?.status === 'done' ? 'selected' : ''}>완료</option>
|
|
</select>
|
|
</div>
|
|
<button type="button" onclick="removeEntry(this)" class="text-red-500 hover:text-red-700">
|
|
<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="M6 18L18 6M6 6l12 12"/>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
<textarea name="entries[${index}][content]"
|
|
placeholder="업무 내용"
|
|
rows="2"
|
|
class="w-full px-2 py-1 border rounded text-sm"
|
|
required>${entry?.content || ''}</textarea>
|
|
</div>
|
|
`;
|
|
|
|
container.insertAdjacentHTML('beforeend', html);
|
|
}
|
|
|
|
// 담당자 옵션 생성
|
|
function getAssigneeOptions(type) {
|
|
const list = type === 'team' ? assignees.teams : assignees.users;
|
|
return list.map(item => `<option value="${item.name}" data-id="${item.id}">`).join('');
|
|
}
|
|
|
|
// 담당자 타입 변경 시 옵션 업데이트
|
|
function updateAssigneeOptions(select, index) {
|
|
const datalist = document.getElementById(`assigneeList${index}`);
|
|
datalist.innerHTML = getAssigneeOptions(select.value);
|
|
}
|
|
|
|
// 항목 삭제
|
|
function removeEntry(btn) {
|
|
btn.closest('.entry-item').remove();
|
|
reindexEntries();
|
|
}
|
|
|
|
// 항목 인덱스 재정렬
|
|
function reindexEntries() {
|
|
const entries = document.querySelectorAll('.entry-item');
|
|
entries.forEach((entry, index) => {
|
|
entry.dataset.index = index;
|
|
entry.querySelectorAll('[name^="entries["]').forEach(input => {
|
|
input.name = input.name.replace(/entries\[\d+\]/, `entries[${index}]`);
|
|
});
|
|
const datalist = entry.querySelector('datalist');
|
|
if (datalist) {
|
|
datalist.id = `assigneeList${index}`;
|
|
}
|
|
const nameInput = entry.querySelector('input[list]');
|
|
if (nameInput) {
|
|
nameInput.setAttribute('list', `assigneeList${index}`);
|
|
}
|
|
});
|
|
}
|
|
|
|
// 폼 제출
|
|
document.getElementById('logForm').addEventListener('submit', function(e) {
|
|
e.preventDefault();
|
|
|
|
const logId = document.getElementById('logId').value;
|
|
const url = logId ? `/api/admin/daily-logs/${logId}` : '/api/admin/daily-logs';
|
|
const method = logId ? 'PUT' : 'POST';
|
|
|
|
const formData = new FormData(this);
|
|
const data = {};
|
|
|
|
// FormData를 객체로 변환
|
|
for (let [key, value] of formData.entries()) {
|
|
const match = key.match(/^entries\[(\d+)\]\[(.+)\]$/);
|
|
if (match) {
|
|
const idx = match[1];
|
|
const field = match[2];
|
|
if (!data.entries) data.entries = [];
|
|
if (!data.entries[idx]) data.entries[idx] = {};
|
|
data.entries[idx][field] = value;
|
|
} else {
|
|
data[key] = value;
|
|
}
|
|
}
|
|
|
|
// 빈 항목 제거
|
|
if (data.entries) {
|
|
data.entries = data.entries.filter(e => e && e.assignee_name && e.content);
|
|
}
|
|
|
|
fetch(url, {
|
|
method: method,
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'Accept': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
},
|
|
body: JSON.stringify(data)
|
|
})
|
|
.then(res => res.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
closeModal();
|
|
htmx.trigger('#log-table', 'filterSubmit');
|
|
} else {
|
|
showToast(result.message || '오류가 발생했습니다.', 'error');
|
|
}
|
|
})
|
|
.catch(err => {
|
|
console.error(err);
|
|
showToast('오류가 발생했습니다.', 'error');
|
|
});
|
|
});
|
|
|
|
// 삭제 확인
|
|
function confirmDelete(id, date) {
|
|
showDeleteConfirm(`"${date}" 일일 로그`, () => {
|
|
fetch(`/api/admin/daily-logs/${id}`, {
|
|
method: 'DELETE',
|
|
headers: {
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
}
|
|
})
|
|
.then(res => res.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
htmx.trigger('#log-table', 'filterSubmit');
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
// 복원 확인
|
|
function confirmRestore(id, date) {
|
|
showConfirm(`"${date}" 일일 로그를 복원하시겠습니까?`, () => {
|
|
fetch(`/api/admin/daily-logs/${id}/restore`, {
|
|
method: 'POST',
|
|
headers: {
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
}
|
|
})
|
|
.then(res => res.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
htmx.trigger('#log-table', 'filterSubmit');
|
|
}
|
|
});
|
|
}, { title: '복원 확인', icon: 'question' });
|
|
}
|
|
|
|
// 영구삭제 확인
|
|
function confirmForceDelete(id, date) {
|
|
showPermanentDeleteConfirm(`"${date}" 일일 로그`, () => {
|
|
fetch(`/api/admin/daily-logs/${id}/force`, {
|
|
method: 'DELETE',
|
|
headers: {
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
}
|
|
})
|
|
.then(res => res.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
htmx.trigger('#log-table', 'filterSubmit');
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
// ========================================
|
|
// 주간 타임라인 → 테이블 행 스크롤 및 아코디언 열기
|
|
// ========================================
|
|
function scrollToTableRow(logId) {
|
|
// 테이블 행 찾기
|
|
const row = document.querySelector(`tr.log-row[data-log-id="${logId}"]`);
|
|
|
|
if (row) {
|
|
// 선택된 카드 하이라이트
|
|
document.querySelectorAll('.day-card').forEach(card => {
|
|
card.classList.remove('ring-2', 'ring-indigo-500');
|
|
});
|
|
const selectedCard = document.querySelector(`.day-card[data-log-id="${logId}"]`);
|
|
if (selectedCard) {
|
|
selectedCard.classList.add('ring-2', 'ring-indigo-500');
|
|
}
|
|
|
|
// 테이블 행으로 스크롤
|
|
row.scrollIntoView({ behavior: 'smooth', block: 'center' });
|
|
|
|
// 스크롤 완료 후 아코디언 열기 (약간의 딜레이)
|
|
setTimeout(() => {
|
|
// 가상 이벤트 객체 생성
|
|
const fakeEvent = { target: row };
|
|
toggleTableAccordion(logId, fakeEvent);
|
|
}, 300);
|
|
} else {
|
|
// 테이블에 해당 로그가 없는 경우 (필터링 등으로 인해)
|
|
showToast('해당 로그가 현재 목록에 표시되지 않습니다. 필터를 확인해주세요.', 'warning');
|
|
}
|
|
}
|
|
|
|
// ========================================
|
|
// 테이블 리스트 아코디언 기능
|
|
// ========================================
|
|
let tableOpenAccordionId = null;
|
|
|
|
// 테이블 아코디언 토글
|
|
function toggleTableAccordion(logId, event) {
|
|
// 클릭한 요소가 버튼이면 무시 (이벤트 버블링 방지)
|
|
if (event.target.closest('button') || event.target.closest('a')) {
|
|
return;
|
|
}
|
|
|
|
const row = document.querySelector(`tr.log-row[data-log-id="${logId}"]`);
|
|
const accordionRow = document.querySelector(`tr.accordion-row[data-accordion-for="${logId}"]`);
|
|
const chevron = row.querySelector('.accordion-chevron');
|
|
|
|
// 같은 행을 다시 클릭하면 닫기
|
|
if (tableOpenAccordionId === logId) {
|
|
accordionRow.classList.add('hidden');
|
|
chevron.classList.remove('rotate-90');
|
|
row.classList.remove('bg-blue-50');
|
|
tableOpenAccordionId = null;
|
|
return;
|
|
}
|
|
|
|
// 다른 열린 아코디언 닫기
|
|
if (tableOpenAccordionId !== null) {
|
|
const prevRow = document.querySelector(`tr.log-row[data-log-id="${tableOpenAccordionId}"]`);
|
|
const prevAccordion = document.querySelector(`tr.accordion-row[data-accordion-for="${tableOpenAccordionId}"]`);
|
|
if (prevRow && prevAccordion) {
|
|
prevAccordion.classList.add('hidden');
|
|
prevRow.querySelector('.accordion-chevron')?.classList.remove('rotate-90');
|
|
prevRow.classList.remove('bg-blue-50');
|
|
}
|
|
}
|
|
|
|
// 현재 아코디언 열기
|
|
accordionRow.classList.remove('hidden');
|
|
chevron.classList.add('rotate-90');
|
|
row.classList.add('bg-blue-50');
|
|
tableOpenAccordionId = logId;
|
|
|
|
// 데이터 로드
|
|
loadTableAccordionContent(logId);
|
|
}
|
|
|
|
// 테이블 아코디언 콘텐츠 로드
|
|
function loadTableAccordionContent(logId) {
|
|
const contentDiv = document.getElementById(`accordion-content-${logId}`);
|
|
contentDiv.innerHTML = '<div class="text-center py-4 text-gray-500">로딩 중...</div>';
|
|
|
|
fetch(`/api/admin/daily-logs/${logId}`, {
|
|
headers: {
|
|
'Accept': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
}
|
|
})
|
|
.then(res => res.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
renderTableAccordionContent(logId, data.data);
|
|
} else {
|
|
contentDiv.innerHTML = '<div class="text-center py-4 text-red-500">데이터 로드 실패</div>';
|
|
}
|
|
})
|
|
.catch(err => {
|
|
contentDiv.innerHTML = '<div class="text-center py-4 text-red-500">데이터 로드 실패</div>';
|
|
});
|
|
}
|
|
|
|
// 테이블 아코디언 콘텐츠 렌더링
|
|
function renderTableAccordionContent(logId, log) {
|
|
const contentDiv = document.getElementById(`accordion-content-${logId}`);
|
|
|
|
const statusColors = {
|
|
'todo': 'bg-gray-100 text-gray-700',
|
|
'in_progress': 'bg-yellow-100 text-yellow-700',
|
|
'done': 'bg-green-100 text-green-700'
|
|
};
|
|
const statusLabels = {
|
|
'todo': '예정',
|
|
'in_progress': '진행중',
|
|
'done': '완료'
|
|
};
|
|
|
|
let entriesHtml = '';
|
|
if (log.entries && log.entries.length > 0) {
|
|
entriesHtml = log.entries.map(entry => `
|
|
<div class="flex items-start gap-3 p-3 bg-white rounded-lg border" data-entry-id="${entry.id}">
|
|
<div class="flex-1">
|
|
<div class="flex items-center gap-2 mb-1">
|
|
<span class="font-medium text-gray-900">${entry.assignee_name}</span>
|
|
<span class="px-2 py-0.5 text-xs rounded-full ${statusColors[entry.status]}">${statusLabels[entry.status]}</span>
|
|
</div>
|
|
<p class="text-sm text-gray-600">${nl2br(entry.content)}</p>
|
|
</div>
|
|
<div class="flex items-center gap-1">
|
|
${entry.status !== 'todo' ? `
|
|
<button onclick="updateTableEntryStatus(${logId}, ${entry.id}, 'todo')" class="p-1 text-gray-400 hover:text-gray-600" title="예정">
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9" stroke-width="2"/></svg>
|
|
</button>` : ''}
|
|
${entry.status !== 'in_progress' ? `
|
|
<button onclick="updateTableEntryStatus(${logId}, ${entry.id}, 'in_progress')" class="p-1 text-yellow-500 hover:text-yellow-700" title="진행중">
|
|
<svg class="w-4 h-4" fill="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9"/></svg>
|
|
</button>` : ''}
|
|
${entry.status !== 'done' ? `
|
|
<button onclick="updateTableEntryStatus(${logId}, ${entry.id}, 'done')" class="p-1 text-green-500 hover:text-green-700" title="완료">
|
|
<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="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/></svg>
|
|
</button>` : ''}
|
|
<button onclick="deleteTableEntry(${logId}, ${entry.id})" class="p-1 text-red-400 hover:text-red-600" title="삭제">
|
|
<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="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>
|
|
`).join('');
|
|
} else {
|
|
entriesHtml = '<div class="text-center py-4 text-gray-400">등록된 항목이 없습니다.</div>';
|
|
}
|
|
|
|
contentDiv.innerHTML = `
|
|
<div class="space-y-3">
|
|
${log.summary ? `<div class="text-sm text-gray-700 mb-3 pb-3 border-b">${nl2br(log.summary)}</div>` : ''}
|
|
<div class="space-y-2">
|
|
${entriesHtml}
|
|
</div>
|
|
<div class="pt-3 border-t flex justify-between items-center">
|
|
<button onclick="openQuickAddTableEntry(${logId})" class="text-sm text-blue-600 hover:text-blue-800">
|
|
+ 항목 추가
|
|
</button>
|
|
<button onclick="editLog(${logId})" class="text-sm text-indigo-600 hover:text-indigo-800">
|
|
전체 수정
|
|
</button>
|
|
</div>
|
|
</div>
|
|
`;
|
|
}
|
|
|
|
// 테이블 항목 상태 업데이트
|
|
function updateTableEntryStatus(logId, entryId, status) {
|
|
fetch(`/api/admin/daily-logs/entries/${entryId}/status`, {
|
|
method: 'PUT',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
},
|
|
body: JSON.stringify({ status })
|
|
})
|
|
.then(res => res.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
loadTableAccordionContent(logId);
|
|
}
|
|
});
|
|
}
|
|
|
|
// 테이블 항목 삭제
|
|
function deleteTableEntry(logId, entryId) {
|
|
showConfirm('이 항목을 삭제하시겠습니까?', () => {
|
|
fetch(`/api/admin/daily-logs/entries/${entryId}`, {
|
|
method: 'DELETE',
|
|
headers: {
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
}
|
|
})
|
|
.then(res => res.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
loadTableAccordionContent(logId);
|
|
}
|
|
});
|
|
}, { title: '항목 삭제', icon: 'warning' });
|
|
}
|
|
|
|
// 테이블 빠른 항목 추가 - 미완료 항목 수정 모달 사용
|
|
function openQuickAddTableEntry(logId) {
|
|
openQuickAddModal(logId);
|
|
}
|
|
|
|
// HTMX 로드 완료 후 테이블 아코디언 상태 리셋
|
|
document.body.addEventListener('htmx:afterSwap', function(event) {
|
|
if (event.detail.target.id === 'log-table') {
|
|
tableOpenAccordionId = null;
|
|
}
|
|
});
|
|
|
|
// ========================================
|
|
// 미완료 항목 펼치기/접기
|
|
// ========================================
|
|
function togglePendingEntry(textEl) {
|
|
const entryDiv = textEl.closest('[data-entry-id]');
|
|
const fullDiv = entryDiv.querySelector('.pending-entry-full');
|
|
|
|
if (fullDiv.classList.contains('hidden')) {
|
|
// 펼치기 - 타이틀은 truncate 유지, 전체 내용 표시
|
|
fullDiv.classList.remove('hidden');
|
|
textEl.classList.add('text-blue-600');
|
|
} else {
|
|
// 접기
|
|
fullDiv.classList.add('hidden');
|
|
textEl.classList.remove('text-blue-600');
|
|
}
|
|
}
|
|
|
|
// ========================================
|
|
// 미완료 항목 상태 업데이트
|
|
// ========================================
|
|
function updatePendingStatus(entryId, status) {
|
|
fetch(`/api/admin/daily-logs/entries/${entryId}/status`, {
|
|
method: 'PUT',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
},
|
|
body: JSON.stringify({ status })
|
|
})
|
|
.then(res => res.json())
|
|
.then(result => {
|
|
if (result.success) {
|
|
const item = document.querySelector(`[data-entry-id="${entryId}"]`);
|
|
if (item) {
|
|
if (status === 'done') {
|
|
// 완료 처리: 항목 제거 애니메이션
|
|
const parentCard = item.closest('.pending-assignee-card');
|
|
item.style.transition = 'opacity 0.3s, transform 0.3s';
|
|
item.style.opacity = '0';
|
|
item.style.transform = 'scale(0.95)';
|
|
setTimeout(() => {
|
|
item.remove();
|
|
// 담당자 카드에 항목이 없으면 카드도 제거
|
|
if (parentCard && parentCard.querySelectorAll('[data-entry-id]').length === 0) {
|
|
parentCard.style.transition = 'opacity 0.3s, transform 0.3s';
|
|
parentCard.style.opacity = '0';
|
|
parentCard.style.transform = 'scale(0.95)';
|
|
setTimeout(() => {
|
|
parentCard.remove();
|
|
// 전체 섹션 체크
|
|
const grid = document.getElementById('pendingEntriesGrid');
|
|
if (grid && grid.children.length === 0) {
|
|
grid.closest('.bg-white')?.remove();
|
|
}
|
|
}, 300);
|
|
}
|
|
}, 300);
|
|
} else {
|
|
// 예정/진행중 변경: 상태 뱃지만 업데이트
|
|
const badge = item.querySelector('span.px-1\\.5');
|
|
if (badge) {
|
|
if (status === 'in_progress') {
|
|
badge.className = 'px-1.5 py-0.5 text-[10px] rounded shrink-0 bg-yellow-100 text-yellow-700';
|
|
badge.textContent = '진행';
|
|
} else {
|
|
badge.className = 'px-1.5 py-0.5 text-[10px] rounded shrink-0 bg-gray-100 text-gray-600';
|
|
badge.textContent = '예정';
|
|
}
|
|
}
|
|
// 상태변경 버튼들도 업데이트
|
|
updatePendingEntryButtons(item, entryId, status);
|
|
}
|
|
}
|
|
// 로그 리스트도 새로고침
|
|
htmx.trigger('#log-table', 'filterSubmit');
|
|
}
|
|
});
|
|
}
|
|
|
|
// 미완료 항목 버튼 업데이트
|
|
function updatePendingEntryButtons(item, entryId, currentStatus) {
|
|
const buttonsDiv = item.querySelector('.flex.gap-0\\.5');
|
|
if (!buttonsDiv) return;
|
|
|
|
let buttonsHtml = '';
|
|
|
|
if (currentStatus !== 'todo') {
|
|
buttonsHtml += `
|
|
<button onclick="event.stopPropagation(); updatePendingStatus(${entryId}, 'todo')"
|
|
class="p-0.5 text-gray-400 hover:bg-gray-100 rounded" title="예정">
|
|
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9" stroke-width="2"/></svg>
|
|
</button>
|
|
`;
|
|
}
|
|
if (currentStatus !== 'in_progress') {
|
|
buttonsHtml += `
|
|
<button onclick="event.stopPropagation(); updatePendingStatus(${entryId}, 'in_progress')"
|
|
class="p-0.5 text-yellow-500 hover:bg-yellow-50 rounded" title="진행중">
|
|
<svg class="w-3.5 h-3.5" fill="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9"/></svg>
|
|
</button>
|
|
`;
|
|
}
|
|
buttonsHtml += `
|
|
<button onclick="event.stopPropagation(); updatePendingStatus(${entryId}, 'done')"
|
|
class="p-0.5 text-green-500 hover:bg-green-50 rounded" title="완료">
|
|
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/>
|
|
</svg>
|
|
</button>
|
|
`;
|
|
|
|
buttonsDiv.innerHTML = buttonsHtml;
|
|
}
|
|
|
|
// ========================================
|
|
// 미완료 항목 수정 모달 관련 함수
|
|
// ========================================
|
|
let pendingEditEntries = [];
|
|
let pendingEditAssigneeName = '';
|
|
let pendingEditFromAccordion = null; // 아코디언에서 열렸으면 logId 저장
|
|
|
|
// 모달 열기 (기존 항목 수정)
|
|
function openPendingEditModal(entriesJson, assigneeName) {
|
|
pendingEditEntries = entriesJson;
|
|
pendingEditAssigneeName = assigneeName;
|
|
pendingEditFromAccordion = null; // 기존 항목 수정 모드
|
|
|
|
// 담당자 텍스트 표시, 입력 필드 숨김
|
|
document.getElementById('pendingEditAssigneeNameText').textContent = assigneeName;
|
|
document.getElementById('pendingEditAssigneeNameText').classList.remove('hidden');
|
|
document.getElementById('pendingEditAssigneeNameInput').classList.add('hidden');
|
|
|
|
// 컨테이너 초기화 및 항목 추가
|
|
const container = document.getElementById('pendingEditEntriesContainer');
|
|
container.innerHTML = '';
|
|
|
|
entriesJson.forEach((entry, index) => {
|
|
addPendingEntryRow(entry, index);
|
|
});
|
|
|
|
document.getElementById('pendingEditModal').classList.remove('hidden');
|
|
}
|
|
|
|
// 모달 열기 (새 항목 추가 - 아코디언에서)
|
|
function openQuickAddModal(logId) {
|
|
pendingEditEntries = [{ id: '', content: '', status: 'todo', daily_log_id: logId }];
|
|
pendingEditAssigneeName = '';
|
|
pendingEditFromAccordion = logId;
|
|
|
|
// 담당자 입력 필드 표시, 텍스트 숨김
|
|
document.getElementById('pendingEditAssigneeNameText').classList.add('hidden');
|
|
const assigneeInput = document.getElementById('pendingEditAssigneeNameInput');
|
|
assigneeInput.value = '';
|
|
assigneeInput.classList.remove('hidden');
|
|
|
|
// 컨테이너 초기화 및 빈 항목 추가
|
|
const container = document.getElementById('pendingEditEntriesContainer');
|
|
container.innerHTML = '';
|
|
addPendingEntryRow({ id: '', content: '', status: 'todo', daily_log_id: logId }, 0);
|
|
|
|
document.getElementById('pendingEditModal').classList.remove('hidden');
|
|
|
|
// 담당자 입력 필드에 포커스
|
|
setTimeout(() => {
|
|
assigneeInput.focus();
|
|
}, 100);
|
|
}
|
|
|
|
// 모달 닫기
|
|
function closePendingEditModal() {
|
|
document.getElementById('pendingEditModal').classList.add('hidden');
|
|
pendingEditEntries = [];
|
|
pendingEditAssigneeName = '';
|
|
pendingEditFromAccordion = null;
|
|
}
|
|
|
|
// 항목 행 추가
|
|
function addPendingEntryRow(entry = null, index = null) {
|
|
const container = document.getElementById('pendingEditEntriesContainer');
|
|
if (index === null) {
|
|
index = container.children.length;
|
|
}
|
|
|
|
const currentStatus = entry?.status || 'todo';
|
|
|
|
// 새 항목인 경우 기존 항목의 daily_log_id 사용
|
|
let dailyLogId = entry?.daily_log_id || '';
|
|
if (!dailyLogId && pendingEditEntries.length > 0) {
|
|
dailyLogId = pendingEditEntries[0].daily_log_id || '';
|
|
}
|
|
|
|
const html = `
|
|
<div class="pending-edit-row flex gap-2 items-start p-2 bg-gray-50 rounded-lg" data-entry-id="${entry?.id || ''}" data-index="${index}">
|
|
<!-- 상태 버튼 그룹 -->
|
|
<div class="flex gap-1 shrink-0 pt-1">
|
|
<button type="button" onclick="togglePendingStatus(this, 'todo')"
|
|
class="status-btn p-1.5 rounded transition-all ${currentStatus === 'todo' ? 'bg-gray-200 ring-2 ring-gray-400' : 'hover:bg-gray-100'}"
|
|
data-status="todo" title="예정">
|
|
<svg class="w-4 h-4 text-gray-500" fill="none" stroke="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9" stroke-width="2"/></svg>
|
|
</button>
|
|
<button type="button" onclick="togglePendingStatus(this, 'in_progress')"
|
|
class="status-btn p-1.5 rounded transition-all ${currentStatus === 'in_progress' ? 'bg-yellow-100 ring-2 ring-yellow-400' : 'hover:bg-yellow-50'}"
|
|
data-status="in_progress" title="진행중">
|
|
<svg class="w-4 h-4 text-yellow-500" fill="currentColor" viewBox="0 0 24 24"><circle cx="12" cy="12" r="9"/></svg>
|
|
</button>
|
|
<button type="button" onclick="togglePendingStatus(this, 'done')"
|
|
class="status-btn p-1.5 rounded transition-all ${currentStatus === 'done' ? 'bg-green-100 ring-2 ring-green-400' : 'hover:bg-green-50'}"
|
|
data-status="done" title="완료">
|
|
<svg class="w-4 h-4 text-green-500" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/></svg>
|
|
</button>
|
|
</div>
|
|
<input type="hidden" name="pending_entries[${index}][status]" value="${currentStatus}">
|
|
<div class="flex-1">
|
|
<textarea name="pending_entries[${index}][content]" rows="2" required
|
|
class="w-full px-3 py-1.5 text-sm border border-gray-300 rounded-lg focus:ring-2 focus:ring-indigo-500 resize-none"
|
|
placeholder="업무 내용">${escapeHtml(entry?.content || '')}</textarea>
|
|
</div>
|
|
<input type="hidden" name="pending_entries[${index}][id]" value="${entry?.id || ''}">
|
|
<input type="hidden" name="pending_entries[${index}][daily_log_id]" value="${dailyLogId}">
|
|
<button type="button" onclick="removePendingEntryRow(this)" class="p-1.5 text-red-500 hover:bg-red-50 rounded shrink-0 pt-1" title="삭제">
|
|
<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="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>
|
|
`;
|
|
|
|
container.insertAdjacentHTML('beforeend', html);
|
|
}
|
|
|
|
// 상태 버튼 토글
|
|
function togglePendingStatus(btn, status) {
|
|
const row = btn.closest('.pending-edit-row');
|
|
const statusInput = row.querySelector('input[name*="[status]"]');
|
|
const buttons = row.querySelectorAll('.status-btn');
|
|
|
|
// 모든 버튼 스타일 초기화
|
|
buttons.forEach(b => {
|
|
b.classList.remove('bg-gray-200', 'ring-2', 'ring-gray-400', 'bg-yellow-100', 'ring-yellow-400', 'bg-green-100', 'ring-green-400');
|
|
});
|
|
|
|
// 선택된 버튼 스타일 적용
|
|
if (status === 'todo') {
|
|
btn.classList.add('bg-gray-200', 'ring-2', 'ring-gray-400');
|
|
} else if (status === 'in_progress') {
|
|
btn.classList.add('bg-yellow-100', 'ring-2', 'ring-yellow-400');
|
|
} else if (status === 'done') {
|
|
btn.classList.add('bg-green-100', 'ring-2', 'ring-green-400');
|
|
}
|
|
|
|
// hidden input 값 업데이트
|
|
statusInput.value = status;
|
|
}
|
|
|
|
// 항목 행 삭제
|
|
function removePendingEntryRow(btn) {
|
|
const row = btn.closest('.pending-edit-row');
|
|
row.remove();
|
|
reindexPendingEntryRows();
|
|
}
|
|
|
|
// 인덱스 재정렬
|
|
function reindexPendingEntryRows() {
|
|
const rows = document.querySelectorAll('.pending-edit-row');
|
|
rows.forEach((row, index) => {
|
|
row.dataset.index = index;
|
|
row.querySelectorAll('[name^="pending_entries["]').forEach(input => {
|
|
input.name = input.name.replace(/pending_entries\[\d+\]/, `pending_entries[${index}]`);
|
|
});
|
|
});
|
|
}
|
|
|
|
// 폼 제출
|
|
function submitPendingEditEntries(event) {
|
|
event.preventDefault();
|
|
|
|
// 새 항목 추가 모드일 때 입력 필드에서 담당자 이름 가져오기
|
|
let assigneeName = pendingEditAssigneeName;
|
|
if (pendingEditFromAccordion) {
|
|
const inputField = document.getElementById('pendingEditAssigneeNameInput');
|
|
assigneeName = inputField.value.trim();
|
|
if (!assigneeName) {
|
|
showToast('담당자 이름을 입력해주세요.', 'warning');
|
|
inputField.focus();
|
|
return;
|
|
}
|
|
}
|
|
|
|
const submitBtn = document.getElementById('pendingEditSubmitBtn');
|
|
submitBtn.disabled = true;
|
|
submitBtn.textContent = '저장 중...';
|
|
|
|
const rows = document.querySelectorAll('.pending-edit-row');
|
|
const updatePromises = [];
|
|
const deletePromises = [];
|
|
|
|
// 기존 항목 ID 수집
|
|
const currentIds = Array.from(rows).map(row => row.dataset.entryId).filter(id => id);
|
|
const originalIds = pendingEditEntries.map(e => String(e.id));
|
|
|
|
// 삭제할 항목
|
|
const deletedIds = originalIds.filter(id => !currentIds.includes(id));
|
|
deletedIds.forEach(id => {
|
|
deletePromises.push(
|
|
fetch(`/api/admin/daily-logs/entries/${id}`, {
|
|
method: 'DELETE',
|
|
headers: { 'X-CSRF-TOKEN': '{{ csrf_token() }}' }
|
|
})
|
|
);
|
|
});
|
|
|
|
// 수정/추가할 항목
|
|
rows.forEach(row => {
|
|
const entryId = row.dataset.entryId;
|
|
const status = row.querySelector('input[name*="[status]"]').value;
|
|
const content = row.querySelector('textarea').value;
|
|
const dailyLogId = row.querySelector('input[name*="[daily_log_id]"]').value;
|
|
|
|
if (!content.trim()) return;
|
|
|
|
if (entryId) {
|
|
// 기존 항목 수정
|
|
updatePromises.push(
|
|
fetch(`/api/admin/daily-logs/entries/${entryId}`, {
|
|
method: 'PUT',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
},
|
|
body: JSON.stringify({ content, status })
|
|
})
|
|
);
|
|
} else if (dailyLogId) {
|
|
// 새 항목 추가
|
|
updatePromises.push(
|
|
fetch(`/api/admin/daily-logs/${dailyLogId}/entries`, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-TOKEN': '{{ csrf_token() }}'
|
|
},
|
|
body: JSON.stringify({
|
|
assignee_type: 'user',
|
|
assignee_name: assigneeName,
|
|
content,
|
|
status
|
|
})
|
|
})
|
|
);
|
|
}
|
|
});
|
|
|
|
Promise.all([...deletePromises, ...updatePromises])
|
|
.then(() => {
|
|
closePendingEditModal();
|
|
// 페이지 새로고침으로 미완료 항목 갱신
|
|
location.reload();
|
|
})
|
|
.catch(err => {
|
|
console.error(err);
|
|
showToast('저장 중 오류가 발생했습니다.', 'error');
|
|
})
|
|
.finally(() => {
|
|
submitBtn.disabled = false;
|
|
submitBtn.textContent = '저장';
|
|
});
|
|
}
|
|
|
|
// ESC 키로 모달 닫기
|
|
document.addEventListener('keydown', function(e) {
|
|
if (e.key === 'Escape') {
|
|
if (!document.getElementById('pendingEditModal').classList.contains('hidden')) {
|
|
closePendingEditModal();
|
|
}
|
|
}
|
|
});
|
|
</script>
|
|
@endpush |