Android:如何通过编程创建滑动手势事件

6

我一直在谷歌上寻找这个特定情况的答案,但惊人的是没有找到任何答案(只有关于检测而不是创建滑动的答案)。我想要实际上在屏幕上从右向左模拟一个滑动(不管在哪里,但理想情况下是从右边缘到左边缘)。我尝试了类似这样的东西,但我得到了NPE:

final float viewWidth = view.getWidth();
    TouchUtils.drag(null,viewWidth,1f,0f,0f,1);

原因是我需要在一个无限的viewpager中强制滑到下一页,但实际上无法确定页面的位置(所以不能使用setCurrentItem)。 有什么好的想法吗?

3
那我不能只使用setCurrentItem方法。那为什么不调用pager.setCurrentItem(pager.getCurrentItem()+1, true);呢?你可能认为它是“无限”的,但ViewPager仍然使用项目位置。 - CommonsWare
由于某种原因,这样做会导致整个应用程序冻结 - 而logcat并没有帮助... - user1548172
2
好的,“TouchUtils”是用于测试而不是生产代码。虽然你至少在理论上可以模拟触摸事件到你自己的应用程序,但我并不相信这将证明是一个稳定和可靠的解决方案来解决你的问题。我建议你设置一些断点,找出“setCurrentItem()”在哪里无法正常工作。 - CommonsWare
我想在生产环境中使用测试类,例如Instrumentation、Touchutils等等。目的是为了远程模拟我们的应用程序。是否可以在生产环境中使用测试类? - Ramesh_D
1个回答

1

正如之前所说的,TouchUtils.drag使用的是只能在androidTest中使用的Instrumentation。 我写了一些扩展函数,它们不需要Instrumentation,因此可以在发布版本中使用:

fun ViewGroup.performSwipeToLeft(target: View) {
    this.performSwipe(target, distanceX = -this.width * .5f, distanceY = 0f)
}
fun ViewGroup.performSwipeToRight(target: View) {
    this.performSwipe(target, distanceX = +this.width * .5f, distanceY = 0f)
}
fun ViewGroup.performSwipeToTop(target: View) {
    this.performSwipe(target, distanceX = 0f, distanceY = -this.height * .5f)
}
fun ViewGroup.performSwipeToBottom(target: View) {
    this.performSwipe(target, distanceX = 0f, distanceY = +this.width * .5f)
}

fun ViewGroup.performSwipe(target: View, distanceX: Float, distanceY: Float) {
    val parentCoords = intArrayOf(0, 0)
    this.getLocationInWindow(parentCoords)

    val childCoords = intArrayOf(0, 0)
    target.getLocationInWindow(childCoords)

    val initGlobalX = childCoords[0].toFloat() + 1f
    val initGlobalY = childCoords[1].toFloat() + 1f

    val initLocalX = (childCoords[0] - parentCoords[0]).toFloat() + 1f
    val initLocalY = (childCoords[1] - parentCoords[1]).toFloat() + 1f

    val downTime = SystemClock.uptimeMillis()
    var eventTime = SystemClock.uptimeMillis()

    this.dispatchTouchEvent(
        MotionEvent.obtain(
            downTime,
            eventTime,
            MotionEvent.ACTION_DOWN,
            initGlobalX,
            initGlobalY,
            0
        ).apply {
            setLocation(initLocalX, initLocalY)
            source = InputDevice.SOURCE_TOUCHSCREEN
        }
    )

    val steps = 20
    var i = 0
    while (i in 0..steps) {
        val globalX = initGlobalX + i * distanceX / steps
        val globalY = initGlobalY + i * distanceY / steps
        val localX = initLocalX + i * distanceX / steps
        val localY = initLocalY + i * distanceY / steps
        if (globalX <= 10f || globalY <= 10f) {
            break
        }
        this.dispatchTouchEvent(
            MotionEvent.obtain(
                downTime,
                ++eventTime,
                MotionEvent.ACTION_MOVE,
                globalX,
                globalY,
                0
            ).apply {
                setLocation(localX, localY)
                source = InputDevice.SOURCE_TOUCHSCREEN
            }
        )
        i++
    }

    this.dispatchTouchEvent(
        MotionEvent.obtain(
            downTime,
            ++eventTime,
            MotionEvent.ACTION_UP,
            initGlobalX + i * distanceX,
            initGlobalY + i * distanceY,
            0
        ).apply {
            setLocation(initLocalX + i * distanceX, initLocalY + i * distanceY)
            source = InputDevice.SOURCE_TOUCHSCREEN
        }
    )
}

要使用它,您需要将当前页面内容作为参数传递,并将viewPager作为此函数的接收者:
val viewPager = ...
viewPager.performSwipeToLeft(viewPager.getChildAt(viewPager.currentItem))

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接