frontend-oj/src/views/course/CourseDetail.vue

808 lines
24 KiB
Vue
Raw Normal View History

2025-03-10 23:38:23 +08:00
<template>
<div class="course-detail">
<div class="course-header">
<div class="container">
<div class="course-info">
<h1 class="course-title">{{ course.title }}</h1>
<p class="course-desc">{{ course.description }}</p>
<div class="course-meta">
<div class="meta-item">
<el-icon><User /></el-icon>
<span>讲师: {{ course.teacher }}</span>
</div>
<div class="meta-item">
<el-icon><Calendar /></el-icon>
<span>更新时间: {{ course.updateTime }}</span>
</div>
<div class="meta-item">
<el-icon><View /></el-icon>
<span>{{ course.studentCount }}人学习</span>
</div>
<div class="meta-item">
<el-rate v-model="course.rating" disabled text-color="#ff9900" />
<span>({{ course.ratingCount }})</span>
</div>
</div>
</div>
</div>
</div>
<div class="container">
<div class="course-content">
<div class="course-main">
<el-tabs v-model="activeTab" class="course-tabs">
<el-tab-pane label="课程章节" name="chapters">
<div class="course-chapters">
<el-collapse v-model="activeChapters">
<el-collapse-item v-for="chapter in course.chapters" :key="chapter.id" :title="chapter.title" :name="chapter.id">
<div class="chapter-sections">
<div
v-for="section in chapter.sections"
:key="section.id"
class="section-item"
:class="{ 'is-free': section.isFree, 'is-active': currentSection && currentSection.id === section.id }"
@click="playSection(chapter, section)"
>
<div class="section-info">
<el-icon v-if="section.type === 'video'"><VideoPlay /></el-icon>
<el-icon v-else-if="section.type === 'document'"><Document /></el-icon>
<span class="section-title">{{ section.title }}</span>
<span v-if="section.isFree" class="free-tag">免费</span>
</div>
<div class="section-meta">
<span>{{ section.duration }}</span>
<el-icon v-if="section.isCompleted"><Check /></el-icon>
</div>
</div>
</div>
</el-collapse-item>
</el-collapse>
</div>
</el-tab-pane>
<el-tab-pane label="课程介绍" name="intro">
<div class="course-intro" v-html="course.introHtml"></div>
</el-tab-pane>
<el-tab-pane label="课程评价" name="reviews">
<div class="course-reviews">
<div class="review-summary">
<div class="rating-overall">
<div class="rating-score">{{ course.rating.toFixed(1) }}</div>
<div class="rating-stars">
<el-rate v-model="course.rating" disabled text-color="#ff9900" />
<div class="rating-count">{{ course.ratingCount }}人评价</div>
</div>
</div>
<div class="rating-distribution">
<div v-for="(count, index) in course.ratingDistribution" :key="index" class="rating-bar">
<span class="rating-level">{{ 5 - index }}</span>
<el-progress
:percentage="Math.round(count / course.ratingCount * 100)"
:stroke-width="12"
:show-text="false"
:color="'#ffc107'"
/>
<span class="rating-percent">{{ Math.round(count / course.ratingCount * 100) }}%</span>
</div>
</div>
</div>
<div class="review-list">
<div v-for="review in course.reviews" :key="review.id" class="review-item">
<div class="review-header">
<div class="reviewer-info">
<el-avatar :size="40" :src="review.avatar"></el-avatar>
<div class="reviewer-meta">
<div class="reviewer-name">{{ review.username }}</div>
<div class="review-date">{{ review.date }}</div>
</div>
</div>
<div class="review-rating">
<el-rate v-model="review.rating" disabled text-color="#ff9900" />
</div>
</div>
<div class="review-content">{{ review.content }}</div>
<div v-if="review.reply" class="review-reply">
<div class="reply-header">讲师回复:</div>
<div class="reply-content">{{ review.reply }}</div>
</div>
</div>
</div>
</div>
</el-tab-pane>
</el-tabs>
</div>
<div class="course-sidebar">
<el-card class="sidebar-card">
<div class="video-preview">
<img :src="course.coverImg" :alt="course.title">
<div class="play-button" @click="playPreview">
<el-icon><VideoPlay /></el-icon>
</div>
</div>
<div class="course-price-info">
<div v-if="course.price > 0" class="price">
<span class="current-price">¥{{ course.price.toFixed(2) }}</span>
<span v-if="course.originalPrice" class="original-price">¥{{ course.originalPrice.toFixed(2) }}</span>
</div>
<div v-else class="price free">免费</div>
</div>
<div class="course-actions">
<el-button v-if="isEnrolled" type="primary" @click="continueLearning">继续学习</el-button>
<el-button v-else type="primary" @click="enrollCourse">立即报名</el-button>
<el-button v-if="isFavorite" @click="removeFavorite">取消收藏</el-button>
<el-button v-else @click="addFavorite">收藏课程</el-button>
</div>
<div class="course-includes">
<h3>课程包含</h3>
<ul>
<li><el-icon><VideoPlay /></el-icon> {{ course.videoCount }}</li>
<li><el-icon><Document /></el-icon> {{ course.documentCount }}</li>
<li><el-icon><Timer /></el-icon> {{ course.totalDuration }}</li>
<li><el-icon><Trophy /></el-icon> </li>
<li><el-icon><Connection /></el-icon> </li>
</ul>
</div>
</el-card>
</div>
</div>
</div>
<el-dialog v-model="videoDialogVisible" title="视频播放" width="70%" destroy-on-close>
<div class="video-player">
<div class="player-placeholder">
<img :src="course.coverImg" alt="视频封面">
<div class="player-overlay">
<p>{{ currentSection ? currentSection.title : '课程预览' }}</p>
<el-icon><VideoPlay /></el-icon>
</div>
</div>
</div>
</el-dialog>
</div>
</template>
<script setup>
import { ref, computed, onMounted } from 'vue'
import { useRoute, useRouter } from 'vue-router'
import { ElMessage, ElMessageBox } from 'element-plus'
import { User, Calendar, View, VideoPlay, Document, Check, Trophy, Timer, Connection } from '@element-plus/icons-vue'
const route = useRoute()
const router = useRouter()
const courseId = route.params.id
const activeTab = ref('chapters')
const activeChapters = ref([1])
const videoDialogVisible = ref(false)
const currentSection = ref(null)
// 模拟课程数据
const course = ref({
id: 1,
title: 'Vue.js 3 完全指南',
description: '从入门到精通全面掌握Vue3的新特性和开发技巧',
coverImg: 'https://picsum.photos/id/1/600/400',
teacher: '张教授',
updateTime: '2023-05-15',
studentCount: 1250,
rating: 4.8,
ratingCount: 245,
ratingDistribution: [180, 50, 10, 3, 2], // 5星到1星的分布
price: 199,
originalPrice: 299,
videoCount: 45,
documentCount: 12,
totalDuration: '12小时30分钟',
introHtml: `
<h2>课程简介</h2>
<p>Vue.js 3是一个流行的JavaScript框架用于构建用户界面它建立在标准HTMLCSS和JavaScript之上并提供了一个声明式的组件化的编程模型帮助你高效地开发用户界面</p>
<h3>你将学到什么</h3>
<ul>
<li>Vue 3的核心概念和基础知识</li>
<li>组合式API的使用方法和最佳实践</li>
<li>Vue Router和Pinia状态管理</li>
<li>Vue 3的高级特性和性能优化</li>
<li>实际项目开发和部署</li>
</ul>
<h3>适合人群</h3>
<p>本课程适合有一定JavaScript基础想要学习Vue.js 3的开发者如果你已经熟悉Vue 2本课程也会帮助你快速过渡到Vue 3</p>
`,
chapters: [
{
id: 1,
title: '第1章Vue 3基础入门',
sections: [
{ id: 101, title: '课程介绍', type: 'video', duration: '10:30', isFree: true, isCompleted: true },
{ id: 102, title: 'Vue 3新特性概述', type: 'video', duration: '15:45', isFree: true, isCompleted: true },
{ id: 103, title: '开发环境搭建', type: 'video', duration: '12:20', isFree: false, isCompleted: true },
{ id: 104, title: '第一个Vue 3应用', type: 'video', duration: '18:15', isFree: false, isCompleted: false }
]
},
{
id: 2,
title: '第2章组合式API详解',
sections: [
{ id: 201, title: 'setup函数介绍', type: 'video', duration: '14:30', isFree: false, isCompleted: false },
{ id: 202, title: 'ref和reactive', type: 'video', duration: '20:15', isFree: false, isCompleted: false },
{ id: 203, title: '计算属性和侦听器', type: 'video', duration: '16:40', isFree: false, isCompleted: false },
{ id: 204, title: '生命周期钩子', type: 'video', duration: '12:55', isFree: false, isCompleted: false },
{ id: 205, title: '组合式函数', type: 'document', duration: '阅读材料', isFree: false, isCompleted: false }
]
},
{
id: 3,
title: '第3章Vue Router路由管理',
sections: [
{ id: 301, title: 'Vue Router 4基础', type: 'video', duration: '18:20', isFree: false, isCompleted: false },
{ id: 302, title: '动态路由和路由参数', type: 'video', duration: '15:10', isFree: false, isCompleted: false },
{ id: 303, title: '路由导航守卫', type: 'video', duration: '22:35', isFree: false, isCompleted: false },
{ id: 304, title: '路由元信息和过渡效果', type: 'document', duration: '阅读材料', isFree: false, isCompleted: false }
]
}
],
reviews: [
{
id: 1,
username: '李明',
avatar: 'https://cube.elemecdn.com/0/88/03b0d39583f48206768a7534e55bcpng.png',
rating: 5,
date: '2023-05-10',
content: '非常棒的课程讲解清晰内容全面对Vue 3的理解有了质的提升。',
reply: '谢谢支持,希望课程对你有所帮助!'
},
{
id: 2,
username: '王芳',
avatar: 'https://cube.elemecdn.com/0/88/03b0d39583f48206768a7534e55bcpng.png',
rating: 4,
date: '2023-05-05',
content: '内容很充实,但有些地方讲解速度有点快,需要多看几遍才能理解。',
reply: null
},
{
id: 3,
username: '张伟',
avatar: 'https://cube.elemecdn.com/0/88/03b0d39583f48206768a7534e55bcpng.png',
rating: 5,
date: '2023-04-28',
content: '老师讲解非常清晰示例也很实用强烈推荐给想学Vue 3的同学',
reply: '感谢您的评价,我们会继续努力提供更好的课程!'
}
]
})
// 模拟用户状态
const isEnrolled = ref(true)
const isFavorite = ref(false)
// 播放课程章节
const playSection = (chapter, section) => {
if (!isEnrolled.value && !section.isFree) {
ElMessage.warning('请先报名课程')
return
}
currentSection.value = section
videoDialogVisible.value = true
}
// 播放预览视频
const playPreview = () => {
currentSection.value = null
videoDialogVisible.value = true
}
// 继续学习
const continueLearning = () => {
// 查找上次学习的位置或第一个未完成的章节
let targetChapter, targetSection
for (const chapter of course.value.chapters) {
const uncompletedSection = chapter.sections.find(section => !section.isCompleted)
if (uncompletedSection) {
targetChapter = chapter
targetSection = uncompletedSection
break
}
}
if (targetChapter && targetSection) {
activeChapters.value = [targetChapter.id]
playSection(targetChapter, targetSection)
} else {
// 如果全部完成,播放第一章第一节
const firstChapter = course.value.chapters[0]
const firstSection = firstChapter.sections[0]
activeChapters.value = [firstChapter.id]
playSection(firstChapter, firstSection)
}
}
// 报名课程
const enrollCourse = () => {
ElMessageBox.confirm('确定要报名该课程吗?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'info'
}).then(() => {
// 在实际应用中这里应该调用API报名课程
isEnrolled.value = true
ElMessage.success('报名成功')
}).catch(() => {
// 用户取消操作
})
}
// 收藏课程
const addFavorite = () => {
// 在实际应用中这里应该调用API收藏课程
isFavorite.value = true
ElMessage.success('已收藏课程')
}
// 取消收藏
const removeFavorite = () => {
ElMessageBox.confirm('确定要取消收藏该课程吗?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
// 在实际应用中这里应该调用API取消收藏
isFavorite.value = false
ElMessage.success('已取消收藏')
}).catch(() => {
// 用户取消操作
})
}
onMounted(() => {
// 在实际应用中这里应该从API获取课程详情
// fetchCourseDetail(courseId)
// 检查URL参数如果有autoplay=true则自动开始播放
if (route.query.autoplay === 'true') {
continueLearning()
}
})
</script>
<style lang="scss" scoped>
.course-detail {
.course-header {
background-color: #f5f7fa;
padding: 40px 0;
.container {
max-width: 1200px;
margin: 0 auto;
padding: 0 20px;
}
.course-info {
max-width: 800px;
.course-title {
font-size: 28px;
margin-bottom: 15px;
color: #303133;
}
.course-desc {
font-size: 16px;
color: #606266;
margin-bottom: 20px;
line-height: 1.6;
}
.course-meta {
display: flex;
flex-wrap: wrap;
gap: 20px;
.meta-item {
display: flex;
align-items: center;
color: #909399;
.el-icon {
margin-right: 5px;
}
}
}
}
}
.container {
max-width: 1200px;
margin: 0 auto;
padding: 30px 20px;
}
.course-content {
display: grid;
grid-template-columns: 1fr 350px;
gap: 30px;
@media (max-width: 992px) {
grid-template-columns: 1fr;
}
.course-main {
background-color: white;
border-radius: 8px;
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
overflow: hidden;
.course-tabs {
padding: 20px;
}
.course-chapters {
.chapter-sections {
.section-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 12px 10px;
border-bottom: 1px solid #ebeef5;
cursor: pointer;
&:last-child {
border-bottom: none;
}
&:hover {
background-color: #f5f7fa;
}
&.is-free {
.section-title {
color: #67c23a;
}
}
&.is-active {
background-color: #ecf5ff;
}
.section-info {
display: flex;
align-items: center;
gap: 8px;
.el-icon {
color: #409eff;
}
.free-tag {
background-color: #67c23a;
color: white;
padding: 2px 6px;
border-radius: 4px;
font-size: 12px;
margin-left: 8px;
}
}
.section-meta {
display: flex;
align-items: center;
gap: 10px;
color: #909399;
font-size: 14px;
.el-icon {
color: #67c23a;
}
}
}
}
}
.course-intro {
padding: 0 10px;
h2 {
font-size: 20px;
margin: 20px 0 15px;
color: #303133;
}
h3 {
font-size: 18px;
margin: 18px 0 12px;
color: #303133;
}
p {
margin-bottom: 15px;
line-height: 1.6;
color: #606266;
}
ul {
margin-bottom: 15px;
padding-left: 20px;
li {
margin-bottom: 8px;
line-height: 1.6;
color: #606266;
}
}
}
.course-reviews {
.review-summary {
display: flex;
gap: 30px;
margin-bottom: 30px;
padding: 20px;
background-color: #f5f7fa;
border-radius: 8px;
@media (max-width: 768px) {
flex-direction: column;
}
.rating-overall {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
min-width: 150px;
.rating-score {
font-size: 48px;
font-weight: bold;
color: #ff9900;
}
.rating-stars {
display: flex;
flex-direction: column;
align-items: center;
.rating-count {
margin-top: 5px;
color: #909399;
}
}
}
.rating-distribution {
flex-grow: 1;
.rating-bar {
display: flex;
align-items: center;
margin-bottom: 8px;
.rating-level {
width: 40px;
text-align: right;
margin-right: 10px;
color: #606266;
}
.el-progress {
flex-grow: 1;
margin: 0 10px;
}
.rating-percent {
width: 40px;
color: #909399;
}
}
}
}
.review-list {
.review-item {
border-bottom: 1px solid #ebeef5;
padding: 20px 0;
&:last-child {
border-bottom: none;
}
.review-header {
display: flex;
justify-content: space-between;
margin-bottom: 15px;
.reviewer-info {
display: flex;
align-items: center;
.reviewer-meta {
margin-left: 10px;
.reviewer-name {
font-weight: bold;
margin-bottom: 5px;
}
.review-date {
font-size: 12px;
color: #909399;
}
}
}
}
.review-content {
margin-bottom: 15px;
line-height: 1.6;
color: #606266;
}
.review-reply {
background-color: #f5f7fa;
padding: 15px;
border-radius: 4px;
.reply-header {
font-weight: bold;
margin-bottom: 8px;
color: #409eff;
}
.reply-content {
color: #606266;
line-height: 1.6;
}
}
}
}
}
}
.course-sidebar {
.sidebar-card {
position: sticky;
top: 20px;
.video-preview {
position: relative;
cursor: pointer;
img {
width: 100%;
height: 200px;
object-fit: cover;
border-radius: 4px;
}
.play-button {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
width: 60px;
height: 60px;
background-color: rgba(0, 0, 0, 0.6);
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
.el-icon {
font-size: 30px;
color: white;
}
&:hover {
background-color: rgba(0, 0, 0, 0.8);
}
}
}
.course-price-info {
margin: 20px 0;
text-align: center;
.price {
.current-price {
font-size: 24px;
font-weight: bold;
color: #f56c6c;
}
.original-price {
font-size: 16px;
color: #909399;
text-decoration: line-through;
margin-left: 10px;
}
&.free {
font-size: 24px;
font-weight: bold;
color: #67c23a;
}
}
}
.course-actions {
display: flex;
gap: 10px;
margin-bottom: 20px;
.el-button {
flex: 1;
}
}
.course-includes {
margin-top: 20px;
h3 {
font-size: 16px;
margin-bottom: 15px;
color: #303133;
}
ul {
list-style: none;
padding: 0;
li {
display: flex;
align-items: center;
margin-bottom: 10px;
color: #606266;
.el-icon {
margin-right: 8px;
color: #409eff;
}
}
}
}
}
}
}
.video-player {
.player-placeholder {
position: relative;
width: 100%;
height: 400px;
background-color: #000;
img {
width: 100%;
height: 100%;
object-fit: cover;
opacity: 0.5;
}
.player-overlay {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
text-align: center;
p {
color: white;
font-size: 18px;
margin-bottom: 15px;
}
.el-icon {
font-size: 48px;
color: white;
}
}
}
}
}
</style>