import initMatrix from '../client/initMatrix'; import HashIC from '../../public/res/ic/outlined/hash.svg'; import HashGlobeIC from '../../public/res/ic/outlined/hash-globe.svg'; import HashLockIC from '../../public/res/ic/outlined/hash-lock.svg'; import SpaceIC from '../../public/res/ic/outlined/space.svg'; import SpaceGlobeIC from '../../public/res/ic/outlined/space-globe.svg'; import SpaceLockIC from '../../public/res/ic/outlined/space-lock.svg'; const WELL_KNOWN_URI = '/.well-known/matrix/client'; export async function getBaseUrl(servername) { let protocol = 'https://'; if (servername.match(/^https?:\/\//) !== null) protocol = ''; const serverDiscoveryUrl = `${protocol}${servername}${WELL_KNOWN_URI}`; try { const result = await (await fetch(serverDiscoveryUrl, { method: 'GET' })).json(); const baseUrl = result?.['m.homeserver']?.base_url; if (baseUrl === undefined) throw new Error(); return baseUrl; } catch (e) { throw new Error(`${protocol}${servername}`); } } export function getUsername(userId) { const mx = initMatrix.matrixClient; const user = mx.getUser(userId); if (user === null) return userId; let username = user.displayName; if (typeof username === 'undefined') { username = userId; } return username; } export function getUsernameOfRoomMember(roomMember) { return roomMember.name || roomMember.userId; } export async function isRoomAliasAvailable(alias) { try { const result = await initMatrix.matrixClient.resolveRoomAlias(alias); if (result.room_id) return false; return false; } catch (e) { if (e.errcode === 'M_NOT_FOUND') return true; return false; } } export function getPowerLabel(powerLevel) { if (powerLevel > 9000) return 'Goku'; if (powerLevel > 100) return 'Founder'; if (powerLevel === 100) return 'Admin'; if (powerLevel >= 50) return 'Mod'; return null; } export function parseReply(rawBody) { if (rawBody?.indexOf('>') !== 0) return null; let body = rawBody.slice(rawBody.indexOf('<') + 1); const user = body.slice(0, body.indexOf('>')); body = body.slice(body.indexOf('>') + 2); const replyBody = body.slice(0, body.indexOf('\n\n')); body = body.slice(body.indexOf('\n\n') + 2); if (user === '') return null; const isUserId = user.match(/^@.+:.+/); return { userId: isUserId ? user : null, displayName: isUserId ? null : user, replyBody, body, }; } export function hasDMWith(userId) { const mx = initMatrix.matrixClient; const directIds = [...initMatrix.roomList.directs]; return directIds.find((roomId) => { const dRoom = mx.getRoom(roomId); const roomMembers = dRoom.getMembers(); if (roomMembers.length <= 2 && dRoom.getMember(userId)) { return true; } return false; }); } export function joinRuleToIconSrc(joinRule, isSpace) { return ({ restricted: () => (isSpace ? SpaceIC : HashIC), invite: () => (isSpace ? SpaceLockIC : HashLockIC), public: () => (isSpace ? SpaceGlobeIC : HashGlobeIC), }[joinRule]?.() || null); } // NOTE: it gives userId with minimum power level 50; function getHighestPowerUserId(room) { const userIdToPower = room.currentState.getStateEvents('m.room.power_levels', '')?.getContent().users; let powerUserId = null; if (!userIdToPower) return powerUserId; Object.keys(userIdToPower).forEach((userId) => { if (userIdToPower[userId] < 50) return; if (powerUserId === null) { powerUserId = userId; return; } if (userIdToPower[userId] > userIdToPower[powerUserId]) { powerUserId = userId; } }); return powerUserId; } export function getIdServer(userId) { const idParts = userId.split(':'); return idParts[1]; } export function getServerToPopulation(room) { const members = room.getMembers(); const serverToPop = {}; members?.forEach((member) => { const { userId } = member; const server = getIdServer(userId); const serverPop = serverToPop[server]; if (serverPop === undefined) { serverToPop[server] = 1; return; } serverToPop[server] = serverPop + 1; }); return serverToPop; } export function genRoomVia(room) { const via = []; const userId = getHighestPowerUserId(room); if (userId) { const server = getIdServer(userId); if (server) via.push(server); } const serverToPop = getServerToPopulation(room); const sortedServers = Object.keys(serverToPop).sort( (svrA, svrB) => serverToPop[svrB] - serverToPop[svrA], ); const mostPop3 = sortedServers.slice(0, 3); if (via.length === 0) return mostPop3; if (mostPop3.includes(via[0])) { mostPop3.splice(mostPop3.indexOf(via[0]), 1); } return via.concat(mostPop3.slice(0, 2)); }