欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

android 控件同时监听单击和双击实例

程序员文章站 2022-07-24 15:41:46
不适用click而用touch自定义监听:class myongesturelistener extends gesturedetector.simpleongesturelistener { @ov...

不适用click而用touch

自定义监听:

class myongesturelistener extends gesturedetector.simpleongesturelistener {
 @override
 public boolean ondoubletap(motionevent e) {
  //点赞
  mloadinglistener.onfinishedloading("0");//取消点赞 是一个接口
  //已经点赞 更换图片 1:已经点赞 0 :没有点赞
  if (liketype.equals("1")){
   string getlike = tvlike.gettext().tostring().trim();
   int il = integer.valueof(getlike) - 1;
   tvlike.settext(string.valueof(il));
   mivvideolike.setimageresource(r.mipmap.video_likegray);
   liketype = "0";
  }else {
   string getlike = tvlike.gettext().tostring().trim();
   int il = integer.valueof(getlike) + 1;
   tvlike.settext(string.valueof(il));
   mivvideolike.setimageresource(r.mipmap.video_xin_red);
   liketype = "1";
  }

  mreltotal.addloveview(e.getrawx(),e.getrawy());
  return super.ondoubletap(e);
 }

 @override
 public boolean onsingletapconfirmed(motionevent e) {

  if (monvideoplayereventlistener.isplaying()){
   monvideoplayereventlistener.pause();
   mivvideoshow.setvisibility(visible);
  }else {
   monvideoplayereventlistener.start();
   mivvideoshow.setvisibility(gone);
  }

  return super.onsingletapconfirmed(e);
 }

}

使用:

mygesturedetector = new gesturedetector(mcontext, new myongesturelistener());

mreltotal.setontouchlistener(new view.ontouchlistener() {

 @override//可以捕获触摸屏幕发生的event事件
 public boolean ontouch(view v, motionevent event) {
  //使用gesturedetector转发motionevent对象给ongesturelistener

  mygesturedetector.ontouchevent(event);
  return true;
 }
});

补充知识:android 利用gesturedetector处理不太常用的一些点击事件

关于gesturedetector ,在网上有很多资料是描述如下常见情况下的回调:

点击一下非常快的(不滑动)touchup:

ondown->onsingletapup->onsingletapconfirmed

点击一下稍微慢点的(不滑动)touchup:

ondown->onshowpress->onsingletapup->onsingletapconfirmed

长按:

ondown-->onshowpress-->onlongpress

两次连续点击(第二次点击之后立即抬起):

(第一次点击)ondown->onsingletapup->(第二次点击)ondoubletap->ondoubletapevent->ondown->onshowpress->ondoubletapevent

点击之后滑动:

ondown->onshowpress->onscroll->......(->onfling)(视速度快慢)

但是这些并不能完美符合我们的需求,我们还会遇到以下需求:

双击之后拖动:

我在每个回调函数打上log,双击之后拖动的log如下:

android 控件同时监听单击和双击实例

(中间若干个都是ontouch: move)

android 控件同时监听单击和双击实例

首先可以看到双击(ondoubletapevent)被回调之后的move事件都被传递到了ondoubletapevent中。但是当你第二次点击时间达到一定之后,onlongpress会被回调,而当onlongpress被回调之后,move动作就被gesturedetector无视了,直到up动作出现,显然这不是我们想要的。

那么我们可以在ondoubletapevent中接收到down动作时,利用setislongpressenabled()使longpress不会触发,然后在ondoubletapevent中接收到up动作时再恢复即可。

 @override
 public boolean ondoubletapevent(motionevent e) {
  log.d(tag, "ondoubletapevent: ");
  switch (e.getaction()) {
   case motionevent.action_down:
    gesturedetector.setislongpressenabled(false);
    //action
    break;
   case motionevent.action_move:
    //action
    break;
   case motionevent.action_up:
    //action
    gesturedetector.setislongpressenabled(true);
    break;
  }
  return true;
 }

更改之后,再进行测试,如下:

android 控件同时监听单击和双击实例

(中间若干个ontouch: move,ondoubletapevent)

android 控件同时监听单击和双击实例

长按拖动:

在onlongpress被回调之后,gesturedetector不会对move动作调用任何函数,除非直到一个up动作出现,但用户的习惯不可能是这样。

因此对于这个需求我们需要在ontouch中对move动作进行识别。

首先修改onlongpress函数,在长按之后更新状态为可拖拽,然后对ontouch中的move动作我们自己调用onscroll(不一定要onscroll),并且在onscroll中完成动作,因此需要记录上一次的motionevent:

 @override
 public void onlongpress(motionevent e) {
  log.d(tag, "onlongpress: ");
  lastmotionevent = e;
  draggable = true;
 }

然后在ontouch函数中:

 @override
 public boolean ontouch(view v, motionevent event) {
  boolean result = gesturedetector.ontouchevent(event);
  // 如果gesturedetector不消费动作
  if (!result) {
   switch (event.getaction()) {
    case motionevent.action_down:
     break;
    case motionevent.action_move:
     // 可拖拽状态下调用onscroll,同时更新lastmotionevent
     if (draggable) {
      onscroll(lastmotionevent, event, lastmotionevent.getx() - event.getx(), lastmotionevent.gety() - event.gety());
      lastmotionevent = motionevent.obtain(event);
     }
     result = true;
     break;
    case motionevent.action_up:
     // 恢复为不可拖拽状态
     if (draggable) {
      onscroll(lastmotionevent, event, lastmotionevent.getx() - event.getx(), lastmotionevent.gety() - event.gety());
      lastmotionevent = null;
      draggable = false;
     }
     result = true;
     break;
   }
  }
  return result;
 }

处理点击-滑动之后的action_up

滑动的回调是这样的

ondown->onshowpress->onscroll->......(->onfling)(视速度快慢)

如果onfling没有被回调的话,我们无法对onscroll之后的up动作响应,因此对于这个动作,我们也要在ontouch中处理。

首先要明确: android 控件同时监听单击和双击实例 从点a滑动到点b,并且在点b松手的话,在没有触发onfling的情况下,会回调onscroll(ea, eb, distancex, distancey),然后gesturedetector不消费点b的up事件,此时我们在ontouch中处理这个up事件。

代码也很简单,在长按拖动的基础上增加一个else即可:

    case motionevent.action_up:
     if (draggable) {
      onscroll(lastmotionevent, event, lastmotionevent.getx() - event.getx(), lastmotionevent.gety() - event.gety());
      lastmotionevent = null;
      draggable = false;
     } else {
      afterscroll(event);
     }
     result = true;
     break;

具体需要处理何种点击事件可根据实际修改,希望分享的内容能给你一点idea。

如果错误,请指出。

以上这篇android 控件同时监听单击和双击实例就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。