Initial commit
This commit is contained in:
BIN
backend/static/android-chrome-192x192.png
Normal file
BIN
backend/static/android-chrome-192x192.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 8.1 KiB |
BIN
backend/static/android-chrome-512x512.png
Normal file
BIN
backend/static/android-chrome-512x512.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 25 KiB |
BIN
backend/static/apple-touch-icon.png
Normal file
BIN
backend/static/apple-touch-icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 7.3 KiB |
BIN
backend/static/favicon-16x16.png
Normal file
BIN
backend/static/favicon-16x16.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 579 B |
BIN
backend/static/favicon-32x32.png
Normal file
BIN
backend/static/favicon-32x32.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.1 KiB |
BIN
backend/static/favicon.ico
Normal file
BIN
backend/static/favicon.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 15 KiB |
264
backend/static/script.js
Normal file
264
backend/static/script.js
Normal file
@@ -0,0 +1,264 @@
|
||||
// script.js - 정리된 버전
|
||||
|
||||
document.addEventListener('DOMContentLoaded', () => {
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// CSRF 토큰
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
const csrfToken = document.querySelector('input[name="csrf_token"]')?.value || '';
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// 진행바 업데이트
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
window.updateProgress = function(percent) {
|
||||
const bar = document.getElementById('progressBar');
|
||||
if (!bar) return;
|
||||
const v = Math.max(0, Math.min(100, Number(percent) || 0));
|
||||
bar.style.width = v + '%';
|
||||
bar.setAttribute('aria-valuenow', v);
|
||||
bar.innerHTML = `<span class="fw-semibold small">${v}%</span>`;
|
||||
};
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// 줄 수 카운터
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
function updateLineCount(textareaId, badgeId) {
|
||||
const textarea = document.getElementById(textareaId);
|
||||
const badge = document.getElementById(badgeId);
|
||||
|
||||
if (!textarea || !badge) return;
|
||||
|
||||
const updateCount = () => {
|
||||
const text = textarea.value.trim();
|
||||
if (text === '') {
|
||||
badge.textContent = '0줄';
|
||||
return;
|
||||
}
|
||||
const lines = text.split('\n').filter(line => line.trim().length > 0);
|
||||
badge.textContent = `${lines.length}줄`;
|
||||
};
|
||||
|
||||
updateCount();
|
||||
textarea.addEventListener('input', updateCount);
|
||||
textarea.addEventListener('change', updateCount);
|
||||
textarea.addEventListener('keyup', updateCount);
|
||||
textarea.addEventListener('paste', () => setTimeout(updateCount, 10));
|
||||
}
|
||||
|
||||
updateLineCount('ips', 'ipLineCount');
|
||||
updateLineCount('server_list_content', 'serverLineCount');
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// 스크립트 선택 시 XML 드롭다운 토글
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
const TARGET_SCRIPT = "02-set_config.py";
|
||||
const scriptSelect = document.getElementById('script');
|
||||
const xmlGroup = document.getElementById('xmlFileGroup');
|
||||
|
||||
function toggleXml() {
|
||||
if (!scriptSelect || !xmlGroup) return;
|
||||
xmlGroup.style.display = (scriptSelect.value === TARGET_SCRIPT) ? 'block' : 'none';
|
||||
}
|
||||
|
||||
if (scriptSelect) {
|
||||
toggleXml();
|
||||
scriptSelect.addEventListener('change', toggleXml);
|
||||
}
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// 파일 보기 모달
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
const modalEl = document.getElementById('fileViewModal');
|
||||
const titleEl = document.getElementById('fileViewModalLabel');
|
||||
const contentEl = document.getElementById('fileViewContent');
|
||||
|
||||
if (modalEl) {
|
||||
modalEl.addEventListener('show.bs.modal', async (ev) => {
|
||||
const btn = ev.relatedTarget;
|
||||
const folder = btn?.getAttribute('data-folder') || '';
|
||||
const date = btn?.getAttribute('data-date') || '';
|
||||
const filename = btn?.getAttribute('data-filename') || '';
|
||||
|
||||
titleEl.innerHTML = `<i class="bi bi-file-text me-2"></i>${filename || '파일'}`;
|
||||
contentEl.textContent = '불러오는 중...';
|
||||
|
||||
const params = new URLSearchParams();
|
||||
if (folder) params.set('folder', folder);
|
||||
if (date) params.set('date', date);
|
||||
if (filename) params.set('filename', filename);
|
||||
|
||||
try {
|
||||
const res = await fetch(`/view_file?${params.toString()}`, { cache: 'no-store' });
|
||||
if (!res.ok) throw new Error('HTTP ' + res.status);
|
||||
|
||||
const data = await res.json();
|
||||
contentEl.textContent = data?.content ?? '(빈 파일)';
|
||||
} catch (e) {
|
||||
contentEl.textContent = '파일을 불러오지 못했습니다: ' + (e?.message || e);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// 공통 POST 함수
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
async function postFormAndHandle(url) {
|
||||
const res = await fetch(url, {
|
||||
method: 'POST',
|
||||
credentials: 'same-origin',
|
||||
headers: {
|
||||
'X-CSRFToken': csrfToken,
|
||||
'Accept': 'application/json, text/html;q=0.9,*/*;q=0.8',
|
||||
},
|
||||
});
|
||||
|
||||
const ct = (res.headers.get('content-type') || '').toLowerCase();
|
||||
|
||||
if (ct.includes('application/json')) {
|
||||
const data = await res.json();
|
||||
if (data.success === false) {
|
||||
throw new Error(data.error || ('HTTP ' + res.status));
|
||||
}
|
||||
return data;
|
||||
}
|
||||
|
||||
return { success: true, html: true };
|
||||
}
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// MAC 파일 이동
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
const macForm = document.getElementById('macMoveForm');
|
||||
if (macForm) {
|
||||
macForm.addEventListener('submit', async (e) => {
|
||||
e.preventDefault();
|
||||
const btn = macForm.querySelector('button');
|
||||
const originalHtml = btn.innerHTML;
|
||||
btn.disabled = true;
|
||||
btn.innerHTML = '<span class="spinner-border spinner-border-sm me-2"></span>처리 중...';
|
||||
|
||||
try {
|
||||
await postFormAndHandle(macForm.action);
|
||||
location.reload();
|
||||
} catch (err) {
|
||||
alert('MAC 이동 중 오류: ' + (err?.message || err));
|
||||
btn.disabled = false;
|
||||
btn.innerHTML = originalHtml;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// GUID 파일 이동
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
const guidForm = document.getElementById('guidMoveForm');
|
||||
if (guidForm) {
|
||||
guidForm.addEventListener('submit', async (e) => {
|
||||
e.preventDefault();
|
||||
const btn = guidForm.querySelector('button');
|
||||
const originalHtml = btn.innerHTML;
|
||||
btn.disabled = true;
|
||||
btn.innerHTML = '<span class="spinner-border spinner-border-sm me-2"></span>처리 중...';
|
||||
|
||||
try {
|
||||
await postFormAndHandle(guidForm.action);
|
||||
location.reload();
|
||||
} catch (err) {
|
||||
alert('GUID 이동 중 오류: ' + (err?.message || err));
|
||||
btn.disabled = false;
|
||||
btn.innerHTML = originalHtml;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// IP 폼 제출 및 진행률 폴링
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
const ipForm = document.getElementById("ipForm");
|
||||
if (ipForm) {
|
||||
ipForm.addEventListener("submit", async (ev) => {
|
||||
ev.preventDefault();
|
||||
|
||||
const formData = new FormData(ipForm);
|
||||
const btn = ipForm.querySelector('button[type="submit"]');
|
||||
const originalHtml = btn.innerHTML;
|
||||
btn.disabled = true;
|
||||
btn.innerHTML = '<span class="spinner-border spinner-border-sm me-2"></span>처리 중...';
|
||||
|
||||
try {
|
||||
const res = await fetch(ipForm.action, {
|
||||
method: "POST",
|
||||
body: formData
|
||||
});
|
||||
|
||||
if (!res.ok) throw new Error("HTTP " + res.status);
|
||||
|
||||
const data = await res.json();
|
||||
console.log("[DEBUG] process_ips 응답:", data);
|
||||
|
||||
if (data.job_id) {
|
||||
pollProgress(data.job_id);
|
||||
} else {
|
||||
window.updateProgress(100);
|
||||
setTimeout(() => location.reload(), 1000);
|
||||
}
|
||||
} catch (err) {
|
||||
console.error("처리 중 오류:", err);
|
||||
alert("처리 중 오류 발생: " + err.message);
|
||||
btn.disabled = false;
|
||||
btn.innerHTML = originalHtml;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// 진행률 폴링 함수
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
function pollProgress(jobId) {
|
||||
const interval = setInterval(async () => {
|
||||
try {
|
||||
const res = await fetch(`/progress_status/${jobId}`);
|
||||
if (!res.ok) {
|
||||
clearInterval(interval);
|
||||
return;
|
||||
}
|
||||
|
||||
const data = await res.json();
|
||||
|
||||
if (data.progress !== undefined) {
|
||||
window.updateProgress(data.progress);
|
||||
}
|
||||
|
||||
if (data.progress >= 100) {
|
||||
clearInterval(interval);
|
||||
window.updateProgress(100);
|
||||
setTimeout(() => location.reload(), 1500);
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('진행률 확인 중 오류:', err);
|
||||
clearInterval(interval);
|
||||
}
|
||||
}, 500);
|
||||
}
|
||||
|
||||
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
// 알림 자동 닫기 (5초 후)
|
||||
// ─────────────────────────────────────────────────────────────
|
||||
setTimeout(() => {
|
||||
document.querySelectorAll('.alert').forEach(alert => {
|
||||
const bsAlert = new bootstrap.Alert(alert);
|
||||
bsAlert.close();
|
||||
});
|
||||
}, 5000);
|
||||
|
||||
});
|
||||
534
backend/static/style.css
Normal file
534
backend/static/style.css
Normal file
@@ -0,0 +1,534 @@
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
기본 레이아웃
|
||||
───────────────────────────────────────────────────────────── */
|
||||
body {
|
||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", "Malgun Gothic",
|
||||
"Apple SD Gothic Neo", "Noto Sans KR", sans-serif;
|
||||
font-weight: 400;
|
||||
background-color: #f8f9fa;
|
||||
padding-top: 56px;
|
||||
}
|
||||
|
||||
.container-card {
|
||||
background-color: #ffffff;
|
||||
padding: 20px;
|
||||
box-shadow: 0 0 15px rgba(0, 0, 0, 0.08);
|
||||
border-radius: 8px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
텍스트 및 제목 - 모두 일반 굵기
|
||||
───────────────────────────────────────────────────────────── */
|
||||
h1, h2, h3, h4, h5, h6 {
|
||||
color: #343a40;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
.card-header h6 {
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
폼 요소 - 모두 일반 굵기
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.form-label {
|
||||
color: #495057;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
.form-control, .form-select {
|
||||
border-radius: 5px;
|
||||
border: 1px solid #ced4da;
|
||||
font-weight: 400;
|
||||
transition: border-color 0.2s ease-in-out, box-shadow 0.2s ease-in-out;
|
||||
}
|
||||
|
||||
.form-control:focus, .form-select:focus {
|
||||
border-color: #80bdff;
|
||||
box-shadow: 0 0 0 0.25rem rgba(0, 123, 255, 0.25);
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
버튼 - 일반 굵기
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.btn {
|
||||
border-radius: 5px;
|
||||
font-weight: 400;
|
||||
transition: all 0.2s ease-in-out;
|
||||
}
|
||||
|
||||
.badge {
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
네비게이션 바
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.navbar {
|
||||
background-color: #343a40 !important;
|
||||
}
|
||||
|
||||
.navbar-brand {
|
||||
font-weight: 700;
|
||||
color: #ffffff !important;
|
||||
}
|
||||
|
||||
.nav-link {
|
||||
color: rgba(255, 255, 255, 0.75) !important;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
.nav-link:hover {
|
||||
color: #ffffff !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
카드 헤더 색상 (1번 이미지와 동일하게)
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.card-header.bg-primary {
|
||||
background-color: #007bff !important;
|
||||
color: #ffffff !important;
|
||||
}
|
||||
|
||||
.card-header.bg-success {
|
||||
background-color: #28a745 !important;
|
||||
color: #ffffff !important;
|
||||
}
|
||||
|
||||
.card-header.bg-primary h6,
|
||||
.card-header.bg-success h6 {
|
||||
color: #ffffff !important;
|
||||
}
|
||||
|
||||
.card-header.bg-primary i,
|
||||
.card-header.bg-success i {
|
||||
color: #ffffff !important;
|
||||
}
|
||||
|
||||
/* 밝은 배경 헤더는 어두운 텍스트 */
|
||||
.card-header.bg-light {
|
||||
background-color: #f8f9fa !important;
|
||||
color: #343a40 !important;
|
||||
}
|
||||
|
||||
.card-header.bg-light h6 {
|
||||
color: #343a40 !important;
|
||||
}
|
||||
|
||||
.card-header.bg-light i {
|
||||
color: #343a40 !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
버튼 색상 (2번 이미지와 동일하게)
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.btn-warning {
|
||||
background-color: #ffc107 !important;
|
||||
border-color: #ffc107 !important;
|
||||
color: #000 !important;
|
||||
}
|
||||
.btn-warning:hover {
|
||||
background-color: #e0a800 !important;
|
||||
border-color: #d39e00 !important;
|
||||
color: #000 !important;
|
||||
}
|
||||
|
||||
.btn-info {
|
||||
background-color: #17a2b8 !important;
|
||||
border-color: #17a2b8 !important;
|
||||
color: #fff !important;
|
||||
}
|
||||
.btn-info:hover {
|
||||
background-color: #138496 !important;
|
||||
border-color: #117a8b !important;
|
||||
color: #fff !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
진행바
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.progress {
|
||||
border-radius: 10px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.progress-bar {
|
||||
transition: width 0.6s ease;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
파일 그리드 레이아웃 - 빈 공간 없이 채우기
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.file-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(140px, 1fr));
|
||||
gap: 1rem;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
파일 카드 (컴팩트)
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.file-card-compact {
|
||||
transition: all 0.2s ease;
|
||||
background: #fff;
|
||||
width: 100%;
|
||||
max-width: 180px; /* 기본값 유지(카드가 너무 넓어지지 않도록) */
|
||||
}
|
||||
.file-card-compact:hover {
|
||||
box-shadow: 0 0.25rem 0.5rem rgba(0, 0, 0, 0.1);
|
||||
transform: translateY(-2px);
|
||||
}
|
||||
.file-card-compact a {
|
||||
font-size: 0.85rem;
|
||||
font-weight: 400;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
max-width: 100%;
|
||||
}
|
||||
|
||||
/* 파일 카드 내 모든 텍스트 일반 굵기 */
|
||||
.file-card-compact,
|
||||
.file-card-compact * {
|
||||
font-weight: 400 !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
(공통) 파일 카드 버튼 컨테이너 기본값 (기존 유지)
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.file-card-buttons { /* 처리된 목록(2버튼) 기본 레이아웃 */
|
||||
display: flex;
|
||||
gap: 0.15rem;
|
||||
}
|
||||
.file-card-buttons > button,
|
||||
.file-card-buttons > form {
|
||||
width: calc(50% - 0.075rem);
|
||||
}
|
||||
.file-card-buttons form { margin: 0; padding: 0; }
|
||||
.file-card-buttons .btn-sm {
|
||||
padding: 0.1rem 0.2rem !important;
|
||||
font-size: 0.65rem !important;
|
||||
width: 100%;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
/* 1버튼(백업) 기본 레이아웃 */
|
||||
.file-card-single-button {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
}
|
||||
.file-card-single-button .btn-sm {
|
||||
padding: 0.15rem 0.3rem !important;
|
||||
font-size: 0.7rem !important;
|
||||
min-width: 50px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
(공통) Outline 기본값 (기존 유지)
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.file-card-compact .btn-outline-primary {
|
||||
background-color: transparent !important;
|
||||
color: #0d6efd !important;
|
||||
border: 1px solid #0d6efd !important;
|
||||
}
|
||||
.file-card-compact .btn-outline-primary:hover {
|
||||
background-color: #0d6efd !important;
|
||||
color: #fff !important;
|
||||
}
|
||||
.file-card-compact .btn-outline-danger {
|
||||
background-color: transparent !important;
|
||||
color: #dc3545 !important;
|
||||
border: 1px solid #dc3545 !important;
|
||||
}
|
||||
.file-card-compact .btn-outline-danger:hover {
|
||||
background-color: #dc3545 !important;
|
||||
color: #fff !important;
|
||||
}
|
||||
|
||||
/* 기존 d-flex gap-2 레거시 대응 */
|
||||
.file-card-compact .d-flex.gap-2 { display: flex; gap: 0.2rem; }
|
||||
.file-card-compact .d-flex.gap-2 > * { flex: 1; }
|
||||
.file-card-compact .d-flex.gap-2 form { display: contents; }
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
!!! 목록별 버튼 스타일 "분리" 규칙 (HTML에 클래스만 달아주면 적용)
|
||||
- processed-list 블록의 보기/삭제
|
||||
- backup-list 블록의 보기
|
||||
───────────────────────────────────────────────────────────── */
|
||||
|
||||
/* 처리된 파일 목록(Processed) : 컨테이너 세부 튜닝 */
|
||||
.processed-list .file-card-buttons {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr; /* 보기/삭제 2열 격자 */
|
||||
gap: 0.2rem;
|
||||
}
|
||||
|
||||
/* 보기(처리된) — 전용 클래스 우선 */
|
||||
.processed-list .btn-view-processed,
|
||||
.processed-list .file-card-buttons .btn-outline-primary { /* (백워드 호환) */
|
||||
border-color: #3b82f6 !important;
|
||||
color: #1d4ed8 !important;
|
||||
background: transparent !important;
|
||||
padding: .35rem .55rem !important;
|
||||
font-size: .8rem !important;
|
||||
font-weight: 600 !important;
|
||||
}
|
||||
.processed-list .btn-view-processed:hover,
|
||||
.processed-list .file-card-buttons .btn-outline-primary:hover {
|
||||
background: rgba(59,130,246,.10) !important;
|
||||
color: #1d4ed8 !important;
|
||||
}
|
||||
|
||||
/* 삭제(처리된) — 전용 클래스 우선(더 작게) */
|
||||
.processed-list .btn-delete-processed,
|
||||
.processed-list .file-card-buttons .btn-outline-danger { /* (백워드 호환) */
|
||||
border-color: #ef4444 !important;
|
||||
color: #b91c1c !important;
|
||||
background: transparent !important;
|
||||
padding: .25rem .45rem !important; /* 더 작게 */
|
||||
font-size: .72rem !important; /* 더 작게 */
|
||||
font-weight: 600 !important;
|
||||
}
|
||||
.processed-list .btn-delete-processed:hover,
|
||||
.processed-list .file-card-buttons .btn-outline-danger:hover {
|
||||
background: rgba(239,68,68,.10) !important;
|
||||
color: #b91c1c !important;
|
||||
}
|
||||
|
||||
/* 백업 파일 목록(Backup) : 1버튼 컨테이너 */
|
||||
.backup-list .file-card-single-button {
|
||||
display: flex;
|
||||
margin-top: .25rem;
|
||||
}
|
||||
|
||||
/* 보기(백업) — 전용 클래스 우선(초록계열), 기존 .btn-outline-primary 사용 시에도 분리 적용 */
|
||||
.backup-list .btn-view-backup,
|
||||
.backup-list .file-card-single-button .btn-outline-primary { /* (백워드 호환) */
|
||||
width: 100%;
|
||||
border-color: #10b981 !important; /* emerald-500 */
|
||||
color: #047857 !important; /* emerald-700 */
|
||||
background: transparent !important;
|
||||
padding: .42rem .7rem !important;
|
||||
font-size: .8rem !important;
|
||||
font-weight: 700 !important; /* 백업은 강조 */
|
||||
}
|
||||
.backup-list .btn-view-backup:hover,
|
||||
.backup-list .file-card-single-button .btn-outline-primary:hover {
|
||||
background: rgba(16,185,129,.12) !important;
|
||||
color: #047857 !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
[★ 보완] 버튼 크기 “완전 통일”(처리/백업 공통)
|
||||
- 폰트/라인하이트/패딩을 변수화해서 두 목록 크기 동일
|
||||
- 기존 개별 padding/font-size를 덮어써서 시각적 높이 통일
|
||||
───────────────────────────────────────────────────────────── */
|
||||
:root{
|
||||
--btn-font: .80rem; /* 버튼 폰트 크기 통일 지점 */
|
||||
--btn-line: 1.2; /* 버튼 라인하이트 통일 지점 */
|
||||
--btn-py: .32rem; /* 수직 패딩 */
|
||||
--btn-px: .60rem; /* 좌우 패딩 */
|
||||
}
|
||||
|
||||
.processed-list .file-card-buttons .btn,
|
||||
.backup-list .file-card-single-button .btn {
|
||||
font-size: var(--btn-font) !important;
|
||||
line-height: var(--btn-line) !important;
|
||||
padding: var(--btn-py) var(--btn-px) !important;
|
||||
min-height: calc(1em * var(--btn-line) + (var(--btn-py) * 2)) !important;
|
||||
}
|
||||
|
||||
/* 이전 규칙보다 더 구체적으로 동일 규격을 한 번 더 보장 */
|
||||
.processed-list .file-card-buttons .btn.btn-outline,
|
||||
.backup-list .file-card-single-button .btn.btn-outline {
|
||||
font-size: var(--btn-font) !important;
|
||||
line-height: var(--btn-line) !important;
|
||||
padding: var(--btn-py) var(--btn-px) !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
[★ 보완] 카드 “자동 한줄 배치”
|
||||
- 기존 Bootstrap .row.g-3를 Grid로 오버라이드(HTML 수정 無)
|
||||
- 우측 여백 최소화, 화면 너비에 맞춰 자연스럽게 줄 수 변경
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.processed-list .card-body > .row.g-3,
|
||||
.backup-list .card-body .row.g-3 {
|
||||
display: grid !important;
|
||||
grid-template-columns: repeat(auto-fill, minmax(160px, 1fr));
|
||||
gap: .75rem;
|
||||
}
|
||||
|
||||
/* 그리드 기준으로 카드 폭이 잘 늘어나도록 제한 완화 */
|
||||
.processed-list .file-card-compact,
|
||||
.backup-list .file-card-compact {
|
||||
max-width: none !important; /* 기존 180px 제한을 목록 구간에 한해 해제 */
|
||||
min-width: 160px;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
반응형 파일 그리드 (기존 유지)
|
||||
───────────────────────────────────────────────────────────── */
|
||||
@media (max-width: 1400px) {
|
||||
.file-grid { grid-template-columns: repeat(auto-fit, minmax(150px, 1fr)); }
|
||||
}
|
||||
@media (max-width: 1200px) {
|
||||
.file-grid { grid-template-columns: repeat(auto-fit, minmax(160px, 1fr)); }
|
||||
}
|
||||
@media (max-width: 992px) {
|
||||
.file-grid { grid-template-columns: repeat(auto-fit, minmax(180px, 1fr)); }
|
||||
}
|
||||
@media (max-width: 768px) {
|
||||
.file-grid { grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); }
|
||||
}
|
||||
@media (max-width: 576px) {
|
||||
.file-grid { grid-template-columns: repeat(auto-fit, minmax(45%, 1fr)); }
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
백업 파일 리스트
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.list-group-item .bg-light {
|
||||
transition: background-color 0.2s ease;
|
||||
}
|
||||
.list-group-item:hover .bg-light {
|
||||
background-color: #e9ecef !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
모달 - 파일 내용 보기
|
||||
───────────────────────────────────────────────────────────── */
|
||||
#fileViewContent {
|
||||
background-color: #f8f9fa;
|
||||
border: 1px solid #e9ecef;
|
||||
padding: 1rem;
|
||||
border-radius: 5px;
|
||||
max-height: 60vh;
|
||||
overflow-y: auto;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
.modal-body pre::-webkit-scrollbar {
|
||||
width: 8px;
|
||||
height: 8px;
|
||||
}
|
||||
.modal-body pre::-webkit-scrollbar-track {
|
||||
background: #f1f1f1;
|
||||
border-radius: 4px;
|
||||
}
|
||||
.modal-body pre::-webkit-scrollbar-thumb {
|
||||
background: #888;
|
||||
border-radius: 4px;
|
||||
}
|
||||
.modal-body pre::-webkit-scrollbar-thumb:hover {
|
||||
background: #555;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
접근성 - Skip to content
|
||||
───────────────────────────────────────────────────────────── */
|
||||
.visually-hidden-focusable:not(:focus):not(:focus-within) {
|
||||
position: absolute !important;
|
||||
width: 1px !important;
|
||||
height: 1px !important;
|
||||
padding: 0 !important;
|
||||
margin: -1px !important;
|
||||
overflow: hidden !important;
|
||||
clip: rect(0, 0, 0, 0) !important;
|
||||
white-space: nowrap !important;
|
||||
border: 0 !important;
|
||||
}
|
||||
.visually-hidden-focusable:focus {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
z-index: 9999;
|
||||
padding: 1rem;
|
||||
background: #000;
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
전역 폰트 굵기 강제 (Bootstrap 오버라이드)
|
||||
───────────────────────────────────────────────────────────── */
|
||||
* { font-weight: inherit; }
|
||||
strong, b { font-weight: 600; }
|
||||
label, .form-label, .card-title, .list-group-item strong {
|
||||
font-weight: 400 !important;
|
||||
}
|
||||
|
||||
|
||||
/* ─────────────────────────────────────────────────────────────
|
||||
반응형
|
||||
───────────────────────────────────────────────────────────── */
|
||||
@media (max-width: 768px) {
|
||||
.card-body {
|
||||
padding: 1.5rem !important;
|
||||
}
|
||||
body {
|
||||
font-size: 0.95rem;
|
||||
}
|
||||
}
|
||||
|
||||
/* === [FIX] 처리된 목록 보기/삭제 버튼 크기 완전 동일화 === */
|
||||
|
||||
/* 1) 그리드 두 칸을 꽉 채우게 강제 */
|
||||
.processed-list .file-card-buttons {
|
||||
display: grid !important;
|
||||
grid-template-columns: 1fr 1fr !important;
|
||||
gap: .2rem !important;
|
||||
align-items: stretch !important; /* 높이도 칸 높이에 맞춰 늘림 */
|
||||
}
|
||||
|
||||
/* 2) 그리드 아이템(버튼/폼) 자체를 칸 너비로 확장 */
|
||||
.processed-list .file-card-buttons > * {
|
||||
width: 100% !important;
|
||||
}
|
||||
|
||||
/* 3) 폼 안의 버튼도 100%로 확장 (폼이 그리드 아이템인 경우 대비) */
|
||||
.processed-list .file-card-buttons > form { display: block !important; }
|
||||
.processed-list .file-card-buttons > form > button {
|
||||
display: block !important;
|
||||
width: 100% !important;
|
||||
}
|
||||
|
||||
/* 4) 예전 플렉스 기반 전역 규칙 덮어쓰기(폭 계산식 무력화) */
|
||||
.processed-list .file-card-buttons > button,
|
||||
.processed-list .file-card-buttons > form {
|
||||
width: 100% !important;
|
||||
}
|
||||
|
||||
/* 5) 폰트/라인하이트/패딩 통일(높이 동일) — 필요 시 수치만 조정 */
|
||||
:root{
|
||||
--btn-font: .80rem;
|
||||
--btn-line: 1.2;
|
||||
--btn-py: .32rem;
|
||||
--btn-px: .60rem;
|
||||
}
|
||||
.processed-list .file-card-buttons .btn {
|
||||
font-size: var(--btn-font) !important;
|
||||
line-height: var(--btn-line) !important;
|
||||
padding: var(--btn-py) var(--btn-px) !important;
|
||||
min-height: calc(1em * var(--btn-line) + (var(--btn-py) * 2)) !important;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
Reference in New Issue
Block a user