fchat-rising/site/character_page/character_page.vue

170 lines
8.2 KiB
Vue

<template>
<div class="row character-page" id="pageBody">
<div class="alert alert-info" v-show="loading">Loading character information.</div>
<div class="alert alert-danger" v-show="error">{{error}}</div>
<div class="col-xs-2" v-if="!loading">
<sidebar :character="character" @memo="memo" @bookmarked="bookmarked"></sidebar>
</div>
<div class="col-xs-10" v-if="!loading">
<div id="characterView" class="row">
<div>
<div v-if="character.ban_reason" id="headerBanReason" class="alert alert-warning">
This character has been banned and is not visible to the public. Reason:
<br/> {{ character.ban_reason }}
<template v-if="character.timeout"><br/>Timeout expires:
<date :time="character.timeout"></date>
</template>
</div>
<div v-if="character.block_reason" id="headerBlocked" class="alert alert-warning">
This character has been blocked and is not visible to the public. Reason:
<br/> {{ character.block_reason }}
</div>
<div v-if="character.memo" id="headerCharacterMemo" class="alert alert-info">Memo: {{ character.memo.memo }}</div>
<ul class="nav nav-tabs" role="tablist" style="margin-bottom:5px">
<li role="presentation" class="active"><a href="#overview" aria-controls="overview" role="tab" data-toggle="tab">Overview</a>
</li>
<li role="presentation"><a href="#infotags" aria-controls="infotags" role="tab" data-toggle="tab">Info</a></li>
<li role="presentation" v-if="!hideGroups"><a href="#groups" aria-controls="groups" role="tab" data-toggle="tab">Groups</a>
</li>
<li role="presentation"><a href="#images" aria-controls="images" role="tab"
data-toggle="tab">Images ({{ character.character.image_count }})</a></li>
<li v-if="character.settings.guestbook" role="presentation"><a href="#guestbook" aria-controls="guestbook"
role="tab" data-toggle="tab">Guestbook</a></li>
<li v-if="character.is_self || character.settings.show_friends" role="presentation"><a href="#friends"
aria-controls="friends" role="tab" data-toggle="tab">Friends</a></li>
</ul>
<div class="tab-content">
<div role="tabpanel" class="tab-pane active" id="overview" aria-labeledby="overview-tab">
<div v-bbcode="character.character.description" class="well"></div>
<character-kinks :character="character"></character-kinks>
</div>
<div role="tabpanel" class="tab-pane" id="infotags" aria-labeledby="infotags-tab">
<character-infotags :character="character"></character-infotags>
</div>
<div role="tabpanel" class="tab-pane" id="groups" aria-labeledby="groups-tab" v-if="!hideGroups">
<character-groups :character="character" ref="groups"></character-groups>
</div>
<div role="tabpanel" class="tab-pane" id="images" aria-labeledby="images-tab">
<character-images :character="character" ref="images" :use-preview="imagePreview"></character-images>
</div>
<div v-if="character.settings.guestbook" role="tabpanel" class="tab-pane" id="guestbook"
aria-labeledby="guestbook-tab">
<character-guestbook :character="character" ref="guestbook"></character-guestbook>
</div>
<div v-if="character.is_self || character.settings.show_friends" role="tabpanel" class="tab-pane" id="friends"
aria-labeledby="friends-tab">
<character-friends :character="character" ref="friends"></character-friends>
</div>
</div>
</div>
</div>
</div>
</div>
</template>
<script lang="ts">
import Vue from 'vue';
import Component from 'vue-class-component';
import {Prop, Watch} from 'vue-property-decorator';
import {initCollapse, standardParser} from '../../bbcode/standard';
import * as Utils from '../utils';
import {methods, Store} from './data_store';
import {Character, SharedStore} from './interfaces';
import DateDisplay from '../../components/date_display.vue';
import FriendsView from './friends.vue';
import GroupsView from './groups.vue';
import GuestbookView from './guestbook.vue';
import ImagesView from './images.vue';
import InfotagsView from './infotags.vue';
import CharacterKinksView from './kinks.vue';
import Sidebar from './sidebar.vue';
interface ShowableVueTab extends Vue {
show?(target: Element): void
}
@Component({
components: {
sidebar: Sidebar,
date: DateDisplay,
'character-friends': FriendsView,
'character-guestbook': GuestbookView,
'character-groups': GroupsView,
'character-infotags': InfotagsView,
'character-images': ImagesView,
'character-kinks': CharacterKinksView
}
})
export default class CharacterPage extends Vue {
//tslint:disable:no-null-keyword
@Prop()
private readonly name?: string;
@Prop()
private readonly characterid?: number;
@Prop({required: true})
private readonly authenticated: boolean;
@Prop()
readonly hideGroups?: true;
@Prop()
readonly imagePreview?: true;
private shared: SharedStore = Store;
private character: Character | null = null;
loading = true;
error = '';
beforeMount(): void {
this.shared.authenticated = this.authenticated;
}
mounted(): void {
if(this.character === null) this._getCharacter().then(); //tslint:disable-line:no-floating-promises
}
beforeDestroy(): void {
$('a[data-toggle="tab"]').off('shown.bs.tab', (e) => this.switchTabHook(e));
}
switchTabHook(evt: JQuery.Event): void {
const targetId = (<HTMLElement>evt.target).getAttribute('aria-controls')!;
//tslint:disable-next-line:strict-type-predicates no-unbound-method
if(typeof this.$refs[targetId] !== 'undefined' && typeof (<ShowableVueTab>this.$refs[targetId]).show === 'function')
(<ShowableVueTab>this.$refs[targetId]).show!(<Element>evt.target);
}
@Watch('name')
onCharacterSet(): void {
this._getCharacter().then(); //tslint:disable-line:no-floating-promises
}
memo(memo: {id: number, memo: string}): void {
Vue.set(this.character!, 'memo', memo);
}
bookmarked(state: boolean): void {
Vue.set(this.character!, 'bookmarked', state);
}
private async _getCharacter(): Promise<void> {
if(this.name === undefined || this.name.length === 0)
return;
try {
this.loading = true;
await methods.fieldsGet();
this.character = await methods.characterData(this.name, this.characterid);
standardParser.allowInlines = true;
standardParser.inlines = this.character.character.inlines;
this.loading = false;
this.$nextTick(() => {
$('a[data-toggle="tab"]').on('shown.bs.tab', (e) => this.switchTabHook(e));
initCollapse();
});
} catch(e) {
if(Utils.isJSONError(e))
this.error = <string>e.response.data.error;
Utils.ajaxError(e, 'Failed to load character information.');
}
}
}
</script>