├── .editorconfig ├── LICENSE ├── README.md ├── app.js ├── app.json ├── app.wxss ├── comm ├── script │ ├── config.js │ └── fetch.js └── style │ ├── animation.wxss │ └── tab.wxss ├── component ├── filmList │ ├── filmList.wxml │ └── filmList.wxss ├── list │ ├── list.wxml │ └── list.wxss ├── message │ ├── message.js │ ├── message.wxml │ └── message.wxss └── nullTip │ ├── nullTip.wxml │ └── nullTip.wxss ├── dist └── images │ ├── coming_active_icon.png │ ├── coming_icon.png │ ├── green_tri.png │ ├── my_active_icon.png │ ├── my_icon.png │ ├── popular_active_icon.png │ ├── popular_icon.png │ ├── search_active_icon.png │ ├── search_icon.png │ ├── top_active_icon.png │ └── top_icon.png ├── pages ├── about │ ├── about.js │ ├── about.json │ ├── about.wxml │ └── about.wxss ├── coming │ ├── coming.js │ ├── coming.json │ ├── coming.wxml │ └── coming.wxss ├── editPersonInfo │ ├── editPersonInfo.js │ ├── editPersonInfo.json │ ├── editPersonInfo.wxml │ └── editPersonInfo.wxss ├── favorite │ ├── favorite.js │ ├── favorite.json │ ├── favorite.wxml │ └── favorite.wxss ├── filmDetail │ ├── filmDetail.js │ ├── filmDetail.json │ ├── filmDetail.wxml │ └── filmDetail.wxss ├── gallery │ ├── gallery.js │ ├── gallery.json │ ├── gallery.wxml │ └── gallery.wxss ├── history │ ├── history.js │ ├── history.json │ ├── history.wxml │ └── history.wxss ├── location │ ├── location.js │ ├── location.json │ ├── location.wxml │ └── location.wxss ├── my │ ├── my.js │ ├── my.json │ ├── my.wxml │ └── my.wxss ├── personDetail │ ├── personDetail.js │ ├── personDetail.json │ ├── personDetail.wxml │ └── personDetail.wxss ├── personInfo │ ├── personInfo.js │ ├── personInfo.json │ ├── personInfo.wxml │ └── personInfo.wxss ├── popular │ ├── popular.js │ ├── popular.json │ ├── popular.wxml │ └── popular.wxss ├── search │ ├── search.js │ ├── search.json │ ├── search.wxml │ └── search.wxss ├── searchResult │ ├── searchResult.js │ ├── searchResult.json │ ├── searchResult.wxml │ └── searchResult.wxss ├── setting │ ├── setting.js │ ├── setting.json │ ├── setting.wxml │ └── setting.wxss ├── shake │ ├── shake.js │ ├── shake.json │ ├── shake.wxml │ └── shake.wxss ├── skin │ ├── skin.js │ ├── skin.json │ ├── skin.wxml │ └── skin.wxss ├── systemInfo │ ├── systemInfo.js │ ├── systemInfo.json │ ├── systemInfo.wxml │ └── systemInfo.wxss └── top │ ├── top.js │ ├── top.json │ ├── top.wxml │ └── top.wxss ├── resource └── logo.png └── util └── util.js /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | charset = utf-8 5 | indent_style = space 6 | indent_size = 2 7 | end_of_line = lf 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | 11 | [*.md] 12 | insert_final_newline = true 13 | trim_trailing_whitespace = false 14 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2018 Sesine 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 |

logo

2 | 3 | # 🎬 电影推荐小程序 v2.0 4 | 5 | 🎉 查看页面UI请点击:[https://sesine.com/mina](https://sesine.com/mina/) 6 | 7 | 🌠 此项目使用了大部分微信的api,封装了一些UI组件。如果对您有帮助,请给一个⭐️Star。如果在使用过程中发现问题,请给我 [反馈](https://github.com/sesine/wechat-weapp-movie/issues/new) 我会尽量在第一时间解决。 8 | 9 | ## 🔌 使用方法 10 | 11 | 1. 克隆或下载此项目 12 | 2. 在微信开发者工具导入此项目 13 | 3. 域名设置 14 | - 如果有自己的APPID,可以使用自己的APPID,并在微信后台添加可信域名 `https://sesine.com/` 和 `https://sesine.com/mina/api/`。 15 | - 如果没有自己的APPID,可以使用测试APPID。开发时勾选 `不校验合法域名、web-view(业务域名)、TLS 版本以及 HTTPS 证书` 的选项,如果需要在手机上预览,并且出现api无法访问的问题。请在 [微信小程序测试号管理](https://developers.weixin.qq.com/sandbox) 里面添加上一条中的二个可信域名。 16 | 4. 开发、体验与反馈。发现问题时,可以给我 [反馈](https://github.com/sesine/wechat-weapp-movie/issues/new) 17 | 18 | 19 | ## ❤️ 感谢与支持 20 | - 豆瓣api https://sesine.com/ (代理豆瓣api) 21 | 22 | > 由于微信服务器请求豆瓣api次数过多,已经禁止微信小程序请求。请使用新的api地址请求,请将原来的api请求地址从 `https://api.douban.com/v2/` 改为 `https://sesine.com/mina/api/` 即可,api请求的参数不变。 23 | 24 | - 百度地图 http://lbsyun.baidu.com/ 25 | 26 | ## 🚀 v2.0更新内容 27 | 28 | - 将搜索从页面转移到首页面的搜索框 29 | - 增加了电影与人物的浏览历史、收藏功能 30 | - 添加“我的页面”(重头戏) 31 | - 添加摇一摇、相册模块 32 | - 添加设置功能,可清除缓存、编辑个人资料、查看手机信息、更新地理位置 33 | - 添加关于页面,里面列举除了项目中所有使用到的api 34 | 35 | ## 📅 更新日志 36 | 37 | > v1.0的旧版本代码可以通过master分支中的 [tag](https://github.com/sesine/wechat-weapp-movie/tree/v1.0) 中找到 38 | 39 | - **v1.8.0** 2016-11-20 40 | + 1.添加下拉刷新功能 41 | + 2.列表改为模板渲染 42 | + 3.封装了api请求的代码提高可复用性 43 | 44 | - **v1.7.0** 2016-11-21 45 | + 1.添加人物的详情页 46 | + 2.将电影详情和人物详情的网络请求进行了封装 47 | 48 | - **v1.6.0** 2016-11-22 49 | + 1.添加了消息通知组件 50 | + 2.搜索页调整 51 | + 3.调整了目录结构,整合静态资源(dist)和组件文件(component) 52 | 53 | - **v1.5.0** 2016-11-25 54 | + 1.将网络请求从wx.request改为fetch的方式 55 | + 2.完成搜索功能 56 | + 3.完成消息通知组件 57 | + 4.删除了没有用到的util文件夹 58 | 59 | - **v1.4.0** 2016-12-03 60 | + 1.将电影列表的下拉刷新从scroll-view的bindscrolltolower改为Page的onReachBottom事件触发 61 | + 2.将“我的”页面的文件补全,功能列表改为数据渲染,添加跳转。新增换肤 62 | + 3.删除了tabBar中的搜索选项,添加“我的”选项 63 | + 4.首页搜索栏(点击跳转到搜索页)添加轮播图 64 | + 5.添加定位功能,在小程序载入时进行定位 65 | + 6.添加浏览记录、收藏、相册、设置、摇一摇功能(都在开发中) 66 | 67 | - **v1.3.0** 2016-12-04 68 | + 1.将api列表 banner列表、搜索关键词列表、皮肤列表整合配置文件(config.js) 69 | + 2.完成换肤、设置、个人资料(还差修改)、摇一摇功能(再次进入不能摇的问题还需解决) 70 | + 3.添加util文件及文件夹,用于封装获取并格式化时间等工具类方法 71 | + 4.消息组件修改,删除了成功、失败等情况,避免与wx.showToast重合,添加了网络不正常的提示 72 | + 5.电影详情页面添加存储浏览历史的功能 73 | 74 | - **v1.2.0** 2016-12-06 75 | + 1.完成电影收藏和人物收藏功能 76 | + 2.搜索页面添加为空时的提示页面并封装成组件 77 | 78 | - **v1.1.0** 2016-12-14 79 | + 1.完成相册功能和关于页面 80 | + 2.添加定位功能(gps) 81 | 82 | - **v1.0.0** 2016-12-24 83 | + 1.相册的背景图片方式改成image标签的方式 84 | + 2.添加摇一摇debug测试开关变量 85 | -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | var config = require('comm/script/config') 2 | App({ 3 | globalData: { 4 | userInfo: null 5 | }, 6 | onLaunch: function() { 7 | // 获取用户信息 8 | this.getUserInfo() 9 | //初始化缓存 10 | this.initStorage() 11 | }, 12 | getUserInfo:function(cb){ 13 | var that = this 14 | wx.login({ 15 | success: function () { 16 | wx.getUserInfo({ 17 | success: function (res) { 18 | that.globalData.userInfo = res.userInfo 19 | typeof cb == "function" && cb(that.globalData.userInfo) 20 | } 21 | }) 22 | } 23 | }) 24 | }, 25 | getCity: function(cb) { 26 | var that = this 27 | wx.getLocation({ 28 | type: 'gcj02', 29 | success: function (res) { 30 | var locationParam = res.latitude + ',' + res.longitude + '1' 31 | wx.request({ 32 | url: config.apiList.baiduMap, 33 | data: { 34 | ak: config.baiduAK, 35 | location: locationParam, 36 | output: 'json', 37 | pois: '1' 38 | }, 39 | method: 'GET', 40 | success: function(res){ 41 | config.city = res.data.result.addressComponent.city.slice(0,-1) 42 | typeof cb == "function" && cb(res.data.result.addressComponent.city.slice(0,-1)) 43 | }, 44 | fail: function(res) { 45 | // 重新定位 46 | that.getCity(); 47 | } 48 | }) 49 | } 50 | }) 51 | }, 52 | initStorage: function() { 53 | wx.getStorageInfo({ 54 | success: function(res) { 55 | // 判断电影收藏是否存在,没有则创建 56 | if (!('film_favorite' in res.keys)) { 57 | wx.setStorage({ 58 | key: 'film_favorite', 59 | data: [] 60 | }) 61 | } 62 | // 判断人物收藏是否存在,没有则创建 63 | if (!('person_favorite' in res.keys)) { 64 | wx.setStorage({ 65 | key: 'person_favorite', 66 | data: [] 67 | }) 68 | } 69 | // 判断电影浏览记录是否存在,没有则创建 70 | if (!('film_history' in res.keys)) { 71 | wx.setStorage({ 72 | key: 'film_history', 73 | data: [] 74 | }) 75 | } 76 | // 判断人物浏览记录是否存在,没有则创建 77 | if (!('person_history' in res.keys)) { 78 | wx.setStorage({ 79 | key: 'person_history', 80 | data: [] 81 | }) 82 | } 83 | // 个人信息默认数据 84 | var personInfo = { 85 | name: '', 86 | nickName: '', 87 | gender: '', 88 | age: '', 89 | birthday: '', 90 | constellation: '', 91 | company: '', 92 | school: '', 93 | tel: '', 94 | email:'', 95 | intro: '' 96 | } 97 | // 判断个人信息是否存在,没有则创建 98 | if (!('person_info' in res.keys)) { 99 | wx.setStorage({ 100 | key: 'person_info', 101 | data: personInfo 102 | }) 103 | } 104 | // 判断相册数据是否存在,没有则创建 105 | if (!('gallery' in res.keys)) { 106 | wx.setStorage({ 107 | key: 'gallery', 108 | data: [] 109 | }) 110 | } 111 | // 判断背景卡选择数据是否存在,没有则创建 112 | if (!('skin' in res.keys)) { 113 | wx.setStorage({ 114 | key: 'skin', 115 | data: '' 116 | }) 117 | } 118 | } 119 | }) 120 | } 121 | }) -------------------------------------------------------------------------------- /app.json: -------------------------------------------------------------------------------- 1 | { 2 | "pages": [ 3 | "pages/popular/popular", 4 | "pages/coming/coming", 5 | "pages/top/top", 6 | "pages/search/search", 7 | "pages/filmDetail/filmDetail", 8 | "pages/personDetail/personDetail", 9 | "pages/searchResult/searchResult", 10 | "pages/my/my", 11 | "pages/favorite/favorite", 12 | "pages/history/history", 13 | "pages/shake/shake", 14 | "pages/gallery/gallery", 15 | "pages/setting/setting", 16 | "pages/skin/skin", 17 | "pages/personInfo/personInfo", 18 | "pages/editPersonInfo/editPersonInfo", 19 | "pages/systemInfo/systemInfo", 20 | "pages/about/about", 21 | "pages/location/location" 22 | ], 23 | "window": { 24 | "navigationBarBackgroundColor": "#47a86c", 25 | "navigationBarTextStyle": "white", 26 | "navigationBarTitleText": "电影推荐", 27 | "backgroundColor": "#fff", 28 | "backgroundTextStyle": "dark", 29 | "enablePullDownRefresh": true 30 | }, 31 | "tabBar": { 32 | "color": "#686868", 33 | "selectedColor": "#47a86c", 34 | "backgroundColor": "#ffffff", 35 | "borderStyle": "white", 36 | "list": [{ 37 | "pagePath": "pages/popular/popular", 38 | "iconPath": "dist/images/popular_icon.png", 39 | "selectedIconPath": "dist/images/popular_active_icon.png", 40 | "text": "热映" 41 | }, { 42 | "pagePath": "pages/coming/coming", 43 | "iconPath": "dist/images/coming_icon.png", 44 | "selectedIconPath": "dist/images/coming_active_icon.png", 45 | "text": "待映" 46 | }, 47 | { 48 | "pagePath": "pages/top/top", 49 | "iconPath": "dist/images/top_icon.png", 50 | "selectedIconPath": "dist/images/top_active_icon.png", 51 | "text": "口碑" 52 | }, 53 | { 54 | "pagePath": "pages/my/my", 55 | "iconPath": "dist/images/my_icon.png", 56 | "selectedIconPath": "dist/images/my_active_icon.png", 57 | "text": "我的" 58 | }] 59 | }, 60 | "networkTimeout": { 61 | "request": 10000, 62 | "downloadFile": 10000 63 | }, 64 | "debug": true 65 | } -------------------------------------------------------------------------------- /app.wxss: -------------------------------------------------------------------------------- 1 | page { 2 | display: block; 3 | background-color: #fff; 4 | font-family: "微软雅黑"; 5 | font-size: 28rpx; 6 | } 7 | 8 | .loading { 9 | position: fixed; 10 | left: 50%; 11 | top: 50%; 12 | width: 200rpx; 13 | height: 200rpx; 14 | margin: -100rpx 0 0 -100rpx; 15 | line-height: 300rpx; 16 | border-radius: 16rpx; 17 | background-image: url(); 18 | background-position: center 40rpx; 19 | background-repeat: no-repeat; 20 | background-size: 80rpx; 21 | background-color: rgba(0, 0, 0, 0.85); 22 | color: #fff; 23 | text-align: center; 24 | font-size: 24rpx; 25 | animation: bounceIn .3s; 26 | } -------------------------------------------------------------------------------- /comm/script/config.js: -------------------------------------------------------------------------------- 1 | /* 2 | 备注 3 | city: 城市(在程序载入时获取一次) 4 | count: 返回结果数量 5 | baiduAK: 百度地图AK 6 | apiList: api列表 7 | hotKeyword: 搜索页热门关键词关键词 8 | hotTag: 搜索页热门类型 9 | bannerList: 首页(热映页)轮播图列表列表 10 | skinList: “我的”页面背景列表 11 | shakeSound: 摇一摇音效地址(带staticUrl表示远程地址) 12 | shakeWelcomeImg: 摇一摇欢迎图片 13 | */ 14 | // 静态资源地址 15 | var staticUrl = 'https://static.sesine.com/wechat-weapp-movie' 16 | // api地址 17 | var apiUrl = 'https://sesine.com/mina/api' 18 | module.exports = { 19 | city: '', 20 | count: 20, 21 | baiduAK: 'Y1R5guY8Y2GNRdDpLz7SUeM3QgADAXec', 22 | apiList: { 23 | popular: apiUrl + '/movie/in_theaters', 24 | coming: apiUrl + '/movie/coming_soon', 25 | top: apiUrl + '/movie/top250', 26 | search: { 27 | byKeyword: apiUrl + '/movie/search?q=', 28 | byTag: apiUrl + '/movie/search?tag=' 29 | }, 30 | filmDetail: apiUrl + '/movie/subject/', 31 | personDetail: apiUrl + '/movie/celebrity/', 32 | baiduMap: 'https://api.map.baidu.com/geocoder/v2/' 33 | }, 34 | hotKeyword: ['功夫熊猫', '烈日灼心', '摆渡人', '长城', '我不是潘金莲', '这个杀手不太冷', '驴得水', '海贼王之黄金城', '西游伏妖片', '我在故宫修文物', '你的名字'], 35 | hotTag: ['动作', '喜剧', '爱情', '悬疑'], 36 | bannerList: [ 37 | {type:'film', id: '26683290', imgUrl: staticUrl + '/images/banner_1.jpg'}, 38 | {type:'film', id: '25793398', imgUrl: staticUrl + '/images/banner_2.jpg'}, 39 | {type:'film', id: '26630781', imgUrl: staticUrl + '/images/banner_3.jpg'}, 40 | {type:'film', id: '26415200', imgUrl: staticUrl + '/images/banner_4.jpg'}, 41 | {type:'film', id: '3025375', imgUrl: staticUrl + '/images/banner_5.jpg'} 42 | ], 43 | skinList: [ 44 | {title: '公路', imgUrl: staticUrl + '/images/user_bg_1.jpg'}, 45 | {title: '黑夜森林', imgUrl: staticUrl + '/images/user_bg_2.jpg'}, 46 | {title: '鱼与水', imgUrl: staticUrl + '/images/user_bg_3.jpg'}, 47 | {title: '山之剪影', imgUrl: staticUrl + '/images/user_bg_4.jpg'}, 48 | {title: '火山', imgUrl: staticUrl + '/images/user_bg_5.jpg'}, 49 | {title: '科技', imgUrl: staticUrl + '/images/user_bg_6.jpg'}, 50 | {title: '沙漠', imgUrl: staticUrl + '/images/user_bg_7.jpg'}, 51 | {title: '叶子', imgUrl: staticUrl + '/images/user_bg_8.jpg'}, 52 | {title: '早餐', imgUrl: staticUrl + '/images/user_bg_9.jpg'}, 53 | {title: '英伦骑车', imgUrl: staticUrl + '/images/user_bg_10.jpg'}, 54 | {title: '草原', imgUrl: staticUrl + '/images/user_bg_11.jpg'}, 55 | {title: '城市', imgUrl: staticUrl + '/images/user_bg_12.jpg'} 56 | ], 57 | shakeSound: { 58 | startUrl: staticUrl + '/sound/shake.mp3', 59 | start: '', 60 | completeUrl: staticUrl + '/sound/shakeComplete.wav', 61 | complete: '' 62 | }, 63 | shakeWelcomeImg: staticUrl + '/images/shake_welcome.png' 64 | } 65 | -------------------------------------------------------------------------------- /comm/script/fetch.js: -------------------------------------------------------------------------------- 1 | var config = require('./config.js') 2 | var message = require('../../component/message/message') 3 | 4 | // 获取电影列表 5 | function fetchFilms(url, start, count, cb, fail_cb) { 6 | var that = this 7 | message.hide.call(that) 8 | if (that.data.hasMore) { 9 | wx.request({ 10 | url: url, 11 | data: { 12 | city: config.city, 13 | start: start, 14 | count: config.count 15 | }, 16 | method: 'GET', 17 | header: { 18 | "Content-Type": "application/json,application/json" 19 | }, 20 | success: function(res){ 21 | if(res.data.subjects.length === 0){ 22 | that.setData({ 23 | hasMore: false, 24 | }) 25 | }else{ 26 | that.setData({ 27 | films: that.data.films.concat(res.data.subjects), 28 | start: that.data.start + res.data.subjects.length, 29 | showLoading: false 30 | }) 31 | console.log(that.data.start); 32 | } 33 | wx.stopPullDownRefresh() 34 | typeof cb == 'function' && cb(res.data) 35 | }, 36 | fail: function() { 37 | that.setData({ 38 | showLoading: false 39 | }) 40 | message.show.call(that,{ 41 | content: '网络开小差了', 42 | icon: 'offline', 43 | duration: 3000 44 | }) 45 | wx.stopPullDownRefresh() 46 | typeof fail_cb == 'function' && fail_cb() 47 | } 48 | }) 49 | } 50 | } 51 | 52 | // 获取电影详情 53 | function fetchFilmDetail(url, id, cb) { 54 | var that = this; 55 | message.hide.call(that) 56 | wx.request({ 57 | url: url + id, 58 | method: 'GET', 59 | header: { 60 | "Content-Type": "application/json,application/json" 61 | }, 62 | success: function(res){ 63 | that.setData({ 64 | filmDetail: res.data, 65 | showLoading: false, 66 | showContent: true 67 | }) 68 | wx.setNavigationBarTitle({ 69 | title: res.data.title 70 | }) 71 | wx.stopPullDownRefresh() 72 | typeof cb == 'function' && cb(res.data) 73 | }, 74 | fail: function() { 75 | that.setData({ 76 | showLoading: false 77 | }) 78 | message.show.call(that,{ 79 | content: '网络开小差了', 80 | icon: 'offline', 81 | duration: 3000 82 | }) 83 | } 84 | }) 85 | } 86 | 87 | // 获取人物详情 88 | function fetchPersonDetail(url, id, cb) { 89 | var that = this; 90 | message.hide.call(that) 91 | wx.request({ 92 | url: url + id, 93 | method: 'GET', 94 | header: { 95 | "Content-Type": "application/json,application/json" 96 | }, 97 | success: function(res){ 98 | that.setData({ 99 | personDetail: res.data, 100 | showLoading: false, 101 | showContent: true 102 | }) 103 | wx.setNavigationBarTitle({ 104 | title: res.data.name 105 | }) 106 | wx.stopPullDownRefresh() 107 | typeof cb == 'function' && cb(res.data) 108 | }, 109 | fail: function() { 110 | that.setData({ 111 | showLoading: false 112 | }) 113 | message.show.call(that,{ 114 | content: '网络开小差了', 115 | icon: 'offline', 116 | duration: 3000 117 | }) 118 | } 119 | }) 120 | } 121 | 122 | // 搜索(关键词或者类型) 123 | function search(url, keyword, start, count, cb){ 124 | var that = this 125 | message.hide.call(that) 126 | var url = decodeURIComponent(url) 127 | if (that.data.hasMore) { 128 | wx.request({ 129 | url: url + keyword, 130 | data: { 131 | start: start, 132 | count: count 133 | }, 134 | method: 'GET', 135 | header: { 136 | "Content-Type": "application/json,application/json" 137 | }, 138 | success: function(res){ 139 | if(res.data.subjects.length === 0){ 140 | that.setData({ 141 | hasMore: false, 142 | showLoading: false 143 | }) 144 | }else{ 145 | that.setData({ 146 | films: that.data.films.concat(res.data.subjects), 147 | start: that.data.start + res.data.subjects.length, 148 | showLoading: false 149 | }) 150 | wx.setNavigationBarTitle({ 151 | title: keyword 152 | }) 153 | } 154 | wx.stopPullDownRefresh() 155 | typeof cb == 'function' && cb(res.data) 156 | }, 157 | fail: function() { 158 | that.setData({ 159 | showLoading: false 160 | }) 161 | message.show.call(that,{ 162 | content: '网络开小差了', 163 | icon: 'offline', 164 | duration: 3000 165 | }) 166 | } 167 | }) 168 | } 169 | } 170 | module.exports = { 171 | fetchFilms: fetchFilms, 172 | fetchFilmDetail: fetchFilmDetail, 173 | fetchPersonDetail: fetchPersonDetail, 174 | search: search 175 | } -------------------------------------------------------------------------------- /comm/style/animation.wxss: -------------------------------------------------------------------------------- 1 | @keyframes fadeIn { 2 | from { 3 | opacity: 0; 4 | } 5 | to { 6 | opacity: 1; 7 | } 8 | } 9 | 10 | @keyframes fadeInDown { 11 | from { 12 | opacity: 0; 13 | transform: translate3d(0, -100%, 0); 14 | } 15 | to { 16 | opacity: 1; 17 | transform: none; 18 | } 19 | } 20 | 21 | @keyframes fadeInUp { 22 | from { 23 | opacity: 0; 24 | transform: translate3d(0, 100%, 0); 25 | } 26 | to { 27 | opacity: 1; 28 | transform: none; 29 | } 30 | } 31 | 32 | @keyframes bounceIn { 33 | from, 20%, 40%, 60%, 80%, to { 34 | animation-timing-function: cubic-bezier(0.215, 0.610, 0.355, 1.000); 35 | } 36 | 0% { 37 | opacity: 0; 38 | transform: scale3d(.3, .3, .3); 39 | } 40 | 20% { 41 | transform: scale3d(1.1, 1.1, 1.1); 42 | } 43 | 40% { 44 | transform: scale3d(.9, .9, .9); 45 | } 46 | 60% { 47 | opacity: 1; 48 | transform: scale3d(1.03, 1.03, 1.03); 49 | } 50 | 80% { 51 | transform: scale3d(.97, .97, .97); 52 | } 53 | to { 54 | opacity: 1; 55 | transform: scale3d(1, 1, 1); 56 | } 57 | } 58 | 59 | @keyframes fadeInRight { 60 | from { 61 | opacity: 0; 62 | -webkit-transform: translate3d(100%, 0, 0); 63 | transform: translate3d(100%, 0, 0); 64 | } 65 | 66 | to { 67 | opacity: 1; 68 | -webkit-transform: none; 69 | transform: none; 70 | } 71 | } 72 | 73 | 74 | @keyframes fadeInLeft { 75 | from { 76 | opacity: 0; 77 | -webkit-transform: translate3d(-100%, 0, 0); 78 | transform: translate3d(-100%, 0, 0); 79 | } 80 | 81 | to { 82 | opacity: 1; 83 | -webkit-transform: none; 84 | transform: none; 85 | } 86 | } -------------------------------------------------------------------------------- /comm/style/tab.wxss: -------------------------------------------------------------------------------- 1 | page{ 2 | background-color: #fcfcfc; 3 | } 4 | 5 | .tab-header { 6 | box-sizing: border-box; 7 | position: fixed; 8 | top: 0; 9 | left: 0; 10 | display: table; 11 | width: 750rpx; 12 | height: 80rpx; 13 | background-color: #fff; 14 | border-bottom: 1px solid #eee; 15 | } 16 | 17 | .tab-header-item { 18 | display: table-cell; 19 | height: 80rpx; 20 | line-height: 80rpx; 21 | text-align: center; 22 | } 23 | 24 | .tab-header-item:active { 25 | background-color: #f8f8f8; 26 | } 27 | 28 | .tab-header-active { 29 | border-bottom: 4rpx solid #47a86c; 30 | } 31 | 32 | .tab-content { 33 | margin-top: 80rpx; 34 | } 35 | 36 | .tab-content-item { 37 | box-sizing: border-box; 38 | padding: 30rpx; 39 | } -------------------------------------------------------------------------------- /component/filmList/filmList.wxml: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /component/filmList/filmList.wxss: -------------------------------------------------------------------------------- 1 | @import "../../comm/style/animation.wxss"; 2 | .film { 3 | box-sizing: border-box; 4 | width: 750rpx; 5 | padding: 10rpx 15rpx; 6 | display: flex; 7 | flex-wrap: wrap; 8 | flex-direction: row; 9 | justify-content: space-between; 10 | box-shadow: 0 0 40rpx #f4f4f4 inset; 11 | } 12 | 13 | .film-item { 14 | width: 350rpx; 15 | margin-bottom: 20rpx; 16 | border-radius: 10rpx; 17 | background-color: #fff; 18 | border: 1px solid #e4e4e4; 19 | box-shadow: 0 20rpx 40rpx #eee; 20 | overflow: hidden; 21 | animation: fadeIn 1s; 22 | } 23 | 24 | .film-cover, .film-cover-img { 25 | width: 350rpx; 26 | height: 508rpx; 27 | } 28 | 29 | .film-cover { 30 | position: relative; 31 | overflow: hidden; 32 | } 33 | 34 | .film-rating { 35 | box-sizing: border-box; 36 | position: absolute; 37 | bottom: 0; 38 | left: 0; 39 | width: 100%; 40 | height: 50rpx; 41 | padding-right: 20rpx; 42 | font-size: 12px; 43 | text-align: right; 44 | line-height: 50rpx; 45 | background-color: rgba(0, 0, 0, .65); 46 | color: #fff; 47 | } 48 | 49 | .file-intro { 50 | padding: 16rpx; 51 | margin-top: -8rpx; 52 | } 53 | 54 | .film-title { 55 | white-space: nowrap; 56 | text-overflow: ellipsis; 57 | overflow: hidden; 58 | } 59 | 60 | .film-tag { 61 | width: 100%; 62 | margin-top: 10rpx; 63 | display: flex; 64 | justify-content: flex-start; 65 | } 66 | 67 | .film-tag-item { 68 | padding: 4rpx 6rpx; 69 | margin-right: 10rpx; 70 | font-size: 24rpx; 71 | box-shadow: 0 0 0 1px #ccc; 72 | border-top: 1px solid #fff; 73 | border-radius: 10rpx; 74 | background-color: #fafafa; 75 | color: #666; 76 | } 77 | 78 | .loading-tip { 79 | width: 100%; 80 | height: 80rpx; 81 | line-height: 80rpx; 82 | text-align: center; 83 | color: #ccc; 84 | } -------------------------------------------------------------------------------- /component/list/list.wxml: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /component/list/list.wxss: -------------------------------------------------------------------------------- 1 | page { 2 | background: #f4f4f4; 3 | } 4 | 5 | .cells { 6 | position: relative; 7 | margin: 20rpx 0; 8 | background: #fff; 9 | overflow: hidden; 10 | } 11 | 12 | .cells::before, .cells::after { 13 | content: ''; 14 | display: block; 15 | position: absolute; 16 | z-index: 99; 17 | left: 0; 18 | width: 100%; 19 | height: 1px; 20 | transform-origin: 0 0; 21 | transform: scaleY(0.5); 22 | background-color: #e5e5e5; 23 | } 24 | 25 | .cells::before { 26 | top: 0; 27 | } 28 | 29 | .cells::after { 30 | bottom: 0; 31 | } 32 | 33 | .cell { 34 | display: flex; 35 | justify-content: space-between; 36 | position: relative; 37 | box-sizing: border-box; 38 | width: 750rpx; 39 | height: 100rpx; 40 | padding: 0 40rpx; 41 | font-size: 32rpx; 42 | line-height: 100rpx; 43 | } 44 | 45 | .cell:first-child::before { 46 | display: none; 47 | } 48 | 49 | .cell::before { 50 | content: ''; 51 | display: block; 52 | position: absolute; 53 | z-index: 99; 54 | top: 0; 55 | left: 40rpx; 56 | width: 100%; 57 | height: 1px; 58 | transform-origin: 0 0; 59 | transform: scaleY(0.5); 60 | background-color: #e5e5e5; 61 | } 62 | 63 | .cell:active { 64 | background-color: #fafafa; 65 | } 66 | 67 | .cell-ft{ 68 | color: #bbb; 69 | } 70 | 71 | .cell-access .cell-ft { 72 | padding-right: 13px; 73 | position: relative; 74 | } 75 | 76 | .cell-access .cell-ft::after { 77 | content: ''; 78 | display: inline-block; 79 | height: 6px; 80 | width: 6px; 81 | border-width: 2px 2px 0 0; 82 | border-color: #C8C8CD; 83 | border-style: solid; 84 | transform: matrix(0.71, 0.71, -0.71, 0.71, 0, 0); 85 | position: absolute; 86 | top: 50%; 87 | margin-top: -4px; 88 | right: 2px; 89 | } -------------------------------------------------------------------------------- /component/message/message.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | show: function(cfg) { 3 | var that = this 4 | that.setData({ 5 | message: { 6 | content: cfg.content, 7 | icon: cfg.icon, 8 | visiable: true 9 | } 10 | }) 11 | if (typeof cfg.duration !== 'undefined') { 12 | setTimeout(function(){ 13 | that.setData({ 14 | message: { 15 | visiable: false 16 | } 17 | }) 18 | }, cfg.duration) 19 | } 20 | }, 21 | hide: function() { 22 | var that = this 23 | that.setData({ 24 | message: { 25 | visiable: false 26 | } 27 | }) 28 | } 29 | } -------------------------------------------------------------------------------- /component/message/message.wxml: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /component/message/message.wxss: -------------------------------------------------------------------------------- 1 | @import "../../comm/style/animation.wxss"; 2 | .message-area { 3 | position: fixed; 4 | width: 100%; 5 | height: 100%; 6 | z-index: 99; 7 | } 8 | 9 | .message { 10 | box-sizing: border-box; 11 | position: fixed; 12 | z-index: 999; 13 | left: 50%; 14 | top: 50%; 15 | width: 250rpx; 16 | height: 250rpx; 17 | padding: 20rpx; 18 | margin-top: -125rpx; 19 | margin-left: -125rpx; 20 | display: flex; 21 | justify-content: center; 22 | flex-direction: column; 23 | align-items: center; 24 | border-radius: 16rpx; 25 | background-color: rgba(0, 0, 0, .75); 26 | color: #fff; 27 | animation: fadeIn .6s; 28 | } 29 | 30 | .message-icon { 31 | height: 100rpx; 32 | width: 100rpx; 33 | background-position: center; 34 | background-repeat: no-repeat; 35 | background-size: 100rpx; 36 | } 37 | 38 | .message-icon-offline { 39 | background-image: url(); 40 | } 41 | 42 | .message-icon-null { 43 | background-image: url(); 44 | } 45 | 46 | .message-content { 47 | margin-top: 15rpx; 48 | text-align: center; 49 | } -------------------------------------------------------------------------------- /component/nullTip/nullTip.wxml: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /component/nullTip/nullTip.wxss: -------------------------------------------------------------------------------- 1 | .null-tip { 2 | height: 400rpx; 3 | margin: 120rpx 0; 4 | background-image: url(); 5 | background-repeat: no-repeat; 6 | background-position: center; 7 | background-size: 140rpx 112rpx; 8 | line-height: 800rpx; 9 | text-align: center; 10 | color: #999; 11 | } 12 | 13 | .null-action { 14 | margin: 0 auto; 15 | width: 200rpx; 16 | height: 80rpx; 17 | line-height: 80rpx; 18 | text-align: center; 19 | border-radius: 8rpx; 20 | border: 1px solid #47a86c; 21 | color: #47a86c; 22 | } 23 | 24 | .null-action:active { 25 | background-color: #47a86c; 26 | color: #fff; 27 | } -------------------------------------------------------------------------------- /dist/images/coming_active_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/coming_active_icon.png -------------------------------------------------------------------------------- /dist/images/coming_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/coming_icon.png -------------------------------------------------------------------------------- /dist/images/green_tri.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/green_tri.png -------------------------------------------------------------------------------- /dist/images/my_active_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/my_active_icon.png -------------------------------------------------------------------------------- /dist/images/my_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/my_icon.png -------------------------------------------------------------------------------- /dist/images/popular_active_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/popular_active_icon.png -------------------------------------------------------------------------------- /dist/images/popular_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/popular_icon.png -------------------------------------------------------------------------------- /dist/images/search_active_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/search_active_icon.png -------------------------------------------------------------------------------- /dist/images/search_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/search_icon.png -------------------------------------------------------------------------------- /dist/images/top_active_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/top_active_icon.png -------------------------------------------------------------------------------- /dist/images/top_icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sesine/wechat-weapp-movie/913c27a10dc3dabaeaa718733c6c599277740206/dist/images/top_icon.png -------------------------------------------------------------------------------- /pages/about/about.js: -------------------------------------------------------------------------------- 1 | Page({ 2 | data:{ 3 | 4 | }, 5 | onLoad:function(options){ 6 | // 页面初始化 options为页面跳转所带来的参数 7 | } 8 | }) -------------------------------------------------------------------------------- /pages/about/about.json: -------------------------------------------------------------------------------- 1 | { 2 | "navigationBarTitleText": "关于", 3 | "enablePullDownRefresh": false 4 | } -------------------------------------------------------------------------------- /pages/about/about.wxml: -------------------------------------------------------------------------------- 1 | 2 | 电影推荐 3 | 4 | 5 | 后端支持 6 | 豆瓣API:https://developers.douban.com/ 7 | 百度地图API:http://lbsyun.baidu.com/ 8 | 9 | 10 | 作者 11 | sesine:https://github.com/yesifeng 12 | 13 | 14 | 此项目所使用到的API 15 | 16 | wx.request、wx.chooseImage、wx.previewImage、wx.playVoice、wx.saveFile、wx.setStorage、wx.getStorage、wx.clearStorage、wx.getStorageInfo、wx.getLocation、wx.getSystemInfo、wx.onAccelerometerChange、wx.showToast、wx.showModal、wx.showActionSheet、wx.setNavigationBarTitle、wx.navigateTo、wx.redirectTo、wx.login 17 | 18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /pages/about/about.wxss: -------------------------------------------------------------------------------- 1 | page { 2 | width: 100%; 3 | height: 100%; 4 | } 5 | 6 | view { 7 | width: 100%; 8 | } 9 | 10 | .weapp-about { 11 | width: 100%; 12 | height: 100%; 13 | display: flex; 14 | flex-wrap: wrap; 15 | justify-content: center; 16 | align-content: space-around; 17 | align-items: center; 18 | } 19 | 20 | .weapp-name { 21 | font-size: 60rpx; 22 | text-align: center; 23 | color: #555; 24 | } 25 | 26 | .webapp-intro { 27 | box-sizing: border-box; 28 | padding: 60rpx; 29 | height: 900rpx; 30 | } 31 | 32 | .webapp-intro-item { 33 | margin-bottom: 40rpx; 34 | } 35 | 36 | .webapp-intro-title { 37 | margin-bottom: 10rpx; 38 | font-size: 32rpx; 39 | color: #666; 40 | } 41 | 42 | .webapp-intro-content { 43 | color: #ccc; 44 | } -------------------------------------------------------------------------------- /pages/coming/coming.js: -------------------------------------------------------------------------------- 1 | var douban = require('../../comm/script/fetch') 2 | var config = require('../../comm/script/config') 3 | Page({ 4 | data: { 5 | films: [], 6 | hasMore: true, 7 | showLoading: true, 8 | start: 0 9 | }, 10 | onLoad: function() { 11 | var that = this 12 | douban.fetchFilms.call(that, config.apiList.coming, that.data.start) 13 | }, 14 | onPullDownRefresh: function() { 15 | var that = this 16 | that.setData({ 17 | films: [], 18 | hasMore: true, 19 | showLoading: true, 20 | start: 0 21 | }) 22 | douban.fetchFilms.call(that, config.apiList.coming, that.data.start) 23 | }, 24 | onReachBottom: function() { 25 | var that = this 26 | if (!that.data.showLoading) { 27 | douban.fetchFilms.call(that, config.apiList.coming, that.data.start) 28 | } 29 | }, 30 | viewFilmDetail: function(e) { 31 | var data = e.currentTarget.dataset; 32 | wx.navigateTo({ 33 | url: "../filmDetail/filmDetail?id=" + data.id 34 | }) 35 | }, 36 | viewFilmByTag: function(e) { 37 | var data = e.currentTarget.dataset 38 | var keyword = data.tag 39 | wx.navigateTo({ 40 | url: '../searchResult/searchResult?url=' + encodeURIComponent(config.apiList.search.byTag) + '&keyword=' + keyword 41 | }) 42 | } 43 | }) -------------------------------------------------------------------------------- /pages/coming/coming.json: -------------------------------------------------------------------------------- 1 | { 2 | "navigationBarTitleText": "即将上映" 3 | } -------------------------------------------------------------------------------- /pages/coming/coming.wxml: -------------------------------------------------------------------------------- 1 | 2 |