├── .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 | ![运行截图](https://share-10039692.file.myqcloud.com/app4.png) 15 | 16 | 17 | ## 部署和运行 18 | 19 | 拿到了本小程序源码的朋友可以尝试自己运行起来。 20 | 21 | ### 整体架构 22 | 23 | ![整体架构](https://share-10039692.file.myqcloud.com/app3.png) 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 | ![配置通信域名](https://easyimage-10028115.file.myqcloud.com/internal/tjzpgjrz.y5a.jpg) 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 | ![证书 Nginx 配置](https://easyimage-10028115.file.myqcloud.com/internal/agfty0fn.gfi.jpg) 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 | ![添加域名解析](https://easyimage-10028115.file.myqcloud.com/internal/uw25hdj2.k1u.jpg) 76 | 77 | 解析生效后,我们在浏览器使用域名就可以进行 HTTPS 访问。 78 | 79 | ![HTTPS 访问效果图](https://easyimage-10028115.file.myqcloud.com/internal/bxfkmjea.g41.jpg) 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 | ![上传转码](https://share-10039692.file.myqcloud.com/app5.png) 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 | ![修改配置文件](https://share-10039692.file.myqcloud.com/app1.png) 137 | 138 | 然后点击调试即可打开新片预告Demo开始体验。 139 | 140 | ![调试](https://share-10039692.file.myqcloud.com/app2.png) 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 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | {{item.nick}} 18 | {{item.content}} 19 | 20 | 21 | 22 | 23 | 24 | 25 | 28 | 29 | -------------------------------------------------------------------------------- /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 | -------------------------------------------------------------------------------- /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 | }; --------------------------------------------------------------------------------