-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathlite_hackathon.js
More file actions
302 lines (269 loc) · 9.36 KB
/
lite_hackathon.js
File metadata and controls
302 lines (269 loc) · 9.36 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
// Helper
const $ = (s, c = document) => c.querySelector(s);
const $$ = (s, c = document) => Array.from(c.querySelectorAll(s));
// Footer year
const year = $('#year');
if (year) year.textContent = new Date().getFullYear();
/* ========== Sidebar ========== */
const openSidebar = () => {
document.body.classList.add('is-open');
$('#sidebar').setAttribute('aria-hidden', 'false');
$('#overlay').setAttribute('aria-hidden', 'false');
};
const closeSidebar = () => {
document.body.classList.remove('is-open');
$('#sidebar').setAttribute('aria-hidden', 'true');
$('#overlay').setAttribute('aria-hidden', 'true');
};
$('#openSidebar').addEventListener('click', openSidebar);
$('#closeSidebar').addEventListener('click', closeSidebar);
$('#overlay').addEventListener('click', closeSidebar);
window.addEventListener('keydown', (e) => {
if (e.key === 'Escape') closeSidebar();
});
/* ========== Login ========== */
$('#loginBtn').addEventListener('click', () => {
window.location.href = 'login.html';
});
/* ========== 🔍 검색 기능 (핵심 업데이트) ========== */
$('#searchForm').addEventListener('submit', (e) => {
e.preventDefault();
const input = $('#q');
const query = (input.value || '').trim(); // 공백 제거
if (!query) return alert('검색어를 입력하세요.');
// 1. 페이지 매핑 리스트
const pageMap = [
{
keywords: ['수행', '평가', '과제'],
url: 'suhang.html',
name: '수행평가',
},
{
keywords: ['성적', '점수', '등급', '내신'],
url: 'score.html',
name: '성적조회',
},
{
keywords: ['답안', '정답', '모범'],
url: 'mobum.html',
name: '정기고사 모범답안',
},
{ keywords: ['채점', '가채점'], url: 'gache.html', name: '가채점' },
{
keywords: ['모의', '학력', '수능'],
url: 'mogo.html',
name: '모의고사 학습',
},
{ keywords: ['알림', '신청'], url: 'allim.html', name: '알림신청' },
{
keywords: ['게시판', '공지', '소통', '자유'],
url: 'board.html',
name: '경희 게시판',
},
{ keywords: ['자료', '파일', '다운'], url: 'files.html', name: '학습자료' },
{ keywords: ['마이', '내정보', '프로필'], url: 'my.html', name: 'My 기능' },
{
keywords: ['상담', '진로', '컨설팅'],
url: 'sangdam.html',
name: '진로상담',
},
{
keywords: ['학사', '일정', '달력', '캘린더'],
url: 'haksa.html',
name: '학사일정',
},
{ keywords: ['로그인', '접속'], url: 'login.html', name: '로그인' },
];
// 2. 키워드 매칭 확인
const target = pageMap.find((item) =>
item.keywords.some((k) => query.includes(k))
);
if (target) {
// 매칭된 페이지가 있으면 이동
// confirm(`'${target.name}' 페이지로 이동하시겠습니까?`) // 확인창이 필요하면 주석 해제
window.location.href = target.url;
} else {
// 3. 페이지가 없으면 위젯 관련 검색인지 확인
if (
query.includes('급식') ||
query.includes('밥') ||
query.includes('메뉴')
) {
alert(
"급식 정보는 '오늘의 급식' 위젯에서 확인할 수 있습니다.\n위젯 추가 화면을 열어드릴게요!"
);
openModal();
} else if (query.includes('시간표')) {
alert('시간표는 위젯으로 제공됩니다.\n위젯 추가 화면을 열어드릴게요!');
openModal();
} else {
// 4. 아무것도 없으면 안내 메시지
alert(
`'${query}'에 대한 페이지를 찾지 못했습니다.\n사이드바 메뉴를 확인해보세요.`
);
openSidebar(); // 사이드바를 열어주는 센스
}
}
// 입력창 초기화 및 포커스 해제 (모바일 키보드 닫기)
input.value = '';
input.blur();
});
/* ========== Widget modal ========== */
const modal = $('#widgetModal');
const openModal = () => document.body.classList.add('is-modal');
const closeModal = () => document.body.classList.remove('is-modal');
$('#fabAdd').addEventListener('click', openModal);
$('#closeModal').addEventListener('click', closeModal);
modal.addEventListener('click', (e) => {
if (e.target === modal) closeModal();
});
/* ========== Widget factory ========== */
function createWidget(type) {
const card = document.createElement('article');
card.className = 'widget';
card.dataset.widget = type;
const header = document.createElement('header');
header.className = 'widget-header';
const h3 = document.createElement('h3');
const closeBtn = document.createElement('button');
closeBtn.className = 'icon-btn sm remove-widget';
closeBtn.title = '위젯 삭제';
closeBtn.innerHTML = `
<svg width="16" height="16" viewBox="0 0 24 24" aria-hidden="true">
<path d="M6 6l12 12M18 6L6 18" stroke="currentColor" stroke-width="2" stroke-linecap="round"/>
</svg>`;
closeBtn.addEventListener('click', () => {
setTimeout(() => {
card.remove();
updateWidgetHint();
}, 0);
});
header.append(h3, closeBtn);
const body = document.createElement('div');
body.className = 'widget-body';
switch (type) {
case 'timetable': {
h3.textContent = '시간표';
const wrap = document.createElement('div');
wrap.className = 'timetable';
const cells = document.createElement('div');
cells.className = 'cells';
for (let i = 0; i < 30; i++)
cells.appendChild(document.createElement('div'));
wrap.appendChild(cells);
body.appendChild(wrap);
break;
}
case 'lunch': {
h3.textContent = '오늘의 급식';
const ul = document.createElement('ul');
ul.className = 'lunch-list';
ul.id = 'todayLunchWidget'; // ID 부여 (데이터 로딩용)
['로딩 중...'].forEach((t) => {
const li = document.createElement('li');
li.textContent = t;
ul.appendChild(li);
});
body.appendChild(ul);
// 위젯 생성 시 데이터 로드 트리거
setTimeout(loadTodayLunch, 100);
break;
}
case 'exam-schedule': {
h3.textContent = '수행평가 일정';
body.innerHTML =
'<p>다가오는 수행평가 일정이 표시됩니다. (연동 예정)</p>';
break;
}
case 'calendar': {
h3.textContent = '캘린더';
body.innerHTML = '<p>월간 학사일정 위젯입니다. (연동 예정)</p>';
break;
}
case 'notice': {
h3.textContent = '알림';
body.innerHTML = '<p>공지/알림 모아보기. (연동 예정)</p>';
break;
}
default: {
h3.textContent = '커스텀 위젯';
body.textContent = '내용을 구성하세요.';
}
}
card.append(header, body);
return card;
}
/* ========== 위젯 안내 문구 ========== */
function updateWidgetHint() {
const widgetCount = document.querySelectorAll('.widget-grid .widget').length;
const hint = document.getElementById('widgetHint');
if (!hint) return;
hint.style.display = widgetCount === 0 ? 'inline-block' : 'none';
}
/* ========== 모달에서 위젯 추가 ========== */
$$('.picker-item', modal).forEach((btn) => {
btn.addEventListener('click', () => {
const w = createWidget(btn.dataset.widget);
$('#widgetGrid').appendChild(w);
updateWidgetHint();
closeModal();
});
});
/* ========== 초기 로드 ========== */
updateWidgetHint();
// === 다크모드 / 라이트모드 토글 ===
document.addEventListener('DOMContentLoaded', () => {
const themeBtn = document.getElementById('themeToggle');
const body = document.body;
const savedTheme = localStorage.getItem('theme');
if (savedTheme === 'light') {
body.classList.add('light-mode');
themeBtn.textContent = '☀️';
} else {
themeBtn.textContent = '🌙';
}
themeBtn.addEventListener('click', () => {
body.classList.add('theme-transition');
setTimeout(() => body.classList.remove('theme-transition'), 500);
body.classList.toggle('light-mode');
const isLight = body.classList.contains('light-mode');
themeBtn.textContent = isLight ? '☀️' : '🌙';
localStorage.setItem('theme', isLight ? 'light' : 'dark');
});
});
document.body.classList.add('theme-transition');
setTimeout(() => {
document.body.classList.remove('theme-transition');
}, 600);
/* ============================= */
/* 🍱 오늘의 급식 데이터 로드 */
/* ============================= */
async function loadTodayLunch() {
// 생성된 위젯이 있는지 확인
const listElement = document.querySelector('.lunch-list');
if (!listElement) return;
try {
// 실제로는 API나 파일을 불러옵니다. 여기서는 예시 데이터
const dummyMeals = [
{
날짜: '11월 8일',
메뉴: '현미밥, 쇠고기미역국, 돈육불고기, 계란말이, 배추김치',
},
];
const today = new Date();
const todayStr = `${today.getMonth() + 1}월 ${today.getDate()}일`;
// 예시 데이터 사용 (실제 구현 시 fetch 사용)
// const meal = dummyMeals.find(...)
// UI 업데이트 (테스트용)
listElement.innerHTML = `
<li><strong>${todayStr} 급식</strong></li>
<li>쌀밥</li>
<li>순두부찌개</li>
<li>제육볶음</li>
<li>깍두기</li>
`;
} catch (err) {
console.error('급식 로드 실패:', err);
listElement.innerHTML = `<li>정보를 불러올 수 없습니다.</li>`;
}
}