fchat-rising/chat/core.ts

126 lines
4.4 KiB
TypeScript
Raw Normal View History

import Vue, {WatchHandler} from 'vue';
2019-07-07 01:37:15 +00:00
import { CacheManager } from '../learn/cache-manager';
2019-09-17 17:14:14 +00:00
import {Channels, Characters} from '../fchat';
2017-09-02 01:50:31 +00:00
import BBCodeParser from './bbcode';
import {Settings as SettingsImpl} from './common';
2019-09-17 17:14:14 +00:00
import Conversations from './conversations';
2017-09-02 01:50:31 +00:00
import {Channel, Character, Connection, Conversation, Logs, Notifications, Settings, State as StateInterface} from './interfaces';
import { AdCoordinatorGuest } from './ads/ad-coordinator-guest';
2017-09-02 01:50:31 +00:00
function createBBCodeParser(): BBCodeParser {
const parser = new BBCodeParser();
for(const tag of state.settings.disallowedTags)
parser.removeTag(tag);
return parser;
}
class State implements StateInterface {
_settings: Settings | undefined = undefined;
hiddenUsers: string[] = [];
2017-09-02 01:50:31 +00:00
get settings(): Settings {
if(this._settings === undefined) throw new Error('Settings load failed.');
return this._settings;
}
set settings(value: Settings) {
this._settings = value;
//tslint:disable-next-line:no-floating-promises
if(data.settingsStore !== undefined) data.settingsStore.set('settings', value);
data.bbCodeParser = createBBCodeParser();
}
}
interface VueState {
readonly channels: Channel.State
readonly characters: Character.State
readonly conversations: Conversation.State
readonly state: StateInterface
}
const state = new State();
const vue = <Vue & VueState>new Vue({
data: {
channels: undefined,
characters: undefined,
conversations: undefined,
state
},
watch: {
2019-09-17 17:14:14 +00:00
'state.hiddenUsers': async(newValue: string[], oldValue: string[]) => {
if(data.settingsStore !== undefined && newValue !== oldValue)
await data.settingsStore.set('hiddenUsers', newValue);
}
2017-09-02 01:50:31 +00:00
}
});
const data = {
connection: <Connection | undefined>undefined,
2018-03-28 13:51:05 +00:00
logs: <Logs | undefined>undefined,
2017-09-02 01:50:31 +00:00
settingsStore: <Settings.Store | undefined>undefined,
state: vue.state,
bbCodeParser: new BBCodeParser(),
2017-09-02 01:50:31 +00:00
conversations: <Conversation.State | undefined>undefined,
channels: <Channel.State | undefined>undefined,
characters: <Character.State | undefined>undefined,
notifications: <Notifications | undefined>undefined,
2019-07-07 01:37:15 +00:00
cache: <CacheManager | undefined>undefined,
adCoordinator: <AdCoordinatorGuest | undefined>undefined,
2019-09-17 17:14:14 +00:00
register<K extends 'characters' | 'conversations' | 'channels'>(module: K, subState: VueState[K]): void {
2017-09-02 01:50:31 +00:00
Vue.set(vue, module, subState);
2019-09-17 17:14:14 +00:00
(<VueState[K]>data[module]) = subState;
2017-09-02 01:50:31 +00:00
},
watch<T>(getter: (this: VueState) => T, callback: WatchHandler<T>): void {
2017-09-02 01:50:31 +00:00
vue.$watch(getter, callback);
},
async reloadSettings(): Promise<void> {
2019-09-17 17:14:14 +00:00
state._settings = Object.assign(new SettingsImpl(), await core.settingsStore.get('settings'));
const hiddenUsers = await core.settingsStore.get('hiddenUsers');
state.hiddenUsers = hiddenUsers !== undefined ? hiddenUsers : [];
2017-09-02 01:50:31 +00:00
}
};
2020-03-15 14:02:31 +00:00
export function init(this: any, connection: Connection, logsClass: new() => Logs, settingsClass: new() => Settings.Store,
2017-09-02 01:50:31 +00:00
notificationsClass: new() => Notifications): void {
data.connection = connection;
data.logs = new logsClass();
data.settingsStore = new settingsClass();
data.notifications = new notificationsClass();
2019-07-07 01:37:15 +00:00
data.cache = new CacheManager();
data.adCoordinator = new AdCoordinatorGuest();
2019-07-07 01:37:15 +00:00
// tslint:disable-next-line no-floating-promises
2019-07-07 01:37:15 +00:00
data.cache.start();
2019-09-17 17:14:14 +00:00
data.register('characters', Characters(connection));
data.register('channels', Channels(connection, core.characters));
data.register('conversations', Conversations());
2017-09-02 01:50:31 +00:00
connection.onEvent('connecting', async() => {
await data.reloadSettings();
data.bbCodeParser = createBBCodeParser();
});
}
2018-01-06 16:14:21 +00:00
export interface Core {
2017-09-02 01:50:31 +00:00
readonly connection: Connection
2018-03-28 13:51:05 +00:00
readonly logs: Logs
2017-09-02 01:50:31 +00:00
readonly state: StateInterface
readonly settingsStore: Settings.Store
readonly conversations: Conversation.State
readonly characters: Character.State
readonly channels: Channel.State
readonly bbCodeParser: BBCodeParser
readonly notifications: Notifications
2019-07-07 01:37:15 +00:00
readonly cache: CacheManager
readonly adCoordinator: AdCoordinatorGuest;
2019-07-07 01:37:15 +00:00
watch<T>(getter: (this: VueState) => T, callback: WatchHandler<T>): void
2018-01-06 16:14:21 +00:00
}
const core = <Core><any>data; /*tslint:disable-line:no-any*///hack
2017-09-02 01:50:31 +00:00
export default core;