da7ebe03c6
- 반짝이는 별(twinkle 애니메이션) - 유성(shooting star) 효과 - 관리자/로그인/공개 페이지 전체 적용 - 기존 글래스모피즘 유지 + 우주 배경 레이어 추가
170 lines
13 KiB
Plaintext
170 lines
13 KiB
Plaintext
<!DOCTYPE html>
|
|
<html lang="ko">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title><%= typeof pageTitle !== 'undefined' ? pageTitle : 'Crawl Manager' %></title>
|
|
<style>
|
|
:root{--bg:#0a0e1a;--bg2:#1f2937;--bg3:#374151;--text:#f9fafb;--muted:#9ca3af;--primary:#6366f1;--primary-hover:#818cf8;--danger:#ef4444;--success:#22c55e;--warning:#f59e0b;--border:#374151;--radius:12px;--glass-bg:rgba(255,255,255,.05);--glass-bg-hover:rgba(255,255,255,.08);--glass-border:rgba(255,255,255,.1);--glass-border-hover:rgba(255,255,255,.18);--glass-blur:blur(12px);--glass-blur-heavy:blur(20px);--glass-shadow:0 8px 32px rgba(0,0,0,.3)}
|
|
*{margin:0;padding:0;box-sizing:border-box}
|
|
body{font-family:-apple-system,BlinkMacSystemFont,'Segoe UI','Noto Sans KR',sans-serif;background:var(--bg);color:var(--text);min-height:100vh;display:flex;position:relative;overflow-x:hidden}
|
|
body::before{content:'';position:fixed;top:0;left:0;width:100%;height:100%;z-index:0;background:radial-gradient(600px circle at 20% 30%,rgba(99,102,241,.1),transparent 50%),radial-gradient(500px circle at 80% 70%,rgba(6,182,212,.07),transparent 50%),radial-gradient(400px circle at 50% 50%,rgba(139,92,246,.05),transparent 50%);animation:bgShift 20s ease-in-out infinite alternate;pointer-events:none}
|
|
body::after{content:'';position:fixed;top:0;left:0;width:100%;height:100%;z-index:0;pointer-events:none;background-image:radial-gradient(1px 1px at 10% 20%,rgba(255,255,255,.6),transparent),radial-gradient(1px 1px at 25% 65%,rgba(255,255,255,.5),transparent),radial-gradient(1.5px 1.5px at 40% 10%,rgba(255,255,255,.7),transparent),radial-gradient(1px 1px at 55% 45%,rgba(255,255,255,.4),transparent),radial-gradient(1.5px 1.5px at 70% 80%,rgba(255,255,255,.6),transparent),radial-gradient(1px 1px at 85% 30%,rgba(255,255,255,.5),transparent),radial-gradient(1px 1px at 15% 85%,rgba(255,255,255,.4),transparent),radial-gradient(1.5px 1.5px at 60% 15%,rgba(255,255,255,.7),transparent),radial-gradient(1px 1px at 90% 55%,rgba(255,255,255,.5),transparent),radial-gradient(1px 1px at 35% 40%,rgba(255,255,255,.3),transparent),radial-gradient(1.5px 1.5px at 5% 50%,rgba(255,255,255,.6),transparent),radial-gradient(1px 1px at 75% 25%,rgba(255,255,255,.4),transparent),radial-gradient(1px 1px at 45% 75%,rgba(255,255,255,.5),transparent),radial-gradient(1.5px 1.5px at 20% 95%,rgba(255,255,255,.6),transparent),radial-gradient(1px 1px at 95% 10%,rgba(255,255,255,.4),transparent),radial-gradient(1px 1px at 50% 90%,rgba(255,255,255,.3),transparent),radial-gradient(1.5px 1.5px at 30% 55%,rgba(255,255,255,.5),transparent),radial-gradient(1px 1px at 65% 35%,rgba(255,255,255,.4),transparent),radial-gradient(1px 1px at 80% 95%,rgba(255,255,255,.6),transparent),radial-gradient(1.5px 1.5px at 12% 40%,rgba(255,255,255,.5),transparent);animation:twinkle 4s ease-in-out infinite alternate}
|
|
@keyframes bgShift{0%{transform:translate(0,0) scale(1)}50%{transform:translate(-30px,20px) scale(1.05)}100%{transform:translate(20px,-15px) scale(1)}}
|
|
@keyframes twinkle{0%{opacity:.6}100%{opacity:1}}
|
|
.shooting-star{position:fixed;width:80px;height:1px;background:linear-gradient(90deg,rgba(255,255,255,.8),transparent);z-index:0;pointer-events:none;opacity:0;animation:shoot 3s ease-in-out infinite}
|
|
.shooting-star:nth-child(2){top:15%;left:60%;animation-delay:2s;width:60px}
|
|
.shooting-star:nth-child(3){top:70%;left:20%;animation-delay:5s;width:100px}
|
|
@keyframes shoot{0%{transform:translateX(0) translateY(0) rotate(-35deg);opacity:0}5%{opacity:1}30%{transform:translateX(300px) translateY(150px) rotate(-35deg);opacity:0}100%{opacity:0}}
|
|
|
|
/* 사이드바 */
|
|
.sidebar{width:220px;background:rgba(15,20,35,.75);backdrop-filter:var(--glass-blur-heavy);-webkit-backdrop-filter:var(--glass-blur-heavy);border-right:1px solid var(--glass-border);padding:1.5rem 0;flex-shrink:0;position:fixed;top:0;left:0;height:100vh;overflow-y:auto;z-index:2}
|
|
.sidebar .logo{padding:0 1.2rem 1.5rem;font-size:1.1rem;font-weight:700;color:var(--primary);border-bottom:1px solid var(--glass-border);margin-bottom:1rem}
|
|
.sidebar nav a{display:flex;align-items:center;gap:.6rem;padding:.7rem 1.2rem;color:var(--muted);text-decoration:none;font-size:.88rem;transition:all .2s;border-left:3px solid transparent}
|
|
.sidebar nav a:hover{color:var(--text);background:rgba(255,255,255,.05)}
|
|
.sidebar nav a.active{color:var(--text);background:rgba(99,102,241,.1);border-left:3px solid var(--primary)}
|
|
|
|
/* 메인 */
|
|
.main{margin-left:220px;flex:1;min-height:100vh;position:relative;z-index:1}
|
|
.topbar{background:rgba(15,20,35,.6);backdrop-filter:var(--glass-blur);-webkit-backdrop-filter:var(--glass-blur);border-bottom:1px solid var(--glass-border);padding:.8rem 1.5rem;display:flex;justify-content:space-between;align-items:center}
|
|
.topbar h1{font-size:1.1rem;font-weight:600}
|
|
.content{padding:1.5rem}
|
|
|
|
/* 카드 */
|
|
.card{background:var(--glass-bg);backdrop-filter:var(--glass-blur);-webkit-backdrop-filter:var(--glass-blur);border:1px solid var(--glass-border);border-radius:var(--radius);padding:1.2rem;margin-bottom:1rem;box-shadow:var(--glass-shadow);transition:all .2s}
|
|
.card:hover{border-color:var(--glass-border-hover)}
|
|
.card-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:1rem;padding-bottom:.8rem;border-bottom:1px solid var(--glass-border)}
|
|
.card-header h2{font-size:1rem;font-weight:600}
|
|
|
|
/* 버튼 */
|
|
.btn{padding:.5rem 1rem;border:none;border-radius:var(--radius);cursor:pointer;font-size:.85rem;font-weight:500;transition:all .2s;display:inline-flex;align-items:center;gap:.4rem}
|
|
.btn-primary{background:linear-gradient(135deg,var(--primary),#7c3aed);color:#fff;box-shadow:0 4px 15px rgba(99,102,241,.3)}
|
|
.btn-primary:hover{background:linear-gradient(135deg,var(--primary-hover),#8b5cf6);transform:translateY(-1px);box-shadow:0 6px 20px rgba(99,102,241,.4)}
|
|
.btn-danger{background:linear-gradient(135deg,var(--danger),#dc2626);color:#fff}
|
|
.btn-danger:hover{opacity:.9;transform:translateY(-1px)}
|
|
.btn-success{background:linear-gradient(135deg,var(--success),#16a34a);color:#fff}
|
|
.btn-warning{background:linear-gradient(135deg,var(--warning),#d97706);color:#000}
|
|
.btn-sm{padding:.35rem .7rem;font-size:.78rem}
|
|
.btn-outline{background:rgba(255,255,255,.03);border:1px solid var(--glass-border);color:var(--text)}
|
|
.btn-outline:hover{background:rgba(255,255,255,.08);border-color:var(--glass-border-hover)}
|
|
|
|
/* 테이블 */
|
|
table{width:100%;border-collapse:collapse}
|
|
th,td{padding:.6rem .8rem;text-align:left;border-bottom:1px solid var(--glass-border);font-size:.85rem}
|
|
th{color:var(--muted);font-weight:500;font-size:.78rem;text-transform:uppercase;letter-spacing:.5px}
|
|
tr:hover td{background:rgba(255,255,255,.04)}
|
|
|
|
/* 폼 */
|
|
.form-group{margin-bottom:1rem}
|
|
.form-group label{display:block;font-size:.82rem;color:var(--muted);margin-bottom:.3rem;font-weight:500}
|
|
.form-group input,.form-group textarea,.form-group select{width:100%;padding:.55rem .8rem;background:rgba(0,0,0,.3);border:1px solid var(--glass-border);border-radius:var(--radius);color:var(--text);font-size:.88rem;font-family:inherit;transition:all .2s}
|
|
.form-group input:focus,.form-group textarea:focus,.form-group select:focus{outline:none;border-color:var(--primary);box-shadow:0 0 0 3px rgba(99,102,241,.15);background:rgba(0,0,0,.4)}
|
|
.form-group textarea{resize:vertical;min-height:80px;font-family:monospace}
|
|
.form-row{display:grid;grid-template-columns:1fr 1fr;gap:1rem}
|
|
|
|
/* 뱃지 */
|
|
.badge{display:inline-block;padding:.15rem .5rem;border-radius:10px;font-size:.72rem;font-weight:600}
|
|
.badge-success{background:rgba(34,197,94,.15);color:var(--success)}
|
|
.badge-danger{background:rgba(239,68,68,.15);color:var(--danger)}
|
|
.badge-warning{background:rgba(245,158,11,.15);color:var(--warning)}
|
|
.badge-info{background:rgba(99,102,241,.15);color:var(--primary-hover)}
|
|
|
|
/* 모달 */
|
|
.modal-overlay{display:none;position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.4);backdrop-filter:blur(4px);-webkit-backdrop-filter:blur(4px);z-index:100;align-items:center;justify-content:center}
|
|
.modal-overlay.active{display:flex}
|
|
.modal{background:rgba(15,20,35,.85);backdrop-filter:var(--glass-blur-heavy);-webkit-backdrop-filter:var(--glass-blur-heavy);border:1px solid var(--glass-border);border-radius:16px;padding:1.5rem;width:90%;max-width:600px;max-height:90vh;overflow-y:auto;box-shadow:0 25px 50px rgba(0,0,0,.5)}
|
|
.modal h3{margin-bottom:1rem;font-size:1rem}
|
|
|
|
/* 유틸 */
|
|
.text-muted{color:var(--muted)}
|
|
.text-success{color:var(--success)}
|
|
.text-danger{color:var(--danger)}
|
|
.mt-1{margin-top:.5rem}
|
|
.mb-1{margin-bottom:.5rem}
|
|
.flex{display:flex;gap:.5rem;align-items:center}
|
|
.flex-between{display:flex;justify-content:space-between;align-items:center}
|
|
|
|
/* 통계 카드 */
|
|
.stats-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(180px,1fr));gap:1rem;margin-bottom:1.5rem}
|
|
.stat-card{background:var(--glass-bg);backdrop-filter:var(--glass-blur);-webkit-backdrop-filter:var(--glass-blur);border:1px solid var(--glass-border);border-radius:var(--radius);padding:1rem;transition:all .2s;box-shadow:var(--glass-shadow)}
|
|
.stat-card:hover{background:var(--glass-bg-hover);transform:translateY(-2px);border-color:var(--glass-border-hover)}
|
|
.stat-card .number{font-size:1.8rem;font-weight:700}
|
|
.stat-card .label{color:var(--muted);font-size:.8rem;margin-top:.2rem}
|
|
|
|
/* 토스트 */
|
|
.toast{position:fixed;top:1rem;right:1rem;background:rgba(15,20,35,.85);backdrop-filter:var(--glass-blur);-webkit-backdrop-filter:var(--glass-blur);border:1px solid var(--glass-border);border-radius:var(--radius);padding:.8rem 1.2rem;z-index:200;display:none;font-size:.85rem;box-shadow:0 8px 32px rgba(0,0,0,.4)}
|
|
.toast.show{display:block;animation:slideIn .3s}
|
|
@keyframes slideIn{from{transform:translateX(100%);opacity:0}to{transform:translateX(0);opacity:1}}
|
|
|
|
/* 크론 프리셋 */
|
|
.cron-presets{display:flex;flex-wrap:wrap;gap:.4rem;margin-top:.4rem}
|
|
.cron-presets .preset{padding:.25rem .6rem;background:rgba(255,255,255,.05);border:1px solid var(--glass-border);border-radius:6px;cursor:pointer;font-size:.75rem;color:var(--muted);transition:all .2s}
|
|
.cron-presets .preset:hover{border-color:var(--primary);color:var(--text);background:rgba(99,102,241,.1)}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div class="shooting-star" style="top:25%;left:40%"></div>
|
|
<div class="shooting-star" style="top:15%;left:60%"></div>
|
|
<div class="shooting-star" style="top:70%;left:20%"></div>
|
|
<aside class="sidebar">
|
|
<div class="logo">Crawl Manager</div>
|
|
<nav>
|
|
<a href="/admin" class="<%= typeof page !== 'undefined' && page === 'dashboard' ? 'active' : '' %>">📊 대시보드</a>
|
|
<a href="/admin/sites" class="<%= typeof page !== 'undefined' && page === 'sites' ? 'active' : '' %>">🌐 사이트 관리</a>
|
|
<a href="/admin/adsense" class="<%= typeof page !== 'undefined' && page === 'adsense' ? 'active' : '' %>">💰 AdSense 관리</a>
|
|
<a href="/admin/mapper" class="<%= typeof page !== 'undefined' && page === 'mapper' ? 'active' : '' %>">🧙 비주얼 매퍼</a>
|
|
<a href="/admin/domains" class="<%= typeof page !== 'undefined' && page === 'domains' ? 'active' : '' %>">🔗 도메인 매핑</a>
|
|
<a href="/admin/logs" class="<%= typeof page !== 'undefined' && page === 'logs' ? 'active' : '' %>">📝 로그</a>
|
|
<div style="border-top:1px solid var(--border);margin-top:auto;padding-top:.5rem;margin-top:1rem">
|
|
<a href="/logout" style="color:var(--danger)">🚪 로그아웃</a>
|
|
</div>
|
|
</nav>
|
|
</aside>
|
|
<div class="main">
|
|
<div class="topbar">
|
|
<h1><%= typeof pageTitle !== 'undefined' ? pageTitle : '' %></h1>
|
|
<div class="flex" style="gap:1rem">
|
|
<span class="text-muted" style="font-size:.8rem">Crawl Manager v1.0</span>
|
|
<a href="/logout" class="btn btn-outline btn-sm" style="font-size:.75rem">로그아웃</a>
|
|
</div>
|
|
</div>
|
|
<div class="content">
|
|
<%- typeof ssrData !== 'undefined' ? ssrData : '' %>
|
|
<script>
|
|
function api(method, url, data) {
|
|
var opts = { method: method, headers: { 'Content-Type': 'application/json' }, credentials: 'same-origin' };
|
|
if (data) opts.body = JSON.stringify(data);
|
|
return fetch(url, opts).then(function(r) {
|
|
var ct = r.headers.get('content-type') || '';
|
|
if (r.status === 401) { window.location.href = '/login?redirect=' + encodeURIComponent(window.location.pathname); return []; }
|
|
if (!ct.includes('application/json')) { console.error('API non-JSON:', r.status); return []; }
|
|
return r.json();
|
|
}).catch(function(err) { console.error('API Error:', err); return []; });
|
|
}
|
|
function toast(msg, type) {
|
|
type = type || 'success';
|
|
var el = document.getElementById('toast');
|
|
el.textContent = msg;
|
|
el.style.borderLeftColor = type === 'success' ? 'var(--success)' : type === 'error' ? 'var(--danger)' : 'var(--warning)';
|
|
el.style.borderLeftWidth = '3px';
|
|
el.classList.add('show');
|
|
setTimeout(function() { el.classList.remove('show'); }, 3000);
|
|
}
|
|
function timeAgo(dateStr) {
|
|
if (!dateStr) return '-';
|
|
var diff = Date.now() - new Date(dateStr).getTime();
|
|
var m = Math.floor(diff / 60000);
|
|
if (m < 1) return '방금';
|
|
if (m < 60) return m + '분 전';
|
|
var h = Math.floor(m / 60);
|
|
if (h < 24) return h + '시간 전';
|
|
return Math.floor(h / 24) + '일 전';
|
|
}
|
|
</script>
|
|
<%- body %>
|
|
</div>
|
|
</div>
|
|
<div class="toast" id="toast"></div>
|
|
</body>
|
|
</html>
|