mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-18 23:27:26 +01:00
* wip * refactor world state * allow resource to be reloaded when the server is updated * fix #9242 * fix event listeners * remove un-needed code * add tests for asyncResourceFactory reloadOnAppVersionChange * fix double cron notifications and party members showing up in the header after a party invitation is accepted * remove console.log * do not send vm info to loggly due to circular dependency + fix typo * fix #12181 * do not load invites multiple times in members modal * add hover to challenge member count * groups: load members only on demand * minor ui fixes * choose class: fix vue duplicate key warning * minor ui fixes * challanges: load members on demand * add loading spinner * change loading mechanism * fix loading gryphon issues * reduce code duplication
264 lines
6.6 KiB
Vue
264 lines
6.6 KiB
Vue
<template>
|
|
<div>
|
|
<invite-modal
|
|
:group="inviteModalGroup"
|
|
:group-type="inviteModalGroupType"
|
|
/>
|
|
<create-party-modal />
|
|
<div
|
|
id="app-header"
|
|
class="row"
|
|
:class="{'hide-header': hideHeader}"
|
|
>
|
|
<members-modal :hide-badge="true" />
|
|
<member-details
|
|
:member="user"
|
|
:class-badge-position="'next-to-name'"
|
|
:is-header="true"
|
|
:disable-name-styling="true"
|
|
/>
|
|
<div
|
|
v-if="hasParty"
|
|
class="view-party d-flex align-items-center"
|
|
>
|
|
<button
|
|
class="btn btn-primary view-party-button"
|
|
@click="showPartyMembers()"
|
|
>
|
|
{{ $t('viewParty') }}
|
|
</button>
|
|
</div>
|
|
<div
|
|
v-if="hasParty"
|
|
ref="partyMembersDiv"
|
|
v-resize="1500"
|
|
class="party-members d-flex"
|
|
@resized="setPartyMembersWidth($event)"
|
|
>
|
|
<!-- eslint-disable vue/no-use-v-if-with-v-for -->
|
|
<member-details
|
|
v-for="(member, $index) in sortedPartyMembers"
|
|
v-if="member._id !== user._id && $index < membersToShow"
|
|
:key="member._id"
|
|
:member="member"
|
|
condensed="condensed"
|
|
:expanded="member._id === expandedMember"
|
|
:is-header="true"
|
|
:class-badge-position="'hidden'"
|
|
@onHover="expandMember(member._id)"
|
|
/>
|
|
<!-- eslint-enable vue/no-use-v-if-with-v-for -->
|
|
</div>
|
|
<div
|
|
v-else
|
|
class="no-party d-flex justify-content-center text-center"
|
|
>
|
|
<div class="align-self-center">
|
|
<h3>{{ $t('battleWithFriends') }}</h3>
|
|
<span
|
|
class="small-text"
|
|
v-html="$t('inviteFriendsParty')"
|
|
></span>
|
|
<br>
|
|
<button
|
|
class="btn btn-primary"
|
|
@click="createOrInviteParty()"
|
|
>
|
|
{{ user.party._id ? $t('inviteFriends') : $t('startAParty') }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss" scoped>
|
|
@import '~@/assets/scss/colors.scss';
|
|
|
|
#app-header {
|
|
padding-left: 24px;
|
|
padding-top: 9px;
|
|
padding-bottom: 8px;
|
|
background: $purple-50;
|
|
color: $header-color;
|
|
flex-wrap: nowrap;
|
|
position: relative;
|
|
max-height: 10.25rem;
|
|
}
|
|
|
|
.hide-header {
|
|
display: none;
|
|
}
|
|
|
|
.no-party, .party-members {
|
|
flex-grow: 1;
|
|
}
|
|
|
|
.view-party {
|
|
position: absolute;
|
|
z-index: 10;
|
|
right: 0;
|
|
padding-right: 40px;
|
|
padding-left: 10px;
|
|
height: calc(100% - 9px);
|
|
background-image: linear-gradient(to right, rgba($purple-50, 0), $purple-50);
|
|
}
|
|
|
|
.no-party {
|
|
.small-text {
|
|
color: $header-color;
|
|
flex-wrap: nowrap;
|
|
}
|
|
|
|
h3 {
|
|
color: $white;
|
|
margin-bottom: 4px;
|
|
}
|
|
|
|
.btn {
|
|
margin-top: 16px;
|
|
}
|
|
}
|
|
|
|
@media only screen and (max-width: 768px) {
|
|
.view-party-button {
|
|
display: none;
|
|
}
|
|
}
|
|
</style>
|
|
|
|
<script>
|
|
import orderBy from 'lodash/orderBy';
|
|
import { mapGetters, mapActions } from '@/libs/store';
|
|
import MemberDetails from '../memberDetails';
|
|
import createPartyModal from '../groups/createPartyModal';
|
|
import inviteModal from '../groups/inviteModal';
|
|
import membersModal from '../groups/membersModal';
|
|
import ResizeDirective from '@/directives/resize.directive';
|
|
|
|
export default {
|
|
directives: {
|
|
resize: ResizeDirective,
|
|
},
|
|
components: {
|
|
MemberDetails,
|
|
createPartyModal,
|
|
inviteModal,
|
|
membersModal,
|
|
},
|
|
data () {
|
|
return {
|
|
expandedMember: null,
|
|
currentWidth: 0,
|
|
inviteModalGroup: undefined,
|
|
inviteModalGroupType: undefined,
|
|
};
|
|
},
|
|
watch: {
|
|
hideHeader () {
|
|
this.$nextTick(() => {
|
|
if (this.$refs.partyMembersDiv) {
|
|
this.setPartyMembersWidth({ width: this.$refs.partyMembersDiv.clientWidth });
|
|
}
|
|
});
|
|
},
|
|
},
|
|
computed: {
|
|
...mapGetters({
|
|
user: 'user:data',
|
|
partyMembers: 'party:members',
|
|
}),
|
|
showHeader () {
|
|
if (this.$store.state.hideHeader) return false;
|
|
return true;
|
|
},
|
|
hasParty () {
|
|
return this.user.party && this.user.party._id
|
|
&& this.partyMembers && this.partyMembers.length > 1;
|
|
},
|
|
membersToShow () {
|
|
return Math.floor(this.currentWidth / 140) + 1;
|
|
},
|
|
sortedPartyMembers () {
|
|
let sortedMembers = this.partyMembers.slice(); // shallow clone to avoid infinite loop
|
|
const { order, orderAscending } = this.user.party;
|
|
|
|
if (order === 'profile.name') {
|
|
// If members are to be sorted by name, use localeCompare for case-
|
|
// insensitive sort
|
|
sortedMembers.sort(
|
|
(a, b) => {
|
|
if (orderAscending === 'desc') {
|
|
return b.profile.name.localeCompare(a.profile.name);
|
|
}
|
|
|
|
return a.profile.name.localeCompare(b.profile.name);
|
|
},
|
|
);
|
|
} else {
|
|
sortedMembers = orderBy(
|
|
sortedMembers,
|
|
[order],
|
|
[orderAscending],
|
|
);
|
|
}
|
|
|
|
return sortedMembers;
|
|
},
|
|
hideHeader () {
|
|
return ['groupPlan', 'privateMessages'].includes(this.$route.name);
|
|
},
|
|
},
|
|
created () {
|
|
if (this.user.party && this.user.party._id) {
|
|
this.$store.state.memberModalOptions.groupId = this.user.party._id;
|
|
this.getPartyMembers();
|
|
}
|
|
},
|
|
mounted () {
|
|
this.$root.$on('inviteModal::inviteToGroup', group => {
|
|
this.inviteModalGroup = group;
|
|
this.inviteModalGroupType = group.type === 'guild' ? 'Guild' : 'Party';
|
|
this.$root.$emit('bv::show::modal', 'invite-modal');
|
|
});
|
|
},
|
|
beforeDestroy () {
|
|
this.$root.$off('inviteModal::inviteToGroup');
|
|
},
|
|
methods: {
|
|
...mapActions({
|
|
getPartyMembers: 'party:getMembers',
|
|
}),
|
|
expandMember (memberId) {
|
|
if (this.expandedMember === memberId) {
|
|
this.expandedMember = null;
|
|
} else {
|
|
this.expandedMember = memberId;
|
|
}
|
|
},
|
|
createOrInviteParty () {
|
|
if (this.user.party._id) {
|
|
this.$root.$emit('inviteModal::inviteToGroup', this.user.party);
|
|
} else {
|
|
this.$root.$emit('bv::show::modal', 'create-party-modal');
|
|
}
|
|
},
|
|
async showPartyMembers () {
|
|
const party = await this.$store.dispatch('party:getParty');
|
|
|
|
this.$root.$emit('habitica:show-member-modal', {
|
|
groupId: party.data._id,
|
|
viewingMembers: this.partyMembers,
|
|
group: party.data,
|
|
fetchMoreMembers: p => this.$store.dispatch('members:getGroupMembers', p),
|
|
});
|
|
},
|
|
setPartyMembersWidth ($event) {
|
|
if (this.currentWidth !== $event.width) {
|
|
this.currentWidth = $event.width;
|
|
}
|
|
},
|
|
},
|
|
};
|
|
</script>
|