mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-14 05:07:22 +01:00
* fix(content): textual tweaks and updates * fix(link): direct to FAQ instead of wiki * fix(faq): correct Markdown * Show orb of rebirth confirmation modal after use (window refresh) * Set and check rebirth confirmation modal from localstorage Set and check rebirth confirmation modal from localstorage after window reload * Don't show orb of rebirth confirmation modal until page reloads * message effective limit optimization * Keep max limit for web (400 recent messages) * Fix amount of messages initially being shown * PM_PER_PAGE set to 50 * Increases number of messages in inbox test * Increases number of messages for inbox pagination test * Set and check rebirth confirmation modal from localstorage Set and check rebirth confirmation modal from localstorage after window reload * Don't show orb of rebirth confirmation modal until page reloads * message effective limit optimization * Keep max limit for web (400 recent messages) * Add UUID validation for 'before' query parameter * add party message stress test tool in admin panel * lint * add MAX_PM_COUNT of 400, admin tool for stress testing messages * comment * update stress test inbox message tool to use logged in user * comment --------- Co-authored-by: Kalista Payne <kalista@habitica.com>
1297 lines
34 KiB
Vue
1297 lines
34 KiB
Vue
<template>
|
|
<div id="private-message">
|
|
<div class="floating-header-shadow"></div>
|
|
<div class="header-bar d-flex w-100">
|
|
<!-- changing w-25 would also need changes in .left-header.w-25 -->
|
|
<div class="d-flex left-header">
|
|
<div
|
|
v-once
|
|
class="mail-icon svg-icon"
|
|
v-html="icons.mail"
|
|
></div>
|
|
<h2
|
|
v-once
|
|
class="flex-fill text-center mail-icon-label"
|
|
>
|
|
{{ $t('messages') }}
|
|
</h2>
|
|
|
|
<button
|
|
class="btn btn-secondary plus-button"
|
|
:class="{'new-message-mode':showStartNewConversationInput}"
|
|
@click="triggerStartNewConversationState()"
|
|
>
|
|
<div
|
|
class="svg-icon icon-10 color"
|
|
v-html="icons.positive"
|
|
></div>
|
|
</button>
|
|
</div>
|
|
|
|
<start-new-conversation-input-header
|
|
v-if="showStartNewConversationInput"
|
|
@startNewConversation="startConversationByUsername($event)"
|
|
@cancelNewConversation="showStartNewConversationInput = false"
|
|
/>
|
|
<div
|
|
v-else-if="selectedConversation && selectedConversation.key"
|
|
class="d-flex selected-conversion"
|
|
>
|
|
<router-link
|
|
:to="{'name': 'userProfile', 'params': {'userId': selectedConversation.key}}"
|
|
>
|
|
<face-avatar
|
|
v-if="selectedConversation.userStyles"
|
|
:member="selectedConversation.userStyles"
|
|
:class="selectedConversationFaceAvatarClass"
|
|
/>
|
|
</router-link>
|
|
<user-link
|
|
:backer="selectedConversation.backer"
|
|
:contributor="selectedConversation.contributor"
|
|
:name="selectedConversation.name"
|
|
:user="selectedConversation"
|
|
:user-id="selectedConversation.key"
|
|
:hide-tooltip="true"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div class="d-flex content">
|
|
<div class="sidebar d-flex flex-column">
|
|
<div class="disable-background">
|
|
<toggle-switch
|
|
:label="optTextSet.switchDescription"
|
|
:checked="user.inbox.optOut"
|
|
:hover-text="optTextSet.popoverText"
|
|
@change="toggleOpt()"
|
|
/>
|
|
</div>
|
|
<pm-conversations-list
|
|
:filters-conversations="filtersConversations"
|
|
:selected-conversation="selectedConversation"
|
|
@selectConversation="selectConversation($event)"
|
|
/>
|
|
<button
|
|
v-if="canLoadMoreConversations"
|
|
class="btn btn-secondary"
|
|
@click="loadConversations()"
|
|
>
|
|
{{ $t('loadMore') }}
|
|
</button>
|
|
</div>
|
|
|
|
<div class="messages-column d-flex flex-column align-items-center">
|
|
<div
|
|
v-if="user.inbox.optOut"
|
|
class="disable-background-in-message-list"
|
|
>
|
|
<span
|
|
v-once
|
|
class="caption"
|
|
> {{ $t('PMDisabledCaptionTitle') }}. </span>
|
|
<span
|
|
v-once
|
|
class="text"
|
|
> {{ $t('PMDisabledCaptionText') }} </span>
|
|
</div>
|
|
|
|
<pm-empty-state
|
|
v-if="uiState === UI_STATES.NO_CONVERSATIONS"
|
|
:chat-revoked="user.flags.chatRevoked"
|
|
@newMessageClicked="showStartNewConversationInput = true"
|
|
/>
|
|
|
|
<pm-new-message-started
|
|
v-if="uiState === UI_STATES.START_NEW_CONVERSATION && selectedConversation.userStyles"
|
|
:member-obj="selectedConversation.userStyles"
|
|
/>
|
|
|
|
<div
|
|
v-if="uiState === UI_STATES.NO_CONVERSATIONS_SELECTED"
|
|
class="empty-messages full-height m-auto text-center"
|
|
>
|
|
<div class="no-messages-box">
|
|
<div
|
|
v-once
|
|
class="svg-icon envelope"
|
|
v-html="icons.messageIcon"
|
|
></div>
|
|
<h2>{{ placeholderTexts.title }}</h2>
|
|
<p v-html="placeholderTexts.description"></p>
|
|
</div>
|
|
</div>
|
|
|
|
<messageList
|
|
v-if="selectedConversation && selectedConversationMessages.length > 0"
|
|
ref="chatscroll"
|
|
class="message-scroll"
|
|
:chat="selectedConversationMessages"
|
|
:conversation-opponent-user="selectedConversation.userStyles"
|
|
:can-load-more="canLoadMore"
|
|
:is-loading="messagesLoading"
|
|
@message-removed="messageRemoved"
|
|
@message-liked="messageLiked"
|
|
@triggerLoad="infiniteScrollTrigger"
|
|
/>
|
|
|
|
<pm-disabled-state
|
|
v-if="disabledTexts?.showBottomInfo"
|
|
:disabled-texts="disabledTexts"
|
|
/>
|
|
<div
|
|
v-if="shouldShowInputPanel"
|
|
class="full-width"
|
|
>
|
|
<div
|
|
class="new-message-row d-flex align-items-center"
|
|
>
|
|
<textarea
|
|
ref="textarea"
|
|
v-model="newMessage"
|
|
dir="auto"
|
|
class="flex-fill"
|
|
:placeholder="$t('needsTextPlaceholder')"
|
|
:maxlength="MAX_MESSAGE_LENGTH"
|
|
:class="{'has-content': newMessage.trim() !== '', 'disabled': newMessageDisabled}"
|
|
@keyup.ctrl.enter="sendPrivateMessage()"
|
|
>
|
|
</textarea>
|
|
</div>
|
|
<div
|
|
class="sub-new-message-row d-flex"
|
|
>
|
|
<div
|
|
v-once
|
|
class="guidelines flex-fill"
|
|
v-html="$t('communityGuidelinesIntro')"
|
|
></div>
|
|
<button
|
|
class="btn btn-primary"
|
|
:class="{'disabled':newMessageDisabled || newMessage === ''}"
|
|
@click="sendPrivateMessage()"
|
|
>
|
|
{{ $t('sendMessage') }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss">
|
|
@import '@/assets/scss/colors';
|
|
@import '@/assets/scss/variables';
|
|
|
|
$pmHeaderHeight: 56px;
|
|
|
|
// Content of Private Message should be always full-size (minus the toolbar/resting banner)
|
|
|
|
#private-message {
|
|
height: calc(100vh - #{$menuToolbarHeight} -
|
|
var(--banner-gift-promo-height, 0px) -
|
|
var(--banner-damage-paused-height, 0px) -
|
|
var(--banner-gems-promo-height, 0px)
|
|
); // css variable magic :), must be 0px, 0 alone won't work
|
|
|
|
.content {
|
|
flex: 1;
|
|
height: calc(100vh - #{$menuToolbarHeight} - #{$pmHeaderHeight} -
|
|
var(--banner-gift-promo-height, 0px) -
|
|
var(--banner-damage-paused-height, 0px) -
|
|
var(--banner-gems-promo-height, 0px)
|
|
);
|
|
}
|
|
|
|
.disable-background {
|
|
.toggle-switch-description {
|
|
white-space: nowrap;
|
|
text-overflow: ellipsis;
|
|
overflow: hidden;
|
|
flex: 1;
|
|
}
|
|
|
|
.toggle-switch-outer {
|
|
display: flex;
|
|
}
|
|
|
|
}
|
|
|
|
.modal-body {
|
|
padding: 0rem;
|
|
}
|
|
|
|
.modal-content {
|
|
width: 66vw;
|
|
}
|
|
|
|
.modal-dialog {
|
|
margin: 10vh 15vw 0rem;
|
|
}
|
|
|
|
.modal-header {
|
|
padding: 1rem 0rem;
|
|
|
|
.close {
|
|
cursor: pointer;
|
|
margin: 0rem 1.5rem;
|
|
min-width: 0.75rem;
|
|
padding: 0rem;
|
|
width: 0.75rem;
|
|
}
|
|
}
|
|
|
|
.toggle-switch-description {
|
|
font-size: 14px;
|
|
font-weight: bold;
|
|
font-style: normal;
|
|
font-stretch: normal;
|
|
line-height: 1.43;
|
|
letter-spacing: normal;
|
|
color: $gray-50;
|
|
}
|
|
}
|
|
</style>
|
|
|
|
<style lang="scss" scoped>
|
|
@import '@/assets/scss/colors.scss';
|
|
@import '@/assets/scss/tiers.scss';
|
|
@import '@/assets/scss/variables.scss';
|
|
|
|
$pmHeaderHeight: 56px;
|
|
$background: $white;
|
|
|
|
.header-bar {
|
|
height: 56px;
|
|
background-color: $white;
|
|
padding-left: 1.5rem;
|
|
padding-right: 1.5rem;
|
|
align-items: center;
|
|
|
|
.mail-icon {
|
|
width: 32px;
|
|
height: 24px;
|
|
object-fit: contain;
|
|
}
|
|
|
|
.mail-icon-label {
|
|
margin-bottom: 0;
|
|
}
|
|
|
|
.placeholder.svg-icon {
|
|
width: 32px;
|
|
}
|
|
}
|
|
|
|
.full-height {
|
|
height: 100%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: center;
|
|
}
|
|
|
|
.user-link {
|
|
margin-left: 12px;
|
|
}
|
|
|
|
.selected-conversion {
|
|
justify-content: center;
|
|
align-items: center;
|
|
}
|
|
|
|
#private-message {
|
|
background-color: $background;
|
|
position: relative;
|
|
}
|
|
|
|
.disable-background {
|
|
.toggle-switch-description {
|
|
white-space: nowrap;
|
|
text-overflow: ellipsis;
|
|
overflow: hidden;
|
|
flex: 1;
|
|
}
|
|
|
|
.toggle-switch-outer {
|
|
display: block !important;
|
|
}
|
|
|
|
.toggle-switch {
|
|
float: right !important;
|
|
}
|
|
|
|
}
|
|
|
|
.modal-body {
|
|
padding: 0rem;
|
|
}
|
|
|
|
.modal-content {
|
|
width: 66vw;
|
|
}
|
|
|
|
.modal-dialog {
|
|
margin: 10vh 15vw 0rem;
|
|
}
|
|
|
|
.modal-header {
|
|
padding: 1rem 0rem;
|
|
|
|
.close {
|
|
cursor: pointer;
|
|
margin: 0rem 1.5rem;
|
|
min-width: 0.75rem;
|
|
padding: 0rem;
|
|
width: 0.75rem;
|
|
}
|
|
}
|
|
|
|
.toggle-switch-description {
|
|
font-size: 14px;
|
|
font-weight: bold;
|
|
font-style: normal;
|
|
font-stretch: normal;
|
|
line-height: 1.43;
|
|
letter-spacing: normal;
|
|
color: $gray-50;
|
|
}
|
|
|
|
.empty-messages {
|
|
flex-flow: column;
|
|
justify-content: center;
|
|
|
|
h3, p {
|
|
color: $gray-200;
|
|
margin: 0rem;
|
|
}
|
|
|
|
h2 {
|
|
color: $gray-200;
|
|
margin-bottom: 1rem;
|
|
}
|
|
|
|
.no-messages-box {
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
width: 330px;
|
|
}
|
|
|
|
.envelope {
|
|
color: $gray-400 !important;
|
|
|
|
svg {
|
|
width: 86px;
|
|
height: 64px;
|
|
}
|
|
}
|
|
}
|
|
</style>
|
|
|
|
<style lang="scss" scoped>
|
|
@import '@/assets/scss/colors';
|
|
@import '@/assets/scss/tiers';
|
|
@import '@/assets/scss/variables';
|
|
|
|
$pmHeaderHeight: 56px;
|
|
$background: $white;
|
|
|
|
.header-bar {
|
|
height: 56px;
|
|
background-color: $white;
|
|
align-items: center;
|
|
|
|
.left-header {
|
|
padding-left: 1.5rem;
|
|
max-width: 330px;
|
|
align-items: center;
|
|
flex: 1;
|
|
}
|
|
|
|
.mail-icon {
|
|
width: 32px;
|
|
height: 24px;
|
|
object-fit: contain;
|
|
}
|
|
|
|
.mail-icon-label {
|
|
margin-bottom: 0;
|
|
}
|
|
|
|
.placeholder.svg-icon {
|
|
width: 32px;
|
|
}
|
|
|
|
.plus-button {
|
|
padding: 10px 14px;
|
|
|
|
&.new-message-mode {
|
|
color: $gray-200;
|
|
}
|
|
}
|
|
}
|
|
|
|
.full-height {
|
|
height: 100%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: center;
|
|
}
|
|
|
|
.user-link {
|
|
margin-left: 12px;
|
|
}
|
|
|
|
.selected-conversion {
|
|
justify-content: center;
|
|
align-items: center;
|
|
}
|
|
|
|
#private-message {
|
|
background-color: $background;
|
|
position: relative;
|
|
}
|
|
|
|
.disable-background {
|
|
height: 44px;
|
|
background-color: $gray-600;
|
|
padding: 0.75rem 1.5rem;
|
|
|
|
border-bottom: 1px solid $gray-500;
|
|
}
|
|
|
|
.disable-background-in-message-list {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
height: 44px;
|
|
|
|
color: $yellow-1;
|
|
background: $yellow-500;
|
|
width: 100%;
|
|
|
|
.caption {
|
|
font-weight: 700;
|
|
line-height: 24px;
|
|
}
|
|
|
|
.text {
|
|
font-weight: 400;
|
|
line-height: 24px;
|
|
}
|
|
}
|
|
|
|
h3 {
|
|
margin: 0rem;
|
|
|
|
.svg-icon {
|
|
width: 10px;
|
|
display: inline-block;
|
|
margin-left: .5em;
|
|
}
|
|
}
|
|
|
|
.header-wrap {
|
|
padding: 0.5em;
|
|
|
|
h2 {
|
|
margin: 0;
|
|
line-height: 1;
|
|
}
|
|
}
|
|
|
|
.messagePreview {
|
|
display: block;
|
|
display: -webkit-box;
|
|
-webkit-line-clamp: 2;
|
|
-webkit-box-orient: vertical;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
word-break: break-word;
|
|
}
|
|
|
|
.selected-conversion {
|
|
flex: 1;
|
|
}
|
|
|
|
.messages-column {
|
|
flex: 1;
|
|
padding: 0rem;
|
|
display: flex;
|
|
flex-direction: column;
|
|
|
|
.empty-messages, .message-scroll {
|
|
flex: 1;
|
|
}
|
|
}
|
|
|
|
.message-scroll {
|
|
overflow-x: hidden;
|
|
padding-top: 0.5rem;
|
|
|
|
@media (min-width: 992px) {
|
|
overflow-x: hidden;
|
|
overflow-y: scroll;
|
|
}
|
|
}
|
|
|
|
.full-width {
|
|
width: 100%;
|
|
}
|
|
|
|
.new-message-row {
|
|
width: 100%;
|
|
padding-left: 1.5rem;
|
|
padding-top: 1.5rem;
|
|
padding-right: 1.5rem;
|
|
|
|
textarea {
|
|
background: $white;
|
|
display: inline-block;
|
|
vertical-align: bottom;
|
|
border-radius: 2px;
|
|
z-index: 5;
|
|
|
|
&.disabled {
|
|
pointer-events: none;
|
|
opacity: 0.64;
|
|
background-color: $gray-500;
|
|
}
|
|
|
|
&.has-content {
|
|
--textarea-auto-height: 80px;
|
|
}
|
|
|
|
height: var(--textarea-auto-height, 40px);
|
|
min-height: var(--textarea-auto-height, 40px);
|
|
max-height: 300px;
|
|
}
|
|
}
|
|
|
|
.sub-new-message-row {
|
|
padding: 1.5rem;
|
|
|
|
.guidelines {
|
|
height: 32px;
|
|
font-size: 12px;
|
|
font-weight: normal;
|
|
font-style: normal;
|
|
font-stretch: normal;
|
|
line-height: 1.33;
|
|
letter-spacing: normal;
|
|
color: $gray-200;
|
|
margin-top: 0.25rem;
|
|
margin-bottom: 0.25rem;
|
|
}
|
|
|
|
button {
|
|
height: 32px;
|
|
border-radius: 4px;
|
|
margin-left: 1.5rem;
|
|
white-space: nowrap;
|
|
|
|
&.disabled {
|
|
cursor: default;
|
|
pointer-events: none;
|
|
opacity: 0.64;
|
|
background-color: $gray-500;
|
|
color: $gray-100;
|
|
}
|
|
}
|
|
}
|
|
|
|
.sidebar {
|
|
width: 330px;
|
|
background-color: $gray-700;
|
|
padding: 0;
|
|
border-bottom-left-radius: 8px;
|
|
|
|
@media only screen and (max-width: 768px) {
|
|
width: 280px;
|
|
}
|
|
}
|
|
|
|
.time {
|
|
font-size: 12px;
|
|
color: $gray-200;
|
|
margin-bottom: 0.5rem;
|
|
}
|
|
|
|
.to-form input {
|
|
width: 60%;
|
|
display: inline-block;
|
|
margin-left: 1em;
|
|
}
|
|
|
|
.empty-sidebar {
|
|
display: flex;
|
|
align-items: center;
|
|
}
|
|
|
|
.floating-message-input {
|
|
background: $background;
|
|
position: fixed;
|
|
bottom: 0;
|
|
}
|
|
|
|
.floating-header-shadow {
|
|
position: absolute;
|
|
top: 0;
|
|
width: 100%;
|
|
height: 56px;
|
|
right: 0;
|
|
z-index: 1;
|
|
pointer-events: none;
|
|
|
|
box-shadow: 0 3px 12px 0 rgba(26, 24, 29, 0.24);
|
|
}
|
|
|
|
.center-avatar {
|
|
margin: 0 auto;
|
|
}
|
|
</style>
|
|
|
|
<script>
|
|
import Vue, { defineComponent } from 'vue';
|
|
import moment from 'moment';
|
|
import groupBy from 'lodash/groupBy';
|
|
import orderBy from 'lodash/orderBy';
|
|
import axios from 'axios';
|
|
import { MAX_MESSAGE_LENGTH } from '@/../../common/script/constants';
|
|
import findIndex from 'lodash/findIndex';
|
|
import { mapState } from '@/libs/store';
|
|
import styleHelper from '@/mixins/styleHelper';
|
|
import toggleSwitch from '@/components/ui/toggleSwitch.vue';
|
|
import userLink from '@/components/userLink.vue';
|
|
|
|
import messageList from '@/components/messages/messageList.vue';
|
|
import messageIcon from '@/assets/svg/message.svg?raw';
|
|
import mail from '@/assets/svg/mail.svg?raw';
|
|
import faceAvatar from '@/components/faceAvatar.vue';
|
|
import { EVENTS } from '@/libs/events';
|
|
import PmConversationsList from './pm-conversations-list.vue';
|
|
import PmEmptyState from './pm-empty-state.vue';
|
|
import PmDisabledState from './pm-disabled-state.vue';
|
|
import PmNewMessageStarted from './pm-new-message-started.vue';
|
|
import StartNewConversationInputHeader from './start-new-conversation-input-header.vue';
|
|
import positiveIcon from '@/assets/svg/positive.svg?raw';
|
|
import NotificationMixins from '@/mixins/notifications';
|
|
|
|
// extract to a shared path
|
|
const CONVERSATIONS_PER_PAGE = 10;
|
|
const PM_PER_PAGE = 50;
|
|
|
|
const UI_STATES = Object.freeze({
|
|
LOADING: 'LOADING',
|
|
NO_CONVERSATIONS: 'NO_CONVERSATIONS',
|
|
NO_CONVERSATIONS_SELECTED: 'NO_CONVERSATIONS_SELECTED',
|
|
START_NEW_CONVERSATION: 'START_NEW_CONVERSATION',
|
|
CONVERSATION_SELECTED: 'CONVERSATION_SELECTED',
|
|
});
|
|
|
|
export default defineComponent({
|
|
components: {
|
|
StartNewConversationInputHeader,
|
|
PmNewMessageStarted,
|
|
PmDisabledState,
|
|
PmEmptyState,
|
|
PmConversationsList,
|
|
messageList,
|
|
toggleSwitch,
|
|
userLink,
|
|
faceAvatar,
|
|
},
|
|
filters: {
|
|
timeAgo (value) {
|
|
return moment(new Date(value)).fromNow();
|
|
},
|
|
},
|
|
mixins: [styleHelper, NotificationMixins],
|
|
beforeRouteEnter (to, from, next) {
|
|
next(vm => {
|
|
const data = vm.$store.state.privateMessageOptions;
|
|
|
|
if ((!data || (data && !data.userIdToMessage)) && vm.$route.query && vm.$route.query.uuid) {
|
|
vm.$store.dispatch('user:userLookup', { uuid: vm.$route.query.uuid }).then(res => {
|
|
if (res && res.data && res.data.data) {
|
|
vm.$store.dispatch('user:newPrivateMessageTo', {
|
|
member: res.data.data,
|
|
});
|
|
}
|
|
});
|
|
} else {
|
|
vm.hasPrivateMessageOptionsOnPageLoad = true;
|
|
}
|
|
});
|
|
},
|
|
data () {
|
|
return {
|
|
icons: Object.freeze({
|
|
messageIcon,
|
|
mail,
|
|
positive: positiveIcon,
|
|
}),
|
|
UI_STATES,
|
|
showStartNewConversationInput: false,
|
|
newConversationTargetUser: null,
|
|
loadingConversations: true,
|
|
showPopover: false,
|
|
|
|
/* Conversation-specific data */
|
|
/**
|
|
* @type {PrivateMessages.InitiatedConversation}
|
|
*/
|
|
initiatedConversation: null,
|
|
updateConversationsCounter: 0,
|
|
selectedConversation: {},
|
|
conversationPage: 0,
|
|
canLoadMoreConversations: false,
|
|
/** @type {PrivateMessages.ConversationSummaryMessageEntry[]} */
|
|
loadedConversations: [],
|
|
/** @type {Record<string, PrivateMessages.PrivateMessageEntry[]>} */
|
|
messagesByConversation: {}, // cache {uuid: []}
|
|
|
|
newMessage: '',
|
|
messages: [],
|
|
messagesLoading: false,
|
|
MAX_MESSAGE_LENGTH: MAX_MESSAGE_LENGTH.toString(),
|
|
};
|
|
},
|
|
computed: {
|
|
...mapState({ user: 'user.data' }),
|
|
canLoadMore () {
|
|
return this.selectedConversation && this.selectedConversation.canLoadMore;
|
|
},
|
|
conversations () {
|
|
const inboxGroup = groupBy(this.loadedConversations, 'uuid');
|
|
|
|
// Add placeholder for new conversations
|
|
if (this.initiatedConversation && this.initiatedConversation.uuid) {
|
|
inboxGroup[this.initiatedConversation.uuid] = [{
|
|
uuid: this.initiatedConversation.uuid,
|
|
user: this.initiatedConversation.user,
|
|
username: this.initiatedConversation.username,
|
|
contributor: this.initiatedConversation.contributor,
|
|
backer: this.initiatedConversation.backer,
|
|
userStyles: this.initiatedConversation.userStyles,
|
|
id: '',
|
|
text: '',
|
|
timestamp: new Date(),
|
|
canReceive: true,
|
|
}];
|
|
}
|
|
// Create conversation objects
|
|
|
|
/** @type {PrivateMessages.ConversationEntry[]} */
|
|
const convos = [];
|
|
|
|
for (const key in inboxGroup) {
|
|
if (Object.prototype.hasOwnProperty.call(inboxGroup, key)) {
|
|
/**
|
|
* @type {PrivateMessages.ConversationSummaryMessageEntry}
|
|
*/
|
|
const recentMessage = inboxGroup[key][0];
|
|
|
|
const convoModel = {
|
|
key: recentMessage.uuid,
|
|
name: recentMessage.user,
|
|
// Handles case where from user sent the only message
|
|
// or the to user sent the only message
|
|
username: recentMessage.username,
|
|
date: recentMessage.timestamp,
|
|
lastMessageText: recentMessage.text,
|
|
contributor: recentMessage.contributor,
|
|
userStyles: recentMessage.userStyles,
|
|
backer: recentMessage.backer,
|
|
canReceive: recentMessage.canReceive,
|
|
canLoadMore: false,
|
|
page: 0,
|
|
};
|
|
|
|
convos.push(convoModel);
|
|
}
|
|
}
|
|
|
|
return convos;
|
|
},
|
|
// Separate from selectedConversation which is not computed
|
|
// so messages don't update automatically
|
|
/* eslint-disable vue/no-side-effects-in-computed-properties */
|
|
selectedConversationMessages () {
|
|
// Vue-subscribe to changes
|
|
const subscribeToUpdate = this.messagesLoading || this.updateConversationsCounter > -1;
|
|
|
|
const selectedConversationKey = this.selectedConversation.key;
|
|
const selectedConversation = this.messagesByConversation[selectedConversationKey];
|
|
this.messages = selectedConversation || [];
|
|
|
|
const ordered = orderBy(this.messages, [m => m.timestamp], ['asc']);
|
|
|
|
if (subscribeToUpdate) {
|
|
return ordered;
|
|
}
|
|
|
|
return [];
|
|
},
|
|
filtersConversations () {
|
|
// Vue-subscribe to changes
|
|
const subscribeToUpdate = this.updateConversationsCounter > -1;
|
|
|
|
const filtered = subscribeToUpdate && this.conversations;
|
|
|
|
const ordered = orderBy(filtered, [o => o.date], ['desc']);
|
|
|
|
return ordered;
|
|
},
|
|
placeholderTexts () {
|
|
if (this.user.flags.chatRevoked) {
|
|
return {
|
|
title: this.$t('PMPlaceholderTitleRevoked'),
|
|
description: this.$t('chatPrivilegesRevoked'),
|
|
};
|
|
}
|
|
return {
|
|
title: this.$t('PMPlaceholderTitle'),
|
|
description: this.$t('PMPlaceholderDescription'),
|
|
};
|
|
},
|
|
|
|
disabledTexts () {
|
|
if (this.user.flags.chatRevoked) {
|
|
return {
|
|
enableInput: false,
|
|
showBottomInfo: true,
|
|
title: this.$t('PMPlaceholderTitleRevoked'),
|
|
description: this.$t('chatPrivilegesRevoked'),
|
|
};
|
|
}
|
|
|
|
if (this.selectedConversation?.key) {
|
|
if (this.user.inbox.blocks.includes(this.selectedConversation.key)) {
|
|
return {
|
|
enableInput: false,
|
|
showBottomInfo: true,
|
|
title: this.$t('PMDisabledCaptionTitle'),
|
|
description: this.$t('PMUnblockUserToSendMessages'),
|
|
};
|
|
}
|
|
|
|
if (!this.selectedConversation.canReceive) {
|
|
return {
|
|
enableInput: false,
|
|
showBottomInfo: true,
|
|
title: this.$t('PMCanNotReply'),
|
|
description: this.$t('PMUserDoesNotReceiveMessages'),
|
|
};
|
|
}
|
|
}
|
|
|
|
if (this.user.inbox.optOut) {
|
|
return {
|
|
enableInput: true,
|
|
showBottomInfo: false,
|
|
title: this.$t('PMDisabledCaptionTitle'),
|
|
description: this.$t('PMDisabledCaptionText'),
|
|
};
|
|
}
|
|
|
|
return null;
|
|
},
|
|
optTextSet () {
|
|
if (!this.user.inbox.optOut) {
|
|
return {
|
|
switchDescription: this.$t('PMDisabled'),
|
|
popoverText: this.$t('PMEnabledOptPopoverText'),
|
|
};
|
|
}
|
|
return {
|
|
switchDescription: this.$t('PMDisabled'),
|
|
popoverText: this.$t('PMDisabledOptPopoverText'),
|
|
};
|
|
},
|
|
selectedConversationFaceAvatarClass () {
|
|
if (this.selectedConversation?.contributor) {
|
|
return `tier${this.selectedConversation.contributor.level}`;
|
|
}
|
|
return '';
|
|
},
|
|
newMessageDisabled () {
|
|
if (this.disabledTexts) {
|
|
return !this.disabledTexts.enableInput;
|
|
}
|
|
|
|
return [
|
|
UI_STATES.NO_CONVERSATIONS_SELECTED,
|
|
UI_STATES.NO_CONVERSATIONS,
|
|
UI_STATES.LOADING,
|
|
].includes(this.uiState);
|
|
},
|
|
uiState () {
|
|
if (this.loadingConversations) {
|
|
return UI_STATES.LOADING;
|
|
}
|
|
|
|
if (this.loadedConversations.length === 0) {
|
|
return UI_STATES.NO_CONVERSATIONS;
|
|
}
|
|
|
|
// Hiding the "Select a conversation on the left" state,
|
|
// and just picking the first conversation once it loads, right away
|
|
// see reload method
|
|
/* if (!this.selectedConversation.key) {
|
|
return UI_STATES.NO_CONVERSATIONS_SELECTED;
|
|
} */
|
|
|
|
if (this.selectedConversationMessages.length === 0) {
|
|
return UI_STATES.START_NEW_CONVERSATION;
|
|
}
|
|
|
|
return UI_STATES.CONVERSATION_SELECTED;
|
|
},
|
|
shouldShowInputPanel () {
|
|
const currentUiState = this.uiState;
|
|
|
|
switch (currentUiState) {
|
|
case UI_STATES.CONVERSATION_SELECTED:
|
|
case UI_STATES.START_NEW_CONVERSATION: {
|
|
return true;
|
|
}
|
|
|
|
default: {
|
|
return false;
|
|
}
|
|
}
|
|
},
|
|
},
|
|
async mounted () {
|
|
this.$store.dispatch('common:setTitle', {
|
|
section: this.$t('messages'),
|
|
});
|
|
// notification click to refresh
|
|
this.$root.$on(EVENTS.PM_REFRESH, async () => {
|
|
await this.reload();
|
|
});
|
|
|
|
// header sync button
|
|
this.$root.$on(EVENTS.RESYNC_COMPLETED, async () => {
|
|
await this.reload();
|
|
});
|
|
|
|
await this.reload();
|
|
|
|
// close modal if the Private Messages page is opened in an existing tab
|
|
this.$root.$emit('bv::hide::modal', 'profile');
|
|
this.$root.$emit('bv::hide::modal', 'members-modal');
|
|
|
|
const data = this.$store.state.privateMessageOptions;
|
|
if (data && data.userIdToMessage) {
|
|
this.initiatedConversation = {
|
|
uuid: data.userIdToMessage,
|
|
user: data.displayName,
|
|
username: data.username,
|
|
backer: data.backer,
|
|
contributor: data.contributor,
|
|
userStyles: data.userStyles,
|
|
canReceive: true,
|
|
};
|
|
|
|
this.$store.state.privateMessageOptions = {};
|
|
|
|
this.selectConversation(this.initiatedConversation.uuid);
|
|
}
|
|
},
|
|
beforeDestroy () {
|
|
this.$root.$off(EVENTS.RESYNC_COMPLETED);
|
|
this.$root.$off(EVENTS.PM_REFRESH);
|
|
},
|
|
|
|
methods: {
|
|
async reload () {
|
|
this.loadingConversations = true;
|
|
this.conversationPage = 0;
|
|
|
|
this.loadedConversations = [];
|
|
this.selectedConversation = {};
|
|
|
|
await this.loadConversations();
|
|
|
|
await this.$store.dispatch('user:markPrivMessagesRead');
|
|
|
|
await this.selectFirstConversation();
|
|
|
|
this.loadingConversations = false;
|
|
},
|
|
async loadConversations () {
|
|
const query = [
|
|
'/api/v4/inbox/conversations',
|
|
`?page=${this.conversationPage}`,
|
|
];
|
|
this.conversationPage += 1;
|
|
|
|
const conversationRes = await axios.get(query.join(''));
|
|
const loadedConversations = conversationRes.data.data;
|
|
this.canLoadMoreConversations = loadedConversations.length === CONVERSATIONS_PER_PAGE;
|
|
this.loadedConversations.push(...loadedConversations);
|
|
},
|
|
messageLiked (message) {
|
|
const messages = this.messagesByConversation[this.selectedConversation.key];
|
|
|
|
const chatIndex = findIndex(messages, chatMessage => chatMessage.id === message.id);
|
|
messages.splice(chatIndex, 1, message);
|
|
},
|
|
messageRemoved (message) {
|
|
const messages = this.messagesByConversation[this.selectedConversation.key];
|
|
|
|
const messageIndex = messages.findIndex(msg => msg.id === message.id);
|
|
if (messageIndex !== -1) messages.splice(messageIndex, 1);
|
|
if (this.selectedConversationMessages.length === 0) {
|
|
this.initiatedConversation = {
|
|
uuid: this.selectedConversation.key,
|
|
user: this.selectedConversation.name,
|
|
username: this.selectedConversation.username,
|
|
backer: this.selectedConversation.backer,
|
|
contributor: this.selectedConversation.contributor,
|
|
};
|
|
}
|
|
},
|
|
toggleOpt () {
|
|
this.$store.dispatch('user:togglePrivateMessagesOpt');
|
|
},
|
|
async selectConversation (key, forceLoadMessage = false) {
|
|
const convoFound = this.conversations.find(conversation => conversation.key === key);
|
|
|
|
this.selectedConversation = convoFound || {};
|
|
|
|
if (!this.messagesByConversation[this.selectedConversation.key] || forceLoadMessage) {
|
|
await this.loadMessages();
|
|
}
|
|
|
|
this.scrollToBottom();
|
|
},
|
|
sendPrivateMessage () {
|
|
if (!this.newMessage) return;
|
|
|
|
const messages = this.messagesByConversation[this.selectedConversation.key];
|
|
|
|
messages.push({
|
|
sent: true,
|
|
text: this.newMessage,
|
|
timestamp: new Date(),
|
|
toUser: this.selectedConversation.name,
|
|
toUserName: this.selectedConversation.username,
|
|
toUserContributor: this.selectedConversation.contributor,
|
|
toUserBacker: this.selectedConversation.backer,
|
|
toUUID: this.selectedConversation.uuid,
|
|
|
|
id: '-1', // will be updated once the result is back
|
|
likes: {},
|
|
ownerId: this.user._id,
|
|
uuid: this.user._id,
|
|
fromUUID: this.user._id,
|
|
user: this.user.profile.name,
|
|
username: this.user.auth.local.username,
|
|
contributor: this.user.contributor,
|
|
backer: this.user.backer,
|
|
canReceive: true,
|
|
});
|
|
|
|
// Remove the placeholder message
|
|
if (this.initiatedConversation
|
|
&& this.initiatedConversation.uuid === this.selectedConversation.key) {
|
|
this.loadedConversations.unshift(this.initiatedConversation);
|
|
this.initiatedConversation = null;
|
|
}
|
|
|
|
this.selectedConversation.lastMessageText = this.newMessage;
|
|
this.selectedConversation.date = new Date();
|
|
|
|
this.$store.dispatch('members:sendPrivateMessage', {
|
|
toUserId: this.selectedConversation.key,
|
|
message: this.newMessage,
|
|
}).then(response => {
|
|
const newMessage = response.data.data.message;
|
|
const messageToReset = messages[messages.length - 1];
|
|
|
|
// just set the id, all other infos already set
|
|
messageToReset.id = newMessage.id;
|
|
messageToReset.text = newMessage.text;
|
|
messageToReset.uniqueMessageId = newMessage.uniqueMessageId;
|
|
|
|
Object.assign(messages[messages.length - 1], messageToReset);
|
|
this.updateConversationsCounter += 1;
|
|
});
|
|
|
|
this.newMessage = '';
|
|
|
|
setTimeout(() => {
|
|
this.scrollToBottom();
|
|
}, 150);
|
|
},
|
|
/**
|
|
* This method does a couple of things:
|
|
* - first round:
|
|
* - tries to scroll down
|
|
* - in the next tick it triggers it again
|
|
* (during testing it seemed that the first trigger still had some space left to scroll)
|
|
* - 2nd round:
|
|
* - tries to scroll down
|
|
* - in the next tick it checks if the scrollTop is to most it can scroll down,
|
|
* if it is, it stops from doing that again
|
|
* if not, it goes into the next round
|
|
* - if we reach round 6 it stops completely,
|
|
* no need to have a endless loop of just scrolling down
|
|
*/
|
|
scrollToBottom (callCount = 0) {
|
|
if (callCount > 5) {
|
|
return;
|
|
}
|
|
|
|
if (!this.$refs.chatscroll) {
|
|
// if the message list component not loaded yet, but scrollToBottom was called
|
|
// just try again at a later time
|
|
setTimeout(() => {
|
|
this.scrollToBottom(callCount + 1);
|
|
}, 125);
|
|
return;
|
|
}
|
|
|
|
const chatscrollEl = this.$refs.chatscroll.$el;
|
|
// chatscrollBeforeTick.scrollTop = chatscrollBeforeTick.scrollHeight;
|
|
chatscrollEl.scrollTo(0, chatscrollEl.scrollHeight);
|
|
|
|
Vue.nextTick(() => {
|
|
if (!this.$refs.chatscroll) {
|
|
return;
|
|
}
|
|
|
|
let shouldRetrigger = true;
|
|
|
|
if (callCount > 1) {
|
|
const maxPossibleScrollPos = chatscrollEl.scrollHeight - chatscrollEl.clientHeight;
|
|
|
|
if (chatscrollEl.scrollTop === maxPossibleScrollPos) {
|
|
shouldRetrigger = false;
|
|
}
|
|
}
|
|
|
|
if (shouldRetrigger) {
|
|
setTimeout(() => {
|
|
this.scrollToBottom(callCount + 1);
|
|
}, 125);
|
|
}
|
|
});
|
|
},
|
|
infiniteScrollTrigger () {
|
|
// show loading and wait until the loadMore debounced
|
|
// or else it would trigger on every scrolling-pixel (while not loading)
|
|
if (this.canLoadMore) {
|
|
this.messagesLoading = true;
|
|
}
|
|
|
|
return this.loadMore();
|
|
},
|
|
loadMore () {
|
|
this.selectedConversation.page += 1;
|
|
return this.loadMessages();
|
|
},
|
|
async loadMessages () {
|
|
this.messagesLoading = true;
|
|
|
|
// use local vars if the loading takes longer
|
|
// and the user switches the conversation while loading
|
|
const conversationKey = this.selectedConversation.key;
|
|
|
|
const requestUrl = `/api/v4/inbox/paged-messages?conversation=${conversationKey}&page=${this.selectedConversation.page}`;
|
|
const res = await axios.get(requestUrl);
|
|
const loadedMessages = res.data.data;
|
|
|
|
/* eslint-disable max-len */
|
|
this.messagesByConversation[conversationKey] = this.messagesByConversation[conversationKey] || [];
|
|
/* eslint-disable max-len */
|
|
const loadedMessagesToAdd = loadedMessages
|
|
.filter(m => this.messagesByConversation[conversationKey].findIndex(mI => mI.id === m.id) === -1);
|
|
this.messagesByConversation[conversationKey].push(...loadedMessagesToAdd);
|
|
|
|
// only show the load more Button if the max count was returned
|
|
this.selectedConversation.canLoadMore = loadedMessages.length === PM_PER_PAGE;
|
|
this.messagesLoading = false;
|
|
},
|
|
async selectFirstConversation () {
|
|
if (this.loadedConversations.length > 0) {
|
|
await this.selectConversation(this.loadedConversations[0].uuid, true);
|
|
}
|
|
},
|
|
triggerStartNewConversationState () {
|
|
this.showStartNewConversationInput = true;
|
|
},
|
|
async startConversationByUsername (targetUserName) {
|
|
// check if the target user exists in current conversations, select that conversation
|
|
/** @type {PrivateMessages.ConversationSummaryMessageEntry} */
|
|
const foundConversation = this.loadedConversations.find(c => c.username === targetUserName);
|
|
|
|
if (foundConversation) {
|
|
this.selectConversation(foundConversation.uuid);
|
|
this.showStartNewConversationInput = false;
|
|
|
|
return;
|
|
}
|
|
|
|
let loadedMember = null;
|
|
|
|
try {
|
|
loadedMember = await this.$store.dispatch('members:fetchMemberByUsername', {
|
|
username: targetUserName,
|
|
});
|
|
} catch {
|
|
loadedMember = null;
|
|
}
|
|
|
|
if (!loadedMember) {
|
|
this.error(this.$t('targetUserNotExist', { userName: targetUserName }));
|
|
return;
|
|
}
|
|
|
|
const loadedMemberUUID = loadedMember.id;
|
|
|
|
this.showStartNewConversationInput = false;
|
|
|
|
// otherwise create a dummy conversation, load messages for that user
|
|
/**
|
|
* @type {PrivateMessages.ConversationSummaryMessageEntry}
|
|
*/
|
|
const newConversationItem = {
|
|
uuid: loadedMemberUUID,
|
|
user: loadedMember.profile.name,
|
|
username: loadedMember.auth.local.username,
|
|
contributor: loadedMember.contributor,
|
|
userStyles: loadedMember,
|
|
canReceive: loadedMember.inbox.canReceive,
|
|
timestamp: new Date(),
|
|
count: 0,
|
|
text: '',
|
|
};
|
|
|
|
this.loadedConversations.splice(0, 0, newConversationItem);
|
|
|
|
this.selectConversation(loadedMemberUUID);
|
|
|
|
if (this.messagesByConversation[loadedMemberUUID]) {
|
|
const messageLengthByConversation = this.messagesByConversation[loadedMemberUUID].length;
|
|
|
|
// if messages already exists, update the sidebar entry last message
|
|
if (messageLengthByConversation > 0) {
|
|
/** @type {PrivateMessages.PrivateMessageEntry} */
|
|
const lastMessage = this.messagesByConversation[loadedMemberUUID][messageLengthByConversation - 1];
|
|
|
|
newConversationItem.lastMessageText = lastMessage.text;
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
this.newConversationTargetUser = loadedMember;
|
|
},
|
|
},
|
|
});
|
|
</script>
|