如何通过webview覆盖一些手势事件,但让其他人通过?

时间:2011-03-02 19:18:09

标签: android webview android-webview gestures

我还在搞清楚SDK的来龙去脉,所以请耐心等待。

所以我正在尝试实现一个Webview,它可以保持多点触控缩放控件和垂直滚动(webview类处理它们),同时在Webview上锁定水平滚动,并实现水平fling以寻找新数据。

我发现了这个:Fling Gesture and Webview in Android 。因此我使用此代码了解如何在我的应用程序中实现此功能,因此我正在那里工作(当然是答案)。

并开始为我的西装编写代码。例如:

   if (event1.getRawX() > event2.getRawX() && StrictMath.abs(event1.getRawY()-event2.getRawY())<100) {
    show_toast("swipe left");
   } else if(event1.getRawX() < event2.getRawX() && StrictMath.abs(event1.getRawY()-event2.getRawY())<100){
    show_toast("swipe right");

实际上只检测水平褶皱。

但是,我还没弄清楚如何确保水平flings触发我选择的动作,而垂直flings仍然控制着相关WebView的滚动。

我的第一个想法是在webview上尝试类似pageUp / pageDown或scrollTo的东西,但是因为我将webview类扩展到MyWebView而因此还没有这种类型的instanciated对象,所以无法工作。 / p>

如果需要,这是完整的代码:

package test.fling;


import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.webkit.WebView;
import android.widget.Toast;

public class testicules extends Activity {
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    MyWebView webview = new MyWebView(this);
    webview.loadUrl("http://en.wikipedia.org/wiki/Android");
    setContentView(webview);
}

class MyWebView extends WebView {
 Context context;
 GestureDetector gd;

public MyWebView(Context context) {
super(context);

this.context = context;
     gd = new GestureDetector(context, sogl);
}

@Override
public boolean onTouchEvent(MotionEvent event) {
return gd.onTouchEvent(event);
}

 GestureDetector.SimpleOnGestureListener sogl = new GestureDetector.SimpleOnGestureListener() {
  public boolean onDown(MotionEvent event) {
   return true;
  }
  public boolean onFling(MotionEvent event1, MotionEvent event2, float velocityX, float velocityY) {
   if (event1.getRawX() > event2.getRawX() && StrictMath.abs(event1.getRawY()-event2.getRawY())<100) {
    show_toast("swipe left");
   } else if(event1.getRawX() < event2.getRawX() && StrictMath.abs(event1.getRawY()-event2.getRawY())<100){
    show_toast("swipe right");
   } else {
     //MyWebView.pageUp(true); /*can't work, as explained above*/
   }
return true;
  }
 };

 void show_toast(final String text) {
  Toast t = Toast.makeText(context, text, Toast.LENGTH_SHORT);
  t.show();
 }
}
}

任何帮助将不胜感激。谢谢!

2 个答案:

答案 0 :(得分:7)

你想要做的是在onTouchEvent中添加代码以检查GestureDetector的onTouchEvent的返回值,如果为false,则使用该事件调用WebView超类,如下所示:

<code>
    @Override 

    public boolean onTouchEvent(MotionEvent event) {

        return (gd.onTouchEvent(event) 
            || super.onTouchEvent(event)); 
    }  
</code>

然后,如果您正在处理事件,则您的onFling方法可以返回true,如果您希望WebView处理事件,则返回false。您还需要更改onDown以返回false,因此WebView有机会获得初始向下事件。

答案 1 :(得分:1)

我编写了从WebView扩展的类,可以轻松检测滑动。

import android.content.Context;
import android.util.AttributeSet;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.GestureDetector.SimpleOnGestureListener;
import android.webkit.WebView;
import android.widget.Toast;

public class HtmlImageView extends WebView {

Context mContext;
GestureDetector gestureDetector = new GestureDetector(new MyGestureDetector());

public HtmlImageView(Context context) {
    super(context);
    mContext=context;
}

public HtmlImageView(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    mContext=context;
}

public HtmlImageView(Context context, AttributeSet attrs) {
    super(context, attrs);
    mContext=context;
}

private static final int SWIPE_MIN_DISTANCE = 120;
private static final int SWIPE_MAX_OFF_PATH = 250;
private static final int SWIPE_THRESHOLD_VELOCITY = 200;

class MyGestureDetector extends SimpleOnGestureListener {

    @Override
    public boolean onDown(MotionEvent e) {
        return true;
    }

    @Override
    public boolean onScroll(MotionEvent e1, MotionEvent e2,
            float distanceX, float distanceY) {
        return super.onScroll(e1, e2, distanceX, distanceY);
    }

    @Override
    public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
            float velocityY) {
        try {
            if (Math.abs(e1.getY() - e2.getY()) > SWIPE_MAX_OFF_PATH)
                return false;
            // right to left swipe
            if (e1.getX() - e2.getX() > SWIPE_MIN_DISTANCE
                    && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {
                Toast.makeText(mContext, "Left Swipe", Toast.LENGTH_SHORT)
                        .show();
            } else if (e2.getX() - e1.getX() > SWIPE_MIN_DISTANCE
                    && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {
                Toast.makeText(mContext, "Right Swipe", Toast.LENGTH_SHORT)
                        .show();
            }
        } catch (Exception e) {
            // nothing
        }
        return false;
    }
}

public boolean onTouchEvent(MotionEvent event) {

    return (gestureDetector.onTouchEvent(event) 
        || super.onTouchEvent(event)); 
}  
}