├── .gitignore ├── README.md ├── lib ├── index.js ├── index.json ├── index.wxml └── index.wxss ├── package.json └── screenshot └── 1.gif /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules/ 3 | ./dist/ 4 | dist/* 5 | dist/ 6 | .wepycache 7 | .idea 8 | npm-debug.log -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # 微信小程序 基于scroll-view 下拉刷新 2 | 3 | 4 | ## 说明 5 | 由于官方scroll-view组件只有滚到顶部刷新的机制,对用户来说不是很友好,所以做一个基于scroll-view的下拉刷新组件(该组件支持原生引用、wexp框架、wepy框架) 6 | 7 | ## 展示 8 | ![Android](./screenshot/1.gif) 9 | ## 使用 10 | 11 | ### 安装组件 12 | ``` 13 | npm i weapp-scroll-view-refresh -S --production 14 | ``` 15 | 16 | ### 使用示例 17 | ```json 18 | usingComponents: { 19 | 'k-scroll': 'weapp-scroll-view-refresh/index' 20 | } 21 | ``` 22 | 23 | ```html 24 | 30 | content 31 | 32 | ``` 33 | 34 | ## API 35 | ### 组件参数 36 | | 参数 | 说明 | 类型 | 可选值 | 默认值 | 37 | |-----------|---------------------------------|-----------|-----------|-------------| 38 | | lowerThresHold | 距底部多远时触发 scrolltolower 事件 | `Number` | ` ` | `50` | 39 | | scrollWithAnimation | 在设置滚动条位置时使用动画过渡 | `Boolean` | `true` `false` | `false` | 40 | | enableBackToTop | iOS点击顶部状态栏、安卓双击标题栏时,滚动条返回顶部,只支持竖向 | `Boolean` | `true` `false` | `false` | 41 | | scrollIntoView | 值应为某子元素id(id不能以数字开头)。设置哪个方向可滚动,则在哪个方向滚动到该元素 | `String` | ` ` | ` ` | 42 | 43 | 44 | ### Event 45 | 46 | | 事件名 | 说明 | 参数 | 47 | |-----------|-----------|-----------| 48 | | bind:scroll | 组件滚动触发该事件 | event.detail: 回传滚动的高度 | 49 | | bind:scrollToLower | 组件滚动到底部触发该事件 | 无回传参数 | 50 | | bind:refresh | 组件刷新操作的时候触发该事件 | 无回传参数 | 51 | 52 | ### 函数 53 | 54 | | 事件名 | 说明 | 参数 | 55 | |-----------|-----------|-----------| 56 | | reset | 重置刷新状态 | `-` | 57 | 58 | 59 | ## 更多说明 60 | 参考[原版插件](https://github.com/Chaunjie/weapp-scroll-view-refresh)。 -------------------------------------------------------------------------------- /lib/index.js: -------------------------------------------------------------------------------- 1 | Component({ 2 | data: { 3 | scrollY: true, 4 | touchYDelta: '', 5 | isLoading: false, 6 | loadWrapH: '', 7 | winfactor: 0.2, 8 | translateVal: '', 9 | isMoved: false, 10 | firstTouchY: '', 11 | initialScroll: '', 12 | friction: 2.5, 13 | scrollTop: 0, 14 | triggerDistance: 100, 15 | className: '', 16 | animationData: {}, 17 | animation: {}, 18 | system: '', 19 | brand: '', 20 | loader: { 21 | height: 500 22 | }, 23 | time: '', 24 | text: '下拉可以刷新' 25 | }, 26 | externalClasses: ['custom-class'], 27 | options: { 28 | multipleSlots: true 29 | }, 30 | properties: { 31 | lowerThresHold: { 32 | type: Number, 33 | value: 50 34 | }, 35 | scrollWithAnimation: { 36 | type: Boolean, 37 | value: false 38 | }, 39 | enableBackToTop: { 40 | type: Boolean, 41 | value: false 42 | }, 43 | scrollIntoView: { 44 | type: String, 45 | value: '' 46 | } 47 | }, 48 | methods: { 49 | scroll (e) { 50 | this.setData({ 51 | scrollTop: e.detail.scrollTop 52 | }) 53 | this.triggerEvent('scroll', e.detail.scrollTop) 54 | }, 55 | scrollToLower () { 56 | this.triggerEvent('lower') 57 | }, 58 | touchstart (ev) { 59 | const {isLoading, scrollTop} = this.data 60 | if (isLoading) { 61 | return 62 | } 63 | const touchobj = ev.touches[0] 64 | this.setData({ 65 | isMoved: false, 66 | sDuration: '0ms', 67 | touchYDelta: '', 68 | firstTouchY: parseInt(touchobj.clientY), 69 | initialScroll: scrollTop 70 | }) 71 | }, 72 | touchmove (ev) { 73 | const {isLoading} = this.data 74 | if (isLoading) { 75 | return 76 | } 77 | let touchobj = ev.touches[0] 78 | let touchY = parseInt(touchobj.clientY) 79 | let touchYDelta = touchY - this.data.firstTouchY 80 | if (this.data.initialScroll > 0 || this.data.scrollTop > 0 || this.data.scrollTop === 0 && touchYDelta < 0) { 81 | this.setData({ 82 | firstTouchY: touchY 83 | }) 84 | return 85 | } 86 | /* eslint-enable */ 87 | const yDelta = this.data.brand === 'devtools' ? touchYDelta ** 0.85 : this.data.system === 'ios' ? touchYDelta ** 0.5 : touchYDelta ** 0.85 88 | // let translateVal = yDelta 89 | this.data.animation.translate3d(0, yDelta, 0).step() 90 | let obj = touchYDelta >= this.data.triggerDistance ? { 91 | className: 'refresh-pull-up', 92 | // text: '松开可以刷新' 93 | } : { 94 | className: 'refresh-pull-down', 95 | // text: '下拉可以刷新' 96 | } 97 | this.setData({ 98 | touchYDelta: touchYDelta, 99 | animationData: this.data.animation.export(), 100 | isMoved: true, 101 | ...obj 102 | }) 103 | }, 104 | touchend (ev) { 105 | if (this.data.isLoading || !this.data.isMoved) { 106 | this.setData({ 107 | isMoved: false 108 | }) 109 | return 110 | } 111 | // 根据下拉高度判断是否加载 112 | if (this.data.touchYDelta >= this.data.triggerDistance) { 113 | this.data.animation.translate3d(0, this.data.loader.height, 0).step() 114 | this.setData({ 115 | isLoading: true, 116 | scrollY: false, 117 | animationData: this.data.animation.export(), 118 | className: 'refreshing', 119 | text: '正在刷新...' 120 | }) 121 | this.triggerEvent('refresh', 'success') 122 | } else { 123 | this.data.animation.translate3d(0, 0, 0).step({duration: 300}) 124 | this.setData({ 125 | animationData: this.data.animation.export() 126 | }) 127 | // this.triggerEvent('refresh', 'error'); 128 | } 129 | this.setData({ 130 | isMoved: false 131 | }) 132 | }, 133 | reset () { 134 | this.setData({ 135 | isLoading: false, 136 | scrollY: true, 137 | className: 'refresh-pull-up', 138 | text: '下拉可以刷新' 139 | }, () => { 140 | this.data.animation.translate3d(0, 0, 0).step({duration: 300}) 141 | const time = this.getTime() 142 | this.setData({ 143 | animationData: this.data.animation.export(), 144 | className: 'refresh-pull-down', 145 | time: time 146 | }) 147 | }) 148 | }, 149 | throttle (fn, delay) { 150 | let allowSample = true 151 | return function(e) { 152 | if (allowSample) { 153 | allowSample = false 154 | setTimeout(function() { allowSample = true }, delay) 155 | fn(e) 156 | } 157 | } 158 | }, 159 | getTime() { 160 | const date = new Date() 161 | return `${date.getFullYear()}-${this.getFriendlyTime(date.getMonth() + 1)}-${this.getFriendlyTime(date.getDate())} ${this.getFriendlyTime(date.getHours())}:${this.getFriendlyTime(date.getMinutes())}:${this.getFriendlyTime(date.getSeconds())}` 162 | }, 163 | getFriendlyTime (time) { 164 | return time < 10 ? '0' + time : time 165 | }, 166 | detached () { 167 | console.log(66666) 168 | } 169 | }, 170 | ready () { 171 | let animation = wx.createAnimation({ 172 | duration: 0, 173 | timingFunction: 'linear' 174 | }) 175 | let system = 'android' 176 | const systemInfo = wx.getSystemInfoSync() 177 | if (/iPhone/.test(systemInfo.model)) { 178 | system = 'ios' 179 | } 180 | const time = this.getTime() 181 | this.setData({ 182 | time: time, 183 | system: system, 184 | animation: animation, 185 | brand: systemInfo.brand, 186 | animationData: animation.export() 187 | }) 188 | wx.createSelectorQuery().in(this).selectAll('.refresh-load').boundingClientRect((res) => { 189 | if (res && res.length) { 190 | this.setData({ 191 | loader: res[0] 192 | }) 193 | } 194 | }).exec() 195 | } 196 | }) -------------------------------------------------------------------------------- /lib/index.json: -------------------------------------------------------------------------------- 1 | { 2 | "component": true 3 | } -------------------------------------------------------------------------------- /lib/index.wxml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | {{text}} 9 | 更新于:{{time}} 10 | 11 | 12 | 13 | 14 | 15 | 16 | -------------------------------------------------------------------------------- /lib/index.wxss: -------------------------------------------------------------------------------- 1 | .scroll-view .new-block { 2 | width: 100%; 3 | height: 100px; 4 | background-color: #333; 5 | } 6 | .scroll-view .new-block:nth-child(even) { 7 | background-color: #fff; 8 | } 9 | .scroll-view .refresh-loader { 10 | width: 100%; 11 | height: 100%; 12 | } 13 | @keyframes rotate { 14 | 0% { 15 | -webkit-transform: rotate(0deg) scale(1); 16 | transform: rotate(0deg) scale(1); 17 | } 18 | 50% { 19 | -webkit-transform: rotate(180deg) scale(1); 20 | transform: rotate(180deg) scale(1); 21 | } 22 | 100% { 23 | -webkit-transform: rotate(360deg) scale(1); 24 | transform: rotate(360deg) scale(1); 25 | } 26 | } 27 | @-webkit-keyframes rotate { 28 | 0% { 29 | -webkit-transform: rotate(0deg) scale(1); 30 | transform: rotate(0deg) scale(1); 31 | } 32 | 50% { 33 | -webkit-transform: rotate(180deg) scale(1); 34 | transform: rotate(180deg) scale(1); 35 | } 36 | 100% { 37 | -webkit-transform: rotate(360deg) scale(1); 38 | transform: rotate(360deg) scale(1); 39 | } 40 | } 41 | .scroll-view .refresh-content { 42 | position: relative; 43 | -webkit-overflow-scrolling: touch; 44 | -webkit-transform: translate3d(0, 0, 0); 45 | transform: translate3d(0, 0, 0); 46 | } 47 | .scroll-view .refresh-load { 48 | background-color: #f5f5f5; 49 | width: 100%; 50 | position: absolute; 51 | top: 0; 52 | left: 0; 53 | padding: 8px 0; 54 | display: flex; 55 | flex-direction: row; 56 | justify-content: center; 57 | align-items: center; 58 | } 59 | .scroll-view .refresh-load .refresh-load__content { 60 | width: 260px; 61 | display: flex; 62 | flex-direction: row; 63 | justify-content: flex-start; 64 | align-items: center; 65 | } 66 | .scroll-view .refresh-load .refresh-load__content .refresh-load__text > view { 67 | font-size: 14px; 68 | display: flex; 69 | flex-direction: row; 70 | justify-content: center; 71 | } 72 | .scroll-view .refresh-pull-arrow { 73 | width: 24px; 74 | height: 24px; 75 | visibility: visible; 76 | margin-right: 20px; 77 | background: no-repeat center; 78 | background-image: url('data:image/svg+xml;charset=utf-8,'); 79 | background-size: cover; 80 | z-index: 10; 81 | -webkit-transform: rotate(0deg) translate3d(0, 0, 0); 82 | transform: rotate(0deg) translate3d(0, 0, 0); 83 | -webkit-transition-duration: 300ms; 84 | transition-duration: 300ms; 85 | } 86 | .scroll-view .refresh-pull-down { 87 | -webkit-transform: rotate(0deg) translate3d(0, 0, 0); 88 | transform: rotate(0deg) translate3d(0, 0, 0); 89 | } 90 | .scroll-view .refresh-pull-up { 91 | -webkit-transform: rotate(180deg) translate3d(0, 0, 0); 92 | transform: rotate(180deg) translate3d(0, 0, 0); 93 | } 94 | .scroll-view .refreshing { 95 | width: 24px; 96 | height: 24px; 97 | background-image: url(''); 98 | -webkit-animation: rotate 1s 0s linear infinite; 99 | animation: rotate 1s 0s linear infinite; 100 | } 101 | .scroll-view .loading-load { 102 | width: 100%; 103 | position: relative; 104 | padding: 8px 0; 105 | } 106 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "weapp-scroll-view-refresh", 3 | "version": "1.0.1", 4 | "description": "wechat weapp scroll-view refresh component", 5 | "main": "./lib/index.js", 6 | "scripts": { 7 | "test": "echo \"Error: no test specified\" && exit 1" 8 | }, 9 | "repository": { 10 | "type": "git", 11 | "url": "git+https://github.com/Chaunjie/weapp-scroll-view-refresh.git" 12 | }, 13 | "author": "chaunjie", 14 | "license": "ISC", 15 | "miniprogram": "lib", 16 | "devDependencies": {}, 17 | "bugs": { 18 | "url": "https://github.com/Chaunjie/weapp-scroll-view-refresh/issues" 19 | }, 20 | "homepage": "https://github.com/Chaunjie/weapp-scroll-view-refresh#readme" 21 | } 22 | -------------------------------------------------------------------------------- /screenshot/1.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Chaunjie/weapp-scroll-view-refresh/c0413b810227c7e71651e73ff53cd5e8ae9f69a2/screenshot/1.gif --------------------------------------------------------------------------------