import { render, html } from '/uhtml.js'; import loadMessages from '/message.js'; import loadSpace from '/space.js'; import loadCall from '/call.js'; function setVisibility() { document.getElementById('visibility').innerText = `${ window.currentThread.permissions.everyone.view.value ? 'this thread is visible to everyone' : 'members can view this thread'} ${window.currentThread.permissions.everyone.post.value ? 'anyone can post' : window.currentThread.permissions.members.post.value ? 'only members can post' : 'some members can post'}`; } function openThread(div, pushState) { if (!document.getElementById('removemember').classList.contains('hidden')) document.getElementById('member').classList.add('hidden'); if (!div) { document.getElementById('thread').classList.add('hidden'); window.currentThread = null; return; } document.getElementById('thread').classList.remove('hidden'); const edit = document.getElementById('edit'); if (div.thread.permissions.admin) { edit.classList.remove('hidden'); document.getElementById('membersadd').classList.remove('hidden'); } else { edit.classList.add('hidden'); document.getElementById('membersadd').classList.add('hidden'); } document.getElementById('threadname').textContent = div.thread.name; if (window.currentThread) window.currentThread.div.classList.remove('active'); div.classList.add('active'); window.currentThread = div.thread; window.currentInstance = div.thread.instance; window.currentInstance.emit('get_thread', { thread: div.thread.id }, async msg => { if (!msg.success) { console.log('get_thread failed:', msg.message); return; } Object.assign(div.thread, msg.thread); setVisibility(); document.getElementById('memberlist').replaceChildren( ...await Promise.all(msg.thread.members.map(async member => { let p = document.createElement('p'); if (member.name) p.append(window.makeUser(member, window.currentInstance.url, 'member')); else try { let user = await window.getUser(member.id); user.id = member.id; user.permissions = member.permissions; p.append(window.makeUser(user, user.id.split?.('@')[1], 'member')); } catch (e) { console.log(`error getting user ${member.id}: ${e}`); p.append(html.node`${member.id}`); } return p; })) ); loadCall(); if (div.tab) { switchTab(document.getElementById(div.tab)); if (div.tab === 'messagetab') loadMessages(); else if (div.tab === 'spacetab') loadSpace(); } else // load first tab that has any content await new Promise(resolve => { loadMessages(messages => { if (messages.length) { switchTab(document.getElementById(div.tab = 'messagetab')); resolve(); } else loadSpace(spans => { if (spans.length) switchTab(document.getElementById(div.tab = 'spacetab')); else if (Object.keys(currentThread.call).length || currentThread.streams.length) switchTab(document.getElementById(div.tab = 'calltab')); else switchTab(document.getElementById( div.tab = spans.length ? 'spacetab' : Object.keys(currentThread.call).length || currentThread.streams.length ? 'calltab' : 'messagetab')); resolve(); }); }); }); if (pushState) { let url = new URL(window.location); // put instance before thread and tab url.searchParams.delete('thread'); url.searchParams.delete('tab'); if (currentInstance === instancelist[0]) url.searchParams.delete('instance'); else url.searchParams.set('instance', currentInstance.url); url.searchParams.set('thread', div.thread.id); url.searchParams.set('tab', div.tab.substring(0, div.tab.length - 3)); window.history.pushState(null, '', url.toString()); } }); } window.onpopstate = event => { let params = new URLSearchParams(window.location.search); let thread = params.get('thread'); if (!thread) { openThread(); return; } let url = params.get('instance'); let div = document.querySelector( `#instance${(url ? instancelist.find(i => i.url === url) : instancelist[0])?.id} #thread${thread}`); if (!div) return; let tab = params.get('tab'); if (['message', 'space', 'call'].includes(tab)) div.tab = tab + 'tab'; openThread(div); }; function switchTab(tab) { document.querySelectorAll('.tab').forEach(tab => tab.classList.remove('active')); document.querySelectorAll('.tabcontent').forEach(tab => tab.classList.add('hidden')); tab.classList.add('active'); document.getElementById(tab.id.slice(0, -3)).classList.remove('hidden'); } function newThread(instance) { let form = document.getElementById('createthread'); if (form) { form.remove(); for (let create of document.querySelectorAll('#newthread')) { create.textContent = 'create'; create.classList.remove('hidden'); } return; } let members = [{ id: String(window.id), name: window.name, permissions: { admin: true } }]; if (instance !== window.instancelist[0]) { members[0].id += '@' + location.host; members[0].name += '@' + location.host; } async function addMember() { const name = document.getElementById('newmembername'); if (!name.value) return; let at = name.value.split('@'); let url = at[1] || instance.url; let error = document.getElementById('newmembererror'); let user = await window.getUser('@' + url, at[0]); if (!user) { error.innerText = 'user not found'; return; } error.innerText = ''; user.id = String(user.id); if (instance.url !== url) { user.id += '@' + url; user.name += '@' + url; } if (members.find(member => member.name === user.value)) return; members.push(user); document.getElementById('newmembers') .append(html.node`
${user.name}
`); name.value = ''; } document.querySelector('#instance + .separator').insertAdjacentElement('afterend', html.node` `); for (let create of document.querySelectorAll('#newthread')) { if (create === this) create.textContent = 'cancel'; else create.classList.add('hidden'); } } function editThread() { let form = document.getElementById('editthread'); if (form) { form.remove(); document.getElementById('edit').textContent = 'edit'; return; } form = html.node` `; form['editthreadname'].value = window.currentThread.name; if (window.currentThread.permissions.everyone.post.value) form['public'].checked = true; else if (window.currentThread.permissions.everyone.view.value) form['private_post'].checked = true; else form['private_view'].checked = true; document.getElementById('thread').append(form); document.getElementById('edit').textContent = 'cancel'; } function clickedTab(event) { switchTab(this); document.getElementById(`thread${window.currentThread.id}`).tab = this.id; if (this.id === 'messagetab') loadMessages(); else if (this.id === 'spacetab') loadSpace(); let url = new URL(location); url.searchParams.set('tab', this.id.substring(0, this.id.length - 3)); window.history.pushState(null, '', url.toString()); } async function addMember() { let name = document.getElementById('addmembername'); let error = document.getElementById('membererror'); function close() { error.innerText = ''; name.value = ''; document.getElementById('addmember').classList.add('hidden'); } if (!name.value) return close(); let at = name.value.split('@'); let url = at[1] || window.currentInstance.url; let user = await window.getUser('@' + url, at[0]); if (!user) { error.innerText = 'user not found'; return; } error.innerText = ''; if (window.currentInstance.url !== url) user.id += '@' + url; for (let p of document.getElementById('memberlist').children) if (p.children[0].user.id == user.id) return close(); window.currentInstance.emit('add_member', { id: String(user.id), thread: window.currentThread.id }, msg => { if (!msg.success) { console.log('add_member failed:', msg.message); error.textContent = 'error: ' + msg.message; return; } close(); let p = document.createElement('p'); p.append(window.makeUser(user, url, 'member')); document.getElementById('memberlist').append(p); }); } async function loadThreads(instancediv, select) { function makeThread(thread) { thread.instance = instancediv.instance; let node = html.node`