├── app
├── app.js
├── app.wxss
├── images
│ ├── logo.png
│ ├── qr.png
│ └── camera.png
├── config.js
├── pages
│ ├── index
│ │ ├── index.json
│ │ ├── index.js
│ │ ├── index.wxml
│ │ └── index.wxss
│ └── album
│ │ ├── album.wxss
│ │ ├── album.wxml
│ │ └── album.js
├── lib
│ ├── api.js
│ ├── request.js
│ └── util.js
└── app.json
├── server
├── globals.js
├── routes
│ ├── index.js
│ └── album
│ │ ├── routehub.js
│ │ └── handlers
│ │ ├── list.js
│ │ ├── delete.js
│ │ └── upload.js
├── process.json
├── services
│ └── cos
│ │ └── index.js
├── common
│ └── routerbase.js
├── package.json
├── config.js
├── app.js
└── middlewares
│ └── route_dispatcher.js
├── .gitignore
├── screenshot.png
├── LICENSE
└── README.md
/app/app.js:
--------------------------------------------------------------------------------
1 | App({
2 |
3 | });
--------------------------------------------------------------------------------
/server/globals.js:
--------------------------------------------------------------------------------
1 | global.SERVER_ROOT = __dirname;
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/
2 | npm-debug.log
3 | tmp
--------------------------------------------------------------------------------
/app/app.wxss:
--------------------------------------------------------------------------------
1 | page {
2 | background-color: #1c1b16;
3 | }
--------------------------------------------------------------------------------
/screenshot.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-album/HEAD/screenshot.png
--------------------------------------------------------------------------------
/server/routes/index.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | '/album': 'album/routehub',
3 | };
4 |
--------------------------------------------------------------------------------
/app/images/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-album/HEAD/app/images/logo.png
--------------------------------------------------------------------------------
/app/images/qr.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-album/HEAD/app/images/qr.png
--------------------------------------------------------------------------------
/app/images/camera.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CFETeam/weapp-demo-album/HEAD/app/images/camera.png
--------------------------------------------------------------------------------
/app/config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | /** 通讯域名 */
3 | host: 'www.qcloud.la',
4 | basePath: '/applet/album',
5 | };
--------------------------------------------------------------------------------
/app/pages/index/index.json:
--------------------------------------------------------------------------------
1 | {
2 | "navigationBarBackgroundColor": "#2277da",
3 | "navigationBarTextStyle": "white"
4 | }
--------------------------------------------------------------------------------
/app/pages/index/index.js:
--------------------------------------------------------------------------------
1 | Page({
2 | // 前往相册页
3 | gotoAlbum() {
4 | wx.navigateTo({ url: '../album/album' });
5 | },
6 | });
7 |
--------------------------------------------------------------------------------
/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 | };
--------------------------------------------------------------------------------
/server/process.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "album",
3 | "script": "app.js",
4 | "cwd": "./",
5 | "exec_mode": "fork",
6 | "watch": true,
7 | "ignore_watch": ["tmp"],
8 | "env": {
9 | "NODE_ENV": "production"
10 | }
11 | }
--------------------------------------------------------------------------------
/server/routes/album/routehub.js:
--------------------------------------------------------------------------------
1 | module.exports = (router) => {
2 | // 获取图片列表
3 | router.get('/list', require('./handlers/list'));
4 |
5 | // 上传图片
6 | router.post('/upload', require('./handlers/upload'));
7 |
8 | // 删除图片
9 | router.post('/delete', require('./handlers/delete'));
10 | };
--------------------------------------------------------------------------------
/app/app.json:
--------------------------------------------------------------------------------
1 | {
2 | "pages": [
3 | "pages/index/index",
4 | "pages/album/album"
5 | ],
6 |
7 | "window": {
8 | "backgroundTextStyle": "dark",
9 | "navigationBarBackgroundColor": "#373b3e",
10 | "navigationBarTitleText": "相册回收站",
11 | "windowBackground": "#1c1b16",
12 | "navigationBarTextStyle": "white"
13 | }
14 | }
--------------------------------------------------------------------------------
/server/services/cos/index.js:
--------------------------------------------------------------------------------
1 | const COS = require('cos-nodejs-sdk-v5')
2 | const _cosConfig = require('cos-nodejs-sdk-v5/sdk/config')
3 | const config = require('../../config')
4 |
5 | /**
6 | * init COS config
7 | * see: https://github.com/tencentyun/cos-nodejs-sdk-v5/blob/master/sdk/config.js#L24
8 | */
9 | _cosConfig.setAppInfo(config.cosAppId, config.cosSecretId, config.cosSecretKey);
10 |
11 | module.exports = COS
12 |
--------------------------------------------------------------------------------
/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/pages/index/index.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 恭喜你
4 | 成功地搭建了一个微信小程序
5 |
6 |
7 |
8 |
9 |
10 | 分享二维码邀请好友结伴一起写小程序!
11 |
12 |
13 |
14 |
--------------------------------------------------------------------------------
/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 | };
--------------------------------------------------------------------------------
/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/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "album-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 | "cos-nodejs-sdk-v5": "^1.0.0",
16 | "express": "^4.14.0",
17 | "file-type": "^3.8.0",
18 | "lodash": "^4.16.1",
19 | "morgan": "^1.7.0",
20 | "multiparty": "^4.1.2",
21 | "qcloud_cos": "^1.0.6",
22 | "read-chunk": "^2.0.0",
23 | "shortid": "^2.2.6"
24 | },
25 | "devDependencies": {
26 | "nodemon": "^1.10.2"
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/server/config.js:
--------------------------------------------------------------------------------
1 | const CONF = {
2 | port: '9993',
3 | ROUTE_BASE_PATH: '/applet',
4 |
5 | /**
6 | * COS 信息配置
7 | * 查看:https://console.qcloud.com/capi
8 | */
9 |
10 | // APPID
11 | cosAppId: '',
12 | /**
13 | * 区域
14 | * 华北:cn-north
15 | * 华东:cn-east
16 | * 华南:cn-south
17 | * 西南:cn-southwest
18 | */
19 | cosRegion: 'cn-north',
20 | // SecretId
21 | cosSecretId: '',
22 | // SecretKey
23 | cosSecretKey: '',
24 | // Bucket 名称
25 | cosFileBucket: '',
26 | // 文件夹
27 | cosUploadFolder: '/'
28 | }
29 |
30 | // 生成访问 COS 的域名,无需修改
31 | CONF.cosDomain = (() => `http://${CONF.cosFileBucket}-${CONF.cosAppId}.costj.myqcloud.com/`)()
32 |
33 | module.exports = CONF
34 |
--------------------------------------------------------------------------------
/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/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;
--------------------------------------------------------------------------------
/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.
--------------------------------------------------------------------------------
/server/routes/album/handlers/list.js:
--------------------------------------------------------------------------------
1 | const _ = require('lodash');
2 | const path = require('path');
3 | const RouterBase = require('../../../common/routerbase');
4 | const config = require('../../../config');
5 | const cos = require('../../../services/cos');
6 |
7 | class ListImages extends RouterBase {
8 | handle() {
9 | const cosParams = {
10 | Bucket : config.cosFileBucket,
11 | Region : config.cosRegion,
12 | MaxKeys : 100
13 | }
14 |
15 | cos.getBucket(cosParams, (err, res) => {
16 |
17 | if (err) {
18 | this.res.json({ code: -1, msg: 'failed', data: {} });
19 | return;
20 | }
21 |
22 | this.res.json({
23 | code: 0,
24 | msg: 'ok',
25 | data: _.map(res.Contents, 'Key').filter(item => {
26 | let extname = String(path.extname(item)).toLowerCase();
27 |
28 | // 只返回`jpg/png`后缀图片
29 | return ['.jpg', '.png'].includes(extname)
30 | }).map(v => config.cosDomain + v),
31 | });
32 | });
33 | }
34 | }
35 |
36 | module.exports = ListImages.makeRouteHandler();
--------------------------------------------------------------------------------
/app/pages/album/album.wxss:
--------------------------------------------------------------------------------
1 | .container {
2 | width: 100%;
3 | height: 100%;
4 | }
5 |
6 | .album-container {
7 | margin: 0.1rem 0;
8 | }
9 |
10 | .item-group {
11 | display: flex;
12 | }
13 |
14 | .album-item {
15 | flex: 1;
16 | margin: 0.1rem;
17 | background: #333;
18 | text-align: center;
19 | height: 6.66rem;
20 | line-height: 6.66rem;
21 | }
22 |
23 | .album-item.empty {
24 | background: transparent;
25 | }
26 |
27 | .upload-image {
28 | color: #ccc;
29 | background: #333;
30 | position: absolute;
31 | left: 0.1rem;
32 | top: 0.2rem;
33 | width: 6.46rem;
34 | height: 6.66rem;
35 | text-align: center;
36 | line-height: 6.66rem;
37 | }
38 |
39 | .upload-image image {
40 | position: absolute;
41 | left: 0;
42 | width: 100%;
43 | height: 2.6rem;
44 | top: 1.2rem;
45 | }
46 |
47 | .upload-image text {
48 | position: absolute;
49 | width: 100%;
50 | height: 2rem;
51 | left: 0;
52 | top: 1.6rem;
53 | }
54 |
55 | .swiper-container {
56 | position: fixed;
57 | left: 0;
58 | top: 0;
59 | width: 100%;
60 | height: 100%;
61 | background: #000;
62 | }
63 |
64 | .swiper-container image {
65 | width: 100%;
66 | height: 100%;
67 | }
68 |
69 | action-sheet-item.warn {
70 | color: #e64340;
71 | }
--------------------------------------------------------------------------------
/server/routes/album/handlers/delete.js:
--------------------------------------------------------------------------------
1 | const _ = require('lodash');
2 | const RouterBase = require('../../../common/routerbase');
3 | const config = require('../../../config');
4 | const cos = require('../../../services/cos');
5 |
6 | class DeleteImage extends RouterBase {
7 | handle() {
8 | let filePath = String(this.req.body.filepath || '');
9 |
10 | try {
11 | filePath = decodeURIComponent(filePath);
12 |
13 | if (!filePath.startsWith(config.cosUploadFolder)) {
14 | throw new Error('operation forbidden');
15 | }
16 |
17 | } catch (err) {
18 | return this.res.json({
19 | code: -1,
20 | msg: 'failed',
21 | data: _.pick(err, ['name', 'message']),
22 | });
23 | }
24 |
25 | filePath = filePath.slice(1)
26 |
27 | const params = {
28 | Bucket: config.cosFileBucket,
29 | Region: config.cosRegion,
30 | Key: filePath
31 | }
32 |
33 | cos.deleteObject(params, (err, data) => {
34 | if (err) {
35 | this.res.json({
36 | code: -1,
37 | msg: 'failed',
38 | data: _.pick(err, ['name', 'message']),
39 | })
40 | } else {
41 | this.res.json({
42 | code: 0,
43 | msg: 'ok',
44 | data: {},
45 | })
46 | }
47 | })
48 | }
49 | }
50 |
51 | module.exports = DeleteImage.makeRouteHandler();
--------------------------------------------------------------------------------
/app/pages/album/album.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 | 上传图片
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 | 保存到本地
30 | 删除图片
31 | 取消
32 |
33 |
34 | {{loadingMessage}}
35 | {{toastMessage}}
--------------------------------------------------------------------------------
/server/routes/album/handlers/upload.js:
--------------------------------------------------------------------------------
1 | const fs = require('fs');
2 | const path = require('path');
3 | const multiparty = require('multiparty');
4 | const readChunk = require('read-chunk');
5 | const fileType = require('file-type');
6 | const shortid = require('shortid');
7 | const RouterBase = require('../../../common/routerbase');
8 | const config = require('../../../config');
9 | const cos = require('../../../services/cos');
10 |
11 | class ImageUploader extends RouterBase {
12 | constructor() {
13 | super(...arguments);
14 |
15 | // 图片允许上传的最大文件大小,单位(M)
16 | this.MAX_FILE_SIZE = 5;
17 | }
18 |
19 | handle() {
20 | const result = { 'code': -1, 'msg': '', 'data': {} };
21 |
22 | this.parseForm()
23 | .then(({ files }) => {
24 | if (!('image' in files)) {
25 | result.msg = '参数错误';
26 | return;
27 | }
28 |
29 | const imageFile = files.image[0];
30 |
31 | const buffer = readChunk.sync(imageFile.path, 0, 262);
32 | const resultType = fileType(buffer);
33 | if (!resultType || !['image/jpeg', 'image/png'].includes(resultType.mime)) {
34 | result.msg = '仅jpg/png格式';
35 | return;
36 | }
37 |
38 | const srcpath = imageFile.path
39 | const imgKey = `${Date.now()}-${shortid.generate()}.${resultType.ext}`
40 | const params = {
41 | Bucket: config.cosFileBucket,
42 | Region: config.cosRegion,
43 | Key: imgKey,
44 | Body: srcpath,
45 | ContentLength: imageFile.size
46 | }
47 |
48 | return new Promise((resolve, reject) => {
49 | cos.putObject(params, (err, data) => {
50 | if (err) reject(err)
51 |
52 | console.log(data)
53 | result.code = 0
54 | result.msg = 'ok'
55 | result.data.imgUrl = config.cosDomain + imgKey
56 | resolve()
57 |
58 | // remove uploaded file
59 | fs.unlink(srcpath)
60 |
61 | });
62 | });
63 |
64 | })
65 | .catch(e => {
66 | console.log(e)
67 | if (e.statusCode === 413) {
68 | result.msg = `单个不超过${this.MAX_FILE_SIZE}MB`;
69 | } else {
70 | result.msg = '图片上传失败,请稍候再试';
71 | }
72 | })
73 | .then(() => {
74 | this.res.json(result);
75 | });
76 | }
77 |
78 | parseForm() {
79 | const form = new multiparty.Form({
80 | encoding: 'utf8',
81 | maxFilesSize: this.MAX_FILE_SIZE * 1024 * 1024,
82 | autoFiles: true,
83 | uploadDir: path.join(global.SERVER_ROOT, 'tmp'),
84 | })
85 |
86 | return new Promise((resolve, reject) => {
87 | form.parse(this.req, (err, fields = {}, files = {}) => {
88 | err ? reject(err) : resolve({ fields, files })
89 | })
90 | })
91 | }
92 | }
93 |
94 | module.exports = ImageUploader.makeRouteHandler();
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # 微信小程序示例 - 小相册
2 |
3 | 小相册是结合腾讯云[对象存储服务](https://www.qcloud.com/product/cos.html)(Cloud Object Service,简称COS)制作的一个微信小程序示例。在代码结构上包含如下两部分:
4 |
5 | - `app`: 小相册应用包代码,可直接在微信开发者工具中作为项目打开
6 | - `server`: 搭建的Node服务端代码,作为服务器和`app`通信,提供 CGI 接口示例用于拉取 COS 图片资源、上传图片到 COS、删除 COS 图片
7 |
8 | 小相册主要功能如下:
9 | * 列出 COS 服务器中的图片列表
10 | * 点击左上角**上传图片**图标,可以调用相机拍照或从手机相册选择图片,并将选中的图片上传到 COS 服务器中
11 | * 轻按任意图片,可进入全屏图片预览模式,并可左右滑动切换预览图片
12 | * 长按任意图片,可将其保存到本地,或从 COS 中删除
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)和[证书申请](https://console.qcloud.com/ssl)服务,还没有域名或者证书的可以去使用
33 |
34 | 域名注册好之后,可以登录[微信公众平台](https://mp.weixin.qq.com)配置通信域名了。
35 |
36 | 
37 |
38 | ### 2. Nginx 和 Node 代码部署
39 |
40 | 小相册服务要运行,需要进行以下几步:
41 |
42 | * 部署 Nginx,Nginx 的安装和部署请大家自行搜索(注意需要把 SSL 模块也编译进去)
43 | * 配置 Nginx 反向代理到 `http://127.0.0.1:9993`
44 | * Node 运行环境,可以安装 [Node V6.6.0](https://nodejs.org/)
45 | * 部署 `server` 目录的代码到服务器上,如 `/data/release/qcloud-applet-album`
46 | * 使用 `npm install` 安装依赖模块
47 | * 使用 `npm install pm2 -g` 安装 pm2
48 |
49 | > 上述环境配置比较麻烦,剪刀石头布的服务器运行代码和配置已经打包成[腾讯云 CVM 镜像](https://buy.qcloud.com/cvm?marketImgId=371),推荐大家直接使用。
50 | > * 镜像部署完成之后,云主机上就有运行 WebSocket 服务的基本环境、代码和配置了。
51 | > * 腾讯云用户可以[免费领取礼包](https://www.qcloud.com/act/event/yingyonghao.html#section-voucher),体验腾讯云小程序解决方案。
52 | > * 镜像已包含「剪刀石头布」和「小相册」两个小程序的服务器环境与代码,需要体验两个小程序的朋友无需重复部署
53 |
54 | ### 3. 配置 HTTPS
55 |
56 | 镜像中已经部署了 nginx,需要在 `/etc/nginx/conf.d` 下修改配置中的域名、证书、私钥。
57 |
58 | 
59 |
60 |
61 | 配置完成后,即可启动 nginx。
62 |
63 | ```sh
64 | nginx
65 | ```
66 |
67 | ### 4. 域名解析
68 |
69 | 我们还需要添加域名记录解析到我们的云服务器上,这样才可以使用域名进行 HTTPS 服务。
70 |
71 | 在腾讯云注册的域名,可以直接使用[云解析控制台](https://console.qcloud.com/cns/domains)来添加主机记录,直接选择上面购买的 CVM。
72 |
73 | 
74 |
75 | 解析生效后,我们在浏览器使用域名就可以进行 HTTPS 访问。
76 |
77 | 
78 |
79 | ### 5. 开通和配置 COS
80 |
81 | 小相册示例的图片资源是存储在 COS 上的,要使用 COS 服务,需要登录 [COS 管理控制台](https://console.qcloud.com/cos/overview),然后在其中完成以下操作:
82 |
83 | - 开通 COS 服务分配得到唯一的`APP ID`
84 | - 使用密钥管理生成一对`SecretID`和`SecretKey`(用于调用 COS API)
85 | - 在 Bucket 列表中创建**公有读私有写**访问权限、**CDN加速**的 bucket(存储图片的目标容器)
86 | - 在创建的 bucket 容器中创建文件夹,命名为`photos`,图片将会上传存储到该目录下
87 |
88 | ### 6. 启动小相册示例 Node 服务
89 |
90 | 在镜像中,小相册示例的 Node 服务代码已部署在目录 `/data/release/qcloud-applet-album` 下:
91 |
92 | 进入该目录:
93 |
94 | ```bash
95 | cd /data/release/qcloud-applet-album
96 | ```
97 |
98 | 在该目录下有个名为`config.js`的配置文件(如下所示),按注释修改对应的 COS 配置:
99 |
100 | ```js
101 | module.exports = {
102 | // Node 监听的端口号
103 | port: '9993',
104 | ROUTE_BASE_PATH: '/applet',
105 |
106 | cosAppId: '填写开通 COS 时分配的 APP ID',
107 | cosSecretId: '填写密钥 SecretID',
108 | cosSecretKey: '填写密钥 SecretKey',
109 | cosFileBucket: '填写创建的公有读私有写的bucket名称',
110 | };
111 | ```
112 |
113 | 代码运行需要临时目录,在部署目录下创建一个临时目录 `tmp`。
114 |
115 | ```sh
116 | mkdir tmp
117 | ```
118 |
119 | 小相册示例使用`pm2`管理 Node 进程,执行以下命令启动 Node 服务:
120 |
121 | ```bash
122 | pm2 start process.json
123 | ```
124 |
125 | ### 7. 微信小程序服务器配置
126 |
127 | 进入微信公众平台管理后台设置服务器配置,配置类似如下设置:
128 |
129 | 
130 |
131 | 注意:需要将 `www.qcloud.la` 设置为上面申请的域名,将 downloadFile 合法域名设置为在 COS 管理控制台中自己创建的 bucket 的相应 **CDN 加速访问地址**,如下图所示:
132 |
133 | 
134 |
135 | ### 8. 启动小相册 Demo
136 |
137 | 在微信开发者工具将小相册应用包源码添加为项目,并把源文件`config.js`中的通讯域名修改成上面申请的域名。
138 |
139 | 
140 |
141 | 然后点击调试即可打开小相册 Demo 开始体验。
142 |
143 | 
144 |
145 | 这里有个问题。截止目前为止,微信小程序提供的上传和下载 API 无法在调试工具中正常工作,需要用手机微信扫码预览体验。
146 |
147 | ## 主要功能实现
148 |
149 | ### 上传图片
150 |
151 | 上传图片使用了微信小程序提供的`wx.chooseImage(OBJECT)`获取需要上传的文件路径,然后调用上传文件接口`wx.request(OBJECT)`发送 HTTPS POST 请求到自己指定的后台服务器。和传统表单文件上传一样,请求头`Content-Type`也是`multipart/form-data`。后台服务器收到请求后,使用 npm 模块 multiparty 解析 `multipart/form-data` 请求,将解析后的数据保存为指定目录下的临时文件。拿到临时文件的路径后,就可直接调用 COS SDK 提供的[文件上传 API](https://www.qcloud.com/doc/product/430/5947#.E6.96.87.E4.BB.B6.E4.B8.8A.E4.BC.A0) 进行图片存储,最后得到图片的存储路径及访问地址(存储的图片路径也可以直接在 COS 管理控制台看到)。
152 |
153 | ### 获取图片列表
154 |
155 | 调用[列举目录下文件&目录 API](https://www.qcloud.com/doc/product/430/5947#.E5.88.97.E4.B8.BE.E7.9B.AE.E5.BD.95.E4.B8.8B.E6.96.87.E4.BB.B6.26amp.3B.E7.9B.AE.E5.BD.95)可以获取到在 COS 服务端指定 bucket 和该 bucket 指定路径下存储的图片。
156 |
157 | ### 下载和保存图片
158 |
159 | 指定图片的访问地址,然后调用微信小程序提供的 `wx.downloadFile(OBJECT)` 和 `wx.saveFile(OBJECT)` 接口可以直接将图片下载和保存本地。这里要注意图片访问地址的域名需要和服务器配置的 downloadFile 合法域名一致,否则无法下载。
160 |
161 | ### 删除图片
162 |
163 | 删除图片也十分简单,直接调用[文件删除 API](https://www.qcloud.com/doc/product/430/5947#.E6.96.87.E4.BB.B6.E5.88.A0.E9.99.A4) 就可以将存储在 COS 服务端的图片删除。
164 |
165 | ## LICENSE
166 |
167 | [MIT](LICENSE)
168 |
--------------------------------------------------------------------------------
/app/pages/album/album.js:
--------------------------------------------------------------------------------
1 | const config = require('../../config.js');
2 | const { listToMatrix, always } = 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 | albumList: [],
10 |
11 | // 图片布局列表(二维数组,由`albumList`计算而得)
12 | layoutList: [],
13 |
14 | // 布局列数
15 | layoutColumnSize: 3,
16 |
17 | // 是否显示loading
18 | showLoading: false,
19 |
20 | // loading提示语
21 | loadingMessage: '',
22 |
23 | // 是否显示toast
24 | showToast: false,
25 |
26 | // 提示消息
27 | toastMessage: '',
28 |
29 | // 是否显示动作命令
30 | showActionsSheet: false,
31 |
32 | // 当前操作的图片
33 | imageInAction: '',
34 |
35 | // 图片预览模式
36 | previewMode: false,
37 |
38 | // 当前预览索引
39 | previewIndex: 0,
40 | },
41 |
42 | // 显示loading提示
43 | showLoading(loadingMessage) {
44 | this.setData({ showLoading: true, loadingMessage });
45 | },
46 |
47 | // 隐藏loading提示
48 | hideLoading() {
49 | this.setData({ showLoading: false, loadingMessage: '' });
50 | },
51 |
52 | // 显示toast消息
53 | showToast(toastMessage) {
54 | this.setData({ showToast: true, toastMessage });
55 | },
56 |
57 | // 隐藏toast消息
58 | hideToast() {
59 | this.setData({ showToast: false, toastMessage: '' });
60 | },
61 |
62 | // 隐藏动作列表
63 | hideActionSheet() {
64 | this.setData({ showActionsSheet: false, imageInAction: '' });
65 | },
66 |
67 | onLoad() {
68 | this.renderAlbumList();
69 |
70 | this.getAlbumList().then((resp) => {
71 | if (resp.code !== 0) {
72 | // 图片列表加载失败
73 | return;
74 | }
75 |
76 | this.setData({ 'albumList': this.data.albumList.concat(resp.data) });
77 | this.renderAlbumList();
78 | });
79 | },
80 |
81 | // 获取相册列表
82 | getAlbumList() {
83 | this.showLoading('加载列表中…');
84 | setTimeout(() => this.hideLoading(), 1000);
85 | return request({ method: 'GET', url: api.getUrl('/list') });
86 | },
87 |
88 | // 渲染相册列表
89 | renderAlbumList() {
90 | let layoutColumnSize = this.data.layoutColumnSize;
91 | let layoutList = [];
92 |
93 | if (this.data.albumList.length) {
94 | layoutList = listToMatrix([0].concat(this.data.albumList), layoutColumnSize);
95 |
96 | let lastRow = layoutList[layoutList.length - 1];
97 | if (lastRow.length < layoutColumnSize) {
98 | let supplement = Array(layoutColumnSize - lastRow.length).fill(0);
99 | lastRow.push(...supplement);
100 | }
101 | }
102 |
103 | this.setData({ layoutList });
104 | },
105 |
106 | // 从相册选择照片或拍摄照片
107 | chooseImage() {
108 | wx.chooseImage({
109 | count: 9,
110 | sizeType: ['original', 'compressed'],
111 | sourceType: ['album', 'camera'],
112 |
113 | success: (res) => {
114 | this.showLoading('正在上传图片…');
115 |
116 | console.log(api.getUrl('/upload'));
117 | wx.uploadFile({
118 | url: api.getUrl('/upload'),
119 | filePath: res.tempFilePaths[0],
120 | name: 'image',
121 |
122 | success: (res) => {
123 | let response = JSON.parse(res.data);
124 |
125 | if (response.code === 0) {
126 | console.log(response);
127 |
128 | let albumList = this.data.albumList;
129 | albumList.unshift(response.data.imgUrl);
130 |
131 | this.setData({ albumList });
132 | this.renderAlbumList();
133 |
134 | this.showToast('图片上传成功');
135 | } else {
136 | console.log(response);
137 | }
138 | },
139 |
140 | fail: (res) => {
141 | console.log('fail', res);
142 | },
143 |
144 | complete: () => {
145 | this.hideLoading();
146 | },
147 | });
148 |
149 | },
150 | });
151 | },
152 |
153 | // 进入预览模式
154 | enterPreviewMode(event) {
155 | if (this.data.showActionsSheet) {
156 | return;
157 | }
158 |
159 | let imageUrl = event.target.dataset.src;
160 | let previewIndex = this.data.albumList.indexOf(imageUrl);
161 |
162 | this.setData({ previewMode: true, previewIndex: previewIndex });
163 | },
164 |
165 | // 退出预览模式
166 | leavePreviewMode() {
167 | this.setData({ previewMode: false, previewIndex: 0 });
168 | },
169 |
170 | // 显示可操作命令
171 | showActions(event) {
172 | this.setData({ showActionsSheet: true, imageInAction: event.target.dataset.src });
173 | },
174 |
175 | // 下载图片
176 | downloadImage() {
177 | this.showLoading('正在保存图片…');
178 | console.log('download_image_url', this.data.imageInAction);
179 |
180 | wx.downloadFile({
181 | url: this.data.imageInAction,
182 | type: 'image',
183 | success: (resp) => {
184 | wx.saveFile({
185 | tempFilePath: resp.tempFilePath,
186 | success: (resp) => {
187 | this.showToast('图片保存成功');
188 | },
189 |
190 | fail: (resp) => {
191 | console.log('fail', resp);
192 | },
193 |
194 | complete: (resp) => {
195 | console.log('complete', resp);
196 | this.hideLoading();
197 | },
198 | });
199 | },
200 |
201 | fail: (resp) => {
202 | console.log('fail', resp);
203 | },
204 | });
205 |
206 | this.setData({ showActionsSheet: false, imageInAction: '' });
207 | },
208 |
209 | // 删除图片
210 | deleteImage() {
211 | let imageUrl = this.data.imageInAction;
212 | let filepath = '/' + imageUrl.split('/').slice(3).join('/');
213 |
214 | this.showLoading('正在删除图片…');
215 | this.setData({ showActionsSheet: false, imageInAction: '' });
216 |
217 | request({
218 | method: 'POST',
219 | url: api.getUrl('/delete'),
220 | data: { filepath },
221 | })
222 | .then((resp) => {
223 | if (resp.code !== 0) {
224 | // 图片删除失败
225 | return;
226 | }
227 |
228 | // 从图片列表中移除
229 | let index = this.data.albumList.indexOf(imageUrl);
230 | if (~index) {
231 | let albumList = this.data.albumList;
232 | albumList.splice(index, 1);
233 |
234 | this.setData({ albumList });
235 | this.renderAlbumList();
236 | }
237 |
238 | this.showToast('图片删除成功');
239 | })
240 | .catch(error => {
241 | console.log('failed', error);
242 | })
243 | .then(() => {
244 | this.hideLoading();
245 | });
246 | },
247 | });
--------------------------------------------------------------------------------
/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 | }
--------------------------------------------------------------------------------