Files
tk-factory-services/system1-factory/web/pages/work/meeting-detail.html
Hyungi Ahn 0de9d5bb48 feat(sso): 인앱 브라우저 SSO 토큰 릴레이 — 카톡 WebView 쿠키 미공유 해결
카카오톡 인앱 WebView는 서브도메인 간 쿠키를 공유하지 않아
tkds에서 로그인 후 tkfb로 리다이렉트 시 인증이 풀리는 문제.

- sso-relay.js: URL hash의 _sso= 토큰을 로컬 쿠키+localStorage로 설정
- gateway dashboard: 로그인 후 redirect URL에 #_sso=<token> 추가
- 전 서비스 HTML: core JS 직전에 sso-relay.js 로드 (81개 파일)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-01 15:44:02 +09:00

201 lines
12 KiB
HTML

<!DOCTYPE html>
<html lang="ko">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>회의록 상세 - TK 공장관리</title>
<script src="https://cdn.tailwindcss.com"></script>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
<link rel="stylesheet" href="/static/css/tkfb.css?v=2026040103">
<style>
.attendee-tag { display: inline-flex; align-items: center; gap: 4px; background: #eff6ff; color: #2563eb; padding: 2px 8px; border-radius: 9999px; font-size: 0.75rem; }
.attendee-tag .remove-btn { cursor: pointer; color: #93c5fd; }
.attendee-tag .remove-btn:hover { color: #dc2626; }
.user-search-results { position: absolute; top: 100%; left: 0; right: 0; background: white; border: 1px solid #e2e8f0; border-radius: 0.5rem; max-height: 200px; overflow-y: auto; z-index: 50; box-shadow: 0 4px 12px rgba(0,0,0,0.1); }
.user-search-item { padding: 0.5rem 0.75rem; cursor: pointer; font-size: 0.875rem; }
.user-search-item:hover { background: #f1f5f9; }
</style>
</head>
<body class="bg-gray-50">
<header class="bg-orange-700 text-white sticky top-0 z-50">
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
<div class="flex justify-between items-center h-14">
<div class="flex items-center gap-3">
<button id="mobileMenuBtn" class="lg:hidden text-orange-200 hover:text-white">
<i class="fas fa-bars text-xl"></i>
</button>
<i class="fas fa-industry text-xl text-orange-200"></i>
<h1 class="text-lg font-semibold">TK 공장관리</h1>
</div>
<div class="flex items-center gap-4">
<span id="headerUserName" class="text-sm hidden sm:block">-</span>
<div id="headerUserAvatar" class="w-8 h-8 bg-orange-600 rounded-full flex items-center justify-center text-sm font-bold">-</div>
<button onclick="doLogout()" class="text-orange-200 hover:text-white" title="로그아웃">
<i class="fas fa-sign-out-alt"></i>
</button>
</div>
</div>
</div>
</header>
<div id="mobileOverlay" class="hidden fixed inset-0 bg-black/50 z-30 lg:hidden"></div>
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-4 fade-in">
<div class="flex gap-6">
<nav id="sideNav" class="hidden lg:flex flex-col gap-1 w-52 flex-shrink-0 pt-2 fixed lg:static z-40 bg-white lg:bg-transparent p-4 lg:p-0 rounded-lg lg:rounded-none shadow-lg lg:shadow-none top-14 left-0 bottom-0 overflow-y-auto"></nav>
<div class="flex-1 min-w-0">
<!-- 상단 -->
<div class="flex items-center gap-3 mb-5">
<a href="/pages/work/meetings.html" class="text-gray-400 hover:text-gray-600"><i class="fas fa-arrow-left"></i></a>
<h2 id="pageTitle" class="text-xl font-bold text-gray-800">새 회의록</h2>
<span id="statusBadge" class="badge badge-gray hidden">초안</span>
</div>
<!-- 기본 정보 -->
<div class="bg-white rounded-xl shadow-sm p-5 mb-4">
<h3 class="font-semibold text-gray-800 mb-3">기본 정보</h3>
<div class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-4">
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">날짜 *</label>
<input type="date" id="meetingDate" class="input-field w-full rounded-lg px-3 py-2 text-sm" required>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">시간</label>
<input type="time" id="meetingTime" class="input-field w-full rounded-lg px-3 py-2 text-sm">
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">장소</label>
<input type="text" id="meetingLocation" class="input-field w-full rounded-lg px-3 py-2 text-sm" placeholder="회의 장소">
</div>
<div class="sm:col-span-2 lg:col-span-3">
<label class="block text-sm font-medium text-gray-700 mb-1">제목 *</label>
<input type="text" id="meetingTitle" class="input-field w-full rounded-lg px-3 py-2 text-sm" placeholder="회의 제목" required>
</div>
<div class="sm:col-span-2 lg:col-span-3">
<label class="block text-sm font-medium text-gray-700 mb-1">요약</label>
<textarea id="meetingSummary" class="input-field w-full rounded-lg px-3 py-2 text-sm" rows="3" placeholder="회의 요약"></textarea>
</div>
</div>
</div>
<!-- 참석자 -->
<div class="bg-white rounded-xl shadow-sm p-5 mb-4">
<h3 class="font-semibold text-gray-800 mb-3">참석자</h3>
<div class="relative mb-3">
<input type="text" id="attendeeSearch" class="input-field w-full rounded-lg px-3 py-2 text-sm" placeholder="이름 또는 아이디로 검색" autocomplete="off">
<div id="attendeeResults" class="user-search-results hidden"></div>
</div>
<div id="attendeeTags" class="flex flex-wrap gap-2"></div>
</div>
<!-- 안건 목록 -->
<div class="bg-white rounded-xl shadow-sm p-5 mb-4">
<div class="flex items-center justify-between mb-3">
<h3 class="font-semibold text-gray-800">안건</h3>
<button id="btnAddItem" class="hidden text-sm text-orange-600 hover:text-orange-700 font-medium" onclick="openItemModal()">
<i class="fas fa-plus mr-1"></i>안건 추가
</button>
</div>
<div id="agendaList" class="space-y-3"></div>
<div id="agendaEmpty" class="text-center py-6 text-gray-400 text-sm">
<i class="fas fa-clipboard-list text-2xl mb-2"></i>
<p>안건이 없습니다.</p>
</div>
</div>
<!-- 하단 버튼 -->
<div class="flex flex-wrap gap-2 justify-end mb-8" id="bottomActions">
<button id="btnSave" class="hidden bg-orange-600 text-white px-5 py-2 rounded-lg text-sm hover:bg-orange-700" onclick="saveMeeting()">
<i class="fas fa-save mr-1"></i>저장
</button>
<button id="btnPublish" class="hidden bg-green-600 text-white px-5 py-2 rounded-lg text-sm hover:bg-green-700" onclick="publishMeeting()">
<i class="fas fa-paper-plane mr-1"></i>발행
</button>
<button id="btnUnpublish" class="hidden bg-gray-500 text-white px-5 py-2 rounded-lg text-sm hover:bg-gray-600" onclick="unpublishMeeting()">
<i class="fas fa-undo mr-1"></i>발행 취소
</button>
<button id="btnDelete" class="hidden bg-red-500 text-white px-5 py-2 rounded-lg text-sm hover:bg-red-600" onclick="deleteMeeting()">
<i class="fas fa-trash mr-1"></i>삭제
</button>
</div>
</div>
</div>
</div>
<!-- 안건 모달 -->
<div id="itemModal" class="modal-overlay hidden">
<div class="modal-content p-6">
<div class="flex justify-between items-center mb-4">
<h3 id="itemModalTitle" class="text-lg font-bold">안건 추가</h3>
<button onclick="closeItemModal()" class="text-gray-400 hover:text-gray-600"><i class="fas fa-times text-xl"></i></button>
</div>
<input type="hidden" id="itemId">
<div class="grid grid-cols-1 sm:grid-cols-2 gap-4">
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">유형</label>
<select id="itemType" class="input-field w-full rounded-lg px-3 py-2 text-sm">
<option value="schedule_update">공정현황</option>
<option value="issue">이슈</option>
<option value="decision">결정사항</option>
<option value="action_item">조치사항</option>
<option value="other">기타</option>
</select>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">관련 프로젝트</label>
<select id="itemProject" class="input-field w-full rounded-lg px-3 py-2 text-sm" onchange="loadItemMilestones()">
<option value="">선택안함</option>
</select>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">관련 마일스톤</label>
<select id="itemMilestone" class="input-field w-full rounded-lg px-3 py-2 text-sm">
<option value="">선택안함</option>
</select>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">상태</label>
<select id="itemStatus" class="input-field w-full rounded-lg px-3 py-2 text-sm">
<option value="open">미처리</option>
<option value="in_progress">진행중</option>
<option value="completed">완료</option>
<option value="cancelled">취소</option>
</select>
</div>
<div class="sm:col-span-2">
<label class="block text-sm font-medium text-gray-700 mb-1">내용 *</label>
<textarea id="itemContent" class="input-field w-full rounded-lg px-3 py-2 text-sm" rows="3" required></textarea>
</div>
<div class="sm:col-span-2">
<label class="block text-sm font-medium text-gray-700 mb-1">결정사항</label>
<textarea id="itemDecision" class="input-field w-full rounded-lg px-3 py-2 text-sm" rows="2"></textarea>
</div>
<div class="sm:col-span-2">
<label class="block text-sm font-medium text-gray-700 mb-1">필요조치</label>
<textarea id="itemAction" class="input-field w-full rounded-lg px-3 py-2 text-sm" rows="2"></textarea>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">담당자</label>
<select id="itemResponsible" class="input-field w-full rounded-lg px-3 py-2 text-sm">
<option value="">선택안함</option>
</select>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">기한</label>
<input type="date" id="itemDueDate" class="input-field w-full rounded-lg px-3 py-2 text-sm">
</div>
</div>
<div class="flex justify-end gap-2 mt-4">
<button type="button" onclick="closeItemModal()" class="px-4 py-2 text-sm border rounded-lg hover:bg-gray-50">취소</button>
<button type="button" onclick="saveItem()" class="px-4 py-2 text-sm bg-orange-600 text-white rounded-lg hover:bg-orange-700">저장</button>
</div>
</div>
</div>
<script src="/js/sso-relay.js?v=20260401"></script>
<script src="/static/js/tkfb-core.js?v=2026040105"></script>
<script src="/js/meeting-detail.js?v=2026031701"></script>
</body>
</html>