156 lines
7.1 KiB
TypeScript
156 lines
7.1 KiB
TypeScript
|
import * as electronRemote from '@electron/remote';
|
||
|
const electronConsole = electronRemote.getGlobal('console') as Console;
|
||
|
import Logger from '../../../logger/logger';
|
||
|
const LOG = new Logger(__filename, electronConsole);
|
||
|
|
||
|
import ClientController from "../client-controller";
|
||
|
import Globals from "../globals";
|
||
|
import BaseElements from "./require/base-elements";
|
||
|
import IState from "./require/elements-state";
|
||
|
import ElementsUtil from "./require/elements-util";
|
||
|
|
||
|
import { $, $$, $$$, $$$$ } from './require/q-module';
|
||
|
|
||
|
export default function createCreateChannelOverlay(state: IState, server: ClientController): HTMLElement {
|
||
|
const { document, ui } = state;
|
||
|
$.setDocument(document);
|
||
|
|
||
|
// See also overlay-modify-channel
|
||
|
|
||
|
let element = BaseElements.createOverlay(document, { class: 'content submit-dialog modify-channel', content: [
|
||
|
{ class: 'preview channel-title', content: [
|
||
|
{ class: 'channel-icon', content: BaseElements.TEXT_CHANNEL_ICON },
|
||
|
{ class: 'channel-name', content: 'channel-name' },
|
||
|
{ class: 'channel-flavor-divider' },
|
||
|
{ class: 'channel-flavor-text', content: '' }
|
||
|
] },
|
||
|
{ class: 'text-input channel-name', placeholder: 'channel-name', content: '', contenteditable: 'plaintext-only' },
|
||
|
{ class: 'text-input channel-flavor-text', placeholder: 'Flavor Text (optional)', content: '' || '', contenteditable: 'plaintext-only' },
|
||
|
{ class: 'lower', content: [
|
||
|
{ class: 'error' },
|
||
|
{ class: 'buttons', content: [
|
||
|
{ class: 'button submit', content: 'Create Channel' }
|
||
|
] }
|
||
|
] }
|
||
|
] });
|
||
|
|
||
|
let newName = '';
|
||
|
let newFlavorText: string | null = '';
|
||
|
function updatePreview() {
|
||
|
newName = $$$(element, '.text-input.channel-name').innerText;
|
||
|
newFlavorText = $$$(element, '.text-input.channel-flavor-text').innerText;
|
||
|
$$$(element, '.channel-title .channel-name').innerText = newName;
|
||
|
$$$(element, '.channel-title .channel-flavor-text').innerText = newFlavorText;
|
||
|
|
||
|
if (newFlavorText != '') {
|
||
|
$$$(element, '.channel-title .channel-flavor-divider').style.visibility = 'visible';
|
||
|
} else {
|
||
|
$$$(element, '.channel-title .channel-flavor-divider').style.visibility = 'hidden';
|
||
|
}
|
||
|
}
|
||
|
updatePreview();
|
||
|
|
||
|
let submitting = false;
|
||
|
async function submit() {
|
||
|
if (submitting) return;
|
||
|
submitting = true;
|
||
|
|
||
|
$$$(element, '.error').innerText = '';
|
||
|
$$$(element, '.button.submit').innerText = 'Submitting...';
|
||
|
$$$(element, '.text-input.channel-name').removeAttribute('contenteditable');
|
||
|
$$$(element, '.text-input.channel-flavor-text').removeAttribute('contenteditable');
|
||
|
|
||
|
let success = false;
|
||
|
if (newName.length == 0) {
|
||
|
LOG.warn('attempted to set empty channel name');
|
||
|
$$$(element, '.error').innerText = 'Channel name cannot be empty';
|
||
|
$$$(element, '.button.submit').innerText = 'Try Again';
|
||
|
await ElementsUtil.shakeElement($$$(element, '.button.submit'), 400);
|
||
|
} else if (newName.length > Globals.MAX_CHANNEL_NAME_LENGTH) {
|
||
|
LOG.warn('attempted to set too long channel name');
|
||
|
$$$(element, '.error').innerText = 'Channel name is too long. ' + newName.length + ' > ' + Globals.MAX_CHANNEL_NAME_LENGTH;
|
||
|
$$$(element, '.button.submit').innerText = 'Try Again';
|
||
|
await ElementsUtil.shakeElement($$$(element, '.button.submit'), 400);
|
||
|
} else if (!(/^[A-Za-z0-9-]+$/.exec(newName))) {
|
||
|
LOG.warn('attempted to set channel name with illegal characters');
|
||
|
$$$(element, '.error').innerText = 'Please use only [A-Za-z0-9-]+ in channel name';
|
||
|
$$$(element, '.button.submit').innerText = 'Try Again';
|
||
|
await ElementsUtil.shakeElement($$$(element, '.button.submit'), 400);
|
||
|
} else if (newFlavorText != null && newFlavorText.length > Globals.MAX_CHANNEL_FLAVOR_TEXT_LENGTH) {
|
||
|
LOG.warn('attempted to set too long flavor text');
|
||
|
$$$(element, '.error').innerText = 'Flavor text is too long. ' + newName.length + ' > ' + Globals.MAX_CHANNEL_NAME_LENGTH;
|
||
|
$$$(element, '.button.submit').innerText = 'Try Again';
|
||
|
await ElementsUtil.shakeElement($$$(element, '.button.submit'), 400);
|
||
|
} else {
|
||
|
if (newFlavorText != null && newFlavorText.length == 0) {
|
||
|
newFlavorText = null;
|
||
|
}
|
||
|
try {
|
||
|
await server.createChannel(newName, newFlavorText);
|
||
|
success = true;
|
||
|
} catch (e) {
|
||
|
LOG.error('error updating channel', e);
|
||
|
$$$(element, '.error').innerText = 'Error updating channel';
|
||
|
$$$(element, '.button.submit').innerText = 'Try Again';
|
||
|
await ElementsUtil.shakeElement($$$(element, '.button.submit'), 400);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (success) {
|
||
|
element.removeSelf();
|
||
|
}
|
||
|
|
||
|
$$$(element, '.text-input.channel-name').setAttribute('contenteditable', 'plaintext-only');
|
||
|
$$$(element, '.text-input.channel-flavor-text').setAttribute('contenteditable', 'plaintext-only');
|
||
|
|
||
|
submitting = false;
|
||
|
}
|
||
|
|
||
|
let textInputs = $$$$(element, '.text-input');
|
||
|
for (let textInput of textInputs) {
|
||
|
textInput.addEventListener('input', () => {
|
||
|
updatePreview();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
$$$(element, '.text-input.channel-name').addEventListener('keydown', async (e) => {
|
||
|
if (e.key == 'Backspace' || e.key == 'Escape' || e.key == 'F4') {
|
||
|
// these keys are good
|
||
|
} else if (e.key == 'Tab') {
|
||
|
// have to hard-code this one because otherwise, it just picks the beginning of the next input
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
$$$(element, '.text-input.channel-flavor-text').focus();
|
||
|
ElementsUtil.setCursorToEnd($$$(element, '.text-input.channel-flavor-text'));
|
||
|
} else if (e.key == 'Enter') {
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
await submit();
|
||
|
} else if (!/^[A-Za-z0-9-]$/.exec(e.key)) {
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
}
|
||
|
});
|
||
|
|
||
|
$$$(element, '.text-input.channel-flavor-text').addEventListener('keydown', async (e) => {
|
||
|
if (e.key == 'Tab' && e.shiftKey) {
|
||
|
// have to hard-code this one because otherwise, it just picks the beginning of the next input
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
$$$(element, '.text-input.channel-name').focus();
|
||
|
ElementsUtil.setCursorToEnd($$$(element, '.text-input.channel-name'));
|
||
|
} else if (e.key == 'Enter') {
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
await submit();
|
||
|
}
|
||
|
});
|
||
|
|
||
|
$$$(element, '.button.submit').addEventListener('click', async () => {
|
||
|
await submit();
|
||
|
});
|
||
|
|
||
|
return element;
|
||
|
}
|
||
|
|
||
|
module.exports = createCreateChannelOverlay;
|