import { renderHeader } from '../components/header.js'; import { authService, state } from '../state.js'; import { toUserMessage } from '../services/ui-error-texts.js'; import { channelNameErrorText, normalizeChannelDescription, normalizeChannelDisplayName, validateChannelDisplayName, } from '../services/channel-name-rules.js'; export const pageMeta = { id: 'add-channel-view', title: 'Создать канал' }; const CREATE_CHANNEL_FLASH_KEY = 'shine-channels-create-success'; const CHANNEL_TYPE_PUBLIC = 1; const CHANNEL_TYPE_PERSONAL = 100; const CHANNEL_TYPE_GROUP = 200; function persistCreateSuccessFlash(message) { try { sessionStorage.setItem(CREATE_CHANNEL_FLASH_KEY, String(message || '').trim()); } catch { // ignore storage errors } } function validateDescription(value) { const normalized = normalizeChannelDescription(value); const bytes = new TextEncoder().encode(normalized).length; if (bytes > 200) { return { ok: false, normalized, bytes, error: 'Описание слишком длинное: максимум 200 байт UTF-8.' }; } return { ok: true, normalized, bytes, error: '' }; } export function render({ navigate }) { const screen = document.createElement('section'); screen.className = 'stack channels-screen channels-screen--add'; screen.append( renderHeader({ title: 'Создать канал', leftAction: { label: '<', onClick: () => navigate('channels-list') }, }), ); const form = document.createElement('form'); form.className = 'card stack'; form.innerHTML = ` Создание канала

Можно использовать только латиницу, цифры, _ и -.

Длина названия: от 3 до 32 символов.

Публичный канал видят все. Писать может только владелец.
0 / 200 байт
`; const nameEl = form.querySelector('#channel-name'); const typeEl = form.querySelector('#channel-type'); const typeHintEl = form.querySelector('#channel-type-hint'); const descriptionEl = form.querySelector('#channel-description'); const nameErrorEl = form.querySelector('#channel-name-error'); const descriptionErrorEl = form.querySelector('#channel-description-error'); const descriptionCounterEl = form.querySelector('#channel-description-counter'); const errorEl = form.querySelector('#channel-create-error'); const submitEl = form.querySelector('#submit-create-channel'); const cancelEl = form.querySelector('#cancel-create-channel'); let submitInFlight = false; const setBusy = (busy) => { submitInFlight = !!busy; submitEl.disabled = submitInFlight; cancelEl.disabled = submitInFlight; nameEl.disabled = submitInFlight; typeEl.disabled = submitInFlight; descriptionEl.disabled = submitInFlight; submitEl.textContent = submitInFlight ? 'Создаём...' : 'Создать'; }; const updateTypeHint = () => { const typeCode = Number(typeEl.value || CHANNEL_TYPE_PUBLIC); if (typeCode === CHANNEL_TYPE_PERSONAL) { typeHintEl.textContent = 'Для персонального канала название должно быть login собеседника.'; nameEl.placeholder = 'Например: aidar'; return; } if (typeCode === CHANNEL_TYPE_GROUP) { typeHintEl.textContent = 'Для группового канала участников добавляют командами /.add и /.remove.'; nameEl.placeholder = 'Например: team_room'; return; } typeHintEl.textContent = 'Публичный канал видят все. Писать может только владелец.'; nameEl.placeholder = 'Например: my_channel-1'; }; const updateValidation = () => { const nameCheck = validateChannelDisplayName(nameEl.value); const descriptionCheck = validateDescription(descriptionEl.value); nameErrorEl.textContent = nameCheck.ok ? '' : channelNameErrorText(nameCheck.code); descriptionErrorEl.textContent = descriptionCheck.error; const descLength = Number(descriptionCheck.bytes || 0); descriptionCounterEl.textContent = `${descLength} / 200 байт`; const ok = nameCheck.ok && descriptionCheck.ok; submitEl.disabled = submitInFlight || !ok; return { ok, name: nameCheck.normalized, description: descriptionCheck.normalized, }; }; nameEl.addEventListener('input', updateValidation); descriptionEl.addEventListener('input', updateValidation); form.addEventListener('submit', async (event) => { event.preventDefault(); if (submitInFlight) return; const login = state.session.login; const storagePwd = state.session.storagePwdInMemory; if (!login || !storagePwd) { errorEl.textContent = 'Сессия недействительна. Выполните вход заново.'; return; } const check = updateValidation(); if (!check.ok) return; setBusy(true); errorEl.textContent = ''; try { const channelType = Number(typeEl.value || CHANNEL_TYPE_PUBLIC); if (channelType === CHANNEL_TYPE_PERSONAL) { const targetLogin = normalizeChannelDisplayName(check.name); const foundUser = await authService.getUser(targetLogin); if (!foundUser?.exists) { throw new Error('Логин для персонального канала не найден.'); } } await authService.addBlockCreateChannel({ login, storagePwd, channelName: normalizeChannelDisplayName(check.name), channelDescription: normalizeChannelDescription(check.description), channelType, channelTypeVersion: 1, }); persistCreateSuccessFlash(`Канал "${normalizeChannelDisplayName(check.name)}" создан.`); navigate('channels-list'); } catch (error) { errorEl.textContent = toUserMessage(error, 'Не удалось создать канал.'); setBusy(false); updateValidation(); } }); cancelEl.addEventListener('click', () => navigate('channels-list')); typeEl.addEventListener('change', updateTypeHint); screen.append(form); nameEl.focus(); updateTypeHint(); updateValidation(); return screen; }