mirror of
https://github.com/Ocelot-Social-Community/Ocelot-Social.git
synced 2025-12-13 07:46:06 +00:00
495 lines
14 KiB
Vue
495 lines
14 KiB
Vue
<template>
|
|
<div>
|
|
<ds-card v-if="user && user.image">
|
|
<p>PROFILE IMAGE</p>
|
|
</ds-card>
|
|
<ds-space />
|
|
<ds-flex
|
|
v-if="user"
|
|
:width="{ base: '100%' }"
|
|
gutter="base"
|
|
>
|
|
<ds-flex-item :width="{ base: '100%', sm: 2, md: 2, lg: 1 }">
|
|
<ds-card
|
|
:class="{'disabled-content': user.disabled}"
|
|
style="position: relative; height: auto;"
|
|
>
|
|
<ds-avatar
|
|
:image="user.avatar"
|
|
:name="user.name || 'Anonymus'"
|
|
class="profile-avatar"
|
|
size="120px"
|
|
/>
|
|
<no-ssr>
|
|
<content-menu
|
|
placement="bottom-end"
|
|
resource-type="user"
|
|
:resource="user"
|
|
:is-owner="myProfile"
|
|
/>
|
|
</no-ssr>
|
|
<ds-space margin="small">
|
|
<ds-heading
|
|
tag="h3"
|
|
align="center"
|
|
no-margin
|
|
>
|
|
{{ user.name }}
|
|
</ds-heading>
|
|
<ds-text
|
|
v-if="user.location"
|
|
align="center"
|
|
color="soft"
|
|
size="small"
|
|
>
|
|
<ds-icon name="map-marker" />
|
|
{{ user.location.name }}
|
|
</ds-text>
|
|
<ds-text
|
|
align="center"
|
|
color="soft"
|
|
size="small"
|
|
>
|
|
{{ $t('profile.memberSince') }} {{ user.createdAt | date('MMMM yyyy') }}
|
|
</ds-text>
|
|
</ds-space>
|
|
<ds-space
|
|
v-if="user.badges && user.badges.length"
|
|
margin="x-small"
|
|
>
|
|
<hc-badges :badges="user.badges" />
|
|
</ds-space>
|
|
<ds-flex>
|
|
<ds-flex-item>
|
|
<no-ssr>
|
|
<ds-number :label="$t('profile.followers')">
|
|
<hc-count-to
|
|
slot="count"
|
|
:end-val="followedByCount"
|
|
/>
|
|
</ds-number>
|
|
</no-ssr>
|
|
</ds-flex-item>
|
|
<ds-flex-item>
|
|
<no-ssr>
|
|
<ds-number :label="$t('profile.following')">
|
|
<hc-count-to
|
|
slot="count"
|
|
:end-val="Number(user.followingCount) || 0"
|
|
/>
|
|
</ds-number>
|
|
</no-ssr>
|
|
</ds-flex-item>
|
|
</ds-flex>
|
|
<ds-space margin="small">
|
|
<hc-follow-button
|
|
v-if="!myProfile"
|
|
:follow-id="user.id"
|
|
:is-followed="user.followedByCurrentUser"
|
|
@optimistic="follow => user.followedByCurrentUser = follow"
|
|
@update="follow => fetchUser()"
|
|
/>
|
|
</ds-space>
|
|
<template v-if="user.about">
|
|
<hr>
|
|
<ds-space
|
|
margin-top="small"
|
|
margin-bottom="small"
|
|
>
|
|
<ds-text
|
|
color="soft"
|
|
size="small"
|
|
>
|
|
{{ user.about }}
|
|
</ds-text>
|
|
</ds-space>
|
|
</template>
|
|
</ds-card>
|
|
<ds-space />
|
|
<ds-heading
|
|
tag="h3"
|
|
soft
|
|
style="text-align: center; margin-bottom: 10px;"
|
|
>
|
|
Netzwerk
|
|
</ds-heading>
|
|
<ds-card style="position: relative; height: auto;">
|
|
<ds-space
|
|
v-if="user.following && user.following.length"
|
|
margin="x-small"
|
|
>
|
|
<ds-text
|
|
tag="h5"
|
|
color="soft"
|
|
>
|
|
Wem folgt {{ user.name | truncate(15) }}?
|
|
</ds-text>
|
|
</ds-space>
|
|
<template v-if="user.following && user.following.length">
|
|
<ds-space
|
|
v-for="follow in uniq(user.following)"
|
|
:key="follow.id"
|
|
margin="x-small"
|
|
>
|
|
<!-- TODO: find better solution for rendering errors -->
|
|
<no-ssr>
|
|
<user
|
|
:user="follow"
|
|
:trunc="15"
|
|
/>
|
|
</no-ssr>
|
|
</ds-space>
|
|
<ds-space
|
|
v-if="user.followingCount - user.following.length"
|
|
margin="small"
|
|
>
|
|
<ds-text
|
|
size="small"
|
|
color="softer"
|
|
>
|
|
und {{ user.followingCount - user.following.length }} weitere
|
|
</ds-text>
|
|
</ds-space>
|
|
</template>
|
|
<template v-else>
|
|
<p style="text-align: center; opacity: .5;">
|
|
{{ user.name }} folgt niemandem
|
|
</p>
|
|
</template>
|
|
</ds-card>
|
|
<ds-space />
|
|
<ds-card style="position: relative; height: auto;">
|
|
<ds-space
|
|
v-if="user.followedBy && user.followedBy.length"
|
|
margin="x-small"
|
|
>
|
|
<ds-text
|
|
tag="h5"
|
|
color="soft"
|
|
>
|
|
Wer folgt {{ user.name | truncate(15) }}?
|
|
</ds-text>
|
|
</ds-space>
|
|
<template v-if="user.followedBy && user.followedBy.length">
|
|
<ds-space
|
|
v-for="follow in uniq(user.followedBy)"
|
|
:key="follow.id"
|
|
margin="x-small"
|
|
>
|
|
<!-- TODO: find better solution for rendering errors -->
|
|
<no-ssr>
|
|
<user
|
|
:user="follow"
|
|
:trunc="15"
|
|
/>
|
|
</no-ssr>
|
|
</ds-space>
|
|
<ds-space
|
|
v-if="user.followedByCount - user.followedBy.length"
|
|
margin="small"
|
|
>
|
|
<ds-text
|
|
size="small"
|
|
color="softer"
|
|
>
|
|
und {{ user.followedByCount - user.followedBy.length }} weitere
|
|
</ds-text>
|
|
</ds-space>
|
|
</template>
|
|
<template v-else>
|
|
<p style="text-align: center; opacity: .5;">
|
|
niemand folgt {{ user.name }}
|
|
</p>
|
|
</template>
|
|
</ds-card>
|
|
<ds-space />
|
|
<ds-space
|
|
v-if="user.socialMedia && user.socialMedia.length"
|
|
margin="x-small"
|
|
/>
|
|
<ds-card style="position: relative; height: auto;">
|
|
<ds-space
|
|
v-if="user.socialMedia && user.socialMedia.length"
|
|
margin="x-small"
|
|
>
|
|
<ds-text
|
|
tag="h5"
|
|
color="soft"
|
|
>
|
|
{{ $t('profile.socialMedia') }} {{ user.name | truncate(15) }}?
|
|
</ds-text>
|
|
<template v-if="user.socialMedia && user.socialMedia.length">
|
|
<ds-space
|
|
v-for="link in socialMediaLinks"
|
|
:key="link.username"
|
|
margin="x-small"
|
|
>
|
|
<!-- TODO: find better solution for rendering errors -->
|
|
<a :href="link.url">
|
|
<ds-avatar
|
|
:image="link.favicon"
|
|
/>
|
|
{{ link.username }}
|
|
</a>
|
|
</ds-space>
|
|
</template>
|
|
</ds-space>
|
|
</ds-card>
|
|
</ds-flex-item>
|
|
<ds-flex-item :width="{ base: '100%', sm: 3, md: 5, lg: 3 }">
|
|
<ds-flex
|
|
:width="{ base: '100%' }"
|
|
gutter="small"
|
|
>
|
|
<ds-flex-item class="profile-top-navigation">
|
|
<ds-card class="ds-tab-nav">
|
|
<ds-flex>
|
|
<ds-flex-item class="ds-tab-nav-item ds-tab-nav-item-active">
|
|
<ds-space margin="small">
|
|
<!-- TODO: find better solution for rendering errors -->
|
|
<no-ssr>
|
|
<ds-number :label="$t('common.post', null, user.contributionsCount)">
|
|
<hc-count-to
|
|
slot="count"
|
|
:end-val="user.contributionsCount"
|
|
/>
|
|
</ds-number>
|
|
</no-ssr>
|
|
</ds-space>
|
|
</ds-flex-item>
|
|
<!--<ds-flex-item class="ds-tab-nav-item">
|
|
<ds-space margin="small">-->
|
|
<!-- TODO: find better solution for rendering errors -->
|
|
<!--
|
|
<no-ssr>
|
|
<ds-number :label="$t('profile.commented')">
|
|
<hc-count-to slot="count" :end-val="user.commentsCount"/>
|
|
</ds-number>
|
|
</no-ssr>
|
|
</ds-space>
|
|
</ds-flex-item>
|
|
-->
|
|
<!--<ds-flex-item class="ds-tab-nav-item">
|
|
<ds-space margin="small">-->
|
|
<!-- TODO: find better solution for rendering errors -->
|
|
<!--<no-ssr>
|
|
<ds-number :label="$t('profile.shouted')">
|
|
<hc-count-to slot="count" :end-val="user.shoutedCount"/>
|
|
</ds-number>
|
|
</no-ssr>
|
|
</ds-space>
|
|
</ds-flex-item>-->
|
|
</ds-flex>
|
|
</ds-card>
|
|
</ds-flex-item>
|
|
<ds-flex-item style="text-align: center">
|
|
<ds-button
|
|
v-if="myProfile"
|
|
v-tooltip="{content: 'Create a new Post', placement: 'left', delay: { show: 500 }}"
|
|
:path="{ name: 'post-create' }"
|
|
class="profile-post-add-button"
|
|
icon="plus"
|
|
size="large"
|
|
primary
|
|
/>
|
|
</ds-flex-item>
|
|
<template v-if="activePosts.length">
|
|
<ds-flex-item
|
|
v-for="post in activePosts"
|
|
:key="post.id"
|
|
:width="{ base: '100%', md: '100%', xl: '50%' }"
|
|
>
|
|
<hc-post-card :post="post" />
|
|
</ds-flex-item>
|
|
</template>
|
|
<template v-else>
|
|
<ds-flex-item :width="{ base: '100%' }">
|
|
<hc-empty
|
|
margin="xx-large"
|
|
icon="file"
|
|
/>
|
|
</ds-flex-item>
|
|
</template>
|
|
</ds-flex>
|
|
<hc-load-more
|
|
v-if="hasMore"
|
|
:loading="$apollo.loading"
|
|
@click="showMoreContributions"
|
|
/>
|
|
</ds-flex-item>
|
|
</ds-flex>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import uniqBy from 'lodash/uniqBy'
|
|
|
|
import User from '~/components/User.vue'
|
|
import HcPostCard from '~/components/PostCard.vue'
|
|
import HcFollowButton from '~/components/FollowButton.vue'
|
|
import HcCountTo from '~/components/CountTo.vue'
|
|
import HcBadges from '~/components/Badges.vue'
|
|
import HcLoadMore from '~/components/LoadMore.vue'
|
|
import HcEmpty from '~/components/Empty.vue'
|
|
import ContentMenu from '~/components/ContentMenu'
|
|
|
|
export default {
|
|
components: {
|
|
User,
|
|
HcPostCard,
|
|
HcFollowButton,
|
|
HcCountTo,
|
|
HcBadges,
|
|
HcLoadMore,
|
|
HcEmpty,
|
|
ContentMenu
|
|
},
|
|
transition: {
|
|
name: 'slide-up',
|
|
mode: 'out-in'
|
|
},
|
|
data() {
|
|
return {
|
|
User: [],
|
|
voted: false,
|
|
page: 1,
|
|
pageSize: 6
|
|
}
|
|
},
|
|
computed: {
|
|
myProfile() {
|
|
return this.$route.params.slug === this.$store.getters['auth/user'].slug
|
|
},
|
|
followedByCount() {
|
|
let count = Number(this.user.followedByCount) || 0
|
|
return count
|
|
},
|
|
user() {
|
|
return this.User ? this.User[0] : {}
|
|
},
|
|
offset() {
|
|
return (this.page - 1) * this.pageSize
|
|
},
|
|
hasMore() {
|
|
return (
|
|
this.user.contributions &&
|
|
this.user.contributions.length < this.user.contributionsCount
|
|
)
|
|
},
|
|
activePosts() {
|
|
if (!this.user.contributions) {
|
|
return []
|
|
}
|
|
return this.uniq(this.user.contributions.filter(post => !post.deleted))
|
|
},
|
|
socialMediaLinks() {
|
|
const { socialMedia = [] } = this.user
|
|
return socialMedia.map(socialMedia => {
|
|
const { url } = socialMedia
|
|
const matches = url.match(
|
|
/^(?:https?:\/\/)?(?:[^@\n])?(?:www\.)?([^:\/\n?]+)/g
|
|
)
|
|
const [domain] = matches || []
|
|
const favicon = domain ? `${domain}/favicon.ico` : null
|
|
const username = url.split('/')[3]
|
|
return { url, username, favicon }
|
|
})
|
|
}
|
|
},
|
|
watch: {
|
|
User(val) {
|
|
if (!val || !val.length) {
|
|
throw new Error('User not found!')
|
|
}
|
|
}
|
|
},
|
|
methods: {
|
|
uniq(items, field = 'id') {
|
|
return uniqBy(items, field)
|
|
},
|
|
fetchUser() {
|
|
// TODO: we should use subscriptions instead of fetching the whole user again
|
|
this.$apollo.queries.User.refetch()
|
|
},
|
|
showMoreContributions() {
|
|
// this.page++
|
|
// Fetch more data and transform the original result
|
|
this.page++
|
|
this.$apollo.queries.User.fetchMore({
|
|
variables: {
|
|
slug: this.$route.params.slug,
|
|
first: this.pageSize,
|
|
offset: this.offset
|
|
},
|
|
// Transform the previous result with new data
|
|
updateQuery: (previousResult, { fetchMoreResult }) => {
|
|
let output = { User: this.User }
|
|
output.User[0].contributions = [
|
|
...previousResult.User[0].contributions,
|
|
...fetchMoreResult.User[0].contributions
|
|
]
|
|
return output
|
|
},
|
|
fetchPolicy: 'cache-and-network'
|
|
})
|
|
}
|
|
},
|
|
apollo: {
|
|
User: {
|
|
query() {
|
|
return require('~/graphql/UserProfileQuery.js').default(this)
|
|
},
|
|
variables() {
|
|
return {
|
|
slug: this.$route.params.slug,
|
|
first: this.pageSize,
|
|
offset: 0
|
|
}
|
|
},
|
|
fetchPolicy: 'cache-and-network'
|
|
}
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style lang="scss">
|
|
.profile-avatar {
|
|
display: block;
|
|
margin: auto;
|
|
margin-top: -60px;
|
|
border: #fff 5px solid;
|
|
}
|
|
|
|
.page-name-profile-id-slug {
|
|
.ds-flex-item:first-child .content-menu {
|
|
position: absolute;
|
|
top: $space-x-small;
|
|
right: $space-x-small;
|
|
}
|
|
}
|
|
|
|
.profile-top-navigation {
|
|
position: sticky;
|
|
top: 53px;
|
|
z-index: 2;
|
|
}
|
|
|
|
.ds-tab-nav {
|
|
.ds-card-content {
|
|
padding: 0 !important;
|
|
|
|
.ds-tab-nav-item {
|
|
&.ds-tab-nav-item-active {
|
|
border-bottom: 3px solid #17b53f;
|
|
&:first-child {
|
|
border-bottom-left-radius: $border-radius-x-large;
|
|
}
|
|
&:last-child {
|
|
border-bottom-right-radius: $border-radius-x-large;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style>
|