| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369 |
- <template>
- <div class="user-home-container">
- <div class="user-header-wrapper">
- <el-card v-if="user" class="user-info-card" :body-style="{ padding: '20px' }">
- <div class="info-content-top">
- <div class="avatar-section">
- <el-avatar :size="90" :src="user.avatarUrl" class="user-avatar-main" />
- <el-tag
- v-if="user.vip"
- size="mini"
- effect="dark"
- type="warning"
- class="vip-tag-overlay"
- >
- VIP
- </el-tag>
- </div>
- <div class="action-section">
- <el-button
- :type="followButton.text === '已关注' ? 'info' : 'danger'"
- round
- size="medium"
- :icon="followButton.icon"
- @click="followUser(user.userId)"
- >
- {{ followButton.text }}
- </el-button>
- <el-button icon="el-icon-message" round size="medium" @click="sendMessage(user.userId)">发消息</el-button>
- <el-dropdown trigger="click" class="more-btn">
- <el-button icon="el-icon-more" circle plain size="medium"></el-button>
- <el-dropdown-menu slot="dropdown">
- <el-dropdown-item v-if="user.biliUserId">
- <a :href="'https://space.bilibili.com/' + user.biliUserId" target="_blank" class="no-link-style">访问B站空间</a>
- </el-dropdown-item>
- <el-dropdown-item divider>举报用户</el-dropdown-item>
- </el-dropdown-menu>
- </el-dropdown>
- </div>
- </div>
- <div class="info-text-bottom">
- <h2 class="user-name">
- {{ user.screenName }}
- <el-tag v-if="user.vip" size="mini" type="warning" effect="plain" class="name-vip-tag">会员</el-tag>
- <el-tag v-if="user.biliUserId" size="mini" effect="plain" class="bili-tag">B站认证</el-tag>
- </h2>
- <p class="user-signature">{{ user.signature || '这个人很心大,居然没有写签名~' }}</p>
- <div class="user-stats">
- <router-link :to="`/user/${user.userId}/following`" class="stat-item">
- <span class="stat-value">{{ user.following }}</span>
- <span class="stat-label">关注</span>
- </router-link>
- <div class="stat-divider"></div>
- <router-link :to="`/user/${user.userId}/follower`" class="stat-item">
- <span class="stat-value">{{ user.follower }}</span>
- <span class="stat-label">粉丝</span>
- </router-link>
- </div>
- </div>
- </el-card>
- </div>
- <div class="user-content-section">
- <el-tabs v-if="userContentData" v-model="activeName" class="custom-tabs" @tab-click="tabClick">
- <el-tab-pane name="video">
- <span slot="label">视频 <span class="tab-count">{{ userContentData.videoCount }}</span></span>
- <el-row :gutter="15" class="grid-container" v-loading="loading">
- <el-col v-for="(video, index) in dataList" :key="index" :xs="12" :sm="8" :md="6">
- <video-card :video="video" class="content-card-hover" />
- </el-col>
- </el-row>
- </el-tab-pane>
- <el-tab-pane name="image">
- <span slot="label">图片 <span class="tab-count">{{ userContentData.imageCount }}</span></span>
- <el-row :gutter="15" class="grid-container" v-loading="loading">
- <el-col v-for="(album, index) in dataList" :key="index" :xs="12" :sm="8" :md="6">
- <image-album-card :image-album="album" class="content-card-hover" />
- </el-col>
- </el-row>
- </el-tab-pane>
- <el-tab-pane name="album">
- <span slot="label">收藏夹 <span class="tab-count">{{ userContentData.albumCount }}</span></span>
- <el-row :gutter="15" class="grid-container" v-loading="loading">
- <el-col v-for="(item, index) in dataList" :key="index" :xs="12" :sm="8" :md="6">
- <el-card :body-style="{ padding: '0px' }" class="playlist-card content-card-hover">
- <router-link :to="`/playlist/${item.albumId}`" class="no-link-style">
- <div class="playlist-cover-wrapper">
- <el-image lazy fit="cover" :src="item.coverUrl" class="playlist-img" />
- <div class="playlist-mask">
- <i class="el-icon-collection"></i> <span>{{ item.total }}</span>
- </div>
- </div>
- <div class="playlist-info">
- <span class="playlist-title">{{ item.albumName }}</span>
- </div>
- </router-link>
- </el-card>
- </el-col>
- </el-row>
- </el-tab-pane>
- </el-tabs>
- <el-empty v-if="showEmpty && !loading" :image-size="200" description="该分类下空空如也" />
- <div class="pagination-wrapper" v-if="totalSize > pageSize">
- <el-pagination
- background
- layout="prev, pager, next"
- :current-page="currentPage"
- :page-size="pageSize"
- :total="totalSize"
- @current-change="handleCurrentChange"
- />
- </div>
- </div>
- </div>
- </template>
- <script>
- import VideoCard from '@/components/card/VideoCard'
- import ImageAlbumCard from '@/components/card/ImageAlbumCard'
- import { getUserInfo, checkRelation, followUser, unfollowUser } from '@/api/user'
- import { getUserContentData, getUserVideos } from '@/api/video'
- import { getAlbumImage1 } from '@/api/image'
- import { getUserPlaylist } from '@/api/collect'
- export default {
- name: 'UserHome',
- components: { VideoCard, ImageAlbumCard },
- data() {
- return {
- user: null,
- userId: null,
- followButton: { icon: 'el-icon-plus', text: '关注' },
- activeName: 'video',
- currentPage: 1,
- pageSize: 12,
- totalSize: 0,
- dataList: [],
- userContentData: null,
- loading: false,
- showEmpty: false
- }
- },
- watch: {
- '$route.query': {
- handler() {
- this.loadDataFromRoute();
- },
- deep: true
- }
- },
- created() {
- this.userId = this.$route.params.id;
- this.initUserContext();
- },
- methods: {
- initUserContext() {
- Promise.all([
- getUserInfo(this.userId),
- checkRelation(this.userId),
- getUserContentData(this.userId)
- ]).then(([info, relation, content]) => {
- if (info.code === 0) {
- this.user = info.data;
- this.loadDataFromRoute();
- }
- if (relation.code === 0 && relation.data) {
- this.followButton = { icon: 'el-icon-check', text: '已关注' };
- }
- if (content.code === 0) {
- this.userContentData = content.data;
- }
- });
- },
- loadDataFromRoute() {
- const query = this.$route.query;
- this.activeName = query.tab || 'video';
- this.currentPage = query.page ? parseInt(query.page) : 1;
- this.updatePageTitle();
- this.fetchContentList();
- },
- syncStateToURL() {
- this.$router.push({
- path: this.$route.path,
- query: { tab: this.activeName, page: this.currentPage }
- }).catch(err => {
- if (err.name !== 'NavigationDuplicated') throw err;
- });
- },
- fetchContentList() {
- this.loading = true;
- this.dataList = [];
- this.showEmpty = false;
- let apiCall;
- if (this.activeName === 'video') {
- apiCall = getUserVideos(this.userId, this.currentPage);
- } else if (this.activeName === 'image') {
- apiCall = getAlbumImage1(this.currentPage, this.userId);
- } else if (this.activeName === 'album') {
- apiCall = getUserPlaylist({ userId: this.userId, pn: this.currentPage });
- }
- apiCall.then(resp => {
- if (resp.code === 0) {
- this.dataList = resp.data.list;
- this.totalSize = resp.data.totalSize;
- this.showEmpty = this.dataList.length === 0;
- }
- }).finally(() => { this.loading = false; });
- },
- tabClick(tab) {
- this.activeName = tab.name;
- this.currentPage = 1;
- this.syncStateToURL();
- },
- handleCurrentChange(page) {
- this.currentPage = page;
- this.syncStateToURL();
- window.scrollTo({ top: 0, behavior: 'smooth' });
- },
- updatePageTitle() {
- if (!this.user) return;
- const suffix = { video: '的视频', image: '的图片', album: '的收藏夹' };
- document.title = `${this.user.screenName}${suffix[this.activeName] || '的主页'}`;
- },
- followUser(userId) {
- const isFollowing = this.followButton.text === '已关注';
- const api = isFollowing ? unfollowUser : followUser;
- api(userId).then(resp => {
- if (resp.code === 0) {
- this.followButton = isFollowing ? { icon: 'el-icon-plus', text: '关注' } : { icon: 'el-icon-check', text: '已关注' };
- this.$message.success(isFollowing ? '已取消关注' : '关注成功');
- }
- });
- },
- sendMessage() { this.$message.info('私信功能暂未开放'); }
- }
- }
- </script>
- <style scoped>
- .user-home-container {
- background-color: #f4f5f7;
- min-height: 100vh;
- padding: 20px 0 50px 0;
- }
- .user-header-wrapper {
- max-width: 1100px;
- margin: 0 auto 20px;
- padding: 0 15px;
- }
- .user-info-card {
- border-radius: 12px;
- border: none;
- box-shadow: 0 4px 12px rgba(0,0,0,0.05);
- }
- .info-content-top {
- display: flex;
- justify-content: space-between;
- align-items: center;
- margin-bottom: 20px;
- }
- .avatar-section {
- position: relative;
- display: flex;
- }
- .user-avatar-main {
- border: 2px solid #fff;
- background: #fff;
- }
- /* 会员 Tag 覆盖在头像上的样式 */
- .vip-tag-overlay {
- position: absolute;
- bottom: 0;
- right: -10px;
- border-radius: 4px;
- border: 1px solid #fff;
- font-weight: bold;
- transform: scale(0.9);
- }
- .name-vip-tag {
- margin-left: 5px;
- font-weight: normal;
- vertical-align: middle;
- }
- .action-section {
- display: flex;
- gap: 10px;
- }
- .user-name {
- margin: 0 0 8px 0;
- font-size: 24px;
- display: flex;
- align-items: center;
- gap: 8px;
- }
- .user-signature {
- color: #9499a0;
- font-size: 14px;
- margin-bottom: 20px;
- }
- .user-stats {
- display: flex;
- align-items: center;
- gap: 30px;
- }
- .stat-item {
- text-decoration: none;
- display: flex;
- gap: 5px;
- font-size: 14px;
- }
- .stat-value { font-weight: bold; color: #18191c; }
- .stat-label { color: #9499a0; }
- .stat-divider { width: 1px; height: 14px; background: #e3e5e7; }
- .user-content-section {
- max-width: 1100px;
- margin: 0 auto;
- padding: 0 15px;
- }
- .custom-tabs /deep/ .el-tabs__header {
- background: #fff;
- padding: 0 20px;
- border-radius: 8px;
- margin-bottom: 20px;
- }
- .tab-count { font-size: 12px; color: #9499a0; margin-left: 4px; }
- .content-card-hover { transition: all 0.3s ease; margin-bottom: 20px; }
- .content-card-hover:hover { transform: translateY(-5px); }
- .no-link-style { text-decoration: none; color: inherit; }
- .playlist-card { border-radius: 8px; overflow: hidden; }
- .playlist-cover-wrapper { position: relative; aspect-ratio: 16/9; }
- .playlist-img { width: 100%; height: 100%; }
- .playlist-mask { position: absolute; bottom: 0; right: 0; background: rgba(0,0,0,0.6); color: #fff; padding: 2px 8px; font-size: 12px; border-top-left-radius: 8px; }
- .playlist-info { padding: 10px; text-align: left; }
- .playlist-title { font-size: 14px; color: #18191c; display: block; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
- .pagination-wrapper { margin-top: 30px; display: flex; justify-content: center; }
- @media screen and (max-width: 768px) {
- .user-home-container { padding-top: 10px; }
- .user-avatar-main { width: 70px !important; height: 70px !important; }
- .user-name { font-size: 20px; }
- .vip-tag-overlay { right: -5px; bottom: -2px; }
- .action-section .el-button { padding: 8px 12px; font-size: 12px; }
- .more-btn { display: none; }
- }
- </style>
|