index.vue 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821
  1. <template>
  2. <view class="s-pull-scroll" :class="customClass">
  3. <scroll-view
  4. :id="scrollId"
  5. class="s-pull-scroll-view"
  6. :class="{'is-fixed':fixed}"
  7. :style="{'padding-top':padTop,'padding-bottom':padBottom,'top':fixedTop,'bottom':fixedBottom}"
  8. :scroll-top="scrollTop"
  9. :scroll-with-animation="false"
  10. :scroll-y="scrollAble"
  11. :enable-back-to-top="true"
  12. @scroll="scroll"
  13. @touchstart="touchstart"
  14. @touchmove="touchmove"
  15. @touchend="touchend"
  16. @touchcancel="touchend"
  17. >
  18. <view :style="{'transform': translateY, 'transition': transition}">
  19. <view
  20. class="s-pull-down-wrap"
  21. :class="[{'is-success': isShowDownTip && isDownSuccess},{'is-error': isShowDownTip && isDownError}]"
  22. :style="{'height':downOffset+'rpx'}"
  23. >
  24. <view
  25. class="s-pull-loading-icon"
  26. v-if="!isShowDownTip"
  27. :class="{'s-pull-loading-rotate':isDownLoading}"
  28. :style="{'transform':downRotate}"
  29. ></view>
  30. <view>{{downText}}</view>
  31. </view>
  32. <slot></slot>
  33. <view v-if="isUpLoading" class="s-pull-up-wrap">
  34. <view class="s-pull-loading-icon s-pull-loading-rotate"></view>
  35. <view>{{upLoadingText}}</view>
  36. </view>
  37. <slot name="empty" v-if="isEmpty && showEmpty">
  38. <view class="s-pull-tip-wrap" v-if="emptyText">{{emptyText}}</view>
  39. </slot>
  40. <slot name="up-error" v-else-if="isUpError && showUpError">
  41. <view class="s-pull-tip-wrap" v-if="upErrorText" @click="onUpErrorClick">{{upErrorText}}</view>
  42. </slot>
  43. <slot name="up-finish" v-else-if="isUpFinish && showUpFinish">
  44. <view class="s-pull-tip-wrap" v-if="upFinishText">{{upFinishText}}</view>
  45. </slot>
  46. </view>
  47. </scroll-view>
  48. <!-- 回到顶部按钮 (fixed元素,需写在scroll-view外面,防止滚动的时候抖动)-->
  49. <view
  50. class="s-pull-back-top"
  51. v-if="backTop"
  52. :class="{'is-show':isShowBackTop}"
  53. @click="onBackTop"
  54. >
  55. <slot name="backtop">
  56. <view class="default-back-top">
  57. <img src="./back-top.png" />
  58. </view>
  59. </slot>
  60. </view>
  61. </view>
  62. </template>
  63. <script>
  64. export default {
  65. name: 's-pull-scroll',
  66. data () {
  67. Object.assign(this, {
  68. pullType: '',
  69. scrollRealTop: 0, // 滚动条的位置
  70. preScrollY: 0,
  71. clientNum: 0,
  72. isExec: false,
  73. scrollHeight: 0,
  74. clientHeight: 0,
  75. bodyHeight: 0,
  76. windowTop: 0, // 可使用窗口的顶部位置
  77. windowBottom: 0, // 可使用窗口的底部位置
  78. page: 0,
  79. startPoint: null,
  80. lastPoint: null,
  81. startTop: 0,
  82. maxTouchmoveY: 0,
  83. inTouchend: false,
  84. moveTime: 0,
  85. moveTimeDiff: 0,
  86. movetype: 0,
  87. isMoveDown: false
  88. });
  89. return {
  90. scrollId: 's-pull-scroll-view-id-' + Math.random().toString(36).substr(2), // 随机生成mescroll的id(不能数字开头,否则找不到元素)
  91. downHight: 0, // 下拉刷新: 容器高度
  92. downRotate: 0, // 下拉刷新: 圆形进度条旋转的角度
  93. downText: '', // 下拉刷新: 提示的文本
  94. isEmpty: false, // 是否显示空布局
  95. isShowDownTip: false, // 下拉刷新提示结果
  96. isDownSuccess: false, // 下拉刷新成功
  97. isDownError: false, // 下拉刷新失败
  98. isDownReset: false, // 下拉刷新: 是否显示重置的过渡动画
  99. isDownLoading: false, // 下拉刷新: 是否显示加载中
  100. isUpLoading: false, // 上拉加载: 是否显示 "加载中..."
  101. isUpFinish: false, // 是否加载完毕
  102. isUpError: false, // 是否上拉加载出错
  103. isShowBackTop: false, // 是否显示回到顶部按钮
  104. scrollAble: true, // 是否禁止下滑 (下拉时禁止,避免抖动)
  105. scrollTop: 0 // 滚动条的位置
  106. };
  107. },
  108. props: {
  109. // class
  110. customClass: {
  111. type: String,
  112. default: ''
  113. },
  114. // 是否通过fixed固定高度, 默认true
  115. fixed: {
  116. type: Boolean,
  117. default: true
  118. },
  119. // 自定义头部时,头部高度(px)
  120. headerHeight: {
  121. type: [Number, String],
  122. default () {
  123. return 0;
  124. }
  125. },
  126. // 距顶部(rpx)
  127. top: {
  128. type: [Number, Array, String],
  129. default () {
  130. return 0;
  131. }
  132. },
  133. // 自定义底部时,底部高度(px)
  134. footerHeight: {
  135. type: [Number, String],
  136. default () {
  137. return 0;
  138. }
  139. },
  140. // 距底部(rpx)
  141. bottom: {
  142. type: [Number, Array, String],
  143. default () {
  144. return 0;
  145. }
  146. },
  147. // 是否阻止默认滚动
  148. preventTouchmove: {
  149. type: Boolean,
  150. default: true
  151. },
  152. // 下拉时文案
  153. pullingText: {
  154. type: String,
  155. default: '下拉刷新'
  156. },
  157. // 下拉释放时文案
  158. loosingText: {
  159. type: String,
  160. default: '释放刷新'
  161. },
  162. // 下拉释放后文案
  163. downLoadingText: {
  164. type: String,
  165. default: '正在刷新 ...'
  166. },
  167. // 上拉加载时文案
  168. upLoadingText: {
  169. type: String,
  170. default: '加载中 ...'
  171. },
  172. // 是否显示空布局
  173. showEmpty: {
  174. type: Boolean,
  175. default: true
  176. },
  177. // 刷新或加载数据为空时文案
  178. emptyText: {
  179. type: String,
  180. default: '暂无数据'
  181. },
  182. // 是否显示下拉刷新成功
  183. showDownSuccess: {
  184. type: Boolean,
  185. default: false
  186. },
  187. // 下拉刷新成功文案
  188. downSuccessText: {
  189. type: String,
  190. default: '刷新成功'
  191. },
  192. // 是否显示下拉刷新失败
  193. showDownError: {
  194. type: Boolean,
  195. default: false
  196. },
  197. // 下拉刷新失败文案
  198. downErrorText: {
  199. type: String,
  200. default: '刷新失败'
  201. },
  202. // 是否显示上拉加载时失败
  203. showUpError: {
  204. type: Boolean,
  205. default: true
  206. },
  207. // 上拉加载失败文案
  208. upErrorText: {
  209. type: String,
  210. default: '加载失败,点击重新加载'
  211. },
  212. // 是否显示上拉加载数据全部完成
  213. showUpFinish: {
  214. type: Boolean,
  215. default: true
  216. },
  217. // 上拉加载完毕文案
  218. upFinishText: {
  219. type: String,
  220. default: '暂无更多了'
  221. },
  222. // 下拉配置
  223. // 下拉回掉,参数为vm
  224. pullDown: Function,
  225. // 是否允许下拉刷新
  226. enablePullDown: {
  227. type: Boolean,
  228. default: true
  229. },
  230. downOffset: {
  231. type: Number,
  232. default: 100
  233. },
  234. downFps: {
  235. type: Number,
  236. default: 40
  237. },
  238. downMinAngle: {
  239. type: Number,
  240. default: 45
  241. },
  242. downInOffsetRate: {
  243. type: Number,
  244. default: 1
  245. },
  246. downOutOffsetRate: {
  247. type: Number,
  248. default: 0.4
  249. },
  250. downStartTop: {
  251. type: Number,
  252. default: 100
  253. },
  254. downBottomOffset: {
  255. type: Number,
  256. default: 20
  257. },
  258. // 上拉配置
  259. // 上拉回掉,参数为vm
  260. pullUp: Function,
  261. // 是否允许上拉加载
  262. enablePullUp: {
  263. type: Boolean,
  264. default: true
  265. },
  266. upOffset: {
  267. type: Number,
  268. default: 160
  269. },
  270. // 回到顶部
  271. backTop: Boolean,
  272. // 滚动距离大于多少rpx时触发
  273. backTopOffset: {
  274. type: Number,
  275. default: 1000
  276. }
  277. },
  278. watch: {
  279. top () {
  280. this.refreshClientHeight();
  281. },
  282. bottom () {
  283. this.refreshClientHeight();
  284. },
  285. headerHeight () {
  286. this.refreshClientHeight();
  287. },
  288. footerHeight () {
  289. this.refreshClientHeight();
  290. }
  291. },
  292. computed: {
  293. numTop () {
  294. return Number(this.headerHeight || 0) + this.upx2px(this.top);
  295. },
  296. numBottom () {
  297. return Number(this.footerHeight || 0) + this.upx2px(this.bottom);
  298. },
  299. numBackTopOffset () {
  300. return this.upx2px(this.backTopOffset);
  301. },
  302. numDownBottomOffset () {
  303. return this.upx2px(this.downBottomOffset);
  304. },
  305. numDownStartTop () {
  306. return this.upx2px(this.downStartTop);
  307. },
  308. numDownOffset () {
  309. return this.upx2px(this.downOffset);
  310. },
  311. numUpOffset () {
  312. return this.upx2px(this.upOffset);
  313. },
  314. fixedTop () {
  315. return this.fixed ? (this.numTop + this.windowTop) + 'px' : 0;
  316. },
  317. padTop () {
  318. return !this.fixed ? this.numTop + 'px' : 0;
  319. },
  320. fixedBottom () {
  321. return this.fixed ? (this.numBottom + this.windowBottom) + 'px' : 0;
  322. },
  323. padBottom () {
  324. return !this.fixed ? this.numBottom + 'px' : 0;
  325. },
  326. transition () {
  327. return this.isDownReset ? 'transform 300ms' : '';
  328. },
  329. translateY () {
  330. return this.downHight > 0 ? 'translateY(' + this.downHight + 'px)' : '';
  331. }
  332. },
  333. methods: {
  334. upx2px (value) {
  335. return (Array.isArray(value) ? value : [value]).map(num => uni.upx2px(Number(num || 0))).reduce((a, b) => a + b) || 0;
  336. },
  337. // 注册列表滚动事件,用于下拉刷新
  338. scroll (e) {
  339. e = e.detail;
  340. // 更新滚动条的位置
  341. this.scrollRealTop = e.scrollTop;
  342. // 更新滚动内容高度
  343. this.scrollHeight = e.scrollHeight;
  344. // 向上滑还是向下滑动
  345. const isScrollUp = e.scrollTop - this.preScrollY > 0;
  346. this.preScrollY = e.scrollTop;
  347. // 上滑 && 检查并触发上拉
  348. isScrollUp && this.triggerPullUp(true);
  349. // 回到顶部功能
  350. if (this.backTop) {
  351. // 返回顶部按钮的显示隐藏
  352. if (e.scrollTop >= this.numBackTopOffset) {
  353. this.isShowBackTop = true;
  354. } else {
  355. this.isShowBackTop = false;
  356. }
  357. }
  358. },
  359. // 注册列表touchstart事件,用于下拉刷新
  360. touchstart (e) {
  361. if (!this.pullDown || !this.enablePullDown) return;
  362. this.startPoint = this.getPoint(e); // 记录起点
  363. this.startTop = this.scrollRealTop; // 记录此时的滚动条位置
  364. this.lastPoint = this.startPoint; // 重置上次move的点
  365. this.maxTouchmoveY = this.bodyHeight - this.numDownBottomOffset; // 手指触摸的最大范围(写在touchstart避免body获取高度为0的情况)
  366. this.inTouchend = false; // 标记不是touchend
  367. },
  368. // 注册列表touchmove事件,用于下拉刷新
  369. touchmove (e) {
  370. if (!this.pullDown || !this.enablePullDown) return;
  371. if (!this.startPoint) return;
  372. // 节流
  373. const t = Date.now();
  374. if (this.moveTime && t - this.moveTime < this.moveTimeDiff) { // 小于节流时间,则不处理
  375. return;
  376. } else {
  377. this.moveTime = t;
  378. this.moveTimeDiff = 1000 / this.downFps;
  379. }
  380. let scrollRealTop = this.scrollRealTop; // 当前滚动条的距离
  381. let curPoint = this.getPoint(e); // 当前点
  382. let moveY = curPoint.y - this.startPoint.y; // 和起点比,移动的距离,大于0向下拉,小于0向上拉
  383. // (向下拉&&在顶部) scroll-view在滚动时不会触发touchmove,当触顶/底/左/右时,才会触发touchmove
  384. // scroll-view滚动到顶部时,scrollTop不一定为0; 在iOS的APP中scrollTop可能为负数,不一定和startTop相等
  385. if (moveY > 0 && (scrollRealTop <= 0 || (scrollRealTop <= this.numDownStartTop && scrollRealTop === this.startTop))) {
  386. // 可下拉的条件
  387. if (this.pullDown && this.enablePullDown && !this.inTouchend && !this.isDownLoading && !this.isUpLoading) {
  388. // 下拉的角度是否在配置的范围内
  389. let x = Math.abs(this.lastPoint.x - curPoint.x);
  390. let y = Math.abs(this.lastPoint.y - curPoint.y);
  391. let z = Math.sqrt(x * x + y * y);
  392. if (z !== 0) {
  393. let angle = Math.asin(y / z) / Math.PI * 180; // 两点之间的角度,区间 [0,90]
  394. if (angle < this.downMinAngle) return; // 如果小于配置的角度,则不往下执行下拉刷新
  395. }
  396. // 如果手指的位置超过配置的距离,则提前结束下拉,避免Webview嵌套导致touchend无法触发
  397. if (this.maxTouchmoveY > 0 && curPoint.y >= this.maxTouchmoveY) {
  398. this.inTouchend = true; // 标记执行touchend
  399. this.touchend(); // 提前触发touchend
  400. return;
  401. }
  402. this.preventDefault(e); // 阻止默认事件
  403. let diff = curPoint.y - this.lastPoint.y; // 和上次比,移动的距离 (大于0向下,小于0向上)
  404. // 下拉距离 < 指定距离
  405. if (this.downHight < this.numDownOffset) {
  406. if (this.movetype !== 1) {
  407. this.movetype = 1; // 加入标记,保证只执行一次
  408. // 下拉的距离进入offset范围内那一刻的回调
  409. this.scrollAble = false; // 禁止下拉,避免抖动
  410. this.isDownReset = false; // 不重置高度
  411. this.isDownLoading = false; // 不显示加载中
  412. this.downText = this.pullingText; // 设置文本
  413. this.isMoveDown = true; // 标记下拉区域高度改变,在touchend重置回来
  414. }
  415. this.downHight += diff * this.downInOffsetRate; // 越往下,高度变化越小
  416. // 指定距离 <= 下拉距离
  417. } else {
  418. if (this.movetype !== 2) {
  419. this.movetype = 2; // 加入标记,保证只执行一次
  420. // 下拉的距离大于offset那一刻的回调
  421. this.scrollAble = false; // 禁止下拉,避免抖动
  422. this.isDownReset = false; // 不重置高度
  423. this.isDownLoading = false; // 不显示加载中
  424. this.downText = this.loosingText; // 设置文本
  425. this.isMoveDown = true; // 标记下拉区域高度改变,在touchend重置回来
  426. }
  427. if (diff > 0) { // 向下拉
  428. this.downHight += Math.round(diff * this.downOutOffsetRate); // 越往下,高度变化越小
  429. } else { // 向上收
  430. this.downHight += diff; // 向上收回高度,则向上滑多少收多少高度
  431. }
  432. }
  433. // 设置旋转角度
  434. this.downRotate = 'rotate(' + 360 * (this.downHight / this.numDownOffset) + 'deg)';
  435. }
  436. }
  437. // 记录本次移动的点
  438. this.lastPoint = curPoint;
  439. },
  440. // 注册列表touchend事件,用于下拉刷新
  441. touchend (e) {
  442. if (!this.pullDown || !this.enablePullDown) return;
  443. // 如果下拉区域高度已改变,则需重置回来
  444. if (this.isMoveDown) {
  445. if (this.downHight >= this.numDownOffset) {
  446. // 符合触发刷新的条件
  447. this.triggerPullDown();
  448. } else {
  449. // 不符合的话 则重置
  450. this.downHight = 0;
  451. this.scrollAble = true; // 开启下拉
  452. this.isDownReset = true; // 重置高度
  453. this.isDownLoading = false; // 不显示加载中
  454. }
  455. this.movetype = 0;
  456. this.isMoveDown = false;
  457. } else if (this.scrollRealTop === this.startTop) { // 到顶/左/右/底的滑动事件
  458. const isScrollUp = this.getPoint(e).y - this.startPoint.y < 0; // 和起点比,移动的距离,大于0向下拉,小于0向上拉
  459. // 上滑 && 检查并触发上拉
  460. isScrollUp && this.triggerPullUp(true);
  461. }
  462. },
  463. preventDefault (e) {
  464. // 小程序不支持e.preventDefault
  465. // app的bounce只能通过配置pages.json的style.app-plus.bounce为"none"来禁止
  466. // cancelable:是否可以被禁用; defaultPrevented:是否已经被禁用
  467. if (e && e.cancelable && !e.defaultPrevented) e.preventDefault();
  468. },
  469. // 点击回到顶部的按钮回调
  470. onBackTop () {
  471. this.isShowBackTop = false; // 回到顶部按钮需要先隐藏,再执行回到顶部,避免闪动
  472. this.scrollTo(0); // 执行回到顶部
  473. },
  474. // 点击失败重新加载
  475. onUpErrorClick () {
  476. this.isUpError = false;
  477. if (this.pullType === 'down') {
  478. this.triggerPullDown();
  479. } else if (this.pullType === 'up') {
  480. this.triggerPullUp();
  481. }
  482. },
  483. scrollTo (y) {
  484. this.scrollTop = this.scrollRealTop;
  485. this.$nextTick(() => {
  486. this.scrollTop = y;
  487. });
  488. },
  489. /* 根据点击滑动事件获取第一个手指的坐标 */
  490. getPoint (e) {
  491. if (!e) {
  492. return {
  493. x: 0,
  494. y: 0
  495. };
  496. }
  497. if (e.touches && e.touches[0]) {
  498. return {
  499. x: e.touches[0].pageX,
  500. y: e.touches[0].pageY
  501. };
  502. } else if (e.changedTouches && e.changedTouches[0]) {
  503. return {
  504. x: e.changedTouches[0].pageX,
  505. y: e.changedTouches[0].pageY
  506. };
  507. } else {
  508. return {
  509. x: e.clientX,
  510. y: e.clientY
  511. };
  512. }
  513. },
  514. /* 滚动条到底部的距离 */
  515. getScrollBottom () {
  516. return this.scrollHeight - this.getClientHeight() - this.scrollRealTop;
  517. },
  518. /* 滚动容器的高度 */
  519. getClientHeight (isReal) {
  520. let h = this.clientHeight || 0;
  521. if (h === 0 && isReal !== true) { // 未获取到容器的高度,可临时取body的高度 (可能会有误差)
  522. h = this.bodyHeight;
  523. }
  524. return h;
  525. },
  526. /* 更新滚动区域的高度 (使内容不满屏和到底,都可继续翻页) */
  527. refreshClientHeight () {
  528. if (!this.isExec) {
  529. this.isExec = true; // 避免多次获取
  530. this.$nextTick(() => { // 确保dom已渲染
  531. uni.createSelectorQuery().in(this).select('#' + this.scrollId).boundingClientRect(data => {
  532. this.isExec = false;
  533. if (data) {
  534. this.clientHeight = data.height;
  535. } else if (this.clientNum != 3) { // 极少部分情况,可能dom还未渲染完毕,递归获取,最多重试3次
  536. this.clientNum = this.clientNum == 0 ? 1 : this.clientNum + 1;
  537. setTimeout(() => {
  538. this.refreshClientHeight();
  539. }, this.clientNum * 100);
  540. }
  541. }).exec();
  542. });
  543. }
  544. },
  545. /* 显示下拉进度布局 */
  546. showDownLoading () {
  547. this.isEmpty = false;
  548. this.isUpLoading = false;
  549. this.isUpError = false;
  550. this.isUpFinish = false;
  551. this.isShowDownTip = false;
  552. this.isDownSuccess = false;
  553. this.isDownError = false;
  554. this.isDownLoading = true; // 显示加载中
  555. this.downHight = this.numDownOffset; // 更新下拉区域高度
  556. this.scrollAble = true; // 开启下拉
  557. this.isDownReset = true; // 重置高度
  558. this.downText = this.downLoadingText; // 设置文本
  559. },
  560. /* 结束下拉刷新 */
  561. hideDownLoading () {
  562. if (this.isDownLoading) {
  563. if (this.isDownSuccess && this.showDownSuccess) {
  564. this.downText = this.downSuccessText;
  565. this.isShowDownTip = true;
  566. } else if (this.isDownError && this.showDownError) {
  567. this.downText = this.downErrorText;
  568. this.isShowDownTip = true;
  569. }
  570. if (this.isShowDownTip) {
  571. setTimeout(() => {
  572. this.downHight = 0;
  573. this.isDownReset = true; // 重置高度
  574. this.scrollHeight = 0;// 重置滚动区域,使数据不满屏时仍可检查触发翻页
  575. setTimeout(() => {
  576. this.scrollAble = true; // 开启下拉
  577. this.isDownLoading = false; // 不显示加载中
  578. this.isShowDownTip = false;
  579. }, 300);
  580. }, 1000);
  581. } else {
  582. this.downHight = 0;
  583. this.isDownReset = true; // 重置高度
  584. this.scrollHeight = 0;// 重置滚动区域,使数据不满屏时仍可检查触发翻页
  585. this.scrollAble = true; // 开启下拉
  586. this.isDownLoading = false; // 不显示加载中
  587. this.isShowDownTip = false;
  588. }
  589. }
  590. },
  591. /* 显示上拉加载中 */
  592. showUpLoading () {
  593. this.isEmpty = false;
  594. this.isUpError = false;
  595. this.isUpFinish = false;
  596. this.isUpLoading = true;
  597. },
  598. /* 结束上拉加载 */
  599. hideUpLoading () {
  600. if (this.isUpLoading) {
  601. this.$nextTick(() => {
  602. this.isUpLoading = false;
  603. });
  604. }
  605. },
  606. /* 触发下拉刷新 */
  607. triggerPullDown () {
  608. if (this.pullDown && this.enablePullDown && !this.isDownLoading && !this.isUpLoading) {
  609. // 下拉加载中...
  610. this.showDownLoading(); // 下拉刷新中...
  611. this.page = 1; // 预先加一页
  612. this.pullType = 'down';
  613. this.pullDown && this.pullDown.call(this.$parent, this);
  614. }
  615. },
  616. /* 触发上拉加载 */
  617. triggerPullUp (isCheck) {
  618. if (this.pullUp && this.enablePullUp && !this.isUpLoading && !this.isDownLoading && !this.isUpError && !this.isUpFinish) {
  619. // 是否校验在底部; 默认不校验
  620. if (isCheck && this.getScrollBottom() > this.numUpOffset) return;
  621. // 上拉加载中...
  622. this.showUpLoading();
  623. this.page++;
  624. this.pullType = 'up';
  625. this.pullUp && this.pullUp.call(this.$parent, this);
  626. // 更新容器的高度
  627. this.refreshClientHeight();
  628. }
  629. },
  630. refresh () {
  631. this.page = 0;
  632. this.isEmpty = false;
  633. this.isDownSuccess = false;
  634. this.isDownError = false;
  635. this.isShowDownTip = false;
  636. this.isUpError = false;
  637. this.isUpFinish = false;
  638. this.isDownLoading = false;
  639. this.isUpLoading = false;
  640. this.scrollTo(0);
  641. if (this.pullDown && this.enablePullDown) {
  642. this.triggerPullDown();
  643. } else if (this.pullUp && this.enablePullUp) {
  644. this.triggerPullUp();
  645. }
  646. },
  647. /* 正常加载成功 */
  648. success () {
  649. if (this.isDownLoading) {
  650. this.isDownSuccess = true;
  651. }
  652. this.hideDownLoading();
  653. this.hideUpLoading();
  654. },
  655. /* 加载失败 */
  656. error () {
  657. if (this.page > 0) {
  658. this.page--;
  659. }
  660. if (this.isDownLoading) {
  661. this.isDownError = true;
  662. } else if (this.isUpLoading) {
  663. this.isUpError = true;
  664. }
  665. this.hideDownLoading();
  666. this.hideUpLoading();
  667. },
  668. /* 没有数据 */
  669. empty () {
  670. if (this.isDownLoading) {
  671. this.isDownSuccess = true;
  672. }
  673. this.isEmpty = true;
  674. this.isUpFinish = true;
  675. this.hideDownLoading();
  676. this.hideUpLoading();
  677. },
  678. /* 全部数据加载完毕 */
  679. finish () {
  680. this.hideDownLoading();
  681. this.hideUpLoading();
  682. this.isUpFinish = true;
  683. }
  684. },
  685. created () {
  686. // 设置高度
  687. uni.getSystemInfo({
  688. success: (res) => {
  689. if (res.windowTop) this.windowTop = res.windowTop; // 修正app和H5的top值
  690. if (res.windowBottom) this.windowBottom = res.windowBottom; // 修正app和H5的bottom值
  691. this.bodyHeight = res.windowHeight;// 使down的bottomOffset生效
  692. }
  693. });
  694. },
  695. mounted () {
  696. // 设置容器的高度
  697. this.refreshClientHeight = this.refreshClientHeight.bind(this);
  698. uni.onWindowResize(this.refreshClientHeight);
  699. this.refreshClientHeight();
  700. this.$el && this.$el.addEventListener && this.$el.addEventListener('touchmove', e => {
  701. this.preventTouchmove && e.preventDefault();
  702. });
  703. },
  704. beforeDestroy () {
  705. uni.offWindowResize(this.refreshClientHeight);
  706. }
  707. };
  708. </script>
  709. <style lang="scss">
  710. .s-pull-scroll {
  711. height: 100%;
  712. -webkit-overflow-scrolling: touch;
  713. .s-pull-scroll-view {
  714. position: relative;
  715. width: 100%;
  716. height: 100%;
  717. overflow-y: auto;
  718. box-sizing: border-box;
  719. }
  720. /* 定位的方式固定高度 */
  721. .is-fixed {
  722. z-index: 1;
  723. position: fixed;
  724. top: 0;
  725. left: 0;
  726. right: 0;
  727. bottom: 0;
  728. width: auto;
  729. height: auto;
  730. }
  731. .s-pull-down-wrap,
  732. .s-pull-up-wrap,
  733. .s-pull-tip-wrap {
  734. display: flex;
  735. justify-content: center;
  736. align-items: center;
  737. font-size: 28rpx;
  738. color: #969799;
  739. }
  740. .s-pull-down-wrap {
  741. position: absolute;
  742. left: 0;
  743. width: 100%;
  744. transform: translateY(-100%);
  745. }
  746. .s-pull-up-wrap,
  747. .s-pull-tip-wrap {
  748. height: 100rpx;
  749. }
  750. /* 旋转loading */
  751. .s-pull-loading-icon {
  752. width: 30rpx;
  753. height: 30rpx;
  754. display: inline-block;
  755. vertical-align: middle;
  756. border-radius: 50%;
  757. border: 2rpx solid #969799;
  758. border-bottom-color: transparent;
  759. box-sizing: border-box;
  760. &:first-child {
  761. margin-right: 16rpx;
  762. }
  763. }
  764. /* 旋转动画 */
  765. .s-pull-loading-rotate {
  766. animation: s-pull-loading-rotate 0.6s linear infinite;
  767. }
  768. @keyframes s-pull-loading-rotate {
  769. 0% {
  770. transform: rotate(0deg);
  771. }
  772. 100% {
  773. transform: rotate(360deg);
  774. }
  775. }
  776. /* 回到顶部的按钮 */
  777. .s-pull-back-top {
  778. position: relative;
  779. z-index: 99;
  780. opacity: 0;
  781. pointer-events: none;
  782. transition: opacity 0.3s linear;
  783. &.is-show {
  784. opacity: 1;
  785. pointer-events: auto;
  786. }
  787. }
  788. .default-back-top {
  789. position: fixed;
  790. right: 20rpx;
  791. bottom: calc(var(--window-bottom) + 25rpx);
  792. img {
  793. width: 72rpx;
  794. height: 72rpx;
  795. border-radius: 50%;
  796. }
  797. }
  798. }
  799. </style>