课程列表页

分页显示数据

rest_framework 里面封装了有分页功能的组件,直接可以拿来即用

在courses/views.py 中新建一个分页器类 (类的嵌套)

from rest_framework.pagination import PageNumberPaginationclass StandardPageNumberPagination(PageNumberPagination):page_size_query_param = 'page_size'max_page_size = 1class CourseAPIView(ListAPIView):queryset = Course.objects.filter(status=0).order_by("-orders","-students")# 设置过滤的字段filter_fields = ('course_category',)serializer_class = CourseSerializerfilter_backends = [OrderingFilter]ordering_fields = ('id', 'students', 'price', 'course_category')pagination_class = StandardPageNumberPagination

客户端请求后端发送数据

<template><div class="course"><Header/><div class="main"><!-- 筛选功能 --><div class="top"><ul class="condition condition1"><li class="cate-condition">课程分类:</li><li class="item" :class="query_params.course_category===0?'current':''"@click="query_params.course_category=0">全部</li><li :class="query_params.course_category===catetory.id?'current':''"@click="query_params.course_category=catetory.id" v-for="catetory in catetory_list":data-key="catetory.id" class="item">{{catetory.name}}</li></ul><ul class="condition condition2"><li class="cate-condition">筛&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;选:</li><li class="item" :class="(query_params.ordering==='-id' || query_params.ordering==='id')?'current':''"@click="select_ordering('id')">默认</li><li class="item":class="(query_params.ordering==='-students' || query_params.ordering==='students')?'current':''"@click="select_ordering('students')">人气</li><li class="item":class="query_params.ordering==='price'?'current price':(query_params.ordering==='-price'?'current price2':'')"@click="select_ordering('price')">价格</li><li class="course-length">共21个课程</li></ul></div><!-- 课程列表 ---><div class="list"><ul><li class="course-item" v-for="course in course_list"><router-link :to="{path: '/detail',query:{id:course.id}}" class="course-link"><div class="course-cover"><img :src="course.course_img" alt=""></div><div class="course-info"><div class="course-title"><h3>{{course.name}}</h3><span>{{course.students}}人已加入学习</span></div><p class="teacher"><span class="info">{{course.teacher.name}} {{course.teacher.title}}</span><span class="lesson">共{{course.lessons}}课时/{{course.lessons===course.pub_lessons?'更新完成':('已更新'+course.pub_lessons+"课时")}}</span></p><ul class="lesson-list"><li v-for="lesson,key in course.lesson_list"><p class="lesson-title">0{{key+1}} | {{lesson.name}}</p><span v-if="lesson.free_trail" class="free">免费</span></li></ul><div class="buy-info"><span class="discount">限时免费</span><span class="present-price">¥0.00元</span><span class="original-price">原价:{{course.price}}元</span><button class="buy-now">立即购买</button></div></div></router-link></li></ul></div><div class="pagination"><el-pagination@current-change="handleCurrentChange":current-page="query_params.current_page"backgroundlayout="prev, pager, next":page-size="course_page_size":total="course_count"></el-pagination></div></div><Footer/></div>
</template><script>import Header from "./common/Header"import Footer from "./common/Footer"export default {name: "Course",data() {return {catetory_list: [],course_list: [],course_count: 0,course_page_size: 1,query_params: {course_category: 0,ordering: "-id",current_page: 1,}}},watch: {// 每次点击不同课程时,要重新获取课程列表"query_params.course_category": function () {this.get_course_list();// 当切换分类的时候,重置页码this.query_params.current_page = 1;},"query_params.ordering": function () {// 当切换排序条件的时候,重置页码// this.query_params.current_page = 1;this.get_course_list();},"query_params.current_page": function () {this.get_course_list();}},components: {Header, Footer},created() {// 获取课程分类this.$axios.get(this.$settings.Host + "/courses/cate/").then(response => {this.catetory_list = response.data}).catch(error => {console.log(error.response)});// 获取课程信息this.get_course_list()},methods: {select_ordering(selector) {// 默认排序if (this.query_params.ordering === ('-' + selector)) {this.query_params.ordering = selector;} else {this.query_params.ordering = '-' + selector;}},get_course_list() {let query_params = {ordering: this.query_params.ordering,page: this.query_params.current_page,};if (this.query_params.course_category !== 0) {query_params.course_category = this.query_params.course_category;}this.$axios.get(this.$settings.Host + "/courses/list/", {params: query_params}).then(response => {// 课程列表this.course_list = response.data.results;// 课程总数量this.course_count = response.data.count;}).catch(error => {console.log(error.response)});},handleCurrentChange(page) {// 页码发生改变this.query_params.current_page = page;}}}
</script><style scoped>.main {width: 1100px;height: auto;margin: 0 auto;padding-top: 35px;}.main .top {margin-bottom: 35px;padding: 25px 30px 25px 20px;background: #fff;border-radius: 4px;box-shadow: 0 2px 4px 0 #f0f0f0;}.condition {border-bottom: 1px solid #333;border-bottom-color: rgba(51, 51, 51, .05);padding-bottom: 18px;margin-bottom: 17px;overflow: hidden;}.condition li {float: left;}.condition .cate-condition {color: #888;font-size: 16px;}.condition .item {padding: 6px 16px;line-height: 16px;margin-left: 14px;position: relative;transition: all .3s ease;border: 1px solid transparent; /*  transparent 透明 */cursor: pointer;color: #4a4a4a;}.condition1 .current {color: #ffc210;border: 1px solid #ffc210 !important;border-radius: 30px;}.condition2 .current {color: #ffc210;}.condition .price:before {content: "";width: 0;border: 5px solid transparent;border-top-color: #d8d8d8;position: absolute;right: 0;bottom: 2.5px;}.condition .price2:before {content: "";width: 0;border: 5px solid transparent;position: absolute;right: 0;bottom: 2.5px;border-top-color: #ffc210;}.condition .price2:after {content: "";width: 0;border: 5px solid transparent;position: absolute;right: 0;top: 2.5px;border-bottom-color: #d8d8d8;}.condition .price:after {content: "";width: 0;border: 5px solid transparent;border-bottom-color: #ffc210;position: absolute;right: 0;top: 2.5px;}.condition2 .course-length {float: right;font-size: 14px;color: #9b9b9b;}.course-item {background: #fff;padding: 20px 30px 20px 20px;margin-bottom: 35px;border-radius: 2px;cursor: pointer;box-shadow: 2px 3px 16px rgba(0, 0, 0, .1);transition: all .2s ease;overflow: hidden;cursor: pointer;}.course-link {overflow: hidden;}.course-cover {width: 423px;height: 210px;margin-right: 30px;float: left;}.course-info {width: 597px;float: left;}.course-title {margin-bottom: 8px;overflow: hidden;}.course-title h3 {font-size: 26px;color: #333;float: left;}.course-title span {float: right;font-size: 14px;color: #9b9b9b;margin-top: 12px;text-indent: 1em; /* 缩进 2字符宽度 */background: url("../assets/people.svg") no-repeat 0px 3px;}.teacher {justify-content: space-between;font-size: 14px;color: #9b9b9b;margin-bottom: 14px;padding-bottom: 14px;border-bottom: 1px solid #333;border-bottom-color: rgba(51, 51, 51, .05);}.teacher .lesson {float: right;}.lesson-list {overflow: hidden;}.lesson-list li {width: 49%;margin-bottom: 15px;cursor: pointer;float: left;margin-right: 1%;}.lesson-list li .player {width: 16px;height: 16px;vertical-align: text-bottom;}.lesson-list li .lesson-title {display: inline-block;max-width: 227px;text-overflow: ellipsis; /* 如果字体太多超出元素的宽度,则添加省略符号 */color: #666;overflow: hidden;white-space: nowrap;font-size: 14px;vertical-align: text-bottom; /* 文本的垂直对齐方式: text-botton 文本底部对齐 */text-indent: 1.5em;background: url(../../static/player.svg) no-repeat 0px 3px;}.lesson-list .free {width: 34px;height: 20px;color: #fd7b4d;margin-left: 10px;border: 1px solid #fd7b4d;border-radius: 2px;text-align: center;font-size: 13px;white-space: nowrap;}.lesson-list li:hover .lesson-title {color: #ffc210;background-image: url(../../static/player2.svg);}.lesson-list li:hover .free {border-color: #ffc210;color: #ffc210;}.buy-info .discount {padding: 0px 10px;font-size: 16px;color: #fff;display: inline-block;height: 36px;text-align: center;margin-right: 8px;background: #fa6240;border: 1px solid #fa6240;border-radius: 10px 0 10px 0;line-height: 36px;}.present-price {font-size: 24px;color: #fa6240;}.original-price {text-decoration: line-through;font-size: 14px;color: #9b9b9b;margin-left: 10px;}.buy-now {width: 120px;height: 38px;background: transparent;color: #fa6240;font-size: 16px;border: 1px solid #fd7b4d;border-radius: 3px;transition: all .2s ease-in-out; /* 过渡动画 */float: right;margin-top: 5px;}.buy-now:hover {color: #fff;background: #ffc210;border: 1px solid #ffc210;cursor: pointer;}.pagination {text-align: center;margin: 20px 0px 50px 0px;}
</style>

View Code

课程详情页

CKEditor富文本编辑器

富文本即具备丰富样式格式的文本。在运营后台,运营人员需要录入课程的相关描述,可以是包含了HTML语法格式的字符串。为了快速简单的让用户能够在页面中编辑带html格式的文本,我们引入富文本编辑器。

富文本编辑器:ueditor、ckeditor、kindeditor

1. 安装

pip install django-ckeditor

2. 添加应用

在INSTALLED_APPS中添加

INSTALLED_APPS = [...'ckeditor',  # 富文本编辑器'ckeditor_uploader',  # 富文本编辑器上传图片模块
    ...
]

3. 添加CKEditor设置

在settings/dev.py中添加

# 富文本编辑器ckeditor配置
CKEDITOR_CONFIGS = {'default': {'toolbar': 'full',  # 工具条功能'height': 300,      # 编辑器高度# 'width': 300,     # 编辑器宽
    },
}
CKEDITOR_UPLOAD_PATH = ''  # 上传图片保存路径,留空则调用django的文件上传功能

4. 添加ckeditor路由

在总路由中添加

path(r'^ckeditor/', include('ckeditor_uploader.urls')),

5. 为模型类添加字段

ckeditor提供了两种类型的Django模型类字段

  • ckeditor.fields.RichTextField 不支持上传文件的富文本字段

  • ckeditor_uploader.fields.RichTextUploadingField 支持上传文件的富文本字段\

修改course/models.py里面的字段信息,记得要重新数据迁移

from ckeditor_uploader.fields import RichTextUploadingField
class Course(models.Model):"""专题课程"""...brief = RichTextUploadingField(max_length=2048, verbose_name="课程概述", null=True, blank=True)

课程详情页显示

因为接下来的组件中使用了vue-video视频播放组件,所以我们需要先预安装。

安装依赖  (前端)

npm install vue-video-player --save

在main.js中注册加载组件require('video.js/dist/video-js.css');
require('vue-video-player/src/custom-theme.css');
import VideoPlayer from 'vue-video-player'
Vue.use(VideoPlayer);

Detail.vue组件(模板)代码:

<template><div class="detail"><Header></Header><div class="warp"><div class="course-info"><div class="warp-left" style="width: 690px;height: 388px;background-color: #000;"></div><div class="warp-right"><h3 class="course-title">Python开发21天入门</h3><p class="course-data">37400人在学&nbsp;&nbsp;&nbsp;&nbsp;课程总时长:154课时/30小时&nbsp;&nbsp;&nbsp;&nbsp;难度:初级</p><div class="preferential"><p class="price-service">限时免费</p><p class="timer">距离结束:仅剩 28天 14小时 10分 <span>57</span> 秒</p></div><p class="course-price"><span>活动价</span><span class="real-price">¥0.00</span><span class="old-price">¥9.00</span></p><div class="buy-course"><p class="buy-btn"><span class="btn1">立即购买</span><span class="btn2">免费试学</span></p><p class="add-cart"><img src="../../static/images/cart.svg" alt="">加入购物车</p></div></div></div><div class="course-tab"><ul><li  class="active">详情介绍</li><li>课程章节 <span>(试学)</span></li><li>用户评论 (83)</li><li>常见问题</li></ul></div><div class="course-section"><section class="course-section-left"><img src="../../static/images/21天01_1547098127.6672518.jpeg" alt=""></section></div></div><Footer></Footer></div>
</template><script>
import Header from "./common/Header"
import Footer from "./common/Footer"
export default {name: 'CourseDetail',data(){return {}},components:{Header,Footer,},methods:{},created(){}
};
</script><style scoped>
.detail{margin-top: 80px;
}
.course-info{padding-top: 30px;width:1200px;height: 388px;margin: auto;
}
.warp-left,.warp-right{float: left;
}
.warp-right{height: 388px;position: relative;
}
.course-title{font-size: 20px;color: #333;padding: 10px 23px;letter-spacing: .45px;font-weight: normal;
}
.course-data{padding-left: 23px;padding-right: 23px;padding-bottom: 16px;font-size: 14px;color: #9b9b9b;
}
.preferential{width: 100%;height: auto;background: #fa6240;font-size: 14px;color: #4a4a4a;display: -ms-flexbox;display: flex;-ms-flex-align: center;align-items: center;-ms-flex-pack: justify;justify-content: space-between;padding: 10px 23px;
}
.price-service{font-size: 16px;color: #fff;letter-spacing: .36px;
}
.timer{font-size: 14px;color: #fff;
}
.course-price{width: 100%;background: #fff;height: auto;font-size: 14px;color: #4a4a4a;display: -ms-flexbox;display: flex;-ms-flex-align: end;align-items: flex-end;padding: 5px 23px;
}
.real-price{font-size: 26px;color: #fa6240;margin-left: 10px;display: inline-block;margin-bottom: -5px;
}
.old-price{font-size: 14px;color: #9b9b9b;margin-left: 10px;text-decoration: line-through;
}
.buy-course{position: absolute;left: 0;bottom: 20px;width: 100%;height: auto;-ms-flex-pack: justify;justify-content: space-between;padding-left: 23px;padding-right: 23px;
}
.buy-btn{float: left;
}
.buy-btn .btn1{display: inline-block;width: 125px;height: 40px;background: #ffc210;border-radius: 4px;color: #fff;cursor: pointer;margin-right: 15px;text-align: center;vertical-align: middle;line-height: 40px;
}
.buy-btn .btn2{width: 125px;height: 40px;border-radius: 4px;cursor: pointer;margin-right: 15px;display: inline-block;background: #fff;color: #ffc210;border: 1px solid #ffc210;text-align: center;vertical-align: middle;line-height: 40px;
}
.add-cart{font-size: 14px;color: #ffc210;text-align: center;cursor: pointer;float: right;margin-top: 10px;
}
.add-cart img{width: 20px;height: auto;margin-right: 7px;
}
.course-tab{width: 100%;height: auto;background: #fff;margin-bottom: 30px;box-shadow: 0 2px 4px 0 #f0f0f0;
}
.course-tab>ul{padding: 0;margin: 0 auto;list-style: none;width: 1200px;height: auto;display: -ms-flexbox;display: flex;-ms-flex-align: center;align-items: center;color: #4a4a4a;
}
.course-tab>ul>li{margin-right: 15px;padding: 26px 20px 16px;font-size: 17px;cursor: pointer;
}
.course-tab>ul>.active{color: #ffc210;border-bottom: 2px solid #ffc210;
}
.course-section{background: #FAFAFA;overflow: hidden;padding-bottom: 40px;width: 1200px;height: auto;margin: 0 auto;
}
.course-section-left{width: 880px;height: auto;padding: 20px;background: #fff;float: left;box-sizing: border-box;overflow: hidden;position: relative;box-shadow: 0 2px 4px 0 #f0f0f0;
}
</style>

View Code

注册路由

routers/index.js

import CourseDetail from "../components/Detail",{name:"Detail",path: "/detail",component: Detail,}

完善从课程列表跳转到课程详情的链接

Course.vue,代码:

<p class="box-title"><router-link :to="{path: '/detail',query:{id:course.id}}">{{course.name}}</router-link></p>

后端提供课程详情页数据接口

序列化器/serializers.py代码:

class TeacherDetailModelSerializer(serializers.ModelSerializer):class Meta:model = Teacherfields = ("id","name","title","role","signature","image","brief")class CourseDetailModelSerializer(serializers.ModelSerializer):"""课程详情页的序列化器"""teacher = TeacherDetailModelSerializer()class Meta:model = Coursefields = ("id","name","course_img","students","lessons","pub_lessons","price","teacher","course_level","brief")

视图代码:

from rest_framework.generics import RetrieveAPIView
from .serializers import CourseDetailModelSerializer
class CourseDeitalAPIView(RetrieveAPIView):queryset = Course.objects.filter(is_delete=False, is_show=True).order_by("orders")serializer_class = CourseDetailModelSerializer

路由代码:

from django.urls import path, re_path
from . import views
urlpatterns = [re_path(r"detail/(?P<pk>\d+)",views.CourseDetailAPIView.as_view())
]

前端请求api接口并显示数据

<template><div class="detail"><Header/><div class="main"><div class="course-info"><div class="wrap-left"><video-player class="video-player vjs-custom-skin"ref="videoPlayer":playsinline="true":options="playerOptions"@play="onPlayerPlay($event)"@pause="onPlayerPause($event)"></video-player></div><div class="wrap-right"><h3 class="course-name">{{course.name}}</h3><p class="data">{{course.students}}人在学&nbsp;&nbsp;&nbsp;&nbsp;课程总时长:{{course.lessons}}课时/{{course.lessons==course.pub_lessons?'更新完成':('已更新'+course.pub_lessons+"课程")}}&nbsp;&nbsp;&nbsp;&nbsp;难度:{{course.course_level}}</p><div class="sale-time"><p class="sale-type">限时免费</p><p class="expire">距离结束:仅剩 01天 04小时 33分 <span class="second">08</span> 秒</p></div><p class="course-price"><span>活动价</span><span class="discount">¥0.00</span><span class="original">¥{{course.price}}</span></p><div class="buy"><div class="buy-btn"><button class="buy-now">立即购买</button><button class="free">免费试学</button></div><div class="add-cart"><img src="@/assets/cart-yellow.svg" alt="">加入购物车</div></div></div></div><div class="course-tab"><ul class="tab-list"><li :class="tabIndex==1?'active':''" @click="tabIndex=1">详情介绍</li><li :class="tabIndex==2?'active':''" @click="tabIndex=2">课程章节 <span :class="tabIndex!=2?'free':''">(试学)</span></li><li :class="tabIndex==3?'active':''" @click="tabIndex=3">用户评论 (42)</li><li :class="tabIndex==4?'active':''" @click="tabIndex=4">常见问题</li></ul></div><div class="course-content"><div class="course-tab-list"><div class="tab-item" v-if="tabIndex==1"><div v-html="course.brief"></div></div><div class="tab-item" v-if="tabIndex==2"><div class="tab-item-title"><p class="chapter">课程章节</p><p class="chapter-length">共11章 147个课时</p></div><div class="chapter-item"><p class="chapter-title"><img src="@/assets/1.svg" alt="">第1章·Linux硬件基础</p><ul class="lesson-list"><li class="lesson-item"><p class="name"><span class="index">1-1</span> 课程介绍-学习流程<span class="free">免费</span></p><p class="time">07:30 <img src="@/assets/chapter-player.svg"></p><button class="try">立即试学</button></li><li class="lesson-item"><p class="name"><span class="index">1-2</span> 服务器硬件-详解<span class="free">免费</span></p><p class="time">07:30 <img src="@/assets/chapter-player.svg"></p><button class="try">立即试学</button></li></ul></div><div class="chapter-item"><p class="chapter-title"><img src="@/assets/1.svg" alt="">第2章·Linux发展过程</p><ul class="lesson-list"><li class="lesson-item"><p class="name"><span class="index">2-1</span> 操作系统组成-Linux发展过程</p><p class="time">07:30 <img src="@/assets/chapter-player.svg"></p><button class="try">立即购买</button></li><li class="lesson-item"><p class="name"><span class="index">2-2</span> 自由软件-GNU-GPL核心讲解</p><p class="time">07:30 <img src="@/assets/chapter-player.svg"></p><button class="try">立即购买</button></li></ul></div></div><div class="tab-item" v-if="tabIndex==3">用户评论</div><div class="tab-item" v-if="tabIndex==4">常见问题</div></div><div class="course-side"><div class="teacher-info"><h4 class="side-title"><span>授课老师</span></h4><div class="teacher-content"><div class="cont1"><img :src="course.teacher.image"><div class="name"><p class="teacher-name">{{course.teacher.name}} {{course.teacher.title}}</p><p class="teacher-title">{{course.teacher.signature}}</p></div></div><p class="narrative" >Linux运维技术专家,老男孩Linux金牌讲师,讲课风趣幽默、深入浅出、声音洪亮到爆炸</p></div></div></div></div></div><Footer/></div>
</template><script>
import Header from "./common/Header"
import Footer from "./common/Footer"import {videoPlayer} from 'vue-video-player';export default {name: "Detail",data(){return {tabIndex:1,  // 当前选项卡显示的下标
        course_id:0, // 当前页面对应的课程ID
        course: {teacher: {},},  // 课程详情信息
        playerOptions: {playbackRates: [0.7, 1.0, 1.5, 2.0], // 播放速度
          autoplay: false, //如果true,则自动播放
          muted: false, // 默认情况下将会消除任何音频。
          loop: false, // 循环播放
          preload: 'auto',  // 建议浏览器在<video>加载元素后是否应该开始下载视频数据。auto浏览器选择最佳行为,立即开始加载视频(如果浏览器支持)
          language: 'zh-CN',aspectRatio: '16:9', // 将播放器置于流畅模式,并在计算播放器的动态大小时使用该值。值应该代表一个比例 - 用冒号分隔的两个数字(例如"16:9"或"4:3")
          fluid: true, // 当true时,Video.js player将拥有流体大小。换句话说,它将按比例缩放以适应其容器。
          sources: [{ // 播放资源和资源格式
            type: "video/mp4",src: "http://img.ksbbs.com/asset/Mon_1703/05cacb4e02f9d9e.mp4" //你的视频地址(必填)
          }],poster: "../static/courses/675076.jpeg", //视频封面图
          width: document.documentElement.clientWidth, // 默认视频全屏时的最大宽度
          notSupportedMessage: '此视频暂无法播放,请稍后再试', //允许覆盖Video.js无法播放媒体源时显示的默认信息。
        }}},watch:{course(data){while(data.brief.search(`"/media`) != -1 ){data.brief = data.brief.replace(`"/media`,`"${this.$settings.Host}/media`)}},tabIndex(){if(tabIndex==2){//获取当前课程对应的章节列表和课时列表
          }}},created(){// 获取当前课程IDthis.course_id = this.$route.query.id - 0;// 判断ID基本有效性
      let _this = this;if( isNaN(this.course_id) || this.course_id < 1 ){_this.$alert("无效的课程ID!","错误",{callback(){_this.$router.go(-1);}});}// 发送请求获取后端课程数据this.$axios.get(this.$settings.Host+`/courses/detail/${this.course_id}/`).then(response=>{this.course = response.data;// 修改视频中的封面图片this.playerOptions.poster = this.course.course_img;}).catch(error=>{console.log(error.response)});},methods: {// 视频播放事件
      onPlayerPlay(player) {alert("play");},// 视频暂停播放事件
      onPlayerPause(player){alert("pause");},// 视频插件初始化
      player() {return this.$refs.videoPlayer.player;}},components:{Header,Footer,videoPlayer,}
}
</script>

View Code

后端提供当前课程对应的章节和课时列表信息

courses/serializers.py,序列化器,代码:

from .models import CourseLesson
class CourseLessonModelSerializer(serializers.ModelSerializer):"""课程课时"""class Meta:model = CourseLessonfields = ["id","name","duration","free_trail"]from .models import CourseChapter
class CourseChapterModelSerializer(serializers.ModelSerializer):"""课程章节"""coursesections = CourseLessonModelSerializer(many=True)class Meta:model = CourseChapterfields = ("id","name","coursesections","chapter")

courses/views.py视图,代码:

from rest_framework.generics import ListAPIView
from .serializers import CourseChapterModelSerializer
from .models import CourseChapter
class CourseChapterAPIView(ListAPIView):"""课程章节信息"""queryset = CourseChapter.objects.filter(is_delete=False, is_show=True).order_by("orders")serializer_class = CourseChapterModelSerializerfilter_backends = [DjangoFilterBackend]filter_fields = ['course']

courses/urls.py路由,代码:

path(r"chapters/",views.CourseChapterAPIView.as_view()),

前端请求章节信息展示到页面中\

<template><div class="detail"><Header/><div class="main"><div class="course-info"><div class="wrap-left"><video-player class="video-player vjs-custom-skin"ref="videoPlayer":playsinline="true":options="playerOptions"@play="onPlayerPlay($event)"@pause="onPlayerPause($event)"></video-player></div><div class="wrap-right"><h3 class="course-name">{{course.name}}</h3><p class="data">{{course.students}}人在学&nbsp;&nbsp;&nbsp;&nbsp;课程总时长:{{course.lessons}}课时/{{course.lessons==course.pub_lessons?'更新完成':('已更新'+course.pub_lessons+"课程")}}&nbsp;&nbsp;&nbsp;&nbsp;难度:{{course.course_level}}</p><div class="sale-time"><p class="sale-type">限时免费</p><p class="expire">距离结束:仅剩 01天 04小时 33分 <span class="second">08</span> 秒</p></div><p class="course-price"><span>活动价</span><span class="discount">¥0.00</span><span class="original">¥{{course.price}}</span></p><div class="buy"><div class="buy-btn"><button class="buy-now">立即购买</button><button class="free">免费试学</button></div><div class="add-cart"><img src="@/assets/cart-yellow.svg" alt="">加入购物车</div></div></div></div><div class="course-tab"><ul class="tab-list"><li :class="tabIndex==1?'active':''" @click="tabIndex=1">详情介绍</li><li :class="tabIndex==2?'active':''" @click="tabIndex=2">课程章节 <span :class="tabIndex!=2?'free':''">(试学)</span></li><li :class="tabIndex==3?'active':''" @click="tabIndex=3">用户评论 (42)</li><li :class="tabIndex==4?'active':''" @click="tabIndex=4">常见问题</li></ul></div><div class="course-content"><div class="course-tab-list"><div class="tab-item" v-if="tabIndex==1"><div v-html="course.brief"></div></div><div class="tab-item" v-if="tabIndex==2"><div class="tab-item-title"><p class="chapter">课程章节</p><p class="chapter-length">共{{chapter_list.length}}章 147个课时</p></div><div class="chapter-item" v-for="chapter in chapter_list"><p class="chapter-title"><img src="@/assets/1.svg" alt="">第{{chapter.chapter}}章·{{chapter.name}}</p><ul class="lesson-list"><li class="lesson-item" v-for="lesson in chapter.coursesections"><p class="name"><span class="index">{{chapter.chapter}}-{{lesson.id}}</span> {{lesson.name}}<span class="free" v-if="lesson.free_trail">免费</span></p><p class="time">{{lesson.duration}} <img src="@/assets/chapter-player.svg"></p><button class="try" v-if="lesson.free_trail">立即试学</button><button class="try" v-else>立即购买</button></li></ul></div></div><div class="tab-item" v-if="tabIndex==3">用户评论</div><div class="tab-item" v-if="tabIndex==4">常见问题</div></div><div class="course-side"><div class="teacher-info"><h4 class="side-title"><span>授课老师</span></h4><div class="teacher-content"><div class="cont1"><img :src="course.teacher.image"><div class="name"><p class="teacher-name">{{course.teacher.name}} {{course.teacher.title}}</p><p class="teacher-title">{{course.teacher.signature}}</p></div></div><p class="narrative" >Linux运维技术专家,老男孩Linux金牌讲师,讲课风趣幽默、深入浅出、声音洪亮到爆炸</p></div></div></div></div></div><Footer/></div>
</template><script>
import Header from "./common/Header"
import Footer from "./common/Footer"import {videoPlayer} from 'vue-video-player';export default {name: "Detail",data(){return {tabIndex:1,  // 当前选项卡显示的下标
        course_id:0, // 当前页面对应的课程ID
        course: {teacher: {},},  // 课程详情信息
        chapter_list:{},playerOptions: {playbackRates: [0.7, 1.0, 1.5, 2.0], // 播放速度
          autoplay: false, //如果true,则自动播放
          muted: false, // 默认情况下将会消除任何音频。
          loop: false, // 循环播放
          preload: 'auto',  // 建议浏览器在<video>加载元素后是否应该开始下载视频数据。auto浏览器选择最佳行为,立即开始加载视频(如果浏览器支持)
          language: 'zh-CN',aspectRatio: '16:9', // 将播放器置于流畅模式,并在计算播放器的动态大小时使用该值。值应该代表一个比例 - 用冒号分隔的两个数字(例如"16:9"或"4:3")
          fluid: true, // 当true时,Video.js player将拥有流体大小。换句话说,它将按比例缩放以适应其容器。
          sources: [{ // 播放资源和资源格式
            type: "video/mp4",src: "http://img.ksbbs.com/asset/Mon_1703/05cacb4e02f9d9e.mp4" //你的视频地址(必填)
          }],poster: "../static/courses/675076.jpeg", //视频封面图
          width: document.documentElement.clientWidth, // 默认视频全屏时的最大宽度
          notSupportedMessage: '此视频暂无法播放,请稍后再试', //允许覆盖Video.js无法播放媒体源时显示的默认信息。
        }}},watch:{course(data){while(data.brief.search(`"/media`) != -1 ){data.brief = data.brief.replace(`"/media`,`"${this.$settings.Host}/media`)}},tabIndex(data){if(data==2){//获取当前课程对应的章节列表和课时列表this.$axios.get(`${this.$settings.Host}/courses/chapters/?course=${this.course_id}`).then(response=>{this.chapter_list = response.data;}).catch(error=>{console.log(error.response)})}}},created(){// 获取当前课程IDthis.course_id = this.$route.query.id - 0;// 判断ID基本有效性
      let _this = this;if( isNaN(this.course_id) || this.course_id < 1 ){_this.$alert("无效的课程ID!","错误",{callback(){_this.$router.go(-1);}});}// 发送请求获取后端课程数据this.$axios.get(this.$settings.Host+`/courses/detail/${this.course_id}/`).then(response=>{this.course = response.data;// 修改视频中的封面图片this.playerOptions.poster = this.course.course_img;}).catch(error=>{console.log(error.response)});},methods: {// 视频播放事件
      onPlayerPlay(player) {alert("play");},// 视频暂停播放事件
      onPlayerPause(player){alert("pause");},// 视频插件初始化
      player() {return this.$refs.videoPlayer.player;}},components:{Header,Footer,videoPlayer,}
}
</script>

View Code

Detail的前端代码

<template><div class="detail"><Header/><div class="main"><div class="course-info"><div class="wrap-left"><video-player class="video-player vjs-custom-skin"ref="videoPlayer":playsinline="true":options="playerOptions"@play="onPlayerPlay($event)"@pause="onPlayerPause($event)"></video-player></div><div class="wrap-right"><h3 class="course-name">{{course.name}}</h3><p class="data">{{course.students}}人在学&nbsp;&nbsp;&nbsp;&nbsp;课程总时长:{{course.lessons}}课时/{{course.lessons==course.pub_lessons?'更新完成':('已更新'+course.pub_lessons+"课程")}}&nbsp;&nbsp;&nbsp;&nbsp;难度:{{course.course_level}}</p><div class="sale-time"><p class="sale-type">限时免费</p><p class="expire">距离结束:仅剩 01天 04小时 33分 <span class="second">08</span> 秒</p></div><p class="course-price"><span>活动价</span><span class="discount">¥0.00</span><span class="original">¥{{course.price}}</span></p><div class="buy"><div class="buy-btn"><button class="buy-now">立即购买</button><button class="free">免费试学</button></div><div class="add-cart"><img src="@/assets/cart-yellow.svg" alt="">加入购物车</div></div></div></div><div class="course-tab"><ul class="tab-list"><li :class="tabIndex==1?'active':''" @click="tabIndex=1">详情介绍</li><li :class="tabIndex==2?'active':''" @click="tabIndex=2">课程章节 <span :class="tabIndex!=2?'free':''">(试学)</span></li><li :class="tabIndex==3?'active':''" @click="tabIndex=3">用户评论 (42)</li><li :class="tabIndex==4?'active':''" @click="tabIndex=4">常见问题</li></ul></div><div class="course-content"><div class="course-tab-list"><div class="tab-item" v-if="tabIndex==1"><div v-html="course.brief"></div></div><div class="tab-item" v-if="tabIndex==2"><div class="tab-item-title"><p class="chapter">课程章节</p><p class="chapter-length">共{{chapter_list.length}}章 147个课时</p></div><div class="chapter-item" v-for="chapter in chapter_list"><p class="chapter-title"><img src="@/assets/1.svg" alt="">第{{chapter.chapter}}章·{{chapter.name}}</p><ul class="lesson-list"><li class="lesson-item" v-for="lesson in chapter.coursesections"><p class="name"><span class="index">{{chapter.chapter}}-{{lesson.id}}</span> {{lesson.name}}<span class="free" v-if="lesson.free_trail">免费</span></p><p class="time">{{lesson.duration}} <img src="@/assets/chapter-player.svg"></p><button class="try" v-if="lesson.free_trail"><router-link :to="{path: '/player',query:{'vid':lesson.section_link}}">立即试学</router-link></button><button class="try" v-else>立即购买</button></li></ul></div></div><div class="tab-item" v-if="tabIndex==3">用户评论</div><div class="tab-item" v-if="tabIndex==4">常见问题</div></div><div class="course-side"><div class="teacher-info"><h4 class="side-title"><span>授课老师</span></h4><div class="teacher-content"><div class="cont1"><img :src="course.teacher.image"><div class="name"><p class="teacher-name">{{course.teacher.name}} {{course.teacher.title}}</p><p class="teacher-title">{{course.teacher.signature}}</p></div></div><p class="narrative" >{{course.teacher.brief}}</p></div></div></div></div></div><Footer/></div>
</template><script>
import Header from "./common/Header"
import Footer from "./common/Footer"import {videoPlayer} from 'vue-video-player';export default {name: "Detail",data(){return {tabIndex:1,  // 当前选项卡显示的下标
        course_id:0, // 当前页面对应的课程ID
        course: {teacher: {},},  // 课程详情信息
        chapter_list:{},playerOptions: {playbackRates: [0.7, 1.0, 1.5, 2.0], // 播放速度
          autoplay: false, //如果true,则自动播放
          muted: false, // 默认情况下将会消除任何音频。
          loop: false, // 循环播放
          preload: 'auto',  // 建议浏览器在<video>加载元素后是否应该开始下载视频数据。auto浏览器选择最佳行为,立即开始加载视频(如果浏览器支持)
          language: 'zh-CN',aspectRatio: '16:9', // 将播放器置于流畅模式,并在计算播放器的动态大小时使用该值。值应该代表一个比例 - 用冒号分隔的两个数字(例如"16:9"或"4:3")
          fluid: true, // 当true时,Video.js player将拥有流体大小。换句话说,它将按比例缩放以适应其容器。
          sources: [{ // 播放资源和资源格式
            type: "video/mp4",src: "http://img.ksbbs.com/asset/Mon_1703/05cacb4e02f9d9e.mp4" //你的视频地址(必填)
          }],poster: "../static/courses/675076.jpeg", //视频封面图
          width: document.documentElement.clientWidth, // 默认视频全屏时的最大宽度
          notSupportedMessage: '此视频暂无法播放,请稍后再试', //允许覆盖Video.js无法播放媒体源时显示的默认信息。
        }}},watch:{course(data){// 替换视频地址
        console.log(data)console.log(this.playerOptions.sources[0].src)this.playerOptions.sources[0].src = data.video;// 替换视频封面this.playerOptions.poster = data.course_img;// 替换科恒信息中的详情介绍里面的图片路径while(data.brief.search(`"/media`) != -1 ){data.brief = data.brief.replace(`"/media`,`"${this.$settings.Host}/media`)}},tabIndex(data){if(data==2){//获取当前课程对应的章节列表和课时列表this.$axios.get(`${this.$settings.Host}/courses/chapters/?course=${this.course_id}`).then(response=>{this.chapter_list = response.data;}).catch(error=>{console.log(error.response)})}}},created(){// 获取当前课程IDthis.course_id = this.$route.query.id - 0;// 判断ID基本有效性
      let _this = this;if( isNaN(this.course_id) || this.course_id < 1 ){_this.$alert("无效的课程ID!","错误",{callback(){_this.$router.go(-1);}});}// 发送请求获取后端课程数据this.$axios.get(this.$settings.Host+`/courses/detail/${this.course_id}/`).then(response=>{this.course = response.data;// 修改视频中的封面图片this.playerOptions.poster = this.course.course_img;}).catch(error=>{console.log(error.response)});},methods: {// 视频播放事件
      onPlayerPlay(player) {alert("play");},// 视频暂停播放事件
      onPlayerPause(player){alert("pause");},// 视频插件初始化
      player() {return this.$refs.videoPlayer.player;}},components:{Header,Footer,videoPlayer,}
}
</script><style scoped>
.main{background: #fff;padding-top: 30px;
}
.course-info{width: 1200px;margin: 0 auto;overflow: hidden;
}
.wrap-left{float: left;width: 690px;height: 388px;background-color: #000;
}
.wrap-right{float: left;position: relative;height: 388px;
}
.course-name{font-size: 20px;color: #333;padding: 10px 23px;letter-spacing: .45px;
}
.data{padding-left: 23px;padding-right: 23px;padding-bottom: 16px;font-size: 14px;color: #9b9b9b;
}
.sale-time{width: 464px;background: #fa6240;font-size: 14px;color: #4a4a4a;padding: 10px 23px;overflow: hidden;
}
.sale-type {font-size: 16px;color: #fff;letter-spacing: .36px;float: left;
}
.sale-time .expire{font-size: 14px;color: #fff;float: right;
}
.sale-time .expire .second{width: 24px;display: inline-block;background: #fafafa;color: #5e5e5e;padding: 6px 0;text-align: center;
}
.course-price{background: #fff;font-size: 14px;color: #4a4a4a;padding: 5px 23px;
}
.discount{font-size: 26px;color: #fa6240;margin-left: 10px;display: inline-block;margin-bottom: -5px;
}
.original{font-size: 14px;color: #9b9b9b;margin-left: 10px;text-decoration: line-through;
}
.buy{width: 464px;padding: 0px 23px;position: absolute;left: 0;bottom: 20px;overflow: hidden;
}
.buy .buy-btn{float: left;
}
.buy .buy-now{width: 125px;height: 40px;border: 0;background: #ffc210;border-radius: 4px;color: #fff;cursor: pointer;margin-right: 15px;outline: none;
}
.buy .free{width: 125px;height: 40px;border-radius: 4px;cursor: pointer;margin-right: 15px;background: #fff;color: #ffc210;border: 1px solid #ffc210;
}
.add-cart{float: right;font-size: 14px;color: #ffc210;text-align: center;cursor: pointer;margin-top: 10px;
}
.add-cart img{width: 20px;height: 18px;margin-right: 7px;vertical-align: middle;
}.course-tab{width: 100%;background: #fff;margin-bottom: 30px;box-shadow: 0 2px 4px 0 #f0f0f0;}
.course-tab .tab-list{width: 1200px;margin: auto;color: #4a4a4a;overflow: hidden;
}
.tab-list li{float: left;margin-right: 15px;padding: 26px 20px 16px;font-size: 17px;cursor: pointer;
}
.tab-list .active{color: #ffc210;border-bottom: 2px solid #ffc210;
}
.tab-list .free{color: #fb7c55;
}
.course-content{width: 1200px;margin: 0 auto;background: #FAFAFA;overflow: hidden;padding-bottom: 40px;
}
.course-tab-list{width: 880px;height: auto;padding: 20px;background: #fff;float: left;box-sizing: border-box;overflow: hidden;position: relative;box-shadow: 0 2px 4px 0 #f0f0f0;
}
.tab-item{width: 880px;background: #fff;padding-bottom: 20px;box-shadow: 0 2px 4px 0 #f0f0f0;
}
.tab-item-title{justify-content: space-between;padding: 25px 20px 11px;border-radius: 4px;margin-bottom: 20px;border-bottom: 1px solid #333;border-bottom-color: rgba(51,51,51,.05);overflow: hidden;
}
.chapter{font-size: 17px;color: #4a4a4a;float: left;
}
.chapter-length{float: right;font-size: 14px;color: #9b9b9b;letter-spacing: .19px;
}
.chapter-title{font-size: 16px;color: #4a4a4a;letter-spacing: .26px;padding: 12px;background: #eee;border-radius: 2px;display: -ms-flexbox;display: flex;-ms-flex-align: center;align-items: center;
}
.chapter-title img{width: 18px;height: 18px;margin-right: 7px;vertical-align: middle;
}
.lesson-list{padding:0 20px;
}
.lesson-list .lesson-item{padding: 15px 20px 15px 36px;cursor: pointer;justify-content: space-between;position: relative;overflow: hidden;
}
.lesson-item .name{font-size: 14px;color: #666;float: left;
}
.lesson-item .index{margin-right: 5px;
}
.lesson-item .free{font-size: 12px;color: #fff;letter-spacing: .19px;background: #ffc210;border-radius: 100px;padding: 1px 9px;margin-left: 10px;
}
.lesson-item .time{font-size: 14px;color: #666;letter-spacing: .23px;opacity: 1;transition: all .15s ease-in-out;float: right;
}
.lesson-item .time img{width: 18px;height: 18px;margin-left: 15px;vertical-align: text-bottom;
}
.lesson-item .try{width: 86px;height: 28px;background: #ffc210;border-radius: 4px;font-size: 14px;color: #fff;position: absolute;right: 20px;top: 10px;opacity: 0;transition: all .2s ease-in-out;cursor: pointer;outline: none;border: none;
}
.lesson-item:hover{background: #fcf7ef;box-shadow: 0 0 0 0 #f3f3f3;
}
.lesson-item:hover .name{color: #333;
}
.lesson-item:hover .try{opacity: 1;
}.course-side{width: 300px;height: auto;margin-left: 20px;float: right;
}
.teacher-info{background: #fff;margin-bottom: 20px;box-shadow: 0 2px 4px 0 #f0f0f0;
}
.side-title{font-weight: normal;font-size: 17px;color: #4a4a4a;padding: 18px 14px;border-bottom: 1px solid #333;border-bottom-color: rgba(51,51,51,.05);
}
.side-title span{display: inline-block;border-left: 2px solid #ffc210;padding-left: 12px;
}.teacher-content{padding: 30px 20px;box-sizing: border-box;
}.teacher-content .cont1{margin-bottom: 12px;overflow: hidden;
}.teacher-content .cont1 img{width: 54px;height: 54px;margin-right: 12px;float: left;
}
.teacher-content .cont1 .name{float: right;
}
.teacher-content .cont1 .teacher-name{width: 188px;font-size: 16px;color: #4a4a4a;padding-bottom: 4px;
}
.teacher-content .cont1 .teacher-title{width: 188px;font-size: 13px;color: #9b9b9b;white-space: nowrap;
}
.teacher-content .narrative{font-size: 14px;color: #666;line-height: 24px;
}
</style>

View Code

视频播放

项目中有两种视频:收费视频[需要加密]和免费视频

使用保利威云视频服务来对视频进行加密

官方网址: http://www.polyv.net/vod/

注意:

开发时通过免费试用注册体验版账号

公司使用酷播尊享版

开发文档地址:http://dev.polyv.net/2017/videoproduct/v-playerapi/html5player/html5-docs/

要开发播放保利威的加密视频功能,需要在用户中心->设置->API接口和加密设置.

http://my.polyv.net/secure/setting/api

配置视频上传加密.

上传视频并记录视频的VID

后端获取保利威的视频播放授权token,提供接口api给前端

参考文档:http://dev.polyv.net/2019/videoproduct/v-api/v-api-play/create-playsafe-token/

根据官方文档的案例,已经有其他人开源了,针对polvy的token生成的python版本了,我们可以直接拿来使用.

在libs下创建polyv.py,编写token生成工具函数

from django.conf import settings
import time
import requests
import hashlibclass PolyvPlayer(object):userId = settings.POLYV_CONFIG['userId']secretkey = settings.POLYV_CONFIG['secretkey']def tomd5(self, value):"""取md5值"""return hashlib.md5(value.encode()).hexdigest()# 获取视频数据的tokendef get_video_token(self, videoId, viewerIp, viewerId=None, viewerName='', extraParams='HTML5'):""":param videoId: 视频id:param viewerId: 看视频用户id:param viewerIp: 看视频用户ip:param viewerName: 看视频用户昵称:param extraParams: 扩展参数:param sign: 加密的sign:return: 返回点播的视频的token"""ts = int(time.time() * 1000)  # 时间戳plain = {"userId": self.userId,'videoId': videoId,'ts': ts,'viewerId': viewerId,'viewerIp': viewerIp,'viewerName': viewerName,'extraParams': extraParams}# 按照ASCKII升序 key + value + key + value... + value 拼接plain_sorted = {}key_temp = sorted(plain)for key in key_temp:plain_sorted[key] = plain[key]print(plain_sorted)plain_string = ''for k, v in plain_sorted.items():plain_string += str(k) + str(v)print(plain_string)sign_data = self.secretkey + plain_string + self.secretkey# 取sign_data的md5的大写sign = self.tomd5(sign_data).upper()# 新的带有sign的字典plain.update({'sign': sign})result = requests.post(url='https://hls.videocc.net/service/v1/token',headers={"Content-type": "application/x-www-form-urlencoded"},data=plain).json()data = {} if isinstance(result, str) else result.get("data", {})return {"token": data}

View Code

配置文件settings/dev.py,代码

# 保利威视频加密服务
POLYV_CONFIG = {"userId":"注册获取","secretkey":"注册获取","servicesUrl":"https://hls.videocc.net/service/v1/token",
}

视图代码:

from rest_framework.response import Response
from luffy.utils.polyv import PolyvPlayerfrom rest_framework.views import APIView
class PolyvAPIView(APIView):def get(self, request):vid = request.query_params.get("vid")remote_addr = request.META.get("REMOTE_ADDR")user_id = 1  # 测试使用user_name = "test"  # 测试使用polyv_video = PolyvPlayer()verify_data = polyv_video.get_video_token(vid, remote_addr, user_id, user_name)return Response(verify_data["token"])

路由代码:

path(r"polyv/token/",views.PolyvAPIView.as_view()),

客户端请求token并播放视频

在 vue项目的入口文件index.html 中加载保利威视频播放器的js核心类库

<script src='https://player.polyv.net/script/polyvplayer.min.js'></script>

创建视频播放页面的组件Player.vue,组件中直接配置保利威播放器需要的参数。

Player.vue,代码:

<template><div class="player"><div id="player"></div></div>
</template><script>
export default {name:"Player",data () {return {}},methods: {},mounted(){let _this = this;var player = polyvObject('#player').videoPlayer({wrap: '#player',width: document.documentElement.clientWidth, // 宽度
        height: document.documentElement.clientHeight, // 高度
        forceH5: true,vid: '62dc475e3f09b6db69447011eed4415a_6',code: '骑士3期', // 一般是用户昵称// 视频加密播放的配置
        playsafe: function (vid, next) { // 向后端发送请求获取加密的token
            _this.$axios.get(_this.$settings.Host+`/courses/polyv/token/`,{params:{vid: "62dc475e3f09b6db69447011eed4415a_6",}}).then(function (response) {console.log(response);next(response.data.token);})}});},computed: {}
}
</script><style scoped>
</style>

View Code

前端路由,代码:

   {name:"Player",path:"/player",component: Player,},

完善点击课程详情页的立即试学按钮跳转到视频播放页面,并发送vid

Detail.vue,代码:

课时章节:

<button class="try" v-if="lesson.free_trail"><router-link :to="{path: '/player',query:{'vid':lesson.section_link}}">立即试学</router-link></button>

View Code

Player.vue,代码:

获取vid视频ID

<template><div class="player"><div id="player"></div></div>
</template><script>
export default {name:"Player",data () {return {}},methods: {},mounted(){let _this = this;let video_id = this.$route.query.vid;var player = polyvObject('#player').videoPlayer({wrap: '#player',width: document.documentElement.clientWidth, // 宽度
        height: document.documentElement.clientHeight, // 高度
        forceH5: true,vid:video_id, // vid:vid,的简写
        code: '骑士3期', // 一般是用户昵称// 视频加密播放的配置
        playsafe: function (vid, next) { // 向后端发送请求获取加密的token
            _this.$axios.get(_this.$settings.Host+`/courses/polyv/token/`,{params:{vid:video_id,}}).then(function (response) {console.log(response);next(response.data.token);})}});},computed: {}
}
</script><style scoped>
</style>

View Code

完善API接口的身份认证

试学必须在用户登录以后才能进行,所以后端的tokenAPI接口必须保证用户登陆以后,

所以后端视图代码中增加对jwt token的识别认证,代码:

from rest_framework.views import APIView
from luffy.libs.polyv import PolyvPlayer
from rest_framework.response import Response
from rest_framework.permissions import IsAuthenticated
class PolyvAPIView(APIView):"""生成播放视频的playsafetoken""""""播放页面的当前访问者只能是用户,不能是游客"""permission_classes = (IsAuthenticated,)def get(self,request):# 获取客户端要播放的视频vidvid = request.query_params.get("vid")# 获取客户端的IP地址remote_addr = request.META.get("REMOTE_ADDR")# 获取用户的ID和用户名[测试]user_id = request.user.iduser_name = request.user.username# 生成tokenpolyv = PolyvPlayer()data = polyv.get_video_token(vid, remote_addr,user_id, user_name)return Response(data["token"])

View Code

前端在请求后端提供视频加密播放的token时需要附带 jwt token

Player.vue,代码:

<template><div class="player"><div id="player"></div></div>
</template><script>
export default {name:"Player",data () {return {token: sessionStorage.token || localStorage.token,user_id: sessionStorage.user_id || localStorage.user_id,user_name: sessionStorage.user_name || localStorage.user_name,}},methods: {},created(){// 判断用户用户是否已经登录了if(!this.token){let _this = this;this.$alert("对不起,您尚未登录!请登录!","警告",{callback(){_this.$router.push("/login");}})}},mounted(){let _this = this;let video_id = this.$route.query.vid;var player = polyvObject('#player').videoPlayer({wrap: '#player',width: document.documentElement.clientWidth, // 宽度
        height: document.documentElement.clientHeight, // 高度
        forceH5: true,vid:video_id, // vid:vid,的简写
        code: _this.user_name, // 跑马灯的显示信息,一般是用户昵称// 视频加密播放的配置
        playsafe: function (vid, next) { // 向后端发送请求获取加密的token
            _this.$axios.get(_this.$settings.Host+`/courses/polyv/token/`,{// 附带jwt token
              headers:{// 注意下方的空格!!!"Authorization":"jwt " + _this.token},params:{vid:video_id,}}).then(function (response) {console.log(response);next(response.data.token);})}});},computed: {}
}
</script><style scoped>
</style>

View Code

详情页的视频免费播放

在课程模型Courses/models.py中新增一个视频的字段

from ckeditor_uploader.fields import RichTextUploadingField
class Course(BaseModel):"""专题课程"""video = models.FileField(upload_to="video", null=True,blank=True,default=None, verbose_name="封面视频")

执行数据迁移

python manage.py makemigrations
python manage.py migrate

前端修改播放器中关于视频地址和视频封面的地址

watch:{course(data){// 替换视频地址this.playerOptions.sources[0].src = data.video;// 替换视频封面this.playerOptions.poster = data.course_img;// 替换科恒信息中的详情介绍里面的图片路径while(data.brief.search(`"/media`) != -1 ){data.brief = data.brief.replace(`"/media`,`"${this.$settings.Host}/media`)}},

转载于:https://www.cnblogs.com/yang950718/p/10879357.html

vue ---05 分页和详情页功能的实现相关推荐

  1. 关于ASP.NET给产品分类,分页,详情页生成静态页面

    之前讲了如何给栏目页生成静态.现在剩下复杂的产品分类,分页,详情页生成静态页面. 我采用的原理是.产品分类通过循环全部生成静态页面. 这个就不说了,跟之前生成栏目页方法一样. 接下来是产品分页和详情页 ...

  2. Vue.js框架简单读取数据库信息并渲染完成news新闻文章列表以及detail详情页功能(小试牛刀)

    项目结构 news.html(新闻列表文件) <!doctype html> <html lang="en"> <head><meta c ...

  3. Vue商城——详情页功能

    详情页实现思路 点击商品进去详情页,根据点击请求更加详细的信息,要传过来goodsItem的iid,根据id去服务器请求更加详细的信息:配置路由映射关系,点击进行跳转,带参数传递跳转. 在GoodsL ...

  4. vue实现商品详情页功能之商品选项卡

    用户点击商品进入商品详情页,默认显示第一个小图对应的大图,然后鼠标滑到小图上,大图也会发生改变,实现效果如下: 实现代码: shopitem.vue的template(HTML),上面是大图,下面是小 ...

  5. vue音乐项目歌手详情页小结

    技术栈 1,vue 2,vuex 3,vue-router(子路由) 需求分析 1)歌手列表点击歌手会跳转到下级页面歌手详情页,歌手详情页由四个部分组成 歌手图片 返回按钮:点击返回歌手tab页 随机 ...

  6. django博客项目-文章详情页功能

    文章详情页左侧边栏数据复用 文章详情页和个人站点的左侧边栏的内容格式都是一样的,从个人站点路由和文章详情路由进入到网页,侧边栏显示的内容是一模一样,解决方案: 方案一: 写一个home_site.ht ...

  7. vue 实现分页和多页签功能

    直接看源码 <template><view class="wrap"><u-navbar back-text="返回" title ...

  8. Angular1.4.6框架简单读取数据库信息并渲染完成news新闻文章列表以及detail详情页功能(小试牛刀)

    项目结构 css/angular-common.css table tr td:first-child {/**背景图片*/width: 200px;height: 100px;/**居中填满*/ba ...

  9. springboot根据id查看详情页功能(就是用id获取全部数据)

    xml: <select id="selectBfhtreswById" resultType="com.icbc.icbc.entity.Bfhtresw&quo ...

最新文章

  1. 阿里巴巴开源技术汇总:115个软件(一)
  2. Anaconda闪退问题
  3. ORACLE开发:创建与管理表空间和数据文件1
  4. python判断是不是整数的命令_介绍python判断一个数是不是正小数和整数的方法
  5. 打勾显示输入的密码 --EditText与setTransformationMethod
  6. Hadoop权威指南-读书笔记
  7. Ubuntu 16.04 安装monaco字体
  8. Power Management of Hybrid DRAM/PRAM-Based Main Memory
  9. VBA字典数组转置维度变化
  10. priority inversion
  11. 《惢客创业日记》2019.05.18(周六)视频通话后的一个创意
  12. 理想主义者与现实主义者的差别
  13. 【人工智能】3.谓词与机器推理
  14. NBD(Network Block Device)简介及基本使用
  15. BPM、BPMN介绍
  16. apache服务构建虚拟web主机
  17. [生存志] 第84节 列子淡泊号冲虚
  18. openvz学习笔记
  19. android自动回复退订,[原创]某聊天app自动回复
  20. 肖锰:浪潮GS开发平台学习札记(一)——服务器端安装

热门文章

  1. c语言迷宫游戏会遇到的问题,C语言迷宫问题
  2. python进行文件运行手机_手机上面如何运行Python
  3. 移动硬盘恢复数据,一招就能做到
  4. iOS 监控体系之电池状态监控【电池的状态处理:电池状态获取及监测、电池电量获取及监测、低电量模式切换监测】应用场景:ASO机刷的场景就需要保证设备的高可用性(UIDeviceBattery)
  5. 专业显卡深度学习_胜任专业学习工作领域,双11就买RTX显卡神舟战神游戏本
  6. 微软做小程序的思路值得我们细细思量
  7. java语言和python语言发展前景哪个好?
  8. 寻路算法——A*算法详解并附带实现代码
  9. Word 如何添加斜线表头(单/多斜线)
  10. C++的string长度和插入函数