combined guild mostly implemented
This commit is contained in:
parent
64490d027f
commit
87a4d8584f
@ -35,7 +35,7 @@ electronMain.initialize();
|
||||
webPreferences: {
|
||||
//@ts-ignore enableRemoteModule is enabled with @electron/remote and not included in electron's typing
|
||||
enableRemoteModule: true, // so we can get console logs properly
|
||||
preload: path.join(__dirname, 'webapp', 'script.js')
|
||||
preload: path.join(__dirname, 'webapp', 'entrypoint.js')
|
||||
}
|
||||
});
|
||||
|
||||
|
@ -1,14 +1,14 @@
|
||||
import ElementsUtil from './require/elements-util';
|
||||
import BaseElements from './require/base-elements';
|
||||
import ClientController from '../client-controller';
|
||||
import { Channel } from '../data-types';
|
||||
import createModifyChannelOverlay from './overlay-modify-channel';
|
||||
import UI from '../ui';
|
||||
import Actions from '../actions';
|
||||
import Q from '../q-module';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createChannel(document: Document, q: Q, ui: UI, guild: CombinedGuild, channel: Channel) {
|
||||
let element = q.create({ class: 'channel text', 'meta-id': channel.id, 'meta-server-id': server.id, content: [
|
||||
let element = q.create({ class: 'channel text', 'meta-id': channel.id, 'meta-server-id': guild.id, content: [
|
||||
// Scraped directly from discord (#)
|
||||
{ class: 'icon', content: BaseElements.TEXT_CHANNEL_ICON },
|
||||
{ class: 'name', content: channel.name },
|
||||
@ -17,8 +17,8 @@ export default function createChannel(document: Document, q: Q, ui: UI, guild: C
|
||||
|
||||
element.addEventListener('click', async () => {
|
||||
if (element.classList.contains('active')) return;
|
||||
await ui.setActiveChannel(server, channel);
|
||||
await Actions.fetchAndUpdateMessagesRecent(q, ui, server, channel);
|
||||
await ui.setActiveChannel(guild, channel);
|
||||
await Actions.fetchAndUpdateMessagesRecent(q, ui, guild, channel);
|
||||
q.$('#text-input').focus();
|
||||
});
|
||||
|
||||
@ -34,7 +34,7 @@ export default function createChannel(document: Document, q: Q, ui: UI, guild: C
|
||||
if (modifyContextElement.parentElement) {
|
||||
modifyContextElement.parentElement.removeChild(modifyContextElement);
|
||||
}
|
||||
let modifyOverlay = createModifyChannelOverlay(document, q, server, channel);
|
||||
let modifyOverlay = createModifyChannelOverlay(document, q, guild, channel);
|
||||
document.body.appendChild(modifyOverlay);
|
||||
q.$$$(modifyOverlay, '.text-input.channel-name').focus();
|
||||
ElementsUtil.setCursorToEnd(q.$$$(modifyOverlay, '.text-input.channel-name'));
|
||||
|
@ -1,10 +1,10 @@
|
||||
import ElementsUtil from './require/elements-util.js';
|
||||
import BaseElements from './require/base-elements.js';
|
||||
|
||||
import ClientController from '../client-controller';
|
||||
import createPersonalizeOverlay from './overlay-personalize.js';
|
||||
import Q from '../q-module.js';
|
||||
import UI from '../ui.js';
|
||||
import CombinedGuild from '../guild-combined.js';
|
||||
|
||||
export default function createConnectionContextMenu(document: Document, q: Q, ui: UI, guild: CombinedGuild) {
|
||||
let statuses = [ 'online', 'away', 'busy', 'invisible' ];
|
||||
@ -26,7 +26,7 @@ export default function createConnectionContextMenu(document: Document, q: Q, ui
|
||||
q.$$$(element, '.personalize').addEventListener('click', async () => {
|
||||
element.removeSelf();
|
||||
if (ui.activeConnection === null) return;
|
||||
let overlayElement = createPersonalizeOverlay(document, q, server, ui.activeConnection);
|
||||
let overlayElement = createPersonalizeOverlay(document, q, guild, ui.activeConnection);
|
||||
document.body.appendChild(overlayElement);
|
||||
q.$$$(overlayElement, '.text-input').focus();
|
||||
ElementsUtil.setCursorToEnd(q.$$$(overlayElement, '.text-input'));
|
||||
@ -35,9 +35,9 @@ export default function createConnectionContextMenu(document: Document, q: Q, ui
|
||||
for (let status of statuses) {
|
||||
q.$$$(element, '.' + status).addEventListener('click', async () => {
|
||||
element.removeSelf();
|
||||
let currentConnection = await server.fetchConnectionInfo();
|
||||
let currentConnection = await guild.fetchConnectionInfo();
|
||||
if (status != currentConnection.status) {
|
||||
await server.setStatus(status);
|
||||
await guild.requestSetStatus(status);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
@ -5,29 +5,29 @@ const LOG = Logger.create(__filename, electronConsole);
|
||||
|
||||
import ElementsUtil from './require/elements-util.js';
|
||||
import BaseElements from './require/base-elements.js';
|
||||
import ClientController from '../client-controller';
|
||||
import { CacheServerData, ServerMetaData } from '../data-types';
|
||||
import { GuildMetadata } from '../data-types';
|
||||
import Q from '../q-module';
|
||||
import UI from '../ui';
|
||||
|
||||
import createErrorMessageOverlay from './overlay-error-message';
|
||||
import createServerSettingsOverlay from './overlay-server-settings';
|
||||
import createGuildSettingsOverlay from './overlay-guild-settings';
|
||||
import createCreateInviteTokenOverlay from './overlay-create-invite-token';
|
||||
import createCreateChannelOverlay from './overlay-create-channel';
|
||||
import createTokenLogOverlay from './overlay-token-log';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createServerTitleContextMenu(document: Document, q: Q, ui: UI, guild: CombinedGuild): HTMLElement {
|
||||
export default function createGuildTitleContextMenu(document: Document, q: Q, ui: UI, guild: CombinedGuild): HTMLElement {
|
||||
if (ui.activeConnection === null) {
|
||||
LOG.warn('no active connection when creating server title context menu');
|
||||
LOG.warn('no active connection when creating guild title context menu');
|
||||
return q.create({}) as HTMLElement;
|
||||
}
|
||||
|
||||
let menuItems: any[] = [];
|
||||
|
||||
if (ui.activeConnection.privileges.includes('modify_profile')) {
|
||||
menuItems.push({ class: 'item server-settings', content: [
|
||||
menuItems.push({ class: 'item guild-settings', content: [
|
||||
{ class: 'icon', content: BaseElements.COG },
|
||||
'Server Settings'
|
||||
'Guild Settings'
|
||||
] });
|
||||
}
|
||||
|
||||
@ -59,23 +59,23 @@ export default function createServerTitleContextMenu(document: Document, q: Q, u
|
||||
}
|
||||
|
||||
let element = BaseElements.createContextMenu(document, {
|
||||
class: 'server-title-context', content: menuItems
|
||||
class: 'guild-title-context', content: menuItems
|
||||
});
|
||||
|
||||
if (ui.activeConnection.privileges.includes('modify_profile')) {
|
||||
q.$$$(element, '.item.server-settings').addEventListener('click', async () => {
|
||||
q.$$$(element, '.item.guild-settings').addEventListener('click', async () => {
|
||||
element.removeSelf();
|
||||
let serverMeta: ServerMetaData | CacheServerData | null = null;
|
||||
let guildMeta: GuildMetadata | null = null;
|
||||
try {
|
||||
serverMeta = await server.grabMetadata();
|
||||
guildMeta = await guild.fetchMetadata();
|
||||
} catch (e) {
|
||||
LOG.error('error fetching server info', e);
|
||||
LOG.error('error fetching guild info', e);
|
||||
}
|
||||
if (serverMeta === null) {
|
||||
let overlay = createErrorMessageOverlay(document, 'Error Opening Settings', 'Could not load server information');
|
||||
if (guildMeta === null) {
|
||||
let overlay = createErrorMessageOverlay(document, 'Error Opening Settings', 'Could not load guild information');
|
||||
document.body.appendChild(overlay);
|
||||
} else {
|
||||
let overlay = createServerSettingsOverlay(document, q, server, serverMeta);
|
||||
let overlay = createGuildSettingsOverlay(document, q, guild, guildMeta);
|
||||
document.body.appendChild(overlay);
|
||||
q.$$$(overlay, '.text-input').focus();
|
||||
ElementsUtil.setCursorToEnd(q.$$$(overlay, '.text-input'));
|
||||
@ -86,7 +86,7 @@ export default function createServerTitleContextMenu(document: Document, q: Q, u
|
||||
if (ui.activeConnection.privileges.includes('modify_channels')) {
|
||||
q.$$$(element, '.item.create-channel').addEventListener('click', () => {
|
||||
element.removeSelf();
|
||||
let overlay = createCreateChannelOverlay(document, q, server);
|
||||
let overlay = createCreateChannelOverlay(document, q, guild);
|
||||
document.body.appendChild(overlay);
|
||||
q.$$$(overlay, '.text-input.channel-name').focus();
|
||||
ElementsUtil.setCursorToEnd(q.$$$(overlay, '.text-input.channel-name'));
|
||||
@ -96,13 +96,13 @@ export default function createServerTitleContextMenu(document: Document, q: Q, u
|
||||
if (ui.activeConnection.privileges.includes('modify_members')) {
|
||||
q.$$$(element, '.item.create-invite-token').addEventListener('click', () => {
|
||||
element.removeSelf();
|
||||
let overlay = createCreateInviteTokenOverlay(document, server);
|
||||
let overlay = createCreateInviteTokenOverlay(document, guild);
|
||||
document.body.appendChild(overlay);
|
||||
//LOG.info('create invite token clicked');
|
||||
});
|
||||
q.$$$(element, '.item.token-log').addEventListener('click', () => {
|
||||
element.removeSelf();
|
||||
let overlay = createTokenLogOverlay(document, q, server);
|
||||
let overlay = createTokenLogOverlay(document, q, guild);
|
||||
document.body.appendChild(overlay);
|
||||
});
|
||||
}
|
@ -5,12 +5,12 @@ const LOG = Logger.create(__filename, electronConsole);
|
||||
|
||||
import BaseElements from './require/base-elements.js';
|
||||
|
||||
import ClientController from '../client-controller.js';
|
||||
import Q from '../q-module';
|
||||
import UI from '../ui';
|
||||
import Controller from '../controller';
|
||||
import GuildsManager from '../guilds-manager';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createServerContextMenu(document: Document, q: Q, ui: UI, controller: Controller, guild: CombinedGuild) {
|
||||
export default function createServerContextMenu(document: Document, q: Q, ui: UI, guildsManager: GuildsManager, guild: CombinedGuild) {
|
||||
let element = BaseElements.createContextMenu(document, {
|
||||
class: 'server-context', content: [
|
||||
{ class: 'item red leave-server', content: 'Leave Server' }
|
||||
@ -19,9 +19,9 @@ export default function createServerContextMenu(document: Document, q: Q, ui: UI
|
||||
|
||||
q.$$$(element, '.leave-server').addEventListener('click', async () => {
|
||||
element.removeSelf();
|
||||
await server.disconnect();
|
||||
await controller.removeServer(server);
|
||||
await ui.removeServer(server);
|
||||
guild.disconnect();
|
||||
await guildsManager.removeServer(guild);
|
||||
await ui.removeGuild(guild);
|
||||
let firstServerElement = q.$_('#server-list .server');
|
||||
if (firstServerElement) {
|
||||
firstServerElement.click();
|
@ -6,8 +6,8 @@ import * as electron from 'electron';
|
||||
import BaseElements from './require/base-elements';
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createImageContextMenu(
|
||||
document: Document,
|
||||
@ -28,7 +28,7 @@ export default function createImageContextMenu(
|
||||
q.$$$(contextMenu, '.copy-image').innerText = 'Copying...';
|
||||
let nativeImage: electron.NativeImage;
|
||||
if (mime != 'image/png' && mime != 'image/jpeg' && mime != 'image/jpg') {
|
||||
// use sharp to convert to png since nativeImage only supports jpeg/png
|
||||
// use sharp to convserver: serverrt to png since nativeImage only supports jpeg/png
|
||||
nativeImage = electron.nativeImage.createFromBuffer(await sharp(buffer).png().toBuffer());
|
||||
} else {
|
||||
nativeImage = electron.nativeImage.createFromBuffer(buffer);
|
||||
@ -37,7 +37,7 @@ export default function createImageContextMenu(
|
||||
q.$$$(contextMenu, '.copy-image').innerText = 'Copied to Clipboard';
|
||||
});
|
||||
q.$$$(contextMenu, '.save-image').addEventListener('click', ElementsUtil.createDownloadListener({
|
||||
downloadBuff: buffer, server: server,
|
||||
downloadBuff: buffer, guild: guild,
|
||||
resourceName: path.basename(resourceName, '.' + ext) + (isPreview ? '-preview.' : '.') + ext,
|
||||
downloadStartFunc: () => {},
|
||||
writeStartFunc: () => { q.$$$(contextMenu, '.save-image').innerText = 'Writing...'; },
|
||||
|
@ -7,25 +7,24 @@ import * as fs from 'fs/promises';
|
||||
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
import createAddServerOverlay, { IAddServerData } from './overlay-add-server';
|
||||
import createAddGuildOverlay, { IAddGuildData } from './overlay-add-guild';
|
||||
import Q from '../q-module';
|
||||
import UI from '../ui';
|
||||
import Controller from '../controller';
|
||||
import GuildsManager from '../guilds-manager';
|
||||
import createErrorMessageOverlay from './overlay-error-message';
|
||||
import Actions from '../actions';
|
||||
|
||||
export default function bindAddServerEvents(document: Document, q: Q, ui: UI, controller: Controller): void {
|
||||
export default function bindAddGuildEvents(document: Document, q: Q, ui: UI, guildsManager: GuildsManager): void {
|
||||
let choosingFile = false;
|
||||
q.$('#add-server').addEventListener('click', async () => {
|
||||
q.$('#add-guild').addEventListener('click', async () => {
|
||||
if (choosingFile) return;
|
||||
choosingFile = true;
|
||||
|
||||
let result = await electronRemote.dialog.showOpenDialog({
|
||||
title: 'Select Server File',
|
||||
title: 'Select Guild File',
|
||||
defaultPath: '.', // TODO: better path name
|
||||
properties: [ 'openFile' ],
|
||||
filters: [
|
||||
{ name: 'Cordis Server Files', extensions: [ 'cordis' ] }
|
||||
{ name: 'Cordis Guild Files', extensions: [ 'cordis' ] }
|
||||
]
|
||||
});
|
||||
|
||||
@ -37,26 +36,26 @@ export default function bindAddServerEvents(document: Document, q: Q, ui: UI, co
|
||||
let filePath = result.filePaths[0];
|
||||
let fileText = (await fs.readFile(filePath)).toString('utf-8'); // TODO: try/catch?
|
||||
|
||||
let addServerData: any | null = null;
|
||||
let addGuildData: any | null = null;
|
||||
try {
|
||||
addServerData = JSON.parse(fileText);
|
||||
addGuildData = JSON.parse(fileText);
|
||||
if (
|
||||
typeof addServerData !== 'object' ||
|
||||
typeof addServerData?.name !== 'string' ||
|
||||
typeof addServerData?.url !== 'string' ||
|
||||
typeof addServerData?.cert !== 'string' ||
|
||||
typeof addServerData?.token !== 'string' ||
|
||||
typeof addServerData?.expires !== 'number' ||
|
||||
typeof addServerData?.iconSrc !== 'string'
|
||||
typeof addGuildData !== 'object' ||
|
||||
typeof addGuildData?.name !== 'string' ||
|
||||
typeof addGuildData?.url !== 'string' ||
|
||||
typeof addGuildData?.cert !== 'string' ||
|
||||
typeof addGuildData?.token !== 'string' ||
|
||||
typeof addGuildData?.expires !== 'number' ||
|
||||
typeof addGuildData?.iconSrc !== 'string'
|
||||
) {
|
||||
LOG.debug('bad server data:', { addServerData, fileText })
|
||||
throw new Error('bad server data');
|
||||
LOG.debug('bad guild data:', { addGuildData, fileText })
|
||||
throw new Error('bad guild data');
|
||||
}
|
||||
let overlayElement = createAddServerOverlay(document, q, ui, controller, addServerData as IAddServerData);
|
||||
let overlayElement = createAddGuildOverlay(document, q, ui, guildsManager, addGuildData as IAddGuildData);
|
||||
document.body.appendChild(overlayElement);
|
||||
} catch (e) {
|
||||
LOG.error('Unable to parse server data', e);
|
||||
let errorOverlayElement = createErrorMessageOverlay(document, 'Unable to parse server file', e.message);
|
||||
LOG.error('Unable to parse guild data', e);
|
||||
let errorOverlayElement = createErrorMessageOverlay(document, 'Unable to parse guild file', e.message);
|
||||
document.body.appendChild(errorOverlayElement);
|
||||
}
|
||||
|
||||
@ -72,14 +71,14 @@ export default function bindAddServerEvents(document: Document, q: Q, ui: UI, co
|
||||
'L 8,0 ' +
|
||||
'Z' }
|
||||
] },
|
||||
{ class: 'content', content: 'Add a Server' }
|
||||
{ class: 'content', content: 'Add a Guild' }
|
||||
]
|
||||
} }) as HTMLElement;
|
||||
q.$('#add-server').addEventListener('mouseenter', () => {
|
||||
q.$('#add-guild').addEventListener('mouseenter', () => {
|
||||
document.body.appendChild(contextElement);
|
||||
ElementsUtil.alignContextElement(contextElement, q.$('#add-server'), { left: 'right', centerY: 'centerY' })
|
||||
ElementsUtil.alignContextElement(contextElement, q.$('#add-guild'), { left: 'right', centerY: 'centerY' })
|
||||
});
|
||||
q.$('#add-server').addEventListener('mouseleave', () => {
|
||||
q.$('#add-guild').addEventListener('mouseleave', () => {
|
||||
if (contextElement.parentElement) {
|
||||
contextElement.parentElement.removeChild(contextElement);
|
||||
}
|
@ -7,7 +7,7 @@ import createConnectionContextMenu from './context-menu-conn';
|
||||
export default function bindConnectionEvents(document: Document, q: Q, ui: UI): void {
|
||||
q.$('#connection').addEventListener('click', () => {
|
||||
if (ui.activeGuild === null) return;
|
||||
if (!ui.activeGuild.isVerified) return;
|
||||
if (!ui.activeGuild.isSocketVerified()) return;
|
||||
|
||||
let contextMenu = createConnectionContextMenu(document, q, ui, ui.activeGuild);
|
||||
document.body.appendChild(contextMenu);
|
||||
|
@ -1,13 +1,13 @@
|
||||
import Q from '../q-module';
|
||||
import UI from '../ui';
|
||||
import createServerTitleContextMenu from './context-menu-srv-title';
|
||||
import createServerTitleContextMenu from './context-menu-guild-title';
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
export default function bindAddServerTitleEvents(document: Document, q: Q, ui: UI) {
|
||||
q.$('#server-name-container').addEventListener('click', () => {
|
||||
if (ui.activeConnection === null) return;
|
||||
if (ui.activeGuild === null) return;
|
||||
if (!ui.activeGuild.isVerified) return;
|
||||
if (!ui.activeGuild.isSocketVerified()) return;
|
||||
if (
|
||||
!ui.activeConnection.privileges.includes('modify_profile') &&
|
||||
!ui.activeConnection.privileges.includes('modify_members')
|
@ -7,12 +7,12 @@ import ElementsUtil from './require/elements-util.js';
|
||||
import Globals from '../globals';
|
||||
|
||||
import { Channel } from '../data-types';
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import UI from '../ui';
|
||||
import createUploadOverlayFromPath from './overlay-upload-path';
|
||||
import createUploadOverlayFromDataTransferItem from './overlay-upload-datatransfer';
|
||||
import createUploadDropTarget from './overlay-upload-drop-target';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function bindTextInputEvents(document: Document, q: Q, ui: UI): void {
|
||||
// Send Current Channel Messages
|
||||
@ -27,10 +27,10 @@ export default function bindTextInputEvents(document: Document, q: Q, ui: UI): v
|
||||
|
||||
sendingMessage = true;
|
||||
|
||||
let server = ui.activeGuild as ClientController;
|
||||
let server = ui.activeGuild as CombinedGuild;
|
||||
let channel = ui.activeChannel as Channel;
|
||||
|
||||
if (!server.isVerified) {
|
||||
if (!server.isSocketVerified()) {
|
||||
LOG.warn('client attempted to send message while not verified');
|
||||
q.$('#send-error').innerText = 'Not Connected to Server';
|
||||
await ElementsUtil.shakeElement(q.$('#send-error'), 400);
|
||||
@ -50,7 +50,7 @@ export default function bindTextInputEvents(document: Document, q: Q, ui: UI): v
|
||||
q.$('#text-input').removeAttribute('contenteditable');
|
||||
q.$('#text-input').classList.add('sending');
|
||||
try {
|
||||
await server.sendMessage(channel.id, text);
|
||||
await server.requestSendMessage(channel.id, text);
|
||||
|
||||
q.$('#send-error').innerText = '';
|
||||
q.$('#text-input').innerText = '';
|
||||
|
@ -10,11 +10,11 @@ import { GuildMetadata } from '../data-types';
|
||||
import Q from '../q-module';
|
||||
import UI from '../ui';
|
||||
import Actions from '../actions';
|
||||
import createGuildContextMenu from './context-menu-srv';
|
||||
import Controller from '../controller';
|
||||
import createGuildContextMenu from './context-menu-guild';
|
||||
import GuildsManager from '../guilds-manager';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createGuildListGuild(document: Document, q: Q, ui: UI, controller: Controller, guild: CombinedGuild) {
|
||||
export default function createGuildListGuild(document: Document, q: Q, ui: UI, guildsManager: GuildsManager, guild: CombinedGuild) {
|
||||
let element = q.create({ class: 'guild', 'meta-id': guild.id, 'meta-name': guild.id, content: [
|
||||
{ class: 'pill' },
|
||||
{ tag: 'img', src: './img/loading.svg', alt: 'guild' }, // src is set later by script.js
|
||||
@ -24,10 +24,10 @@ export default function createGuildListGuild(document: Document, q: Q, ui: UI, c
|
||||
(async () => {
|
||||
let guildData: GuildMetadata;
|
||||
try {
|
||||
guildData = await guild.grabMetadata();
|
||||
guildData = await guild.fetchMetadata();
|
||||
if (!guildData.iconResourceId) throw new Error('guild icon not identified yet');
|
||||
let guildIcon = await guild.fetchResource(guildData.iconResourceId);
|
||||
let guildIconSrc = await ElementsUtil.getImageBufferSrc(guildIcon);
|
||||
let guildIconSrc = await ElementsUtil.getImageBufferSrc(guildIcon.data);
|
||||
(q.$$$(element, 'img') as HTMLImageElement).src = guildIconSrc;
|
||||
} catch (e) {
|
||||
LOG.error('Error fetching guild icon', e);
|
||||
@ -83,7 +83,7 @@ export default function createGuildListGuild(document: Document, q: Q, ui: UI, c
|
||||
(async () => {
|
||||
// Explicitly not using a withPotentialError to make this simpler
|
||||
try {
|
||||
let guildData = await guild.grabMetadata();
|
||||
let guildData = await guild.fetchMetadata();
|
||||
ui.updateGuildName(guild, guildData.name);
|
||||
} catch (e) {
|
||||
LOG.error('Error fetching guild name', e);
|
||||
@ -103,7 +103,7 @@ export default function createGuildListGuild(document: Document, q: Q, ui: UI, c
|
||||
});
|
||||
|
||||
element.addEventListener('contextmenu', (e) => {
|
||||
let contextMenu = createGuildContextMenu(document, q, ui, controller, guild);
|
||||
let contextMenu = createGuildContextMenu(document, q, ui, guildsManager, guild);
|
||||
document.body.appendChild(contextMenu);
|
||||
let relativeTo = { x: e.pageX, y: e.pageY };
|
||||
ElementsUtil.alignContextElement(contextMenu, relativeTo, { top: 'centerY', left: 'centerX' });
|
||||
|
@ -1,5 +1,5 @@
|
||||
import ClientController from "../client-controller";
|
||||
import { Member } from "../data-types";
|
||||
import CombinedGuild from "../guild-combined";
|
||||
import Q from "../q-module";
|
||||
|
||||
import ElementsUtil from "./require/elements-util";
|
||||
@ -18,7 +18,7 @@ export default function createMember(q: Q, guild: CombinedGuild, member: Member)
|
||||
] }) as HTMLElement;
|
||||
(async () => {
|
||||
(q.$$$(element, 'img.avatar') as HTMLImageElement).src =
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(server, member.avatarResourceId);
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, member.avatarResourceId);
|
||||
})();
|
||||
return element;
|
||||
}
|
||||
|
@ -1,5 +1,5 @@
|
||||
import ClientController from '../client-controller';
|
||||
import { Message } from '../data-types';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
import Q from '../q-module';
|
||||
import createImageResourceMessage from './msg-img-res';
|
||||
import createImageResourceMessageContinued from './msg-img-res-cont';
|
||||
@ -13,22 +13,22 @@ export default function createMessage(document: Document, q: Q, guild: CombinedG
|
||||
if (message.hasResource()) {
|
||||
if (message.isImageResource()) {
|
||||
if (message.isContinued(lastMessage)) {
|
||||
element = createImageResourceMessageContinued(document, q, server, message);
|
||||
element = createImageResourceMessageContinued(document, q, guild, message);
|
||||
} else {
|
||||
element = createImageResourceMessage(document, q, server, message);
|
||||
element = createImageResourceMessage(document, q, guild, message);
|
||||
}
|
||||
} else {
|
||||
if (message.isContinued(lastMessage)) {
|
||||
element = createResourceMessageContinued(q, server, message);
|
||||
element = createResourceMessageContinued(q, guild, message);
|
||||
} else {
|
||||
element = createResourceMessage(q, server, message);
|
||||
element = createResourceMessage(q, guild, message);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (message.isContinued(lastMessage)) {
|
||||
element = createTextMessageContinued(q, server, message);
|
||||
element = createTextMessageContinued(q, guild, message);
|
||||
} else {
|
||||
element = createTextMessage(q, server, message);
|
||||
element = createTextMessage(q, guild, message);
|
||||
}
|
||||
}
|
||||
return element;
|
||||
|
@ -9,17 +9,17 @@ import * as FileType from 'file-type';
|
||||
import ElementsUtil from './require/elements-util.js';
|
||||
|
||||
import { Message, ShouldNeverHappenError } from '../data-types';
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import createImageOverlay from './overlay-image';
|
||||
import createImageContextMenu from './context-menu-img';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createImageResourceMessageContinued(document: Document, q: Q, guild: CombinedGuild, message: Message): HTMLElement {
|
||||
if (!message.resourceId || !message.resourcePreviewId || !message.resourceName) {
|
||||
throw new ShouldNeverHappenError('Message is not a resource message');
|
||||
}
|
||||
|
||||
let element = q.create({ class: 'message continued', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-server-id': server.id, content: [
|
||||
let element = q.create({ class: 'message continued', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-guild-id': guild.id, content: [
|
||||
{ class: 'timestamp', content: moment(message.sent).format('HH:mm') },
|
||||
{ class: 'right', content: [
|
||||
{ class: 'content image', style: `width: ${message.previewWidth}px; height: ${message.previewHeight}px;`, content:
|
||||
@ -28,20 +28,20 @@ export default function createImageResourceMessageContinued(document: Document,
|
||||
] }
|
||||
] }) as HTMLElement;
|
||||
q.$$$(element, '.content.image').addEventListener('click', () => {
|
||||
document.body.appendChild(createImageOverlay(document, q, server, message.resourceId as string, message.resourceName as string));
|
||||
document.body.appendChild(createImageOverlay(document, q, guild, message.resourceId as string, message.resourceName as string));
|
||||
});
|
||||
(async () => {
|
||||
try {
|
||||
let buffer = await server.fetchResource(message.resourcePreviewId as string);
|
||||
let src = await ElementsUtil.getImageBufferSrc(buffer);
|
||||
let resource = await guild.fetchResource(message.resourcePreviewId as string);
|
||||
let src = await ElementsUtil.getImageBufferSrc(resource.data);
|
||||
(q.$$$(element, '.content.image img') as HTMLImageElement).src = src;
|
||||
|
||||
let { mime, ext } = (await FileType.fromBuffer(buffer)) ?? { mime: null, ext: null };
|
||||
let { mime, ext } = (await FileType.fromBuffer(resource.data)) ?? { mime: null, ext: null };
|
||||
|
||||
if (mime === null || ext === null) throw new Error('unable to get mime/ext');
|
||||
|
||||
q.$$$(element, '.content.image').addEventListener('contextmenu', (e) => {
|
||||
let contextMenu = createImageContextMenu(document, q, server, message.resourceName as string, buffer, mime as string, ext as string, true);
|
||||
let contextMenu = createImageContextMenu(document, q, guild, message.resourceName as string, resource.data, mime as string, ext as string, true);
|
||||
document.body.appendChild(contextMenu);
|
||||
let relativeTo = { x: e.pageX, y: e.pageY };
|
||||
ElementsUtil.alignContextElement(contextMenu, relativeTo, { top: 'centerY', left: 'right' });
|
||||
|
@ -9,10 +9,10 @@ import * as FileType from 'file-type';
|
||||
import ElementsUtil from './require/elements-util.js';
|
||||
|
||||
import { Message, Member, ShouldNeverHappenError } from '../data-types';
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import createImageOverlay from './overlay-image';
|
||||
import createImageContextMenu from './context-menu-img';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createImageResourceMessage(document: Document, q: Q, guild: CombinedGuild, message: Message) {
|
||||
if (!message.resourceId || !message.resourcePreviewId || !message.resourceName) {
|
||||
@ -39,7 +39,7 @@ export default function createImageResourceMessage(document: Document, q: Q, gui
|
||||
}
|
||||
|
||||
let nameStyle = memberInfo.roleColor != null ? 'color: ' + memberInfo.roleColor : '';
|
||||
let element = q.create({ class: 'message', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-server-id': server.id, content: [
|
||||
let element = q.create({ class: 'message', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-guild-id': guild.id, content: [
|
||||
{ class: 'member-avatar', content: { tag: 'img', src: './img/loading.svg', alt: memberInfo.displayName } },
|
||||
{ class: 'right', content: [
|
||||
{ class: 'header', content: [
|
||||
@ -52,23 +52,23 @@ export default function createImageResourceMessage(document: Document, q: Q, gui
|
||||
] }
|
||||
] }) as HTMLElement;
|
||||
q.$$$(element, '.content.image').addEventListener('click', (e) => {
|
||||
document.body.appendChild(createImageOverlay(document, q, server, message.resourceId as string, message.resourceName as string));
|
||||
document.body.appendChild(createImageOverlay(document, q, guild, message.resourceId as string, message.resourceName as string));
|
||||
});
|
||||
(async () => {
|
||||
(q.$$$(element, '.member-avatar img') as HTMLImageElement).src =
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(server, memberInfo.avatarResourceId);
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, memberInfo.avatarResourceId);
|
||||
})();
|
||||
(async () => {
|
||||
try {
|
||||
let buffer = await server.fetchResource(message.resourcePreviewId as string);
|
||||
let src = await ElementsUtil.getImageBufferSrc(buffer);
|
||||
let resource = await guild.fetchResource(message.resourcePreviewId as string);
|
||||
let src = await ElementsUtil.getImageBufferSrc(resource.data);
|
||||
(q.$$$(element, '.content.image img') as HTMLImageElement).src = src;
|
||||
|
||||
let { mime, ext } = (await FileType.fromBuffer(buffer)) ?? { mime: null, ext: null };
|
||||
let { mime, ext } = (await FileType.fromBuffer(resource.data)) ?? { mime: null, ext: null };
|
||||
if (mime === null || ext === null) throw new Error('unable to get mime/ext');
|
||||
|
||||
q.$$$(element, '.content.image').addEventListener('contextmenu', (e) => {
|
||||
let contextMenu = createImageContextMenu(document, q, server, message.resourceName as string, buffer, mime as string, ext as string, true);
|
||||
let contextMenu = createImageContextMenu(document, q, guild, message.resourceName as string, resource.data, mime as string, ext as string, true);
|
||||
document.body.appendChild(contextMenu);
|
||||
let relativeTo = { x: e.pageX, y: e.pageY };
|
||||
ElementsUtil.alignContextElement(contextMenu, relativeTo, { top: 'centerY', left: 'right' });
|
||||
|
@ -1,6 +1,6 @@
|
||||
import * as moment from "moment";
|
||||
import ClientController from "../client-controller";
|
||||
import { Message } from '../data-types';
|
||||
import CombinedGuild from "../guild-combined";
|
||||
import Q from "../q-module";
|
||||
|
||||
import ElementsUtil from "./require/elements-util";
|
||||
@ -17,7 +17,7 @@ export default function createResourceMessageContinued(q: Q, guild: CombinedGuil
|
||||
throw new ShouldNeverHappenError('Message is not a resource message');
|
||||
}
|
||||
|
||||
let element = q.create({ class: 'message continued', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-server-id': server.id, content: [
|
||||
let element = q.create({ class: 'message continued', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-guild-id': guild.id, content: [
|
||||
{ class: 'timestamp', content: moment(message.sent).format('HH:mm') },
|
||||
{ class: 'right', content: [
|
||||
{ class: 'content resource', content: [
|
||||
@ -31,7 +31,7 @@ export default function createResourceMessageContinued(q: Q, guild: CombinedGuil
|
||||
] }
|
||||
] }) as HTMLElement;
|
||||
q.$$$(element, '.resource').addEventListener('click', ElementsUtil.createDownloadListener({
|
||||
server: server, resourceId: message.resourceId, resourceName: message.resourceName,
|
||||
guild: guild, resourceId: message.resourceId, resourceName: message.resourceName,
|
||||
downloadStartFunc: () => {
|
||||
q.$$$(element, '.resource .download-status').innerText = 'Downloading...';
|
||||
},
|
||||
|
@ -1,6 +1,6 @@
|
||||
import * as moment from 'moment';
|
||||
import ClientController from "../client-controller";
|
||||
import { Message, Member } from '../data-types';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
import Q from '../q-module';
|
||||
|
||||
import ElementsUtil from './require/elements-util';
|
||||
@ -37,7 +37,7 @@ export default function createResourceMessage(q: Q, guild: CombinedGuild, messag
|
||||
}
|
||||
|
||||
let nameStyle = memberInfo.roleColor ? 'color: ' + memberInfo.roleColor : '';
|
||||
let element = q.create({ class: 'message', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-server-id': server.id, content: [
|
||||
let element = q.create({ class: 'message', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-guild-id': guild.id, content: [
|
||||
{ class: 'member-avatar', content: { tag: 'img', src: './img/loading.svg', alt: memberInfo.displayName } },
|
||||
{ class: 'right', content: [
|
||||
{ class: 'header', content: [
|
||||
@ -55,7 +55,7 @@ export default function createResourceMessage(q: Q, guild: CombinedGuild, messag
|
||||
] }
|
||||
] }) as HTMLElement;
|
||||
q.$$$(element, '.resource').addEventListener('click', ElementsUtil.createDownloadListener({
|
||||
server: server, resourceId: message.resourceId, resourceName: message.resourceName,
|
||||
guild: guild, resourceId: message.resourceId, resourceName: message.resourceName,
|
||||
downloadStartFunc: () => {
|
||||
q.$$$(element, '.resource .download-status').innerText = 'Downloading...';
|
||||
},
|
||||
@ -76,7 +76,7 @@ export default function createResourceMessage(q: Q, guild: CombinedGuild, messag
|
||||
}));
|
||||
(async () => {
|
||||
(q.$$$(element, '.member-avatar img') as HTMLImageElement).src =
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(server, memberInfo.avatarResourceId);
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, memberInfo.avatarResourceId);
|
||||
})();
|
||||
return element;
|
||||
}
|
||||
|
@ -1,12 +1,12 @@
|
||||
import * as moment from 'moment';
|
||||
import ClientController from '../client-controller.js';
|
||||
import { Message } from '../data-types';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
import Q from '../q-module.js';
|
||||
|
||||
import ElementsUtil from './require/elements-util.js';
|
||||
|
||||
export default function createTextMessageContinued(q: Q, guild: CombinedGuild, message: Message): HTMLElement {
|
||||
return q.create({ class: 'message continued', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-server-id': server.id, content: [
|
||||
return q.create({ class: 'message continued', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-guild-id': guild.id, content: [
|
||||
{ class: 'timestamp', content: moment(message.sent).format('HH:mm') },
|
||||
{ class: 'right', content: [
|
||||
{ class: 'content text', content: ElementsUtil.parseMessageText(message.text ?? '') }
|
||||
|
@ -3,8 +3,8 @@ import * as moment from 'moment';
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
import { Message, Member, IDummyTextMessage } from '../data-types';
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createTextMessage(q: Q, guild: CombinedGuild, message: Message | IDummyTextMessage): HTMLElement {
|
||||
let memberInfo: {
|
||||
@ -35,7 +35,7 @@ export default function createTextMessage(q: Q, guild: CombinedGuild, message: M
|
||||
}
|
||||
|
||||
let nameStyle = memberInfo.roleColor ? 'color: ' + memberInfo.roleColor : '';
|
||||
let element = q.create({ class: 'message', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-server-id': server.id, content: [
|
||||
let element = q.create({ class: 'message', 'meta-id': message.id, 'meta-member-id': message.member.id, 'meta-guild-id': guild.id, content: [
|
||||
{ class: 'member-avatar', content: { tag: 'img', src: './img/loading.svg', alt: memberInfo.displayName } },
|
||||
{ class: 'right', content: [
|
||||
{ class: 'header', content: [
|
||||
@ -47,7 +47,7 @@ export default function createTextMessage(q: Q, guild: CombinedGuild, message: M
|
||||
] }) as HTMLElement;
|
||||
(async () => {
|
||||
(q.$$$(element, '.member-avatar img') as HTMLImageElement).src =
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(server, memberInfo.avatarResourceId);
|
||||
await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, memberInfo.avatarResourceId);
|
||||
})();
|
||||
return element;
|
||||
}
|
||||
|
@ -14,13 +14,12 @@ import Globals from '../globals';
|
||||
import BaseElements from './require/base-elements';
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import UI from '../ui';
|
||||
import Controller from '../controller';
|
||||
import Actions from '../actions';
|
||||
import GuildsManager from '../guilds-manager';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export interface IAddServerData {
|
||||
export interface IAddGuildData {
|
||||
name: string,
|
||||
url: string,
|
||||
cert: string,
|
||||
@ -51,23 +50,23 @@ function getExampleAvatarPath(): string {
|
||||
return paths[Math.floor(Math.random() * paths.length)];
|
||||
}
|
||||
|
||||
export default function createAddServerOverlay(document: Document, q: Q, ui: UI, controller: Controller, addServerData: IAddServerData): HTMLElement {
|
||||
let expired = addServerData.expires < new Date().getTime();
|
||||
export default function createAddGuildOverlay(document: Document, q: Q, ui: UI, guildsManager: GuildsManager, addGuildData: IAddGuildData): HTMLElement {
|
||||
let expired = addGuildData.expires < new Date().getTime();
|
||||
|
||||
let displayName = getExampleDisplayName();
|
||||
let avatarPath = getExampleAvatarPath();
|
||||
|
||||
//LOG.debug('addserverdata:', { addServerData });
|
||||
//LOG.debug('addguilddata:', { addGuildData });
|
||||
|
||||
let element = BaseElements.createOverlay(document, {
|
||||
class: 'content add-server', content: [
|
||||
class: 'content add-guild', content: [
|
||||
{ class: 'preview', content: [
|
||||
{ tag: 'img', class: 'icon', src: addServerData.iconSrc, alt: 'icon' },
|
||||
{ tag: 'img', class: 'icon', src: addGuildData.iconSrc, alt: 'icon' },
|
||||
{ content: [
|
||||
{ class: 'name', content: addServerData.name },
|
||||
{ class: 'url', content: addServerData.url },
|
||||
{ class: 'name', content: addGuildData.name },
|
||||
{ class: 'url', content: addGuildData.url },
|
||||
{ class: 'expires',
|
||||
content: (expired ? 'Invite Expired ' : 'Invite Expires ') + moment(addServerData.expires).fromNow() }
|
||||
content: (expired ? 'Invite Expired ' : 'Invite Expires ') + moment(addGuildData.expires).fromNow() }
|
||||
] }
|
||||
] },
|
||||
{ class: 'divider' },
|
||||
@ -93,7 +92,7 @@ export default function createAddServerOverlay(document: Document, q: Q, ui: UI,
|
||||
{ class: 'lower', content: [
|
||||
{ class: 'error' },
|
||||
{ class: 'buttons', content: [
|
||||
{ class: 'button submit', content: 'Add Server' }
|
||||
{ class: 'button submit', content: 'Add Guild' }
|
||||
] }
|
||||
] }
|
||||
]
|
||||
@ -177,12 +176,12 @@ export default function createAddServerOverlay(document: Document, q: Q, ui: UI,
|
||||
|
||||
q.$$$(element, '.display-name-input').removeAttribute('contenteditable');
|
||||
|
||||
let newguild: CombinedGuild | null = null;
|
||||
if (addServerData == null) {
|
||||
q.$$$(element, '.error').innerText = 'Very bad server file';
|
||||
let newGuild: CombinedGuild | null = null;
|
||||
if (addGuildData == null) {
|
||||
q.$$$(element, '.error').innerText = 'Very bad guild file';
|
||||
q.$$$(element, '.submit').innerText = 'Try Again';
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.submit'), 400);
|
||||
} else if (addServerData.expires < new Date().getTime()) {
|
||||
} else if (addGuildData.expires < new Date().getTime()) {
|
||||
q.$$$(element, '.error').innerText = 'Token expired';
|
||||
q.$$$(element, '.submit').innerText = 'Try Again';
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.submit'), 400);
|
||||
@ -201,20 +200,20 @@ export default function createAddServerOverlay(document: Document, q: Q, ui: UI,
|
||||
} else { // NOTE: Avatar size is checked above
|
||||
q.$$$(element, '.submit').innerText = 'Registering...';
|
||||
try {
|
||||
newServer = await controller.addNewGuild(addServerData, displayName, avatarBuff);
|
||||
newGuild = await guildsManager.addNewGuild(addGuildData, displayName, avatarBuff);
|
||||
} catch (e) {
|
||||
LOG.warn('error adding new server: ' + e.message, e); // explicitly not printing stack trace here
|
||||
LOG.warn('error adding new guild: ' + e.message, e); // explicitly not printing stack trace here
|
||||
q.$$$(element, '.error').innerText = e.message;
|
||||
q.$$$(element, '.submit').innerText = 'Try Again';
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.submit'), 400);
|
||||
newServer = null;
|
||||
newGuild = null;
|
||||
}
|
||||
}
|
||||
|
||||
if (newServer !== null) {
|
||||
let serverElement = await ui.addServer(controller, newServer);
|
||||
element.removeSelf(); // close the overlay since we got a new server
|
||||
serverElement.click(); // click on the new server
|
||||
if (newGuild !== null) {
|
||||
let guildElement = await ui.addGuild(guildsManager, newGuild);
|
||||
element.removeSelf(); // close the overlay since we got a new guild
|
||||
guildElement.click(); // click on the new guild
|
||||
}
|
||||
|
||||
q.$$$(element, '.display-name-input').setAttribute('contenteditable', 'plaintext-only');
|
@ -3,12 +3,12 @@ const electronConsole = electronRemote.getGlobal('console') as Console;
|
||||
import Logger from '../../../logger/logger';
|
||||
const LOG = Logger.create(__filename, electronConsole);
|
||||
|
||||
import ClientController from "../client-controller";
|
||||
import Globals from "../globals";
|
||||
|
||||
import ElementsUtil from "./require/elements-util";
|
||||
import BaseElements from "./require/base-elements";
|
||||
import Q from '../q-module';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createCreateChannelOverlay(document: Document, q: Q, guild: CombinedGuild): HTMLElement {
|
||||
|
||||
@ -83,7 +83,7 @@ export default function createCreateChannelOverlay(document: Document, q: Q, gui
|
||||
newFlavorText = null;
|
||||
}
|
||||
try {
|
||||
await server.createChannel(newName, newFlavorText);
|
||||
await guild.requestDoCreateChannel(newName, newFlavorText);
|
||||
success = true;
|
||||
} catch (e) {
|
||||
LOG.error('error updating channel', e);
|
||||
|
@ -1,5 +1,4 @@
|
||||
import ClientController from "../client-controller";
|
||||
|
||||
import CombinedGuild from "../guild-combined";
|
||||
import BaseElements from "./require/base-elements";
|
||||
|
||||
export default function createCreateInviteTokenOverlay(document: Document, guild: CombinedGuild): HTMLElement {
|
||||
|
@ -8,20 +8,20 @@ import Globals from '../globals';
|
||||
import BaseElements from './require/base-elements';
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
import ClientController from '../client-controller';
|
||||
import { CacheServerData, ServerMetaData } from '../data-types';
|
||||
import { GuildMetadata } from '../data-types';
|
||||
import Q from '../q-module';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createServerSettingsOverlay(document: Document, q: Q, guild: CombinedGuild, serverMeta: ServerMetaData | CacheServerData): HTMLElement {
|
||||
export default function createGuildSettingsOverlay(document: Document, q: Q, guild: CombinedGuild, guildMeta: GuildMetadata): HTMLElement {
|
||||
let element = BaseElements.createOverlay(document, {
|
||||
class: 'content submit-dialog server-settings', content: [
|
||||
{ class: 'server preview', content: [
|
||||
class: 'content submit-dialog guild-settings', content: [
|
||||
{ class: 'guild preview', content: [
|
||||
{ class: 'icon', content: { tag: 'img', src: './img/loading.svg', alt: 'icon' } },
|
||||
{ class: 'name', content: serverMeta.name }
|
||||
{ class: 'name', content: guildMeta.name }
|
||||
] },
|
||||
{ class: 'text-input server-name', placeholder: 'New Server Name',
|
||||
contenteditable: 'plaintext-only', content: serverMeta.name },
|
||||
{ class: 'image-input server-icon', content: [
|
||||
{ class: 'text-input guild-name', placeholder: 'New Guild Name',
|
||||
contenteditable: 'plaintext-only', content: guildMeta.name },
|
||||
{ class: 'image-input guild-icon', content: [
|
||||
{ tag: 'label', class: 'image-input-label button', content: [
|
||||
'Select New Icon',
|
||||
{ class: 'image-input-upload', tag: 'input', type: 'file', accept: '.png,.jpg,.jpeg', style: 'display: none;' }
|
||||
@ -37,7 +37,7 @@ export default function createServerSettingsOverlay(document: Document, q: Q, gu
|
||||
});
|
||||
|
||||
(async () => {
|
||||
(q.$$$(element, '.icon img') as HTMLImageElement).src = await ElementsUtil.getImageBufferFromResourceFailSoftly(server, serverMeta.iconResourceId);
|
||||
(q.$$$(element, '.icon img') as HTMLImageElement).src = await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, guildMeta.iconResourceId);
|
||||
})();
|
||||
|
||||
let newIconBuff: Buffer | null = null;
|
||||
@ -48,11 +48,11 @@ export default function createServerSettingsOverlay(document: Document, q: Q, gu
|
||||
onCleared: () => {},
|
||||
onError: async (errMsg) => {
|
||||
q.$$$(element, '.error').innerText = errMsg;
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.image-input-upload.server-icon'), 400);
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.image-input-upload.guild-icon'), 400);
|
||||
},
|
||||
onLoaded: (buff, src) => {
|
||||
newIconBuff = buff;
|
||||
(q.$$$(element, '.server .icon img') as HTMLImageElement).src = src;
|
||||
(q.$$$(element, '.guild .icon img') as HTMLImageElement).src = src;
|
||||
}
|
||||
});
|
||||
|
||||
@ -64,7 +64,7 @@ export default function createServerSettingsOverlay(document: Document, q: Q, gu
|
||||
});
|
||||
|
||||
q.$$$(element, '.text-input').addEventListener('input', () => {
|
||||
q.$$$(element, '.server.preview .name').innerText = q.$$$(element, '.text-input').innerText;
|
||||
q.$$$(element, '.guild.preview .name').innerText = q.$$$(element, '.text-input').innerText;
|
||||
});
|
||||
|
||||
let submitting = false;
|
||||
@ -76,33 +76,34 @@ export default function createServerSettingsOverlay(document: Document, q: Q, gu
|
||||
|
||||
let newName = q.$$$(element, '.text-input').innerText;
|
||||
|
||||
if (newName == serverMeta.name && newIconBuff == null) {
|
||||
if (newName == guildMeta.name && newIconBuff == null) {
|
||||
// nothing changed, close the dialog
|
||||
element.removeSelf();
|
||||
return;
|
||||
}
|
||||
|
||||
let success = false;
|
||||
if (newName != serverMeta.name && newName.length == 0) {
|
||||
LOG.warn('attempted to set empty server name');
|
||||
if (newName != guildMeta.name && newName.length == 0) {
|
||||
LOG.warn('attempted to set empty guild name');
|
||||
q.$$$(element, '.button.submit').innerText = 'Try Again';
|
||||
q.$$$(element, '.error').innerText = 'New name is empty';
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.button.submit'), 400);
|
||||
} else if (newName != serverMeta.name && newName.length > Globals.MAX_SERVER_NAME_LENGTH) {
|
||||
LOG.warn('attempted to oversized server name');
|
||||
} else if (newName != guildMeta.name && newName.length > Globals.MAX_GUILD_NAME_LENGTH) {
|
||||
LOG.warn('attempted to oversized guild name');
|
||||
q.$$$(element, '.button.submit').innerText = 'Try Again';
|
||||
q.$$$(element, '.error').innerText = 'New name is too long. ' + newName.length + ' > ' + Globals.MAX_SERVER_NAME_LENGTH;
|
||||
q.$$$(element, '.error').innerText = 'New name is too long. ' + newName.length + ' > ' + Globals.MAX_GUILD_NAME_LENGTH;
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.button.submit'), 400);
|
||||
} else { // client-size icon size checks are handled above
|
||||
let failed = false;
|
||||
// Set Name
|
||||
if (newName != serverMeta.name) {
|
||||
if (newName != guildMeta.name) {
|
||||
try {
|
||||
serverMeta = await server.setName(newName);
|
||||
await guild.requestSetServerName(newName);
|
||||
guildMeta = await guild.fetchMetadata();
|
||||
} catch (e) {
|
||||
LOG.error('error setting new server name', e);
|
||||
LOG.error('error setting new guild name', e);
|
||||
q.$$$(element, '.button.submit').innerText = 'Try Again';
|
||||
q.$$$(element, '.error').innerText = 'Error setting new server name';
|
||||
q.$$$(element, '.error').innerText = 'Error setting new guild name';
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.button.submit'), 400);
|
||||
failed = true;
|
||||
}
|
||||
@ -111,12 +112,12 @@ export default function createServerSettingsOverlay(document: Document, q: Q, gu
|
||||
// Set Icon
|
||||
if (!failed && newIconBuff != null) {
|
||||
try {
|
||||
await server.setIcon(newIconBuff);
|
||||
await guild.requestSetServerIcon(newIconBuff);
|
||||
newIconBuff = null; // prevent resubmit
|
||||
} catch (e) {
|
||||
LOG.error('error setting new server icon', e);
|
||||
LOG.error('error setting new guild icon', e);
|
||||
q.$$$(element, '.button.submit').innerText = 'Try Again';
|
||||
q.$$$(element, '.error').innerText = 'Error setting new server icon';
|
||||
q.$$$(element, '.error').innerText = 'Error setting new guild icon';
|
||||
await ElementsUtil.shakeElement(q.$$$(element, '.button.submit'), 400);
|
||||
failed = true;
|
||||
}
|
@ -8,9 +8,9 @@ import * as FileType from 'file-type'
|
||||
import BaseElements from './require/base-elements';
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import createImageContextMenu from './context-menu-img';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createImageOverlay(document: Document, q: Q, guild: CombinedGuild, resourceId: string, resourceName: string): HTMLElement {
|
||||
let element = BaseElements.createOverlay(document, { class: 'content popup-image', content: [
|
||||
@ -26,16 +26,16 @@ export default function createImageOverlay(document: Document, q: Q, guild: Comb
|
||||
|
||||
(async () => {
|
||||
try {
|
||||
let resourceBuff = await server.fetchResource(resourceId);
|
||||
let src = await ElementsUtil.getImageBufferSrc(resourceBuff);
|
||||
let resource = await guild.fetchResource(resourceId);
|
||||
let src = await ElementsUtil.getImageBufferSrc(resource.data);
|
||||
(q.$$$(element, '.content img') as HTMLImageElement).src = src;
|
||||
q.$$$(element, '.download .size').innerText = ElementsUtil.humanSize(resourceBuff.length);
|
||||
q.$$$(element, '.download .size').innerText = ElementsUtil.humanSize(resource.data.length);
|
||||
|
||||
let { mime, ext } = (await FileType.fromBuffer(resourceBuff)) ?? { mime: null, ext: null };
|
||||
let { mime, ext } = (await FileType.fromBuffer(resource.data)) ?? { mime: null, ext: null };
|
||||
if (mime === null || ext === null) throw new Error('unable to get mime/ext');
|
||||
|
||||
q.$$$(element, '.content img').addEventListener('contextmenu', (e) => {
|
||||
let contextMenu = createImageContextMenu(document, q, server, resourceName, resourceBuff, mime as string, ext as string, false);
|
||||
let contextMenu = createImageContextMenu(document, q, guild, resourceName, resource.data, mime as string, ext as string, false);
|
||||
document.body.appendChild(contextMenu);
|
||||
let relativeTo = { x: e.pageX, y: e.pageY };
|
||||
ElementsUtil.alignContextElement(contextMenu, relativeTo, { top: 'centerY', left: 'right' });
|
||||
@ -43,7 +43,7 @@ export default function createImageOverlay(document: Document, q: Q, guild: Comb
|
||||
|
||||
q.$$$(element, '.button').innerText = 'Save';
|
||||
q.$$$(element, '.button').addEventListener('click', ElementsUtil.createDownloadListener({
|
||||
downloadBuff: resourceBuff,
|
||||
downloadBuff: resource.data,
|
||||
resourceName: resourceName,
|
||||
downloadStartFunc: () => {
|
||||
q.$$$(element, '.button').innerText = 'Downloading...';
|
||||
|
@ -4,12 +4,12 @@ import Logger from '../../../logger/logger';
|
||||
const LOG = Logger.create(__filename, electronConsole);
|
||||
|
||||
import { Channel } from '../data-types';
|
||||
import ClientController from '../client-controller.js';
|
||||
import Globals from '../globals.js';
|
||||
|
||||
import BaseElements from './require/base-elements.js';
|
||||
import ElementsUtil from './require/elements-util.js';
|
||||
import Q from '../q-module';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createModifyChannelOverlay(document: Document, q: Q, guild: CombinedGuild, channel: Channel): HTMLElement {
|
||||
// See also overlay-create-channel
|
||||
@ -85,7 +85,7 @@ export default function createModifyChannelOverlay(document: Document, q: Q, gui
|
||||
newFlavorText = null;
|
||||
}
|
||||
try {
|
||||
await server.updateChannel(channel.id, newName, newFlavorText);
|
||||
await guild.requestDoUpdateChannel(channel.id, newName, newFlavorText);
|
||||
success = true;
|
||||
} catch (e) {
|
||||
LOG.error('error updating channel', e);
|
||||
|
@ -8,14 +8,14 @@ import ElementsUtil from './require/elements-util';
|
||||
|
||||
import Globals from '../globals';
|
||||
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import createTextMessage from './msg-txt';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createPersonalizeOverlay(document: Document, q: Q, guild: CombinedGuild, connection: any): HTMLElement {
|
||||
let element = BaseElements.createOverlay(document, {
|
||||
class: 'content submit-dialog personalize', content: [
|
||||
createTextMessage(q, server, { id: 'test-message', member: connection, sent: new Date(), text: 'Example Message' }),
|
||||
createTextMessage(q, guild, { id: 'test-message', member: connection, sent: new Date(), text: 'Example Message' }),
|
||||
{ class: 'text-input', placeholder: 'New Display Name',
|
||||
spellcheck: 'false', contenteditable: 'plaintext-only', content: connection.display_name },
|
||||
{ class: 'image-input avatar-input', content: [
|
||||
@ -95,7 +95,7 @@ export default function createPersonalizeOverlay(document: Document, q: Q, guild
|
||||
let failed = false;
|
||||
if (newDisplayName != connection.display_name) {
|
||||
try {
|
||||
await server.setDisplayName(newDisplayName);
|
||||
await guild.requestSetDisplayName(newDisplayName);
|
||||
connection.display_name = newDisplayName; // prevent resubmit
|
||||
} catch (e) {
|
||||
LOG.error('error setting display name', e);
|
||||
@ -109,7 +109,7 @@ export default function createPersonalizeOverlay(document: Document, q: Q, guild
|
||||
// Set New Avatar
|
||||
if (!failed && newAvatarBuffer != null) {
|
||||
try {
|
||||
await server.setAvatar(newAvatarBuffer);
|
||||
await guild.requestSetAvatar(newAvatarBuffer);
|
||||
newAvatarBuffer = null; // prevent resubmit
|
||||
} catch (e) {
|
||||
LOG.error('error setting avatar buffer', e);
|
||||
|
@ -9,9 +9,9 @@ import BaseElements from './require/base-elements';
|
||||
import ElementsUtil from './require/elements-util';
|
||||
|
||||
import Util from '../util';
|
||||
import ClientController from '../client-controller';
|
||||
import { Member } from '../data-types';
|
||||
import Q from '../q-module';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createTokenLogOverlay(document: Document, q: Q, guild: CombinedGuild): HTMLElement {
|
||||
let element = BaseElements.createOverlay(document, {
|
||||
@ -23,7 +23,7 @@ export default function createTokenLogOverlay(document: Document, q: Q, guild: C
|
||||
(async () => {
|
||||
Util.withPotentialErrorWarnOnCancel(q, {
|
||||
taskFunc: async () => {
|
||||
let tokens = await server.queryTokens();
|
||||
let tokens = await guild.fetchTokens();
|
||||
Q.clearChildren(q.$$$(element, '.tokens'));
|
||||
let displayed = 0;
|
||||
for (let token of tokens) {
|
||||
@ -81,7 +81,7 @@ export default function createTokenLogOverlay(document: Document, q: Q, guild: C
|
||||
q.$$$(contextElement, '.content').innerText = 'Revoking...';
|
||||
ElementsUtil.alignContextElement(contextElement, revokeElement, alignment);
|
||||
try {
|
||||
await server.revokeToken(token.token);
|
||||
await guild.requestDoRevokeToken(token.token);
|
||||
} catch (e) {
|
||||
LOG.error('unable to revoke token', e);
|
||||
q.$$$(contextElement, '.content').innerText = 'Unable to Revoke';
|
||||
|
@ -1,13 +1,13 @@
|
||||
import BaseElements from './require/base-elements.js';
|
||||
|
||||
import { Channel, ShouldNeverHappenError } from '../data-types';
|
||||
import ClientController from '../client-controller.js';
|
||||
import CombinedGuild from '../guild-combined.js';
|
||||
|
||||
export default function createUploadOverlayFromDataTransferItem(document: Document, guild: CombinedGuild, channel: Channel, dataTransferItem: DataTransferItem): HTMLElement {
|
||||
let file = dataTransferItem.getAsFile();
|
||||
if (file === null) throw new ShouldNeverHappenError('no file in the data transfer item');
|
||||
let element = BaseElements.createUploadOverlay(document, {
|
||||
server: server, channel: channel, resourceName: file.name,
|
||||
guild: guild, channel: channel, resourceName: file.name,
|
||||
resourceBuffFunc: async () => {
|
||||
if (file === null) throw new ShouldNeverHappenError('no file in the data transfer item');
|
||||
return Buffer.from(await file.arrayBuffer());
|
||||
|
@ -5,9 +5,9 @@ const LOG = Logger.create(__filename, electronConsole);
|
||||
|
||||
import { Channel } from '../data-types';
|
||||
import BaseElements from './require/base-elements';
|
||||
import ClientController from '../client-controller';
|
||||
import Q from '../q-module';
|
||||
import createUploadOverlayFromDataTransferItem from './overlay-upload-datatransfer';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createUploadDropTarget(document: Document, q: Q, guild: CombinedGuild, channel: Channel): HTMLElement {
|
||||
let element = BaseElements.createOverlay(document, { class: 'content drop-target', content: [
|
||||
@ -37,7 +37,7 @@ export default function createUploadDropTarget(document: Document, q: Q, guild:
|
||||
}
|
||||
}
|
||||
if (fileTransferItem) {
|
||||
let element = createUploadOverlayFromDataTransferItem(document, server, channel, fileTransferItem);
|
||||
let element = createUploadOverlayFromDataTransferItem(document, guild, channel, fileTransferItem);
|
||||
document.body.appendChild(element);
|
||||
q.$$$(element, '.text-input').focus();
|
||||
} else {
|
||||
|
@ -4,12 +4,12 @@ import * as path from 'path';
|
||||
import BaseElements from './require/base-elements';
|
||||
|
||||
import { Channel } from '../data-types';
|
||||
import ClientController from '../client-controller';
|
||||
import CombinedGuild from '../guild-combined';
|
||||
|
||||
export default function createUploadOverlayFromPath(document: Document, guild: CombinedGuild, channel: Channel, resourcePath: string): HTMLElement {
|
||||
let resourceName = path.basename(resourcePath);
|
||||
let element = BaseElements.createUploadOverlay(document, {
|
||||
server: server, channel: channel, resourceName: resourceName,
|
||||
guild: guild, channel: channel, resourceName: resourceName,
|
||||
resourceBuffFunc: async () => {
|
||||
return await fs.readFile(resourcePath);
|
||||
},
|
||||
|
@ -10,7 +10,7 @@ import Globals from '../../globals';
|
||||
|
||||
import ElementsUtil from './elements-util';
|
||||
import { Channel } from '../../data-types';
|
||||
import ClientController from '../../client-controller';
|
||||
import CombinedGuild from '../../guild-combined';
|
||||
import Q from '../../q-module';
|
||||
|
||||
interface HTMLElementWithRemoveSelf extends HTMLElement {
|
||||
@ -209,7 +209,7 @@ export default class BaseElements {
|
||||
static createUploadOverlay(document: Document, props: CreateUploadOverlayProps): HTMLElementWithRemoveSelf {
|
||||
const q = new Q(document);
|
||||
|
||||
const { server, channel, resourceName, resourceBuffFunc, resourceSizeFunc } = props;
|
||||
const { guild, channel, resourceName, resourceBuffFunc, resourceSizeFunc } = props;
|
||||
|
||||
let element = BaseElements.createOverlay(document, { class: 'content upload', content: [
|
||||
{ class: 'title', content: [
|
||||
@ -248,7 +248,7 @@ export default class BaseElements {
|
||||
return;
|
||||
}
|
||||
sending = true;
|
||||
if (!server.isVerified) {
|
||||
if (!guild.isSocketVerified()) {
|
||||
LOG.warn('client attempted to send message with resource while not verified');
|
||||
q.$$$(element, '.error').innerText = 'Not Connected to Server';
|
||||
q.$$$(element, '.button.upload').innerText = 'Try Again';
|
||||
@ -290,7 +290,7 @@ export default class BaseElements {
|
||||
return;
|
||||
}
|
||||
try {
|
||||
await server.sendMessageWithResource(channel.id, text, resourceBuff, resourceName);
|
||||
await guild.requestSendMessageWithResource(channel.id, text, resourceBuff, resourceName);
|
||||
} catch (e) {
|
||||
q.$$$(element, '.error').innerText = 'Error uploading resource.';
|
||||
q.$$$(element, '.button.upload').innerText = 'Try Again';
|
||||
|
@ -13,7 +13,7 @@ import * as FileType from 'file-type';
|
||||
|
||||
import Util from '../../util';
|
||||
import Globals from '../../globals';
|
||||
import ClientController from '../../client-controller';
|
||||
import CombinedGuild from '../../guild-combined';
|
||||
import { ShouldNeverHappenError } from '../../data-types';
|
||||
|
||||
// TODO: pass-through Globals in init function
|
||||
@ -36,7 +36,7 @@ interface IHTMLElementWithRemovalType extends HTMLElement {
|
||||
|
||||
interface CreateDownloadListenerProps {
|
||||
downloadBuff?: Buffer;
|
||||
server?: ClientController;
|
||||
guild?: CombinedGuild;
|
||||
resourceId?: string;
|
||||
resourceName: string;
|
||||
downloadStartFunc: (() => Promise<void> | void);
|
||||
@ -101,15 +101,15 @@ export default class ElementsUtil {
|
||||
|
||||
static async getImageBufferFromResourceFailSoftly(guild: CombinedGuild, resourceId: string | null): Promise<string> {
|
||||
if (!resourceId) {
|
||||
LOG.warn('no server resource specified, showing error instead', new Error());
|
||||
LOG.warn('no guild resource specified, showing error instead', new Error());
|
||||
return './img/error.png';
|
||||
}
|
||||
try {
|
||||
let resourceBuff = await server.fetchResource(resourceId);
|
||||
let src = await ElementsUtil.getImageBufferSrc(resourceBuff);
|
||||
let resource = await guild.fetchResource(resourceId);
|
||||
let src = await ElementsUtil.getImageBufferSrc(resource.data);
|
||||
return src;
|
||||
} catch (e) {
|
||||
LOG.warn('unable to fetch server resource, showing error instead', e);
|
||||
LOG.warn('unable to fetch guild resource, showing error instead', e);
|
||||
return './img/error.png';
|
||||
}
|
||||
}
|
||||
@ -340,7 +340,7 @@ export default class ElementsUtil {
|
||||
static createDownloadListener(props: CreateDownloadListenerProps): (() => Promise<void>) {
|
||||
const {
|
||||
downloadBuff, // pre-downloaded buffer to save rather than submit a download request (downloadStartFunc still required)
|
||||
server, resourceId, resourceName,
|
||||
guild, resourceId, resourceName,
|
||||
downloadStartFunc, downloadFailFunc,
|
||||
writeStartFunc, writeFailFunc,
|
||||
successFunc
|
||||
@ -362,10 +362,10 @@ export default class ElementsUtil {
|
||||
if (downloadBuff) {
|
||||
resourceBuff = downloadBuff;
|
||||
} else {
|
||||
if (!server) throw new ShouldNeverHappenError('server is null and we are not using a pre-download');
|
||||
if (!guild) throw new ShouldNeverHappenError('guild is null and we are not using a pre-download');
|
||||
if (!resourceId) throw new ShouldNeverHappenError('resourceId is null and we are not using a pre-download');
|
||||
try {
|
||||
resourceBuff = await server.fetchResource(resourceId);
|
||||
resourceBuff = (await guild.fetchResource(resourceId)).data;
|
||||
} catch (e) {
|
||||
LOG.error('Error downloading resource', e);
|
||||
if (downloadFailFunc) await downloadFailFunc(e);
|
||||
|
257
client/webapp/entrypoint.ts
Normal file
257
client/webapp/entrypoint.ts
Normal file
@ -0,0 +1,257 @@
|
||||
import * as electronRemote from '@electron/remote';
|
||||
const electronConsole = electronRemote.getGlobal('console') as Console;
|
||||
import Logger from '../../logger/logger';
|
||||
const LOG = Logger.create(__filename, electronConsole);
|
||||
|
||||
LOG.silly('script begins');
|
||||
|
||||
import * as path from 'path';
|
||||
import * as fs from 'fs/promises';
|
||||
|
||||
import GuildsManager from './guilds-manager';
|
||||
|
||||
import Globals from './globals';
|
||||
|
||||
import UI from './ui';
|
||||
import Actions from './actions';
|
||||
import { Changes, Channel, ConnectionInfo, GuildMetadata, Member, Message, Resource, Token } from './data-types';
|
||||
import Q from './q-module';
|
||||
import bindWindowButtonEvents from './elements/events-window-buttons';
|
||||
import bindTextInputEvents from './elements/events-text-input';
|
||||
import bindInfiniteScrollEvents from './elements/events-infinite-scroll';
|
||||
import bindConnectionEvents from './elements/events-connection';
|
||||
import bindAddGuildTitleEvents from './elements/events-guild-title';
|
||||
import bindAddGuildEvents from './elements/events-add-guild';
|
||||
import PersonalDB from './personal-db';
|
||||
import MessageRAMCache from './message-ram-cache';
|
||||
import ResourceRAMCache from './resource-ram-cache';
|
||||
import CombinedGuild from './guild-combined';
|
||||
|
||||
LOG.silly('modules loaded');
|
||||
|
||||
if (Globals.MESSAGES_PER_REQUEST >= Globals.MAX_CURRENT_MESSAGES) throw new Error('messages per request must be less than max current messages');
|
||||
|
||||
window.addEventListener('unhandledrejection', (e) => {
|
||||
LOG.error('Unhandled Promise Rejection', e.reason);
|
||||
});
|
||||
|
||||
window.addEventListener('error', (e) => {
|
||||
LOG.error('Uncaught Error', e.error);
|
||||
});
|
||||
|
||||
window.addEventListener('DOMContentLoaded', () => {
|
||||
document.body.classList.remove('preload');
|
||||
|
||||
(async () => {
|
||||
// Wait for the log to load the typescript source maps so that
|
||||
// logs will include typescript files+line numbers instead of
|
||||
// compiled javascript ones.
|
||||
await LOG.ensureSourceMaps();
|
||||
LOG.silly('web client log source maps loaded');
|
||||
|
||||
// make sure the personaldb directory exists
|
||||
await fs.mkdir(path.dirname(Globals.PERSONALDB_FILE), { recursive: true });
|
||||
const personalDB = await PersonalDB.create(Globals.PERSONALDB_FILE);
|
||||
await personalDB.init();
|
||||
|
||||
LOG.silly('personal db initialized');
|
||||
|
||||
let messageRAMCache = new MessageRAMCache();
|
||||
let resourceRAMCache = new ResourceRAMCache();
|
||||
|
||||
LOG.silly('ram caches initialized');
|
||||
|
||||
const guildsManager = new GuildsManager(messageRAMCache, resourceRAMCache, personalDB);
|
||||
await guildsManager.init();
|
||||
|
||||
LOG.silly('controller initialized');
|
||||
|
||||
const q = new Q(document);
|
||||
const ui = new UI(document, q);
|
||||
|
||||
LOG.silly('action classes initialized');
|
||||
|
||||
bindWindowButtonEvents(q);
|
||||
bindTextInputEvents(document, q, ui);
|
||||
bindInfiniteScrollEvents(q, ui);
|
||||
bindConnectionEvents(document, q, ui);
|
||||
bindAddGuildTitleEvents(document, q, ui);
|
||||
bindAddGuildEvents(document, q, ui, guildsManager);
|
||||
|
||||
LOG.silly('events bound');
|
||||
|
||||
// Add guild icons
|
||||
await ui.setGuilds(guildsManager, guildsManager.guilds);
|
||||
|
||||
if (guildsManager.guilds.length > 0) {
|
||||
// Click on the first guild in the list
|
||||
q.$('#guild-list .guild').click();
|
||||
}
|
||||
|
||||
// Connection Events
|
||||
|
||||
guildsManager.on('verified', async (guild: CombinedGuild) => {
|
||||
(async () => { // update connection info
|
||||
await Actions.fetchAndUpdateConnection(ui, guild);
|
||||
})();
|
||||
(async () => { // refresh members list
|
||||
await Actions.fetchAndUpdateMembers(q, ui, guild);
|
||||
})();
|
||||
(async () => { // refresh channels list
|
||||
await Actions.fetchAndUpdateChannels(q, ui, guild);
|
||||
})();
|
||||
(async () => { // refresh current channel messages
|
||||
if (ui.activeChannel === null) return;
|
||||
if (ui.messagePairs.size == 0) {
|
||||
// fetch messages again since there are no messages yet
|
||||
await Actions.fetchAndUpdateMessagesRecent(q, ui, guild, ui.activeChannel);
|
||||
} else {
|
||||
// If we already have messages, just update the infinite scroll.
|
||||
// NOTE: this will not add/remove new/deleted messages
|
||||
ui.messagesAtTop = false;
|
||||
ui.messagesAtBottom = false;
|
||||
(q.$('#channel-feed-content-wrapper') as any).updateInfiniteScroll();
|
||||
}
|
||||
})();
|
||||
});
|
||||
|
||||
guildsManager.on('disconnect', (guild: CombinedGuild) => {
|
||||
// Update everyone with the 'unknown' status
|
||||
(async () => {
|
||||
await Actions.fetchAndUpdateConnection(ui, guild);
|
||||
})();
|
||||
(async () => {
|
||||
await Actions.fetchAndUpdateMembers(q, ui, guild);
|
||||
})();
|
||||
});
|
||||
|
||||
// Change Events
|
||||
|
||||
guildsManager.on('new-messages', async (guild: CombinedGuild, messages: Message[]) => {
|
||||
if (ui.activeGuild === null || ui.activeGuild.id !== guild.id) return;
|
||||
for (let message of messages) {
|
||||
if (ui.activeChannel === null || ui.activeChannel.id !== message.channel.id) return;
|
||||
if (ui.messagesAtBottom) {
|
||||
// add the message to the bottom of the message feed
|
||||
await ui.addMessages(guild, [ message ]);
|
||||
ui.jumpMessagesToBottom();
|
||||
} else if (message.member.id == guild.memberId) {
|
||||
// this set of messages will include the new messageguildId
|
||||
LOG.debug('not at bottom, jumping down since message was sent by the current user');
|
||||
await Actions.fetchAndUpdateMessagesRecent(q, ui, guild, message.channel);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
guildsManager.on('update-metadata', async (guild: CombinedGuild, guildMeta: GuildMetadata) => {
|
||||
LOG.debug(`g#${guild.id} metadata updated`)
|
||||
await ui.updateGuildName(guild, guildMeta.name);
|
||||
|
||||
// Not using withPotentialError since keeping the old icon is a fine fallback
|
||||
if (guildMeta.iconResourceId) {
|
||||
try {
|
||||
let icon = await guild.fetchResource(guildMeta.iconResourceId);
|
||||
await ui.updateGuildIcon(guild, icon.data);
|
||||
} catch (e) {
|
||||
LOG.error('Error fetching new guild icon', e);
|
||||
// Keep the old guild icon, just log an error.
|
||||
// Should go through another try after a restart
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
guildsManager.on('remove-members', async (guild: CombinedGuild, members: Member[]) => {
|
||||
LOG.debug(members.length + ' removed members');
|
||||
await ui.deleteMembers(guild, members);
|
||||
});
|
||||
|
||||
guildsManager.on('update-members', async (guild: CombinedGuild, updatedMembers: Member[]) => {
|
||||
LOG.debug(updatedMembers.length + ' updated members g#' + guild.id);
|
||||
await ui.updateMembers(guild, updatedMembers);
|
||||
if (
|
||||
ui.activeConnection !== null &&
|
||||
updatedMembers.find(member => member.id === (ui.activeConnection as ConnectionInfo).id)
|
||||
) {
|
||||
await Actions.fetchAndUpdateConnection(ui, guild);
|
||||
}
|
||||
});
|
||||
|
||||
guildsManager.on('new-members', async (guild: CombinedGuild, members: Member[]) => {
|
||||
LOG.debug(members.length + ' new members');
|
||||
await ui.addMembers(guild, members);
|
||||
});
|
||||
|
||||
guildsManager.on('remove-channels', async (guild: CombinedGuild, channels: Channel[]) => {
|
||||
LOG.debug(channels.length + ' removed channels');
|
||||
await ui.deleteChannels(guild, channels);
|
||||
});
|
||||
|
||||
guildsManager.on('update-channels', async (guild: CombinedGuild, updatedChannels: Channel[]) => {
|
||||
LOG.debug(updatedChannels.length + ' updated channels');
|
||||
await ui.updateChannels(guild, updatedChannels);
|
||||
});
|
||||
|
||||
guildsManager.on('new-channels', async (guild: CombinedGuild, channels: Channel[]) => {
|
||||
LOG.debug(channels.length + ' added channels');
|
||||
await ui.addChannels(guild, channels);
|
||||
});
|
||||
|
||||
guildsManager.on('remove-messages', async (guild: CombinedGuild, messages: Message[]) => {
|
||||
LOG.debug(messages.length + ' deleted messages');
|
||||
await ui.deleteMessages(guild, messages);
|
||||
});
|
||||
|
||||
guildsManager.on('update-messages', async (guild: CombinedGuild, updatedMessages: Message[]) => {
|
||||
LOG.debug(updatedMessages.length + ' updated messages');
|
||||
await ui.updateMessages(guild, updatedMessages);
|
||||
});
|
||||
|
||||
guildsManager.on('new-messages', async (guild: CombinedGuild, messages: Message[]) => {
|
||||
LOG.debug(messages.length + ' new messages');
|
||||
await ui.addMessages(guild, messages);
|
||||
});
|
||||
|
||||
// Conflict Events
|
||||
|
||||
guildsManager.on('conflict-metadata', async (guild: CombinedGuild, guildMeta: GuildMetadata) => {
|
||||
LOG.debug('metadata conflict', { newMetadata: guildMeta });
|
||||
(async () => { await ui.updateGuildName(guild, guildMeta.name); })();
|
||||
(async () => {
|
||||
let icon = await guild.fetchResource(guildMeta.iconResourceId);
|
||||
await ui.updateGuildIcon(guild, icon.data);
|
||||
})();
|
||||
});
|
||||
|
||||
guildsManager.on('conflict-channels', async (guild: CombinedGuild, changes: Changes<Channel>) => {
|
||||
LOG.debug('channels conflict', { changes });
|
||||
if (changes.deleted.length > 0) await ui.deleteChannels(guild, changes.deleted);
|
||||
if (changes.added.length > 0) await ui.addChannels(guild, changes.added);
|
||||
if (changes.updated.length > 0) await ui.updateChannels(guild, changes.updated.map(pair => pair.newDataPoint));
|
||||
});
|
||||
|
||||
guildsManager.on('conflict-members', async (guild: CombinedGuild, changes: Changes<Member>) => {
|
||||
LOG.debug('members conflict', { changes });
|
||||
if (changes.deleted.length > 0) await ui.deleteMembers(guild, changes.deleted);
|
||||
if (changes.added.length > 0) await ui.addMembers(guild, changes.added);
|
||||
if (changes.updated.length > 0) await ui.updateMembers(guild, changes.updated.map(pair => pair.newDataPoint));
|
||||
});
|
||||
|
||||
guildsManager.on('conflict-messages', async (guild: CombinedGuild, changes: Changes<Message>) => {
|
||||
LOG.debug('messages conflict', { changes });
|
||||
if (changes.deleted.length > 0) await ui.deleteMessages(guild, changes.deleted);
|
||||
if (changes.added.length > 0) await ui.addMessages(guild, changes.added);
|
||||
if (changes.updated.length > 0) await ui.updateMessages(guild, changes.updated.map(pair => pair.newDataPoint));
|
||||
});
|
||||
|
||||
guildsManager.on('conflict-tokens', async (guild: CombinedGuild, changes: Changes<Token>) => {
|
||||
LOG.debug('tokens conflict', { changes });
|
||||
// TODO
|
||||
});
|
||||
|
||||
guildsManager.on('conflict-resource', async (guild: CombinedGuild, oldResource: Resource, newResource: Resource) => {
|
||||
LOG.debug('resource conflict', { oldResource, newResource });
|
||||
// TODO (these changes should not happen often if at all)
|
||||
});
|
||||
})();
|
||||
});
|
||||
|
@ -1,26 +1,27 @@
|
||||
export default class Globals {
|
||||
static DOWNLOAD_DIR = '/home/michael/Downloads'; // TODO: not hard coded
|
||||
static PERSONALDB_FILE = './db/personal.db';
|
||||
|
||||
static DEFAULT_SOCKET_TIMEOUT = 5000; // Wait up to 5000ms for the server to respond before rejecting/throwing an error
|
||||
static DEFAULT_SOCKET_TIMEOUT = 5000; // Wait up to 5000ms for the guild to respond before rejecting/throwing an error
|
||||
|
||||
static MAX_CURRENT_MESSAGES = 300;
|
||||
static MESSAGES_PER_REQUEST = 100;
|
||||
|
||||
static MAX_TEXT_MESSAGE_LENGTH = 1024 * 2; // 2 KB character message max (for readability)
|
||||
static MAX_RESOURCE_SIZE = 1024 * 1024 * 10; // 10 MB max resource size (server is 50, client-side is limited until socket.io-stream)
|
||||
static MAX_RESOURCE_SIZE = 1024 * 1024 * 10; // 10 MB max resource size (guild is 50, client-side is limited until socket.io-stream)
|
||||
|
||||
static MAX_AVATAR_SIZE = 1024 * 128; // 128 KB max avatar size
|
||||
static MAX_DISPLAY_NAME_LENGTH = 32; // 32 char max display name length
|
||||
|
||||
static MAX_ICON_SIZE = 1024 * 128; // 128 KB max server icon size
|
||||
static MAX_SERVER_NAME_LENGTH = 64; // 64 char max server name length
|
||||
static MAX_ICON_SIZE = 1024 * 128; // 128 KB max guild icon size
|
||||
static MAX_GUILD_NAME_LENGTH = 64; // 64 char max guild name length
|
||||
|
||||
static MAX_CHANNEL_NAME_LENGTH = 32; // 32 char max channel name length
|
||||
static MAX_CHANNEL_FLAVOR_TEXT_LENGTH = 256; // 256 char max channel flavor text length
|
||||
|
||||
static MAX_CACHED_CHANNEL_MESSAGES = 1000; // the 1000 most recent messages in each text channel are cached (in the sqlite db)
|
||||
|
||||
static MAX_SERVER_RESOURCE_CACHE_SIZE = 1024 * 1024 * 1024; // 1 GB max resource cache per server
|
||||
static MAX_SERVER_RESOURCE_CACHE_SIZE = 1024 * 1024 * 1024; // 1 GB max resource cache per guild
|
||||
static MAX_CACHED_RESOURCE_SIZE = 1024 * 1024 * 4; // 4 MB is the biggest resource that will be cached
|
||||
|
||||
static MAX_RAM_CACHED_MESSAGES_CHANNEL_CHARACTERS = 1024 * 1024 * 64; // at most, 64 MB of channel
|
||||
|
@ -7,7 +7,7 @@ import * as socketio from 'socket.io-client';
|
||||
import PersonalDBGuild from './guild-personal-db';
|
||||
import RAMGuild from './guild-ram';
|
||||
import SocketGuild from './guild-socket';
|
||||
import { Changes, Channel, GuildMetadata, GuildMetadataWithIds, Member, Message, Resource, ServerMetaData, SocketConfig, Token } from './data-types';
|
||||
import { Changes, Channel, ConnectionInfo, GuildMetadata, GuildMetadataWithIds, Member, Message, Resource, ServerMetaData, SocketConfig, Token } from './data-types';
|
||||
|
||||
import MessageRAMCache from "./message-ram-cache";
|
||||
import PersonalDB from "./personal-db";
|
||||
@ -28,7 +28,7 @@ export default class CombinedGuild extends EventEmitter<Connectable & Conflictab
|
||||
|
||||
constructor(
|
||||
public readonly id: number,
|
||||
private readonly memberId: string,
|
||||
public readonly memberId: string,
|
||||
socket: socketio.Socket,
|
||||
socketVerifier: SocketVerifier,
|
||||
messageRAMCache: MessageRAMCache,
|
||||
@ -41,16 +41,15 @@ export default class CombinedGuild extends EventEmitter<Connectable & Conflictab
|
||||
this.personalDBGuild = new PersonalDBGuild(personalDB, this.id, this.memberId);
|
||||
this.socketGuild = new SocketGuild(socket, socketVerifier);
|
||||
|
||||
// TODO: Only unverify the personaldb->socket connection on d/c?
|
||||
|
||||
// Connect/Disconnect
|
||||
this.socketGuild.on('connect', () => {
|
||||
LOG.info(`g#${this.id} connected`);
|
||||
this.emit('connect');
|
||||
});
|
||||
this.socketGuild.on('disconnect', () => {
|
||||
this.socketGuild.on('disconnect', async () => {
|
||||
LOG.info(`g#${this.id} disconnected`);
|
||||
this.unverify();
|
||||
await personalDB.clearAllMembersStatus(this.id);
|
||||
this.emit('disconnect');
|
||||
});
|
||||
|
||||
@ -175,12 +174,20 @@ export default class CombinedGuild extends EventEmitter<Connectable & Conflictab
|
||||
);
|
||||
}
|
||||
|
||||
public isSocketVerified(): boolean {
|
||||
return this.socketGuild.verifier.isVerified;
|
||||
}
|
||||
|
||||
private unverify(): void {
|
||||
for (let pairVerifier of this.pairVerifiers) {
|
||||
pairVerifier.unverify();
|
||||
}
|
||||
}
|
||||
|
||||
public disconnect(): void {
|
||||
this.socketGuild.disconnect();
|
||||
}
|
||||
|
||||
private async ensureRAMMembers(): Promise<void> {
|
||||
if (this.ramGuild.getMembers().size === 0) {
|
||||
await this.fetchMembers();
|
||||
@ -195,16 +202,60 @@ export default class CombinedGuild extends EventEmitter<Connectable & Conflictab
|
||||
if (this.ramGuild.getChannels().size === 0) throw new Error('RAM Channels was not updated through fetchChannels');
|
||||
}
|
||||
|
||||
public async grabRAMMembersMap(): Promise<Map<string, Member>> {
|
||||
async grabRAMMembersMap(): Promise<Map<string, Member>> {
|
||||
await this.ensureRAMMembers();
|
||||
return this.ramGuild.getMembers();
|
||||
}
|
||||
|
||||
public async grabRAMChannelsMap(): Promise<Map<string, Channel>> {
|
||||
async grabRAMChannelsMap(): Promise<Map<string, Channel>> {
|
||||
await this.ensureRAMChannels();
|
||||
return this.ramGuild.getChannels();
|
||||
}
|
||||
|
||||
async fetchConnectionInfo(): Promise<ConnectionInfo> {
|
||||
let connection: ConnectionInfo = {
|
||||
id: null,
|
||||
avatarResourceId: null,
|
||||
displayName: 'Connecting...',
|
||||
status: '',
|
||||
privileges: [],
|
||||
roleName: null,
|
||||
roleColor: null,
|
||||
rolePriority: null
|
||||
};
|
||||
if (this.socketGuild.verifier.isVerified) {
|
||||
let members = await this.grabRAMMembersMap();
|
||||
let member = members.get(this.memberId);
|
||||
if (member) {
|
||||
connection.id = member.id;
|
||||
connection.avatarResourceId = member.avatarResourceId;
|
||||
connection.displayName = member.displayName;
|
||||
connection.status = member.status;
|
||||
connection.roleName = member.roleName;
|
||||
connection.roleColor = member.roleColor;
|
||||
connection.rolePriority = member.rolePriority;
|
||||
connection.privileges = member.privileges;
|
||||
} else {
|
||||
LOG.warn('unable to find self in members');
|
||||
}
|
||||
} else {
|
||||
let members = await this.personalDBGuild.fetchMembers();
|
||||
if (members) {
|
||||
let member = members.find(m => m.id === this.memberId);
|
||||
if (member) {
|
||||
connection.id = member.id;
|
||||
connection.avatarResourceId = member.avatarResourceId;
|
||||
connection.displayName = member.displayName;
|
||||
connection.status = 'connecting';
|
||||
connection.privileges = [];
|
||||
} else {
|
||||
LOG.warn('unable to find self in cached members');
|
||||
}
|
||||
}
|
||||
}
|
||||
return connection;
|
||||
}
|
||||
|
||||
// Fetched through the triple-cache system (RAM -> Disk -> Server)
|
||||
async fetchMetadata(): Promise<GuildMetadata> {
|
||||
return await this.fetchable.fetchMetadata();
|
||||
@ -247,6 +298,7 @@ export default class CombinedGuild extends EventEmitter<Connectable & Conflictab
|
||||
async requestSetAvatar(avatar: Buffer): Promise<void> {
|
||||
await this.socketGuild.requestSetAvatar(avatar);
|
||||
}
|
||||
// TODO: Rename Server -> Guild
|
||||
async requestSetServerName(serverName: string): Promise<void> {
|
||||
await this.socketGuild.requestSetServerName(serverName);
|
||||
}
|
||||
|
@ -18,7 +18,7 @@ export default class SocketGuild extends EventEmitter<Connectable> implements As
|
||||
|
||||
constructor(
|
||||
private socket: socketio.Socket,
|
||||
private verifier: SocketVerifier
|
||||
public verifier: SocketVerifier
|
||||
) {
|
||||
super();
|
||||
this.socket.on('connect', async () => {
|
||||
@ -63,6 +63,10 @@ export default class SocketGuild extends EventEmitter<Connectable> implements As
|
||||
});
|
||||
}
|
||||
|
||||
public disconnect() {
|
||||
this.socket.disconnect();
|
||||
}
|
||||
|
||||
// server helper functions
|
||||
|
||||
private async query(timeout: number, endpoint: string, ...args: any[]): Promise<any> {
|
||||
|
@ -107,23 +107,25 @@ export interface SyncLackable {
|
||||
|
||||
export type Lackable = AsyncLackable | SyncLackable;
|
||||
|
||||
|
||||
|
||||
// A Connectable can emit server-like events
|
||||
export type Connectable = {
|
||||
'connect': () => void;
|
||||
'disconnect': () => void;
|
||||
'verified': () => void;
|
||||
|
||||
'update-metadata': (guildMeta: GuildMetadata) => void;
|
||||
|
||||
'new-channels': (channels: Channel[]) => void;
|
||||
'update-channels': (updatedChannels: Channel[]) => void;
|
||||
'remove-channels': (removedChannels: Channel[]) => void;
|
||||
|
||||
'new-members': (members: Member[]) => void;
|
||||
'update-members': (updatedMembers: Member[]) => void;
|
||||
'remove-members': (removedMembers: Member[]) => void;
|
||||
|
||||
'new-messages': (messages: Message[]) => void;
|
||||
'update-messages': (updatedMessages: Message[]) => void;
|
||||
'remove-messages': (removedMessages: Message[]) => void;
|
||||
}
|
||||
|
||||
// A Conflictable could emit conflict-based events if data changed based on verification
|
||||
|
@ -11,26 +11,30 @@ import * as crypto from 'crypto';
|
||||
|
||||
import { Changes, Channel, GuildMetadata, GuildMetadataWithIds, Member, Message, Resource, ServerConfig, SocketConfig, Token } from './data-types';
|
||||
import { IAddServerData } from './elements/overlay-add-server';
|
||||
import { DefaultEventMap, EventEmitter } from 'tsee';
|
||||
import { EventEmitter } from 'tsee';
|
||||
import CombinedGuild from './guild-combined';
|
||||
import PersonalDB from './personal-db';
|
||||
import MessageRAMCache from './message-ram-cache';
|
||||
import ResourceRAMCache from './resource-ram-cache';
|
||||
|
||||
export default class Controller extends EventEmitter<{
|
||||
export default class GuildsManager extends EventEmitter<{
|
||||
'connect': (guild: CombinedGuild) => void;
|
||||
'disconnect': (guild: CombinedGuild) => void;
|
||||
'verified': (guild: CombinedGuild) => void;
|
||||
|
||||
'update-metadata': (guild: CombinedGuild, guildMeta: GuildMetadata) => void;
|
||||
|
||||
'new-channels': (guild: CombinedGuild, channels: Channel[]) => void;
|
||||
'update-channels': (guild: CombinedGuild, updatedChannels: Channel[]) => void;
|
||||
'remove-channels': (guild: CombinedGuild, removedChannels: Channel[]) => void;
|
||||
|
||||
'new-members': (guild: CombinedGuild, members: Member[]) => void;
|
||||
'update-members': (guild: CombinedGuild, updatedMembers: Member[]) => void;
|
||||
'remove-members': (guild: CombinedGuild, removedMembers: Member[]) => void;
|
||||
|
||||
'new-messages': (guild: CombinedGuild, messages: Message[]) => void;
|
||||
'update-messages': (guild: CombinedGuild, updatedMessages: Message[]) => void;
|
||||
'remove-messages': (guild: CombinedGuild, removedMessages: Message[]) => void;
|
||||
|
||||
'conflict-metadata': (guild: CombinedGuild, oldGuildMeta: GuildMetadata, newGuildMeta: GuildMetadata) => void;
|
||||
'conflict-channels': (guild: CombinedGuild, changes: Changes<Channel>) => void;
|
||||
@ -141,7 +145,7 @@ export default class Controller extends EventEmitter<{
|
||||
|
||||
return await new Promise((resolve, reject) => {
|
||||
let clientPublicKeyDerBuff = publicKey.export({ type: 'spki', format: 'der' });
|
||||
Controller._socketEmitTimeout(socket, 5000, 'register-with-token',
|
||||
GuildsManager._socketEmitTimeout(socket, 5000, 'register-with-token',
|
||||
token, clientPublicKeyDerBuff, displayName, avatarBuff, async (errStr: string, dataMember: any, dataMetadata: any) => {
|
||||
if (errStr) {
|
||||
reject(new Error(errStr));
|
@ -14,7 +14,7 @@ export default class PersonalDB {
|
||||
private readonly db: sqlite.Database
|
||||
) {}
|
||||
|
||||
public async create(filePath: string): Promise<PersonalDB> {
|
||||
public static async create(filePath: string): Promise<PersonalDB> {
|
||||
return new PersonalDB(
|
||||
await sqlite.open({ driver: sqlite3.Database, filename: filePath })
|
||||
);
|
||||
|
@ -1,292 +0,0 @@
|
||||
import * as electronRemote from '@electron/remote';
|
||||
const electronConsole = electronRemote.getGlobal('console') as Console;
|
||||
import Logger from '../../logger/logger';
|
||||
const LOG = Logger.create(__filename, electronConsole);
|
||||
|
||||
LOG.silly('script begins');
|
||||
|
||||
import Controller from './controller';
|
||||
|
||||
import DBCache from './db-cache';
|
||||
import Globals from './globals';
|
||||
|
||||
import UI from './ui';
|
||||
import Actions from './actions';
|
||||
import { CacheServerData, Channel, ConnectionInfo, Member, Message, ServerMetaData } from './data-types';
|
||||
import ClientController from './client-controller';
|
||||
import Q from './q-module';
|
||||
import bindWindowButtonEvents from './elements/events-window-buttons';
|
||||
import bindTextInputEvents from './elements/events-text-input';
|
||||
import bindInfiniteScrollEvents from './elements/events-infinite-scroll';
|
||||
import bindConnectionEvents from './elements/events-connection';
|
||||
import bindAddServerTitleEvents from './elements/events-server-title';
|
||||
import bindAddServerEvents from './elements/events-add-server';
|
||||
|
||||
LOG.silly('modules loaded');
|
||||
|
||||
if (Globals.MESSAGES_PER_REQUEST >= Globals.MAX_CURRENT_MESSAGES) throw new Error('messages per request must be less than max current messages');
|
||||
|
||||
window.addEventListener('unhandledrejection', (e) => {
|
||||
LOG.error('Unhandled Promise Rejection', e.reason);
|
||||
});
|
||||
|
||||
window.addEventListener('error', (e) => {
|
||||
LOG.error('Uncaught Error', e.error);
|
||||
});
|
||||
|
||||
window.addEventListener('DOMContentLoaded', () => {
|
||||
document.body.classList.remove('preload');
|
||||
|
||||
(async () => {
|
||||
await LOG.ensureSourceMaps();
|
||||
LOG.silly('web client log source maps loaded');
|
||||
|
||||
const dbCache = await DBCache.connect();
|
||||
await dbCache.init();
|
||||
|
||||
LOG.silly('cache initialized');
|
||||
|
||||
const controller = new Controller(dbCache);
|
||||
await controller.init();
|
||||
|
||||
LOG.silly('controller initialized');
|
||||
|
||||
const q = new Q(document);
|
||||
const ui = new UI(document, q);
|
||||
|
||||
LOG.silly('action classes initialized');
|
||||
|
||||
bindWindowButtonEvents(q);
|
||||
bindTextInputEvents(document, q, ui);
|
||||
bindInfiniteScrollEvents(q, ui);
|
||||
bindConnectionEvents(document, q, ui);
|
||||
bindAddServerTitleEvents(document, q, ui);
|
||||
bindAddServerEvents(document, q, ui, controller);
|
||||
|
||||
LOG.silly('events bound');
|
||||
|
||||
// Add server icons
|
||||
await ui.setServers(controller, controller.servers);
|
||||
|
||||
if (controller.servers.length > 0) {
|
||||
// Click on the first server in the list
|
||||
q.$('#server-list .server').click();
|
||||
}
|
||||
|
||||
// Receive Current Channel Messages
|
||||
controller.on('new-message', async (guild: CombinedGuild, message: Message) => {
|
||||
if (ui.activeGuild === null || ui.activeGuild.id !== server.id) return;
|
||||
if (ui.activeChannel === null || ui.activeChannel.id !== message.channel.id) return;
|
||||
if (ui.messagesAtBottom) {
|
||||
// add the message to the bottom of the message feed
|
||||
await ui.addMessagesAfter(server, message.channel, [ message ], null);
|
||||
ui.jumpMessagesToBottom();
|
||||
} else if (message.member.id == server.memberId) {
|
||||
// this set of messages will include the new messageguildId
|
||||
LOG.debug('not at bottom, jumping down since message was sent by the current user');
|
||||
await Actions.fetchAndUpdateMessagesRecent(q, ui, server, message.channel);
|
||||
}
|
||||
});
|
||||
|
||||
controller.on('verified', async (guild: CombinedGuild) => {
|
||||
(async () => { // update connection info
|
||||
await Actions.fetchAndUpdateConnection(ui, server);
|
||||
})();
|
||||
(async () => { // refresh members cache
|
||||
if (server.members) {
|
||||
await server.fetchMembers();
|
||||
} else {
|
||||
await Actions.fetchAndUpdateMembers(q, ui, server);
|
||||
}
|
||||
})();
|
||||
(async () => { // refresh channels cache
|
||||
if (server.channels) {
|
||||
await server.fetchChannels();
|
||||
} else {
|
||||
await Actions.fetchAndUpdateChannels(q, ui, server);
|
||||
}
|
||||
})();
|
||||
(async () => { // refresh current channel messages
|
||||
if (ui.activeChannel === null) return;
|
||||
if (ui.messagePairs.size == 0) {
|
||||
// fetch messages again since there are no messages yet
|
||||
await Actions.fetchAndUpdateMessagesRecent(q, ui, server, ui.activeChannel);
|
||||
} else {
|
||||
// Just update the infinite scroll. NOTE: this will not remove deleted messages
|
||||
ui.messagesAtTop = false;
|
||||
ui.messagesAtBottom = false;
|
||||
(q.$('#channel-feed-content-wrapper') as any).updateInfiniteScroll();
|
||||
}
|
||||
})();
|
||||
});
|
||||
|
||||
controller.on('disconnected', (guild: CombinedGuild) => {
|
||||
(async () => {
|
||||
await Actions.fetchAndUpdateConnection(ui, server);
|
||||
})();
|
||||
(async () => {
|
||||
await Actions.fetchAndUpdateMembers(q, ui, server);
|
||||
})();
|
||||
});
|
||||
|
||||
controller.on('update-server', async (guild: CombinedGuild, serverData: ServerMetaData | CacheServerData) => {
|
||||
LOG.debug(`s#${server.id} metadata updated`)
|
||||
await ui.updateServerName(server, serverData.name);
|
||||
|
||||
// Not using withPotentialError since keeping the old icon is a fine fallback
|
||||
if (serverData.iconResourceId) {
|
||||
try {
|
||||
let iconBuff = await server.fetchResource(serverData.iconResourceId);
|
||||
await ui.updateServerIcon(server, iconBuff);
|
||||
} catch (e) {
|
||||
LOG.error('Error fetching new server icon', e);
|
||||
// Keep the old server icon, just log an error.
|
||||
// Should go through another try after a restart
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
controller.on('deleted-members', async (guild: CombinedGuild, members: Member[]) => {
|
||||
LOG.debug(members.length + ' deleted members');
|
||||
await ui.deleteMembers(server, members);
|
||||
});
|
||||
|
||||
controller.on('updated-members', async (guild: CombinedGuild, data: { oldMember: Member, newMember: Member }[]) => {
|
||||
LOG.debug(data.length + ' updated members s#' + server.id);
|
||||
await ui.updateMembers(server, data);
|
||||
if (
|
||||
ui.activeConnection !== null &&
|
||||
data.find(change => change.newMember.id === (ui.activeConnection as ConnectionInfo).id)
|
||||
) {
|
||||
await Actions.fetchAndUpdateConnection(ui, server);
|
||||
}
|
||||
});
|
||||
|
||||
controller.on('added-members', async (guild: CombinedGuild, members: Member[]) => {
|
||||
LOG.debug(members.length + ' added members');
|
||||
await ui.addMembers(server, members);
|
||||
});
|
||||
|
||||
controller.on('deleted-channels', async (guild: CombinedGuild, channels: Channel[]) => {
|
||||
LOG.debug(channels.length + ' deleted channels');
|
||||
await ui.deleteChannels(server, channels);
|
||||
});
|
||||
|
||||
controller.on('updated-channels', async (guild: CombinedGuild, data: { oldChannel: Channel, newChannel: Channel }[]) => {
|
||||
LOG.debug(data.length + ' updated channels');
|
||||
await ui.updateChannels(server, data);
|
||||
});
|
||||
|
||||
controller.on('added-channels', async (guild: CombinedGuild, channels: Channel[]) => {
|
||||
LOG.debug(channels.length + ' added channels');
|
||||
await ui.addChannels(server, channels);
|
||||
});
|
||||
|
||||
controller.on('deleted-messages', async (guild: CombinedGuild, channel: Channel, messages: Message[]) => {
|
||||
LOG.debug(messages.length + ' deleted messages');
|
||||
//LOG.debug('deleted messages:', { messages: deletedMessages.map(message => message.text) });
|
||||
// messages were deleted but the cache still had them
|
||||
await ui.deleteMessages(server, channel, messages);
|
||||
});
|
||||
|
||||
controller.on('updated-messages', async (guild: CombinedGuild, channel: Channel, data: { oldMessage: Message, newMessage: Message }[]) => {
|
||||
LOG.debug(data.length + ' updated messages');
|
||||
// messages were updated on the server-side
|
||||
await ui.updateMessages(server, channel, data);
|
||||
});
|
||||
|
||||
controller.on('added-messages', async (guild: CombinedGuild, channel: Channel, addedAfter: Map<string, Message>, addedBefore: Map<string, Message>) => {
|
||||
LOG.debug(addedAfter.size + ' added messages'); // addedBefore.size should equal addedAfter.size
|
||||
//LOG.debug('added messages', { messages: Array.from(addedAfter.values()).map(message => message.text) });
|
||||
// messages were added in a place that the cache did not have them
|
||||
|
||||
if (!ui.isMessagePairsServer(server)) return; // these messages are not from the ones in the feed
|
||||
if (!ui.isMessagePairsChannel(channel)) return; // these messages are not from the ones in the feed
|
||||
|
||||
let currentMessagesSorted = Array.from(ui.messagePairs.values()).sort((a, b) => {
|
||||
return a.message.sent.getTime() - b.message.sent.getTime();
|
||||
});
|
||||
|
||||
// length could be 0 if all previous messages were 'deleted'
|
||||
if (currentMessagesSorted.length == 0) {
|
||||
// Simply set the text channel messages rather than calculating where to add them
|
||||
ui.setMessages(server, channel, Array.from(addedAfter.values()), { atTop: false, atBottom: true });
|
||||
return;
|
||||
}
|
||||
|
||||
let firstMessage = currentMessagesSorted[0].message;
|
||||
let lastMessage = currentMessagesSorted[currentMessagesSorted.length - 1].message;
|
||||
|
||||
// messages that were updated (currently extraneous)
|
||||
// Note: this should never happen since these should be handled by updated-messages
|
||||
// Note: this may be used to replace dummy pre-sent messages
|
||||
let toUpdate: { newMessage: Message, oldMessage: Message }[] = [];
|
||||
for (let addedMessage of addedBefore.values()) {
|
||||
if (ui.messagePairs.has(addedMessage.id)) {
|
||||
toUpdate.push({
|
||||
newMessage: addedMessage,
|
||||
oldMessage: (ui.messagePairs.get(addedMessage.id) as { message: Message, element: HTMLElement }).message
|
||||
});
|
||||
}
|
||||
}
|
||||
if (toUpdate.length > 0) {
|
||||
LOG.warn('updating messages in added-messages... this was intended to be extraneous...', { toUpdate: toUpdate, toUpdateLength: toUpdate.length });
|
||||
await ui.updateMessages(server, channel, toUpdate);
|
||||
}
|
||||
|
||||
// messages before the first message
|
||||
let toAddBefore: Message[] = [];
|
||||
let nextFirstMessage = firstMessage;
|
||||
while (addedBefore.has(nextFirstMessage.id)) {
|
||||
nextFirstMessage = addedBefore.get(nextFirstMessage.id) as Message;
|
||||
toAddBefore.unshift(nextFirstMessage);
|
||||
}
|
||||
if (toAddBefore.length > 0) {
|
||||
LOG.debug('adding ' + toAddBefore.length + ' before');
|
||||
await ui.addMessagesBefore(server, channel, toAddBefore, firstMessage);
|
||||
}
|
||||
|
||||
// messages after the last message
|
||||
let toAddAfter: Message[] = [];
|
||||
let nextLastMessage = lastMessage;
|
||||
while (addedAfter.has(nextLastMessage.id)) {
|
||||
nextLastMessage = addedAfter.get(nextLastMessage.id) as Message;
|
||||
toAddAfter.push(nextLastMessage);
|
||||
}
|
||||
if (toAddAfter.length > 0) {
|
||||
LOG.debug('adding ' + toAddAfter.length + ' after');
|
||||
await ui.addMessagesAfter(server, channel, toAddAfter, lastMessage);
|
||||
}
|
||||
|
||||
// messages added between messages already in the feed
|
||||
let toAddBetween: { messageTop: HTMLElement, messageBottom: HTMLElement, betweenMessages: Message[] }[] = [];
|
||||
for (let i = 1; i < currentMessagesSorted.length; ++i) {
|
||||
let messageTop = currentMessagesSorted[i - 1].message;
|
||||
let messageBottom = currentMessagesSorted[i].message;
|
||||
let betweenMessages: Message[] = [];
|
||||
let followMessage = messageTop;
|
||||
// this should never be null as long as there is an addedAfter (will throw error if this is not the case)
|
||||
let lastFollowMessage = addedBefore.get(messageBottom.id) as Message;
|
||||
while (addedAfter.has(followMessage.id) && followMessage.id != lastFollowMessage.id) {
|
||||
followMessage = addedAfter.get(followMessage.id) as Message;
|
||||
betweenMessages.push(followMessage);
|
||||
}
|
||||
if (betweenMessages.length > 0) {
|
||||
toAddBetween.push({
|
||||
messageTop: (ui.messagePairs.get(messageTop.id) as { message: Message, element: HTMLElement }).element,
|
||||
messageBottom: (ui.messagePairs.get(messageBottom.id) as { message: Message, element: HTMLElement }).element,
|
||||
betweenMessages: betweenMessages
|
||||
});
|
||||
}
|
||||
}
|
||||
// add messages in between
|
||||
if (toAddBetween.length > 0) {
|
||||
LOG.debug('adding ' + toAddBetween.length + ' between sets');
|
||||
}
|
||||
for (let messageSet of toAddBetween) {
|
||||
await ui.addMessagesBetween(server, channel, messageSet.betweenMessages, messageSet.messageTop, messageSet.messageBottom);
|
||||
}
|
||||
});
|
||||
})();
|
||||
});
|
||||
|
@ -1,11 +1,12 @@
|
||||
import * as crypto from 'crypto';
|
||||
import * as socketio from 'socket.io-client';
|
||||
import { EventEmitter } from 'tsee';
|
||||
|
||||
import DedupAwaiter from "./dedup-awaiter";
|
||||
import Util from './util';
|
||||
|
||||
// Automatically re-verifies the socket when connected
|
||||
export default class SocketVerifier {
|
||||
export default class SocketVerifier extends EventEmitter<{ 'verified': () => void }> {
|
||||
public isVerified = false;
|
||||
private memberId: string | null = null;
|
||||
private verifyDedup = new DedupAwaiter(async () => { return await this.doVerify(); });
|
||||
@ -15,6 +16,7 @@ export default class SocketVerifier {
|
||||
private publicKey: crypto.KeyObject,
|
||||
private privateKey: crypto.KeyObject
|
||||
) {
|
||||
super();
|
||||
socket.on('connect', async () => {
|
||||
await this.verify();
|
||||
});
|
||||
@ -23,7 +25,6 @@ export default class SocketVerifier {
|
||||
});
|
||||
}
|
||||
|
||||
// TODO: Move this to a "query/queryDedup" request
|
||||
/** Verifies this client with the server. This function must be called before the server will send messages or give results */
|
||||
private async doVerify(): Promise<string> {
|
||||
if (this.socket.disconnected) throw new Error('socket is disconnected');
|
||||
@ -48,6 +49,7 @@ export default class SocketVerifier {
|
||||
}
|
||||
this.memberId = null;
|
||||
resolve(memberId);
|
||||
this.emit('verified');
|
||||
});
|
||||
});
|
||||
});
|
||||
|
@ -15,7 +15,7 @@ import Q from './q-module';
|
||||
import createGuildListGuild from './elements/guild-list-guild';
|
||||
import createChannel from './elements/channel';
|
||||
import createMember from './elements/member';
|
||||
import Controller from './controller';
|
||||
import GuildsManager from './guilds-manager';
|
||||
import createMessage from './elements/message';
|
||||
|
||||
interface SetMessageProps {
|
||||
@ -161,20 +161,20 @@ export default class UI {
|
||||
});
|
||||
}
|
||||
|
||||
public async setGuilds(controller: Controller, guilds: CombinedGuild[]): Promise<void> {
|
||||
public async setGuilds(guildsManager: GuildsManager, guilds: CombinedGuild[]): Promise<void> {
|
||||
await this._guildsLock.push(() => {
|
||||
Q.clearChildren(this.q.$('#guild-list'));
|
||||
for (let guild of guilds) {
|
||||
let element = createGuildListGuild(this.document, this.q, this, controller, guild);
|
||||
let element = createGuildListGuild(this.document, this.q, this, guildsManager, guild);
|
||||
this.q.$('#guild-list').appendChild(element);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
public async addGuild(controller: Controller, guild: CombinedGuild): Promise<HTMLElement> {
|
||||
public async addGuild(guildsManager: GuildsManager, guild: CombinedGuild): Promise<HTMLElement> {
|
||||
let element: HTMLElement | null = null;
|
||||
await this._guildsLock.push(() => {
|
||||
element = createGuildListGuild(this.document, this.q, this, controller, guild) as HTMLElement;
|
||||
element = createGuildListGuild(this.document, this.q, this, guildsManager, guild) as HTMLElement;
|
||||
this.q.$('#guild-list').appendChild(element);
|
||||
});
|
||||
if (element == null) throw new ShouldNeverHappenError('element was not set');
|
||||
@ -262,22 +262,22 @@ export default class UI {
|
||||
});
|
||||
}
|
||||
|
||||
public async updateChannels(guild: CombinedGuild, data: { oldChannel: Channel, newChannel: Channel }[]): Promise<void> {
|
||||
public async updateChannels(guild: CombinedGuild, updatedChannels: Channel[]): Promise<void> {
|
||||
await this.lockChannels(guild, async () => {
|
||||
for (const { oldChannel, newChannel } of data) {
|
||||
let oldElement = this.q.$('#channel-list .channel[meta-id="' + newChannel.id + '"]');
|
||||
let newElement = createChannel(this.document, this.q, this, guild, newChannel);
|
||||
for (const channel of updatedChannels) {
|
||||
let oldElement = this.q.$('#channel-list .channel[meta-id="' + channel.id + '"]');
|
||||
let newElement = createChannel(this.document, this.q, this, guild, channel);
|
||||
oldElement.parentElement?.replaceChild(newElement, oldElement);
|
||||
await this.updateChannelPosition(guild, newElement);
|
||||
|
||||
if (this.activeChannel !== null && this.activeChannel.id === newChannel.id) {
|
||||
if (this.activeChannel !== null && this.activeChannel.id === channel.id) {
|
||||
newElement.classList.add('active');
|
||||
|
||||
// See also setActiveChannel
|
||||
this.q.$('#channel-name').innerText = newChannel.name;
|
||||
this.q.$('#channel-flavor-text').innerText = newChannel.flavorText ?? '';
|
||||
this.q.$('#channel-flavor-divider').style.visibility = newChannel.flavorText ? 'visible' : 'hidden';
|
||||
this.q.$('#text-input').setAttribute('placeholder', 'Message #' + newChannel.name);
|
||||
this.q.$('#channel-name').innerText = channel.name;
|
||||
this.q.$('#channel-flavor-text').innerText = channel.flavorText ?? '';
|
||||
this.q.$('#channel-flavor-divider').style.visibility = channel.flavorText ? 'visible' : 'hidden';
|
||||
this.q.$('#text-input').setAttribute('placeholder', 'Message #' + channel.name);
|
||||
}
|
||||
}
|
||||
});
|
||||
@ -358,24 +358,26 @@ export default class UI {
|
||||
});
|
||||
}
|
||||
|
||||
public async updateMembers(guild: CombinedGuild, data: { oldMember: Member, newMember: Member }[]): Promise<void> {
|
||||
public async updateMembers(guild: CombinedGuild, updatedMembers: Member[]): Promise<void> {
|
||||
await this.lockMembers(guild, async () => {
|
||||
for (const { oldMember, newMember } of data) {
|
||||
let oldElement = this.q.$_('#guild-members .member[meta-id="' + newMember.id + '"]');
|
||||
for (const member of updatedMembers) {
|
||||
let oldElement = this.q.$_('#guild-members .member[meta-id="' + member.id + '"]');
|
||||
if (oldElement) {
|
||||
let newElement = createMember(this.q, guild, newMember);
|
||||
let newElement = createMember(this.q, guild, member);
|
||||
oldElement.parentElement?.replaceChild(newElement, oldElement);
|
||||
await this.updateMemberPosition(guild, newElement);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// Update the messages too
|
||||
if (this.activeChannel === null) return;
|
||||
await this.lockMessages(guild, this.activeChannel, () => {
|
||||
for (const { oldMember, newMember } of data) {
|
||||
let newStyle = newMember.roleColor ? 'color: ' + newMember.roleColor : null;
|
||||
let newName = newMember.displayName;
|
||||
for (const member of updatedMembers) {
|
||||
let newStyle = member.roleColor ? 'color: ' + member.roleColor : null;
|
||||
let newName = member.displayName;
|
||||
// the extra query selectors may be overkill
|
||||
for (let messageElement of this.q.$$(`.message[meta-member-id="${newMember.id}"]`)) {
|
||||
for (let messageElement of this.q.$$(`.message[meta-member-id="${member.id}"]`)) {
|
||||
let nameElement = this.q.$$$_(messageElement, '.member-name');
|
||||
if (nameElement) { // continued messages will still show up but need to be skipped
|
||||
if (newStyle) nameElement.setAttribute('style', newStyle);
|
||||
@ -408,7 +410,32 @@ export default class UI {
|
||||
return element && this.messagePairs.get(element.getAttribute('meta-id')) || null;
|
||||
}
|
||||
|
||||
public async addMessagesBefore(guild: CombinedGuild, channel: Channel, messages: Message[], prevTopMessage: Message | null): Promise<void> {
|
||||
public async addMessages(guild: CombinedGuild, messages: Message[]) {
|
||||
let channelIds = new Set(messages.map(message => message.channel.id));
|
||||
for (let channelId of channelIds) {
|
||||
let channelMessages = messages.filter(message => message.channel.id === channelId);
|
||||
channelMessages = channelMessages.sort((a, b) => a.sent.getTime() - b.sent.getTime());
|
||||
|
||||
// No Previous Messages is an easy case
|
||||
if (this.messagePairs.size === 0) {
|
||||
await this.addMessagesBefore(guild, { id: channelId }, channelMessages, null);
|
||||
continue;
|
||||
}
|
||||
|
||||
let topMessagePair = this.getTopMessagePair() as { message: Message, element: HTMLElement };
|
||||
let bottomMessagePair = this.getBottomMessagePair() as { message: Message, element: HTMLElement };
|
||||
|
||||
let aboveMessages = messages.filter(message => message.sent < topMessagePair.message.sent);
|
||||
let belowMessages = messages.filter(message => message.sent > bottomMessagePair.message.sent);
|
||||
let betweenMessages = messages.filter(message => message.sent >= topMessagePair.message.sent && message.sent <= bottomMessagePair.message.sent);
|
||||
|
||||
if (aboveMessages.length > 0) await this.addMessagesBefore(guild, { id: channelId }, aboveMessages, topMessagePair.message);
|
||||
if (belowMessages.length > 0) await this.addMessagesAfter(guild, { id: channelId }, belowMessages, bottomMessagePair.message);
|
||||
if (betweenMessages.length > 0) await this.addMessagesBetween(guild, { id: channelId }, betweenMessages, topMessagePair.element, bottomMessagePair.element);
|
||||
}
|
||||
}
|
||||
|
||||
public async addMessagesBefore(guild: CombinedGuild, channel: Channel | { id: string }, messages: Message[], prevTopMessage: Message | null): Promise<void> {
|
||||
this.lockMessages(guild, channel, () => {
|
||||
if (prevTopMessage && this.getTopMessagePair()?.message.id !== prevTopMessage.id) return;
|
||||
|
||||
@ -495,7 +522,7 @@ export default class UI {
|
||||
}
|
||||
|
||||
// TODO: use topMessage, bottomMessage / topMessageId, bottomMessageId instead?
|
||||
public async addMessagesBetween(guild: CombinedGuild, channel: Channel, messages: Message[], topElement: HTMLElement, bottomElement: HTMLElement): Promise<void> {
|
||||
private async addMessagesBetween(guild: CombinedGuild, channel: Channel | { id: string }, messages: Message[], topElement: HTMLElement, bottomElement: HTMLElement): Promise<void> {
|
||||
await this.lockMessages(guild, channel, () => {
|
||||
if (!(messages.length > 0 && topElement != null && bottomElement != null && bottomElement == Q.nextElement(topElement))) {
|
||||
LOG.error('invalid messages between', { messages, top: topElement.innerText, bottom: bottomElement.innerText, afterTop: Q.nextElement(topElement)?.innerText });
|
||||
@ -618,9 +645,12 @@ export default class UI {
|
||||
this.messagesAtBottom = true;
|
||||
}
|
||||
|
||||
public async deleteMessages(guild: CombinedGuild, channel: Channel, messages: Message[]) {
|
||||
await this.lockMessages(guild, channel, () => {
|
||||
for (let message of messages) {
|
||||
public async deleteMessages(guild: CombinedGuild, messages: Message[]) {
|
||||
let channelIds = new Set(messages.map(message => message.channel.id));
|
||||
for (let channelId of channelIds) {
|
||||
let channelMessages = messages.filter(message => message.channel.id === channelId);
|
||||
await this.lockMessages(guild, { id: channelId }, () => {
|
||||
for (let message of channelMessages) {
|
||||
if (this.messagePairs.has(message.id)) {
|
||||
let messagePair = this.messagePairs.get(message.id) as { message: Message, element: HTMLElement };
|
||||
messagePair.element.parentElement?.removeChild(messagePair.element);
|
||||
@ -631,23 +661,28 @@ export default class UI {
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
public async updateMessages(guild: CombinedGuild, channel: Channel, data: { oldMessage: Message, newMessage: Message }[]): Promise<void> {
|
||||
await this.lockMessages(guild, channel, () => {
|
||||
for (const { oldMessage, newMessage } of data) {
|
||||
if (this.messagePairs.has(oldMessage.id)) {
|
||||
let oldElement = (this.messagePairs.get(oldMessage.id) as { message: Message, element: HTMLElement }).element;
|
||||
public async updateMessages(guild: CombinedGuild, updatedMessages: Message[]): Promise<void> {
|
||||
let channelIds = new Set(updatedMessages.map(message => message.channel.id));
|
||||
for (let channelId of channelIds) {
|
||||
let channelMessages = updatedMessages.filter(message => message.channel.id === channelId);
|
||||
await this.lockMessages(guild, { id: channelId }, () => {
|
||||
for (const message of channelMessages) {
|
||||
if (this.messagePairs.has(message.id)) {
|
||||
let oldElement = (this.messagePairs.get(message.id) as { message: Message, element: HTMLElement }).element;
|
||||
let prevElement = Q.previousElement(oldElement);
|
||||
let prevMessage = prevElement && (this.messagePairs.get(prevElement.getAttribute('meta-id')) as { message: Message, element: HTMLElement }).message;
|
||||
let newElement = createMessage(this.document, this.q, guild, newMessage, prevMessage);
|
||||
let newElement = createMessage(this.document, this.q, guild, message, prevMessage);
|
||||
oldElement.parentElement?.replaceChild(newElement, oldElement);
|
||||
// TODO: we should be updating messages sent below this message
|
||||
// however, these events should be relatively rare so that's for the future
|
||||
this.messagePairs.set(oldMessage.id, { message: newMessage, element: newElement });
|
||||
this.messagePairs.set(message.id, { message: message, element: newElement });
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
public async addMessagesErrorIndicatorBefore(guild: CombinedGuild, channel: Channel, errorIndicatorElement: HTMLElement): Promise<void> {
|
||||
await this.lockMessages(guild, channel, () => {
|
||||
|
Loading…
Reference in New Issue
Block a user