├── LICENSE ├── README.md ├── _config.yml ├── legoflow.json ├── package.json └── src ├── _index.html ├── assets ├── img │ ├── gift-lit │ │ ├── level1.png │ │ ├── level2.png │ │ ├── level3.png │ │ ├── level4.png │ │ └── level5.png │ └── like │ │ └── heart │ │ ├── audience │ │ ├── like01.png │ │ ├── like02.png │ │ ├── like03.png │ │ ├── like04.png │ │ ├── like05.png │ │ ├── like06.png │ │ └── like07.png │ │ └── user │ │ ├── like01.png │ │ ├── like02.png │ │ └── like03.png ├── js │ └── svga │ │ └── svga.min.js └── svga │ ├── angel.svga │ ├── girl.svga │ ├── kingset.svga │ ├── posche.svga │ ├── rainbowrose.svga │ ├── rose.svga │ └── sample.svga ├── css └── main.css ├── js ├── component │ ├── danmaku │ │ ├── index.js │ │ ├── style │ │ │ └── index.scss │ │ └── tpl │ │ │ └── index.js │ ├── download-app │ │ └── index.js │ ├── gift │ │ ├── gift-large │ │ │ ├── config.js │ │ │ ├── index.js │ │ │ └── style │ │ │ │ └── index.scss │ │ └── gift-lit │ │ │ ├── config.js │ │ │ ├── index.js │ │ │ ├── module │ │ │ └── combo.js │ │ │ ├── style │ │ │ └── index.scss │ │ │ └── tpl │ │ │ └── index.js │ ├── header │ │ ├── index.js │ │ ├── style │ │ │ └── index.scss │ │ └── tpl │ │ │ └── index.js │ ├── layout │ │ ├── index.js │ │ ├── style │ │ │ └── index.scss │ │ └── tpl │ │ │ └── index.js │ ├── like │ │ ├── config.js │ │ ├── index.js │ │ ├── module │ │ │ ├── anim.js │ │ │ ├── anim │ │ │ │ ├── anim.js │ │ │ │ ├── anim.like.js │ │ │ │ ├── random.js │ │ │ │ └── ticker.js │ │ │ └── frequent.js │ │ └── style │ │ │ └── index.scss │ ├── message │ │ ├── config.js │ │ ├── img │ │ │ ├── chat.png │ │ │ ├── gift.png │ │ │ ├── live.png │ │ │ └── share.png │ │ ├── index.js │ │ ├── style │ │ │ └── index.scss │ │ └── tpl │ │ │ └── index.js │ ├── player │ │ ├── config.js │ │ ├── img │ │ │ ├── icon-play.png │ │ │ └── loading.svg │ │ ├── index.js │ │ ├── style │ │ │ ├── live-end.scss │ │ │ └── live-in.scss │ │ └── tpl │ │ │ ├── live-end.js │ │ │ └── live-in.js │ ├── recommend-list │ │ ├── img │ │ │ └── ico-empty.png │ │ ├── index.js │ │ ├── style │ │ │ └── index.scss │ │ └── tpl │ │ │ └── index.js │ └── top-bar │ │ ├── img │ │ ├── app-logo.png │ │ └── icon-share.png │ │ ├── index.js │ │ ├── style │ │ └── index.scss │ │ └── tpl │ │ └── index.js ├── lib │ ├── anim │ │ ├── anim.js │ │ ├── anim.like.js │ │ ├── custom.js │ │ └── modules │ │ │ ├── random.js │ │ │ └── ticker.js │ ├── classlist-polyfill │ │ └── index.js │ ├── delegate │ │ └── index.js │ ├── es6-promise │ │ └── index.js │ ├── event │ │ └── index.js │ ├── fastclick │ │ └── index.js │ ├── jsonp │ │ └── index.js │ ├── lazy-load-img │ │ └── index.js │ ├── queue │ │ └── index.js │ ├── scrollTo │ │ └── index.js │ ├── socket.io │ │ └── index.js │ ├── sprite │ │ └── index.js │ ├── svga │ │ ├── svga-db.min.js │ │ ├── svga-worker.min.js │ │ └── svga.min.js │ └── swipe │ │ └── index.js ├── main.js └── public │ ├── api │ └── index.js │ ├── config │ └── index.js │ ├── index.js │ └── util │ ├── delegate-event.js │ ├── detect-android.js │ ├── get-url-param.js │ ├── index.js │ ├── preload-script.js │ ├── scroll-to.js │ ├── sleep.js │ └── sprite.js └── sass ├── _img.scss ├── base ├── _base.scss ├── _flexible.scss ├── _normalize.scss ├── _orientation.scss ├── _reset.scss └── _tool.scss └── main.scss /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2017 YY UED 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 | 为了提升品牌影响的影响力和推广产品,H5直播间已经成为了一款直播产品的必备要素。在网络直播盛行的这个风口浪尖,直播产品的模式越来越成熟,一个直播间所需要的元素和交互已经形成了稳定的模式。于是,你没理由完全要自己手动搭建一个H5直播间,因为H5Live提供了一套成熟稳定的H5直播间方案: 2 | 3 | 1. 组件化、配置化搭建,5分钟快速上手; 4 | 2. 涵盖公屏、大礼物、小礼物、弹幕、点赞等功能模块; 5 | 3. 配备高性能的礼物动画方案; 6 | 7 | ---- 8 | 9 | ![h5live](https://user-images.githubusercontent.com/1295348/31772948-c3eb733e-b4a6-11e7-83a8-2fbce55a62f8.png) 10 | 11 | 12 | ## 入门 13 | 14 | 为了加快开发效率,推荐你使用开箱即用的[LegoFlow](https://legoflow.com/)进行开发。当然,你也可以自己配置webpack进行开发,只需要支持ES6编译和SASS loader即可。 15 | 16 | ### 目录结构 17 | 18 | ```` 19 | js/ 20 | ├── component //组件 21 | |   ├── palyer 22 | |   ├── ... 23 | │   └── top-bar 24 | ├── lib //第三方库 25 | ├── public //公共,例如api,全局配置,工具等 26 | └── main.js //主入口 27 | ```` 28 | 29 | ## Issues 30 | 31 | 如有使用问题或建议,欢迎提issues 32 | 33 | -------------------------------------------------------------------------------- /_config.yml: -------------------------------------------------------------------------------- 1 | theme: jekyll-theme-cayman -------------------------------------------------------------------------------- /legoflow.json: -------------------------------------------------------------------------------- 1 | { 2 | "version": "1.0.43", 3 | "type": "mobile", 4 | "assets": "", 5 | "es6": true, 6 | "webp": false, 7 | "legolib": false, 8 | "https": false, 9 | "hot": false, 10 | "isUglifyJs": true, 11 | "isPackVueStyle": false, 12 | "vue@2.1": false, 13 | "watch": [], 14 | "publicPath": "", 15 | "packImgSize": "", 16 | "packCommon": "", 17 | "other": "", 18 | "dist": "", 19 | "output": "", 20 | "shell": "", 21 | "alias": { 22 | }, 23 | "global": { 24 | }, 25 | "externals": {}, 26 | "proxy": {}, 27 | "ts": {}, 28 | "cache": "timestamp", 29 | "user.dev.args": {}, 30 | "user.build.args": {}, 31 | "env": {}, 32 | "build.move": [], 33 | "build.env": "", 34 | "build.zip": "", 35 | "img.folder": "", 36 | "html.folder": "", 37 | "js.folder": "", 38 | "export.folder": "" 39 | } -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "H5Live", 3 | "version": "1.1.0", 4 | "repository": {}, 5 | "author": "UED.David", 6 | "license": "MIT" 7 | } -------------------------------------------------------------------------------- /src/_index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | H5Live 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | -------------------------------------------------------------------------------- /src/assets/img/gift-lit/level1.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/gift-lit/level1.png -------------------------------------------------------------------------------- /src/assets/img/gift-lit/level2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/gift-lit/level2.png -------------------------------------------------------------------------------- /src/assets/img/gift-lit/level3.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/gift-lit/level3.png -------------------------------------------------------------------------------- /src/assets/img/gift-lit/level4.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/gift-lit/level4.png -------------------------------------------------------------------------------- /src/assets/img/gift-lit/level5.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/gift-lit/level5.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/audience/like01.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/audience/like01.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/audience/like02.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/audience/like02.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/audience/like03.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/audience/like03.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/audience/like04.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/audience/like04.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/audience/like05.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/audience/like05.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/audience/like06.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/audience/like06.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/audience/like07.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/audience/like07.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/user/like01.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/user/like01.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/user/like02.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/user/like02.png -------------------------------------------------------------------------------- /src/assets/img/like/heart/user/like03.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/img/like/heart/user/like03.png -------------------------------------------------------------------------------- /src/assets/svga/angel.svga: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/svga/angel.svga -------------------------------------------------------------------------------- /src/assets/svga/girl.svga: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/svga/girl.svga -------------------------------------------------------------------------------- /src/assets/svga/kingset.svga: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/svga/kingset.svga -------------------------------------------------------------------------------- /src/assets/svga/posche.svga: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/svga/posche.svga -------------------------------------------------------------------------------- /src/assets/svga/rainbowrose.svga: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/svga/rainbowrose.svga -------------------------------------------------------------------------------- /src/assets/svga/rose.svga: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/svga/rose.svga -------------------------------------------------------------------------------- /src/assets/svga/sample.svga: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/assets/svga/sample.svga -------------------------------------------------------------------------------- /src/js/component/danmaku/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 弹幕模块 3 | * @author wilson 4 | */ 5 | 6 | import style from'./style/index.scss'; 7 | import createTpl from './tpl/index.js'; 8 | 9 | const danmakuElement = document.querySelector('[data-component="danmaku"]'); 10 | danmakuElement.innerHTML = ``; 11 | 12 | let main = { 13 | // 记录插入的位置 14 | slotArr: [0, 0], 15 | 16 | // 选择下一条弹幕插入的位置 17 | chooseMinPos: (slotArr) => { 18 | let minNum = Math.min.apply(Math, slotArr); 19 | let minPos = slotArr.indexOf(minNum); 20 | if(minPos < 0) return 0; 21 | return minPos; 22 | }, 23 | 24 | // 准备下一条弹幕 25 | followUp: (wrapDomTemp, minPos, danmaku) => { 26 | let slotArr = main.slotArr; 27 | let minPosTemp = slotArr[minPos]; 28 | if (minPosTemp === 0) { 29 | return; 30 | }else if (minPosTemp < 0) { 31 | minPosTemp = 0 32 | } else if(minPosTemp === 1) { 33 | minPosTemp -= 1; 34 | if(danmaku.paused) { 35 | danmaku.resume(); 36 | } 37 | } 38 | slotArr[minPos] = minPosTemp; 39 | main.slotArr = slotArr; 40 | }, 41 | 42 | // 创建弹幕 43 | create: (data) => { 44 | let danmakuItem = document.createElement('div'); 45 | danmakuItem.setAttribute('class', 'live-danmaku__item'); 46 | const tpl = createTpl(data); 47 | danmakuItem.innerHTML = tpl; 48 | danmakuElement.appendChild(danmakuItem); 49 | return danmakuItem; 50 | }, 51 | 52 | // 初始化队列 53 | initQueue: () => { 54 | const speed = 96; 55 | let wrapDom = danmakuElement; 56 | let danmaku = queue.queue(function (task, callback) { 57 | let danmakuItem; 58 | let danmakuItemWidth; 59 | let danmakuItemHeight; 60 | let translateWidth; 61 | let durationTime; 62 | let slotArr = main.slotArr; 63 | let minPos = main.chooseMinPos(slotArr); 64 | if(slotArr[minPos] === 0) { 65 | slotArr[minPos] += 1; 66 | danmakuItem = main.create(task); 67 | danmakuItemWidth = danmakuItem.offsetWidth; 68 | danmakuItemHeight = danmakuItem.offsetHeight; 69 | translateWidth = document.body.clientWidth + danmakuItemWidth + 8; 70 | durationTime = (translateWidth / speed); 71 | let topPos = minPos * (danmakuItemHeight + 4); 72 | danmakuItem.style.top = `${topPos}px`; 73 | danmakuItem.style.webkitTransform = `translate(${-translateWidth}px, 0)`; 74 | danmakuItem.style.transform = `translate(${-translateWidth}px, 0)`; 75 | danmakuItem.style.webkitTransitionDuration = `${durationTime}s`; 76 | danmakuItem.style.transitionDuration = `${durationTime}s`; 77 | } 78 | 79 | main.slotArr = slotArr; 80 | util.sleep(Math.floor(durationTime * 1000 / 1.3)).then(() => { 81 | main.followUp(wrapDom, minPos, danmaku) 82 | }) 83 | callback(danmakuItem, slotArr, durationTime); 84 | }, 2); 85 | 86 | return danmaku; 87 | }, 88 | 89 | // 添加队列 90 | addQueue: (danmakuItemObj, danmaku, isLast) => { 91 | if(isLast) { 92 | // 插到队列尾部 93 | danmaku.push(danmakuItemObj, function (danmakuItem, slotArr, durationTime) { 94 | if(slotArr[0] == 1 && slotArr[1] == 1) { 95 | if(!danmaku.paused) { 96 | danmaku.pause(); 97 | } 98 | } 99 | 100 | util.sleep(Math.ceil(durationTime * 1000)).then(() => { 101 | danmakuElement.removeChild(danmakuItem) 102 | }); 103 | }); 104 | } else { 105 | // 插到队列首部 106 | danmaku.unshift(danmakuItemObj, function (danmakuItem, slotArr, durationTime) { 107 | if(slotArr[0] == 1 && slotArr[1] == 1) { 108 | if(!danmaku.paused) { 109 | danmaku.pause(); 110 | } 111 | } 112 | 113 | util.sleep(Math.ceil(durationTime * 1000)).then(() => { 114 | danmakuElement.removeChild(danmakuItem) 115 | }); 116 | }); 117 | } 118 | } 119 | } 120 | 121 | export default { 122 | init(socket) { 123 | let danmaku = main.initQueue(); 124 | socket.ioSocket.on(socket.socketName, (data) => { 125 | let {nick, avatar, content} = data.data; 126 | let danmakuItemObj = { 127 | "nick": nick, 128 | "avatar": avatar, 129 | "content": content 130 | } 131 | main.addQueue(danmakuItemObj, danmaku, true); 132 | }); 133 | 134 | // 用户发弹幕优先显示 135 | // this.unShiftQueue = (danmakuItemObj) => { 136 | // main.addQueue(danmakuItemObj, danmaku, false); 137 | // } 138 | 139 | // 销毁弹幕 140 | this.destroy = () => { 141 | danmakuElement.style['display'] = 'none'; 142 | danmaku.tasks.length = 0; 143 | } 144 | } 145 | } 146 | -------------------------------------------------------------------------------- /src/js/component/danmaku/style/index.scss: -------------------------------------------------------------------------------- 1 | .live-danmaku { 2 | position: absolute; 3 | left: 0; 4 | top: 3rem; 5 | width: 100%; 6 | height: 1.2rem; 7 | word-break: normal; 8 | word-wrap: normal; 9 | white-space:nowrap; 10 | pointer-events: none; 11 | // HACK: camera pc mode 12 | .camera-pc ~ & { 13 | top: 1.3rem; 14 | } 15 | } 16 | 17 | // .live-danmaku__row { 18 | // position: absolute; 19 | // left: 100%; 20 | // height: 0.54rem; 21 | // word-break: normal; 22 | // word-wrap: normal; 23 | // white-space:nowrap; 24 | // &.ext-row1 { 25 | // bottom: 0; 26 | // } 27 | // &.ext-row2 { 28 | // top: 0; 29 | // } 30 | // } 31 | 32 | // .live-danmaku__rowInner { 33 | // height: 0.54rem; 34 | // } 35 | 36 | .live-danmaku__item { 37 | display: inline-block; 38 | position: absolute; 39 | height: 0.54rem; 40 | left: 100%; 41 | padding-right: 0.24rem; 42 | margin-right: 0.24rem; 43 | background-color: rgba(46, 46, 46, 0.6); 44 | border-radius: 0.26rem; 45 | vertical-align: top; 46 | font-size: 0; 47 | transition: transform 8s linear; 48 | } 49 | 50 | .live-danmaku__avatar { 51 | position: absolute; 52 | left: 0.03rem; 53 | top: 0.05rem; 54 | width: 0.44rem; 55 | height: 0.44rem; 56 | border-radius: 50%; 57 | } 58 | 59 | .live-danmaku__nick { 60 | display: inline-block; 61 | vertical-align: top; 62 | max-width: 3rem; 63 | height: 0.54rem; 64 | padding: 0 0.18rem 0 0.54rem; 65 | font-size: 0.32rem; 66 | color: #e0b66c; 67 | font-weight: 500; 68 | line-height: 0.56rem; 69 | white-space: nowrap; 70 | word-wrap: normal; 71 | overflow: hidden; 72 | text-overflow: ellipsis; 73 | text-align: left; 74 | } 75 | 76 | .live-danmaku__content { 77 | display: inline-block; 78 | vertical-align: top; 79 | font-size: 0.32rem; 80 | color: #fff; 81 | font-weight: 500; 82 | line-height: 0.56rem; 83 | } 84 | -------------------------------------------------------------------------------- /src/js/component/danmaku/tpl/index.js: -------------------------------------------------------------------------------- 1 | let renderTpl = (data)=>{ 2 | let tpl = ` 3 | ${data.nick} 4 |

${data.nick}

5 |

${data.content}

6 | `; 7 | return tpl; 8 | } 9 | 10 | export default renderTpl; 11 | -------------------------------------------------------------------------------- /src/js/component/download-app/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 下载逻辑 3 | * @description 点击下载按钮后的逻辑 4 | * @author wuweishuan 5 | */ 6 | /** 7 | * 状态类名 8 | * @type {String} 9 | */ 10 | const active = 'is-active'; 11 | /** 12 | * 按钮选择器 13 | * @type {String} 14 | */ 15 | const selector = '[data-role="download-app"]'; 16 | 17 | import delegate from 'delegate'; 18 | import downloadTips from 'component/download-tips/index.js'; // 下载弹窗 19 | 20 | document.addEventListener('touchstart', delegate(selector, function ( event ) { 21 | this.classList.add(active); 22 | }), false); 23 | 24 | document.addEventListener('touchend', delegate(selector, function ( event ) { 25 | this.classList.remove(active); 26 | }), false); 27 | 28 | document.addEventListener('touchcancel', delegate(selector, function ( event ) { 29 | this.classList.remove(active); 30 | }), false); 31 | 32 | document.addEventListener('click', delegate(selector, function ( event ) { 33 | if(window.CONFIG.downloadLink) { 34 | window.location.href = window.CONFIG.downloadLink; 35 | } else { 36 | // 如果没有下载链接就打开下载窗 37 | downloadTips.show(); 38 | } 39 | }), false); 40 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-large/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file 大礼物配置 3 | */ 4 | 5 | 6 | let config = { 7 | 'use-svga' : true, //大礼物动画使用SVGA(如果使用svga播放,svga文件需和页面同域) 8 | urlSVGA : '/assets/js/svga/svga.min.js', // svga库路径 9 | }; 10 | 11 | export default config; 12 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-large/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 大礼物播放器 3 | * @author lixinliang 4 | */ 5 | 6 | import Event from '../../../lib/event/index.js'; 7 | import style from'./style/index.scss'; 8 | import config from'./config.js'; 9 | 10 | const componentSVGAPlayer = document.querySelector('[data-component="SVGAPlayer"]'); 11 | 12 | const giftLargeEvent = new Event; 13 | 14 | let cache = {}; 15 | let giftFromWeb = []; 16 | let giftFromUser = []; 17 | let readyToPlay = Promise.resolve(); 18 | 19 | let onSendGift = function () {}; 20 | 21 | let loadSVGA; 22 | 23 | export default { 24 | init ( giftSocket, giftMapObj ) { 25 | loadSVGA = Promise.all([ 26 | util.preloadScript(config.urlSVGA), 27 | // util.preloadScript(config.urlSVGADB), 28 | ]); 29 | return new Promise(( resolve ) => { 30 | componentSVGAPlayer.innerHTML = ``; 31 | this.send = ( svga ) => { 32 | giftFromUser.push(svga); 33 | // console.log('push'); 34 | readyToPlay = readyToPlay.then(play); 35 | }; 36 | // TODO: dev 37 | giftSocket.ioSocket.on(giftSocket.socketName, ( responese ) => { 38 | let data = responese.data; 39 | let svga = giftMapObj[data.propId].svga; 40 | if (svga) { 41 | giftFromWeb.push(svga); 42 | readyToPlay = readyToPlay.then(play); 43 | onSendGift({ 44 | uid: '', 45 | nickname: data.nick, 46 | name: giftMapObj[data.propId].name, 47 | }); 48 | } 49 | }); 50 | // TODO: dev 51 | this.on = giftLargeEvent.on.bind(giftLargeEvent); 52 | resolve(); 53 | }).catch((err) => console.log(err)); 54 | }, 55 | onSendGift ( callback ) { 56 | onSendGift = callback; 57 | }, 58 | setPosition () { 59 | let offsetTop = componentSVGAPlayer.offsetTop; 60 | let style = componentSVGAPlayer.style; 61 | style['transform'] = style['-webkit-transform'] = `translateY(${ -offsetTop/2 }px)`; 62 | }, 63 | destroy () { 64 | return new Promise(( resolve ) => { 65 | componentSVGAPlayer.style.display = 'none'; 66 | resolve(); 67 | }).catch((err) => console.log(err)); 68 | }, 69 | }; 70 | 71 | function play () { 72 | return new Promise(( resolve ) => { 73 | let url = giftFromUser.shift() || giftFromWeb.shift(); 74 | if (url) { 75 | getSVGA(url).then(playSVGA).then(resolve); 76 | } else { 77 | resolve(); 78 | } 79 | }).catch((err) => console.log(err)); 80 | } 81 | 82 | function getSVGA ( url ) { 83 | return new Promise(( resolve ) => { 84 | loadSVGA.then(() => { 85 | if (cache[url]) { 86 | resolve(cache[url]); 87 | } else { 88 | let a = document.createElement('a'); 89 | a.href = url; 90 | let canvas = document.createElement('canvas'); 91 | canvas.id = `svga-player__${ Object.keys(cache).length }`; 92 | canvas.width = canvas.height = 500; 93 | canvas.style.width = `400px`; 94 | canvas.style.height = `400px`; 95 | canvas.clearsAfterStop = true; 96 | componentSVGAPlayer.appendChild(canvas); 97 | 98 | var player = new SVGA.Player(`#${ canvas.id }`); 99 | var parser = new SVGA.Parser(); 100 | player.loops = 1; 101 | parser.load(url, function(videoItem) { 102 | player.canvas = canvas; 103 | componentSVGAPlayer.removeChild(canvas); 104 | canvas.classList.add('is-active'); 105 | resolve({ 106 | player, 107 | videoItem, 108 | }); 109 | }) 110 | } 111 | }); 112 | }).catch((err) => console.log(err)); 113 | } 114 | 115 | function playSVGA ( options ) { 116 | let {player, videoItem} = options; 117 | return new Promise(( resolve ) => { 118 | componentSVGAPlayer.appendChild(player.canvas); 119 | player.onFinished(() => { 120 | componentSVGAPlayer.removeChild(player.canvas); 121 | resolve(); 122 | }); 123 | player.setVideoItem(videoItem); 124 | player.startAnimation(); 125 | }).catch((err) => console.log(err)); 126 | } 127 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-large/style/index.scss: -------------------------------------------------------------------------------- 1 | .live-SVGAPlayer { 2 | position: absolute; 3 | left: 0; 4 | box-sizing: content-box; 5 | height: 0; 6 | width: 100%; 7 | padding-top: 100%; 8 | // top: 50%; 9 | // transform: translateY(-50%); 10 | bottom: 3.3rem + 0.9; 11 | // bottom: 3.3rem + .9rem + .46rem; 12 | // margin-top: (-3.3rem - .9rem - .46rem)/2; 13 | // margin-top: -1.7rem; 14 | pointer-events: none; 15 | canvas { 16 | position: absolute; 17 | display: block; 18 | width: 100%; 19 | height: 100%; 20 | top: 0; 21 | left: 0; 22 | opacity: 0; 23 | &.is-active { 24 | opacity: 1; 25 | } 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-lit/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file 礼物配置 3 | */ 4 | 5 | 6 | let config = { 7 | // 小礼物连送时显示的等级图片 8 | levelMap: [ 9 | { 10 | "levelNum": 1, 11 | "levelImg": "https://legox.org/assets/project/h5live/img/gift-lit/level1.png" 12 | }, 13 | { 14 | "levelNum": 10, 15 | "levelImg": "https://legox.org/assets/project/h5live/img/gift-lit/level2.png" 16 | }, 17 | { 18 | "levelNum": 20, 19 | "levelImg": "https://legox.org/assets/project/h5live/img/gift-lit/level3.png" 20 | }, 21 | { 22 | "levelNum": 30, 23 | "levelImg": "https://legox.org/assets/project/h5live/img/gift-lit/level4.png" 24 | }, 25 | { 26 | "levelNum": 40, 27 | "levelImg": "https://legox.org/assets/project/h5live/img/gift-lit/level5.png" 28 | } 29 | ], 30 | 31 | // 礼物显示时间ms 32 | duration: 3000 33 | } 34 | 35 | export default config; 36 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-lit/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 礼物模块 3 | * @author wilson 4 | */ 5 | 6 | import sprite from '../../../lib/sprite/index.js'; // 礼物动画 7 | import combo from './module/combo.js'; // 礼物连击 8 | import style from'./style/index.scss'; 9 | import createTpl from './tpl/index.js'; 10 | 11 | const componentGift = document.querySelector('[data-component="gift"]'); 12 | componentGift.innerHTML = ``; 13 | 14 | let giftMap = null; 15 | 16 | let main = { 17 | // 创建dom 18 | create: (data, position) => { 19 | main.comboId[`line${position}`] = data.comboId; 20 | let giftItem = document.createElement('div'); 21 | giftItem.setAttribute('class', `live-gift__item ext-line${position} gift-ani`); 22 | const tpl = createTpl(data); 23 | giftItem.innerHTML = tpl; 24 | componentGift.appendChild(giftItem); 25 | }, 26 | 27 | // 礼物连送ID 28 | comboId: { 29 | 'line1': '', 30 | 'line2': '', 31 | }, 32 | 33 | // 礼物连送对象 34 | comboObj: { 35 | 'line1': null, 36 | 'line2': null, 37 | }, 38 | 39 | // 礼物位置数 40 | giftQueueNum: 0, 41 | 42 | // 礼物推送 43 | pushGift: (queue, position) => { 44 | // 创建礼物dom,并指定显示位置 45 | main.create(queue.task, position); 46 | let currentGiftElement = document.querySelector(`.ext-line${position}`); 47 | let giftImgDom = currentGiftElement.querySelector('.live-gift__giftImg'); // 礼物图片动画 48 | let aniDom = currentGiftElement.querySelector('.live-gift__giftNum'); 49 | let comboImgDom = aniDom.children[1]; 50 | let originalDom = aniDom.children[2]; 51 | 52 | sprite(giftImgDom, giftMap[queue.task.propId].spriteImage, giftMap[queue.task.propId].spriteDuration); 53 | giftImgDom.dataset.status = 'play'; 54 | 55 | main.comboObj[`line${position}`] = new combo({ 56 | endNum: queue.task.giftNum, 57 | aniDom: aniDom, 58 | comboImgDom: comboImgDom, 59 | numDom: originalDom, 60 | cb: function(aniDom) { 61 | currentGiftElement.classList.add('is-hide'); 62 | 63 | //重置 64 | main.comboObj[`line${position}`] = null; 65 | main.comboId[`line${position}`] = ''; 66 | 67 | setTimeout(() => { 68 | componentGift.removeChild(currentGiftElement); 69 | queue.callback(queue.task); 70 | }, 500); 71 | } 72 | }) 73 | }, 74 | 75 | // 初始化队列 76 | initQueue: () => { 77 | let timer = null; 78 | //礼物消息队列 79 | let giftLit = queue.queue(function (task, callback) { 80 | let extPos1Dom = document.querySelector('.ext-line1'); 81 | let extPos2Dom = document.querySelector('.ext-line2'); 82 | let currentQuene = { 83 | task, 84 | callback, 85 | }; 86 | 87 | main.giftQueueNum++; 88 | if(main.giftQueueNum <= 2) { 89 | if(extPos1Dom) { 90 | main.pushGift(currentQuene, 2); 91 | } else { 92 | main.pushGift(currentQuene, 1); 93 | } 94 | } else { 95 | // 超出队列显示,队列暂停 96 | giftLit.pause(); 97 | } 98 | }, 2); 99 | 100 | return giftLit; 101 | }, 102 | 103 | // 加入队列 104 | addQueue: (data, giftLitle, isLast) => { 105 | let {nick, avatar, propId, propCount, comboId} = data.data; 106 | let giftItemObj = { 107 | "propId": propId, 108 | "nick":nick, 109 | "avatar": avatar, 110 | "giftNum": propCount, 111 | "giftName": giftMap[propId].name, 112 | "giftImg": giftMap[propId].thumb, 113 | "comboId": comboId 114 | }; 115 | 116 | if(isLast) { 117 | // 插到队列尾部 118 | giftLitle.push(giftItemObj, main.queueCallback(giftLitle)); 119 | } else { 120 | // 插到队列首部 121 | giftLitle.unshift(giftItemObj, main.queueCallback(giftLitle)); 122 | } 123 | 124 | }, 125 | 126 | // 队列回调 127 | queueCallback: (giftLitle) => { 128 | if(giftLitle.tasks.length > 0) { 129 | let tempArr = giftLitle.tasks, 130 | tempArrLen = tempArr.length; 131 | let comboArr = []; 132 | let comboTotal = 0; 133 | let beLeftArr = []; 134 | let firstTaskComboid = tempArr[0].data.comboId; 135 | for(let i = 0; i < tempArrLen;i++) { 136 | if(tempArr[i].data.comboId == firstTaskComboid) { 137 | comboArr.push(tempArr[i]); 138 | comboTotal += tempArr[i].data.giftNum; 139 | } else { 140 | beLeftArr.push(tempArr[i]); 141 | } 142 | } 143 | 144 | if(comboArr.length > 1) { 145 | // 累积comboid相同的礼物 146 | comboArr[0].data.giftNum = comboTotal; 147 | comboArr.length = 1; 148 | } 149 | 150 | // 拼接队列 151 | giftLitle.tasks = comboArr.concat(beLeftArr, giftLitle.tasks.slice(tempArrLen)); 152 | } 153 | main.giftQueueNum--; 154 | // 恢复队列 155 | if(giftLitle.paused) { 156 | giftLitle.resume(); 157 | } 158 | }, 159 | 160 | // 礼物推送判断 161 | giftHandle: (data, giftLit, callback) => { 162 | if(giftMap[data.propId]) { 163 | let svga = giftMap[data.propId].svga; 164 | if(!svga) { 165 | let comboId = data.comboId; 166 | if(comboId == main.comboId['line1']) { 167 | main.comboObj['line1'].update(data.propCount); 168 | } else if(comboId == main.comboId['line2']) { 169 | main.comboObj['line2'].update(data.propCount); 170 | } else { 171 | callback(data, giftMap, giftLit); 172 | } 173 | } 174 | } else { 175 | console.log('没有匹配到礼物!') 176 | } 177 | } 178 | } 179 | 180 | let onSendGift = function () {}; 181 | 182 | export default { 183 | // 初始化 184 | init(socket, giftMapObj) { 185 | giftMap = giftMapObj; 186 | if (componentGift.matches('.camera-pc ~ [data-component="gift"]')) { 187 | // HACK: camera pc mode 188 | const videoHeight = document.querySelector('[data-component="player"]').dataset.videoHeight; 189 | componentGift.style['top'] = `${ videoHeight }px`; 190 | componentGift.style['bottom'] = `0`; 191 | componentGift.style['transform'] = componentGift.style['-webkit-transform'] = `translateY(-100%)`; 192 | // HACK: image padding 193 | componentGift.style['margin-top'] = `-10px`; 194 | } 195 | 196 | let giftLit = main.initQueue(); 197 | 198 | socket.ioSocket.on(socket.socketName, (data) => { 199 | main.giftHandle(data.data, giftLit, function() { 200 | let {uid, nick, propId} = data.data; 201 | onSendGift({ 202 | uid: uid, 203 | nickname: nick, 204 | name: giftMap[propId].name, 205 | }); 206 | main.addQueue(data, giftLit, true); 207 | }) 208 | }); 209 | 210 | // 礼物面板送礼优先显示 211 | // this.unShiftQueue = (data) => { 212 | // main.giftHandle(data, giftLit, function() { 213 | // main.addQueue(data, giftLit, false); 214 | // }) 215 | // }; 216 | }, 217 | destroy () { 218 | componentGift.style['display'] = 'none'; 219 | }, 220 | onSendGift ( callback ) { 221 | onSendGift = callback; 222 | }, 223 | } 224 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-lit/module/combo.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 礼物连击 3 | * @author wilson 4 | */ 5 | 6 | import config from '../config.js'; // 连击参数配置 7 | 8 | let levelMap = config.levelMap; // 礼物连击图片映射数据 9 | 10 | 11 | let combo = function(option) { 12 | this.curNum = 0; 13 | this.endNum = option.endNum; 14 | this.aniDom = option.aniDom; 15 | this.comboImgDom = option.comboImgDom; 16 | this.numDom = option.numDom; 17 | this.cb = option.cb; 18 | 19 | this.hasCritModel = false; // 是否启动了暴击模式 20 | this.start(); 21 | } 22 | 23 | combo.prototype = { 24 | start() { 25 | this.countUp(); 26 | }, 27 | 28 | // 查询连击图片更换 29 | changeComboImg(curNum) { 30 | let levelMapLen = levelMap.length; 31 | for(let i = 0;i < levelMapLen;i++) { 32 | if(curNum == levelMap[i].levelNum) { 33 | return i; 34 | } else if(curNum > levelMap[i].levelNum) { 35 | if((i + 1) < levelMapLen && curNum < levelMap[i + 1].levelNum) { 36 | return i; 37 | } else if((i + 1) == levelMapLen) { 38 | return i; 39 | } 40 | } 41 | } 42 | 43 | return (levelMapLen - 1); 44 | }, 45 | 46 | // 计数 47 | countUp() { 48 | if(this.curNum < this.endNum) { // 累加 49 | 50 | this.remainNum = this.endNum - this.curNum; 51 | if(this.remainNum > 10) { 52 | this.curNum += 10; 53 | this.hasCritModel = true; 54 | } else { 55 | if(this.hasCritModel) { 56 | this.curNum += this.remainNum; 57 | } else { 58 | this.curNum++; 59 | } 60 | } 61 | this.numDom.textContent = this.curNum; 62 | setTimeout(() => { 63 | this.aniDom.classList.add('is-ani'); 64 | if(this.aniDom.classList.contains('is-ani')) { 65 | // 判断是否需要更换连击图片 66 | let level = this.changeComboImg(this.curNum); 67 | this.comboImgDom.style.backgroundImage = `url(${levelMap[level].levelImg})`; 68 | } 69 | }, 0); 70 | 71 | setTimeout(() => { 72 | this.aniDom.classList.remove('is-ani'); 73 | this.comboImgDom.style.backgroundImage = ''; 74 | setTimeout(() => { 75 | this.countUp(); 76 | }, 100) 77 | }, 250) 78 | } else { // 连击结束 79 | let checkTimer = setInterval(() => { 80 | if(this.curNum < this.endNum) { 81 | this.countUp(); 82 | clearTimeout(overTimer); 83 | clearInterval(checkTimer); 84 | } 85 | }, 200); 86 | 87 | let overTimer = setTimeout(() => { 88 | this.cb(this.aniDom); 89 | clearInterval(checkTimer); 90 | }, config.duration); 91 | } 92 | }, 93 | 94 | // 更新 95 | update(offsetNum) { 96 | this.endNum += offsetNum; 97 | } 98 | } 99 | 100 | export default combo; 101 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-lit/style/index.scss: -------------------------------------------------------------------------------- 1 | .live-gift { 2 | position: absolute; 3 | left: 0.2rem; 4 | bottom: 4.3rem; 5 | height: 1.64rem; 6 | pointer-events: none; 7 | } 8 | 9 | .live-gift__item { 10 | position: relative; 11 | width: 3.84rem; 12 | height: 0.64rem; 13 | transform: translateX(-4.5rem); 14 | background-color: rgba(46, 46, 46, 0.6); 15 | border-radius: 0.32rem; 16 | transition: opacity 0.3s ease-out 0.2s; 17 | &.ext-line1 { 18 | position: absolute; 19 | left: 0; 20 | bottom: 0; 21 | } 22 | &.ext-line2 { 23 | position: absolute; 24 | left: 0; 25 | top: 0; 26 | } 27 | &.gift-ani { 28 | animation-name: giftAni; 29 | animation-duration: 0.3s; 30 | animation-timing-function: ease-out; 31 | animation-fill-mode: forwards; 32 | } 33 | 34 | &.is-hide { 35 | opacity: 0; 36 | } 37 | } 38 | 39 | @keyframes giftAni{ 40 | 0% { 41 | transform: translateX(-4.5rem); 42 | } 43 | 100% { 44 | transform: translateX(0); 45 | } 46 | } 47 | 48 | .live-gift__userImg { 49 | position: absolute; 50 | left: 0.02rem; 51 | top: 0.02rem; 52 | width: 0.6rem; 53 | height: 0.6rem; 54 | border-radius: 50%; 55 | } 56 | 57 | .live-gift__userName { 58 | width: 2.44rem; 59 | font-size: 0.24rem; 60 | line-height: 0.3rem; 61 | color: #e0b66c; 62 | font-weight: 500; 63 | padding: 0.05rem 0 0 0.76rem; 64 | white-space: nowrap; 65 | word-wrap: normal; 66 | overflow: hidden; 67 | text-overflow: ellipsis; 68 | text-align: left; 69 | } 70 | 71 | .live-gift__giftName { 72 | width: 2.44rem; 73 | font-size: 0.2rem; 74 | color: #fff; 75 | line-height: 0.26rem; 76 | padding-left: 0.76rem; 77 | white-space: nowrap; 78 | word-wrap: normal; 79 | overflow: hidden; 80 | text-overflow: ellipsis; 81 | text-align: left; 82 | font-weight: 400; 83 | span { 84 | margin-left: 0.08rem; 85 | color: #ffdda0; 86 | font-weight: 500; 87 | } 88 | } 89 | 90 | .live-gift__giftImg { 91 | position: absolute; 92 | left: 2.5rem; 93 | top: -0.15rem; 94 | width: 0.9rem; 95 | height: 0.9rem; 96 | } 97 | 98 | .live-gift__giftNum { 99 | position: absolute; 100 | left: 3.5rem; 101 | top: 0.09rem; 102 | width: 2rem; 103 | height: 0.44rem; 104 | line-height: 0.44rem; 105 | height: .44rem; 106 | font-size: 0.42rem; 107 | color: #ffc40f; 108 | font-weight: bold; 109 | -webkit-text-stroke: 0.02rem #fff; 110 | span { 111 | &:nth-of-type(1) { 112 | position: absolute; 113 | top: 0; 114 | left: 0; 115 | } 116 | &:nth-of-type(2) { 117 | position: absolute; 118 | top: -.65rem; 119 | left: -.4rem; 120 | width: 1.9rem; 121 | height: 1.86rem; 122 | text-align: center; 123 | line-height: 2.34rem; 124 | // background-image: url(http://page.yy.com/channel/assets/images/combo-effect/level1.png); 125 | background-size: 11.4rem 1.86rem; 126 | } 127 | &:nth-of-type(3) { 128 | position: absolute; 129 | top: -.9rem; 130 | left: .3rem; 131 | // width: 2.4rem; 132 | height: 2.34rem; 133 | // text-align: center; 134 | line-height: 2.34rem; 135 | } 136 | } 137 | &.is-ani { 138 | span { 139 | &:nth-of-type(1) { 140 | animation: txtani 0.25s ease-out 0s; 141 | animation-fill-mode: forwards; 142 | } 143 | &:nth-of-type(2) { 144 | animation: giftani 0.25s steps(6,end) 0s; 145 | } 146 | 147 | &:nth-of-type(3) { 148 | animation: txtani 0.25s ease-out 0s; 149 | animation-fill-mode: forwards; 150 | } 151 | } 152 | } 153 | } 154 | 155 | @keyframes giftani { 156 | from{ 157 | background-position: 0 0; 158 | } 159 | to { 160 | background-position: -11.4rem 0; 161 | } 162 | } 163 | 164 | @keyframes txtani { 165 | from{ 166 | opacity: 0; 167 | transform: scale(2.4); 168 | } 169 | to { 170 | opacity: 1; 171 | transform: scale(1); 172 | } 173 | } 174 | -------------------------------------------------------------------------------- /src/js/component/gift/gift-lit/tpl/index.js: -------------------------------------------------------------------------------- 1 | let renderTpl = (data)=>{ 2 | let tpl = ` 3 | 4 |

${data.nick}

5 |

送了${data.giftName}

6 |
7 |

x

8 | `; 9 | return tpl; 10 | } 11 | 12 | export default renderTpl; 13 | -------------------------------------------------------------------------------- /src/js/component/header/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 头部主播名片 3 | * @author David 4 | */ 5 | 6 | import Event from '../../lib/event/index.js'; 7 | import style from './style/index.scss'; 8 | import renderTpl from './tpl/index.js'; 9 | 10 | const componentHeader = document.querySelector('[data-component="header"]'); 11 | 12 | const headerEvent = new Event; 13 | 14 | export default { 15 | init ( response ) { 16 | let tpl = renderTpl(response.data); 17 | componentHeader.innerHTML = `${ tpl }`; 18 | document.addEventListener('click', window.util.delegate('[data-role="follow"]', function () { 19 | headerEvent.trigger('follow', ( handler ) => { 20 | handler.apply(this); 21 | }); 22 | }, 1), false); 23 | this.on = headerEvent.on.bind(headerEvent); 24 | }, 25 | destroy () { 26 | componentHeader.innerHTML = ''; 27 | }, 28 | }; 29 | -------------------------------------------------------------------------------- /src/js/component/header/style/index.scss: -------------------------------------------------------------------------------- 1 | @mixin ellipsis($width: auto) { 2 | @if($width) { 3 | max-width: $width; 4 | } 5 | white-space: nowrap; 6 | word-wrap: normal; 7 | overflow: hidden; 8 | -o-text-overflow: ellipsis; 9 | text-overflow: ellipsis; 10 | text-align: left; 11 | } 12 | 13 | .live-header { 14 | position: absolute; 15 | // top: .4rem + 1.1rem; 16 | top: .15rem; 17 | left: 0; 18 | width: 100%; 19 | padding: 0 .2rem; 20 | } 21 | .live-hostInfo { 22 | float: left; 23 | position: relative; 24 | padding: .09rem .1rem; 25 | // padding-right: .06rem + .72rem + .14rem; 26 | left: .08rem; 27 | // height: 0.62rem; 28 | // padding-right: .92rem; 29 | // padding-left: .1rem; 30 | border-radius: .1rem; 31 | background: rgba(0, 0, 0, 0.2); 32 | color: #fff; 33 | font-size: 0; 34 | } 35 | .live-hostInfo__avatar { 36 | display: inline-block; 37 | vertical-align: top; 38 | width: .56rem; 39 | height: .56rem; 40 | border-radius: 50%; 41 | background-size: 100% 100%; 42 | background-repeat: no-repeat; 43 | position: relative; 44 | // top: 50%; 45 | // transform: translateY(-50%); 46 | } 47 | .live-hostInfo__cnt { 48 | margin-left: .1rem; 49 | display: inline-block; 50 | vertical-align: top; 51 | // font-size: .20rem; 52 | font-size: .24rem; 53 | line-height: .28rem; 54 | position: relative; 55 | // top: 50%; 56 | // transform: translateY(-50%); 57 | // margin-top: .04rem; 58 | } 59 | .live-hostInfo__nick { 60 | @include ellipsis(1.5rem); 61 | font-weight: 700; 62 | } 63 | .live-hostInfo__online { 64 | @include ellipsis(1.5rem); 65 | } 66 | .live-hostInfo__follow { 67 | // position: absolute; 68 | // top: 50%; 69 | // right: .06rem; 70 | // transform: translate(0, -50%); 71 | display: inline-block; 72 | margin-left: .08rem; 73 | width: .72rem; 74 | height: .56rem; 75 | line-height: .56rem; 76 | text-align: center; 77 | border-radius: .1rem; 78 | font-size: .24rem; 79 | background-color: rgba(256, 256, 256, 1); 80 | color: #2e2e2e; 81 | font-weight: 500; 82 | } 83 | .live-uid { 84 | position: absolute; 85 | // top: .05rem; 86 | top: .03rem; 87 | right: .4rem; 88 | right: .3rem; 89 | font-size: .24rem; 90 | color: #fff; 91 | // text-shadow: 0 1px 1px rgba(0, 0, 0, 0.2); 92 | text-shadow: 0 0 1px rgba(0, 0, 0, 1); 93 | } 94 | -------------------------------------------------------------------------------- /src/js/component/header/tpl/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file header主播信息 3 | */ 4 | 5 | let renderTpl = (data)=>{ 6 | let tpl = ` 7 |
8 |
9 |
10 |

${data.anchorNick}

11 |

${data.users}人

12 |
13 |
关注
14 |
15 |
主播ID: ${data.anchorId}
16 | `; 17 | return tpl; 18 | } 19 | 20 | export default renderTpl; 21 | 22 | 23 | 24 | -------------------------------------------------------------------------------- /src/js/component/layout/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component UI布局 3 | */ 4 | 5 | import renderTpl from './tpl/index.js'; 6 | import style from './style/index.scss'; 7 | 8 | let wrapSelector = window.wrapSelector?window.wrapSelector:'body'; 9 | document.querySelector(wrapSelector).innerHTML = `${renderTpl()}`; 10 | -------------------------------------------------------------------------------- /src/js/component/layout/style/index.scss: -------------------------------------------------------------------------------- 1 | html, body { 2 | min-width: 320px; 3 | max-width: 750px; 4 | } 5 | 6 | .live-content { 7 | width: 100%; 8 | position: relative; 9 | // padding-top: 1.1rem; 10 | background: #2e2e2e; 11 | overflow: hidden; 12 | user-select: none; 13 | } 14 | .live-player__uid, 15 | .live-player__nick, 16 | .live-player__avatar, 17 | .live-player__status, 18 | .live-player__playBtn { 19 | z-index: 2;// 覆盖遮罩或者边框等UI元素 20 | } 21 | .live-chatBar__switch { 22 | span { 23 | z-index: 2;// 覆盖遮罩或者边框等UI元素 24 | } 25 | } 26 | .live-giftPanel__cover, 27 | .live-giftPanel__border { 28 | z-index: 2;// 覆盖遮罩或者边框等UI元素 29 | } 30 | .live-giftPanel__present { 31 | z-index: 6;// 涉及动画的层级 32 | } 33 | .live-giftPanel__comboWrap { 34 | z-index: 5;// 涉及动画的层级 35 | } 36 | .live-giftPanel__comboButton { 37 | z-index: 7;// 涉及动画的层级 38 | } 39 | .live-topBar { 40 | z-index: 9;// 保持层级置顶 41 | } 42 | .live-downloadTips { 43 | z-index: 9;// 保持层级置顶 44 | } 45 | -------------------------------------------------------------------------------- /src/js/component/layout/tpl/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file UI布局 3 | */ 4 | 5 | let renderTpl = (data)=>{ 6 | let tpl = ` 7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 | `; 20 | return tpl; 21 | } 22 | 23 | export default renderTpl; 24 | -------------------------------------------------------------------------------- /src/js/component/like/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file 点赞配置 3 | */ 4 | 5 | let config = { 6 | // 点赞图片基础路径 7 | likeImageBaseUrl: '', 8 | 9 | // 点赞区域宽度 10 | wrapperWidth: 60, 11 | 12 | // 点赞区域高度 13 | wrapperHeight: 280, 14 | 15 | // 点赞显示的图片 16 | likeImg: { 17 | user01 : '/assets/img/like/heart/user/like01.png', 18 | user02 : '/assets/img/like/heart/user/like02.png', 19 | user03 : '/assets/img/like/heart/user/like03.png', 20 | audience01 : '/assets/img/like/heart/audience/like01.png', 21 | audience02 : '/assets/img/like/heart/audience/like02.png', 22 | audience03 : '/assets/img/like/heart/audience/like03.png', 23 | audience04 : '/assets/img/like/heart/audience/like04.png', 24 | audience05 : '/assets/img/like/heart/audience/like05.png', 25 | audience06 : '/assets/img/like/heart/audience/like06.png', 26 | audience07 : '/assets/img/like/heart/audience/like07.png', 27 | }, 28 | 29 | // 自动显示点赞的图片数量 30 | likeImgUserNum: 3, 31 | 32 | // 用户点击显示点赞的图片数量 33 | likeImgAudienceNum: 7, 34 | } 35 | 36 | export default config; 37 | -------------------------------------------------------------------------------- /src/js/component/like/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 点赞 3 | * @author lixinliang 4 | */ 5 | 6 | import style from './style/index.scss'; 7 | import anim from './module/anim.js'; //点赞动画 8 | import frequent from './module/frequent.js'; 9 | 10 | const componentLike = document.querySelector('[data-component="like"]'); 11 | 12 | export default { 13 | init () { 14 | componentLike.innerHTML = ``; 15 | componentLike.appendChild(anim.canvas); 16 | // 根据在线人数模拟点赞量 17 | let online = document.querySelector('.live-hostInfo__online'); 18 | this.autoLike = true; 19 | let autoLike = () => { 20 | let viewer = (parseInt(online ? online.innerText : 0) + '').length; 21 | let delay = 50 + 2520 / (viewer > 6 ? 6 : viewer); 22 | frequent(() => { 23 | if (this.autoLike) { 24 | anim.show('audience'); 25 | autoLike(); 26 | } 27 | }, delay) 28 | }; 29 | autoLike(); 30 | this.delegate = delegate('.live-content', function ( event ) { 31 | anim.show('user'); 32 | }); 33 | document.addEventListener('click', this.delegate, false); 34 | }, 35 | destroy () { 36 | this.autoLike = false; 37 | document.removeEventListener('click', this.delegate, false); 38 | componentLike.innerHTML = ''; 39 | }, 40 | }; 41 | -------------------------------------------------------------------------------- /src/js/component/like/module/anim.js: -------------------------------------------------------------------------------- 1 | import Anim from './anim/anim.js'; // 创建一个canvas 2 | import Like from './anim/anim.like.js'; // 点赞效果的逻辑代码 3 | import config from '../config.js'; 4 | 5 | // 把点赞引擎添加 animjs 上 6 | Anim.setup('like', Like); 7 | // 创建点赞实例 8 | let anim = new Anim('like'); 9 | // 设置尺寸 10 | anim.width(60).height(280); 11 | // 设置资源集合 与 映射分组 12 | anim.mainfest(mainfest()).group(mainfest.group); 13 | 14 | export default anim; 15 | 16 | function mainfest () { 17 | let result = {}; 18 | let user = append.call(result, 'user', config.likeImgUserNum); 19 | let audience = append.call(result, 'audience', config.likeImgAudienceNum); 20 | mainfest.group = { user, audience }; 21 | return result; 22 | } 23 | 24 | function append ( directory, length ) { 25 | return Array.apply(null, { length }).map((value, index) => fixed(index + 1 + '')).map((id) => { 26 | let key = directory + id; 27 | this[key] = config.likeImg[key]; 28 | return key; 29 | }); 30 | } 31 | 32 | function fixed ( value ) { 33 | return value.length == 1 ? '0' + value : value; 34 | } 35 | -------------------------------------------------------------------------------- /src/js/component/like/module/anim/anim.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component animjs 包含以下功能:点赞图片预加载、点赞图片分组、定义点赞的canvas样式 3 | * @author lixinliang 4 | */ 5 | 6 | import Ticker from './ticker.js'; 7 | 8 | const DPR = window.devicePixelRatio || 1; 9 | 10 | let closure = {}; 11 | let storage = []; 12 | 13 | // animjs 不涉及动画的实现 调用show方法时 把图片集合交给 动画引擎处理 14 | class Anim { 15 | 16 | /** 17 | * new的时候决定该实例使用那种动画引擎 18 | * @type {String} name 动画引擎 19 | */ 20 | constructor ( name = '' ) { 21 | let canvas = document.createElement('canvas'); 22 | let context = canvas.getContext('2d'); 23 | this.canvas = canvas; 24 | this.context = context; 25 | this.id = storage.length; 26 | this.name = name; 27 | let imageSet = {}; 28 | let imageMap = {}; 29 | storage.push({ imageSet, imageMap }); 30 | let ticker = new Ticker(); 31 | ticker.on(() => { 32 | context.clearRect(0, 0, canvas.width, canvas.height); 33 | }); 34 | this.ticker = ticker; 35 | this.dpr = DPR; 36 | if (DPR > 1) { 37 | this.canvas.style['transform'] = this.canvas.style['-webkit-transform'] = `scale(${ 1/DPR })`; 38 | } 39 | } 40 | 41 | /** 42 | * 传入一组图片集合 进行预加载 43 | * @param {Object} imageSet 资源集合 { 图片id : 图片url } 44 | * @return {Anim} this anim 45 | */ 46 | mainfest ( imageSet ) { 47 | let mySet = storage[this.id].imageSet; 48 | for (let key in imageSet) { 49 | let value = imageSet[key]; 50 | let image = { 51 | url : value, 52 | status : false 53 | }; 54 | image.img = loadImage(image.url, () => { 55 | image.status = true; 56 | }); 57 | mySet[key] = image; 58 | } 59 | return this 60 | } 61 | 62 | /** 63 | * 传入图片分组的映射关系 64 | * @param {Object} imageMap 资源分组映射 { 分组id : [图片id,...] } 65 | * @return {Anim} this anim 66 | */ 67 | group ( imageMap ) { 68 | let myMap = storage[this.id].imageMap; 69 | for (let key in imageMap) { 70 | myMap[key] = imageMap[key].slice(); 71 | } 72 | return this 73 | } 74 | 75 | /** 76 | * 随机播放动画 77 | * @param {String} groupName 资源分组名 78 | * @return {Anim} this anim 79 | */ 80 | show ( groupName ) { 81 | let { imageSet, imageMap } = storage[this.id]; 82 | let images = []; 83 | imageMap[groupName].forEach((imageName) => { 84 | let image = imageSet[imageName]; 85 | if (image.status) { 86 | images.push(image.img); 87 | } 88 | }); 89 | if (this.name) { 90 | return new closure[this.name](this, images); 91 | } 92 | return this 93 | } 94 | 95 | /** 96 | * 设置宽度 97 | * @param {Number} canvasWidth 画布宽度 98 | * @return {Anim} this anim 99 | */ 100 | width ( canvasWidth ) { 101 | let result = correct(canvasWidth); 102 | if (result) { 103 | let { num, str } = result; 104 | this.canvas.width = num; 105 | this.canvas.style.width = str; 106 | } 107 | return this 108 | } 109 | 110 | /** 111 | * 设置高度 112 | * @param {Number} canvasHeight 画布高度 113 | * @return {Anim} this anim 114 | */ 115 | height ( canvasHeight ) { 116 | 117 | let result = correct(canvasHeight); 118 | if (result) { 119 | let { num, str } = result; 120 | this.canvas.height = num; 121 | this.canvas.style.height = str; 122 | } 123 | 124 | return this 125 | } 126 | } 127 | 128 | export default Anim; 129 | 130 | /** 131 | * 加载图片 132 | * @param {String} url 图片地址 133 | * @param {Function} cb 加载回调 134 | * @return {HTMLImageElement} img 图片标签 135 | */ 136 | function loadImage ( url, cb ) { 137 | let img = new Image; 138 | img.onload = cb; 139 | img.src = url; 140 | return img; 141 | }; 142 | 143 | /** 144 | * 对canvas的宽高修正 在hidpi屏上 尺寸需要加倍 145 | * @param {Number} value 尺寸 146 | * @return {Number} value 尺寸 147 | */ 148 | function correct ( value ) { 149 | value *= DPR; 150 | let num; 151 | let str; 152 | if (typeof value == 'number') { 153 | num = value; 154 | str = value + 'px'; 155 | } else if (typeof value == 'string') { 156 | str = value; 157 | num = parseInt(value); 158 | } 159 | if (num && str) { 160 | return { num, str } 161 | } else { 162 | return null 163 | } 164 | }; 165 | 166 | /** 167 | * 定义动画引擎 168 | * @param {String} name 动画引擎名 169 | * @param {Class} plugin 动画引擎 170 | */ 171 | Anim.setup = function ( name, plugin ) { 172 | closure[name] = plugin; 173 | }; 174 | -------------------------------------------------------------------------------- /src/js/component/like/module/anim/anim.like.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component anim.like 基于 animjs 的点赞动画引擎 3 | * @author lixinliang 4 | */ 5 | 6 | import { 7 | getRandomByMedian, 8 | getRandomBetweenInt, 9 | getRandomWithBinomial, 10 | } from './random.js'; 11 | 12 | // 点赞动画引擎 13 | class Like { 14 | /** 15 | * 调用 new Like 创建一个赞的动画 16 | * @param {Anim} anim Anim实例 17 | * @param {Array} images 素材集合 18 | * @return {Like} like 点赞实例 19 | */ 20 | constructor ( anim, images ) { 21 | // 从图片集合 随机抽取一张 22 | let image = images[getRandomBetweenInt(0, images.length - 1)]; 23 | 24 | this.image = image; 25 | this.canvas = anim.canvas; 26 | 27 | let DPR = this.dpr = anim.dpr; 28 | 29 | let imageWidth = image.width; 30 | let imageHeight = image.height; 31 | 32 | // @return 当前动画进度 0 ~ 100 33 | let getImageTimeline = this.createImageTimeline(); 34 | // @return 旋转后的图片 canvas 35 | let getImageData = this.createImageData(); 36 | // @return 获取当前缩放曲线 number 37 | let getImageScaleSize = this.createImageScaleSize(); 38 | // @return 获取当前透明度曲线 number 39 | let getImageOpacityValue = this.createImageOpacityValue(); 40 | // @return 获取当前位移曲线 { x, y } 41 | let getImagePathPosition = this.createImagePathPosition(); 42 | 43 | let handler = () => { 44 | let percentage = getImageTimeline(); 45 | let { imageCanvas, imageContext } = getImageData(); 46 | let scaleSize = getImageScaleSize(percentage); 47 | let globalAlpha = getImageOpacityValue(percentage); 48 | let { x, y } = getImagePathPosition(percentage); 49 | 50 | // 最后渲染的时候 再把 dpr 计算进去 51 | if (scaleSize == 1 && globalAlpha == 1) { 52 | anim.context.drawImage(imageCanvas, x * DPR, y * DPR, imageWidth * DPR, imageHeight * DPR); 53 | } else { 54 | if (globalAlpha == 1) { 55 | let scaleWidth = imageWidth * scaleSize; 56 | let deltaWidth = (imageWidth - scaleWidth) / 2; 57 | let scaleHeight = imageHeight * scaleSize; 58 | let deltaHeight = (imageHeight - scaleHeight) / 2; 59 | anim.context.drawImage(imageCanvas, (x + deltaWidth) * DPR, (y + deltaHeight) * DPR, scaleWidth * DPR, scaleHeight * DPR); 60 | } else { 61 | anim.context.globalAlpha = globalAlpha; 62 | anim.context.drawImage(imageCanvas, x * DPR, y * DPR, imageWidth * DPR, imageHeight * DPR); 63 | anim.context.globalAlpha = 1; 64 | } 65 | } 66 | if (percentage == 100) { 67 | getImageTimeline = null; 68 | getImageData = null; 69 | getImageScaleSize = null; 70 | getImageOpacityValue = null; 71 | getImagePathPosition = null; 72 | anim.ticker.off(handler); 73 | } 74 | }; 75 | anim.ticker.on(handler); 76 | } 77 | 78 | /** 79 | * 创建每一个动画的独立时间线 80 | * @return {Function} getImageTimeline 获得该动画当前时间线的函数 81 | */ 82 | createImageTimeline () { 83 | // 动画总时基数 3000ms 浮动系数 20% 84 | let originDuration = 3000; 85 | let range = 0.2; 86 | 87 | let duration = getRandomByMedian(originDuration, range); 88 | let times = Math.round(duration / 16.7); 89 | let now = 0; 90 | return () => { 91 | now++; 92 | let percentage = parseInt(now * 1000 / times) / 10; 93 | return percentage > 100 ? 100 : percentage 94 | } 95 | } 96 | 97 | /** 98 | * 创建每一个动画的独立形态 99 | * @return {Function} getImageData 获得该动画形态的对象的函数 100 | */ 101 | createImageData () { 102 | // 图像旋转幅度 -30 ~ 30deg 103 | let minRotate = -30; 104 | let maxRotate = 30; 105 | // 缩放系数的二项分布 70 ~ 120 出现95的概率最高 106 | let minScale = 70; 107 | let maxScale = 120; 108 | // 缩放最大值为105 超出105的概率以105为对称轴 添加到 90 ~ 105 的区间 109 | let maxLimitScale = 105; 110 | 111 | let image = this.image; 112 | let { width, height } = image; 113 | let rotate = getRandomBetweenInt(minRotate, maxRotate); 114 | let scale = getRandomWithBinomial(minScale, maxScale); 115 | if (scale > maxLimitScale) { 116 | scale -= (maxScale - maxLimitScale); 117 | } 118 | scale = scale / 100; 119 | 120 | let imageCanvas = document.createElement('canvas'); 121 | let imageContext = imageCanvas.getContext('2d'); 122 | 123 | // 使用离屏canvas保存旋转后的图片并多次复用 为了避免旋转后图片被切割 画布的尺寸是图片的2倍 124 | imageCanvas.width = width * 2; 125 | imageCanvas.height = height * 2; 126 | 127 | imageContext.translate(width, height); 128 | imageContext.rotate(rotate * Math.PI / 180); 129 | imageContext.scale(scale, scale); 130 | imageContext.translate(-width, -height); 131 | 132 | imageContext.drawImage(image, width / 2, height / 2, width, height); 133 | 134 | return () => ({ imageCanvas, imageContext }) 135 | } 136 | 137 | /** 138 | * 创建每一个动画的独立放大曲线 139 | * @return {Function} getImageScaleSize 获得该动画当前尺寸的函数 140 | */ 141 | createImageScaleSize () { 142 | // 完成缩放动画的百分比阶段 9 ~ 11 143 | let minAppearStage = 9; 144 | let maxAppearStage = 11; 145 | 146 | let appearStage = getRandomBetweenInt(minAppearStage, maxAppearStage); 147 | return ( percentage ) => { 148 | return percentage >= appearStage ? 1 : ((parseInt(100 * percentage / appearStage) / 100) || 0.1) 149 | } 150 | } 151 | 152 | /** 153 | * 创建每一个动画的独立透明曲线 154 | * @return {Function} getImageOpacityValue 获得该动画当前透明度的函数 155 | */ 156 | 157 | createImageOpacityValue () { 158 | // 开始淡出动画的百分比阶段 82 ~ 86 159 | let minDisappearStage = 82; 160 | let maxDisappearStage = 86; 161 | 162 | let disappearStage = getRandomBetweenInt(minDisappearStage, maxDisappearStage); 163 | let area = 100 - disappearStage; 164 | return ( percentage ) => { 165 | if (percentage < disappearStage) { 166 | return 1; 167 | } else { 168 | let opacity = ((percentage - disappearStage) / area).toFixed(2) - 0; 169 | return 1 - opacity; 170 | } 171 | } 172 | } 173 | 174 | /** 175 | * 创建每一个动画的独立位移曲线 176 | * @return {Function} getImagePathPosition 获得该动画当前位移值的函数 177 | */ 178 | createImagePathPosition () { 179 | // 以下计算均以 dpr 为 1作为参考值 180 | // 动画初始位置的横向定位的浮动系数 10% 181 | let startX = 10; 182 | // 动画纵向位移距离delta y是总距离的 95% 浮动系数 5% 183 | let deltaY = 0.95; 184 | let deltaYRange = 1 - deltaY; 185 | // 动画横向位移距离符合正弦曲线 正弦波峰最小值10% 最大值15% 186 | let minCrest = 10; 187 | let maxCrest = 15; 188 | // 正弦波长系数 最小值16 最大值32 189 | let minWavelength = 16; 190 | let maxWavelength = 32; 191 | 192 | let DPR = this.dpr; 193 | // @2x 图片要先砍半 194 | let imageWidth = this.image.width / 2; 195 | let imageHeight = this.image.height / 2; 196 | // 画布尺寸根据dpr计算 197 | let canvasWidth = this.canvas.width / DPR; 198 | let canvasHeight = this.canvas.height / DPR; 199 | // console.log(imageWidth, imageHeight, canvasWidth, canvasHeight); 200 | let originX = canvasWidth * (0.5 + getRandomBetweenInt(-startX, startX) / 100); 201 | let originY = canvasHeight; 202 | // 初始值居中是画布总长度减去图片长度后的一半 203 | // 再减去旋转的偏移量 也是图片尺寸的一半 204 | // 刚好减去一倍的图片尺寸 205 | originX -= imageWidth; 206 | originY -= imageHeight; 207 | 208 | let translateX = canvasWidth * getRandomBetweenInt(minCrest, maxCrest) / 100; 209 | let translateY = getRandomByMedian(originY * deltaY, deltaYRange); 210 | 211 | let wavelength = getRandomBetweenInt(minWavelength, maxWavelength); 212 | 213 | // 方向是 -1/1 214 | let direction = getRandomBetweenInt(0, 1) * 2 - 1; 215 | 216 | return ( percentage ) => { 217 | let x = originX + translateX * Math.sin(percentage / wavelength) * direction; 218 | let y = originY - translateY * percentage / 100; 219 | return { x, y } 220 | } 221 | } 222 | } 223 | 224 | export default Like; 225 | -------------------------------------------------------------------------------- /src/js/component/like/module/anim/random.js: -------------------------------------------------------------------------------- 1 | 2 | /** 3 | * 在区间 [min, max] 上的等概率随机整数 4 | * @param {Number} min 最小值 5 | * @param {Number} max 最大值 6 | * @return {Number} value 随机数 7 | */ 8 | let getRandomBetweenInt = function ( min, max ) { 9 | min = parseInt(min) || 0; 10 | max = parseInt(max) || 0; 11 | if (min > max) { 12 | [min, max] = [max, min]; 13 | } 14 | let value = min + Math.round(Math.random() * (max + 1 - min)); 15 | return value == max + 1 ? min : value; 16 | }; 17 | 18 | /** 19 | * 以 value 为中数上下浮动 percentage 的区间上的等概率随机整数 20 | * @param {Number} value 中数 21 | * @param {Number} percentage 浮动系数 22 | * @return {Number} value 随机数 23 | */ 24 | let getRandomByMedian = function ( value, percentage ) { 25 | value = parseInt(value) || 0; 26 | if (percentage < 0) { 27 | percentage = 0; 28 | } 29 | if (percentage > 1) { 30 | percentage = 1; 31 | } 32 | return getRandomBetweenInt(value * (1 - percentage), value * (1 + percentage)); 33 | }; 34 | 35 | /** 36 | * 在区间 [min, max] 上的遵循二项分布的随机整数 其中中数概率最高 37 | * @param {Number} min 最小值 38 | * @param {Number} max 最大值 39 | * @return {Number} value 随机数 40 | */ 41 | let getRandomWithBinomial = function ( min, max ) { 42 | min = parseInt(min) || 0; 43 | max = parseInt(max) || 0; 44 | if (min > max) { 45 | [min, max] = [max, min]; 46 | } 47 | let median = parseInt((max - min) / 2); 48 | return min + getRandomBetweenInt(0, median) + getRandomBetweenInt(0, median); 49 | }; 50 | 51 | export { 52 | getRandomByMedian, 53 | getRandomBetweenInt, 54 | getRandomWithBinomial, 55 | }; 56 | -------------------------------------------------------------------------------- /src/js/component/like/module/anim/ticker.js: -------------------------------------------------------------------------------- 1 | const TICK = 'tick'; 2 | const TICKER = new Event(TICK); 3 | 4 | let elements = []; 5 | 6 | let requestAnimationFrame = window.requestAnimationFrame || window.webkitRequestAnimationFrame || ((fn) => setTimeout(fn, 16)); 7 | 8 | let looper = () => { 9 | elements.forEach((elem) => elem.dispatchEvent(TICKER)); 10 | requestAnimationFrame(looper); 11 | }; 12 | 13 | requestAnimationFrame(looper); 14 | 15 | class Ticker { 16 | constructor () { 17 | this.elem = document.createElement('div'); 18 | elements.push(this.elem); 19 | } 20 | on ( fn ) { 21 | this.elem.addEventListener(TICK, fn, false); 22 | return this 23 | } 24 | off ( fn ) { 25 | this.elem.removeEventListener(TICK, fn, false); 26 | return this 27 | } 28 | } 29 | 30 | export default Ticker; 31 | -------------------------------------------------------------------------------- /src/js/component/like/module/frequent.js: -------------------------------------------------------------------------------- 1 | let requestAnimationFrame = window.requestAnimationFrame || window.webkitRequestAnimationFrame || ((fn) => setTimeout(fn, 16)); 2 | 3 | let frequent = (fn, delay) => { 4 | let counter = parseInt(delay / 16.7); 5 | requestAnimationFrame(function looper () { 6 | if (!--counter) { 7 | fn(); 8 | } else { 9 | requestAnimationFrame(looper); 10 | } 11 | }); 12 | } 13 | export default frequent; 14 | -------------------------------------------------------------------------------- /src/js/component/like/style/index.scss: -------------------------------------------------------------------------------- 1 | .live-like { 2 | canvas { 3 | transform-origin : 100% 100%; 4 | pointer-events: none; 5 | position: absolute; 6 | bottom: 0.46rem; 7 | right : 0; 8 | } 9 | } 10 | -------------------------------------------------------------------------------- /src/js/component/message/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @description 配置文件 3 | */ 4 | 5 | let config = { 6 | // 消息上限 7 | limit: 50, 8 | 9 | // 固定时消息新增上限 10 | limitWhenFixed: 30, 11 | 12 | // 消息更新间隔 13 | interval: 800, 14 | 15 | // 有消息自动滚到底部 16 | autoScrollToBottom: true, 17 | 18 | // 系统消息 19 | systemMessage: '欢迎来到ME星球,玩得开心。', 20 | } 21 | 22 | export default config; 23 | -------------------------------------------------------------------------------- /src/js/component/message/img/chat.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/message/img/chat.png -------------------------------------------------------------------------------- /src/js/component/message/img/gift.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/message/img/gift.png -------------------------------------------------------------------------------- /src/js/component/message/img/live.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/message/img/live.png -------------------------------------------------------------------------------- /src/js/component/message/img/share.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/message/img/share.png -------------------------------------------------------------------------------- /src/js/component/message/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 消息流 3 | * @description 聊天公屏消息流 4 | * @author lixinliang 5 | */ 6 | 7 | import scrollTo from '../../lib/scrollTo/index.js'; 8 | import style from './style/index.scss'; 9 | import createTpl from './tpl/index.js'; 10 | import config from './config'; 11 | 12 | const componentMessage = document.querySelector('[data-component="message"]'); 13 | 14 | // 消息上限 15 | let {limit} = config; 16 | // 固定时消息新增上限 17 | let {limitWhenFixed} = config; 18 | let length = 0; 19 | // 消息更新间隔 20 | let {interval} = config; 21 | let scrollingToBottom = false; 22 | let {autoScrollToBottom} = config; 23 | let tipsShowing = false; 24 | let messages = document.createDocumentFragment(); 25 | 26 | export default { 27 | init (socket) { 28 | componentMessage.innerHTML = `${ createTpl() }`; 29 | this.messageList = componentMessage.querySelector('.live-message__list'); 30 | this.messageScreen = componentMessage.querySelector('.live-message__screen'); 31 | if (componentMessage.matches('.camera-pc ~ [data-component="message"]')) { 32 | // HACK: camera pc mode 33 | const videoHeight = document.querySelector('[data-component="player"]').dataset.videoHeight; 34 | this.messageScreen.style['max-height'] = `${ videoHeight }px`; 35 | this.messageScreen.style['-webkit-mask-size'] = `100% ${ videoHeight }px`; 36 | } 37 | this.sid = setInterval(() => { 38 | this.update(); 39 | }, interval); 40 | const selector = '.live-message__icon'; 41 | const active = 'is-active'; 42 | let fadeOutId = 0; 43 | let fadeOutFn = delegate(selector, function ( event ) { 44 | clearTimeout(fadeOutId); 45 | fadeOutId = setTimeout(() => { 46 | this.classList.remove(active); 47 | }, 100); 48 | }); 49 | document.addEventListener('touchstart', delegate(selector, function ( event ) { 50 | clearTimeout(fadeOutId); 51 | this.classList.add(active); 52 | }), false); 53 | document.addEventListener('touchend', fadeOutFn, false); 54 | document.addEventListener('touchcancel', fadeOutFn, false); 55 | this.messageScreen.addEventListener('scroll', () => { 56 | let { scrollTop, scrollHeight, offsetHeight } = this.messageScreen; 57 | if ((scrollTop + offsetHeight) >= (scrollHeight - 1)) { 58 | autoScrollToBottom = true; 59 | if (tipsShowing) { 60 | tipsShowing = false; 61 | componentMessage.classList.remove('has-tips'); 62 | } 63 | } else { 64 | autoScrollToBottom = false; 65 | } 66 | }, false); 67 | 68 | let button = componentMessage.querySelector('.live-message__tips'); 69 | button.addEventListener('click', () => { 70 | scrollingToBottom = true; 71 | scrollTo('bottom', { 72 | elem : this.messageScreen, 73 | // transition : false, 74 | }).then(() => { 75 | componentMessage.classList.remove('has-tips'); 76 | scrollingToBottom = false; 77 | autoScrollToBottom = true; 78 | }); 79 | }, false); 80 | 81 | this.addSystem(config.systemMessage); 82 | socket.ioSocket.on(socket.socketName, ( data ) => { 83 | let {nick, content} = data.data; 84 | this.addChat(nick, content); 85 | }); 86 | }, 87 | add ( text, insertBefore ) { 88 | let message = document.createElement('p'); 89 | message.classList.add('live-message__item'); 90 | message.innerHTML = text; 91 | if (insertBefore) { 92 | messages.insertBefore(message, messages.firstElementChild); 93 | } else { 94 | messages.appendChild(message); 95 | } 96 | if (insertBefore) { 97 | this.update(); 98 | } 99 | }, 100 | update () { 101 | if (scrollingToBottom) { 102 | return; 103 | } 104 | if (!messages.children.length) { 105 | return; 106 | } 107 | length++; 108 | if (!autoScrollToBottom) { 109 | if (length > (limit + limitWhenFixed)) { 110 | autoScrollToBottom = true; 111 | tipsShowing = false; 112 | componentMessage.classList.remove('has-tips'); 113 | for (let i = 0; i < limitWhenFixed; i++) { 114 | this.messageList.removeChild(this.messageList.firstElementChild); 115 | } 116 | length -= limitWhenFixed; 117 | } 118 | } 119 | if (autoScrollToBottom) { 120 | if (length > limit) { 121 | this.messageList.removeChild(this.messageList.firstElementChild); 122 | length--; 123 | } 124 | } 125 | this.messageList.appendChild(messages.firstElementChild); 126 | if (autoScrollToBottom) { 127 | scrollTo('bottom', { 128 | elem : this.messageScreen, 129 | transition : false, 130 | }); 131 | } else { 132 | if (!tipsShowing) { 133 | tipsShowing = true; 134 | componentMessage.classList.add('has-tips'); 135 | } 136 | } 137 | }, 138 | addSystem ( text ) { 139 | this.add(`系统通知 ${ text }`); 140 | }, 141 | addShare ( nickname ) { 142 | this.add(`${ nickname } 分享了这个直播`); 143 | }, 144 | addChat ( nickname, text, insertBefore ) { 145 | this.add(`${ nickname } ${ text }`, insertBefore); 146 | }, 147 | addPresent ( nickname, gift, insertBefore ) { 148 | this.add(`${ nickname } 送了 ${ gift }`, insertBefore); 149 | }, 150 | addFollow ( nickname, anchorName ) { 151 | this.add(`${ nickname } 关注了 ${ anchorName }`, true); 152 | }, 153 | destroy () { 154 | clearInterval(this.sid); 155 | componentMessage.innerHTML = ''; 156 | }, 157 | show () { 158 | componentMessage.classList.remove('fade-out'); 159 | }, 160 | hide () { 161 | componentMessage.classList.add('fade-out'); 162 | }, 163 | }; 164 | -------------------------------------------------------------------------------- /src/js/component/message/style/index.scss: -------------------------------------------------------------------------------- 1 | .live-message { 2 | position: absolute; 3 | // left: 0.2rem; 4 | left: 0; 5 | bottom: .2rem; 6 | // height: .56rem * 5 + .68; 7 | // padding-bottom: .68rem; 8 | // width: 8rem; 9 | width: 100%; 10 | // display: none; 11 | transition: opacity 200ms ease-in; 12 | &.fade-out { 13 | pointer-events: none; 14 | opacity: 0; 15 | } 16 | &.has-tips { 17 | .live-message__tips { 18 | display: inline-block; 19 | } 20 | } 21 | } 22 | .live-message__screen { 23 | max-height: .56rem * 5; 24 | margin-bottom: 0.2rem; 25 | overflow-y: scroll; 26 | -webkit-overflow-scrolling: touch; 27 | -webkit-mask-image: linear-gradient(to bottom, rgba(0, 0, 0, 0), rgba(0, 0, 0, 1) 15%, rgba(0, 0, 0, 1)); 28 | -webkit-mask-repeat: no-repeat; 29 | -webkit-mask-size: 100% .56rem * 5; 30 | -webkit-mask-position: 100% 100%; 31 | // position: relative; 32 | &::-webkit-scrollbar { 33 | display: none; 34 | } 35 | } 36 | .live-message__list { 37 | // position: absolute; 38 | // left: 0; 39 | // bottom: 0; 40 | width: 6rem; 41 | position: relative; 42 | left: 0.2rem; 43 | .is-system { 44 | color: #76ae15; 45 | } 46 | .is-share { 47 | color: #ffdda0; 48 | } 49 | .is-nickname { 50 | color: #e0b66c; 51 | } 52 | .is-present { 53 | color: #e0b66c; 54 | } 55 | .is-follow { 56 | color: #e0b66c; 57 | } 58 | } 59 | .live-message__item { 60 | line-height: .56rem; 61 | font-size: 0.32rem; 62 | color: #fff; 63 | // text-shadow: 0 0 1px rgba(0, 0, 0, .6); 64 | // text-shadow: 0 0 1px rgba(0, 0, 0, 1); 65 | text-shadow: 1px 0 3px rgba(0, 0, 0, 1); 66 | &, 67 | * { 68 | font-weight: 500; 69 | } 70 | } 71 | .live-message__action { 72 | height: 0.68rem; 73 | width: 0.68rem * 4 + 0.2 * 3; 74 | margin-left: 0.2rem; 75 | // position: absolute; 76 | // left: 0; 77 | // bottom: 0; 78 | &::after { 79 | content: ""; 80 | clear: both; 81 | display: table; 82 | } 83 | } 84 | .live-message__icon { 85 | float: left; 86 | width: 0.68rem; 87 | height: 0.68rem; 88 | margin-left: 0.2rem; 89 | background-repeat: no-repeat; 90 | background-size: 100% 100%; 91 | // box-shadow: 0 0 3px 1px rgba(0, 0, 0, .4); 92 | border-radius: 50%; 93 | cursor: pointer; 94 | &:first-child { 95 | margin-left: 0; 96 | } 97 | &.is-chat { 98 | background-image: url(../img/chat.png); 99 | } 100 | &.is-share { 101 | background-image: url(../img/share.png); 102 | } 103 | &.is-live { 104 | background-image: url(../img/live.png); 105 | } 106 | &.is-gift { 107 | background-image: url(../img/gift.png); 108 | } 109 | &.is-active { 110 | filter: brightness(.7); 111 | } 112 | } 113 | .live-message__tips { 114 | display: none; 115 | position: absolute; 116 | background-color: #ffd893; 117 | padding-left: 0.2rem; 118 | padding-right: 0.2rem + 0.25; 119 | height: 0.64rem; 120 | line-height: 0.64rem; 121 | font-size: 0.26rem; 122 | color: #261300; 123 | left: 0.2rem; 124 | bottom: 0rem + 0.68 + 0.2 + 0.1; 125 | border-radius: 0.12rem; 126 | cursor: pointer; 127 | i { 128 | position: absolute; 129 | top: 50%; 130 | width: 0; 131 | height: 0; 132 | right: 0.35rem; 133 | margin-top: -0.07rem; 134 | &::before { 135 | content: ""; 136 | position: absolute; 137 | top: 0; 138 | left: 0; 139 | border-top: 4px solid #261300; 140 | border-left: 4px solid transparent; 141 | border-right: 4px solid transparent; 142 | } 143 | &::after { 144 | content: ""; 145 | position: absolute; 146 | left: 1px; 147 | top: 0px; 148 | border-top: 3px solid #ffd893; 149 | border-left: 3px solid transparent; 150 | border-right: 3px solid transparent; 151 | } 152 | } 153 | } 154 | -------------------------------------------------------------------------------- /src/js/component/message/tpl/index.js: -------------------------------------------------------------------------------- 1 | let renderTpl = (data)=>{ 2 | let tpl = ` 3 |
4 |
5 |
6 |
7 | 8 | 9 | 10 | 11 |
12 |
有新消息
13 | `; 14 | return tpl; 15 | } 16 | 17 | export default renderTpl; 18 | -------------------------------------------------------------------------------- /src/js/component/player/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @description 配置文件 3 | */ 4 | 5 | let config = { 6 | checkLiveStatusSpeek: 1500, // 检查直播状态的速度频率(毫秒) 7 | } 8 | 9 | export default config; 10 | -------------------------------------------------------------------------------- /src/js/component/player/img/icon-play.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/player/img/icon-play.png -------------------------------------------------------------------------------- /src/js/component/player/img/loading.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 7 | 11 | 12 | 13 | 17 | 21 | 22 | 23 | 27 | 31 | 32 | 33 | 37 | 41 | 42 | 43 | 47 | 51 | 52 | 53 | -------------------------------------------------------------------------------- /src/js/component/player/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 播放器 3 | * @description 根据不同状态渲染不同的UI界面 4 | * @author David 5 | */ 6 | 7 | import config from './config.js'; 8 | import livingStyle from './style/live-in.scss'; 9 | import liveEndStyle from './style/live-end.scss'; 10 | 11 | import livingTpl from './tpl/live-in.js'; // 直播中状态 12 | import liveEndTpl from './tpl/live-end.js'; // 直播结束状态 13 | 14 | let componentPlayer = document.querySelector('[data-component="player"]'); 15 | let video = null; 16 | 17 | export default { 18 | 19 | init({data}) { 20 | this.baseLiveData = data; 21 | return new Promise(( resolve, reject ) => { 22 | // 是否在开播 23 | if(data.living) { 24 | this.renderLiving(data).then(()=>{ 25 | resolve(); 26 | }); 27 | } 28 | else { 29 | this.renderLiveEnd(data); 30 | } 31 | 32 | }).catch((err) => { 33 | console.log(err); 34 | throw err; 35 | }); 36 | }, 37 | 38 | // 监听直播状态 39 | listenLiveStatus(callback) { 40 | RequqestApi.getLiveStatus().then(({data})=>{ 41 | if (data.status === 1 && !this.scopeLiveStatus) { 42 | this.renderLiving(this.baseLiveData, true); 43 | this.scopeLiveStatus = 1; 44 | } 45 | else if (data.status !== 1 && this.scopeLiveStatus) { 46 | this.renderLiveEnd(this.baseLiveData); 47 | this.scopeLiveStatus = 0; 48 | } 49 | callback && callback(data.status); 50 | }); 51 | setTimeout(()=>{ 52 | this.listenLiveStatus(callback); 53 | }, config.checkLiveStatusSpeek); 54 | }, 55 | 56 | // 初始化视频 57 | createVideo(data) { 58 | return new Promise(( resolve, reject ) => { 59 | componentPlayer.classList.add('is-waiting'); 60 | window.util.sleep(1000).then(()=>{ 61 | if (!video) { 62 | let newVideo = document.createElement('video'); 63 | document.querySelector('#player').appendChild(newVideo); 64 | setTimeout(resolve, 10); 65 | video = newVideo; 66 | video.src = data.liveUrl; 67 | video.setAttribute('playsinline', true); 68 | video.removeAttribute('poster'); 69 | } 70 | video.play(); 71 | this.scopeLiveStatus = 1; 72 | }) 73 | }); 74 | }, 75 | 76 | // 渲染直播中状态UI 77 | renderLiving(data, autoplay) { 78 | return new Promise(( resolve, reject ) => { 79 | let isAutoPlay = autoplay?autoplay:(Number(window.util.getUrlParam('autoplay')) || 0); 80 | componentPlayer.innerHTML = `${ livingTpl(data) }`; 81 | componentPlayer.classList.add('has-poster', 'has-button'); 82 | // 自动播放 83 | isAutoPlay && this.createVideo(data).then(resolve); 84 | 85 | document.addEventListener('click', window.util.delegate('[data-role="play-video"]', () => { 86 | this.createVideo(data).then(resolve); 87 | }), false); 88 | }) 89 | .then(this.bindEvent) 90 | .then(() => { 91 | this.setPlayerSize('mobile'); 92 | }); 93 | }, 94 | 95 | // 渲染开播结束状态UI 96 | renderLiveEnd(data) { 97 | const posterURL = data.cover; 98 | componentPlayer.style['background-image'] = `url(${ posterURL })`; 99 | componentPlayer.innerHTML = `${ liveEndTpl(data) }`; 100 | componentPlayer.classList.add('has-poster'); 101 | video = null; 102 | }, 103 | 104 | // 动态计算视频窗口 105 | setPlayerSize( type ) { 106 | const winHeight = window.innerHeight; 107 | const topBarHeight = document.querySelector('[data-component="topBar"]').offsetHeight; 108 | if (type == 'mobile') { 109 | const playerHeight = winHeight - topBarHeight + 1; 110 | const videoHeight = '120%'; 111 | const videoTop = '-10%'; 112 | video.style['height'] = videoHeight; 113 | video.style['top'] = videoTop; 114 | componentPlayer.classList.remove('has-poster', 'has-button', 'is-waiting'); 115 | componentPlayer.classList.add('has-live'); 116 | return new Promise(( resolve ) => { 117 | componentPlayer.addEventListener('transitionend', resolve, false); 118 | componentPlayer.style['height'] = `${ playerHeight }px`; 119 | }); 120 | } 121 | if (type == 'pc') { 122 | const winHeight = window.innerHeight; 123 | const topBarHeight = document.querySelector('[data-component="topBar"]').offsetHeight; 124 | // HACK: message bottom + message action height + message screen margin bottom 125 | const messageActionZone = Math.round((0.2 + 0.68 + 0.2) * parseFloat(getComputedStyle(document.documentElement)['font-size'])); 126 | const playerHeight = winHeight - topBarHeight + 1; 127 | const videoHeight = (playerHeight - messageActionZone) * 0.5; 128 | video.style['height'] = '50%'; 129 | video.style['width'] = `${ videoHeight * 16 / 9 }px`; 130 | document.querySelector('#player').style['padding-bottom'] = `${ messageActionZone }px`; 131 | componentPlayer.classList.remove('has-poster', 'has-button', 'is-waiting'); 132 | componentPlayer.classList.add('has-live', 'camera-pc'); 133 | componentPlayer.dataset.videoHeight = videoHeight; 134 | return new Promise(( resolve ) => { 135 | componentPlayer.addEventListener('transitionend', resolve, false); 136 | componentPlayer.style['height'] = `${ playerHeight }px`; 137 | }); 138 | } 139 | }, 140 | 141 | // 添加视频事件 142 | bindEvent() { 143 | // TODO: 暴露回调 其他组件可以获取直播状态 144 | video.addEventListener('pause', () => { 145 | componentPlayer.classList.add('has-button'); 146 | this.scopeLiveStatus = 0; 147 | }, false); 148 | video.addEventListener('play', () => { 149 | componentPlayer.classList.remove('has-button'); 150 | }, false) 151 | }, 152 | 153 | } 154 | 155 | -------------------------------------------------------------------------------- /src/js/component/player/style/live-end.scss: -------------------------------------------------------------------------------- 1 | .live-player { 2 | position: relative; 3 | height: 6.87rem; 4 | color: #fff; 5 | overflow: hidden; 6 | text-align: center; 7 | background-size: cover; 8 | background-position: center; 9 | background-repeat: no-repeat; 10 | pointer-events: none; 11 | &:after { 12 | content: ''; 13 | position: absolute; 14 | top: 0; 15 | left: 0; 16 | width: 100%; 17 | height: 100%; 18 | background: rgba(0, 0, 0, .5); 19 | } 20 | } 21 | .live-player__status, 22 | .live-player__avatar, 23 | .live-player__nick, 24 | .live-player__uid { 25 | position: relative; 26 | } 27 | .live-player__status { 28 | // padding-top: .9rem; 29 | font-size: .4rem; 30 | margin-top: .9rem; 31 | } 32 | .live-player__avatar { 33 | // padding-top: .52rem; 34 | margin: .52rem auto 0; 35 | height: 1.48rem; 36 | width: 1.48rem; 37 | // border-radius: 1.48rem; 38 | border-radius: 50%; 39 | overflow: hidden; 40 | background-size: 100% 100%; 41 | background-repeat: no-repeat; 42 | // img { 43 | // width: 1.48rem; 44 | // height: 1.48rem; 45 | // border-radius: 1.48rem; 46 | // } 47 | } 48 | .live-player__nick { 49 | // padding-top: .05rem; 50 | margin-top: .05rem; 51 | font-size: .34rem; 52 | } 53 | .live-player__uid { 54 | // padding-top: .1rem; 55 | margin-top: .1rem; 56 | font-size: .24rem; 57 | } 58 | -------------------------------------------------------------------------------- /src/js/component/player/style/live-in.scss: -------------------------------------------------------------------------------- 1 | %center { 2 | position: absolute; 3 | left: 50%; 4 | top: 50%; 5 | transform: translate(-50%, -50%); 6 | } 7 | .live-player { 8 | width: 100%; 9 | height: 6.87rem; 10 | overflow: hidden; 11 | position: relative; 12 | background-size: 0 0; 13 | background-position: center; 14 | background-repeat: no-repeat; 15 | font-size: 0.8rem; 16 | line-height: 5rem; 17 | transition: height .4s linear 0s; 18 | color: #fff; 19 | text-align: center; 20 | pointer-events: none; 21 | #player { 22 | position: relative; 23 | width: 100%; 24 | height: 100%; 25 | // top: 50%; 26 | // transform: translateY(-50%); 27 | } 28 | video { 29 | width: 100%; 30 | height: 100%; 31 | display: block; 32 | font-size: 0.28rem; 33 | position: relative; 34 | } 35 | #player { 36 | opacity: 0; 37 | } 38 | &.has-poster { 39 | background-size: cover; 40 | &:after { 41 | content: ''; 42 | position: absolute; 43 | top: 0; 44 | left: 0; 45 | width: 100%; 46 | height: 100%; 47 | background: rgba(0, 0, 0, .5); 48 | } 49 | } 50 | &.has-live { 51 | #player { 52 | opacity: 1; 53 | } 54 | } 55 | &.camera-pc { 56 | #player { 57 | video { 58 | left: 50%; 59 | transform: translateX(-50%); 60 | } 61 | } 62 | } 63 | &.has-button { 64 | pointer-events: auto; 65 | } 66 | &.has-button.is-waiting { 67 | .live-player__playBtn { 68 | &:after { 69 | background-image: url(); 70 | } 71 | } 72 | } 73 | } 74 | 75 | .live-player__playBtn { 76 | // width: 1.84rem; 77 | display: none; 78 | width: 90px; 79 | height: 0; 80 | padding-top: 65px; 81 | // padding-top: 1.3rem; 82 | overflow: hidden; 83 | background-color: #000; 84 | border-radius: .18rem; 85 | cursor: pointer; 86 | @extend %center; 87 | &:after { 88 | content: ''; 89 | width: .48rem; 90 | height: .6rem; 91 | width: .48rem; 92 | height: .6rem; 93 | background: url(../img/icon-play.png) 0 0 no-repeat; 94 | background-size: cover; 95 | @extend %center; 96 | } 97 | .live-player.has-button & { 98 | display: block; 99 | } 100 | } 101 | 102 | @keyframes loading{ 103 | 0%{ 104 | background-position: 0px 0; 105 | } 106 | 100%{ 107 | background-position: 0px -630px; 108 | } 109 | } 110 | -------------------------------------------------------------------------------- /src/js/component/player/tpl/live-end.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file 直播结束UI 3 | */ 4 | 5 | let renderTpl = (data)=>{ 6 | let tpl = ` 7 |

直播已结束

8 |

9 |

${data.anchorNick}

10 |

ME号:${data.anchorId}

11 | `; 12 | return tpl; 13 | } 14 | 15 | export default renderTpl; 16 | 17 | -------------------------------------------------------------------------------- /src/js/component/player/tpl/live-in.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file 直播中UI 3 | */ 4 | 5 | let renderTpl = (data)=>{ 6 | let tpl = ` 7 | 12 |
13 |
播放
14 | `; 15 | return tpl; 16 | } 17 | 18 | export default renderTpl; 19 | 20 | 21 | 22 | 23 | -------------------------------------------------------------------------------- /src/js/component/recommend-list/img/ico-empty.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/recommend-list/img/ico-empty.png -------------------------------------------------------------------------------- /src/js/component/recommend-list/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 直播推荐列表(大家都在看) 3 | * @author lixinliang 4 | */ 5 | 6 | import style from './style/index.scss'; 7 | import renderTpl from './tpl/index.js'; 8 | import LazyloadImg from '../../lib/lazy-load-img/index.js'; 9 | 10 | let liveContent = document.querySelector('[data-role="liveContent"]'); 11 | let componentRecommendList = document.createElement('div'); 12 | componentRecommendList.setAttribute('data-component', 'recommendList'); 13 | componentRecommendList.classList.add('live-recommendList'); 14 | liveContent.parentNode.insertBefore(componentRecommendList, liveContent.nextSibling); 15 | 16 | export default { 17 | init() { 18 | RequqestApi.getRecommendList().then((response)=>{ 19 | let list = response.data; 20 | componentRecommendList.innerHTML = `${ renderTpl(list) }`; 21 | new LazyloadImg({ 22 | el: componentRecommendList.querySelector('.live-recommendList__content'), 23 | mode: 'diy', 24 | position: { 25 | top: -150, 26 | right: -150, 27 | bottom: -150, 28 | left: -150, 29 | }, 30 | diy: { 31 | backgroundSize: 'cover', 32 | backgroundRepeat: 'no-repeat', 33 | backgroundPosition: 'center center', 34 | }, 35 | }).start(); 36 | }) 37 | 38 | }, 39 | }; 40 | -------------------------------------------------------------------------------- /src/js/component/recommend-list/style/index.scss: -------------------------------------------------------------------------------- 1 | .live-recommendList { 2 | overflow: hidden; 3 | background: #ededed; 4 | &::after { 5 | content: ""; 6 | clear: both; 7 | display: table; 8 | } 9 | .live-recommendList__head { 10 | font-size: 0.28rem; 11 | color: #2e2e2e; 12 | height: .88rem; 13 | line-height: .88rem; 14 | padding: 0 .20rem; 15 | overflow: hidden; 16 | user-select: none; 17 | } 18 | .live-recommendList__item { 19 | float: left; 20 | width: 50%; 21 | a { 22 | display: block; 23 | background: #f7f7f7; 24 | } 25 | &:nth-child(2n+1) { 26 | padding-right: 2px; 27 | } 28 | &:nth-child(2n) { 29 | padding-left: 2px; 30 | } 31 | } 32 | .live-recommendList__img { 33 | position: relative; 34 | padding-bottom: 100%; 35 | background-color: #f5f5f5; 36 | img { 37 | position: absolute; 38 | top: 0; 39 | left: 0; 40 | width: 100%; 41 | height: 100%; 42 | font-size: 0.28rem; 43 | } 44 | } 45 | .live-recommendList__nick { 46 | font-size: 0.24rem; 47 | height: 0.76rem; 48 | padding: 0 0.18rem 0.16rem; 49 | line-height: 0.60rem; 50 | overflow: hidden; 51 | word-break: normal; 52 | word-wrap: normal; 53 | white-space: nowrap; 54 | text-overflow: ellipsis; 55 | // &::after { 56 | // content: ""; 57 | // clear: both; 58 | // display: table; 59 | // } 60 | } 61 | .live-recommendList__count { 62 | font-size: 0.20rem; 63 | color: #484848; 64 | height: 0.60rem; 65 | line-height: 0.60rem; 66 | float: right; 67 | margin-left: 0.1rem; 68 | } 69 | } 70 | -------------------------------------------------------------------------------- /src/js/component/recommend-list/tpl/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file 推荐列表 3 | */ 4 | 5 | let renderTpl = (list)=>{ 6 | let tpl = ` 7 |
大家都在看
8 |
9 | ${list.map(item => 10 | `
11 | 12 |
13 |

${item.users}人${item.nick}

14 |
15 |
` 16 | ).join('')} 17 |
18 | `; 19 | return tpl; 20 | } 21 | 22 | export default renderTpl; 23 | -------------------------------------------------------------------------------- /src/js/component/top-bar/img/app-logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/top-bar/img/app-logo.png -------------------------------------------------------------------------------- /src/js/component/top-bar/img/icon-share.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/js/component/top-bar/img/icon-share.png -------------------------------------------------------------------------------- /src/js/component/top-bar/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 顶部工具条 3 | * @author David 4 | */ 5 | 6 | import style from './style/index.scss'; 7 | import renderTpl from './tpl/index.js'; 8 | 9 | export default { 10 | init () { 11 | let componentTopBar = document.querySelector('[data-component="topBar"]'); 12 | componentTopBar.innerHTML = `${ renderTpl() }`; 13 | }, 14 | }; 15 | -------------------------------------------------------------------------------- /src/js/component/top-bar/style/index.scss: -------------------------------------------------------------------------------- 1 | body { 2 | padding-top: 1.05rem; 3 | } 4 | 5 | .live-topBar { 6 | position: fixed; 7 | left: 0; 8 | top: 0; 9 | width: 100%; 10 | height: 1.05rem; 11 | overflow: hidden; 12 | user-select: none; 13 | // padding: .13rem 0; 14 | // border-bottom: .02rem solid #7b7360; 15 | background: #2e2e2e; 16 | // background: rgba(46, 46, 46, 0.8); 17 | &::before { 18 | position: absolute; 19 | left: .28rem; 20 | top: 50%; 21 | transform: translate(0, -50%); 22 | // margin-top: .01rem; 23 | content: ''; 24 | width: .76rem; 25 | height: .76rem; 26 | background: url(../img/app-logo.png) 0 0 no-repeat; 27 | background-size: cover; 28 | } 29 | &::after { 30 | content: ""; 31 | position: absolute; 32 | width: 100%; 33 | height: 1px; 34 | left: 0; 35 | bottom: 0; 36 | background-color: #7b7360; 37 | transform: scaleY(.5); 38 | } 39 | } 40 | .live-topBar__cnt { 41 | padding-left: 1.33rem; 42 | font-size: .24rem; 43 | height: .24rem; 44 | line-height: .24rem; 45 | color: #fff; 46 | &:first-of-type { 47 | margin-top: .25rem; 48 | margin-bottom: .1rem; 49 | } 50 | } 51 | .live-topBar__btn { 52 | position: absolute; 53 | right: .25rem; 54 | top: 50%; 55 | transform: translate(0, -50%); 56 | width: 2.43rem; 57 | height: .66rem; 58 | // line-height: .66rem; 59 | // text-align: center; 60 | 61 | // text-align: center; 62 | // border: .03rem solid #7d7051; 63 | border: 1px solid #7d7051; 64 | border-radius: .15rem; 65 | background: #524c3e; 66 | // outline: none; 67 | cursor: pointer; 68 | &.is-active { 69 | background: #736b50; 70 | } 71 | &::after { 72 | content: "立即打开"; 73 | position: absolute; 74 | top: 50%; 75 | left: 50%; 76 | transform: translateX(-50%) translateY(-50%); 77 | font-size: .28rem; 78 | height: .28rem; 79 | line-height: .28rem; 80 | color: #e0c37e; 81 | } 82 | } 83 | -------------------------------------------------------------------------------- /src/js/component/top-bar/tpl/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file top-bar 3 | */ 4 | 5 | let renderTpl = (data)=>{ 6 | let tpl = ` 7 |

下载ME立即与TA

8 |

零距离实时互动

9 |
10 | `; 11 | return tpl; 12 | } 13 | 14 | export default renderTpl; 15 | 16 | -------------------------------------------------------------------------------- /src/js/lib/anim/anim.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component animjs 包含以下功能:点赞图片预加载、点赞图片分组、定义点赞的canvas样式 3 | * @author lixinliang 4 | */ 5 | 6 | import Ticker from './modules/ticker.js'; 7 | 8 | const DPR = window.devicePixelRatio || 1; 9 | 10 | let closure = {}; 11 | let storage = []; 12 | 13 | // animjs 不涉及动画的实现 调用show方法时 把图片集合交给 动画引擎处理 14 | class Anim { 15 | 16 | /** 17 | * new的时候决定该实例使用那种动画引擎 18 | * @type {String} name 动画引擎 19 | */ 20 | constructor ( name = '' ) { 21 | let canvas = document.createElement('canvas'); 22 | let context = canvas.getContext('2d'); 23 | this.canvas = canvas; 24 | this.context = context; 25 | this.id = storage.length; 26 | this.name = name; 27 | let imageSet = {}; 28 | let imageMap = {}; 29 | storage.push({ imageSet, imageMap }); 30 | let ticker = new Ticker(); 31 | ticker.on(() => { 32 | context.clearRect(0, 0, canvas.width, canvas.height); 33 | }); 34 | this.ticker = ticker; 35 | this.dpr = DPR; 36 | if (DPR > 1) { 37 | this.canvas.style['transform'] = this.canvas.style['-webkit-transform'] = `scale(${ 1/DPR })`; 38 | } 39 | } 40 | 41 | /** 42 | * 传入一组图片集合 进行预加载 43 | * @param {Object} imageSet 资源集合 { 图片id : 图片url } 44 | * @return {Anim} this anim 45 | */ 46 | mainfest ( imageSet ) { 47 | let mySet = storage[this.id].imageSet; 48 | for (let key in imageSet) { 49 | let value = imageSet[key]; 50 | let image = { 51 | url : value, 52 | status : false 53 | }; 54 | image.img = loadImage(image.url, () => { 55 | image.status = true; 56 | }); 57 | mySet[key] = image; 58 | } 59 | return this 60 | } 61 | 62 | /** 63 | * 传入图片分组的映射关系 64 | * @param {Object} imageMap 资源分组映射 { 分组id : [图片id,...] } 65 | * @return {Anim} this anim 66 | */ 67 | group ( imageMap ) { 68 | let myMap = storage[this.id].imageMap; 69 | for (let key in imageMap) { 70 | myMap[key] = imageMap[key].slice(); 71 | } 72 | return this 73 | } 74 | 75 | /** 76 | * 随机播放动画 77 | * @param {String} groupName 资源分组名 78 | * @return {Anim} this anim 79 | */ 80 | show ( groupName ) { 81 | let { imageSet, imageMap } = storage[this.id]; 82 | let images = []; 83 | imageMap[groupName].forEach((imageName) => { 84 | let image = imageSet[imageName]; 85 | if (image.status) { 86 | images.push(image.img); 87 | } 88 | }); 89 | if (this.name) { 90 | return new closure[this.name](this, images); 91 | } 92 | return this 93 | } 94 | 95 | /** 96 | * 设置宽度 97 | * @param {Number} canvasWidth 画布宽度 98 | * @return {Anim} this anim 99 | */ 100 | width ( canvasWidth ) { 101 | let result = correct(canvasWidth); 102 | if (result) { 103 | let { num, str } = result; 104 | this.canvas.width = num; 105 | this.canvas.style.width = str; 106 | } 107 | return this 108 | } 109 | 110 | /** 111 | * 设置高度 112 | * @param {Number} canvasHeight 画布高度 113 | * @return {Anim} this anim 114 | */ 115 | height ( canvasHeight ) { 116 | 117 | let result = correct(canvasHeight); 118 | if (result) { 119 | let { num, str } = result; 120 | this.canvas.height = num; 121 | this.canvas.style.height = str; 122 | } 123 | 124 | return this 125 | } 126 | } 127 | 128 | export default Anim; 129 | 130 | /** 131 | * 加载图片 132 | * @param {String} url 图片地址 133 | * @param {Function} cb 加载回调 134 | * @return {HTMLImageElement} img 图片标签 135 | */ 136 | function loadImage ( url, cb ) { 137 | let img = new Image; 138 | img.onload = cb; 139 | img.src = url; 140 | return img; 141 | }; 142 | 143 | /** 144 | * 对canvas的宽高修正 在hidpi屏上 尺寸需要加倍 145 | * @param {Number} value 尺寸 146 | * @return {Number} value 尺寸 147 | */ 148 | function correct ( value ) { 149 | value *= DPR; 150 | let num; 151 | let str; 152 | if (typeof value == 'number') { 153 | num = value; 154 | str = value + 'px'; 155 | } else if (typeof value == 'string') { 156 | str = value; 157 | num = parseInt(value); 158 | } 159 | if (num && str) { 160 | return { num, str } 161 | } else { 162 | return null 163 | } 164 | }; 165 | 166 | /** 167 | * 定义动画引擎 168 | * @param {String} name 动画引擎名 169 | * @param {Class} plugin 动画引擎 170 | */ 171 | Anim.setup = function ( name, plugin ) { 172 | closure[name] = plugin; 173 | }; 174 | -------------------------------------------------------------------------------- /src/js/lib/anim/anim.like.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component anim.like 基于 animjs 的点赞动画引擎 3 | * @author lixinliang 4 | */ 5 | 6 | import { 7 | getRandomByMedian, 8 | getRandomBetweenInt, 9 | getRandomWithBinomial, 10 | } from './modules/random.js'; 11 | 12 | // 点赞动画引擎 13 | class Like { 14 | /** 15 | * 调用 new Like 创建一个赞的动画 16 | * @param {Anim} anim Anim实例 17 | * @param {Array} images 素材集合 18 | * @return {Like} like 点赞实例 19 | */ 20 | constructor ( anim, images ) { 21 | // 从图片集合 随机抽取一张 22 | let image = images[getRandomBetweenInt(0, images.length - 1)]; 23 | 24 | this.image = image; 25 | this.canvas = anim.canvas; 26 | 27 | let DPR = this.dpr = anim.dpr; 28 | 29 | let imageWidth = image.width; 30 | let imageHeight = image.height; 31 | 32 | // @return 当前动画进度 0 ~ 100 33 | let getImageTimeline = this.createImageTimeline(); 34 | // @return 旋转后的图片 canvas 35 | let getImageData = this.createImageData(); 36 | // @return 获取当前缩放曲线 number 37 | let getImageScaleSize = this.createImageScaleSize(); 38 | // @return 获取当前透明度曲线 number 39 | let getImageOpacityValue = this.createImageOpacityValue(); 40 | // @return 获取当前位移曲线 { x, y } 41 | let getImagePathPosition = this.createImagePathPosition(); 42 | 43 | let handler = () => { 44 | let percentage = getImageTimeline(); 45 | let { imageCanvas, imageContext } = getImageData(); 46 | let scaleSize = getImageScaleSize(percentage); 47 | let globalAlpha = getImageOpacityValue(percentage); 48 | let { x, y } = getImagePathPosition(percentage); 49 | 50 | // 最后渲染的时候 再把 dpr 计算进去 51 | if (scaleSize == 1 && globalAlpha == 1) { 52 | anim.context.drawImage(imageCanvas, x * DPR, y * DPR, imageWidth * DPR, imageHeight * DPR); 53 | } else { 54 | if (globalAlpha == 1) { 55 | let scaleWidth = imageWidth * scaleSize; 56 | let deltaWidth = (imageWidth - scaleWidth) / 2; 57 | let scaleHeight = imageHeight * scaleSize; 58 | let deltaHeight = (imageHeight - scaleHeight) / 2; 59 | anim.context.drawImage(imageCanvas, (x + deltaWidth) * DPR, (y + deltaHeight) * DPR, scaleWidth * DPR, scaleHeight * DPR); 60 | } else { 61 | anim.context.globalAlpha = globalAlpha; 62 | anim.context.drawImage(imageCanvas, x * DPR, y * DPR, imageWidth * DPR, imageHeight * DPR); 63 | anim.context.globalAlpha = 1; 64 | } 65 | } 66 | if (percentage == 100) { 67 | getImageTimeline = null; 68 | getImageData = null; 69 | getImageScaleSize = null; 70 | getImageOpacityValue = null; 71 | getImagePathPosition = null; 72 | anim.ticker.off(handler); 73 | } 74 | }; 75 | anim.ticker.on(handler); 76 | } 77 | 78 | /** 79 | * 创建每一个动画的独立时间线 80 | * @return {Function} getImageTimeline 获得该动画当前时间线的函数 81 | */ 82 | createImageTimeline () { 83 | // 动画总时基数 3000ms 浮动系数 20% 84 | let originDuration = 3000; 85 | let range = 0.2; 86 | 87 | let duration = getRandomByMedian(originDuration, range); 88 | let times = Math.round(duration / 16.7); 89 | let now = 0; 90 | return () => { 91 | now++; 92 | let percentage = parseInt(now * 1000 / times) / 10; 93 | return percentage > 100 ? 100 : percentage 94 | } 95 | } 96 | 97 | /** 98 | * 创建每一个动画的独立形态 99 | * @return {Function} getImageData 获得该动画形态的对象的函数 100 | */ 101 | createImageData () { 102 | // 图像旋转幅度 -30 ~ 30deg 103 | let minRotate = -30; 104 | let maxRotate = 30; 105 | // 缩放系数的二项分布 70 ~ 120 出现95的概率最高 106 | let minScale = 70; 107 | let maxScale = 120; 108 | // 缩放最大值为105 超出105的概率以105为对称轴 添加到 90 ~ 105 的区间 109 | let maxLimitScale = 105; 110 | 111 | let image = this.image; 112 | let { width, height } = image; 113 | let rotate = getRandomBetweenInt(minRotate, maxRotate); 114 | let scale = getRandomWithBinomial(minScale, maxScale); 115 | if (scale > maxLimitScale) { 116 | scale -= (maxScale - maxLimitScale); 117 | } 118 | scale = scale / 100; 119 | 120 | let imageCanvas = document.createElement('canvas'); 121 | let imageContext = imageCanvas.getContext('2d'); 122 | 123 | // 使用离屏canvas保存旋转后的图片并多次复用 为了避免旋转后图片被切割 画布的尺寸是图片的2倍 124 | imageCanvas.width = width * 2; 125 | imageCanvas.height = height * 2; 126 | 127 | imageContext.translate(width, height); 128 | imageContext.rotate(rotate * Math.PI / 180); 129 | imageContext.scale(scale, scale); 130 | imageContext.translate(-width, -height); 131 | 132 | imageContext.drawImage(image, width / 2, height / 2, width, height); 133 | 134 | return () => ({ imageCanvas, imageContext }) 135 | } 136 | 137 | /** 138 | * 创建每一个动画的独立放大曲线 139 | * @return {Function} getImageScaleSize 获得该动画当前尺寸的函数 140 | */ 141 | createImageScaleSize () { 142 | // 完成缩放动画的百分比阶段 9 ~ 11 143 | let minAppearStage = 9; 144 | let maxAppearStage = 11; 145 | 146 | let appearStage = getRandomBetweenInt(minAppearStage, maxAppearStage); 147 | return ( percentage ) => { 148 | return percentage >= appearStage ? 1 : ((parseInt(100 * percentage / appearStage) / 100) || 0.1) 149 | } 150 | } 151 | 152 | /** 153 | * 创建每一个动画的独立透明曲线 154 | * @return {Function} getImageOpacityValue 获得该动画当前透明度的函数 155 | */ 156 | 157 | createImageOpacityValue () { 158 | // 开始淡出动画的百分比阶段 82 ~ 86 159 | let minDisappearStage = 82; 160 | let maxDisappearStage = 86; 161 | 162 | let disappearStage = getRandomBetweenInt(minDisappearStage, maxDisappearStage); 163 | let area = 100 - disappearStage; 164 | return ( percentage ) => { 165 | if (percentage < disappearStage) { 166 | return 1; 167 | } else { 168 | let opacity = ((percentage - disappearStage) / area).toFixed(2) - 0; 169 | return 1 - opacity; 170 | } 171 | } 172 | } 173 | 174 | /** 175 | * 创建每一个动画的独立位移曲线 176 | * @return {Function} getImagePathPosition 获得该动画当前位移值的函数 177 | */ 178 | createImagePathPosition () { 179 | // 以下计算均以 dpr 为 1作为参考值 180 | // 动画初始位置的横向定位的浮动系数 10% 181 | let startX = 10; 182 | // 动画纵向位移距离delta y是总距离的 95% 浮动系数 5% 183 | let deltaY = 0.95; 184 | let deltaYRange = 1 - deltaY; 185 | // 动画横向位移距离符合正弦曲线 正弦波峰最小值10% 最大值15% 186 | let minCrest = 10; 187 | let maxCrest = 15; 188 | // 正弦波长系数 最小值16 最大值32 189 | let minWavelength = 16; 190 | let maxWavelength = 32; 191 | 192 | let DPR = this.dpr; 193 | // @2x 图片要先砍半 194 | let imageWidth = this.image.width / 2; 195 | let imageHeight = this.image.height / 2; 196 | // 画布尺寸根据dpr计算 197 | let canvasWidth = this.canvas.width / DPR; 198 | let canvasHeight = this.canvas.height / DPR; 199 | // console.log(imageWidth, imageHeight, canvasWidth, canvasHeight); 200 | let originX = canvasWidth * (0.5 + getRandomBetweenInt(-startX, startX) / 100); 201 | let originY = canvasHeight; 202 | // 初始值居中是画布总长度减去图片长度后的一半 203 | // 再减去旋转的偏移量 也是图片尺寸的一半 204 | // 刚好减去一倍的图片尺寸 205 | originX -= imageWidth; 206 | originY -= imageHeight; 207 | 208 | let translateX = canvasWidth * getRandomBetweenInt(minCrest, maxCrest) / 100; 209 | let translateY = getRandomByMedian(originY * deltaY, deltaYRange); 210 | 211 | let wavelength = getRandomBetweenInt(minWavelength, maxWavelength); 212 | 213 | // 方向是 -1/1 214 | let direction = getRandomBetweenInt(0, 1) * 2 - 1; 215 | 216 | return ( percentage ) => { 217 | let x = originX + translateX * Math.sin(percentage / wavelength) * direction; 218 | let y = originY - translateY * percentage / 100; 219 | return { x, y } 220 | } 221 | } 222 | } 223 | 224 | export default Like; 225 | -------------------------------------------------------------------------------- /src/js/lib/anim/custom.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @desc 把点赞源码打包成一个js 可以提供给其他团队快速使用 3 | * @version 订制版 开发者自定义点赞图片 4 | * @author lixinliang 5 | * @doc 使用文档等待补充 6 | */ 7 | 8 | import Anim from 'anim'; 9 | import Like from 'anim.like'; 10 | 11 | // 把点赞引擎添加 animjs 上 12 | Anim.setup('like', Like); 13 | // 创建实例 14 | let anim = new Anim('like'); 15 | // 设置尺寸 16 | anim.width(60).height(280); 17 | // 设置资源 18 | let base64 = { 19 | user01 : require('../../../assets/like/heart/user/like01.png'), 20 | user02 : require('../../../assets/like/heart/user/like02.png'), 21 | user03 : require('../../../assets/like/heart/user/like03.png'), 22 | audience01 : require('../../../assets/like/heart/audience/like01.png'), 23 | audience02 : require('../../../assets/like/heart/audience/like02.png'), 24 | audience03 : require('../../../assets/like/heart/audience/like03.png'), 25 | audience04 : require('../../../assets/like/heart/audience/like04.png'), 26 | audience05 : require('../../../assets/like/heart/audience/like05.png'), 27 | audience06 : require('../../../assets/like/heart/audience/like06.png'), 28 | audience07 : require('../../../assets/like/heart/audience/like07.png'), 29 | }; 30 | // 设置资源集合 与 映射分组 31 | anim.mainfest(mainfest()).group(mainfest.group); 32 | 33 | export default anim; 34 | 35 | function mainfest () { 36 | let result = {}; 37 | let user = append.call(result, 'user', 3); 38 | let audience = append.call(result, 'audience', 7); 39 | // let user = result::append('user', 3); 40 | // let audience = result::append('audience', 7); 41 | // let user = mainfest::append('user', 6); 42 | // let audience = user; 43 | mainfest.group = { user, audience }; 44 | return result; 45 | } 46 | 47 | function append ( directory, length ) { 48 | const baseImageUrl = './assets/like/heart/'; 49 | // const baseImageUrl = './assets/like/luhan/'; 50 | const defaultFileName = 'like'; 51 | const defaultFileType = 'png'; 52 | return Array.apply(null, { length }).map((value, index) => fixed(index + 1 + '')).map((id) => { 53 | let key = directory + id; 54 | // this[key] = [baseImageUrl, directory, '/', defaultFileName, id, '.', defaultFileType].join(''); 55 | this[key] = base64[key]; 56 | return key; 57 | }); 58 | } 59 | 60 | function fixed ( value ) { 61 | return value.length == 1 ? '0' + value : value; 62 | } 63 | 64 | export default anim; 65 | -------------------------------------------------------------------------------- /src/js/lib/anim/modules/random.js: -------------------------------------------------------------------------------- 1 | 2 | /** 3 | * 在区间 [min, max] 上的等概率随机整数 4 | * @param {Number} min 最小值 5 | * @param {Number} max 最大值 6 | * @return {Number} value 随机数 7 | */ 8 | let getRandomBetweenInt = function ( min, max ) { 9 | min = parseInt(min) || 0; 10 | max = parseInt(max) || 0; 11 | if (min > max) { 12 | [min, max] = [max, min]; 13 | } 14 | let value = min + Math.round(Math.random() * (max + 1 - min)); 15 | return value == max + 1 ? min : value; 16 | }; 17 | 18 | /** 19 | * 以 value 为中数上下浮动 percentage 的区间上的等概率随机整数 20 | * @param {Number} value 中数 21 | * @param {Number} percentage 浮动系数 22 | * @return {Number} value 随机数 23 | */ 24 | let getRandomByMedian = function ( value, percentage ) { 25 | value = parseInt(value) || 0; 26 | if (percentage < 0) { 27 | percentage = 0; 28 | } 29 | if (percentage > 1) { 30 | percentage = 1; 31 | } 32 | return getRandomBetweenInt(value * (1 - percentage), value * (1 + percentage)); 33 | }; 34 | 35 | /** 36 | * 在区间 [min, max] 上的遵循二项分布的随机整数 其中中数概率最高 37 | * @param {Number} min 最小值 38 | * @param {Number} max 最大值 39 | * @return {Number} value 随机数 40 | */ 41 | let getRandomWithBinomial = function ( min, max ) { 42 | min = parseInt(min) || 0; 43 | max = parseInt(max) || 0; 44 | if (min > max) { 45 | [min, max] = [max, min]; 46 | } 47 | let median = parseInt((max - min) / 2); 48 | return min + getRandomBetweenInt(0, median) + getRandomBetweenInt(0, median); 49 | }; 50 | 51 | export { 52 | getRandomByMedian, 53 | getRandomBetweenInt, 54 | getRandomWithBinomial, 55 | }; 56 | -------------------------------------------------------------------------------- /src/js/lib/anim/modules/ticker.js: -------------------------------------------------------------------------------- 1 | const TICK = 'tick'; 2 | const TICKER = new Event(TICK); 3 | 4 | let elements = []; 5 | 6 | let requestAnimationFrame = window.requestAnimationFrame || window.webkitRequestAnimationFrame || ((fn) => setTimeout(fn, 16)); 7 | 8 | let looper = () => { 9 | elements.forEach((elem) => elem.dispatchEvent(TICKER)); 10 | requestAnimationFrame(looper); 11 | }; 12 | 13 | requestAnimationFrame(looper); 14 | 15 | class Ticker { 16 | constructor () { 17 | this.elem = document.createElement('div'); 18 | elements.push(this.elem); 19 | } 20 | on ( fn ) { 21 | this.elem.addEventListener(TICK, fn, false); 22 | return this 23 | } 24 | off ( fn ) { 25 | this.elem.removeEventListener(TICK, fn, false); 26 | return this 27 | } 28 | } 29 | 30 | export default Ticker; 31 | -------------------------------------------------------------------------------- /src/js/lib/classlist-polyfill/index.js: -------------------------------------------------------------------------------- 1 | /* 2 | * classList.js: Cross-browser full element.classList implementation. 3 | * 1.1.20170427 4 | * 5 | * By Eli Grey, http://eligrey.com 6 | * License: Dedicated to the public domain. 7 | * See https://github.com/eligrey/classList.js/blob/master/LICENSE.md 8 | */ 9 | 10 | /*global self, document, DOMException */ 11 | 12 | /*! @source http://purl.eligrey.com/github/classList.js/blob/master/classList.js */ 13 | 14 | if ("document" in window.self) { 15 | 16 | // Full polyfill for browsers with no classList support 17 | // Including IE < Edge missing SVGElement.classList 18 | if (!("classList" in document.createElement("_")) 19 | || document.createElementNS && !("classList" in document.createElementNS("http://www.w3.org/2000/svg","g"))) { 20 | 21 | (function (view) { 22 | 23 | "use strict"; 24 | 25 | if (!('Element' in view)) return; 26 | 27 | var 28 | classListProp = "classList" 29 | , protoProp = "prototype" 30 | , elemCtrProto = view.Element[protoProp] 31 | , objCtr = Object 32 | , strTrim = String[protoProp].trim || function () { 33 | return this.replace(/^\s+|\s+$/g, ""); 34 | } 35 | , arrIndexOf = Array[protoProp].indexOf || function (item) { 36 | var 37 | i = 0 38 | , len = this.length 39 | ; 40 | for (; i < len; i++) { 41 | if (i in this && this[i] === item) { 42 | return i; 43 | } 44 | } 45 | return -1; 46 | } 47 | // Vendors: please allow content code to instantiate DOMExceptions 48 | , DOMEx = function (type, message) { 49 | this.name = type; 50 | this.code = DOMException[type]; 51 | this.message = message; 52 | } 53 | , checkTokenAndGetIndex = function (classList, token) { 54 | if (token === "") { 55 | throw new DOMEx( 56 | "SYNTAX_ERR" 57 | , "An invalid or illegal string was specified" 58 | ); 59 | } 60 | if (/\s/.test(token)) { 61 | throw new DOMEx( 62 | "INVALID_CHARACTER_ERR" 63 | , "String contains an invalid character" 64 | ); 65 | } 66 | return arrIndexOf.call(classList, token); 67 | } 68 | , ClassList = function (elem) { 69 | var 70 | trimmedClasses = strTrim.call(elem.getAttribute("class") || "") 71 | , classes = trimmedClasses ? trimmedClasses.split(/\s+/) : [] 72 | , i = 0 73 | , len = classes.length 74 | ; 75 | for (; i < len; i++) { 76 | this.push(classes[i]); 77 | } 78 | this._updateClassName = function () { 79 | elem.setAttribute("class", this.toString()); 80 | }; 81 | } 82 | , classListProto = ClassList[protoProp] = [] 83 | , classListGetter = function () { 84 | return new ClassList(this); 85 | } 86 | ; 87 | // Most DOMException implementations don't allow calling DOMException's toString() 88 | // on non-DOMExceptions. Error's toString() is sufficient here. 89 | DOMEx[protoProp] = Error[protoProp]; 90 | classListProto.item = function (i) { 91 | return this[i] || null; 92 | }; 93 | classListProto.contains = function (token) { 94 | token += ""; 95 | return checkTokenAndGetIndex(this, token) !== -1; 96 | }; 97 | classListProto.add = function () { 98 | var 99 | tokens = arguments 100 | , i = 0 101 | , l = tokens.length 102 | , token 103 | , updated = false 104 | ; 105 | do { 106 | token = tokens[i] + ""; 107 | if (checkTokenAndGetIndex(this, token) === -1) { 108 | this.push(token); 109 | updated = true; 110 | } 111 | } 112 | while (++i < l); 113 | 114 | if (updated) { 115 | this._updateClassName(); 116 | } 117 | }; 118 | classListProto.remove = function () { 119 | var 120 | tokens = arguments 121 | , i = 0 122 | , l = tokens.length 123 | , token 124 | , updated = false 125 | , index 126 | ; 127 | do { 128 | token = tokens[i] + ""; 129 | index = checkTokenAndGetIndex(this, token); 130 | while (index !== -1) { 131 | this.splice(index, 1); 132 | updated = true; 133 | index = checkTokenAndGetIndex(this, token); 134 | } 135 | } 136 | while (++i < l); 137 | 138 | if (updated) { 139 | this._updateClassName(); 140 | } 141 | }; 142 | classListProto.toggle = function (token, force) { 143 | token += ""; 144 | 145 | var 146 | result = this.contains(token) 147 | , method = result ? 148 | force !== true && "remove" 149 | : 150 | force !== false && "add" 151 | ; 152 | 153 | if (method) { 154 | this[method](token); 155 | } 156 | 157 | if (force === true || force === false) { 158 | return force; 159 | } else { 160 | return !result; 161 | } 162 | }; 163 | classListProto.toString = function () { 164 | return this.join(" "); 165 | }; 166 | 167 | if (objCtr.defineProperty) { 168 | var classListPropDesc = { 169 | get: classListGetter 170 | , enumerable: true 171 | , configurable: true 172 | }; 173 | try { 174 | objCtr.defineProperty(elemCtrProto, classListProp, classListPropDesc); 175 | } catch (ex) { // IE 8 doesn't support enumerable:true 176 | // adding undefined to fight this issue https://github.com/eligrey/classList.js/issues/36 177 | // modernie IE8-MSW7 machine has IE8 8.0.6001.18702 and is affected 178 | if (ex.number === undefined || ex.number === -0x7FF5EC54) { 179 | classListPropDesc.enumerable = false; 180 | objCtr.defineProperty(elemCtrProto, classListProp, classListPropDesc); 181 | } 182 | } 183 | } else if (objCtr[protoProp].__defineGetter__) { 184 | elemCtrProto.__defineGetter__(classListProp, classListGetter); 185 | } 186 | 187 | }(window.self)); 188 | 189 | } 190 | 191 | // There is full or partial native classList support, so just check if we need 192 | // to normalize the add/remove and toggle APIs. 193 | 194 | (function () { 195 | "use strict"; 196 | 197 | var testElement = document.createElement("_"); 198 | 199 | testElement.classList.add("c1", "c2"); 200 | 201 | // Polyfill for IE 10/11 and Firefox <26, where classList.add and 202 | // classList.remove exist but support only one argument at a time. 203 | if (!testElement.classList.contains("c2")) { 204 | var createMethod = function(method) { 205 | var original = DOMTokenList.prototype[method]; 206 | 207 | DOMTokenList.prototype[method] = function(token) { 208 | var i, len = arguments.length; 209 | 210 | for (i = 0; i < len; i++) { 211 | token = arguments[i]; 212 | original.call(this, token); 213 | } 214 | }; 215 | }; 216 | createMethod('add'); 217 | createMethod('remove'); 218 | } 219 | 220 | testElement.classList.toggle("c3", false); 221 | 222 | // Polyfill for IE 10 and Firefox <24, where classList.toggle does not 223 | // support the second argument. 224 | if (testElement.classList.contains("c3")) { 225 | var _toggle = DOMTokenList.prototype.toggle; 226 | 227 | DOMTokenList.prototype.toggle = function(token, force) { 228 | if (1 in arguments && !this.contains(token) === !force) { 229 | return force; 230 | } else { 231 | return _toggle.call(this, token); 232 | } 233 | }; 234 | 235 | } 236 | 237 | testElement = null; 238 | }()); 239 | 240 | } 241 | -------------------------------------------------------------------------------- /src/js/lib/delegate/index.js: -------------------------------------------------------------------------------- 1 | if (!Element.prototype.matches) { 2 | Element.prototype.matches = 3 | Element.prototype.matchesSelector || 4 | Element.prototype.webkitMatchesSelector || 5 | function(s) { 6 | var matches = (this.document || this.ownerDocument).querySelectorAll(s), 7 | i = matches.length; 8 | while (--i >= 0 && matches.item(i) !== this) {} 9 | return i > -1; 10 | }; 11 | } 12 | 13 | export default function delegate ( selector, handler, counter ) { 14 | if (counter) { 15 | let callback = function ( event ) { 16 | for (let target = event.target; target && target != this; target = target.parentNode) { 17 | if (target.matches(selector)) { 18 | counter--; 19 | if (!counter) { 20 | this.removeEventListener(event.type, callback, false); 21 | } 22 | return handler.apply(target, arguments); 23 | } 24 | } 25 | }; 26 | return callback; 27 | } 28 | return function ( event ) { 29 | for (let target = event.target; target && target != this; target = target.parentNode) { 30 | if (target.matches(selector)) { 31 | return handler.apply(target, arguments); 32 | } 33 | } 34 | }; 35 | }; 36 | -------------------------------------------------------------------------------- /src/js/lib/event/index.js: -------------------------------------------------------------------------------- 1 | export default class Event { 2 | constructor() { 3 | this.handlers = {}; 4 | } 5 | on ( type, handler ) { 6 | let handlers = this.handlers; 7 | if (handlers[type]) { 8 | handlers[type].push(handler); 9 | } else { 10 | handlers[type] = [handler]; 11 | } 12 | return this; 13 | } 14 | trigger ( type, callback ) { 15 | let handlers = this.handlers; 16 | if (handlers[type] instanceof Array) { 17 | handlers[type].forEach(( handler ) => { 18 | callback(handler); 19 | }); 20 | } 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /src/js/lib/jsonp/index.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | var JSONP, computedUrl, createElement, encode, noop, objectToURI, random, randomString; 3 | 4 | createElement = function(tag) { 5 | return window.document.createElement(tag); 6 | }; 7 | 8 | encode = window.encodeURIComponent; 9 | 10 | random = Math.random; 11 | 12 | JSONP = function(options) { 13 | var callback, callbackFunc, callbackName, done, head, params, script; 14 | if (options == null) { 15 | options = {}; 16 | } 17 | params = { 18 | data: options.data || {}, 19 | error: options.error || noop, 20 | success: options.success || noop, 21 | beforeSend: options.beforeSend || noop, 22 | complete: options.complete || noop, 23 | url: options.url || '' 24 | }; 25 | params.computedUrl = computedUrl(params); 26 | if (params.url.length === 0) { 27 | throw new Error('MissingUrl'); 28 | } 29 | done = false; 30 | if (params.beforeSend({}, params) !== false) { 31 | callbackName = options.callbackName || 'callback'; 32 | callbackFunc = options.callbackFunc || 'jsonp_' + randomString(15); 33 | callback = params.data[callbackName] = callbackFunc; 34 | window[callback] = function(data) { 35 | window[callback] = null; 36 | params.success(data, params); 37 | return params.complete(data, params); 38 | }; 39 | script = createElement('script'); 40 | script.src = computedUrl(params); 41 | script.async = true; 42 | 43 | script.onerror = function(evt) { 44 | params.error({ 45 | url: script.src, 46 | event: evt 47 | }); 48 | return params.complete({ 49 | url: script.src, 50 | event: evt 51 | }, params); 52 | }; 53 | script.onload = script.onreadystatechange = function() { 54 | var ref, ref1; 55 | 56 | if (done) { 57 | return; 58 | } 59 | done = true; 60 | if (script) { 61 | script.onload = script.onreadystatechange = null; 62 | if ((ref1 = script.parentNode) != null) { 63 | ref1.removeChild(script); 64 | } 65 | return script = null; 66 | } 67 | }; 68 | head = window.document.getElementsByTagName('head')[0] || window.document.documentElement; 69 | head.insertBefore(script, head.firstChild); 70 | } 71 | return { 72 | abort: function() { 73 | window[callback] = function() { 74 | return window[callback] = null; 75 | }; 76 | done = true; 77 | if (script != null ? script.parentNode : void 0) { 78 | script.onload = script.onreadystatechange = null; 79 | script.parentNode.removeChild(script); 80 | return script = null; 81 | } 82 | } 83 | }; 84 | }; 85 | 86 | noop = function() { 87 | return void 0; 88 | }; 89 | 90 | computedUrl = function(params) { 91 | var url; 92 | url = params.url; 93 | url += params.url.indexOf('?') < 0 ? '?' : '&'; 94 | url += objectToURI(params.data); 95 | return url; 96 | }; 97 | 98 | randomString = function(length) { 99 | var str; 100 | str = ''; 101 | while (str.length < length) { 102 | str += random().toString(36).slice(2, 3); 103 | } 104 | return str; 105 | }; 106 | 107 | objectToURI = function(obj) { 108 | var data, key, value; 109 | data = (function() { 110 | var results; 111 | results = []; 112 | for (key in obj) { 113 | value = obj[key]; 114 | results.push(encode(key) + '=' + encode(value)); 115 | } 116 | return results; 117 | })(); 118 | return data.join('&'); 119 | }; 120 | 121 | if (typeof define !== "undefined" && define !== null ? define.amd : void 0) { 122 | define(function() { 123 | return JSONP; 124 | }); 125 | } else if (typeof module !== "undefined" && module !== null ? module.exports : void 0) { 126 | module.exports = JSONP; 127 | } else { 128 | this.JSONP = JSONP; 129 | } 130 | 131 | }).call(this); 132 | -------------------------------------------------------------------------------- /src/js/lib/lazy-load-img/index.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * version: 3.0.6 3 | * github:https://github.com/lzxb/lazy-load-img 4 | */ 5 | 6 | (function (global, factory) { 7 | typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : 8 | typeof define === 'function' && define.amd ? define(factory) : 9 | (global.LazyLoadImg = factory()); 10 | }(this, (function () { 'use strict'; 11 | 12 | var testMeet = function (el) { 13 | var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; 14 | 15 | // 取得元素在可视区的位置(相对浏览器视窗)左右上下 16 | var bcr = el.getBoundingClientRect 17 | // padding+border+width 18 | ();var mw = el.offsetWidth; // 元素自身宽度 19 | var mh = el.offsetHeight; // 元素自身的高度 20 | // 包含了导航栏 21 | var w = window.innerWidth; // 视窗的宽度 22 | var h = window.innerHeight; // 视窗的高度 23 | 24 | var boolX = !(bcr.right - options.left <= 0 && bcr.left + mw - options.left <= 0) && !(bcr.left + options.right >= w && bcr.right + options.right >= mw + w); // 左右符合条件 25 | var boolY = !(bcr.bottom - options.top <= 0 && bcr.top + mh - options.top <= 0) && !(bcr.top + options.bottom >= h && bcr.bottom + options.bottom >= mh + h); // 上下符合条件 26 | return el.width !== 0 && el.height !== 0 && boolX && boolY; 27 | }; 28 | 29 | var canvas = document.createElement('canvas'); 30 | canvas.getContext('2d').globalAlpha = 0.0; 31 | var images = {}; 32 | 33 | var getTransparent = function (src, w, h) { 34 | if (images[src]) return images[src]; 35 | canvas.width = w; 36 | canvas.height = h; 37 | var data = canvas.toDataURL('image/png'); 38 | images[src] = data; 39 | return data; 40 | }; 41 | 42 | var classCallCheck = function (instance, Constructor) { 43 | if (!(instance instanceof Constructor)) { 44 | throw new TypeError("Cannot call a class as a function"); 45 | } 46 | }; 47 | 48 | var createClass = function () { 49 | function defineProperties(target, props) { 50 | for (var i = 0; i < props.length; i++) { 51 | var descriptor = props[i]; 52 | descriptor.enumerable = descriptor.enumerable || false; 53 | descriptor.configurable = true; 54 | if ("value" in descriptor) descriptor.writable = true; 55 | Object.defineProperty(target, descriptor.key, descriptor); 56 | } 57 | } 58 | 59 | return function (Constructor, protoProps, staticProps) { 60 | if (protoProps) defineProperties(Constructor.prototype, protoProps); 61 | if (staticProps) defineProperties(Constructor, staticProps); 62 | return Constructor; 63 | }; 64 | }(); 65 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | var _extends = Object.assign || function (target) { 73 | for (var i = 1; i < arguments.length; i++) { 74 | var source = arguments[i]; 75 | 76 | for (var key in source) { 77 | if (Object.prototype.hasOwnProperty.call(source, key)) { 78 | target[key] = source[key]; 79 | } 80 | } 81 | } 82 | 83 | return target; 84 | }; 85 | 86 | var _win = window; 87 | 88 | var LazyLoadImg = function () { 89 | // 构造函数 初始化参数 90 | function LazyLoadImg() { 91 | var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; 92 | classCallCheck(this, LazyLoadImg); 93 | 94 | this.options = { 95 | el: document.querySelector('body'), // 选择的元素 96 | mode: 'default', // 默认模式,将显示原图,diy模式,将自定义剪切,默认剪切居中部分 97 | time: 300, // 设置一个检测时间间隔 98 | done: true, // 页面内所有数据图片加载完成后,是否自己销毁程序,true默认销毁,false不销毁:FALSE应用场景:页面异步不断获取数据的情况下 需要实时监听则不销毁 99 | diy: { // 此属性,只有在设置diy 模式时才生效 100 | backgroundSize: 'cover', 101 | backgroundRepeat: 'no-repeat', 102 | backgroundPosition: 'center center' 103 | }, 104 | position: { // 只要其中一个位置符合条件,都会触发加载机制 105 | top: 0, // 元素距离顶部 106 | right: 0, // 元素距离右边 107 | bottom: 0, // 元素距离下面 108 | left: 0 // 元素距离左边 109 | }, 110 | before: function before(el) {// 图片加载之前,执行钩子函数 111 | 112 | }, 113 | success: function success(el) {// 图片加载成功,执行钩子函数 114 | 115 | }, 116 | error: function error(el) {// 图片加载失败,执行的钩子函数 117 | 118 | } 119 | }; 120 | // 深拷贝 如果都有 则右面的值 option.position会覆盖this.options.position 121 | options.position = _extends({}, this.options.position, options.position); 122 | options.diy = _extends({}, this.options.diy, options.diy); 123 | _extends(this.options, options); 124 | this.start(); 125 | } 126 | 127 | createClass(LazyLoadImg, [{ 128 | key: 'start', 129 | value: function start() { 130 | this._timer = true; 131 | this._start(); 132 | } 133 | }, { 134 | key: '_start', 135 | value: function _start() { 136 | var _this = this; 137 | 138 | var options = this.options; 139 | 140 | clearTimeout(this._timer // 清除定时器 141 | );if (!this._timer) return; 142 | this._timer = setTimeout(function () { 143 | var list = Array.prototype.slice.apply(options.el.querySelectorAll('[data-src]')); 144 | if (!list.length && options.done) return clearTimeout(_this._timer); 145 | list.forEach(function (el) { 146 | if (!el.dataset.LazyLoadImgState && testMeet(el, options.position)) { 147 | _this.loadImg(el); 148 | } 149 | }); 150 | _this._start(); 151 | }, options.time); 152 | } 153 | }, { 154 | key: 'loadImg', 155 | value: function loadImg(el) { 156 | var _this2 = this; 157 | 158 | // 加载图片 159 | var options = this.options; 160 | 161 | el.dataset.LazyLoadImgState = 'start'; 162 | options.before.call(this, el); 163 | var img = new _win.Image(); 164 | img.src = el.dataset.src; 165 | // 图片加载成功 166 | img.addEventListener('load', function () { 167 | if (options.mode === 'diy') { 168 | el.src = getTransparent(el.src, el.width, el.height); 169 | options.diy.backgroundImage = 'url(' + img.src + ')'; 170 | _extends(el.style, options.diy); 171 | } else { 172 | el.src = img.src; 173 | } 174 | delete el.dataset.src; 175 | delete el.dataset.LazyLoadImgState; 176 | return options.success.call(_this2, el); 177 | }, false 178 | 179 | // 图片加载失败 180 | );img.addEventListener('error', function () { 181 | delete el.dataset.src; 182 | delete el.dataset.LazyLoadImgState; 183 | options.error.call(_this2, el); 184 | }, false); 185 | } 186 | }, { 187 | key: 'destroy', 188 | value: function destroy() { 189 | // 解除事件绑定 190 | delete this._timer; 191 | } 192 | }]); 193 | return LazyLoadImg; 194 | }(); 195 | 196 | return LazyLoadImg; 197 | 198 | }))); 199 | 200 | -------------------------------------------------------------------------------- /src/js/lib/queue/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @component 队列 3 | * @author wilson 4 | */ 5 | 6 | (function () { 7 | 8 | var queue = {}; 9 | function noop() {} 10 | 11 | var previous_async; 12 | 13 | var root = typeof self === 'object' && self.self === self && self || 14 | typeof global === 'object' && global.global === global && global || 15 | this; 16 | 17 | if (root != null) { 18 | previous_async = root.queue; 19 | } 20 | 21 | function only_once(fn) { 22 | return function() { 23 | if (fn === null) throw new Error("Callback was already called."); 24 | fn.apply(this, arguments); 25 | fn = null; 26 | }; 27 | } 28 | 29 | var _toString = Object.prototype.toString; 30 | 31 | var _isArray = Array.isArray || function (obj) { 32 | return _toString.call(obj) === '[object Array]'; 33 | }; 34 | 35 | function _arrayEach(arr, iterator) { 36 | var index = -1, 37 | length = arr.length; 38 | 39 | while (++index < length) { 40 | iterator(arr[index], index, arr); 41 | } 42 | } 43 | 44 | function _map(arr, iterator) { 45 | var index = -1, 46 | length = arr.length, 47 | result = Array(length); 48 | 49 | while (++index < length) { 50 | result[index] = iterator(arr[index], index, arr); 51 | } 52 | return result; 53 | } 54 | 55 | var _setImmediate = typeof setImmediate === 'function' && setImmediate; 56 | 57 | var _delay = _setImmediate ? function(fn) { 58 | // not a direct alias for IE10 compatibility 59 | _setImmediate(fn); 60 | } : function(fn) { 61 | setTimeout(fn, 0); 62 | }; 63 | 64 | queue.setImmediate = _delay; 65 | 66 | function _queue(worker, concurrency, payload) { 67 | if (concurrency == null) { 68 | concurrency = 1; 69 | } 70 | else if(concurrency === 0) { 71 | throw new Error('Concurrency must not be zero'); 72 | } 73 | function _insert(q, data, pos, callback) { 74 | if (callback != null && typeof callback !== "function") { 75 | throw new Error("task callback must be a function"); 76 | } 77 | q.started = true; 78 | if (!_isArray(data)) { 79 | data = [data]; 80 | } 81 | if(data.length === 0 && q.idle()) { 82 | // call drain immediately if there are no tasks 83 | return queue.setImmediate(function() { 84 | q.drain(); 85 | }); 86 | } 87 | _arrayEach(data, function(task) { 88 | var item = { 89 | data: task, 90 | callback: callback || noop 91 | }; 92 | 93 | if (pos) { 94 | q.tasks.unshift(item); 95 | } else { 96 | q.tasks.push(item); 97 | } 98 | 99 | if (q.tasks.length === q.concurrency) { 100 | q.saturated(); 101 | } 102 | }); 103 | queue.setImmediate(q.process); 104 | } 105 | function _next(q, tasks) { 106 | return function(){ 107 | workers -= 1; 108 | 109 | var removed = false; 110 | var args = arguments; 111 | _arrayEach(tasks, function (task) { 112 | _arrayEach(workersList, function (worker, index) { 113 | if (worker === task && !removed) { 114 | workersList.splice(index, 1); 115 | removed = true; 116 | } 117 | }); 118 | 119 | task.callback.apply(task, args); 120 | }); 121 | if (q.tasks.length + workers === 0) { 122 | q.drain(); 123 | } 124 | q.process(); 125 | }; 126 | } 127 | 128 | var workers = 0; 129 | var workersList = []; 130 | var q = { 131 | tasks: [], 132 | concurrency: concurrency, 133 | payload: payload, 134 | saturated: noop, 135 | empty: noop, 136 | drain: noop, 137 | started: false, 138 | paused: false, 139 | push: function (data, callback) { 140 | _insert(q, data, false, callback); 141 | }, 142 | unshift: function (data, callback) { 143 | _insert(q, data, true, callback); 144 | }, 145 | process: function () { 146 | while(!q.paused && workers < q.concurrency && q.tasks.length){ 147 | 148 | var tasks = q.payload ? 149 | q.tasks.splice(0, q.payload) : 150 | q.tasks.splice(0, q.tasks.length); 151 | 152 | var data = _map(tasks, function (task) { 153 | return task.data; 154 | }); 155 | 156 | if (q.tasks.length === 0) { 157 | q.empty(); 158 | } 159 | workers += 1; 160 | workersList.push(tasks[0]); 161 | var cb = only_once(_next(q, tasks)); 162 | worker(data, cb); 163 | } 164 | }, 165 | length: function () { 166 | return q.tasks.length; 167 | }, 168 | running: function () { 169 | return workers; 170 | }, 171 | workersList: function () { 172 | return workersList; 173 | }, 174 | idle: function() { 175 | return q.tasks.length + workers === 0; 176 | }, 177 | pause: function () { 178 | q.paused = true; 179 | }, 180 | resume: function () { 181 | if (q.paused === false) { return; } 182 | q.paused = false; 183 | var resumeCount = Math.min(q.concurrency, q.tasks.length); 184 | // Need to call q.process once per concurrent 185 | // worker to preserve full concurrency after pause 186 | for (var w = 1; w <= resumeCount; w++) { 187 | queue.setImmediate(q.process); 188 | } 189 | } 190 | }; 191 | return q; 192 | } 193 | 194 | queue.queue = function (worker, concurrency) { 195 | var q = _queue(function (items, cb) { 196 | worker(items[0], cb); 197 | }, concurrency, 1); 198 | 199 | return q; 200 | }; 201 | 202 | module.exports = queue; 203 | 204 | }()); 205 | -------------------------------------------------------------------------------- /src/js/lib/scrollTo/index.js: -------------------------------------------------------------------------------- 1 | export default function scrollTo ( direction, config ) { 2 | let { elem, speed = 20, interval = 16, transition = true } = config; 3 | return new Promise(( resolve ) => { 4 | if (transition) { 5 | let scrolling = true; 6 | let now = elem.scrollTop; 7 | let next = () => { 8 | let destination; 9 | if (direction == 'top') { 10 | destination = 0; 11 | now -= speed; 12 | if (now < destination) { 13 | now = destination; 14 | scrolling = false; 15 | } 16 | } else { 17 | destination = elem.scrollHeight; 18 | now += speed; 19 | if (now > destination) { 20 | now = destination; 21 | scrolling = false; 22 | } 23 | } 24 | elem.scrollTop = now; 25 | if (scrolling) { 26 | setTimeout(next, interval); 27 | } else { 28 | resolve(); 29 | } 30 | }; 31 | next(); 32 | } else { 33 | if (direction == 'top') { 34 | elem.scrollTop = 0; 35 | } else { 36 | elem.scrollTop = elem.scrollHeight; 37 | } 38 | resolve(); 39 | } 40 | }); 41 | } 42 | -------------------------------------------------------------------------------- /src/js/lib/sprite/index.js: -------------------------------------------------------------------------------- 1 | // 从时序图尺寸 动态生成动画函数 2 | 3 | let record = {}; 4 | 5 | export default function ( elem, src, duration ) { 6 | return new Promise(( resolve ) => { 7 | let img = new Image; 8 | img.onload = onload; 9 | img.src = src; 10 | elem.dataset.status = 'pause'; 11 | function onload () { 12 | let frame; 13 | let direction; 14 | if (this.width > this.height) { 15 | direction = 'x'; 16 | frame = Math.round(this.width / this.height); 17 | // elem.style['background-size'] = `auto ${ getComputedStyle(elem).width }`; 18 | elem.style['background-size'] = `auto 100%`; 19 | } else { 20 | direction = 'y'; 21 | frame = Math.round(this.height / this.width); 22 | // elem.style['background-size'] = `${ getComputedStyle(elem).height } auto`; 23 | elem.style['background-size'] = `100% auto`; 24 | } 25 | elem.dataset.frame = frame; 26 | elem.dataset.direction = direction; 27 | elem.style['background-image'] = `url(${ src })`; 28 | let unique = direction + frame; 29 | if (!record[unique]) { 30 | record[unique] = true; 31 | let style = document.createElement('style'); 32 | style.innerHTML = ` 33 | [data-status="play"][data-frame="${ frame }"][data-direction="${ direction }"] { 34 | animation-duration: ${ duration * frame }ms; 35 | -webkit-animation-duration: ${ duration * frame }ms; 36 | animation-name: spirit-animation-${ unique }; 37 | -webkit-animation-name: spirit-animation-${ unique }; 38 | animation-timing-function: steps(${ frame }); 39 | -webkit-animation-timing-function: steps(${ frame }); 40 | animation-iteration-count: infinite; 41 | -webkit-animation-iteration-count: infinite; 42 | } 43 | @keyframes spirit-animation-${ unique } { 44 | 0% { 45 | background-position: 0 0; 46 | } 47 | 100% { 48 | background-position: ${ direction == 'x' ? `-${ frame }00% 0` : `0 -${ frame }00%` }; 49 | } 50 | } 51 | @-webkit-keyframes spirit-animation-${ unique } { 52 | 0% { 53 | background-position: 0 0; 54 | } 55 | 100% { 56 | background-position: ${ direction == 'x' ? `-${ frame }00% 0` : `0 -${ frame }00%` }; 57 | } 58 | } 59 | `; 60 | document.head.appendChild(style); 61 | } 62 | resolve(); 63 | } 64 | }); 65 | }; 66 | -------------------------------------------------------------------------------- /src/js/lib/svga/svga-db.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * @project : svga-web-canvas 3 | * @version : 0.0.1 4 | * @author : UED.lijialiang 5 | * @update : 2016-10-21 9:35:38 am 6 | * @email : lijialiang@yy.com 7 | */ !function(e,t){"object"==typeof exports&&"object"==typeof module?module.exports=t():"function"==typeof define&&define.amd?define([],t):"object"==typeof exports?exports.SvgaDb=t():e.SvgaDb=t()}(this,function(){return function(e){function t(a){if(n[a])return n[a].exports;var r=n[a]={exports:{},id:a,loaded:!1};return e[a].call(r.exports,r,r.exports,t),r.loaded=!0,r.exports}var n={};return t.m=e,t.c=n,t.p="",t(0)}([function(e,t){function n(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}var a,r,o=function(){function e(e,t){for(var n=0;n0?n(JSON.parse(t.rows.item(0).images),JSON.parse(t.rows.item(0).movie),!1):n("","",!0)},a._errorCallBack)})}},{key:"clear",value:function(e){"string"==typeof e?this._clearByUrl(e):"number"==typeof e&&this._clearByDate(e)}},{key:"_clearByUrl",value:function(t){var n=this;e.db.transaction(function(a){a.executeSql("DELETE FROM "+e.table.name+" WHERE url LIKE ?",[t],function(e,t){console.info("[Svga Web Canvas DB]: clear success")},n._errorCallBack)})}},{key:"_clearByDate",value:function(t){var n=this,a=new Date;e.db.transaction(function(r){r.executeSql("SELECT * FROM "+e.table.name,[],function(e,r){for(var o=r.rows.length,i=0;i=t&&n._clearByUrl(c.url)}},n._errorCallBack)})}}]),e}(),a.table={name:"svga",attribute:"url, movie, images, time"},r)}])}); -------------------------------------------------------------------------------- /src/js/lib/swipe/index.js: -------------------------------------------------------------------------------- 1 | /* 2 | * Swipe 2.0 3 | * 4 | * Brad Birdsall 5 | * Copyright 2013, MIT License 6 | * 7 | */ 8 | 9 | function Swipe(container, options) { 10 | 11 | "use strict"; 12 | 13 | // utilities 14 | var noop = function() {}; // simple no operation function 15 | var offloadFn = function(fn) { setTimeout(fn || noop, 0) }; // offload a functions execution 16 | 17 | // check browser capabilities 18 | var browser = { 19 | addEventListener: !!window.addEventListener, 20 | touch: ('ontouchstart' in window) || window.DocumentTouch && document instanceof DocumentTouch, 21 | transitions: (function(temp) { 22 | var props = ['transitionProperty', 'WebkitTransition', 'MozTransition', 'OTransition', 'msTransition']; 23 | for ( var i in props ) if (temp.style[ props[i] ] !== undefined) return true; 24 | return false; 25 | })(document.createElement('swipe')) 26 | }; 27 | 28 | // quit if no root element 29 | if (!container) return; 30 | var element = container.children[0]; 31 | var slides, slidePos, width, length; 32 | options = options || {}; 33 | var index = parseInt(options.startSlide, 10) || 0; 34 | var speed = options.speed || 300; 35 | options.continuous = options.continuous !== undefined ? options.continuous : true; 36 | 37 | function setup() { 38 | 39 | // cache slides 40 | slides = element.children; 41 | length = slides.length; 42 | 43 | // set continuous to false if only one slide 44 | if (slides.length < 2) options.continuous = false; 45 | 46 | //special case if two slides 47 | if (browser.transitions && options.continuous && slides.length < 3) { 48 | element.appendChild(slides[0].cloneNode(true)); 49 | element.appendChild(element.children[1].cloneNode(true)); 50 | slides = element.children; 51 | } 52 | 53 | // create an array to store current positions of each slide 54 | slidePos = new Array(slides.length); 55 | 56 | // determine width of each slide 57 | width = container.getBoundingClientRect().width || container.offsetWidth; 58 | 59 | element.style.width = (slides.length * width) + 'px'; 60 | 61 | // stack elements 62 | var pos = slides.length; 63 | while(pos--) { 64 | 65 | var slide = slides[pos]; 66 | 67 | slide.style.width = width + 'px'; 68 | slide.setAttribute('data-index', pos); 69 | 70 | if (browser.transitions) { 71 | slide.style.left = (pos * -width) + 'px'; 72 | move(pos, index > pos ? -width : (index < pos ? width : 0), 0); 73 | } 74 | 75 | } 76 | 77 | // reposition elements before and after index 78 | if (options.continuous && browser.transitions) { 79 | move(circle(index-1), -width, 0); 80 | move(circle(index+1), width, 0); 81 | } 82 | 83 | if (!browser.transitions) element.style.left = (index * -width) + 'px'; 84 | 85 | container.style.visibility = 'visible'; 86 | 87 | } 88 | 89 | function prev() { 90 | 91 | if (options.continuous) slide(index-1); 92 | else if (index) slide(index-1); 93 | 94 | } 95 | 96 | function next() { 97 | 98 | if (options.continuous) slide(index+1); 99 | else if (index < slides.length - 1) slide(index+1); 100 | 101 | } 102 | 103 | function circle(index) { 104 | 105 | // a simple positive modulo using slides.length 106 | return (slides.length + (index % slides.length)) % slides.length; 107 | 108 | } 109 | 110 | function slide(to, slideSpeed) { 111 | 112 | // do nothing if already on requested slide 113 | if (index == to) return; 114 | 115 | if (browser.transitions) { 116 | 117 | var direction = Math.abs(index-to) / (index-to); // 1: backward, -1: forward 118 | 119 | // get the actual position of the slide 120 | if (options.continuous) { 121 | var natural_direction = direction; 122 | direction = -slidePos[circle(to)] / width; 123 | 124 | // if going forward but to < index, use to = slides.length + to 125 | // if going backward but to > index, use to = -slides.length + to 126 | if (direction !== natural_direction) to = -direction * slides.length + to; 127 | 128 | } 129 | 130 | var diff = Math.abs(index-to) - 1; 131 | 132 | // move all the slides between index and to in the right direction 133 | while (diff--) move( circle((to > index ? to : index) - diff - 1), width * direction, 0); 134 | 135 | to = circle(to); 136 | 137 | move(index, width * direction, slideSpeed || speed); 138 | move(to, 0, slideSpeed || speed); 139 | 140 | if (options.continuous) move(circle(to - direction), -(width * direction), 0); // we need to get the next in place 141 | 142 | } else { 143 | 144 | to = circle(to); 145 | animate(index * -width, to * -width, slideSpeed || speed); 146 | //no fallback for a circular continuous if the browser does not accept transitions 147 | } 148 | 149 | index = to; 150 | offloadFn(options.callback && options.callback(index, slides[index])); 151 | } 152 | 153 | function move(index, dist, speed) { 154 | 155 | translate(index, dist, speed); 156 | slidePos[index] = dist; 157 | 158 | } 159 | 160 | function translate(index, dist, speed) { 161 | 162 | var slide = slides[index]; 163 | var style = slide && slide.style; 164 | 165 | if (!style) return; 166 | 167 | style.webkitTransitionDuration = 168 | style.MozTransitionDuration = 169 | style.msTransitionDuration = 170 | style.OTransitionDuration = 171 | style.transitionDuration = speed + 'ms'; 172 | 173 | style.webkitTransform = 'translate(' + dist + 'px,0)' + 'translateZ(0)'; 174 | style.msTransform = 175 | style.MozTransform = 176 | style.OTransform = 'translateX(' + dist + 'px)'; 177 | 178 | } 179 | 180 | function animate(from, to, speed) { 181 | 182 | // if not an animation, just reposition 183 | if (!speed) { 184 | 185 | element.style.left = to + 'px'; 186 | return; 187 | 188 | } 189 | 190 | var start = +new Date; 191 | 192 | var timer = setInterval(function() { 193 | 194 | var timeElap = +new Date - start; 195 | 196 | if (timeElap > speed) { 197 | 198 | element.style.left = to + 'px'; 199 | 200 | if (delay) begin(); 201 | 202 | options.transitionEnd && options.transitionEnd.call(event, index, slides[index]); 203 | 204 | clearInterval(timer); 205 | return; 206 | 207 | } 208 | 209 | element.style.left = (( (to - from) * (Math.floor((timeElap / speed) * 100) / 100) ) + from) + 'px'; 210 | 211 | }, 4); 212 | 213 | } 214 | 215 | // setup auto slideshow 216 | var delay = options.auto || 0; 217 | var interval; 218 | 219 | function begin() { 220 | 221 | interval = setTimeout(next, delay); 222 | 223 | } 224 | 225 | function stop() { 226 | 227 | delay = 0; 228 | clearTimeout(interval); 229 | 230 | } 231 | 232 | 233 | // setup initial vars 234 | var start = {}; 235 | var delta = {}; 236 | var isScrolling; 237 | 238 | // setup event capturing 239 | var events = { 240 | 241 | handleEvent: function(event) { 242 | 243 | switch (event.type) { 244 | case 'touchstart': this.start(event); break; 245 | case 'touchmove': this.move(event); break; 246 | case 'touchend': offloadFn(this.end(event)); break; 247 | case 'webkitTransitionEnd': 248 | case 'msTransitionEnd': 249 | case 'oTransitionEnd': 250 | case 'otransitionend': 251 | case 'transitionend': offloadFn(this.transitionEnd(event)); break; 252 | case 'resize': offloadFn(setup); break; 253 | } 254 | 255 | if (options.stopPropagation) event.stopPropagation(); 256 | 257 | }, 258 | start: function(event) { 259 | 260 | var touches = event.touches[0]; 261 | 262 | // measure start values 263 | start = { 264 | 265 | // get initial touch coords 266 | x: touches.pageX, 267 | y: touches.pageY, 268 | 269 | // store time to determine touch duration 270 | time: +new Date 271 | 272 | }; 273 | 274 | // used for testing first move event 275 | isScrolling = undefined; 276 | 277 | // reset delta and end measurements 278 | delta = {}; 279 | 280 | // attach touchmove and touchend listeners 281 | element.addEventListener('touchmove', this, false); 282 | element.addEventListener('touchend', this, false); 283 | 284 | }, 285 | move: function(event) { 286 | 287 | // ensure swiping with one touch and not pinching 288 | if ( event.touches.length > 1 || event.scale && event.scale !== 1) return 289 | 290 | if (options.disableScroll) event.preventDefault(); 291 | 292 | var touches = event.touches[0]; 293 | 294 | // measure change in x and y 295 | delta = { 296 | x: touches.pageX - start.x, 297 | y: touches.pageY - start.y 298 | } 299 | 300 | // determine if scrolling test has run - one time test 301 | if ( typeof isScrolling == 'undefined') { 302 | isScrolling = !!( isScrolling || Math.abs(delta.x) < Math.abs(delta.y) ); 303 | } 304 | 305 | // if user is not trying to scroll vertically 306 | if (!isScrolling) { 307 | 308 | // prevent native scrolling 309 | event.preventDefault(); 310 | 311 | // stop slideshow 312 | stop(); 313 | 314 | // increase resistance if first or last slide 315 | if (options.continuous) { // we don't add resistance at the end 316 | 317 | translate(circle(index-1), delta.x + slidePos[circle(index-1)], 0); 318 | translate(index, delta.x + slidePos[index], 0); 319 | translate(circle(index+1), delta.x + slidePos[circle(index+1)], 0); 320 | 321 | } else { 322 | 323 | delta.x = 324 | delta.x / 325 | ( (!index && delta.x > 0 // if first slide and sliding left 326 | || index == slides.length - 1 // or if last slide and sliding right 327 | && delta.x < 0 // and if sliding at all 328 | ) ? 329 | ( Math.abs(delta.x) / width + 1 ) // determine resistance level 330 | : 1 ); // no resistance if false 331 | 332 | // translate 1:1 333 | translate(index-1, delta.x + slidePos[index-1], 0); 334 | translate(index, delta.x + slidePos[index], 0); 335 | translate(index+1, delta.x + slidePos[index+1], 0); 336 | } 337 | 338 | } 339 | 340 | }, 341 | end: function(event) { 342 | 343 | // measure duration 344 | var duration = +new Date - start.time; 345 | 346 | // determine if slide attempt triggers next/prev slide 347 | var isValidSlide = 348 | Number(duration) < 250 // if slide duration is less than 250ms 349 | && Math.abs(delta.x) > 20 // and if slide amt is greater than 20px 350 | || Math.abs(delta.x) > width/2; // or if slide amt is greater than half the width 351 | 352 | // determine if slide attempt is past start and end 353 | var isPastBounds = 354 | !index && delta.x > 0 // if first slide and slide amt is greater than 0 355 | || index == slides.length - 1 && delta.x < 0; // or if last slide and slide amt is less than 0 356 | 357 | if (options.continuous) isPastBounds = false; 358 | 359 | // determine direction of swipe (true:right, false:left) 360 | var direction = delta.x < 0; 361 | 362 | // if not scrolling vertically 363 | if (!isScrolling) { 364 | 365 | if (isValidSlide && !isPastBounds) { 366 | 367 | if (direction) { 368 | 369 | if (options.continuous) { // we need to get the next in this direction in place 370 | 371 | move(circle(index-1), -width, 0); 372 | move(circle(index+2), width, 0); 373 | 374 | } else { 375 | move(index-1, -width, 0); 376 | } 377 | 378 | move(index, slidePos[index]-width, speed); 379 | move(circle(index+1), slidePos[circle(index+1)]-width, speed); 380 | index = circle(index+1); 381 | 382 | } else { 383 | if (options.continuous) { // we need to get the next in this direction in place 384 | 385 | move(circle(index+1), width, 0); 386 | move(circle(index-2), -width, 0); 387 | 388 | } else { 389 | move(index+1, width, 0); 390 | } 391 | 392 | move(index, slidePos[index]+width, speed); 393 | move(circle(index-1), slidePos[circle(index-1)]+width, speed); 394 | index = circle(index-1); 395 | 396 | } 397 | 398 | options.callback && options.callback(index, slides[index]); 399 | 400 | } else { 401 | 402 | if (options.continuous) { 403 | 404 | move(circle(index-1), -width, speed); 405 | move(index, 0, speed); 406 | move(circle(index+1), width, speed); 407 | 408 | } else { 409 | 410 | move(index-1, -width, speed); 411 | move(index, 0, speed); 412 | move(index+1, width, speed); 413 | } 414 | 415 | } 416 | 417 | } 418 | 419 | // kill touchmove and touchend event listeners until touchstart called again 420 | element.removeEventListener('touchmove', events, false) 421 | element.removeEventListener('touchend', events, false) 422 | 423 | }, 424 | transitionEnd: function(event) { 425 | 426 | if (parseInt(event.target.getAttribute('data-index'), 10) == index) { 427 | 428 | if (delay) begin(); 429 | 430 | options.transitionEnd && options.transitionEnd.call(event, index, slides[index]); 431 | 432 | } 433 | 434 | } 435 | 436 | } 437 | 438 | // trigger setup 439 | setup(); 440 | 441 | // start auto slideshow if applicable 442 | if (delay) begin(); 443 | 444 | 445 | // add event listeners 446 | if (browser.addEventListener) { 447 | 448 | // set touchstart event on element 449 | if (browser.touch) element.addEventListener('touchstart', events, false); 450 | 451 | if (browser.transitions) { 452 | element.addEventListener('webkitTransitionEnd', events, false); 453 | element.addEventListener('msTransitionEnd', events, false); 454 | element.addEventListener('oTransitionEnd', events, false); 455 | element.addEventListener('otransitionend', events, false); 456 | element.addEventListener('transitionend', events, false); 457 | } 458 | 459 | // set resize event on window 460 | window.addEventListener('resize', events, false); 461 | 462 | } else { 463 | 464 | window.onresize = function () { setup() }; // to play nice with old IE 465 | 466 | } 467 | 468 | // expose the Swipe API 469 | return { 470 | setup: function() { 471 | 472 | setup(); 473 | 474 | }, 475 | slide: function(to, speed) { 476 | 477 | // cancel slideshow 478 | stop(); 479 | 480 | slide(to, speed); 481 | 482 | }, 483 | prev: function() { 484 | 485 | // cancel slideshow 486 | stop(); 487 | 488 | prev(); 489 | 490 | }, 491 | next: function() { 492 | 493 | // cancel slideshow 494 | stop(); 495 | 496 | next(); 497 | 498 | }, 499 | stop: function() { 500 | 501 | // cancel slideshow 502 | stop(); 503 | 504 | }, 505 | getPos: function() { 506 | 507 | // return current index position 508 | return index; 509 | 510 | }, 511 | getNumSlides: function() { 512 | 513 | // return total number of slides 514 | return length; 515 | }, 516 | kill: function() { 517 | 518 | // cancel slideshow 519 | stop(); 520 | 521 | // reset element 522 | element.style.width = ''; 523 | element.style.left = ''; 524 | 525 | // reset slides 526 | var pos = slides.length; 527 | while(pos--) { 528 | 529 | var slide = slides[pos]; 530 | slide.style.width = ''; 531 | slide.style.left = ''; 532 | 533 | if (browser.transitions) translate(pos, 0, 0); 534 | 535 | } 536 | 537 | // removed event listeners 538 | if (browser.addEventListener) { 539 | 540 | // remove current event listeners 541 | element.removeEventListener('touchstart', events, false); 542 | element.removeEventListener('webkitTransitionEnd', events, false); 543 | element.removeEventListener('msTransitionEnd', events, false); 544 | element.removeEventListener('oTransitionEnd', events, false); 545 | element.removeEventListener('otransitionend', events, false); 546 | element.removeEventListener('transitionend', events, false); 547 | window.removeEventListener('resize', events, false); 548 | 549 | } 550 | else { 551 | 552 | window.onresize = null; 553 | 554 | } 555 | 556 | } 557 | } 558 | 559 | } 560 | 561 | 562 | // if ( window.jQuery || window.Zepto ) { 563 | // (function($) { 564 | // $.fn.Swipe = function(params) { 565 | // return this.each(function() { 566 | // $(this).data('Swipe', new Swipe($(this)[0], params)); 567 | // }); 568 | // } 569 | // })( window.jQuery || window.Zepto ) 570 | // } 571 | 572 | module.exports = Swipe; 573 | -------------------------------------------------------------------------------- /src/js/main.js: -------------------------------------------------------------------------------- 1 | import './public/index.js'; 2 | import './component/layout/index.js'; 3 | import player from './component/player/index.js'; 4 | 5 | import topBar from './component/top-bar/index.js'; 6 | import header from './component/header/index.js'; 7 | import recommendList from './component/recommend-list/index.js'; 8 | 9 | import danmaku from './component/danmaku/index.js'; 10 | import message from './component/message/index.js'; 11 | import giftLit from './component/gift/gift-lit/index.js'; 12 | import giftLarge from './component/gift/gift-large/index.js'; 13 | import like from './component/like/index.js'; 14 | 15 | // 直播间基础信息准备就绪 16 | RequqestApi.getLiveData().then((liveData)=>{ 17 | 18 | player.init(liveData).then(()=>{ 19 | 20 | // 弹幕初始化 21 | RequqestApi.getDanmaku().then(( socket ) => { 22 | danmaku.init(socket); 23 | }); 24 | 25 | // 公屏初始化 26 | RequqestApi.getMessage().then((socket) => { 27 | message.init(socket); 28 | }) 29 | 30 | // 请求礼物映射 31 | let giftMapReady = RequqestApi.getGiftMap(); 32 | 33 | // 小礼物初始化 34 | Promise.all([RequqestApi.getGiftLit(), giftMapReady]).then(([ giftSocket, giftMapObj ]) => { 35 | giftLit.init(giftSocket, giftMapObj); 36 | }); 37 | 38 | // 大礼物初始化 39 | Promise.all([RequqestApi.getGiftLarge(), giftMapReady]).then(([ giftSocket, giftMapObj ]) => { 40 | giftLarge.init(giftSocket, giftMapObj); 41 | }); 42 | 43 | // 点赞 44 | like.init(); 45 | 46 | 47 | player.listenLiveStatus((status)=>{ 48 | if (status) return; 49 | header.destroy(); 50 | danmaku.destroy(); 51 | message.destroy(); 52 | giftLit.destroy(); 53 | giftLarge.destroy(); 54 | like.destroy(); 55 | console.log('直播结束'); 56 | }); 57 | }); 58 | 59 | (typeof topBar !== "undefined") && topBar.init(); 60 | (typeof header !== "undefined") && header.init(liveData); 61 | (typeof recommendList !== "undefined") && recommendList.init(); 62 | }) 63 | -------------------------------------------------------------------------------- /src/js/public/api/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file API接口 3 | * 4 | */ 5 | 6 | let jsonpPromise = (url) => { 7 | return new Promise(( resolve, reject ) => { 8 | window.util.jsonp({ 9 | url: url, 10 | success: resolve, 11 | error: reject, 12 | }); 13 | }).catch((err) => { 14 | console.log(err); 15 | throw err; 16 | }); 17 | } 18 | 19 | let socketPromise = (key, socketName) => { 20 | let ioUrl = 'http://legox.org:5353'; 21 | return new Promise(( resolve ) => { 22 | let ioSocket = io.connect(ioUrl); 23 | ioSocket.emit(socketName,{ key: key, time: 8000 }); 24 | resolve({ 25 | ioSocket, 26 | socketName, 27 | }); 28 | }); 29 | }; 30 | 31 | window.RequqestApi = { 32 | 33 | // 获取直播间基础信息 34 | getLiveData(){ 35 | return jsonpPromise('https://legox.org/mock/e7507320-b270-11e7-ba75-c111f832c2e3'); 36 | }, 37 | 38 | // 获取视频推荐列表 39 | getRecommendList(){ 40 | return jsonpPromise('https://legox.org/mock/7d57faa0-b3c0-11e7-ba75-c111f832c2e3'); 41 | }, 42 | 43 | // 获取直播状态 44 | getLiveStatus(){ 45 | return jsonpPromise('https://legox.org/mock/a3e67a40-863c-11e7-9085-0ba4558c07dc'); 46 | // return jsonpPromise('https://legox.org/mock/fbec5f60-b3c9-11e7-ba75-c111f832c2e3'); 47 | }, 48 | 49 | // 获取登录用户信息 50 | getUserInfo(){ 51 | return jsonpPromise('http://uedfe.yypm.com/mock/api/344'); 52 | }, 53 | 54 | // 获取弹幕 55 | getDanmaku() { 56 | return socketPromise('64af4540-b4c3-11e7-ba75-c111f832c2e3', 'mock'); 57 | }, 58 | 59 | // 获取公屏 60 | getMessage() { 61 | return socketPromise('4a2691b0-b4c8-11e7-ba75-c111f832c2e3', 'mock'); 62 | }, 63 | 64 | // 获取礼物映射 65 | getGiftMap() { 66 | return jsonpPromise('https://legox.org/mock/10e15760-b53f-11e7-ba75-c111f832c2e3'); 67 | }, 68 | 69 | // 获取小礼物 70 | getGiftLit() { 71 | return socketPromise('52741350-b541-11e7-ba75-c111f832c2e3', 'mock'); 72 | }, 73 | 74 | // 获取大礼物 75 | getGiftLarge() { 76 | return socketPromise('a11dce30-b56c-11e7-ba75-c111f832c2e3', 'mock'); 77 | }, 78 | } 79 | -------------------------------------------------------------------------------- /src/js/public/config/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @description 全局配置 3 | * @author wuweishuan 4 | */ 5 | 6 | window.CONFIG = { 7 | 8 | type: 'mobile', //终端类型 mobile||pc 9 | 10 | wrapSelector: 'body', //最外层的选择器 11 | 12 | // 组件引入 13 | components: { 14 | 'message' : true, //公屏消息 15 | 'gift-danmaku' : true, //弹幕礼物 16 | 'gift-big' : true, //大礼物 17 | 'gift-little' : true, //下礼物 18 | 'like' : true, //点赞 19 | 'top-bar' : true, //顶部条 20 | 'download-tips' : true, //下载提示 21 | 'recommend-list' : true, //视频推荐列表 22 | }, 23 | 24 | // 功能启用 25 | function: { 26 | 'send-message' : false, //发送公屏消息 27 | 'send-gift' : false, //发送礼物 28 | 'use-svga' : true, //大礼物动画使用SVGA 29 | }, 30 | }; 31 | -------------------------------------------------------------------------------- /src/js/public/index.js: -------------------------------------------------------------------------------- 1 | import '../lib/es6-promise/index.js'; 2 | import '../lib/classlist-polyfill/index.js'; 3 | import io from '../lib/socket.io/index.js'; 4 | import fastclick from '../lib/fastclick/index.js'; 5 | import queue from '../lib/queue/index.js'; 6 | import delegate from '../lib/delegate/index.js'; 7 | import './util/index.js'; 8 | import './config/index.js'; 9 | import './api/index.js'; 10 | 11 | document.addEventListener("DOMContentLoaded", function(event) { 12 | fastclick.attach(document.body); 13 | }); 14 | window.io = window.io ? window.io : io; 15 | window.queue = window.queue ? window.queue : queue; 16 | window.delegate = window.delegate ? window.delegate : delegate; 17 | -------------------------------------------------------------------------------- /src/js/public/util/delegate-event.js: -------------------------------------------------------------------------------- 1 | if (!Element.prototype.matches) { 2 | Element.prototype.matches = 3 | Element.prototype.matchesSelector || 4 | Element.prototype.webkitMatchesSelector || 5 | function(s) { 6 | var matches = (this.document || this.ownerDocument).querySelectorAll(s), 7 | i = matches.length; 8 | while (--i >= 0 && matches.item(i) !== this) {} 9 | return i > -1; 10 | }; 11 | } 12 | function delegate ( selector, handler, counter ) { 13 | if (counter) { 14 | let callback = function ( event ) { 15 | for (let target = event.target; target && target != this; target = target.parentNode) { 16 | if (target.matches(selector)) { 17 | counter--; 18 | if (!counter) { 19 | this.removeEventListener(event.type, callback, false); 20 | } 21 | return handler.apply(target, arguments); 22 | } 23 | } 24 | }; 25 | return callback; 26 | } 27 | return function ( event ) { 28 | for (let target = event.target; target && target != this; target = target.parentNode) { 29 | if (target.matches(selector)) { 30 | return handler.apply(target, arguments); 31 | } 32 | } 33 | }; 34 | }; 35 | 36 | export default delegate; 37 | -------------------------------------------------------------------------------- /src/js/public/util/detect-android.js: -------------------------------------------------------------------------------- 1 | /** 2 | * 检测当前环境是否为Android 3 | * 4 | * @return {boolean} true|false 5 | */ 6 | 7 | 8 | function isAndroid() { 9 | let userAgent = window.navigator.userAgent; 10 | return /Android/i.test(userAgent); 11 | } 12 | 13 | export default isAndroid 14 | 15 | -------------------------------------------------------------------------------- /src/js/public/util/get-url-param.js: -------------------------------------------------------------------------------- 1 | /** 2 | * 获取url中指定参数的值 3 | * @param {string} name 需要获取的参数名 4 | * @param {string} url 需要被处理的url,默认为当前url 5 | * @return {string} 对应的参数值 6 | */ 7 | 8 | function getUrlParam(name, url) { 9 | var paramReg = new RegExp("[\\?&#]" + name + "=([^&#]+)","gi"); 10 | var paramMatch = decodeURIComponent(url || location.href).match(paramReg); 11 | var paramResult = []; 12 | 13 | if (paramMatch && paramMatch.length > 0) { 14 | paramResult = (paramMatch[paramMatch.length-1]).split("="); 15 | if (paramResult && paramResult.length > 1) { 16 | return paramResult[1]; 17 | } 18 | return '' 19 | } 20 | return ''; 21 | } 22 | 23 | export default getUrlParam; 24 | -------------------------------------------------------------------------------- /src/js/public/util/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @file util 3 | * 4 | */ 5 | 6 | import jsonp from '../../lib/jsonp/index.js'; 7 | import delegate from './delegate-event.js'; 8 | import sleep from './sleep.js'; 9 | import scrollTo from './scroll-to.js'; 10 | import sprite from './sprite.js'; 11 | import getUrlParam from './get-url-param.js'; 12 | import isAndroid from './detect-android.js'; 13 | import preloadScript from './preload-script.js'; 14 | 15 | window.util = { 16 | jsonp, 17 | delegate, 18 | sleep, 19 | scrollTo, 20 | sprite, 21 | getUrlParam, 22 | isAndroid, 23 | preloadScript, 24 | } 25 | -------------------------------------------------------------------------------- /src/js/public/util/preload-script.js: -------------------------------------------------------------------------------- 1 | function preloadScript(url) { 2 | new Promise(( resolve ) => { 3 | let script = document.createElement('script'); 4 | script.src = url; 5 | script.onload = () => { 6 | document.head.removeChild(script); 7 | resolve(); 8 | }; 9 | document.head.appendChild(script); 10 | }); 11 | } 12 | 13 | export default preloadScript; 14 | -------------------------------------------------------------------------------- /src/js/public/util/scroll-to.js: -------------------------------------------------------------------------------- 1 | function scrollTo ( direction, config ) { 2 | let { elem, speed = 20, interval = 16, transition = true } = config; 3 | return new Promise(( resolve ) => { 4 | if (transition) { 5 | let scrolling = true; 6 | let now = elem.scrollTop; 7 | let next = () => { 8 | let destination; 9 | if (direction == 'top') { 10 | destination = 0; 11 | now -= speed; 12 | if (now < destination) { 13 | now = destination; 14 | scrolling = false; 15 | } 16 | } else { 17 | destination = elem.scrollHeight; 18 | now += speed; 19 | if (now > destination) { 20 | now = destination; 21 | scrolling = false; 22 | } 23 | } 24 | elem.scrollTop = now; 25 | if (scrolling) { 26 | setTimeout(next, interval); 27 | } else { 28 | resolve(); 29 | } 30 | }; 31 | next(); 32 | } else { 33 | if (direction == 'top') { 34 | elem.scrollTop = 0; 35 | } else { 36 | elem.scrollTop = elem.scrollHeight; 37 | } 38 | resolve(); 39 | } 40 | }); 41 | } 42 | 43 | export default scrollTo; 44 | -------------------------------------------------------------------------------- /src/js/public/util/sleep.js: -------------------------------------------------------------------------------- 1 | function sleep(delay) { 2 | return new Promise(( resolve ) => { 3 | setTimeout(resolve, delay); 4 | }); 5 | } 6 | 7 | export default sleep; 8 | -------------------------------------------------------------------------------- /src/js/public/util/sprite.js: -------------------------------------------------------------------------------- 1 | // 从时序图尺寸 动态生成动画函数 2 | 3 | let record = {}; 4 | function sprite( elem, src, duration ) { 5 | return new Promise(( resolve ) => { 6 | let img = new Image; 7 | img.onload = onload; 8 | img.src = src; 9 | elem.dataset.status = 'pause'; 10 | function onload () { 11 | let frame; 12 | let direction; 13 | if (this.width > this.height) { 14 | direction = 'x'; 15 | frame = Math.round(this.width / this.height); 16 | // elem.style['background-size'] = `auto ${ getComputedStyle(elem).width }`; 17 | elem.style['background-size'] = `auto 100%`; 18 | } else { 19 | direction = 'y'; 20 | frame = Math.round(this.height / this.width); 21 | // elem.style['background-size'] = `${ getComputedStyle(elem).height } auto`; 22 | elem.style['background-size'] = `100% auto`; 23 | } 24 | elem.dataset.frame = frame; 25 | elem.dataset.direction = direction; 26 | elem.style['background-image'] = `url(${ src })`; 27 | let unique = direction + frame; 28 | if (!record[unique]) { 29 | record[unique] = true; 30 | let style = document.createElement('style'); 31 | style.innerHTML = ` 32 | [data-status="play"][data-frame="${ frame }"][data-direction="${ direction }"] { 33 | animation-duration: ${ duration * frame }ms; 34 | -webkit-animation-duration: ${ duration * frame }ms; 35 | animation-name: spirit-animation-${ unique }; 36 | -webkit-animation-name: spirit-animation-${ unique }; 37 | animation-timing-function: steps(${ frame }); 38 | -webkit-animation-timing-function: steps(${ frame }); 39 | animation-iteration-count: infinite; 40 | -webkit-animation-iteration-count: infinite; 41 | } 42 | @keyframes spirit-animation-${ unique } { 43 | 0% { 44 | background-position: 0 0; 45 | } 46 | 100% { 47 | background-position: ${ direction == 'x' ? `-${ frame }00% 0` : `0 -${ frame }00%` }; 48 | } 49 | } 50 | @-webkit-keyframes spirit-animation-${ unique } { 51 | 0% { 52 | background-position: 0 0; 53 | } 54 | 100% { 55 | background-position: ${ direction == 'x' ? `-${ frame }00% 0` : `0 -${ frame }00%` }; 56 | } 57 | } 58 | `; 59 | document.head.appendChild(style); 60 | } 61 | resolve(); 62 | } 63 | }); 64 | } 65 | export default sprite; 66 | -------------------------------------------------------------------------------- /src/sass/_img.scss: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yyued/H5Live/20d407f361d56190ca82496c22f6e159af8c7d29/src/sass/_img.scss -------------------------------------------------------------------------------- /src/sass/base/_base.scss: -------------------------------------------------------------------------------- 1 | /* 2 | * 基础模块 & 全局变量配置 3 | * 4 | */ 5 | 6 | // 文本配置 7 | $b-font-size : 14px !default; 8 | $b-font-family : "San Francisco", "Helvetica Neue", Helvetica, sans-serif !default; 9 | $b-font-weight : 300 !default; 10 | $b-line-height : 1.5 !default; 11 | $b-link-decoration : underline !default; 12 | 13 | // 颜色配置 14 | $c-text : #333 !default; 15 | $c-background : #fff !default; 16 | $c-link : #333 #fa0 !default; 17 | 18 | // 基础模块 19 | @import "normalize"; 20 | @import "reset"; 21 | @import "flexible"; 22 | // @import "orientation"; 23 | @import "tool"; 24 | 25 | // todo: 横屏提示在安卓弹出键盘时被触发 -------------------------------------------------------------------------------- /src/sass/base/_flexible.scss: -------------------------------------------------------------------------------- 1 | /* 2 | 3 | 纯 CSS 自适应方案 4 | 5 | 1,设计稿尺寸 6 | 7 | 默认设计稿宽带 750px,可通过变量 $psd-size 进行修改 8 | 9 | 2,对应的 meta 标签写法: 10 | 11 | 12 | 13 | 3,px to rem 需要用到小学算术 14 | 15 | 100px = 1rem / 1px = 0.01rem 16 | 17 | Notes: 18 | 19 | 断点参考: 20 | http://screensiz.es/phone 21 | http://mydevice.io/devices/#sortSmartphones 22 | 23 | 横屏断点:384px, 480px, 533px, 568px, 640px, 667px, 699px, 736px 24 | 某些手机厂商系统的虚拟 SmartBar 会改变 Screen 的值,又是坑,建议统一横屏提示规避 25 | 26 | 避免使用 min-device-width 27 | https://developers.google.com/web/fundamentals/design-and-ui/responsive/fundamentals/use-media-queries?hl=zh-cn#min-device-width- 28 | 29 | Changelog: 30 | 31 | 2016.05.04:添加 393px 699px 断点,小米note 1080 屏幕,DPR=2.75 32 | 2016.11.11: 参考 http://mydevice.io/devices/#sortSmartphones 添加部分安卓手机宽度 33 | 34 | */ 35 | 36 | // 定义基准数 37 | $base-fontSize: 100px !default; 38 | 39 | // 定义设计稿尺寸 40 | $psd-size: 750px !default; 41 | 42 | // 定义竖屏断点 43 | $responsives: 320px, 346px, 360px, 375px, 384px, 390px, 393px, 400px, 412px, 414px, 432px; 44 | 45 | // 定义根元素font-size 46 | @mixin rem($values){ 47 | font-size: $values * $base-fontSize / $psd-size; 48 | } 49 | 50 | // 遍历输出断点 51 | @each $responsive in $responsives{ 52 | @media only screen and (min-width: #{$responsive}) { 53 | html{ @include rem($responsive);} 54 | } 55 | } 56 | -------------------------------------------------------------------------------- /src/sass/base/_normalize.scss: -------------------------------------------------------------------------------- 1 | /* normalize.css v5.0.0 | MIT License | github.com/necolas/normalize.css */ 2 | 3 | /** 4 | * 1. Change the default font family in all browsers (opinionated). 5 | * 2. Correct the line height in all browsers. 6 | * 3. Prevent adjustments of font size after orientation changes in 7 | * IE on Windows Phone and in iOS. 8 | */ 9 | 10 | /* Document 11 | ========================================================================== */ 12 | 13 | html { 14 | font-family: sans-serif; /* 1 */ 15 | line-height: 1.15; /* 2 */ 16 | -ms-text-size-adjust: 100%; /* 3 */ 17 | -webkit-text-size-adjust: 100%; /* 3 */ 18 | } 19 | 20 | /* Sections 21 | ========================================================================== */ 22 | 23 | /** 24 | * Remove the margin in all browsers (opinionated). 25 | */ 26 | 27 | body { 28 | margin: 0; 29 | } 30 | 31 | /** 32 | * Add the correct display in IE 9-. 33 | */ 34 | 35 | article, 36 | aside, 37 | footer, 38 | header, 39 | nav, 40 | section { 41 | display: block; 42 | } 43 | 44 | /** 45 | * Correct the font size and margin on `h1` elements within `section` and 46 | * `article` contexts in Chrome, Firefox, and Safari. 47 | */ 48 | 49 | h1 { 50 | font-size: 2em; 51 | margin: 0.67em 0; 52 | } 53 | 54 | /* Grouping content 55 | ========================================================================== */ 56 | 57 | /** 58 | * Add the correct display in IE 9-. 59 | * 1. Add the correct display in IE. 60 | */ 61 | 62 | figcaption, 63 | figure, 64 | main { /* 1 */ 65 | display: block; 66 | } 67 | 68 | /** 69 | * Add the correct margin in IE 8. 70 | */ 71 | 72 | figure { 73 | margin: 1em 40px; 74 | } 75 | 76 | /** 77 | * 1. Add the correct box sizing in Firefox. 78 | * 2. Show the overflow in Edge and IE. 79 | */ 80 | 81 | hr { 82 | box-sizing: content-box; /* 1 */ 83 | height: 0; /* 1 */ 84 | overflow: visible; /* 2 */ 85 | } 86 | 87 | /** 88 | * 1. Correct the inheritance and scaling of font size in all browsers. 89 | * 2. Correct the odd `em` font sizing in all browsers. 90 | */ 91 | 92 | pre { 93 | font-family: monospace, monospace; /* 1 */ 94 | font-size: 1em; /* 2 */ 95 | } 96 | 97 | /* Text-level semantics 98 | ========================================================================== */ 99 | 100 | /** 101 | * 1. Remove the gray background on active links in IE 10. 102 | * 2. Remove gaps in links underline in iOS 8+ and Safari 8+. 103 | */ 104 | 105 | a { 106 | background-color: transparent; /* 1 */ 107 | -webkit-text-decoration-skip: objects; /* 2 */ 108 | } 109 | 110 | /** 111 | * Remove the outline on focused links when they are also active or hovered 112 | * in all browsers (opinionated). 113 | */ 114 | 115 | a:active, 116 | a:hover { 117 | outline-width: 0; 118 | } 119 | 120 | /** 121 | * 1. Remove the bottom border in Firefox 39-. 122 | * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari. 123 | */ 124 | 125 | abbr[title] { 126 | border-bottom: none; /* 1 */ 127 | text-decoration: underline; /* 2 */ 128 | text-decoration: underline dotted; /* 2 */ 129 | } 130 | 131 | /** 132 | * Prevent the duplicate application of `bolder` by the next rule in Safari 6. 133 | */ 134 | 135 | b, 136 | strong { 137 | font-weight: inherit; 138 | } 139 | 140 | /** 141 | * Add the correct font weight in Chrome, Edge, and Safari. 142 | */ 143 | 144 | b, 145 | strong { 146 | font-weight: bolder; 147 | } 148 | 149 | /** 150 | * 1. Correct the inheritance and scaling of font size in all browsers. 151 | * 2. Correct the odd `em` font sizing in all browsers. 152 | */ 153 | 154 | code, 155 | kbd, 156 | samp { 157 | font-family: monospace, monospace; /* 1 */ 158 | font-size: 1em; /* 2 */ 159 | } 160 | 161 | /** 162 | * Add the correct font style in Android 4.3-. 163 | */ 164 | 165 | dfn { 166 | font-style: italic; 167 | } 168 | 169 | /** 170 | * Add the correct background and color in IE 9-. 171 | */ 172 | 173 | mark { 174 | background-color: #ff0; 175 | color: #000; 176 | } 177 | 178 | /** 179 | * Add the correct font size in all browsers. 180 | */ 181 | 182 | small { 183 | font-size: 80%; 184 | } 185 | 186 | /** 187 | * Prevent `sub` and `sup` elements from affecting the line height in 188 | * all browsers. 189 | */ 190 | 191 | sub, 192 | sup { 193 | font-size: 75%; 194 | line-height: 0; 195 | position: relative; 196 | vertical-align: baseline; 197 | } 198 | 199 | sub { 200 | bottom: -0.25em; 201 | } 202 | 203 | sup { 204 | top: -0.5em; 205 | } 206 | 207 | /* Embedded content 208 | ========================================================================== */ 209 | 210 | /** 211 | * Add the correct display in IE 9-. 212 | */ 213 | 214 | audio, 215 | video { 216 | display: inline-block; 217 | } 218 | 219 | /** 220 | * Add the correct display in iOS 4-7. 221 | */ 222 | 223 | audio:not([controls]) { 224 | display: none; 225 | height: 0; 226 | } 227 | 228 | /** 229 | * Remove the border on images inside links in IE 10-. 230 | */ 231 | 232 | img { 233 | border-style: none; 234 | } 235 | 236 | /** 237 | * Hide the overflow in IE. 238 | */ 239 | 240 | svg:not(:root) { 241 | overflow: hidden; 242 | } 243 | 244 | /* Forms 245 | ========================================================================== */ 246 | 247 | /** 248 | * 1. Change the font styles in all browsers (opinionated). 249 | * 2. Remove the margin in Firefox and Safari. 250 | */ 251 | 252 | button, 253 | input, 254 | optgroup, 255 | select, 256 | textarea { 257 | font-family: sans-serif; /* 1 */ 258 | font-size: 100%; /* 1 */ 259 | line-height: 1.15; /* 1 */ 260 | margin: 0; /* 2 */ 261 | } 262 | 263 | /** 264 | * Show the overflow in IE. 265 | * 1. Show the overflow in Edge. 266 | */ 267 | 268 | button, 269 | input { /* 1 */ 270 | overflow: visible; 271 | } 272 | 273 | /** 274 | * Remove the inheritance of text transform in Edge, Firefox, and IE. 275 | * 1. Remove the inheritance of text transform in Firefox. 276 | */ 277 | 278 | button, 279 | select { /* 1 */ 280 | text-transform: none; 281 | } 282 | 283 | /** 284 | * 1. Prevent a WebKit bug where (2) destroys native `audio` and `video` 285 | * controls in Android 4. 286 | * 2. Correct the inability to style clickable types in iOS and Safari. 287 | */ 288 | 289 | button, 290 | html [type="button"], /* 1 */ 291 | [type="reset"], 292 | [type="submit"] { 293 | -webkit-appearance: button; /* 2 */ 294 | } 295 | 296 | /** 297 | * Remove the inner border and padding in Firefox. 298 | */ 299 | 300 | button::-moz-focus-inner, 301 | [type="button"]::-moz-focus-inner, 302 | [type="reset"]::-moz-focus-inner, 303 | [type="submit"]::-moz-focus-inner { 304 | border-style: none; 305 | padding: 0; 306 | } 307 | 308 | /** 309 | * Restore the focus styles unset by the previous rule. 310 | */ 311 | 312 | button:-moz-focusring, 313 | [type="button"]:-moz-focusring, 314 | [type="reset"]:-moz-focusring, 315 | [type="submit"]:-moz-focusring { 316 | outline: 1px dotted ButtonText; 317 | } 318 | 319 | /** 320 | * Change the border, margin, and padding in all browsers (opinionated). 321 | */ 322 | 323 | fieldset { 324 | border: 1px solid #c0c0c0; 325 | margin: 0 2px; 326 | padding: 0.35em 0.625em 0.75em; 327 | } 328 | 329 | /** 330 | * 1. Correct the text wrapping in Edge and IE. 331 | * 2. Correct the color inheritance from `fieldset` elements in IE. 332 | * 3. Remove the padding so developers are not caught out when they zero out 333 | * `fieldset` elements in all browsers. 334 | */ 335 | 336 | legend { 337 | box-sizing: border-box; /* 1 */ 338 | color: inherit; /* 2 */ 339 | display: table; /* 1 */ 340 | max-width: 100%; /* 1 */ 341 | padding: 0; /* 3 */ 342 | white-space: normal; /* 1 */ 343 | } 344 | 345 | /** 346 | * 1. Add the correct display in IE 9-. 347 | * 2. Add the correct vertical alignment in Chrome, Firefox, and Opera. 348 | */ 349 | 350 | progress { 351 | display: inline-block; /* 1 */ 352 | vertical-align: baseline; /* 2 */ 353 | } 354 | 355 | /** 356 | * Remove the default vertical scrollbar in IE. 357 | */ 358 | 359 | textarea { 360 | overflow: auto; 361 | } 362 | 363 | /** 364 | * 1. Add the correct box sizing in IE 10-. 365 | * 2. Remove the padding in IE 10-. 366 | */ 367 | 368 | [type="checkbox"], 369 | [type="radio"] { 370 | box-sizing: border-box; /* 1 */ 371 | padding: 0; /* 2 */ 372 | } 373 | 374 | /** 375 | * Correct the cursor style of increment and decrement buttons in Chrome. 376 | */ 377 | 378 | [type="number"]::-webkit-inner-spin-button, 379 | [type="number"]::-webkit-outer-spin-button { 380 | height: auto; 381 | } 382 | 383 | /** 384 | * 1. Correct the odd appearance in Chrome and Safari. 385 | * 2. Correct the outline style in Safari. 386 | */ 387 | 388 | [type="search"] { 389 | -webkit-appearance: textfield; /* 1 */ 390 | outline-offset: -2px; /* 2 */ 391 | } 392 | 393 | /** 394 | * Remove the inner padding and cancel buttons in Chrome and Safari on macOS. 395 | */ 396 | 397 | [type="search"]::-webkit-search-cancel-button, 398 | [type="search"]::-webkit-search-decoration { 399 | -webkit-appearance: none; 400 | } 401 | 402 | /** 403 | * 1. Correct the inability to style clickable types in iOS and Safari. 404 | * 2. Change font properties to `inherit` in Safari. 405 | */ 406 | 407 | ::-webkit-file-upload-button { 408 | -webkit-appearance: button; /* 1 */ 409 | font: inherit; /* 2 */ 410 | } 411 | 412 | /* Interactive 413 | ========================================================================== */ 414 | 415 | /* 416 | * Add the correct display in IE 9-. 417 | * 1. Add the correct display in Edge, IE, and Firefox. 418 | */ 419 | 420 | details, /* 1 */ 421 | menu { 422 | display: block; 423 | } 424 | 425 | /* 426 | * Add the correct display in all browsers. 427 | */ 428 | 429 | summary { 430 | display: list-item; 431 | } 432 | 433 | /* Scripting 434 | ========================================================================== */ 435 | 436 | /** 437 | * Add the correct display in IE 9-. 438 | */ 439 | 440 | canvas { 441 | display: inline-block; 442 | } 443 | 444 | /** 445 | * Add the correct display in IE. 446 | */ 447 | 448 | template { 449 | display: none; 450 | } 451 | 452 | /* Hidden 453 | ========================================================================== */ 454 | 455 | /** 456 | * Add the correct display in IE 10-. 457 | */ 458 | 459 | [hidden] { 460 | display: none; 461 | } -------------------------------------------------------------------------------- /src/sass/base/_orientation.scss: -------------------------------------------------------------------------------- 1 | /*! 2 | 旋转提示 tips 方案 3 | */ 4 | 5 | body{ 6 | &:after{ 7 | position: fixed; 8 | top: 0; 9 | left: 0; 10 | width: 1px; 11 | height: 1px; 12 | overflow: hidden; 13 | content: ""; 14 | background-color: rgba(0,0,0,.8); 15 | background-image: url(''); 16 | background-repeat: no-repeat; 17 | background-position: center; 18 | background-size: 738/2 + px 37/2 + px; 19 | display: none; 20 | z-index: 99999; 21 | } 22 | } 23 | 24 | @media all and (orientation : landscape) { 25 | body{ 26 | &:after{ 27 | width: 100%; 28 | height: 100%; 29 | display: block; 30 | } 31 | } 32 | } 33 | -------------------------------------------------------------------------------- /src/sass/base/_reset.scss: -------------------------------------------------------------------------------- 1 | /* 2 | * @file: 重置样式 3 | */ 4 | 5 | /* 全局设置 */ 6 | *, 7 | *:before, 8 | *:after{ 9 | font-family: "San Francisco", "Helvetica Neue", Helvetica, sans-serif; 10 | font-weight: 300; 11 | box-sizing: border-box; 12 | appearance: none; 13 | -webkit-tap-highlight-color: rgba(0,0,0,0); 14 | } 15 | 16 | /* 设置基本html,body样式 */ 17 | html,body{ 18 | margin: 0 auto; 19 | padding: 0; 20 | } 21 | 22 | /* 设置图片最大宽度 */ 23 | img{ 24 | max-width: 100%; 25 | } 26 | 27 | /* 清除默认按钮表现形式 */ 28 | button{ 29 | -webkit-appearance: none; 30 | appearance: none; 31 | } 32 | 33 | /* 设置表格默认样式 */ 34 | table { 35 | border-collapse: collapse; 36 | border-spacing: 0; 37 | } 38 | 39 | /* 文本域禁止拖拉放大 */ 40 | textarea { 41 | resize: none; 42 | } 43 | 44 | a { 45 | text-decoration: none; 46 | color: currentColor; 47 | } 48 | 49 | img { 50 | vertical-align: top; 51 | } 52 | 53 | i, em { 54 | font-style: normal; 55 | } 56 | 57 | ol, ul, li, menu { 58 | list-style: none outside none; 59 | } 60 | 61 | fieldset, iframe, abbr, acronym { 62 | border: 0 none; 63 | } 64 | 65 | dl, dt, dd, ul, ol, li, h1, h2, h3, h4, h5, h6, pre, code, form, fieldset, legend, input, button, textarea, p, blockquote, th, td, hr, 66 | article, aside, canvas, details, embed, figure, figcaption, footer, header, hgroup, menu, nav, output, ruby, section, summary, time, mark, audio, video { 67 | margin: 0; 68 | padding: 0; 69 | } 70 | 71 | del { 72 | text-decoration:line-through; 73 | } 74 | -------------------------------------------------------------------------------- /src/sass/base/_tool.scss: -------------------------------------------------------------------------------- 1 | /* @name: 转为REM单位fn 2 | * @ps: 默认宽度为750px 3 | */ 4 | @function toRem($px , $width: 750px) { 5 | // 动态尺寸单位 6 | $ppr: $width / 16 / 1rem; 7 | @return ($px / $ppr); 8 | } 9 | 10 | /* @name: 上下左右居中 (未知高度) 11 | * @ps: 需要设置父元素 display:table 12 | */ 13 | @mixin centers{ 14 | display:table-cell; 15 | vertical-align:middle; 16 | } 17 | 18 | /* @name: 上下左右居中 (确定高度)*/ 19 | @mixin centersHight{ 20 | position: absolute; 21 | left: 0; 22 | right: 0; 23 | top: 0; 24 | bottom: 0; 25 | margin: auto; 26 | } 27 | 28 | /* 29 | * @name: 文字溢出显示省略号 30 | * @ps: 需要设置宽度 31 | */ 32 | @mixin ellipsis{ 33 | white-space: nowrap; 34 | word-wrap: normal; 35 | overflow: hidden; 36 | text-overflow: ellipsis; 37 | text-align: left; 38 | } 39 | 40 | /* @name: 强制不换行 */ 41 | @mixin nowrap{ 42 | white-space: nowrap; 43 | word-wrap: normal; 44 | word-break: keep-all; 45 | } 46 | 47 | /* @name: 清除浮动 */ 48 | @mixin clearFix{ 49 | overflow: hidden; 50 | clear: both; 51 | } 52 | 53 | /* 54 | * @name: 三角形 55 | * @param: $size 大小 56 | * @param: $color 颜色 57 | * @param: $type {string} 类型 (up || down || left || right || topleft || topright || bottomleft || bottomright) 58 | */ 59 | @mixin triangle( 60 | $size: 50px, 61 | $type: up, 62 | $color: #6699FF 63 | ){ 64 | width: 0; 65 | height: 0; 66 | @if $type == up { 67 | border-left: $size/2 solid transparent; 68 | border-right: $size/2 solid transparent; 69 | border-bottom: $size solid $color; 70 | }@else if $type == down { 71 | border-left: $size/2 solid transparent; 72 | border-right: $size/2 solid transparent; 73 | border-top: $size solid $color; 74 | }@else if $type == left { 75 | border-top: $size/2 solid transparent; 76 | border-right: $size/2 solid $color; 77 | border-bottom: $size solid transparent; 78 | }@else if $type == right { 79 | border-top: $size/2 solid transparent; 80 | border-left: $size/2 solid $color; 81 | border-bottom: $size solid transparent; 82 | }@else if $type == topleft { 83 | border-top: $size solid $color; 84 | border-right: $size solid transparent; 85 | }@else if $type == topright { 86 | border-top: $size solid $color; 87 | border-left: $size solid transparent; 88 | }@else if $type == bottomleft { 89 | border-bottom: $size solid $color; 90 | border-right: $size solid transparent; 91 | }@else if $type == bottomright { 92 | border-bottom: $size solid $color; 93 | border-left: $size solid transparent; 94 | } 95 | } 96 | -------------------------------------------------------------------------------- /src/sass/main.scss: -------------------------------------------------------------------------------- 1 | @charset "UTF-8"; 2 | 3 | @import "base/base"; 4 | @import "img"; 5 | 6 | body { 7 | font: $b-font-weight #{$b-font-size}/#{$b-line-height} $b-font-family; 8 | color: $c-text; 9 | a { 10 | text-decoration: none; 11 | color: nth($c-link, 1); 12 | &:hover { 13 | text-decoration: underline; 14 | color: nth($c-link, 2); 15 | } 16 | } 17 | } 18 | --------------------------------------------------------------------------------