jsx is for professionals
This commit is contained in:
parent
6979c450e6
commit
6b17f569bb
@ -19,6 +19,9 @@ import UI from '../ui';
|
|||||||
import GuildsManager from '../guilds-manager';
|
import GuildsManager from '../guilds-manager';
|
||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
import ReactHelper from './require/react-helper';
|
||||||
|
|
||||||
export interface IAddGuildData {
|
export interface IAddGuildData {
|
||||||
name: string,
|
name: string,
|
||||||
url: string,
|
url: string,
|
||||||
@ -58,45 +61,44 @@ export default function createAddGuildOverlay(document: Document, q: Q, ui: UI,
|
|||||||
|
|
||||||
//LOG.debug('addguilddata:', { addGuildData });
|
//LOG.debug('addguilddata:', { addGuildData });
|
||||||
|
|
||||||
const element = BaseElements.createOverlay(document, {
|
|
||||||
class: 'content add-guild', content: [
|
const element = BaseElements.createOverlay(document, (
|
||||||
{ class: 'preview', content: [
|
<div className="content add-guild">
|
||||||
{ tag: 'img', class: 'icon', src: addGuildData.iconSrc, alt: 'icon' },
|
<div className="preview">
|
||||||
{ content: [
|
<img className="icon" src={addGuildData.iconSrc} alt="icon"></img>
|
||||||
{ class: 'name', content: addGuildData.name },
|
<div>
|
||||||
{ class: 'url', content: addGuildData.url },
|
<div className="name">{addGuildData.name}</div>
|
||||||
{ class: 'expires',
|
<div className="url">{addGuildData.url}</div>
|
||||||
content: (expired ? 'Invite Expired ' : 'Invite Expires ') + moment(addGuildData.expires).fromNow() }
|
<div className="expires">{(expired ? 'Invite Expired ' : 'Invite Expires ') + moment(addGuildData.expires).fromNow()}</div>
|
||||||
] }
|
</div>
|
||||||
] },
|
</div>
|
||||||
{ class: 'divider' },
|
<div className="divider"></div>
|
||||||
{ class: 'message-preview message', content: [
|
<div className="message-preview message">
|
||||||
{ class: 'member-avatar', content: { tag: 'img', src: './img/loading.svg', alt: 'avatar' } },
|
<div className="member-avatar"><img src="./img/loading.svg" alt="avatar"></img></div>
|
||||||
{ class: 'right', content: [
|
<div className="right">
|
||||||
{ class: 'header', content: [
|
<div className="header">
|
||||||
{ class: 'member-name', content: displayName },
|
<div className="member-name">{displayName}</div>
|
||||||
{ class: 'timestamp', content: moment().calendar(ElementsUtil.calendarFormats) }
|
<div className="timestamp">{moment().calendar(ElementsUtil.calendarFormats)}</div>
|
||||||
] },
|
</div>
|
||||||
{ class: 'content text', content: 'Example Message' }
|
<div className="content text">What's up, gamers?</div>
|
||||||
] }
|
</div>
|
||||||
] },
|
</div>
|
||||||
{ class: 'display-name-input', placeholder: 'Display Name',
|
<div className="display-name-input" placeholder="Display Name" spellCheck="false"
|
||||||
spellcheck: 'false', contenteditable: 'plaintext-only', content: displayName },
|
contentEditable={'plaintext-only' as unknown as boolean /* React doesn't have plaintext-only in its types (https://github.com/DefinitelyTyped/DefinitelyTyped/pull/54779) */}></div>
|
||||||
{ class: 'avatar-input', content: [
|
<div className="avatar-input">
|
||||||
{ tag: 'label', class: 'avatar-upload-label button', content: [
|
<label className="avatar-upload-label button">
|
||||||
'Select Avatar',
|
Select Avatar
|
||||||
{ class: 'avatar-upload', tag: 'input',
|
<input className="avatar-upload" type="file" accept=".png,.jpg.,.jpeg" style={{ display: 'none' }}></input>
|
||||||
type: 'file', accept: '.png,.jpg,.jpeg', style: 'display: none;' },
|
</label>
|
||||||
] }
|
</div>
|
||||||
] },
|
<div className="lower">
|
||||||
{ class: 'lower', content: [
|
<div className="error"></div>
|
||||||
{ class: 'error' },
|
<div className="buttons">
|
||||||
{ class: 'buttons', content: [
|
<div className="button submit">Add Guild</div>
|
||||||
{ class: 'button submit', content: 'Add Guild' }
|
</div>
|
||||||
] }
|
</div>
|
||||||
] }
|
</div>
|
||||||
]
|
));
|
||||||
});
|
|
||||||
|
|
||||||
let avatarBuff: Buffer | null;
|
let avatarBuff: Buffer | null;
|
||||||
let defaultAvatarBuff: Buffer | null;
|
let defaultAvatarBuff: Buffer | null;
|
@ -10,26 +10,30 @@ import BaseElements from "./require/base-elements";
|
|||||||
import Q from '../q-module';
|
import Q from '../q-module';
|
||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
export default function createCreateChannelOverlay(document: Document, q: Q, guild: CombinedGuild): HTMLElement {
|
export default function createCreateChannelOverlay(document: Document, q: Q, guild: CombinedGuild): HTMLElement {
|
||||||
|
|
||||||
// See also overlay-modify-channel
|
// See also overlay-modify-channel
|
||||||
|
|
||||||
const element = BaseElements.createOverlay(document, { class: 'content submit-dialog modify-channel', content: [
|
const element = BaseElements.createOverlay(document, (
|
||||||
{ class: 'preview channel-title', content: [
|
<div className="content submit-dialog modify-channel">
|
||||||
{ class: 'channel-icon', content: BaseElements.Q_TEXT_CHANNEL_ICON },
|
<div className="preview channel-title">
|
||||||
{ class: 'channel-name', content: 'channel-name' },
|
<div className="channel-icon">{BaseElements.TEXT_CHANNEL_ICON}</div>
|
||||||
{ class: 'channel-flavor-divider' },
|
<div className="channel-name">channel-name</div>
|
||||||
{ class: 'channel-flavor-text', content: '' }
|
<div className="channel-flavor-divider"></div>
|
||||||
] },
|
<div className="channel-flavor-text"></div>
|
||||||
{ class: 'text-input channel-name', 'data-placeholder': 'channel-name', content: '', contenteditable: 'plaintext-only' },
|
</div>
|
||||||
{ class: 'text-input channel-flavor-text', 'data-placeholder': 'Flavor Text (optional)', content: '' || '', contenteditable: 'plaintext-only' },
|
<div className="text-input channel-name" data-placeholder="channel-name" contentEditable={'plaintext-only' as unknown as boolean /* React doesn't have plaintext-only in its typings (https://github.com/DefinitelyTyped/DefinitelyTyped/pull/54779) */}></div>
|
||||||
{ class: 'lower', content: [
|
<div className="text-input channel-flavor-text" data-placeholder="Flavor Text (optional)" contentEditable={'plaintext-only' as unknown as boolean /* React doesn't have plaintext-only in its typings (https://github.com/DefinitelyTyped/DefinitelyTyped/pull/54779) */}></div>
|
||||||
{ class: 'error' },
|
<div className="lower">
|
||||||
{ class: 'buttons', content: [
|
<div className="error"></div>
|
||||||
{ class: 'button submit', content: 'Create Channel' }
|
<div className="buttons">
|
||||||
] }
|
<div className="button submit">Create Channel</div>
|
||||||
] }
|
</div>
|
||||||
] });
|
</div>
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
|
||||||
let newName = '';
|
let newName = '';
|
||||||
let newFlavorText: string | null = '';
|
let newFlavorText: string | null = '';
|
@ -1,24 +0,0 @@
|
|||||||
import CombinedGuild from "../guild-combined";
|
|
||||||
import BaseElements from "./require/base-elements";
|
|
||||||
|
|
||||||
export default function createCreateInviteTokenOverlay(document: Document, guild: CombinedGuild): HTMLElement {
|
|
||||||
const element = BaseElements.createOverlay(document, { class: 'content submit-dialog', content: [
|
|
||||||
{ class: 'role-select category-select', content: [
|
|
||||||
{ class: 'label', content: 'Select Starting Roles' },
|
|
||||||
{ class: 'categories', content: [
|
|
||||||
{ class: 'category suggestion-1', content: '+Gamer' },
|
|
||||||
{ class: 'category suggestion-2', content: '+Chodist' },
|
|
||||||
{ class: 'category suggestion-3', content: '+Gamer' },
|
|
||||||
{ class: 'category more', content: 'More...' },
|
|
||||||
] }
|
|
||||||
] },
|
|
||||||
{ class: 'lower', content: [
|
|
||||||
{ class: 'error' },
|
|
||||||
{ class: 'buttons', content: [
|
|
||||||
{ class: 'button submit', content: 'Create Invite Token' }
|
|
||||||
] }
|
|
||||||
] }
|
|
||||||
] });
|
|
||||||
|
|
||||||
return element;
|
|
||||||
}
|
|
28
src/client/webapp/elements/overlay-create-invite-token.tsx
Normal file
28
src/client/webapp/elements/overlay-create-invite-token.tsx
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import CombinedGuild from "../guild-combined";
|
||||||
|
import BaseElements, { HTMLElementWithRemoveSelf } from "./require/base-elements";
|
||||||
|
|
||||||
|
import React from "react";
|
||||||
|
|
||||||
|
export default function createCreateInviteTokenOverlay(document: Document, guild: CombinedGuild): HTMLElementWithRemoveSelf {
|
||||||
|
const element = BaseElements.createOverlay(document, (
|
||||||
|
<div className="content submit-dialog">
|
||||||
|
<div className="role-select category-select">
|
||||||
|
<div className="label">Select Starting Roles</div>
|
||||||
|
<div className="categories">
|
||||||
|
<div className="category suggestion-1">+Gamer</div>
|
||||||
|
<div className="category suggestion-2">+Chodist</div>
|
||||||
|
<div className="category suggestion-3">+Gamer</div>
|
||||||
|
<div className="category more">More...</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="lower">
|
||||||
|
<div className="error"></div>
|
||||||
|
<div className="buttons">
|
||||||
|
<div className="button submit">Create Invite Token</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
|
||||||
|
return element;
|
||||||
|
}
|
@ -1,10 +0,0 @@
|
|||||||
import BaseElements from './require/base-elements.js';
|
|
||||||
|
|
||||||
export default function createErrorMessageOverlay(document: Document, title: string, message: string): HTMLElement {
|
|
||||||
return BaseElements.createOverlay(document, {
|
|
||||||
class: 'content error-message', content: [
|
|
||||||
{ class: 'title', content: title },
|
|
||||||
{ class: 'message', content: message }
|
|
||||||
]
|
|
||||||
});
|
|
||||||
}
|
|
11
src/client/webapp/elements/overlay-error-message.tsx
Normal file
11
src/client/webapp/elements/overlay-error-message.tsx
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import BaseElements, { HTMLElementWithRemoveSelf } from './require/base-elements.js';
|
||||||
|
|
||||||
|
export default function createErrorMessageOverlay(document: Document, title: string, message: string): HTMLElementWithRemoveSelf {
|
||||||
|
return BaseElements.createOverlay(document, (
|
||||||
|
<div className="content error-message">
|
||||||
|
<div className="title">{title}</div>
|
||||||
|
<div className="message">{message}</div>
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
}
|
@ -6,64 +6,61 @@ const LOG = Logger.create(__filename, electronConsole);
|
|||||||
|
|
||||||
import Globals from '../globals';
|
import Globals from '../globals';
|
||||||
|
|
||||||
import BaseElements from './require/base-elements';
|
import BaseElements, { HTMLElementWithRemoveSelf } from './require/base-elements';
|
||||||
import ElementsUtil from './require/elements-util';
|
import ElementsUtil from './require/elements-util';
|
||||||
|
|
||||||
import { GuildMetadata } from '../data-types';
|
import { GuildMetadata } from '../data-types';
|
||||||
import Q from '../q-module';
|
import Q from '../q-module';
|
||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
|
|
||||||
export default function createGuildSettingsOverlay(document: Document, q: Q, guild: CombinedGuild, guildMeta: GuildMetadata): HTMLElement {
|
import React from 'react';
|
||||||
const element = BaseElements.createOverlay(document, {
|
|
||||||
class: 'content display-swapper guild-settings', content: [
|
export default function createGuildSettingsOverlay(document: Document, q: Q, guild: CombinedGuild, guildMeta: GuildMetadata): HTMLElementWithRemoveSelf {
|
||||||
{ class: 'options', content: [
|
const element = BaseElements.createOverlay(document, (
|
||||||
{ class: 'title', content: guildMeta.name }, // TODO: update on change
|
<div className="content display-swapper guild-settings">
|
||||||
{ class: 'choosable chosen', content: 'Overview' },
|
<div className="options">
|
||||||
{ class: 'choosable', content: 'Channels' },
|
<div className="title">{guildMeta.name}</div>
|
||||||
{ class: 'choosable', content: 'Roles' },
|
<div className="choosable chosen">Overview</div>
|
||||||
{ class: 'choosable', content: 'Invites' },
|
<div className="choosable">Channels</div>
|
||||||
] },
|
<div className="choosable">Roles</div>
|
||||||
{ class: 'display', content: [
|
<div className="choosable">Invites</div>
|
||||||
{ class: 'scroll', content: [
|
</div>
|
||||||
{ class: 'metadata', content: [
|
<div className="display">
|
||||||
{ tag: 'label', class: 'image-input-label', content: [
|
<div className="scroll">
|
||||||
{ class: 'image-input', content: [
|
<div className="metadata">
|
||||||
{ class: 'icon', content: [
|
<label className="image-input-label">
|
||||||
{ tag: 'img', class: 'guild-icon', src: './img/loading.svg', alt: 'icon' },
|
<div className="icon">
|
||||||
{ class: 'modify', content: [
|
<img className="guild-icon" src="./img/loading.svg" alt="icon"></img>
|
||||||
{ tag: 'img', src: './img/pencil-icon.png' }
|
<div className="modify"><img src="./img/pencil-icon.png" alt="modify"></img></div>
|
||||||
] }
|
</div>
|
||||||
] },
|
<input className="image-input-upload" type="file" accept=".png,.jpg,.jpeg" style={{ display: 'none' }}></input>
|
||||||
] },
|
</label>
|
||||||
{ tag: 'input', class: 'image-input-upload', type: 'file', accept: '.png,.jpg,.jpeg', style: 'display: none' }
|
<div className="name">
|
||||||
] },
|
<div className="label">Guild Name</div>
|
||||||
{ class: 'name', content: [
|
<input className="guild-name" type="text" placeholder={guildMeta.name}></input>
|
||||||
{ class: 'label', content: 'Guild Name' },
|
</div>
|
||||||
{ tag: 'input', type: 'text', class: 'guild-name', 'placeholder': guildMeta.name,
|
</div>
|
||||||
contenteditable: 'plaintext-only', content: guildMeta.name },
|
</div>
|
||||||
] },
|
<div className="popup changes">
|
||||||
] }
|
<div className="content">
|
||||||
] },
|
<div className="tip">You have unsaved changes</div>
|
||||||
{ class: 'popup changes', content: [
|
<div className="actions">
|
||||||
{ class: 'content', content: [
|
<div className="button perdu reset">Reset</div>
|
||||||
{ class: 'tip', content: 'You have unsaved changes' },
|
<div className="button positive save-changes">Save Changes</div>
|
||||||
{ class: 'actions', content: [
|
</div>
|
||||||
{ class: 'button perdu reset', content: 'Reset' },
|
</div>
|
||||||
{ class: 'button positive save-changes', content: 'Save Changes' }
|
</div>
|
||||||
] }
|
<div className="popup error">
|
||||||
]}
|
<div className="content">
|
||||||
] },
|
<div className="tip"></div>
|
||||||
{ class: 'popup error', content: [
|
<div className="actions">
|
||||||
{ class: 'content', content: [
|
<div className="button perdu close">X</div>
|
||||||
{ class: 'tip' },
|
</div>
|
||||||
{ class: 'actions', content: [
|
</div>
|
||||||
{ class: 'button perdu close', content: 'X' },
|
</div>
|
||||||
] }
|
</div>
|
||||||
] }
|
</div>
|
||||||
] },
|
));
|
||||||
] }
|
|
||||||
]
|
|
||||||
});
|
|
||||||
|
|
||||||
let newIconBuff: Buffer | null = null;
|
let newIconBuff: Buffer | null = null;
|
||||||
let oldName = guildMeta.name;
|
let oldName = guildMeta.name;
|
||||||
@ -87,8 +84,10 @@ export default function createGuildSettingsOverlay(document: Document, q: Q, gui
|
|||||||
}
|
}
|
||||||
|
|
||||||
function reset() {
|
function reset() {
|
||||||
|
q.$$$<HTMLInputElement>(element, '.image-input-upload').value = '';
|
||||||
q.$$$<HTMLInputElement>(element, 'input.guild-name').value = oldName;
|
q.$$$<HTMLInputElement>(element, 'input.guild-name').value = oldName;
|
||||||
(async () => {
|
(async () => {
|
||||||
|
LOG.debug('resetting icon');
|
||||||
q.$$$<HTMLImageElement>(element, '.icon img').src = await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, guildMeta.iconResourceId);
|
q.$$$<HTMLImageElement>(element, '.icon img').src = await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, guildMeta.iconResourceId);
|
||||||
})();
|
})();
|
||||||
|
|
||||||
@ -114,6 +113,7 @@ export default function createGuildSettingsOverlay(document: Document, q: Q, gui
|
|||||||
onCleared: () => {},
|
onCleared: () => {},
|
||||||
onError: async (errMsg) => await updatePopups(errMsg),
|
onError: async (errMsg) => await updatePopups(errMsg),
|
||||||
onLoaded: async (buff, src) => {
|
onLoaded: async (buff, src) => {
|
||||||
|
LOG.debug('image loaded');
|
||||||
newIconBuff = buff;
|
newIconBuff = buff;
|
||||||
(q.$$$(element, 'img.guild-icon') as HTMLImageElement).src = src;
|
(q.$$$(element, 'img.guild-icon') as HTMLImageElement).src = src;
|
||||||
await updatePopups();
|
await updatePopups();
|
@ -5,24 +5,28 @@ const LOG = Logger.create(__filename, electronConsole);
|
|||||||
|
|
||||||
import * as FileType from 'file-type'
|
import * as FileType from 'file-type'
|
||||||
|
|
||||||
import BaseElements from './require/base-elements';
|
import BaseElements, { HTMLElementWithRemoveSelf } from './require/base-elements';
|
||||||
import ElementsUtil from './require/elements-util';
|
import ElementsUtil from './require/elements-util';
|
||||||
|
|
||||||
import Q from '../q-module';
|
import Q from '../q-module';
|
||||||
import createImageContextMenu from './context-menu-img';
|
import createImageContextMenu from './context-menu-img';
|
||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
|
|
||||||
export default function createImageOverlay(document: Document, q: Q, guild: CombinedGuild, resourceId: string, resourceName: string): HTMLElement {
|
import React from 'react';
|
||||||
const element = BaseElements.createOverlay(document, { class: 'content popup-image', content: [
|
|
||||||
{ tag: 'img', src: './img/loading.svg', alt: resourceName, title: resourceName },
|
export default function createImageOverlay(document: Document, q: Q, guild: CombinedGuild, resourceId: string, resourceName: string): HTMLElementWithRemoveSelf {
|
||||||
{ class: 'download', content: [
|
const element = BaseElements.createOverlay(document, (
|
||||||
{ class: 'info', content: [
|
<div className="content popup-image">
|
||||||
{ class: 'name', content: resourceName },
|
<img src="./img/loading.svg" alt={resourceName} title={resourceName}></img>
|
||||||
{ class: 'size', content: 'Loading Size...' }
|
<div className="download">
|
||||||
] },
|
<div className="info">
|
||||||
{ class: 'button', content: 'Loading...' }
|
<div className="name">{resourceName}</div>
|
||||||
] }
|
<div className="size">Loading Size...</div>
|
||||||
] });
|
</div>
|
||||||
|
<div className="button">Loading...</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
|
||||||
(async () => {
|
(async () => {
|
||||||
try {
|
try {
|
@ -6,30 +6,39 @@ const LOG = Logger.create(__filename, electronConsole);
|
|||||||
import { Channel } from '../data-types';
|
import { Channel } from '../data-types';
|
||||||
import Globals from '../globals.js';
|
import Globals from '../globals.js';
|
||||||
|
|
||||||
import BaseElements from './require/base-elements.js';
|
import BaseElements, { HTMLElementWithRemoveSelf } from './require/base-elements.js';
|
||||||
import ElementsUtil from './require/elements-util.js';
|
import ElementsUtil from './require/elements-util.js';
|
||||||
import Q from '../q-module';
|
import Q from '../q-module';
|
||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
|
|
||||||
export default function createModifyChannelOverlay(document: Document, q: Q, guild: CombinedGuild, channel: Channel): HTMLElement {
|
import React from 'react';
|
||||||
|
|
||||||
|
export default function createModifyChannelOverlay(document: Document, q: Q, guild: CombinedGuild, channel: Channel): HTMLElementWithRemoveSelf {
|
||||||
// See also overlay-create-channel
|
// See also overlay-create-channel
|
||||||
|
|
||||||
const element = BaseElements.createOverlay(document, { class: 'content submit-dialog modify-channel', content: [
|
const element = BaseElements.createOverlay(document, (
|
||||||
{ class: 'preview channel-title', content: [
|
<div className="content submit-dialog modify-channel">
|
||||||
{ class: 'channel-icon', content: BaseElements.Q_TEXT_CHANNEL_ICON },
|
<div className="preview channel-title">
|
||||||
{ class: 'channel-name', content: channel.name },
|
<div className="channel-icon">{BaseElements.TEXT_CHANNEL_ICON}</div>
|
||||||
{ class: 'channel-flavor-divider' },
|
<div className="channel-name">{channel.name}</div>
|
||||||
{ class: 'channel-flavor-text', content: channel.flavorText || '' }
|
<div className="channel-flavor-divider"></div>
|
||||||
] },
|
<div className="channel-flavor-text">{channel.flavorText || ''}</div>
|
||||||
{ class: 'text-input channel-name', 'data-placeholder': 'channel-name', content: channel.name, contenteditable: 'plaintext-only' },
|
</div>
|
||||||
{ class: 'text-input channel-flavor-text', 'data-placeholder': 'Flavor Text (optional)', content: channel.flavorText || '', contenteditable: 'plaintext-only' },
|
<div className="text-input channel-name" data-placeholder="channel-name"
|
||||||
{ class: 'lower', content: [
|
contentEditable={'plaintext-only' as unknown as boolean /* React doesn't have plaintext-only in its typings (https://github.com/DefinitelyTyped/DefinitelyTyped/pull/54779) */}></div>
|
||||||
{ class: 'error' },
|
<div className="text-input channel-flavor-text" data-placeholder="Flavor Text (optional)"
|
||||||
{ class: 'buttons', content: [
|
contentEditable={'plaintext-only' as unknown as boolean /* React doesn't have plaintext-only in its typings (https://github.com/DefinitelyTyped/DefinitelyTyped/pull/54779) */}></div>
|
||||||
{ class: 'button submit', content: 'Save Changes' }
|
<div className="lower">
|
||||||
] }
|
<div className="error"></div>
|
||||||
] }
|
<div className="buttons">
|
||||||
] });
|
<div className="button submit">Save Changes</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
|
||||||
|
q.$$$(element, '.text-input.channel-name').innerText = channel.name;
|
||||||
|
q.$$$(element, '.text-input.channel-flavor-text').innerText = channel.flavorText || '';
|
||||||
|
|
||||||
let newName = channel.name;
|
let newName = channel.name;
|
||||||
let newFlavorText = channel.flavorText;
|
let newFlavorText = channel.flavorText;
|
@ -4,7 +4,7 @@ const electronConsole = electronRemote.getGlobal('console') as Console;
|
|||||||
import Logger from '../../../logger/logger';
|
import Logger from '../../../logger/logger';
|
||||||
const LOG = Logger.create(__filename, electronConsole);
|
const LOG = Logger.create(__filename, electronConsole);
|
||||||
|
|
||||||
import BaseElements from './require/base-elements';
|
import BaseElements, { HTMLElementWithRemoveSelf } from './require/base-elements';
|
||||||
import ElementsUtil from './require/elements-util';
|
import ElementsUtil from './require/elements-util';
|
||||||
|
|
||||||
import Globals from '../globals';
|
import Globals from '../globals';
|
||||||
@ -14,27 +14,46 @@ import createTextMessage from './msg-txt';
|
|||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
import { ConnectionInfo } from '../data-types';
|
import { ConnectionInfo } from '../data-types';
|
||||||
|
|
||||||
export default function createPersonalizeOverlay(document: Document, q: Q, guild: CombinedGuild, connection: ConnectionInfo): HTMLElement {
|
import React from 'react';
|
||||||
const element = BaseElements.createOverlay(document, {
|
import moment from 'moment';
|
||||||
class: 'content submit-dialog personalize', content: [
|
|
||||||
createTextMessage(q, guild, { id: 'test-message', member: connection, sent: new Date(), text: 'Example Message' }),
|
export default function createPersonalizeOverlay(document: Document, q: Q, guild: CombinedGuild, connection: ConnectionInfo): HTMLElementWithRemoveSelf {
|
||||||
{ class: 'text-input', placeholder: 'New Display Name',
|
const element = BaseElements.createOverlay(document, (
|
||||||
spellcheck: 'false', contenteditable: 'plaintext-only', content: connection.displayName },
|
<div className="content submit-dialog personalize">
|
||||||
{ class: 'image-input avatar-input', content: [
|
<div className="message">
|
||||||
{ tag: 'label', class: 'image-input-label avatar-input-label button', content: [
|
<div className="member-avatar">
|
||||||
'Select New Avatar',
|
<img src="./img/loading.svg" alt={connection.displayName}></img>
|
||||||
{ class: 'image-input-upload avatar-upload', tag: 'input',
|
</div>
|
||||||
type: 'file', accept: '.png,.jpg,.jpeg', style: 'display: none;' },
|
<div className="right">
|
||||||
] }
|
<div className="header">
|
||||||
] },
|
<div className="member-name">{connection.displayName}</div>
|
||||||
{ class: 'lower', content: [
|
<div className="timestamp">{moment(new Date()).calendar(ElementsUtil.calendarFormats)}</div>
|
||||||
{ class: 'error' },
|
</div>
|
||||||
{ class: 'buttons', content: [
|
<div className="content text">Example Message</div>
|
||||||
{ class: 'button submit', content: 'Save Changes' }
|
</div>
|
||||||
] }
|
</div>
|
||||||
] }
|
<div className="text-input display-name-input" data-placeholder="New Display Name"
|
||||||
]
|
contentEditable={'plaintext-only' as unknown as boolean /* React doesn't have plaintext-only in its typings (https://github.com/DefinitelyTyped/DefinitelyTyped/pull/54779) */}></div>
|
||||||
});
|
<div className="image-input avatar-input">
|
||||||
|
<label className="image-input-label avatar-input-label button">
|
||||||
|
Select New Avatar
|
||||||
|
<input className="image-input-upload avatar-upload" type="file" accept=".png,.jpg,.jpeg" style={{ display: 'none' }}></input>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div className="lower">
|
||||||
|
<div className="error"></div>
|
||||||
|
<div className="buttons">
|
||||||
|
<div className="button submit">Save Changes</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
|
||||||
|
q.$$$(element, '.text-input.display-name-input').innerText = connection.displayName;
|
||||||
|
(async () => {
|
||||||
|
(q.$$$(element, '.member-avatar img') as HTMLImageElement).src =
|
||||||
|
await ElementsUtil.getImageBufferFromResourceFailSoftly(guild, connection.avatarResourceId);
|
||||||
|
})();
|
||||||
|
|
||||||
let newAvatarBuffer: Buffer | null = null;
|
let newAvatarBuffer: Buffer | null = null;
|
||||||
BaseElements.bindImageUploadEvents(q.$$$(element, '.avatar-upload') as HTMLInputElement, {
|
BaseElements.bindImageUploadEvents(q.$$$(element, '.avatar-upload') as HTMLInputElement, {
|
@ -13,12 +13,14 @@ import { Member } from '../data-types';
|
|||||||
import Q from '../q-module';
|
import Q from '../q-module';
|
||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
export default function createTokenLogOverlay(document: Document, q: Q, guild: CombinedGuild): HTMLElement {
|
export default function createTokenLogOverlay(document: Document, q: Q, guild: CombinedGuild): HTMLElement {
|
||||||
const element = BaseElements.createOverlay(document, {
|
const element = BaseElements.createOverlay(document, (
|
||||||
class: 'content token-log', content: [
|
<div className="content token-log">
|
||||||
{ class: 'tokens', content: { tag: 'img', src: './img/loading.svg', alt: 'loading...' } },
|
<div className="tokens"><img src="./img/loading.svg" alt="Loading..."></img></div>
|
||||||
]
|
</div>
|
||||||
});
|
));
|
||||||
|
|
||||||
(async () => {
|
(async () => {
|
||||||
Util.withPotentialErrorWarnOnCancel(q, {
|
Util.withPotentialErrorWarnOnCancel(q, {
|
@ -9,11 +9,16 @@ import Q from '../q-module';
|
|||||||
import createUploadOverlayFromDataTransferItem from './overlay-upload-datatransfer';
|
import createUploadOverlayFromDataTransferItem from './overlay-upload-datatransfer';
|
||||||
import CombinedGuild from '../guild-combined';
|
import CombinedGuild from '../guild-combined';
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
export default function createUploadDropTarget(document: Document, q: Q, guild: CombinedGuild, channel: Channel): HTMLElement {
|
export default function createUploadDropTarget(document: Document, q: Q, guild: CombinedGuild, channel: Channel): HTMLElement {
|
||||||
const element = BaseElements.createOverlay(document, { class: 'content drop-target', content: [
|
const element = BaseElements.createOverlay(document, (
|
||||||
// TODO: icon?
|
<div className="content drop-target">
|
||||||
{ class: 'message', content: 'Upload to #' + channel.name }
|
{/* TODO: icon? */}
|
||||||
] });
|
<div className="message">Upload to #{channel.name}</div>
|
||||||
|
</div>
|
||||||
|
));
|
||||||
|
|
||||||
element.addEventListener('dragover', (e) => {
|
element.addEventListener('dragover', (e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
@ -250,12 +250,18 @@ export default class BaseElements {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
static createOverlay(document: Document, content: any): HTMLElementWithRemoveSelf {
|
static createOverlay(document: Document, content: JSX.Element): HTMLElementWithRemoveSelf {
|
||||||
const q = new Q(document);
|
const q = new Q(document);
|
||||||
|
|
||||||
let wasDownInternal = false; // because 'click' fires on the overlay element anyway
|
let wasDownInternal = false; // because 'click' fires on the overlay element anyway
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
const element = q.create({ class: 'overlay', content: content }) as any;
|
const element: HTMLElementWithRemoveSelf = ReactHelper.createElementFromJSX(<div className="overlay">{content}</div>) as HTMLElementWithRemoveSelf;
|
||||||
|
element.removeSelf = () => {
|
||||||
|
if (element.parentElement) {
|
||||||
|
element.parentElement.removeChild(element);
|
||||||
|
}
|
||||||
|
window.removeEventListener('keydown', onKeyEscape);
|
||||||
|
}
|
||||||
const onKeyEscape = (e: KeyboardEvent) => {
|
const onKeyEscape = (e: KeyboardEvent) => {
|
||||||
if (e.key == 'Escape') {
|
if (e.key == 'Escape') {
|
||||||
element.removeSelf();
|
element.removeSelf();
|
||||||
@ -269,12 +275,7 @@ export default class BaseElements {
|
|||||||
}
|
}
|
||||||
element.removeSelf();
|
element.removeSelf();
|
||||||
});
|
});
|
||||||
element.removeSelf = () => {
|
|
||||||
if (element.parentElement) {
|
|
||||||
element.parentElement.removeChild(element);
|
|
||||||
}
|
|
||||||
window.removeEventListener('keydown', onKeyEscape);
|
|
||||||
}
|
|
||||||
q.$$$(element, '.content').addEventListener('click', (e) => {
|
q.$$$(element, '.content').addEventListener('click', (e) => {
|
||||||
e.stopPropagation(); // prevent the element from closing if the content is clicked on
|
e.stopPropagation(); // prevent the element from closing if the content is clicked on
|
||||||
});
|
});
|
||||||
@ -290,23 +291,25 @@ export default class BaseElements {
|
|||||||
|
|
||||||
const { guild, channel, resourceName, resourceBuffFunc, resourceSizeFunc } = props;
|
const { guild, channel, resourceName, resourceBuffFunc, resourceSizeFunc } = props;
|
||||||
|
|
||||||
const element = BaseElements.createOverlay(document, { class: 'content upload', content: [
|
const element = BaseElements.createOverlay(document, (
|
||||||
{ class: 'title', content: [
|
<div className="content upload">
|
||||||
{ tag: 'img', src: './img/loading.svg', alt: resourceName },
|
<div className="title">
|
||||||
{ class: 'right', content: [
|
<img src="./img/loading.svg" alt={resourceName}></img>
|
||||||
{ class: 'name', content: resourceName },
|
<div className="right">
|
||||||
{ class: 'size', content: '? B' }
|
<div className="name">{resourceName}</div>
|
||||||
] }
|
<div className="size">? B</div>
|
||||||
] },
|
</div>
|
||||||
{ class: 'text-input', placeholder: 'Add a comment (optional)', contenteditable: 'plaintext-only' },
|
</div>
|
||||||
{ class: 'lower', content: [
|
<div className="text-input" data-placeholder="Add a comment (optional)" contentEditable={'plaintext-only' as unknown as boolean /* React doesn't have plaintext-only in its typings (https://github.com/DefinitelyTyped/DefinitelyTyped/pull/54779) */}></div>
|
||||||
{ class: 'error' },
|
<div className="lower">
|
||||||
{ class: 'buttons', content: [
|
<div className="error"></div>
|
||||||
{ class: 'button upload', content: 'Upload to #' + channel.name }
|
<div className="buttons">
|
||||||
] }
|
<div className="button upload">Upload to #{channel.name}</div>
|
||||||
] }
|
</div>
|
||||||
|
</div>
|
||||||
] });
|
</div>
|
||||||
|
));
|
||||||
|
|
||||||
q.$$$(element, '.text-input').addEventListener('keydown', async (e) => {
|
q.$$$(element, '.text-input').addEventListener('keydown', async (e) => {
|
||||||
if (e.key == 'Enter' && !e.shiftKey) {
|
if (e.key == 'Enter' && !e.shiftKey) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
@ -9,6 +9,7 @@ import Logger from '../../../../logger/logger';
|
|||||||
const LOG = Logger.create(__filename, electronConsole);
|
const LOG = Logger.create(__filename, electronConsole);
|
||||||
|
|
||||||
import * as FileType from 'file-type';
|
import * as FileType from 'file-type';
|
||||||
|
import * as uuid from 'uuid';
|
||||||
|
|
||||||
import Util from '../../util';
|
import Util from '../../util';
|
||||||
import Globals from '../../globals';
|
import Globals from '../../globals';
|
||||||
@ -17,11 +18,6 @@ import { ShouldNeverHappenError } from '../../data-types';
|
|||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
// TODO: pass-through Globals in init function
|
|
||||||
// alignment: {
|
|
||||||
// centerY: 'top'
|
|
||||||
// left: 'right + 20'
|
|
||||||
// }
|
|
||||||
interface IAlignment {
|
interface IAlignment {
|
||||||
left?: string;
|
left?: string;
|
||||||
centerX?: string;
|
centerX?: string;
|
||||||
@ -37,7 +33,6 @@ interface IHTMLElementWithRemovalType extends HTMLElement {
|
|||||||
|
|
||||||
interface SimpleQElement {
|
interface SimpleQElement {
|
||||||
tag: 'span',
|
tag: 'span',
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
content: (SimpleQElement | string)[],
|
content: (SimpleQElement | string)[],
|
||||||
class: string | null
|
class: string | null
|
||||||
}
|
}
|
||||||
@ -124,11 +119,8 @@ export default class ElementsUtil {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// creates <span class="bold"/"italic"/"bold italic"/"underline"> spans to format the text (all in q.js element markup that gets converted to jsx [aka hypergaming])
|
// creates <span class="bold"/"italic"/"bold italic"/"underline"> spans to format the text (all in q.js element markup that gets converted to jsx [aka hypergaming])
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
static parseMessageText(text: string): JSX.Element {
|
static parseMessageText(text: string): JSX.Element {
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
const obj: SimpleQElement = { tag: 'span', content: [], class: null };
|
||||||
const obj: SimpleQElement = { tag: 'span', content: [] as (SimpleQElement | string)[], class: null };
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
const stack: SimpleQElement[] = [ obj ];
|
const stack: SimpleQElement[] = [ obj ];
|
||||||
let idx = 0;
|
let idx = 0;
|
||||||
function makeEscape(regex: RegExp, len: number, str: string): { matcher: RegExp, response: ((i: number) => void)} { // function for readability
|
function makeEscape(regex: RegExp, len: number, str: string): { matcher: RegExp, response: ((i: number) => void)} { // function for readability
|
||||||
@ -152,8 +144,7 @@ export default class ElementsUtil {
|
|||||||
// TODO: optimise out empty elements
|
// TODO: optimise out empty elements
|
||||||
stack.pop();
|
stack.pop();
|
||||||
} else { // italic begins
|
} else { // italic begins
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
const obj: SimpleQElement = { tag: 'span', class: cls, content: [] }
|
||||||
const obj: SimpleQElement = { tag: 'span', class: cls, content: [] as any[] }
|
|
||||||
top.content.push(obj);
|
top.content.push(obj);
|
||||||
stack.push(obj);
|
stack.push(obj);
|
||||||
}
|
}
|
||||||
@ -195,7 +186,7 @@ export default class ElementsUtil {
|
|||||||
return qjsObj
|
return qjsObj
|
||||||
} else {
|
} else {
|
||||||
const content = qjsObj.content.map(qjsElement => createReactElement(qjsElement));
|
const content = qjsObj.content.map(qjsElement => createReactElement(qjsElement));
|
||||||
return React.createElement(qjsObj.tag, { className: qjsObj.class }, content);
|
return React.createElement(qjsObj.tag, { key: uuid.v4(), className: qjsObj.class }, content);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user