fchat-rising/site/character_page/character_page.vue

753 lines
23 KiB
Vue
Raw Normal View History

<template>
<div class="row character-page" id="pageBody" ref="pageBody">
2019-01-03 17:38:17 +00:00
<div class="col-12" style="min-height:0">
<div class="alert alert-info" v-show="loading">Loading character information.</div>
<div class="alert alert-danger" v-show="error">{{error}}</div>
</div>
2019-06-29 20:59:29 +00:00
<div class="col-md-4 col-lg-3 col-xl-2" v-if="!loading && character && character.character && characterMatch && selfCharacter">
2019-06-29 01:37:41 +00:00
<sidebar :character="character" :characterMatch="characterMatch" @memo="memo" @bookmarked="bookmarked" :oldApi="oldApi"></sidebar>
</div>
2019-06-29 20:59:29 +00:00
<div class="col-md-8 col-lg-9 col-xl-10 profile-body" v-if="!loading && character && character.character && characterMatch && selfCharacter">
2018-01-06 16:14:21 +00:00
<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 class="tab-count" v-if="groupCount !== null">({{ groupCount }})</span></span>
<span>Images <span class="tab-count">({{ character.character.image_count }})</span></span>
<span v-if="character.settings.guestbook">Guestbook <span class="tab-count" v-if="guestbookPostCount !== null">({{ guestbookPostCount }})</span></span>
<span v-if="character.is_self || character.settings.show_friends">Friends <span class="tab-count" v-if="friendCount !== null">({{ friendCount }})</span></span>
</tabs>
</div>
<div class="card-body">
<div class="tab-content">
2019-01-03 17:38:17 +00:00
<div role="tabpanel" class="tab-pane" :class="{active: tab === '0'}" id="overview">
2019-07-04 19:34:21 +00:00
<match-report :characterMatch="characterMatch" :minimized="character.is_self"></match-report>
2018-03-28 13:51:05 +00:00
<div v-bbcode="character.character.description" style="margin-bottom: 10px"></div>
<character-kinks :character="character" :oldApi="oldApi" ref="tab0"></character-kinks>
</div>
2019-01-03 17:38:17 +00:00
<div role="tabpanel" class="tab-pane" :class="{active: tab === '1'}" id="infotags">
2019-06-29 01:37:41 +00:00
<character-infotags :character="character" ref="tab1" :characterMatch="characterMatch"></character-infotags>
</div>
2019-01-03 17:38:17 +00:00
<div role="tabpanel" class="tab-pane" id="groups" :class="{active: tab === '2'}" v-if="!oldApi">
<character-groups :character="character" ref="tab2"></character-groups>
</div>
2019-01-03 17:38:17 +00:00
<div role="tabpanel" class="tab-pane" id="images" :class="{active: tab === '3'}">
<character-images :character="character" ref="tab3" :use-preview="imagePreview"></character-images>
</div>
2019-01-03 17:38:17 +00:00
<div v-if="character.settings.guestbook" role="tabpanel" class="tab-pane" :class="{active: tab === '4'}"
id="guestbook">
<character-guestbook :character="character" :oldApi="oldApi" ref="tab4"></character-guestbook>
</div>
<div v-if="character.is_self || character.settings.show_friends" role="tabpanel" class="tab-pane"
2019-01-03 17:38:17 +00:00
: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 * as _ from 'lodash';
2019-01-03 17:38:17 +00:00
import {Component, Hook, Prop, Watch} from '@f-list/vue-ts';
import Vue from 'vue';
import {standardParser} from '../../bbcode/standard';
import { CharacterCacheRecord } from '../../learn/profile-cache';
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';
import core from '../../chat/core';
2019-07-07 21:44:32 +00:00
import { Matcher, MatchReport } from '../../learn/matcher';
2019-06-29 20:59:29 +00:00
import MatchReportView from './match-report.vue';
2019-06-29 01:37:41 +00:00
const CHARACTER_CACHE_EXPIRE = 7 * 24 * 60 * 60 * 1000;
const CHARACTER_COUNT_CACHE_EXPIRE = 10 * 24 * 60 * 60 * 1000;
2019-07-07 01:37:15 +00:00
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,
2019-06-29 20:59:29 +00:00
'character-kinks': CharacterKinksView,
'match-report': MatchReportView
}
})
export default class CharacterPage extends Vue {
@Prop()
2019-01-03 17:38:17 +00:00
readonly name?: string;
@Prop()
2019-01-03 17:38:17 +00:00
readonly characterid?: number;
@Prop({required: true})
2019-01-03 17:38:17 +00:00
readonly authenticated!: boolean;
@Prop()
2018-01-06 16:14:21 +00:00
readonly oldApi?: true;
@Prop()
readonly imagePreview?: true;
2019-01-03 17:38:17 +00:00
shared: SharedStore = Store;
character: Character | undefined;
loading = true;
error = '';
tab = '0';
guestbookPostCount: number | null = null;
friendCount: number | null = null;
groupCount: number | null = null;
selfCharacter: Character | undefined;
2019-06-29 01:37:41 +00:00
characterMatch: MatchReport | undefined;
2019-01-03 17:38:17 +00:00
@Hook('beforeMount')
beforeMount(): void {
this.shared.authenticated = this.authenticated;
2019-07-12 22:11:55 +00:00
// console.log('Beforemount');
}
2019-01-03 17:38:17 +00:00
@Hook('mounted')
2018-01-06 16:14:21 +00:00
async mounted(): Promise<void> {
await this.load(false);
2019-07-12 22:11:55 +00:00
// console.log('mounted');
}
@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> {
2018-03-28 13:51:05 +00:00
this.tab = '0';
await this.load();
// Kludge kluge
this.$nextTick(
() => {
const el = document.querySelector('.modal .profile-viewer .modal-body');
if (!el) {
console.error('Could not find modal body for profile view');
return;
}
el.scrollTo(0, 0);
}
);
}
2019-07-06 16:49:19 +00:00
async load(mustLoad: boolean = true): Promise<void> {
this.loading = true;
2019-06-29 20:59:29 +00:00
this.error = '';
try {
2019-07-06 16:49:19 +00:00
const due: Promise<void>[] = [];
if(this.name === undefined || this.name.length === 0)
return;
await methods.fieldsGet();
2019-07-06 16:49:19 +00:00
if ((this.selfCharacter === undefined) && (Utils.Settings.defaultCharacter >= 0))
due.push(this.loadSelfCharacter());
2019-07-06 16:49:19 +00:00
if((mustLoad) || (this.character === undefined))
due.push(this._getCharacter());
await Promise.all(due);
} catch(e) {
2019-06-29 01:37:41 +00:00
console.error(e);
this.error = Utils.isJSONError(e) ? <string>e.response.data.error : (<Error>e).message;
Utils.ajaxError(e, 'Failed to load character information.');
}
this.loading = false;
}
2019-07-06 16:49:19 +00:00
async countGuestbookPosts(): Promise<void> {
try {
if ((!this.character) || (!_.get(this.character, 'settings.guestbook'))) {
this.guestbookPostCount = null;
return;
}
const guestbookState = await methods.guestbookPageGet(this.character.character.id, 1, false);
this.guestbookPostCount = guestbookState.posts.length;
// `${guestbookState.posts.length}${guestbookState.nextPage ? '+' : ''}`;
} catch (err) {
console.error(err);
this.guestbookPostCount = null;
}
}
2019-07-06 16:49:19 +00:00
async countGroups(): Promise<void> {
try {
if ((!this.character) || (this.oldApi)) {
this.groupCount = null;
return;
}
const groups = await methods.groupsGet(this.character.character.id);
this.groupCount = groups.length; // `${groups.length}`;
} catch (err) {
console.error(err);
this.groupCount = null;
}
}
2019-07-06 16:49:19 +00:00
async countFriends(): Promise<void> {
try {
2019-07-04 19:34:21 +00:00
if (
(!this.character)
|| (!this.character.is_self) && (!this.character.settings.show_friends)
) {
this.friendCount = null;
return;
}
const friends = await methods.friendsGet(this.character.character.id);
this.friendCount = friends.length; // `${friends.length}`;
} catch (err) {
console.error(err);
this.friendCount = null;
}
}
async updateCounts(name: string): Promise<void> {
await this.countGuestbookPosts();
await this.countFriends();
await this.countGroups();
await core.cache.profileCache.registerCount(
name,
{
lastCounted: Date.now() / 1000,
groupCount: this.groupCount,
friendCount: this.friendCount,
guestbookCount: this.guestbookPostCount
}
);
}
memo(memo: {id: number, memo: string}): void {
Vue.set(this.character!, 'memo', memo);
}
bookmarked(state: boolean): void {
Vue.set(this.character!, 'bookmarked', state);
}
2019-07-06 16:49:19 +00:00
protected async loadSelfCharacter(): Promise<void> {
2019-06-29 01:37:41 +00:00
// console.log('SELF');
2019-07-07 01:37:15 +00:00
// const ownChar = core.characters.ownCharacter;
2019-07-07 01:37:15 +00:00
// this.selfCharacter = await methods.characterData(ownChar.name, -1);
this.selfCharacter = core.characters.ownProfile;
2019-06-29 01:37:41 +00:00
// console.log('SELF LOADED');
this.updateMatches();
}
private async fetchCharacterCache(): Promise<CharacterCacheRecord | null> {
2019-07-07 01:37:15 +00:00
if (!this.name) {
throw new Error('A man must have a name');
}
// tslint:disable-next-line: await-promise
const cachedCharacter = await core.cache.profileCache.get(this.name);
if (cachedCharacter) {
if (Date.now() - cachedCharacter.lastFetched.getTime() <= CHARACTER_CACHE_EXPIRE) {
return cachedCharacter;
2019-07-07 01:37:15 +00:00
}
}
return null;
2019-07-07 01:37:15 +00:00
}
private async _getCharacter(): Promise<void> {
2019-01-03 17:38:17 +00:00
this.character = undefined;
this.friendCount = null;
this.groupCount = null;
this.guestbookPostCount = null;
2019-06-29 20:59:29 +00:00
2019-07-07 01:37:15 +00:00
if (!this.name) {
return;
}
const cache = await this.fetchCharacterCache();
this.character = (cache)
? cache.character
: await methods.characterData(this.name, this.characterid, false);
2019-07-07 01:37:15 +00:00
standardParser.allowInlines = true;
standardParser.inlines = this.character.character.inlines;
2019-06-29 01:37:41 +00:00
if (
(cache)
&& (cache.counts)
&& (cache.counts.lastCounted)
&& ((Date.now() / 1000) - cache.counts.lastCounted < CHARACTER_COUNT_CACHE_EXPIRE)
) {
this.guestbookPostCount = cache.counts.guestbookCount;
this.friendCount = cache.counts.friendCount;
this.groupCount = cache.counts.groupCount;
} else {
// No awaits on these on purpose:
// tslint:disable-next-line no-floating-promises
this.updateCounts(this.name);
}
2019-07-04 19:34:21 +00:00
// console.log('LoadChar', this.name, this.character);
2019-07-04 19:34:21 +00:00
this.updateMatches();
2019-06-29 01:37:41 +00:00
}
private updateMatches(): void {
2019-07-06 16:49:19 +00:00
if ((!this.selfCharacter) || (!this.character))
2019-06-29 01:37:41 +00:00
return;
2019-06-29 20:59:29 +00:00
this.characterMatch = Matcher.generateReport(this.selfCharacter.character, this.character.character);
2019-07-07 01:37:15 +00:00
// console.log('Match', this.selfCharacter.character.name, this.character.character.name, this.characterMatch);
}
}
</script>
<style lang="scss">
.compare-highlight-block {
margin-bottom: 3px;
.quick-compare-block button {
margin-left: 2px;
}
}
.character-kinks-block {
i.fa {
margin-right: 0.25rem;
}
.character-kink {
.popover {
min-width: 200px;
margin-bottom: 0;
padding-bottom: 0;
}
p {
line-height: 125%;
}
p:last-child {
margin-bottom:0;
}
2019-06-30 19:15:23 +00:00
&.comparison-result {
margin: -4px;
padding: 4px;
padding-top: 2px;
padding-bottom: 2px;
margin-top: 1px;
margin-bottom: 1px;
border-radius: 3px;
}
}
}
.expanded-custom-kink {
.custom-kink {
margin-top: 14px;
margin-bottom: 14px;
}
}
.custom-kink {
&:first-child {
margin-top: 0;
}
&:last-child {
margin-bottom: 0;
}
.kink-name {
font-weight: bold;
color: #f2cd00;
}
i {
color: #f2cd00;
}
margin-top: 7px;
margin-bottom: 7px;
margin-left: -6px;
margin-right: -6px;
border: 1px rgba(255, 255, 255, 0.1) solid;
border-radius: 2px;
/* border-collapse: collapse; */
padding: 5px;
}
.stock-kink {
.kink-name, i {
color: #ededf6;
font-weight: normal;
}
}
.kink-custom-desc {
display: block;
font-weight: normal;
font-size: 0.9rem;
color: rgba(255, 255, 255, 0.7);
line-height: 125%;
}
.infotag-label {
display: block;
/* margin-bottom: 1rem; */
font-weight: normal !important;
line-height: 120%;
font-size: 85%;
color: rgba(255, 255, 255, 0.7);
}
.infotag-value {
display: block;
margin-bottom: 1rem;
font-weight: bold;
line-height: 120%;
}
.quick-info-value {
display: block;
font-weight: bold;
}
.quick-info-label {
display: block;
/* margin-bottom: 1rem; */
font-weight: normal !important;
line-height: 120%;
font-size: 85%;
color: rgba(255, 255, 255, 0.7);
}
.quick-info {
margin-bottom: 1rem;
font-size: 0.9rem;
color: rgba(255, 255, 255, 0.7);
}
.guestbook-post {
margin-bottom: 15px;
margin-top: 15px;
background-color: rgba(0,0,0,0.15);
border-radius: 5px;
padding: 15px;
border: 1px solid rgba(255, 255, 255, 0.1);
.characterLink {
font-size: 20pt;
}
.guestbook-timestamp {
color: rgba(255, 255, 255, 0.3);
font-size: 85%
}
.guestbook-message {
margin-top: 10px;
}
.guestbook-reply {
margin-top: 20px;
background-color: rgba(0,0,0, 0.1);
padding: 15px;
border-radius: 4px;
}
}
.contact-block {
margin-top: 25px !important;
margin-bottom: 25px !important;
.contact-method {
font-size: 80%;
display: block;
margin-bottom: 2px;
img {
border-radius: 2px;
}
}
}
#character-page-sidebar .character-list-block {
.character-avatar.icon {
height: 43px !important;
width: 43px !important;
border-radius: 3px;
}
.characterLink {
font-size: 85%;
padding-left: 3px;
}
}
.character-images {
2019-06-30 19:15:23 +00:00
column-width: auto;
column-count: 2;
column-gap: 0.5rem;
.character-image-wrapper {
display: inline-block;
background-color: rgba(0,0,0, 0.2);
border-radius: 5px;
box-sizing: border-box;
margin: 5px;
a {
border: none;
img {
max-width: 100% !important;
width: 100% !important;
height: auto !important;
object-fit: contain;
object-position: top center;
vertical-align: top !important;
border-radius: 6px;
}
}
.image-description {
font-size: 85%;
padding-top: 5px;
padding-bottom: 5px;
padding-left: 10px;
padding-right: 10px;
}
}
}
2019-06-29 01:37:41 +00:00
.infotag {
2019-06-29 20:59:29 +00:00
&.match-score {
padding-top: 2px;
padding-left: 4px;
padding-right: 4px;
margin-left: -4px;
margin-right: -4px;
border-radius: 2px;
padding-bottom: 2px;
margin-bottom: 1rem;
.infotag-value {
margin-bottom: 0;
}
}
}
.match-report {
display: flex;
flex-direction: row;
background-color: rgba(0,0,0,0.2);
/* width: 100%; */
margin-top: -1.2rem;
margin-left: -1.2rem;
margin-right: -1.2rem;
padding: 1rem;
margin-bottom: 1rem;
padding-bottom: 0;
padding-top: 0.5rem;
2019-06-29 21:28:47 +00:00
.thumbnail {
width: 50px;
height: 50px;
}
&.minimized {
height: 0;
overflow: hidden;
background-color: transparent;
.vs, .scores {
display: none;
}
}
2019-06-29 20:59:29 +00:00
h3 {
font-size: 1.25rem;
2019-06-29 01:37:41 +00:00
}
2019-06-29 21:28:47 +00:00
.minimize-btn {
position: absolute;
display: block;
right: 0.5rem;
background-color: rgba(0,0,0,0.2);
padding: 0.4rem;
padding-top: 0.2rem;
padding-bottom: 0.2rem;
font-size: 0.8rem;
color: rgba(255, 255, 255, 0.7);
border-radius: 4px;
z-index: 1000;
}
2019-06-29 20:59:29 +00:00
.scores {
float: left;
flex: 1;
margin-right: 1rem;
max-width: 25rem;
ul {
padding: 0;
2019-06-29 21:28:47 +00:00
padding-left: 0rem;
2019-06-29 20:59:29 +00:00
list-style: none;
}
.match-score {
font-size: 0.85rem;
border-radius: 2px;
margin-bottom: 4px;
padding: 2px;
padding-left: 4px;
padding-right: 4px;
span {
color: white;
font-weight: bold;
}
}
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
.vs {
margin-left: 1rem;
margin-right: 1rem;
text-align: center;
font-size: 5rem;
line-height: 0;
color: rgba(255, 255, 255, 0.3);
margin-top: auto;
margin-bottom: auto;
font-style: italic;
font-family: 'Times New Roman', Georgia, serif;
}
2019-06-29 01:37:41 +00:00
}
2019-06-30 19:15:23 +00:00
.character-kinks-block .character-kink.comparison-favorite,
.match-report .scores .match-score.match,
.infotag.match {
background-color: rgb(0, 142, 0);
border: solid 1px rgb(0, 113, 0);
// background-color: #007700;
// border: 1px solid #003e00;
}
.character-kinks-block .character-kink.comparison-yes,
.match-report .scores .match-score.weak-match,
.infotag.weak-match {
background-color: rgb(0, 80, 0);
border: 1px solid rgb(0, 64, 0);
// background-color: #004200;
// border: 1px solid #002900;
}
.character-kinks-block .character-kink.comparison-maybe,
.match-report .scores .match-score.weak-mismatch,
.infotag.weak-mismatch {
background-color: rgb(152, 134, 0);
border: 1px solid rgb(142, 126, 0);
// border: 1px solid #613e00;
// background-color: #905d01;
}
.character-kinks-block .character-kink.comparison-no,
.match-report .scores .match-score.mismatch,
.infotag.mismatch {
background-color: rgb(171, 0, 0);
border: 1px solid rgb(128, 0, 0);
// border: 1px solid #420200;
// background-color: #710300;
}
.tab-count {
color: rgba(255, 255, 255, 0.5);
}
</style>