经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 移动开发 » Android » 查看文章
Android Banner - ViewPager 02
来源:cnblogs  作者:真菜啊  时间:2023/4/6 8:43:19  对本文有异议

Android Banner - ViewPager 02

现在来给viewpager实现的banenr加上自动轮播

自动轮播的原理,使用handler的延迟消息来实现。

自动轮播实现如下内容

  1. 开始轮播&停止轮播

  2. 可配置轮播时长、轮播方向

  3. 通过自定义属性来配置轮播时长,方向

  4. 感知生命周期,可见时开始轮播,不可见时停止轮播

  5. 感知手指触摸,触摸按下时停止轮播,抬起重新计时

开始&停止轮播

banner对外提供接口,开始轮播

  1. fun startLoop(){
  2. }
  3. fun stopLoop(){
  4. }

定义handler实现轮播

  1. // 创建handler
  2. fun startLoop() {
  3. if (loopHandler == null) {
  4. loopHandler = Handler(Looper.getMainLooper()) { message ->
  5. return@Handler when (message.what) {
  6. LOOP_NEXT -> {
  7. // 定义消息处理
  8. loopNext()
  9. true
  10. }
  11. else -> false
  12. }
  13. }
  14. }
  15. // 移除正在轮播的消息
  16. loopHandler?.removeMessages(LOOP_NEXT)
  17. // 发送延迟轮播的消息
  18. loopHandler?.sendEmptyMessageDelayed(LOOP_NEXT, mLoopDuration)
  19. }
  20. private fun loopNext() {
  21. val count = adapter?.count ?: 0
  22. // 当pager数量为0或者1时,不用轮播
  23. if (count in 0..1) return
  24. val curr = when (currentItem) {
  25. in 0..count - 2 -> {
  26. currentItem + 1
  27. }
  28. count - 1 -> 0
  29. else -> 0
  30. }
  31. setCurrentItem(curr, true)
  32. loopHandler?.sendEmptyMessageDelayed(LOOP_NEXT, mLoopDuration)
  33. }

可配置轮播时长、轮播方向

定义接口

  1. /**
  2. * 设置轮播时长,有效数据必须大于0,否则使用默认数据5S
  3. * @param duration Long
  4. */
  5. fun setLoopDuration(duration: Long) {
  6. if (duration < 0) {
  7. // 小于0的数据认为是非法数据,使用默认设置
  8. return
  9. }
  10. this.mLoopDuration = duration
  11. }
  12. /**
  13. * 设置轮播方向,默认[LoopOrientation.LTR]
  14. * @param orientation Int
  15. */
  16. fun setLoopOrientation(@LoopOrientation orientation: Int) {
  17. this.mLoopOrientation = orientation
  18. }

轮播处理参数

  1. private fun loopNext() {
  2. val count = adapter?.count ?: 0
  3. // 当pager数量为0或者1时,不用轮播
  4. if (count in 0..1) return
  5. val curr = when (mLoopOrientation) {
  6. LoopOrientation.RTL -> {
  7. when (currentItem) {
  8. in 1..count - 1 -> {
  9. currentItem - 1
  10. }
  11. else -> count - 1 // 0
  12. }
  13. }
  14. else -> {
  15. when (currentItem) {
  16. in 0..count - 2 -> {
  17. currentItem + 1
  18. }
  19. else -> 0 // count - 1
  20. }
  21. }
  22. }
  23. setCurrentItem(curr, true)
  24. mLoopHandler?.sendEmptyMessageDelayed(LOOP_NEXT, mLoopDuration)
  25. }

通过自定义属性来配置轮播时长,方向

  1. <resources>
  2. <declare-styleable name="VPBanner">
  3. <attr name="vp_loop_duration" format="integer" />
  4. <attr name="vp_loop_orientation" format="enum" >
  5. <enum name="ltr" value="1" />
  6. <enum name="rtl" value="0" />
  7. </attr>
  8. <attr name="vp_auto_loop" format="boolean" />
  9. </declare-styleable>
  10. </resources>

读取属性

  1. constructor(context: Context, attrs: AttributeSet?) : super(context, attrs) {
  2. // 读取自定义的属性
  3. val typedArray = context.obtainStyledAttributes(attrs, R.styleable.VPBanner)
  4. this.mLoopDuration = typedArray.getInt(
  5. R.styleable.VPBanner_vp_loop_duration,
  6. DEFAULT_LOOP_DURATION
  7. ).toLong()
  8. this.mAutoLoop = typedArray.getBoolean(R.styleable.VPBanner_vp_auto_loop, false)
  9. this.mLoopOrientation =
  10. typedArray.getInt(R.styleable.VPBanner_vp_loop_orientation, LoopOrientation.LTR)
  11. Log.d("VPBanner","ld:${this.mLoopDuration},al:$mAutoLoop,lo:$mLoopOrientation")
  12. typedArray?.recycle()
  13. }

感知生命周期,可见时开始轮播,不可见时停止轮播

实现生命周期感知

  1. class VPBanner : ViewPager, DefaultLifecycleObserver {
  2. override fun onResume(owner: LifecycleOwner) {
  3. Log.d(TAG, "onResume")
  4. if (this.mAutoLoop) {
  5. startLoop()
  6. }
  7. }
  8. override fun onPause(owner: LifecycleOwner) {
  9. Log.d(TAG, "onResume")
  10. stopLoop()
  11. }
  12. }

感知手指触摸,触摸按下时停止轮播,抬起重新计时

重写onTouchEvent方法

  1. override fun onTouchEvent(ev: MotionEvent?): Boolean {
  2. when (ev?.action) {
  3. MotionEvent.ACTION_DOWN -> stopLoop()
  4. MotionEvent.ACTION_UP, MotionEvent.ACTION_CANCEL -> {
  5. prepareLoop()
  6. }
  7. }
  8. return super.onTouchEvent(ev)
  9. }
  10. private fun prepareLoop() {
  11. if (this.mAutoLoop && this.mResumed) {
  12. startLoop()
  13. }
  14. }

原文链接:https://www.cnblogs.com/android-lol/p/17291146.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号