import Vue, {WatchHandler} from 'vue'; import { CacheManager } from '../learn/cache-manager'; import BBCodeParser from './bbcode'; import {Settings as SettingsImpl} from './common'; import {Channel, Character, Connection, Conversation, Logs, Notifications, Settings, State as StateInterface} from './interfaces'; 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[] = []; 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 = new Vue({ data: { channels: undefined, characters: undefined, conversations: undefined, state }, watch: { 'state.hiddenUsers': async(newValue: string[]) => { if(data.settingsStore !== undefined) await data.settingsStore.set('hiddenUsers', newValue); } } }); const data = { connection: undefined, logs: undefined, settingsStore: undefined, state: vue.state, bbCodeParser: new BBCodeParser(), conversations: undefined, channels: undefined, characters: undefined, notifications: undefined, cache: undefined, register(this: void | never, module: 'characters' | 'conversations' | 'channels', subState: Channel.State | Character.State | Conversation.State): void { Vue.set(vue, module, subState); data[module] = subState; }, watch(getter: (this: VueState) => T, callback: WatchHandler): void { vue.$watch(getter, callback); }, async reloadSettings(): Promise { const settings = new SettingsImpl(); const loadedSettings = await core.settingsStore.get('settings'); if(loadedSettings !== undefined) for(const key in loadedSettings) settings[key] = loadedSettings[key]; state._settings = settings; const hiddenUsers = await core.settingsStore.get('hiddenUsers'); state.hiddenUsers = hiddenUsers !== undefined ? hiddenUsers : []; } }; export function init(this: void, connection: Connection, logsClass: new() => Logs, settingsClass: new() => Settings.Store, notificationsClass: new() => Notifications): void { data.connection = connection; data.logs = new logsClass(); data.settingsStore = new settingsClass(); data.notifications = new notificationsClass(); data.cache = new CacheManager(); data.cache.start(); connection.onEvent('connecting', async() => { await data.reloadSettings(); data.bbCodeParser = createBBCodeParser(); }); } export interface Core { readonly connection: Connection readonly logs: Logs 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 readonly cache: CacheManager register(module: 'conversations', state: Conversation.State): void register(module: 'channels', state: Channel.State): void register(module: 'characters', state: Character.State): void watch(getter: (this: VueState) => T, callback: WatchHandler): void } const core = data; /*tslint:disable-line:no-any*///hack export default core;