fchat-rising/site/character_page/character_page.vue

164 lines
7.6 KiB
Vue
Raw Normal View History

<template>
<div class="row character-page" id="pageBody">
<div class="alert alert-info" v-show="loading" style="margin:0 15px;flex:1">Loading character information.</div>
<div class="alert alert-danger" v-show="error" style="margin:0 15px;flex:1">{{error}}</div>
2018-01-06 16:14:21 +00:00
<div class="col-sm-3 col-md-2" v-if="!loading">
<sidebar :character="character" @memo="memo" @bookmarked="bookmarked" :oldApi="oldApi"></sidebar>
</div>
2018-01-06 16:14:21 +00:00
<div class="col-sm-9 col-md-10 profile-body" v-if="!loading">
<div id="characterView">
<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>
<div class="card bg-light">
<div class="card-header">
<tabs class="card-header-tabs" v-model="tab">
<span>Overview</span>
<span>Info</span>
<span v-if="!oldApi">Groups</span>
<span>Images ({{ character.character.image_count }}</span>
<span v-if="character.settings.guestbook">Guestbook</span>
<span v-if="character.is_self || character.settings.show_friends">Friends</span>
</tabs>
</div>
<div class="card-body">
<div class="tab-content">
<div role="tabpanel" class="tab-pane" :class="{active: tab == 0}" id="overview">
<div v-bbcode="character.character.description"></div>
<character-kinks :character="character" :oldApi="oldApi" ref="tab0"></character-kinks>
</div>
<div role="tabpanel" class="tab-pane" :class="{active: tab == 1}" id="infotags">
<character-infotags :character="character" ref="tab1"></character-infotags>
</div>
<div role="tabpanel" class="tab-pane" id="groups" :class="{active: tab == 2}" v-if="!oldApi">
<character-groups :character="character" ref="tab2"></character-groups>
</div>
<div role="tabpanel" class="tab-pane" id="images" :class="{active: tab == 3}">
<character-images :character="character" ref="tab3" :use-preview="imagePreview"></character-images>
</div>
<div v-if="character.settings.guestbook" role="tabpanel" class="tab-pane" :class="{active: tab == 4}"
id="guestbook">
<character-guestbook :character="character" ref="tab4"></character-guestbook>
</div>
<div v-if="character.is_self || character.settings.show_friends" role="tabpanel" class="tab-pane"
:class="{active: tab == 5}" id="friends">
<character-friends :character="character" ref="tab5"></character-friends>
</div>
</div>
</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 {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 Tabs from '../../components/tabs';
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?(): void
}
@Component({
components: {
sidebar: Sidebar,
date: DateDisplay, tabs: Tabs,
'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()
2018-01-06 16:14:21 +00:00
readonly oldApi?: true;
@Prop()
readonly imagePreview?: true;
private shared: SharedStore = Store;
private character: Character | null = null;
loading = true;
error = '';
tab = '0';
beforeMount(): void {
this.shared.authenticated = this.authenticated;
}
2018-01-06 16:14:21 +00:00
async mounted(): Promise<void> {
if(this.character === null) await this._getCharacter();
}
@Watch('tab')
switchTabHook(): void {
const target = <ShowableVueTab>this.$refs[`tab${this.tab}`];
//tslint:disable-next-line:no-unbound-method
if(typeof target.show === 'function') target.show();
}
@Watch('name')
2018-01-06 16:14:21 +00:00
async onCharacterSet(): Promise<void> {
return this._getCharacter();
}
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;
} catch(e) {
if(Utils.isJSONError(e))
this.error = <string>e.response.data.error;
Utils.ajaxError(e, 'Failed to load character information.');
}
}
}
</script>