Мне нужно создать небольшую область текста. В этой области текста, когда я дважды щелкнул, она переместится к следующей активности. Как я могу это сделать?
DoubleTap в android
Ответ 1
Если вы правильно настроите настройку, OnDoubleTapListener
, в пределах GestureListener
очень полезен. Вам не нужно обрабатывать каждый отдельный кран и подсчитывать время между ними. Вместо этого пусть Android ручка для вас, что кран, двойное нажатие, свиток или бросок может быть. С помощью класса-помощника SimpleGestureListener
, который реализует GestureListener
и OnDoubleTapListener
, вам не нужно много делать.
findViewById(R.id.touchableText).setOnTouchListener(new OnTouchListener() {
private GestureDetector gestureDetector = new GestureDetector(Test.this, new GestureDetector.SimpleOnGestureListener() {
@Override
public boolean onDoubleTap(MotionEvent e) {
Log.d("TEST", "onDoubleTap");
return super.onDoubleTap(e);
}
... // implement here other callback methods like onFling, onScroll as necessary
});
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.d("TEST", "Raw event: " + event.getAction() + ", (" + event.getRawX() + ", " + event.getRawY() + ")");
gestureDetector.onTouchEvent(event);
return true;
}
});
Примечание. Я проверил довольно много времени, чтобы узнать, какая правильная смесь return true
и return false
. Это была действительно сложная часть.
Другое примечание. Когда вы проверите это, сделайте это на реальном устройстве вместо эмулятора. У меня была настоящая проблема с тем, чтобы мышь была достаточно быстрой, чтобы создать событие onFling. Реальные пальцы на реальных устройствах выглядят намного быстрее.
Ответ 2
Лучшей альтернативой является создание легкого абстрактного класса
public abstract class DoubleClickListener implements OnClickListener {
private static final long DOUBLE_CLICK_TIME_DELTA = 300;//milliseconds
long lastClickTime = 0;
@Override
public void onClick(View v) {
long clickTime = System.currentTimeMillis();
if (clickTime - lastClickTime < DOUBLE_CLICK_TIME_DELTA){
onDoubleClick(v);
lastClickTime = 0;
} else {
onSingleClick(v);
}
lastClickTime = clickTime;
}
public abstract void onSingleClick(View v);
public abstract void onDoubleClick(View v);
}
И используйте его как
view.setOnClickListener(new DoubleClickListener() {
@Override
public void onSingleClick(View v) {
}
@Override
public void onDoubleClick(View v) {
}
});
Ответ 3
очень простое логическое использование ниже кода
boolean firstTouch = false;
@Override
public boolean onTouchEvent(MotionEvent event) {
if(event.getAction() == event.ACTION_DOWN){
if(firstTouch && (Helper.getCurrentTimeInMilliSeconds() - time) <= 300) {
//do stuff here for double tap
Log.e("** DOUBLE TAP**"," second tap ");
firstTouch = false;
} else {
firstTouch = true;
time = Helper.getCurrentTimeInMilliSeconds();
Log.e("** SINGLE TAP**"," First Tap time "+time);
return false;
}
}
return true;
}
Ответ 4
----------
Я использовал другой подход для реализации двойного касания на представлениях Android. Я создал свою собственную логику для обнаружения двойного нажатия и ее очень легко реализовать.
Вот шаги для этого:
1. Установите onTouchListener на представление, которое вы хотите получить прикосновением.
2. Внедрить метод onTouch (просмотр, событие). (При двойном нажатии ключ должен обнаружить два события ACTION_DOWN и ACTION_UP. Для этого нам нужно будет вычислить продолжительность времени между двумя последовательными событиями понижения).
Вот логика для достижения этого:
/* variable for counting two successive up-down events */
int clickCount = 0;
/*variable for storing the time of first click*/
long startTime;
/* variable for calculating the total time*/
long duration;
/* constant for defining the time duration between the click that can be considered as double-tap */
static final MAX_DURATION = 500;
@Override
public boolean onTouch (View v, MotionEvent event)
{
switch(event.getAction() & MotionEvent.ACTION_MASK)
{
case MotionEvent.ACTION_DOWN:
startTime = System.currentTimeMillis();
clickCount++;
break;
case MotionEvent.ACTION_UP:
long time = System.currentTimeMillis() - startTime;
duration= duration + time;
if(clickCount == 2)
{
if(totalTime <= DURATION)
{
Toast.makeText(captureActivity.this, "double tap",Toast.LENGTH_LONG).show();
}
clickCount = 0;
duration = 0;
break;
}
}
return true;
}
==== EDIT ======
Для меня приведенное выше неприемлемо с изменениями, предложенными в коментах - тайм-аут не работает для вышеуказанной логики.
используйте это вместо
@Override public boolean onTouch (просмотр paramView, событие MotionEvent) { switch (event.getAction() и MotionEvent.ACTION_MASK) {
case MotionEvent.ACTION_UP:
clickCount++;
if (clickCount==1){
startTime = System.currentTimeMillis();
}
else if(clickCount == 2)
{
long duration = System.currentTimeMillis() - startTime;
if(duration <= ONE_SECOND)
{
Toast.makeText(captureActivity.this, "double tap",Toast.LENGTH_LONG).show();
clickCount = 0;
duration = 0;
}else{
clickCount = 1;
startTime = System.currentTimeMillis();
}
break;
}
}
return true;
}
Ответ 5
import android.app.Activity;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.GestureDetector.SimpleOnGestureListener;
import android.widget.Toast;
public class SimpleGestureFilter extends SimpleOnGestureListener
{
public final static int SWIPE_UP = 1;
public final static int SWIPE_DOWN = 2;
public final static int SWIPE_LEFT = 3;
public final static int SWIPE_RIGHT = 4;
public final static int MODE_TRANSPARENT = 0;
public final static int MODE_SOLID = 1;
public final static int MODE_DYNAMIC = 2;
private final static int ACTION_FAKE = -13;
private int swipe_Min_Distance = 100;
private int swipe_Max_Distance = 350;
private int swipe_Min_Velocity = 100;
private int mode = MODE_DYNAMIC;
private boolean running = true;
private boolean tapIndicator = false;
private Activity context;
private GestureDetector detector;
private SimpleGestureListener listener;
public SimpleGestureFilter(Activity context,SimpleGestureListener sgf)
{
this.context = context;
this.detector = new GestureDetector(context, this);
this.listener = sgf;
}
public void onTouchEvent(MotionEvent me)
{
// TODO Auto-generated method stub
if(!this.running)
return;
boolean result=this.detector.onTouchEvent(me);
if(this.mode==MODE_SOLID)
me.setAction(MotionEvent.ACTION_CANCEL);
else if(this.mode==MODE_DYNAMIC)
{
if(me.getAction()==ACTION_FAKE)
me.setAction(MotionEvent.ACTION_UP);
else if(result)
me.setAction(MotionEvent.ACTION_CANCEL);
else if(this.tapIndicator)
{
me.setAction(MotionEvent.ACTION_DOWN);
this.tapIndicator=false;
}
}
}
public void setMode(int m)
{
this.mode=m;
}
public int getMode()
{
return this.mode;
}
public void setEnabled(boolean status)
{
this.running=status;
}
public void setSwipeMaxDistance(int distance)
{
this.swipe_Max_Distance=distance;
}
public void setSwipeMinDistance(int distance)
{
this.swipe_Min_Distance=distance;
}
public int getSwipeMaxDistance()
{
return this.swipe_Max_Distance;
}
public int getSwipeMinDistance()
{
return this.swipe_Min_Distance;
}
public int getSwipeMinVelocity()
{
return this.swipe_Min_Velocity;
}
public boolean onFling(MotionEvent e1,MotionEvent e2,float velocityX,float velocityY)
{
final float xDistance=Math.abs(e1.getX()-e2.getX());
final float yDistance=Math.abs(e1.getY()-e2.getY());
if(xDistance>this.swipe_Max_Distance || yDistance> this.swipe_Max_Distance)
return false;
velocityX = Math.abs(velocityX);
velocityY = Math.abs(velocityY);
boolean result=false;
if(velocityX > this.swipe_Min_Velocity && xDistance > this.swipe_Min_Distance)
{
if(e1.getX() > e2.getX()) // right to left Move
this.listener.onSwipe(SWIPE_LEFT);
else
this.listener.onSwipe(SWIPE_RIGHT);
result=true;
}
else if(velocityY > this.swipe_Min_Velocity && yDistance > this.swipe_Min_Distance)
{
if(e1.getY() > e2.getY()) // bottom to top Move
this.listener.onSwipe(SWIPE_UP);
else
this.listener.onSwipe(SWIPE_DOWN);
result=true;
}
return result;
}
public boolean onSingleTapUp(MotionEvent e)
{
this.tapIndicator=true;
return false;
}
public boolean onDoubleTap(MotionEvent e)
{
this.listener.onDoubleTap();
return false;
}
public boolean onDoubleTapEvent(MotionEvent e)
{
return true;
}
public boolean onSingleTapConfirmed(MotionEvent e)
{
if(this.mode==MODE_DYNAMIC)
{
e.setAction(ACTION_FAKE);
this.context.dispatchTouchEvent(e);
}
return false;
}
static interface SimpleGestureListener
{
void onSwipe(int direction);
void onDoubleTap();
}
}
Ответ 6
У меня была аналогичная проблема, и решения работают до тех пор, пока я не захочу делать другие события касания, такие как swiping и onLongPress. Эти методы никогда не вызывались, поэтому мне пришлось реализовать OnDoubleTapListener. Я сделал следующее:
public class MainActivity extends Activity implements OnDoubleTapListener
Затем просто реализуйте три метода
@Override
public boolean onDoubleTapEvent(MotionEvent e) {
if(e.getAction()==1)
{
Toast.makeText(getApplicationContext(), "DOUBLE TAP",Toast.LENGTH_SHORT).show();
// TODO Auto-generated method stub
// Implement code here!!!
}
return true;
}
@Override
public boolean onSingleTapConfirmed(MotionEvent e) {
return true;
}
@Override
public boolean onDoubleTap(MotionEvent e) {
return true;
}
Просто реализуйте метод onDoubleTapEvent. Я не знаю, когда вызывается два других метода, но это работает для меня.
Ответ 7
X_View.setOnClickListener(new View.OnClickListener() {
@Override
public void onItemClick(View view)
{
long timeNow=Calendar.getInstance().getTimeInMillis();
long timeLastTapped=Long.valueOf(view.getTag().toString()); // Initially set to zero in adapter
final int minDurationBetweenDoubleTap=500;
if(timeLastTapped != 0)
if( timeNow- timeLastTapped < minDurationBetweenDoubleTap)
{
Toast.makeText(getApplicationContext(), "DoubleTapped", 10).show();
}
view.setTag(""+timeNow);
}