fchat-rising/chat/UserMenu.vue

215 lines
9.5 KiB
Vue
Raw Normal View History

2017-09-02 01:50:31 +00:00
<template>
<div>
<div id="userMenu" class="dropdown-menu" v-show="showContextMenu" :style="position"
style="position:fixed;padding:10px 10px 5px;display:block;width:200px;z-index:1100" ref="menu">
<div v-if="character">
<div style="min-height: 65px;" @click.stop>
<img :src="characterImage" style="width: 60px; height:60px; margin-right: 5px; float: left;" v-if="showAvatars"/>
<h4 style="margin:0;">{{character.name}}</h4>
{{l('status.' + character.status)}}
</div>
<bbcode :text="character.statusText" @click.stop></bbcode>
<ul class="dropdown-menu border-top" role="menu"
style="display:block; position:static; border-width:1px 0 0 0; box-shadow:none; padding:0; width:100%; border-radius:0;">
<li><a tabindex="-1" :href="profileLink" target="_blank" v-if="showProfileFirst">
<span class="fa fa-fw fa-user"></span>{{l('user.profile')}}</a></li>
<li><a tabindex="-1" href="#" @click.prevent="openConversation(true)">
<span class="fa fa-fw fa-comment"></span>{{l('user.messageJump')}}</a></li>
<li><a tabindex="-1" href="#" @click.prevent="openConversation(false)">
<span class="fa fa-fw fa-plus"></span>{{l('user.message')}}</a></li>
<li><a tabindex="-1" :href="profileLink" target="_blank" v-if="!showProfileFirst">
<span class="fa fa-fw fa-user"></span>{{l('user.profile')}}</a></li>
<li><a tabindex="-1" href="#" @click.prevent="showMemo">
<span class="fa fa-fw fa-sticky-note-o"></span>{{l('user.memo')}}</a></li>
<li><a tabindex="-1" href="#" @click.prevent="setBookmarked">
<span class="fa fa-fw fa-bookmark-o"></span>{{l('user.' + (character.isBookmarked ? 'unbookmark' : 'bookmark'))}}
</a></li>
<li><a tabindex="-1" href="#" @click.prevent="setIgnored">
<span class="fa fa-fw fa-minus-circle"></span>{{l('user.' + (character.isIgnored ? 'unignore' : 'ignore'))}}
</a></li>
<li><a tabindex="-1" href="#" @click.prevent="report">
<span class="fa fa-fw fa-exclamation-triangle"></span>{{l('user.report')}}</a></li>
<li v-show="isChannelMod"><a tabindex="-1" href="#" @click.prevent="channelKick">
<span class="fa fa-fw fa-ban"></span>{{l('user.channelKick')}}</a></li>
<li v-show="isChatOp"><a tabindex="-1" href="#" @click.prevent="chatKick" style="color:#f00">
<span class="fa fa-fw fa-trash-o"></span>{{l('user.chatKick')}}</a>
</li>
</ul>
</div>
</div>
<modal :action="l('user.memo.action')" ref="memo" :disabled="memoLoading" @submit="updateMemo">
<div style="float:right;text-align:right;">{{getByteLength(memo)}} / 1000</div>
<textarea class="form-control" v-model="memo" :disabled="memoLoading" maxlength="1000"></textarea>
</modal>
</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 {BBCodeView} from './bbcode';
import {characterImage, errorToString, getByteLength, profileLink} from './common';
import core from './core';
import {Channel, Character} from './interfaces';
import l from './localize';
import ReportDialog from './ReportDialog.vue';
@Component({
components: {bbcode: BBCodeView, modal: Modal}
})
export default class UserMenu extends Vue {
//tslint:disable:no-null-keyword
@Prop({required: true})
readonly reportDialog: ReportDialog;
l = l;
showContextMenu = false;
getByteLength = getByteLength;
character: Character | null = null;
position = {left: '', top: ''};
characterImage: string | null = null;
2017-10-16 23:58:57 +00:00
touchTimer: number | undefined;
2017-09-02 01:50:31 +00:00
channel: Channel | null = null;
memo = '';
memoId: number;
memoLoading = false;
openConversation(jump: boolean): void {
const conversation = core.conversations.getPrivate(this.character!);
if(jump) conversation.show();
}
setIgnored(): void {
core.connection.send('IGN', {action: this.character!.isIgnored ? 'delete' : 'add', character: this.character!.name});
}
setBookmarked(): void {
core.connection.queryApi(`bookmark-${this.character!.isBookmarked ? 'remove' : 'add'}.php`, {name: this.character!.name})
.catch((e: object) => alert(errorToString(e)));
}
report(): void {
this.reportDialog.report(this.character!);
}
channelKick(): void {
core.connection.send('CKU', {channel: this.channel!.id, character: this.character!.name});
}
chatKick(): void {
core.connection.send('KIK', {character: this.character!.name});
}
async showMemo(): Promise<void> {
this.memoLoading = true;
this.memo = '';
(<Modal>this.$refs['memo']).show();
try {
const memo = <{note: string, id: number}>await core.connection.queryApi('character-memo-get.php',
{target: this.character!.name});
this.memoId = memo.id;
this.memo = memo.note;
this.memoLoading = false;
} catch(e) {
alert(errorToString(e));
}
}
updateMemo(): void {
core.connection.queryApi('character-memo-save.php', {target: this.memoId, note: this.memo})
.catch((e: object) => alert(errorToString(e)));
}
get isChannelMod(): boolean {
if(this.channel === null) return false;
if(core.characters.ownCharacter.isChatOp) return true;
const member = this.channel.members[core.connection.character];
return member !== undefined && member.rank > Channel.Rank.Member;
}
get isChatOp(): boolean {
return core.characters.ownCharacter.isChatOp;
}
get showProfileFirst(): boolean {
return core.state.settings.clickOpensMessage;
}
get showAvatars(): boolean {
return core.state.settings.showAvatars;
}
get profileLink(): string | undefined {
return profileLink(this.character!.name);
}
handleEvent(e: MouseEvent | TouchEvent): void {
2017-10-16 23:58:57 +00:00
const touch = e instanceof TouchEvent ? e.changedTouches[0] : e;
2017-09-02 01:50:31 +00:00
let node = <Node & {character?: Character, channel?: Channel}>touch.target;
while(node !== document.body) {
2017-10-18 23:29:28 +00:00
if(e.type === 'touchstart' && node === this.$refs['menu']) return;
2017-09-02 01:50:31 +00:00
if(node.character !== undefined || node.parentNode === null) break;
node = node.parentNode;
}
if(node.character === undefined) {
this.showContextMenu = false;
return;
}
switch(e.type) {
case 'click':
2017-10-16 23:58:57 +00:00
this.onClick(node.character);
2017-09-02 01:50:31 +00:00
break;
case 'touchstart':
2017-10-16 23:58:57 +00:00
this.touchTimer = window.setTimeout(() => {
this.openMenu(touch, node.character!, node.channel);
this.touchTimer = undefined;
}, 500);
break;
case 'touchend':
if(this.touchTimer !== undefined) {
clearTimeout(this.touchTimer);
this.touchTimer = undefined;
this.onClick(node.character);
}
2017-09-02 01:50:31 +00:00
break;
case 'contextmenu':
this.openMenu(touch, node.character, node.channel);
}
e.preventDefault();
}
2017-10-16 23:58:57 +00:00
private onClick(character: Character): void {
this.character = character;
if(core.state.settings.clickOpensMessage) this.openConversation(true);
else window.open(this.profileLink);
this.showContextMenu = false;
}
2017-09-02 01:50:31 +00:00
private openMenu(touch: MouseEvent | Touch, character: Character, channel: Channel | undefined): void {
this.channel = channel !== undefined ? channel : null;
this.character = character;
this.characterImage = null;
this.showContextMenu = true;
this.position = {left: `${touch.clientX}px`, top: `${touch.clientY}px`};
this.$nextTick(() => {
const menu = <HTMLElement>this.$refs['menu'];
this.characterImage = characterImage(character.name);
if((parseInt(this.position.left, 10) + menu.offsetWidth) > window.innerWidth)
this.position.left = `${window.innerWidth - menu.offsetWidth - 1}px`;
if((parseInt(this.position.top, 10) + menu.offsetHeight) > window.innerHeight)
this.position.top = `${window.innerHeight - menu.offsetHeight - 1}px`;
});
}
}
</script>
<style>
#userMenu li a {
padding: 3px 0;
}
.user-view {
cursor: pointer;
}
</style>