import { renderHeader } from '../components/header.js'; import { addAppLogEntry, authService, closeCurrentSessionAndSignOut, state } from '../state.js'; import { canInstallPwa, isStandalonePwaMode, onPwaInstallAvailabilityChange, promptPwaInstall, } from '../services/pwa-install-service.js'; import { initPwaPush } from '../services/pwa-push-service.js'; import { getArweaveWalletFromStoredDeviceKey } from '../services/arweave-wallet-service.js'; import { prepareAvatarImageFile, uploadArweaveFile, validateAvatarSourceFile, } from '../services/arweave-file-service.js'; export const pageMeta = { id: 'settings-view', title: 'Настройки' }; function formatBuildStamp(rawValue) { const value = String(rawValue || '').trim(); if (!/^\d{14}$/.test(value)) return value; const yyyy = value.slice(0, 4); const mm = value.slice(4, 6); const dd = value.slice(6, 8); const hh = value.slice(8, 10); const min = value.slice(10, 12); const ss = value.slice(12, 14); return `${yyyy}-${mm}-${dd}/${hh}:${min}__${ss}`; } function formatVersionForUi(rawValue) { const value = String(rawValue || '').trim(); if (!value) return 'n/a'; const formatted = formatBuildStamp(value); if (formatted && formatted !== value) { return `${formatted} (${value})`; } return value; } function clearArweaveJwk(walletCtx) { if (!walletCtx?.jwk || typeof walletCtx.jwk !== 'object') return; Object.keys(walletCtx.jwk).forEach((key) => { walletCtx.jwk[key] = ''; }); walletCtx.jwk = null; } function formatBytes(bytes) { const value = Number(bytes || 0); if (!Number.isFinite(value) || value <= 0) return '0 B'; if (value < 1024) return `${value} B`; if (value < 1024 * 1024) return `${(value / 1024).toFixed(1)} KB`; return `${(value / (1024 * 1024)).toFixed(2)} MB`; } function openDeveloperAvatarUploadModal({ walletLogin, storagePwd, gateway } = {}) { const root = document.getElementById('modal-root'); if (!root) return; root.innerHTML = ` `; const modal = root.querySelector('#settings-dev-avatar-modal'); const loginInput = root.querySelector('#settings-dev-avatar-login'); const fileInput = root.querySelector('#settings-dev-avatar-file'); const metaEl = root.querySelector('#settings-dev-avatar-meta'); const errorEl = root.querySelector('#settings-dev-avatar-error'); const cancelBtn = root.querySelector('#settings-dev-avatar-cancel'); const uploadBtn = root.querySelector('#settings-dev-avatar-upload'); const resultCard = root.querySelector('#settings-dev-avatar-result'); const txidEl = root.querySelector('#settings-dev-avatar-txid'); const copyBtn = root.querySelector('#settings-dev-avatar-copy'); if (loginInput instanceof HTMLInputElement) { loginInput.value = String(walletLogin || '').trim(); } let isClosed = false; const closeModal = () => { if (isClosed) return; isClosed = true; root.innerHTML = ''; }; const setError = (text) => { if (errorEl) errorEl.textContent = String(text || ''); }; const setMeta = (text) => { if (metaEl) metaEl.textContent = String(text || ''); }; const setTxId = (txId) => { if (!(resultCard instanceof HTMLElement) || !(txidEl instanceof HTMLElement)) return; const value = String(txId || '').trim(); txidEl.textContent = value; resultCard.hidden = !value; }; modal?.addEventListener('click', (event) => { if (event.target === modal) closeModal(); }); cancelBtn?.addEventListener('click', closeModal); copyBtn?.addEventListener('click', async () => { const txId = String(txidEl?.textContent || '').trim(); if (!txId) return; try { await navigator.clipboard.writeText(txId); setError(''); } catch { setError('Не удалось скопировать TX ID.'); } }); uploadBtn?.addEventListener('click', async () => { setError(''); setTxId(''); const targetLogin = String(loginInput?.value || '').trim(); const file = fileInput?.files?.[0] || null; if (!targetLogin) { setError('Введите логин пользователя.'); return; } if (!file) { setError('Выберите файл изображения.'); return; } if (!String(walletLogin || '').trim() || !String(storagePwd || '').trim()) { setError('Нет активной сессии. Войдите заново и повторите.'); return; } uploadBtn.disabled = true; let walletCtx = null; try { validateAvatarSourceFile(file); setMeta('Подготовка изображения...'); const optimized = await prepareAvatarImageFile(file); setMeta( `Файл подготовлен: ${formatBytes(optimized.originalSizeBytes)} → ${formatBytes(optimized.optimizedSizeBytes)} ` + `(${optimized.width}x${optimized.height}, ${optimized.contentType})`, ); walletCtx = await getArweaveWalletFromStoredDeviceKey({ login: walletLogin, storagePwd, onStatus: (message) => setMeta(message), }); setMeta('Загрузка в Arweave...'); const uploaded = await uploadArweaveFile({ gateway, jwk: walletCtx?.jwk, file: optimized.file, tags: [ { name: 'SHiNE-Profile-Login', value: targetLogin }, ], }); const txId = String(uploaded?.id || '').trim(); if (!txId) throw new Error('Пустой Transaction ID'); setMeta('Загрузка завершена.'); setTxId(txId); } catch (error) { const message = String(error?.message || ''); if ( message === 'Выберите файл изображения.' || message === 'Поддерживаются только JPEG, PNG или WebP.' || message === 'Файл слишком большой. Максимум 10 MB.' || message === 'Не удалось подготовить изображение.' ) { setError(message); } else { setError('Не удалось загрузить аватар в Arweave.'); } } finally { clearArweaveJwk(walletCtx); uploadBtn.disabled = false; } }); } export function render({ navigate }) { const screen = document.createElement('section'); screen.className = 'stack'; let isDisposed = false; screen.append( renderHeader({ title: 'Настройки', leftAction: { label: '←', onClick: () => navigate('profile-view') }, }), ); const card = document.createElement('div'); card.className = 'card stack'; card.innerHTML = ` `; card.querySelector('#settings-device').addEventListener('click', () => navigate('device-view')); card.querySelector('#settings-servers').addEventListener('click', () => navigate('server-settings-view')); card.querySelector('#settings-language').addEventListener('click', () => navigate('language-view')); const signOutBtn = card.querySelector('#settings-signout'); const developerCard = document.createElement('div'); developerCard.className = 'card stack settings-developer-card'; developerCard.innerHTML = ` `; const developerToggleBtn = developerCard.querySelector('#settings-dev-toggle'); const developerPanel = developerCard.querySelector('#settings-dev-panel'); const appLogBtn = developerCard.querySelector('#settings-app-log'); const diagnosticsBtn = developerCard.querySelector('#settings-pwa-diagnostics'); const pwaInstallBtn = developerCard.querySelector('#settings-pwa-install'); const uploadAvatarBtn = developerCard.querySelector('#settings-upload-avatar'); developerToggleBtn?.addEventListener('click', () => { const isHidden = developerPanel?.hidden !== false; if (developerPanel) developerPanel.hidden = !isHidden; if (developerToggleBtn) { developerToggleBtn.textContent = isHidden ? 'Для разработчиков ▲' : 'Для разработчиков ▼'; } }); appLogBtn?.addEventListener('click', () => navigate('app-log-view')); diagnosticsBtn?.addEventListener('click', () => navigate('pwa-diagnostics-view')); uploadAvatarBtn?.addEventListener('click', () => { openDeveloperAvatarUploadModal({ walletLogin: state.session.login, storagePwd: state.session.storagePwdInMemory, gateway: state.entrySettings.arweaveServer, }); }); const syncPwaButtonLabel = () => { if (isStandalonePwaMode()) { pwaInstallBtn.textContent = 'PWA установлено (проверить WebPush)'; return; } if (canInstallPwa()) { pwaInstallBtn.textContent = 'Зарегистрировать PWA'; return; } pwaInstallBtn.textContent = 'Как установить PWA'; }; const unsubscribeInstallAvailability = onPwaInstallAvailabilityChange(() => { syncPwaButtonLabel(); }); syncPwaButtonLabel(); signOutBtn.addEventListener('click', async () => { const confirmed = window.confirm( 'Завершить текущую сессию на сервере, отключиться, очистить локальные данные и перейти на стартовый экран?' ); if (!confirmed) return; signOutBtn.disabled = true; try { addAppLogEntry({ level: 'info', source: 'session', message: `Запрошено завершение текущей сессии: ${state.session.sessionId || 'unknown'}`, }); await closeCurrentSessionAndSignOut({ infoMessage: 'Сеанс завершён. Выполните вход заново.', }); } finally { signOutBtn.disabled = false; } }); pwaInstallBtn.addEventListener('click', async () => { pwaInstallBtn.disabled = true; try { await initPwaPush({ authService, onLog: (entry) => addAppLogEntry(entry), }); if (canInstallPwa()) { const result = await promptPwaInstall(); const accepted = result.outcome === 'accepted'; addAppLogEntry({ level: 'info', source: 'pwa-install', message: accepted ? 'Пользователь принял установку PWA' : 'Пользователь отклонил установку PWA', details: { outcome: result.outcome || 'unknown' }, }); if (accepted) { window.alert('Установка PWA подтверждена. Проверьте приложение на главном экране устройства.'); } } else if (!isStandalonePwaMode()) { window.alert('Для установки откройте меню браузера и выберите "Установить приложение" или "Добавить на главный экран".'); } else { window.alert('PWA уже установлено. WebPush перерегистрирован.'); } } catch (error) { addAppLogEntry({ level: 'warn', source: 'pwa-install', message: 'Не удалось зарегистрировать PWA/WebPush', details: { error: error?.message || 'unknown' }, }); window.alert(`Ошибка регистрации PWA: ${error?.message || 'unknown'}`); } finally { pwaInstallBtn.disabled = false; syncPwaButtonLabel(); } }); const versionCard = document.createElement('div'); versionCard.className = 'card stack'; const title = document.createElement('p'); title.className = 'field-label'; title.textContent = 'Версии'; const clientVersion = document.createElement('p'); clientVersion.className = 'meta-muted'; clientVersion.textContent = `Клиент: ${formatVersionForUi(window.__SHINE_CLIENT_VERSION__)}`; const uiBuild = document.createElement('p'); uiBuild.className = 'meta-muted'; uiBuild.textContent = `Сборка UI: ${formatVersionForUi(window.__SHINE_BUILD_HASH__)}`; const serverVersion = document.createElement('p'); serverVersion.className = 'meta-muted'; serverVersion.textContent = 'Сервер: загружается...'; versionCard.append(title, clientVersion, uiBuild, serverVersion); void (async () => { try { let value = ''; try { const pingResp = await authService.ws.request('Ping', { ts: Date.now() }, 7000); value = String(pingResp?.payload?.serverVersion || pingResp?.serverVersion || '').trim(); } catch { // fallback below } if (!value) { const infoResp = await authService.ws.request('GetServerInfo', {}); value = String(infoResp?.payload?.version || '').trim(); } if (!isDisposed) serverVersion.textContent = `Сервер: ${formatVersionForUi(value)}`; } catch { if (!isDisposed) { serverVersion.textContent = 'Сервер: недоступно'; } } })(); screen.append(card); screen.cleanup = () => { isDisposed = true; unsubscribeInstallAvailability(); }; screen.append(versionCard); screen.append(developerCard); return screen; }