485 lines
22 KiB
C++
485 lines
22 KiB
C++
#ifndef SETTINGS_H
|
||
#define SETTINGS_H
|
||
|
||
const char settings_html[] PROGMEM = R"rawliteral(
|
||
<!DOCTYPE html>
|
||
<html>
|
||
<head>
|
||
<meta charset="UTF-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no">
|
||
<title>Settings - Byun CAN Logger</title>
|
||
<style>
|
||
:root {
|
||
--bg: #0e1117;
|
||
--panel: #161b24;
|
||
--card: #1c2230;
|
||
--border: #2d3748;
|
||
--accent: #43cea2;
|
||
--green: #3fb950;
|
||
--blue: #58a6ff;
|
||
--red: #f85149;
|
||
--yellow: #e3b341;
|
||
--purple: #bc8cff;
|
||
--text: #e6edf3;
|
||
--muted: #8b949e;
|
||
--r: 8px;
|
||
}
|
||
* { margin:0; padding:0; box-sizing:border-box; }
|
||
html, body { min-height:100%; }
|
||
body {
|
||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
|
||
background: var(--bg); color: var(--text);
|
||
overflow-x: hidden; font-size: 14px;
|
||
}
|
||
|
||
/* Header */
|
||
.header {
|
||
background: linear-gradient(135deg,#1a2744 0%,#1e1a3a 100%);
|
||
padding: 11px 16px; border-bottom: 1px solid var(--border);
|
||
display: flex; align-items: center; gap: 10px; flex-wrap: wrap;
|
||
}
|
||
.header h1 {
|
||
font-size: 1.0em; font-weight: 700; color: var(--accent);
|
||
display: flex; align-items: center; gap: 8px;
|
||
}
|
||
.header p { font-size: 0.78em; color: var(--muted); margin:0; }
|
||
.badge {
|
||
background: rgba(67,206,162,.15); color: var(--accent);
|
||
padding: 2px 8px; border-radius: 12px; font-size: 0.65em;
|
||
font-weight: 600; border: 1px solid rgba(67,206,162,.3);
|
||
white-space: nowrap;
|
||
}
|
||
.header-spacer { flex:1; }
|
||
|
||
/* Nav */
|
||
.nav {
|
||
background: var(--panel); border-bottom: 1px solid var(--border);
|
||
display: flex; overflow-x: auto; -webkit-overflow-scrolling: touch;
|
||
scrollbar-width: none;
|
||
}
|
||
.nav::-webkit-scrollbar { display:none; }
|
||
.nav a {
|
||
display: inline-flex; align-items: center;
|
||
padding: 10px 13px; text-decoration: none; color: var(--muted);
|
||
font-size: 0.78em; font-weight: 500;
|
||
border-bottom: 2px solid transparent; white-space: nowrap;
|
||
transition: color .2s, border-color .2s;
|
||
}
|
||
.nav a:hover { color: var(--text); }
|
||
.nav a.active { color: var(--accent); border-bottom-color: var(--accent); }
|
||
|
||
/* Content */
|
||
.content { padding: 12px; }
|
||
|
||
/* Headings */
|
||
h2 {
|
||
color: var(--accent); margin: 14px 0 10px;
|
||
font-size: 0.82em; font-weight: 700; text-transform: uppercase;
|
||
letter-spacing: .5px; padding-bottom: 6px;
|
||
border-bottom: 1px solid var(--border);
|
||
}
|
||
h3 {
|
||
color: var(--text); font-size: 0.85em; font-weight: 600;
|
||
margin-bottom: 10px; padding-bottom: 6px;
|
||
border-bottom: 1px solid var(--border);
|
||
}
|
||
|
||
/* Buttons */
|
||
.btn, button {
|
||
padding: 6px 13px; border: 1px solid var(--border);
|
||
border-radius: var(--r); background: var(--bg); color: var(--muted);
|
||
font-size: 0.8em; font-weight: 600; cursor: pointer;
|
||
font-family: inherit; transition: all .15s; white-space: nowrap;
|
||
-webkit-tap-highlight-color: transparent; touch-action: manipulation;
|
||
}
|
||
.btn:hover, button:hover { border-color: var(--accent); color: var(--accent); }
|
||
.btn:active, button:active { transform: scale(.97); }
|
||
.btn-primary { border-color:var(--blue); color:var(--blue); }
|
||
.btn-success { border-color:var(--accent); color:var(--accent); }
|
||
.btn-danger { border-color:var(--red); color:var(--red); }
|
||
.btn-warning { border-color:var(--yellow); color:var(--yellow); }
|
||
.btn-secondary{ border-color:var(--muted); color:var(--muted); }
|
||
.btn-primary:hover { background:rgba(88,166,255,.10); }
|
||
.btn-success:hover { background:rgba(67,206,162,.10); }
|
||
.btn-danger:hover { background:rgba(248,81,73,.10); }
|
||
.btn-warning:hover { background:rgba(227,179,65,.10); }
|
||
.btn-secondary:hover{ background:rgba(139,148,158,.10); }
|
||
.btn-small { padding:4px 9px; font-size:.75em; }
|
||
.button-group, .btn-group { display:flex; gap:6px; flex-wrap:wrap; margin-top:10px; }
|
||
|
||
/* Forms */
|
||
label {
|
||
display:block; font-weight:600; color:var(--muted);
|
||
margin-bottom:4px; font-size:.78em;
|
||
text-transform:uppercase; letter-spacing:.3px;
|
||
}
|
||
input[type="text"], input[type="password"], input[type="number"],
|
||
select, textarea {
|
||
width:100%; padding:7px 10px;
|
||
border:1px solid var(--border); border-radius:var(--r);
|
||
font-size:.85em; font-family:inherit;
|
||
background:var(--bg); color:var(--text);
|
||
transition:border-color .2s;
|
||
}
|
||
input:focus, select:focus, textarea:focus {
|
||
outline:none; border-color:var(--accent);
|
||
box-shadow:0 0 0 2px rgba(67,206,162,.10);
|
||
}
|
||
input[type="checkbox"], input[type="radio"] {
|
||
width:15px; height:15px; cursor:pointer; accent-color:var(--accent);
|
||
}
|
||
select option { background:var(--panel); color:var(--text); }
|
||
|
||
/* Panels */
|
||
.section, .settings-section, .send-panel, .stats,
|
||
.control-panel, .auto-trigger-section {
|
||
background:var(--panel); border:1px solid var(--border);
|
||
border-radius:var(--r); padding:13px; margin-bottom:10px;
|
||
}
|
||
.form-group { margin-bottom:12px; }
|
||
.form-group:last-child { margin-bottom:0; }
|
||
.help-text { font-size:.76em; color:var(--muted); margin-top:4px; line-height:1.4; }
|
||
.form-row {
|
||
display:grid;
|
||
grid-template-columns:repeat(auto-fit,minmax(min(100%,190px),1fr));
|
||
gap:10px; margin-bottom:10px;
|
||
}
|
||
.form-row label { text-transform:none; font-size:.78em; }
|
||
.checkbox-group, .checkbox-row {
|
||
display:flex; align-items:center; gap:8px;
|
||
margin-bottom:8px; padding:7px 10px;
|
||
background:var(--card); border-radius:6px; border:1px solid var(--border);
|
||
}
|
||
.checkbox-group label, .checkbox-row label {
|
||
text-transform:none; cursor:pointer; margin-bottom:0;
|
||
color:var(--text); font-size:.85em; letter-spacing:0;
|
||
}
|
||
|
||
/* Alerts */
|
||
.alert {
|
||
padding:9px 13px; border-radius:var(--r);
|
||
margin-bottom:10px; display:none;
|
||
align-items:center; gap:9px;
|
||
font-size:.83em; font-weight:600;
|
||
border:1px solid var(--border); background:var(--panel); color:var(--text);
|
||
}
|
||
.alert.show { display:flex; }
|
||
.alert-success { border-color:rgba(67,206,162,.4); color:var(--accent); background:rgba(67,206,162,.07); }
|
||
.alert-info { border-color:rgba(88,166,255,.4); color:var(--blue); background:rgba(88,166,255,.07); }
|
||
.alert-warning { border-color:rgba(227,179,65,.4); color:var(--yellow); background:rgba(227,179,65,.07); }
|
||
.alert-icon { font-size:1.15em; }
|
||
.alert-text { flex:1; }
|
||
@keyframes slideDown { from{opacity:0;transform:translateY(-6px);}to{opacity:1;transform:translateY(0);} }
|
||
|
||
/* Connection status */
|
||
.connection-status {
|
||
position:fixed; top:10px; right:10px;
|
||
padding:4px 11px; border-radius:20px;
|
||
font-size:.75em; font-weight:600; z-index:1000;
|
||
border:1px solid var(--border); background:var(--panel); color:var(--muted);
|
||
}
|
||
.connection-status.connected { border-color:rgba(67,206,162,.5); color:var(--accent); background:rgba(67,206,162,.08); }
|
||
.connection-status.disconnected { border-color:rgba(248,81,73,.5); color:var(--red); background:rgba(248,81,73,.08); }
|
||
|
||
@media (max-width:480px) { .content{padding:8px;} h2{font-size:.78em;} }
|
||
|
||
/* settings.h specific */
|
||
.section-title {
|
||
color:var(--accent); font-size:.85em; font-weight:700;
|
||
margin-bottom:14px; padding-bottom:7px; border-bottom:1px solid var(--border);
|
||
display:flex; align-items:center; gap:7px;
|
||
text-transform:uppercase; letter-spacing:.4px;
|
||
}
|
||
.sta-settings {
|
||
background:var(--card); padding:13px; border-radius:var(--r);
|
||
margin-top:10px; border:1px solid var(--border); border-left:3px solid var(--blue);
|
||
}
|
||
.sta-settings.disabled { opacity:.4; pointer-events:none; }
|
||
.btn-save { border-color:var(--accent); color:var(--accent); flex:1; min-width:130px; padding:9px 18px; }
|
||
.btn-save:hover { background:rgba(67,206,162,.10); }
|
||
.btn-cancel { border-color:var(--muted); color:var(--muted); flex:1; min-width:130px; padding:9px 18px; }
|
||
.btn-cancel:hover { background:rgba(139,148,158,.10); }
|
||
.info-box {
|
||
background:rgba(88,166,255,.06); border:1px solid rgba(88,166,255,.2);
|
||
border-left:3px solid var(--blue); padding:11px 13px; border-radius:var(--r); margin-top:10px;
|
||
}
|
||
.info-box-title { font-weight:700; color:var(--blue); margin-bottom:5px; display:flex; align-items:center; gap:5px; font-size:.83em; }
|
||
.info-box-text { color:var(--muted); font-size:.80em; line-height:1.6; }
|
||
</style>
|
||
</head>
|
||
<body>
|
||
<div class="header">
|
||
<h1>⚙️ Settings</h1>
|
||
<p>Configure WiFi and System Settings</p>
|
||
</div>
|
||
|
||
<div class="nav">
|
||
<a href="/">📊 Monitor</a>
|
||
<a href="/transmit">📤 Transmit</a>
|
||
<a href="/graph">📈 Graph</a>
|
||
<a href="/graph-view">📊 Graph View</a>
|
||
<a href="/settings" class="active">⚙️ Settings</a>
|
||
<a href="/serial">📟 Serial1</a>
|
||
<a href="/serial2">📟 Serial2</a>
|
||
</div>
|
||
|
||
<div class="content">
|
||
<div id="alert-success" class="alert alert-success">
|
||
<span class="alert-icon">✓</span>
|
||
<span class="alert-text">Settings saved successfully!</span>
|
||
</div>
|
||
|
||
<div id="alert-loading" class="alert alert-info">
|
||
<span class="alert-icon">⏳</span>
|
||
<span class="alert-text">Loading settings...</span>
|
||
</div>
|
||
|
||
<div class="settings-section">
|
||
<div class="section-title">
|
||
<span>📶</span>
|
||
<span>WiFi Configuration</span>
|
||
</div>
|
||
|
||
<h3 style="color: #333; font-size: 1.1em; margin-bottom: 15px;">AP Mode (Access Point)</h3>
|
||
|
||
<div class="form-group">
|
||
<label for="wifi-ssid">AP SSID (네트워크 이름)</label>
|
||
<input type="text" id="wifi-ssid" placeholder="Byun_CAN_Logger" maxlength="31">
|
||
<div class="help-text">ESP32가 생성할 WiFi 네트워크 이름입니다 (최대 31자)</div>
|
||
</div>
|
||
|
||
<div class="form-group">
|
||
<label for="wifi-password">AP Password (비밀번호)</label>
|
||
<input type="password" id="wifi-password" placeholder="최소 8자 이상" minlength="8" maxlength="63">
|
||
<div class="help-text">WiFi 접속 시 필요한 비밀번호입니다 (8-63자)</div>
|
||
</div>
|
||
|
||
<hr style="margin: 25px 0; border: none; border-top: 1px solid #ddd;">
|
||
|
||
<h3 style="color: #333; font-size: 1.1em; margin-bottom: 15px;">APSTA Mode (AP + Station)</h3>
|
||
|
||
<div class="checkbox-group">
|
||
<input type="checkbox" id="sta-enable" onchange="toggleSTASettings()">
|
||
<label for="sta-enable">Station 모드 활성화 (외부 WiFi에 연결)</label>
|
||
</div>
|
||
<div class="help-text" style="margin-left: 30px; margin-bottom: 15px;">
|
||
AP와 Station을 동시에 사용하여 인터넷 접속 가능
|
||
</div>
|
||
|
||
<div id="sta-settings" class="sta-settings disabled">
|
||
<div class="form-group">
|
||
<label for="sta-ssid">연결할 WiFi SSID</label>
|
||
<input type="text" id="sta-ssid" placeholder="공유기 이름 입력" maxlength="31">
|
||
<div class="help-text">연결할 외부 WiFi 네트워크 이름</div>
|
||
</div>
|
||
|
||
<div class="form-group">
|
||
<label for="sta-password">연결할 WiFi Password</label>
|
||
<input type="password" id="sta-password" placeholder="공유기 비밀번호 입력" maxlength="63">
|
||
<div class="help-text">외부 WiFi 비밀번호</div>
|
||
</div>
|
||
|
||
<div id="sta-status" style="display:none;margin-top:9px;padding:8px 12px;background:rgba(67,206,162,.10);border-radius:6px;color:var(--accent);font-weight:600;border:1px solid rgba(67,206,162,.2);">
|
||
✓ WiFi 연결됨: <span id="sta-ip"></span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="button-group">
|
||
<button class="btn-save" onclick="saveSettings()">💾 Save Settings</button>
|
||
<button class="btn-cancel" onclick="location.href='/'">← Back to Monitor</button>
|
||
</div>
|
||
|
||
<div class="info-box" style="margin-top: 30px;">
|
||
<div class="info-box-title">
|
||
<span>⚠️</span>
|
||
<span>중요 안내</span>
|
||
</div>
|
||
<div class="info-box-text">
|
||
• WiFi 설정을 변경한 경우, ESP32를 재부팅해야 새 SSID/비밀번호가 적용됩니다.<br>
|
||
• <strong>APSTA 모드:</strong> Station 모드를 활성화하면 ESP32가 AP와 Station을 동시에 사용합니다.<br>
|
||
• Station 모드로 외부 WiFi에 연결하면 인터넷 접속이 가능해집니다.<br>
|
||
• Station 연결 실패 시에도 AP 모드는 정상 동작합니다.<br>
|
||
• 설정 저장 후 ESP32의 리셋 버튼을 눌러 재부팅하세요.
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<script>
|
||
let ws;
|
||
|
||
function initWebSocket() {
|
||
ws = new WebSocket('ws://' + window.location.hostname + ':81/');
|
||
|
||
ws.onopen = function() {
|
||
console.log('WebSocket connected');
|
||
loadSettings();
|
||
};
|
||
|
||
ws.onclose = function() {
|
||
console.log('WebSocket disconnected');
|
||
showAlert('alert-loading', '연결 끊김. 재연결 시도 중...', 'alert-warning');
|
||
setTimeout(initWebSocket, 3000);
|
||
};
|
||
|
||
ws.onerror = function(error) {
|
||
console.error('WebSocket error:', error);
|
||
};
|
||
|
||
ws.onmessage = function(event) {
|
||
try {
|
||
const data = JSON.parse(event.data);
|
||
|
||
if (data.type === 'settings') {
|
||
document.getElementById('wifi-ssid').value = data.ssid || 'Byun_CAN_Logger';
|
||
document.getElementById('wifi-password').value = data.password || '';
|
||
|
||
// STA 모드 설정 로드
|
||
document.getElementById('sta-enable').checked = data.staEnable || false;
|
||
document.getElementById('sta-ssid').value = data.staSSID || '';
|
||
document.getElementById('sta-password').value = data.staPassword || '';
|
||
|
||
// STA 설정 표시/숨김
|
||
toggleSTASettings();
|
||
|
||
// STA 연결 상태 표시
|
||
if (data.staConnected && data.staIP && data.staIP !== '0.0.0.0') {
|
||
document.getElementById('sta-ip').textContent = data.staIP;
|
||
document.getElementById('sta-status').style.display = 'block';
|
||
} else {
|
||
document.getElementById('sta-status').style.display = 'none';
|
||
}
|
||
|
||
hideAlert('alert-loading');
|
||
console.log('Settings loaded:', data);
|
||
} else if (data.type === 'settingsSaved') {
|
||
if (data.success) {
|
||
showAlert('alert-success', '설정이 저장되었습니다! 재부팅 후 적용됩니다.', 'alert-success');
|
||
}
|
||
}
|
||
} catch (e) {
|
||
console.error('Parse error:', e);
|
||
}
|
||
};
|
||
}
|
||
|
||
function loadSettings() {
|
||
if (ws && ws.readyState === WebSocket.OPEN) {
|
||
ws.send(JSON.stringify({cmd: 'getSettings'}));
|
||
showAlert('alert-loading', '설정을 불러오는 중...', 'alert-info');
|
||
}
|
||
}
|
||
|
||
function saveSettings() {
|
||
const ssid = document.getElementById('wifi-ssid').value.trim();
|
||
const password = document.getElementById('wifi-password').value;
|
||
|
||
// STA 모드 설정
|
||
const staEnable = document.getElementById('sta-enable').checked;
|
||
const staSSID = document.getElementById('sta-ssid').value.trim();
|
||
const staPassword = document.getElementById('sta-password').value;
|
||
|
||
// 입력 검증
|
||
if (ssid.length === 0) {
|
||
alert('WiFi SSID를 입력하세요.');
|
||
return;
|
||
}
|
||
|
||
if (ssid.length > 31) {
|
||
alert('WiFi SSID는 최대 31자까지 입력 가능합니다.');
|
||
return;
|
||
}
|
||
|
||
if (password.length > 0 && password.length < 8) {
|
||
alert('WiFi 비밀번호는 최소 8자 이상이어야 합니다.');
|
||
return;
|
||
}
|
||
|
||
if (password.length > 63) {
|
||
alert('WiFi 비밀번호는 최대 63자까지 입력 가능합니다.');
|
||
return;
|
||
}
|
||
|
||
// STA 모드 검증
|
||
if (staEnable) {
|
||
if (staSSID.length === 0) {
|
||
alert('Station 모드를 활성화하려면 WiFi SSID를 입력하세요.');
|
||
return;
|
||
}
|
||
if (staSSID.length > 31) {
|
||
alert('Station WiFi SSID는 최대 31자까지 입력 가능합니다.');
|
||
return;
|
||
}
|
||
if (staPassword.length > 0 && staPassword.length < 8) {
|
||
alert('Station WiFi 비밀번호는 최소 8자 이상이어야 합니다.');
|
||
return;
|
||
}
|
||
}
|
||
|
||
const settings = {
|
||
cmd: 'saveSettings',
|
||
ssid: ssid,
|
||
password: password,
|
||
staEnable: staEnable,
|
||
staSSID: staSSID,
|
||
staPassword: staPassword
|
||
};
|
||
|
||
if (ws && ws.readyState === WebSocket.OPEN) {
|
||
ws.send(JSON.stringify(settings));
|
||
console.log('Settings saved:', settings);
|
||
} else {
|
||
alert('WebSocket 연결이 끊겼습니다. 페이지를 새로고침하세요.');
|
||
}
|
||
}
|
||
|
||
function showAlert(alertId, message, className) {
|
||
const alert = document.getElementById(alertId);
|
||
if (alert) {
|
||
const textElement = alert.querySelector('.alert-text');
|
||
if (textElement && message) {
|
||
textElement.textContent = message;
|
||
}
|
||
|
||
// 기존 클래스 제거
|
||
alert.className = 'alert ' + className;
|
||
alert.classList.add('show');
|
||
|
||
// 3초 후 자동 숨김 (success 알림만)
|
||
if (className === 'alert-success') {
|
||
setTimeout(() => {
|
||
hideAlert(alertId);
|
||
}, 5000);
|
||
}
|
||
}
|
||
}
|
||
|
||
function hideAlert(alertId) {
|
||
const alert = document.getElementById(alertId);
|
||
if (alert) {
|
||
alert.classList.remove('show');
|
||
}
|
||
}
|
||
|
||
function toggleSTASettings() {
|
||
const staEnable = document.getElementById('sta-enable').checked;
|
||
const staSettings = document.getElementById('sta-settings');
|
||
|
||
if (staEnable) {
|
||
staSettings.classList.remove('disabled');
|
||
} else {
|
||
staSettings.classList.add('disabled');
|
||
}
|
||
}
|
||
|
||
// 페이지 로드 시 WebSocket 연결
|
||
window.addEventListener('load', function() {
|
||
initWebSocket();
|
||
});
|
||
</script>
|
||
</body>
|
||
</html>
|
||
)rawliteral";
|
||
|
||
#endif |