<template>
    <div>
        <div id="userMenu" class="list-group" v-show="showContextMenu" :style="position" v-if="character"
            style="position:fixed;padding:10px 10px 5px;display:block;width:220px;z-index:1100" ref="menu">
            <div style="min-height: 65px;padding:5px;overflow:auto" class="list-group-item" @click.stop>
                <img :src="characterImage" style="width:60px;height:60px;margin-right:5px;float:left" v-if="showAvatars"/>
                <h5 style="margin:0;line-height:1">{{character.name}}</h5>
                {{l('status.' + character.status)}}
            </div>
            <bbcode id="userMenuStatus" :text="character.statusText" v-show="character.statusText" class="list-group-item"
                style="max-height:200px;overflow:auto;clear:both"></bbcode>
            <a tabindex="-1" :href="profileLink" target="_blank" v-if="showProfileFirst" class="list-group-item list-group-item-action">
                <span class="fa fa-fw fa-user"></span>{{l('user.profile')}}</a>
            <a tabindex="-1" href="#" @click.prevent="openConversation(true)" class="list-group-item list-group-item-action">
                <span class="fa fa-fw fa-comment"></span>{{l('user.messageJump')}}</a>
            <a tabindex="-1" href="#" @click.prevent="openConversation(false)" class="list-group-item list-group-item-action">
                <span class="fa fa-fw fa-plus"></span>{{l('user.message')}}</a>
            <a tabindex="-1" :href="profileLink" target="_blank" v-if="!showProfileFirst" class="list-group-item list-group-item-action">
                <span class="fa fa-fw fa-user"></span>{{l('user.profile')}}</a>
            <a tabindex="-1" href="#" @click.prevent="showMemo()" class="list-group-item list-group-item-action">
                <span class="far fa-fw fa-sticky-note"></span>{{l('user.memo')}}</a>
            <a tabindex="-1" href="#" @click.prevent="setBookmarked()" class="list-group-item list-group-item-action">
                <span class="far fa-fw fa-bookmark"></span>{{l('user.' + (character.isBookmarked ? 'unbookmark' : 'bookmark'))}}</a>
            <a tabindex="-1" href="#" @click.prevent="setHidden()" class="list-group-item list-group-item-action" v-show="!isChatOp">
                <span class="fa fa-fw fa-eye-slash"></span>{{l('user.' + (isHidden ? 'unhide' : 'hide'))}}</a>
            <a tabindex="-1" href="#" @click.prevent="report()" class="list-group-item list-group-item-action" style="border-top-width:1px">
                <span class="fa fa-fw fa-exclamation-triangle"></span>{{l('user.report')}}</a>
            <a tabindex="-1" href="#" @click.prevent="setIgnored()" class="list-group-item list-group-item-action">
                <span class="fa fa-fw fa-minus-circle"></span>{{l('user.' + (character.isIgnored ? 'unignore' : 'ignore'))}}</a>
            <a tabindex="-1" href="#" @click.prevent="channelKick()" class="list-group-item list-group-item-action" v-show="isChannelMod">
                <span class="fa fa-fw fa-ban"></span>{{l('user.channelKick')}}</a>
            <a tabindex="-1" href="#" @click.prevent="chatKick()" style="color:#f00" class="list-group-item list-group-item-action"
                v-show="isChatOp"><span class="fas fa-fw fa-trash"></span>{{l('user.chatKick')}}</a>
        </div>
        <modal :action="l('user.memo.action')" ref="memo" :disabled="memoLoading" @submit="updateMemo" dialogClass="w-100">
            <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 {Component, Prop} from '@f-list/vue-ts';
    import Vue from 'vue';
    import {BBCodeView} from '../bbcode/view';
    import Modal from '../components/Modal.vue';
    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(core.bbCodeParser), modal: Modal}
    })
    export default class UserMenu extends Vue {
        @Prop({required: true})
        readonly reportDialog!: ReportDialog;
        l = l;
        showContextMenu = false;
        getByteLength = getByteLength;
        character: Character | undefined;
        position = {left: '', top: ''};
        characterImage: string | undefined;
        touchedElement: HTMLElement | undefined;
        channel: Channel | undefined;
        memo = '';
        memoId = 0;
        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)));
        }

        setHidden(): void {
            const index = core.state.hiddenUsers.indexOf(this.character!.name);
            if(index !== -1) core.state.hiddenUsers.splice(index, 1);
            else core.state.hiddenUsers.push(this.character!.name);
        }

        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 = await core.connection.queryApi<{note: string | null, id: number}>('character-memo-get2.php',
                    {target: this.character!.name});
                this.memoId = memo.id;
                this.memo = memo.note !== null ? 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 === undefined) 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 isHidden(): boolean {
            return core.state.hiddenUsers.indexOf(this.character!.name) !== -1;
        }

        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 {
            const touch = e.type === 'touchstart' ? (<TouchEvent>e).changedTouches[0] : <MouseEvent>e;
            let node = <HTMLElement & {character?: Character, channel?: Channel, touched?: boolean}>touch.target;
            while(node !== document.body) {
                if(e.type !== 'click' && node === this.$refs['menu'] || node.id === 'userMenuStatus') return;
                if(node.character !== undefined || node.dataset['character'] !== undefined || node.parentNode === null) break;
                node = node.parentElement!;
            }
            if(node.dataset['touch'] === 'false' && e.type !== 'contextmenu') return;
            if(!node.character)
                if(node.dataset['character'] !== undefined) node.character = core.characters.get(node.dataset['character']!);
                else {
                    this.showContextMenu = false;
                    this.touchedElement = undefined;
                    return;
                }
            switch(e.type) {
                case 'click':
                    if(node.dataset['character'] === undefined)
                        if(node === this.touchedElement) this.openMenu(touch, node.character, node.channel || undefined);
                        else this.onClick(node.character);
                    e.preventDefault();
                    break;
                case 'touchstart':
                    this.touchedElement = node;
                    break;
                case 'contextmenu':
                    this.openMenu(touch, node.character, node.channel || undefined);
                    e.preventDefault();
            }
        }

        private onClick(character: Character): void {
            this.character = character;
            if(core.state.settings.clickOpensMessage) this.openConversation(true);
            else window.open(this.profileLink);
            this.showContextMenu = false;
        }

        private openMenu(touch: MouseEvent | Touch, character: Character, channel: Channel | undefined): void {
            this.channel = channel;
            this.character = character;
            this.characterImage = undefined;
            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 .list-group-item {
        padding: 3px;
    }

    #userMenu .list-group-item-action {
        border-top-width: 0;
        z-index: -1;
    }
</style>