├── .gitignore
├── LICENSE
├── README.md
├── app
├── app.js
├── app.json
├── app.wxss
├── config.js
├── images
│ ├── loading.gif
│ ├── logo.png
│ ├── play.png
│ └── qr.png
├── lib
│ ├── api.js
│ ├── request.js
│ └── util.js
└── pages
│ ├── detail
│ ├── detail.js
│ ├── detail.wxml
│ └── detail.wxss
│ ├── index
│ ├── index.js
│ ├── index.wxml
│ └── index.wxss
│ └── video
│ ├── video.js
│ ├── video.wxml
│ └── video.wxss
└── server
├── app.js
├── common
└── routerbase.js
├── config.js
├── globals.js
├── middlewares
└── route_dispatcher.js
├── package.json
├── process.json
└── routes
├── index.js
└── video
├── handlers
├── comment.js
├── commentlist.js
└── list.js
└── routehub.js
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | LICENSE - "MIT License"
2 |
3 | Copyright (c) 2016 by Tencent Cloud
4 |
5 | Permission is hereby granted, free of charge, to any person
6 | obtaining a copy of this software and associated documentation
7 | files (the "Software"), to deal in the Software without
8 | restriction, including without limitation the rights to use,
9 | copy, modify, merge, publish, distribute, sublicense, and/or sell
10 | copies of the Software, and to permit persons to whom the
11 | Software is furnished to do so, subject to the following
12 | conditions:
13 |
14 | The above copyright notice and this permission notice shall be
15 | included in all copies or substantial portions of the Software.
16 |
17 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
19 | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
21 | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
22 | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
23 | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
24 | OTHER DEALINGS IN THE SOFTWARE.
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # 微信小程序示例 - 新片预告
2 |
3 | 新片预告是结合腾讯云[点播 VOD](https://www.qcloud.com/product/vod.html?utm_source=jiaocheng&utm_medium=vod-introduction&utm_campaign=qcloud)和[云数据库 MySQL](https://www.qcloud.com/product/cdb.html?utm_source=jiaocheng&utm_medium=cdb-introduction&utm_campaign=qcloud)制作的一个微信小程序示例。在代码结构上包含如下两部分:
4 |
5 | - `app`: 新片预告应用包代码,可直接在微信开发者工具中作为项目打开
6 | - `server`: 搭建的Node服务端代码,作为服务器和`app`通信,提供 CGI 接口示例用于拉取云数据库上的视频列表、评论列表,将评论数据提交到云数据库
7 |
8 | 新片预告主要功能如下:
9 | * 支持分页滚动加载视频列表
10 | * 点击海报跳转至详情页播放视频
11 | * 对视频进行评论
12 | * 展示视频的评论列表
13 |
14 | 
15 |
16 |
17 | ## 部署和运行
18 |
19 | 拿到了本小程序源码的朋友可以尝试自己运行起来。
20 |
21 | ### 整体架构
22 |
23 | 
24 |
25 | ### 1. 准备域名和证书
26 |
27 | 在微信小程序中,所有的网路请求受到严格限制,不满足条件的域名和协议无法请求,具体包括:
28 |
29 | * 只允许和在 MP 中配置好的域名进行通信,如果还没有域名,需要注册一个。
30 | * 网络请求必须走 HTTPS 协议,所以你还需要为你的域名申请一个 SSL 证书。
31 |
32 | > 腾讯云提供[域名注册](https://www.qcloud.com/product/dm.html?utm_source=jiaocheng&utm_medium=domain2&utm_campaign=qcloud)和[证书申请](https://console.qcloud.com/ssl?utm_source=jiaocheng&utm_medium=ssl2&utm_campaign=qcloud)服务,还没有域名或者证书的可以去使用
33 |
34 | 域名注册好之后,可以登录[微信公众平台](https://mp.weixin.qq.com)配置通信域名了。
35 |
36 | 
37 |
38 | 注意:需要将 `www.qcloud.la` 设置为上面申请的域名
39 |
40 | ### 2. Nginx 和 Node 代码部署
41 |
42 | 小程序服务要运行,需要进行以下几步:
43 |
44 | * 部署 Nginx,Nginx 的安装和部署请大家自行搜索(注意需要把 SSL 模块也编译进去)
45 | * 配置 Nginx 反向代理到 `http://127.0.0.1:9994`
46 | * Node 运行环境,可以安装 [Node V6.6.0](https://nodejs.org/)
47 | * 部署 `server` 目录的代码到服务器上,如 `/data/release/qcloud-applet-video`
48 | * 使用 `npm install` 安装依赖模块
49 | * 使用 `npm install pm2 -g` 安装 pm2
50 |
51 | > 上述环境配置比较麻烦,新片预告的服务器运行代码和配置已经打包成[腾讯云 CVM 镜像](https://buy.qcloud.com/cvm?marketImgId=371?utm_source=jiaocheng&utm_medium=cvm2&utm_campaign=qcloud),推荐大家直接使用。
52 | > * 镜像部署完成之后,云主机上就有运行 WebSocket 服务的基本环境、代码和配置了。
53 | > * 腾讯云用户可以[免费领取礼包](https://www.qcloud.com/act/event/yingyonghao.html#section-voucher),体验腾讯云小程序解决方案。
54 | > * 镜像已包含所有小程序的服务器环境与代码,需要体验小程序的朋友无需重复部署
55 |
56 | ### 3. 配置 HTTPS
57 |
58 | 镜像中已经部署了 nginx,需要在 `/etc/nginx/conf.d` 下修改配置中的域名、证书、私钥。
59 |
60 | 
61 |
62 |
63 | 配置完成后,即可启动 nginx。
64 |
65 | ```sh
66 | nginx
67 | ```
68 |
69 | ### 4. 域名解析
70 |
71 | 我们还需要添加域名记录解析到我们的云服务器上,这样才可以使用域名进行 HTTPS 服务。
72 |
73 | 在腾讯云注册的域名,可以直接使用[云解析控制台](https://console.qcloud.com/cns/domains?utm_source=jiaocheng&utm_medium=cns&utm_campaign=qcloud)来添加主机记录,直接选择上面购买的 CVM。
74 |
75 | 
76 |
77 | 解析生效后,我们在浏览器使用域名就可以进行 HTTPS 访问。
78 |
79 | 
80 |
81 | ### 5. 开通 点播服务
82 |
83 | 新片预告示例的播放资源是存储在 腾讯云点播 上的mp4文件,要使用 点播 服务,需要登录 [点播 管理控制台](http://console.qcloud.com/video?utm_source=jiaocheng&utm_medium=vod-console&utm_campaign=qcloud),然后在其中完成以下操作:
84 |
85 | - 上传视频资源,点播几乎支持所有主流的[视频格式](https://www.qcloud.com/doc/product/266/2846)上传
86 | - 转码成功后获取mp4或m3u8源地址
87 |
88 | 
89 |
90 | > 目前微信小程序`video`组件经测试支持`mp4`和`m3u8`格式,其中 m3u8 格式只能在手机上使用,开发者可以使用腾讯云点播控制台将视频源转码成 mp4 或 m3u8 格式,并且腾讯云点播会对播放的资源进行CDN加速。
91 |
92 | ### 6. 准备 云数据库MySQL
93 | 示例中拉取的视频和评论列表都是存储在 云数据库 上,要使用 [云数据库](https://www.qcloud.com/product/cdb.html?utm_source=jiaocheng&utm_medium=cdb-introduction&utm_campaign=qcloud) 服务需要完成以下操作
94 |
95 | - [购买](https://buy.qcloud.com/cdb?utm_source=jiaocheng&utm_medium=cdb-purchase&utm_campaign=qcloud),注意购买的云数据库需要与云服务器同在一个地域分区
96 | - [初始化流程](https://www.qcloud.com/doc/product/236/3128),本示例选用的是`utf8`编码
97 | - 点击[云数据库 控制台](https://console.qcloud.com/cdb?utm_source=jiaocheng&utm_medium=cdb-console&utm_campaign=qcloud)操作栏的`登录`按钮,登录到phpMyAdmin`创建数据库`并在当前数据库中导入本示例中的[SQL文件](https://share-10039692.file.myqcloud.com/wechat_app.sql)
98 |
99 | > 注意:导入SQL文件中包含了 点播 上传的视频列表,开发者可以基于云数据库自行开发维护一个视频发布管理系统,因为此内容跟本示例暂不相关,所以不再详述。
100 |
101 | ### 7. 启动新片预告示例 Node 服务
102 |
103 | 在镜像中,新片预告示例的 Node 服务代码已部署在目录`/data/release/qcloud-applet-video`下:
104 |
105 | 进入该目录:
106 |
107 | ```bash
108 | cd /data/release/qcloud-applet-video
109 | ```
110 |
111 | 在该目录下有个名为`config.js`的配置文件(如下所示),按注释修改对应的 MySQL 配置:
112 |
113 | ```js
114 | module.exports = {
115 | // Node 监听的端口号
116 | port: '9994',
117 | ROUTE_BASE_PATH: '/applet',
118 |
119 | host: '填写开通 MySQL 时分配的内网IP',
120 | user: '填写MySQL用户名',
121 | password: '填写MySQL密码',
122 | database: '填写上一步中创建的MySQL数据名',
123 | };
124 | ```
125 |
126 | 示例使用`pm2`管理 Node 进程,执行以下命令启动 node 服务:
127 |
128 | ```bash
129 | pm2 start process.json
130 | ```
131 |
132 | ### 8. 启动新片预告 Demo
133 |
134 | 在微信开发者工具将新片预告应用包源码添加为项目,并把源文件`config.js`中的通讯域名修改成上面申请的域名。
135 |
136 | 
137 |
138 | 然后点击调试即可打开新片预告Demo开始体验。
139 |
140 | 
141 |
142 |
143 | ## 主要功能实现
144 |
145 | ### 获取视频列表、展示评论、提交评论
146 | 通过node的mysql模块连接mysql,进行查询,插入操作
147 | 以下是查询评论列表的示例代码
148 |
149 | ```js
150 | const mysql = require('mysql');
151 | const config = require('../../../config');
152 |
153 | let vid = this.req.query.vid;
154 | if (!vid) {
155 | this.res.json({ code: -1, msg: 'failed', data: {} });
156 | return;
157 | }
158 |
159 | //CDB Mysql配置
160 | let connection = mysql.createConnection({
161 | host: config.host,
162 | password: config.password,
163 | user: config.user,
164 | database: config.database
165 | });
166 |
167 | //开启数据库连接
168 | connection.connect((err) => {
169 | if (err) {
170 | this.res.json({ code: -1, msg: 'failed', data: {} });
171 | }
172 | });
173 |
174 | //查询列表
175 | connection.query('SELECT * from comment where vid = ? order by id desc', [vid], (err, result) => {
176 | if (err) {
177 | this.res.json({ code: -1, msg: 'failed', data: {} });
178 | return;
179 | }
180 |
181 | this.res.json({
182 | code: 0,
183 | msg: 'ok',
184 | data: result,
185 | });
186 | });
187 |
188 | //查询完后关闭连接
189 | connection.end();
190 | ```
191 |
192 | ### 播放视频
193 | ```js
194 |
195 | ```
196 |
197 | | 属性名 | 类型 | 说明 |
198 | | :------: | :------: | :------------: |
199 | | src | String | 要播放视频的资源地址 |
200 | | binderror| EventHandle | 当发生错误时触发error事件,event.detail = {errMsg: 'something wrong'} |
201 |
202 | 播放视频使用的是video标签,目前官方文档上只给出了两个参数说明,笔者测试了src支持加载`mp4`和`m3u8`格式视频,
203 | video标签的控制条暂时没办法自定义样式以及隐藏
204 |
205 | ## LICENSE
206 |
207 | [MIT](LICENSE)
208 |
--------------------------------------------------------------------------------
/app/app.js:
--------------------------------------------------------------------------------
1 | App({
2 |
3 | });
--------------------------------------------------------------------------------
/app/app.json:
--------------------------------------------------------------------------------
1 | {
2 | "pages": [
3 | "pages/index/index",
4 | "pages/video/video",
5 | "pages/detail/detail"
6 | ],
7 |
8 | "window": {
9 | "backgroundTextStyle": "dark",
10 | "navigationBarBackgroundColor": "#202020",
11 | "navigationBarTitleText": "新片预告",
12 | "windowBackground": "white",
13 | "navigationBarTextStyle": "white"
14 | }
15 | }
--------------------------------------------------------------------------------
/app/app.wxss:
--------------------------------------------------------------------------------
1 | page {
2 | background-color: #1c1b16;
3 | }
--------------------------------------------------------------------------------
/app/config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | /** 通讯域名 */
3 | host: 'www.qcloud.la',
4 | basePath: '/applet/video',
5 | };
--------------------------------------------------------------------------------
/app/images/loading.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-video/da6180572efd6f4c4e32b2213dd96a39574bb6d5/app/images/loading.gif
--------------------------------------------------------------------------------
/app/images/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-video/da6180572efd6f4c4e32b2213dd96a39574bb6d5/app/images/logo.png
--------------------------------------------------------------------------------
/app/images/play.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-video/da6180572efd6f4c4e32b2213dd96a39574bb6d5/app/images/play.png
--------------------------------------------------------------------------------
/app/images/qr.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-video/da6180572efd6f4c4e32b2213dd96a39574bb6d5/app/images/qr.png
--------------------------------------------------------------------------------
/app/lib/api.js:
--------------------------------------------------------------------------------
1 | var config = require('../config.js');
2 |
3 | module.exports = {
4 | getUrl(route) {
5 | return `https://${config.host}${config.basePath}${route}`;
6 | },
7 | };
--------------------------------------------------------------------------------
/app/lib/request.js:
--------------------------------------------------------------------------------
1 | module.exports = (options) => {
2 | return new Promise((resolve, reject) => {
3 | options = Object.assign(options, {
4 | success(result) {
5 | if (result.statusCode === 200) {
6 | resolve(result.data);
7 | } else {
8 | reject(result);
9 | }
10 | },
11 |
12 | fail: reject,
13 | });
14 |
15 | wx.request(options);
16 | });
17 | };
--------------------------------------------------------------------------------
/app/lib/util.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | // 一维数组转二维数组
3 | listToMatrix(list, elementsPerSubArray) {
4 | let matrix = [], i, k;
5 |
6 | for (i = 0, k = -1; i < list.length; i += 1) {
7 | if (i % elementsPerSubArray === 0) {
8 | k += 1;
9 | matrix[k] = [];
10 | }
11 |
12 | matrix[k].push(list[i]);
13 | }
14 |
15 | return matrix;
16 | },
17 |
18 | // 为promise设置简单回调(无论成功或失败都执行)
19 | always(promise, callback) {
20 | promise.then(callback, callback);
21 | return promise;
22 | },
23 | };
--------------------------------------------------------------------------------
/app/pages/detail/detail.js:
--------------------------------------------------------------------------------
1 | const request = require('../../lib/request.js');
2 | const api = require('../../lib/api.js');
3 |
4 | Page({
5 | data: {
6 | nick: '',
7 | avatar: '',
8 | vid: 0,
9 | commentList: [],
10 | showSubmitLoading: false,
11 | showSubmitSuccessToast: false,
12 | },
13 |
14 | onReady() {
15 | let app = getApp();
16 | wx.setNavigationBarTitle({ title: '播放-' + app.currentVideoTitle});
17 | },
18 |
19 | onLoad() {
20 | let app = getApp();
21 | let self = this;
22 | this.setData({ videoUrl: app.currentVideoUrl, vid: +app.vid });
23 |
24 | this.getUserInfo().then((userInfo) => {
25 | self.setData({
26 | nick: userInfo.nickName,
27 | avatar: userInfo.avatarUrl
28 | });
29 | })
30 |
31 | this.getCommentList().then((resp) => {
32 | if (resp.code !== 0) {
33 | // 视频列表加载失败
34 | return;
35 | }
36 |
37 | this.setData({ commentList: resp.data });
38 | });
39 | },
40 |
41 | getUserInfo() {
42 | return new Promise((resolve, reject) => {
43 | wx.getUserInfo({ success: (resp) => {
44 | resolve(resp.userInfo);
45 | }, fail: reject })
46 | })
47 | },
48 |
49 | // 获取评论列表
50 | getCommentList() {
51 | let promise = request({
52 | method: 'GET',
53 | data: {vid: this.data.vid},
54 | url: api.getUrl('/commentList'),
55 | });
56 | return promise;
57 | },
58 |
59 | commentInputChange(e) {
60 | this.data.commentContent = e.detail.value.trim();
61 | },
62 |
63 | submitComment() {
64 | if (!this.data.commentContent || this.data.isSubmiting) return;
65 |
66 | let params = {
67 | vid: this.data.vid,
68 | nick: this.data.nick,
69 | avatar: this.data.avatar,
70 | content: this.data.commentContent
71 | };
72 |
73 | this.setData({isSubmiting: true, showSubmitLoading: true});
74 | request({
75 | method: 'POST',
76 | url: api.getUrl('/comment'),
77 | data: params
78 | }).then((resp) => {
79 | this.setData({isSubmiting: false, showSubmitLoading: false});
80 |
81 | if (resp.code == 0) {
82 | this.setData({
83 | commentList: [params].concat(this.data.commentList),
84 | showSubmitSuccessToast: true,
85 | });
86 |
87 | setTimeout(() => {
88 | this.setData( {showSubmitSuccessToast: false} );
89 | }, 2000)
90 | }
91 | });
92 | }
93 | });
--------------------------------------------------------------------------------
/app/pages/detail/detail.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
23 |
24 |
25 |
26 | 发表中...
27 |
28 |
29 |
30 | 发表成功
31 |
--------------------------------------------------------------------------------
/app/pages/detail/detail.wxss:
--------------------------------------------------------------------------------
1 | .container {
2 | width: 100%;
3 | height: 100%;
4 | background: #f0f0f0;
5 | }
6 |
7 | .container video {
8 | width: 100%;
9 | height: 15rem;
10 | background: black;
11 | }
12 |
13 | .comment {
14 | background: white;
15 | height: auto;
16 | margin-top: 1rem;
17 | }
18 |
19 | .list{
20 | padding: 20px;
21 | }
22 | .item{
23 | display: flex;
24 | margin-bottom: 20px;
25 | }
26 | .item.first-child{
27 | border-bottom: #eee 1px solid;
28 | padding-bottom: 15px;
29 | margin-bottom: 30px;
30 | }
31 | .photo{
32 | display: inline-block;
33 | width:45px;
34 | height: 45px;
35 | border: #eee 1px solid;
36 | }
37 | .photo image{
38 | display: block;
39 | width: 100%;
40 | height: 100%;
41 | }
42 | .content{
43 | flex: 1;
44 | padding-left: 20px;
45 | font-size: 14px;
46 | text-align: left;
47 | }
48 | .input{
49 | box-sizing: border-box;
50 | border: #eee 1px solid;
51 | width: 100%;
52 | height: 45px;
53 | font-size: 14px;
54 | padding:0 10px;
55 | }
56 | .top{
57 | color: #666;
58 | display: block;
59 | margin-bottom: 10px;
60 | }
61 | .text{
62 | color: #333;
63 | display: block;
64 | margin-bottom: 10px;
65 | }
66 | .bottom{
67 | color: #999;
68 | font-size: 12px;
69 | display: block;
70 | }
71 |
72 | .button{
73 | margin-top:10px;
74 | width: 120px;
75 | height: 35px;
76 | font-size: 12px;
77 | line-height: 35px;
78 | }
79 |
--------------------------------------------------------------------------------
/app/pages/index/index.js:
--------------------------------------------------------------------------------
1 | Page({
2 | // 前往视频列表
3 | gotoVideo() {
4 | wx.navigateTo({ url: '../video/video' });
5 | },
6 | });
7 |
--------------------------------------------------------------------------------
/app/pages/index/index.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 恭喜你
4 | 成功地搭建了一个微信小程序
5 |
6 |
7 |
8 |
9 |
10 | 分享二维码邀请好友结伴一起写小程序!
11 |
12 |
13 |
14 |
--------------------------------------------------------------------------------
/app/pages/index/index.wxss:
--------------------------------------------------------------------------------
1 | .page-top {
2 | width: 750rpx;
3 | height: 594rpx;
4 | background-image: url();
5 | background-repeat: no-repeat;
6 | background-size: 750rpx 694rpx;
7 | background-position:0 -40rpx;
8 | position: relative;
9 | z-index: 2;
10 | }
11 | .username,.text-info {
12 | position: absolute;
13 | left:50%;
14 | transform: translateX(-50%);
15 | white-space: nowrap;
16 | }
17 | .username {
18 | font-size: 40rpx;
19 | color: #fff;
20 | top:339rpx;
21 | }
22 | .text-info {
23 | font-size: 32rpx;
24 | color:#bdd0ee;
25 | top:400rpx;
26 | }
27 | .page-btn-wrap {
28 | position: absolute;
29 | top: 470rpx;
30 | width: 100%;
31 | text-align: center;
32 | }
33 | .page-btn {
34 | position:relative;
35 | margin:0 20rpx;
36 | padding:0;
37 | box-sizing:border-box;
38 | font-size:32rpx;
39 | text-decoration:none;
40 | -webkit-tap-highlight-color:transparent;
41 | overflow:hidden;
42 | display: inline-block;
43 | width: 300rpx;
44 | height: 85rpx;
45 | background-color: #fff;
46 | color: #2277da;
47 | line-height: 85rpx;
48 | }
49 | .page-bottom {
50 | background-color: #fff;
51 | width: 100%;
52 | height: 100%;
53 | position: absolute;
54 | top: 0;
55 | left: 0;
56 | padding: 624rpx 0 0;
57 | z-index: 1;
58 | box-sizing: border-box;
59 | }
60 | .qr-img {
61 | display: block;
62 | width: 300rpx;
63 | height: 300rpx;
64 | margin: 40rpx auto;
65 | }
66 | .qr-txt {
67 | display: block;
68 | color: #666;
69 | font-size: 32rpx;
70 | margin: 20rpx auto 0;
71 | text-align: center;
72 | }
73 | .page-logo {
74 | display: block;
75 | width: 200rpx;
76 | height: 54rpx;
77 | margin: 40rpx auto 40rpx;
78 | }
--------------------------------------------------------------------------------
/app/pages/video/video.js:
--------------------------------------------------------------------------------
1 | const config = require('../../config.js');
2 | const util = require('../../lib/util.js');
3 | const request = require('../../lib/request.js');
4 | const api = require('../../lib/api.js');
5 |
6 | Page({
7 | data: {
8 | // 视频列表数据
9 | videoList: [],
10 |
11 | // 列表布局数据,
12 | layerList: [],
13 |
14 | //一行几个
15 | layoutColumnSize: 2,
16 |
17 | //当前加载页
18 | curPage: 1,
19 |
20 | //总页数
21 | totalPage: 1,
22 |
23 | //是否显示加载图标
24 | showLoding: true
25 | },
26 |
27 | onLoad() {
28 | this.getVideoList();
29 | },
30 |
31 | // 获取相册列表
32 | getVideoList() {
33 | request({
34 | data: {pageNo: this.data.curPage},
35 | method: 'GET',
36 | url: api.getUrl('/list'),
37 | }).then((resp) => {
38 | if (resp.code !== 0) {
39 | // 视频列表加载失败
40 | return;
41 | }
42 | this.setData({totalPage: resp.data.totalPage});
43 |
44 | //对vid进行初始化,如果有的话就保留,如果没有的话,就默认时间戳
45 | for( let i = 0 ; i < resp.data.list.length ; i++ ){
46 | let cur = resp.data.list[i];
47 | if( cur ){
48 | cur.id = cur.id || Date.now();
49 | }
50 | }
51 |
52 | this.renderVideoList(resp.data.list);
53 | });
54 | },
55 |
56 | // 渲染影片列表
57 | renderVideoList(videoList) {
58 | let layoutColumnSize = this.data.layoutColumnSize;
59 | videoList = this.data.videoList.concat(videoList);
60 | this.setData({ videoList: videoList });
61 |
62 | let layoutList = [];
63 | if (videoList.length) {
64 | layoutList = util.listToMatrix(videoList, layoutColumnSize);
65 | }
66 |
67 | setTimeout(() => {
68 | this.setData({ layoutList: layoutList, showLoding: false});
69 | }, 500)
70 | },
71 |
72 | //滚动到底部时触发
73 | scrollToLower(e) {
74 | if (this.data.showLoding || this.data.curPage == this.data.totalPage) return;
75 |
76 | this.setData( {curPage: ++this.data.curPage, showLoding: true} )
77 | this.getVideoList();
78 | },
79 |
80 | //播放影片
81 | gotoPlay(event) {
82 | let currentVideoUrl = event.currentTarget.dataset.src;
83 | let currentVideoTitle = event.currentTarget.dataset.title;
84 | let vid = event.currentTarget.dataset.vid;
85 | if (!currentVideoUrl || !currentVideoTitle || !vid) return;
86 |
87 | let app = getApp();
88 | app.currentVideoUrl = currentVideoUrl;
89 | app.currentVideoTitle = currentVideoTitle;
90 | app.vid = vid;
91 |
92 | wx.navigateTo({ url: '../detail/detail' });
93 | },
94 | });
95 |
--------------------------------------------------------------------------------
/app/pages/video/video.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
10 |
11 |
12 | {{item.video_title}}
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 | 加载中...
21 |
--------------------------------------------------------------------------------
/app/pages/video/video.wxss:
--------------------------------------------------------------------------------
1 | .container {
2 | width: 100%;
3 | height: 100%;
4 | background: white;
5 | }
6 |
7 | .item-group {
8 | display: flex;
9 | height: 15rem;
10 | margin-bottom: 0.5rem;
11 | }
12 |
13 | .video-item {
14 | flex: 1;
15 | margin: 0.1rem;
16 | text-align: center;
17 | width: 9.7rem;
18 | height: 15rem;
19 | border: 0.1rem solid #ebebeb;
20 | position:relative;
21 | }
22 |
23 | .video-image {
24 | text-align: center;
25 | width: 9.7rem;
26 | height: 13.5rem;
27 | }
28 |
29 | .play-image {
30 | height: 3rem;
31 | width: 3rem;
32 | position:absolute;
33 | z-index: 2;
34 | top: 5.5rem;
35 | left: 3.8rem;
36 | }
37 |
38 | .video-title {
39 | position: absolute;
40 | color: #333;
41 | width: 9.7rem;
42 | top: 13.7rem;
43 | left: 0.1rem;
44 | font-size: 0.8rem;
45 | }
46 |
47 | .loading{
48 | width: 100%;
49 | text-align: center;
50 | }
51 |
52 | .loading image {
53 | width: 0.9rem;
54 | height: 0.9rem;
55 | }
56 |
--------------------------------------------------------------------------------
/server/app.js:
--------------------------------------------------------------------------------
1 | require('./globals');
2 |
3 | const http = require('http');
4 | const express = require('express');
5 | const bodyParser = require('body-parser');
6 | const morgan = require('morgan');
7 | const config = require('./config');
8 |
9 | const app = express();
10 |
11 | app.set('query parser', 'simple');
12 | app.set('case sensitive routing', true);
13 | app.set('jsonp callback name', 'callback');
14 | app.set('strict routing', true);
15 | app.set('trust proxy', true);
16 |
17 | app.disable('x-powered-by');
18 |
19 | // 记录请求日志
20 | app.use(morgan('tiny'));
21 |
22 | // parse `application/x-www-form-urlencoded`
23 | app.use(bodyParser.urlencoded({ extended: true }));
24 |
25 | // parse `application/json`
26 | app.use(bodyParser.json());
27 |
28 | app.use(require('./middlewares/route_dispatcher'));
29 |
30 | // 打印异常日志
31 | process.on('uncaughtException', error => {
32 | console.log(error);
33 | });
34 |
35 | // 启动server
36 | http.createServer(app).listen(config.port, () => {
37 | console.log('Express server listening on port: %s', config.port);
38 | });
39 |
--------------------------------------------------------------------------------
/server/common/routerbase.js:
--------------------------------------------------------------------------------
1 | /**
2 | * 封装的路由公共基类,用于添加公用方法,不能直接实例化
3 | */
4 |
5 | class RouterBase {
6 | constructor(req, res, next) {
7 | Object.assign(this, { req, res, next });
8 | }
9 |
10 | /**
11 | * 静态工厂方法:创建用以响应路由的回调函数
12 | */
13 | static makeRouteHandler() {
14 | return (req, res, next) => new this(req, res, next).handle();
15 | }
16 |
17 | /**
18 | * 子类实现该方法处理请求
19 | */
20 | handle() {
21 | throw new Error(`Please implement instance method \`${this.constructor.name}::handle\`.`);
22 | }
23 | }
24 |
25 | module.exports = RouterBase;
--------------------------------------------------------------------------------
/server/config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | port: '9994',
3 | ROUTE_BASE_PATH: '/applet',
4 |
5 | host: '填写开通 MySQL 时分配的内网IP',
6 | user: '填写MySQL用户名',
7 | password: '填写MySQL密码',
8 | database: '填写上一步中创建的MySQL数据名',
9 | };
--------------------------------------------------------------------------------
/server/globals.js:
--------------------------------------------------------------------------------
1 | global.SERVER_ROOT = __dirname;
--------------------------------------------------------------------------------
/server/middlewares/route_dispatcher.js:
--------------------------------------------------------------------------------
1 | /**
2 | * 通用路由分发器
3 | */
4 |
5 | const express = require('express');
6 | const path = require('path');
7 | const _ = require('lodash');
8 | const config = require('../config');
9 | const routes = require('../routes');
10 |
11 | const routeOptions = { 'caseSensitive': true, 'strict': true };
12 | const routeDispatcher = express.Router(routeOptions);
13 |
14 | _.each(routes, (route, subpath) => {
15 | const router = express.Router(routeOptions);
16 |
17 | let routePath;
18 |
19 | // ignore `config.ROUTE_BASE_PATH` if `subpath` begin with `~`
20 | if (subpath[0] === '~') {
21 | routePath = subpath.slice(1);
22 | } else {
23 | routePath = config.ROUTE_BASE_PATH + subpath;
24 | }
25 |
26 | require(path.join(global.SERVER_ROOT, 'routes', route))(router);
27 |
28 | routeDispatcher.use(routePath, router, (err, req, res, next) => {
29 | // mute `URIError` error
30 | if (err instanceof URIError) {
31 | return next();
32 | }
33 |
34 | throw err;
35 | });
36 | });
37 |
38 | module.exports = routeDispatcher;
--------------------------------------------------------------------------------
/server/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "video-server",
3 | "version": "1.0.0",
4 | "description": "",
5 | "main": "index.js",
6 | "scripts": {
7 | "start-dev": "nodemon app.js",
8 | "start": "pm2 start process.json"
9 | },
10 | "keywords": [],
11 | "author": "",
12 | "license": "ISC",
13 | "dependencies": {
14 | "body-parser": "^1.15.2",
15 | "express": "^4.14.0",
16 | "morgan": "^1.7.0",
17 | "lodash": "^4.16.1",
18 | "mysql": "^2.11.1"
19 | },
20 | "devDependencies": {
21 | "nodemon": "^1.10.2"
22 | }
23 | }
24 |
--------------------------------------------------------------------------------
/server/process.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "video",
3 | "script": "app.js",
4 | "cwd": "./",
5 | "exec_mode": "fork",
6 | "watch": true,
7 | "env": {
8 | "NODE_ENV": "production"
9 | }
10 | }
--------------------------------------------------------------------------------
/server/routes/index.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | '/video': 'video/routehub',
3 | };
4 |
--------------------------------------------------------------------------------
/server/routes/video/handlers/comment.js:
--------------------------------------------------------------------------------
1 | const RouterBase = require('../../../common/routerbase');
2 | const config = require('../../../config');
3 | const mysql = require('mysql');
4 |
5 | class Comment extends RouterBase {
6 | handle() {
7 | let req = this.req;
8 | let comment = Object.assign({}, {
9 | vid: req.body.vid,
10 | nick: req.body.nick,
11 | avatar: req.body.avatar,
12 | content: req.body. content
13 | });
14 |
15 | //CDB Mysql配置
16 | let connection = mysql.createConnection({
17 | host: config.host,
18 | password: config.password,
19 | user: config.user,
20 | database: config.database
21 | });
22 |
23 | //开启数据库连接
24 | connection.connect((err) => {
25 | if (err) {
26 | this.res.json({ code: -1, msg: 'failed'});
27 | }
28 | });
29 |
30 | //提交评论
31 | connection.query('INSERT INTO comment SET ?', comment, (err, result) => {
32 | if (err) {
33 | this.res.json({ code: -1, msg: 'failed'});
34 | return;
35 | }
36 |
37 | this.res.json({
38 | code: 0,
39 | msg: 'ok'
40 | });
41 | });
42 |
43 | //提交完后关闭连接
44 | connection.end();
45 | }
46 | }
47 |
48 | module.exports = Comment.makeRouteHandler();
--------------------------------------------------------------------------------
/server/routes/video/handlers/commentlist.js:
--------------------------------------------------------------------------------
1 | const RouterBase = require('../../../common/routerbase');
2 | const config = require('../../../config');
3 | const mysql = require('mysql');
4 |
5 | class CommentList extends RouterBase {
6 | handle() {
7 | let vid = this.req.query.vid;
8 | if (!vid) {
9 | this.res.json({ code: -1, msg: 'failed', data: {} });
10 | return;
11 | }
12 |
13 | //CDB Mysql配置
14 | let connection = mysql.createConnection({
15 | host: config.host,
16 | password: config.password,
17 | user: config.user,
18 | database: config.database
19 | });
20 |
21 | //开启数据库连接
22 | connection.connect((err) => {
23 | if (err) {
24 | this.res.json({ code: -1, msg: 'failed', data: {} });
25 | }
26 | });
27 |
28 | //查询列表
29 | connection.query('SELECT * from comment where vid = ? order by id desc', [vid], (err, result) => {
30 | if (err) {
31 | this.res.json({ code: -1, msg: 'failed', data: {} });
32 | return;
33 | }
34 |
35 | this.res.json({
36 | code: 0,
37 | msg: 'ok',
38 | data: result,
39 | });
40 | });
41 |
42 | //查询完后关闭连接
43 | connection.end();
44 | }
45 | }
46 |
47 | module.exports = CommentList.makeRouteHandler();
--------------------------------------------------------------------------------
/server/routes/video/handlers/list.js:
--------------------------------------------------------------------------------
1 | const RouterBase = require('../../../common/routerbase');
2 | const config = require('../../../config');
3 | const mysql = require('mysql');
4 |
5 | class VideoList extends RouterBase {
6 | handle() {
7 | //CDB Mysql配置
8 | let connection = mysql.createConnection({
9 | host: config.host,
10 | password: config.password,
11 | user: config.user,
12 | database: config.database
13 | });
14 |
15 | let pageNo = this.req.query.pageNo;
16 | let pageSize = 6;
17 | let start = (pageNo - 1) * pageSize;
18 |
19 | //开启数据库连接
20 | connection.connect((err) => {
21 | if (err) {
22 | this.res.json({ code: -1, msg: 'failed', data: {} });
23 | }
24 | });
25 |
26 | //查询列表
27 | Promise.all([
28 | this.queryList(start, pageSize, connection),
29 | this.queryTotalPage(pageSize, connection)
30 | ]).then(([list, totalPage]) => {
31 | this.res.json({
32 | code: 0,
33 | msg: 'ok',
34 | data: {list, totalPage}
35 | });
36 | })
37 |
38 | //查询完后关闭连接
39 | connection.end();
40 | }
41 |
42 | //查询分页列表
43 | queryList(start, pageSize, connection) {
44 | return new Promise((resolve, rejct) => {
45 | connection.query('SELECT * from video limit ?,?', [start, pageSize], (err, result) => {
46 | if (err) {
47 | this.res.json({ code: -1, msg: 'failed', data: {} });
48 | return;
49 | }
50 |
51 | resolve(result);
52 | })
53 | })
54 | }
55 |
56 | //查询总页数
57 | queryTotalPage(pageSize, connection) {
58 | return new Promise((resolve, rejct) => {
59 | connection.query('SELECT count(*) as count from video', (err, result) => {
60 | if (err) {
61 | this.res.json({ code: -1, msg: 'failed', data: {} });
62 | return;
63 | }
64 |
65 | resolve(Math.ceil(result[0].count / pageSize));
66 | })
67 | })
68 | }
69 | }
70 |
71 | module.exports = VideoList.makeRouteHandler();
--------------------------------------------------------------------------------
/server/routes/video/routehub.js:
--------------------------------------------------------------------------------
1 | module.exports = (router) => {
2 | // 获取视频列表
3 | router.get('/list', require('./handlers/list'));
4 |
5 | // 获取评论列表
6 | router.get('/commentList', require('./handlers/commentlist'));
7 |
8 | // 提交评论
9 | router.post('/comment', require('./handlers/comment'));
10 | };
--------------------------------------------------------------------------------