2017-09-02 01:50:31 +00:00
|
|
|
<template>
|
|
|
|
<div style="display:flex; flex-direction: column; height:100%; justify-content: center">
|
2018-03-04 02:32:26 +00:00
|
|
|
<div class="card bg-light" style="width:400px;max-width:100%;margin:0 auto" v-if="!connected">
|
2017-09-02 01:50:31 +00:00
|
|
|
<div class="alert alert-danger" v-show="error">{{error}}</div>
|
2018-04-08 00:22:32 +00:00
|
|
|
<h3 class="card-header" style="margin-top:0;display:flex">
|
|
|
|
{{l('title')}}
|
|
|
|
<a href="#" @click.prevent="$refs['logsDialog'].show()" class="btn" style="flex:1;text-align:right">
|
|
|
|
<span class="fa fa-file-alt"></span> <span class="btn-text">{{l('logs.title')}}</span>
|
|
|
|
</a>
|
|
|
|
</h3>
|
2018-03-04 02:32:26 +00:00
|
|
|
<div class="card-body">
|
2017-09-02 01:50:31 +00:00
|
|
|
<h4 class="card-title">{{l('login.selectCharacter')}}</h4>
|
2018-03-04 02:32:26 +00:00
|
|
|
<select v-model="selectedCharacter" class="form-control custom-select">
|
2017-09-02 01:50:31 +00:00
|
|
|
<option v-for="character in ownCharacters" :value="character">{{character}}</option>
|
|
|
|
</select>
|
2018-03-04 02:32:26 +00:00
|
|
|
<div style="text-align:right;margin-top:10px">
|
2017-09-02 01:50:31 +00:00
|
|
|
<button class="btn btn-primary" @click="connect" :disabled="connecting">
|
|
|
|
{{l(connecting ? 'login.connecting' : 'login.connect')}}
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<chat v-else></chat>
|
|
|
|
<modal :action="l('chat.disconnected.title')" :buttonText="l('action.cancel')" ref="reconnecting" @submit="cancelReconnect"
|
|
|
|
:showCancel="false" buttonClass="btn-danger">
|
|
|
|
<div class="alert alert-danger" v-show="error">{{error}}</div>
|
|
|
|
{{l('chat.disconnected')}}
|
|
|
|
</modal>
|
2018-04-08 00:22:32 +00:00
|
|
|
<logs ref="logsDialog"></logs>
|
|
|
|
<div v-if="version && !connected" style="position:absolute;bottom:0;right:0">{{version}}</div>
|
2017-09-02 01:50:31 +00:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
|
|
|
import Vue from 'vue';
|
|
|
|
import Component from 'vue-class-component';
|
|
|
|
import {Prop} from 'vue-property-decorator';
|
|
|
|
import Modal from '../components/Modal.vue';
|
|
|
|
import Channels from '../fchat/channels';
|
|
|
|
import Characters from '../fchat/characters';
|
2018-03-28 13:51:05 +00:00
|
|
|
import {Keys} from '../keys';
|
2017-09-02 01:50:31 +00:00
|
|
|
import ChatView from './ChatView.vue';
|
2018-03-28 13:51:05 +00:00
|
|
|
import {errorToString, getKey} from './common';
|
2017-09-02 01:50:31 +00:00
|
|
|
import Conversations from './conversations';
|
|
|
|
import core from './core';
|
|
|
|
import l from './localize';
|
2018-04-08 00:22:32 +00:00
|
|
|
import Logs from './Logs.vue';
|
2017-09-02 01:50:31 +00:00
|
|
|
|
2018-03-04 02:32:26 +00:00
|
|
|
type BBCodeNode = Node & {bbcodeTag?: string, bbcodeParam?: string, bbcodeHide?: boolean};
|
|
|
|
|
2018-04-16 23:14:13 +00:00
|
|
|
function copyNode(str: string, node: BBCodeNode, end: Node, range: Range, flags: {endFound?: true}): string {
|
|
|
|
if(node === end) flags.endFound = true;
|
2018-03-04 02:32:26 +00:00
|
|
|
if(node.bbcodeTag !== undefined)
|
|
|
|
str = `[${node.bbcodeTag}${node.bbcodeParam !== undefined ? `=${node.bbcodeParam}` : ''}]${str}[/${node.bbcodeTag}]`;
|
|
|
|
if(node.nextSibling !== null && !flags.endFound) {
|
2018-03-28 13:51:05 +00:00
|
|
|
if(node instanceof HTMLElement && getComputedStyle(node).display === 'block') str += '\r\n';
|
2018-04-16 23:14:13 +00:00
|
|
|
str += scanNode(node.nextSibling!, end, range, flags);
|
2018-03-04 02:32:26 +00:00
|
|
|
}
|
2018-03-28 13:51:05 +00:00
|
|
|
if(node.parentElement === null) return str;
|
2018-04-16 23:14:13 +00:00
|
|
|
return copyNode(str, node.parentNode!, end, range, flags);
|
2018-03-04 02:32:26 +00:00
|
|
|
}
|
|
|
|
|
2018-04-16 23:14:13 +00:00
|
|
|
function scanNode(node: BBCodeNode, end: Node, range: Range, flags: {endFound?: true}, hide?: boolean): string {
|
2018-03-28 13:51:05 +00:00
|
|
|
let str = '';
|
|
|
|
hide = hide || node.bbcodeHide;
|
2018-04-16 23:14:13 +00:00
|
|
|
if(node === end) flags.endFound = true;
|
2018-03-04 02:32:26 +00:00
|
|
|
if(node.bbcodeTag !== undefined) str += `[${node.bbcodeTag}${node.bbcodeParam !== undefined ? `=${node.bbcodeParam}` : ''}]`;
|
2018-03-28 13:51:05 +00:00
|
|
|
if(node instanceof Text) str += node === range.endContainer ? node.nodeValue!.substr(0, range.endOffset) : node.nodeValue;
|
|
|
|
else if(node instanceof HTMLImageElement) str += node.alt;
|
2018-04-16 23:14:13 +00:00
|
|
|
if(node.firstChild !== null && !flags.endFound) str += scanNode(node.firstChild, end, range, flags, hide);
|
2018-03-04 02:32:26 +00:00
|
|
|
if(node.bbcodeTag !== undefined) str += `[/${node.bbcodeTag}]`;
|
2018-03-28 13:51:05 +00:00
|
|
|
if(node instanceof HTMLElement && getComputedStyle(node).display === 'block') str += '\r\n';
|
2018-04-16 23:14:13 +00:00
|
|
|
if(node.nextSibling !== null && !flags.endFound) str += scanNode(node.nextSibling, end, range, flags, hide);
|
2018-03-28 13:51:05 +00:00
|
|
|
return hide ? '' : str;
|
2018-03-04 02:32:26 +00:00
|
|
|
}
|
|
|
|
|
2017-09-02 01:50:31 +00:00
|
|
|
@Component({
|
2018-04-08 00:22:32 +00:00
|
|
|
components: {chat: ChatView, modal: Modal, logs: Logs}
|
2017-09-02 01:50:31 +00:00
|
|
|
})
|
|
|
|
export default class Chat extends Vue {
|
|
|
|
@Prop({required: true})
|
2018-03-04 02:32:26 +00:00
|
|
|
readonly ownCharacters!: string[];
|
2017-09-02 01:50:31 +00:00
|
|
|
@Prop({required: true})
|
2018-03-04 02:32:26 +00:00
|
|
|
readonly defaultCharacter!: string | undefined;
|
2018-01-06 16:14:21 +00:00
|
|
|
selectedCharacter = this.defaultCharacter || this.ownCharacters[0]; //tslint:disable-line:strict-boolean-expressions
|
2018-04-08 00:22:32 +00:00
|
|
|
@Prop()
|
|
|
|
readonly version?: string;
|
2017-09-02 01:50:31 +00:00
|
|
|
error = '';
|
|
|
|
connecting = false;
|
|
|
|
connected = false;
|
|
|
|
l = l;
|
2018-03-28 13:51:05 +00:00
|
|
|
copyPlain = false;
|
2017-09-02 01:50:31 +00:00
|
|
|
|
|
|
|
mounted(): void {
|
2018-04-08 00:22:32 +00:00
|
|
|
document.title = l('title', core.connection.character);
|
2018-03-04 02:32:26 +00:00
|
|
|
document.addEventListener('copy', ((e: ClipboardEvent) => {
|
2018-03-28 13:51:05 +00:00
|
|
|
if(this.copyPlain) {
|
|
|
|
this.copyPlain = false;
|
|
|
|
return;
|
|
|
|
}
|
2018-03-04 02:32:26 +00:00
|
|
|
const selection = document.getSelection();
|
|
|
|
if(selection.isCollapsed) return;
|
|
|
|
const range = selection.getRangeAt(0);
|
2018-04-16 23:14:13 +00:00
|
|
|
let start = range.startContainer, end = range.endContainer;
|
|
|
|
let startValue: string;
|
|
|
|
if(start instanceof HTMLElement) {
|
|
|
|
start = start.childNodes[range.startOffset];
|
|
|
|
startValue = start instanceof HTMLImageElement ? start.alt : scanNode(start.firstChild!, end, range, {});
|
|
|
|
} else
|
|
|
|
startValue = start.nodeValue!.substring(range.startOffset, start === range.endContainer ? range.endOffset : undefined);
|
|
|
|
if(end instanceof HTMLElement) end = end.childNodes[range.endOffset - 1];
|
|
|
|
e.clipboardData.setData('text/plain', copyNode(startValue, start, end, range, {}));
|
2018-03-04 02:32:26 +00:00
|
|
|
e.preventDefault();
|
|
|
|
}) as EventListener);
|
2018-03-28 13:51:05 +00:00
|
|
|
window.addEventListener('keydown', (e) => {
|
|
|
|
if(getKey(e) === Keys.KeyC && e.shiftKey && (e.ctrlKey || e.metaKey) && !e.altKey) {
|
|
|
|
this.copyPlain = true;
|
|
|
|
document.execCommand('copy');
|
2018-04-08 00:22:32 +00:00
|
|
|
e.preventDefault();
|
2018-03-28 13:51:05 +00:00
|
|
|
}
|
|
|
|
});
|
2017-09-02 01:50:31 +00:00
|
|
|
core.register('characters', Characters(core.connection));
|
|
|
|
core.register('channels', Channels(core.connection, core.characters));
|
|
|
|
core.register('conversations', Conversations());
|
|
|
|
core.connection.onEvent('closed', (isReconnect) => {
|
|
|
|
if(isReconnect) (<Modal>this.$refs['reconnecting']).show(true);
|
|
|
|
if(this.connected) core.notifications.playSound('logout');
|
|
|
|
this.connected = false;
|
2018-01-06 16:14:21 +00:00
|
|
|
this.connecting = false;
|
2018-04-08 00:22:32 +00:00
|
|
|
document.title = l('title');
|
2017-09-02 01:50:31 +00:00
|
|
|
});
|
|
|
|
core.connection.onEvent('connecting', async() => {
|
|
|
|
this.connecting = true;
|
2018-01-06 16:14:21 +00:00
|
|
|
if(core.state.settings.notifications) await core.notifications.requestPermission();
|
2017-09-02 01:50:31 +00:00
|
|
|
});
|
|
|
|
core.connection.onEvent('connected', () => {
|
|
|
|
(<Modal>this.$refs['reconnecting']).hide();
|
|
|
|
this.error = '';
|
|
|
|
this.connecting = false;
|
|
|
|
this.connected = true;
|
|
|
|
core.notifications.playSound('login');
|
2018-04-08 00:22:32 +00:00
|
|
|
document.title = l('title.connected', core.connection.character);
|
|
|
|
});
|
|
|
|
core.watch(() => core.conversations.hasNew, (hasNew) => {
|
|
|
|
document.title = (hasNew ? '💬 ' : '') + l(core.connection.isOpen ? 'title.connected' : 'title', core.connection.character);
|
2017-09-02 01:50:31 +00:00
|
|
|
});
|
|
|
|
core.connection.onError((e) => {
|
2018-04-11 19:17:58 +00:00
|
|
|
if((<Error & {request?: object}>e).request !== undefined) {//catch axios network errors
|
|
|
|
this.error = l('login.connectError', errorToString(e));
|
|
|
|
this.connecting = false;
|
|
|
|
} else throw e;
|
2017-09-02 01:50:31 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
cancelReconnect(): void {
|
|
|
|
core.connection.close();
|
|
|
|
(<Modal>this.$refs['reconnecting']).hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
connect(): void {
|
|
|
|
this.connecting = true;
|
2018-04-11 19:17:58 +00:00
|
|
|
core.connection.connect(this.selectedCharacter);
|
2017-09-02 01:50:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|