2011-06-20 10 views
43

एंड्रॉइड में, हम कैसे पता लगा सकते हैं कि कोई उपयोगकर्ता बटन पर छूता है और इस बटन के क्षेत्र से बाहर निकलता है?एंड्रॉइड: पता लगाएँ कि क्या उपयोगकर्ता बटन क्षेत्र से छूता है और ड्रैग करता है?

उत्तर

79

चेक MotionEvent.MOVE_OUTSIDE: चेक MotionEvent.MOVE:

private Rect rect; // Variable rect to hold the bounds of the view 

public boolean onTouch(View v, MotionEvent event) { 
    if(event.getAction() == MotionEvent.ACTION_DOWN){ 
     // Construct a rect of the view's bounds 
     rect = new Rect(v.getLeft(), v.getTop(), v.getRight(), v.getBottom()); 

    } 
    if(event.getAction() == MotionEvent.ACTION_MOVE){ 
     if(!rect.contains(v.getLeft() + (int) event.getX(), v.getTop() + (int) event.getY())){ 
      // User moved outside bounds 
     } 
    } 
    return false; 
} 

नोट: यदि आप Android 4.0 को लक्षित करना चाहते हैं, तो नई संभावनाओं की एक पूरी दुनिया को खोलता है: http://developer.android.com/reference/android/view/MotionEvent.html#ACTION_HOVER_ENTER

+1

:

myView.setOnTouchListener(new SimpleTouchListener() { @Override public void onDownTouchAction() { // do something when the View is touched down } @Override public void onUpTouchAction() { // do something when the down touch is released on the View } @Override public void onCancelTouchAction() { // do something when the down touch is canceled // (e.g. because the down touch moved outside the bounds of the View } }); 

यहाँ वर्ग के स्रोत आप अपने प्रोजेक्ट को जोड़ सकते हैं जो है क्षमा करें लेकिन यह काम नहीं करता है। मैं केवल ACTION_UP, ACTION_DOWN और ACTION_MOVE को पकड़ सकता हूं लेकिन ACTION_OUTSIDE नहीं। – anticafe

+0

यह एपीआई स्तर 3 से आगे उपलब्ध है: http://developer.android.com/reference/android/view/MotionEvent.html#ACTION_OUTSIDE – Entreco

+4

इस उत्तर के अनुसार (http://stackoverflow.com/questions/6162497/when- क्या-एक्शन-आउट-आउट-ट्रिगर/8059266 # 8059266) मेरा मानना ​​है कि हम केवल ACTION_OUTSIDE का उपयोग यह पता लगाने के लिए कर सकते हैं कि स्पर्श गतिविधि के बाहर है या नहीं। – anticafe

21

एंट्रेको द्वारा पोस्ट किए गए उत्तर को मेरे मामले में कुछ मामूली tweaking की जरूरत है।

if(!rect.contains((int)event.getX(), (int)event.getY())) 

के लिए

if(!rect.contains(v.getLeft() + (int) event.getX(), v.getTop() + (int) event.getY())) 

क्योंकि event.getX() और event.getY() केवल imageView ही है, पूरे स्क्रीन नहीं करने के लिए लागू: मैं स्थानापन्न करना पड़ा।

+0

+1 धन्यवाद, यह कस्टम दृश्य कार्यान्वयन में अधिक सार्वभौमिक दृष्टिकोण है और काम करता है – Knickedi

1

जबकि @FrostRocket से जवाब सही आप अनुवाद के लिए खाते view.getX() और वाई का उपयोग करना चाहिए है और साथ ही बदल देता है:

view.getHitRect(viewRect); 
if(viewRect.contains(
     Math.round(view.getX() + event.getX()), 
     Math.round(view.getY() + event.getY()))) { 
    // inside 
} else { 
    // outside 
} 
0

यहाँ एक View.OnTouchListener है कि आप देखने के लिए अगर MotionEvent.ACTION_UP था का उपयोग कर सकते है भेजा जबकि उपयोगकर्ता को देखने के बाहर उसका/उसकी उंगली था:

private OnTouchListener mOnTouchListener = new View.OnTouchListener() { 

    private Rect rect; 

    @Override 
    public boolean onTouch(View v, MotionEvent event) { 
     if (v == null) return true; 
     switch (event.getAction()) { 
     case MotionEvent.ACTION_DOWN: 
      rect = new Rect(v.getLeft(), v.getTop(), v.getRight(), v.getBottom()); 
      return true; 
     case MotionEvent.ACTION_UP: 
      if (rect != null 
        && !rect.contains(v.getLeft() + (int) event.getX(), 
         v.getTop() + (int) event.getY())) { 
       // The motion event was outside of the view, handle this as a non-click event 

       return true; 
      } 
      // The view was clicked. 
      // TODO: do stuff 
      return true; 
     default: 
      return true; 
     } 
    } 
}; 
2

शीर्ष 2 जवाब जब दृश्य एक scrollview के अंदर है, सिवाय इसके ठीक हैं: जब स्क्रॉल जगह लेता है क्योंकि आप अपनी उंगली ले जाते हैं, यह अभी भी पंजीकृत किया गया है एक स्पर्श घटना के रूप में लेकिन एक मोशनवेन्ट.एक्शन_MOVE घटना के रूप में नहीं। तो जवाब में सुधार करने के (जो केवल जरूरत है, तो आपके विचार एक स्क्रॉल तत्व के अंदर है):

private Rect rect; // Variable rect to hold the bounds of the view 

public boolean onTouch(View v, MotionEvent event) { 
    if(event.getAction() == MotionEvent.ACTION_DOWN){ 
     // Construct a rect of the view's bounds 
     rect = new Rect(v.getLeft(), v.getTop(), v.getRight(), v.getBottom()); 

    } else if(rect != null && !rect.contains(v.getLeft() + (int) event.getX(), v.getTop() + (int) event.getY())){ 
     // User moved outside bounds 
    } 
    return false; 
} 

मैं Android 4.3 और Android 4.4

पर इस परीक्षण किया मैं मोरित्ज़ के जवाब के बीच कोई अंतर नहीं दिखाई दी और शीर्ष पर 2 लेकिन यह भी अपने जवाब के लिए लागू होता है:

private Rect rect; // Variable rect to hold the bounds of the view 

public boolean onTouch(View v, MotionEvent event) { 
    if(event.getAction() == MotionEvent.ACTION_DOWN){ 
     // Construct a rect of the view's bounds 
     rect = new Rect(v.getLeft(), v.getTop(), v.getRight(), v.getBottom()); 

    } else if (rect != null){ 
     v.getHitRect(rect); 
     if(rect.contains(
       Math.round(v.getX() + event.getX()), 
       Math.round(v.getY() + event.getY()))) { 
      // inside 
     } else { 
      // outside 
     } 
    } 
    return false; 
} 
+0

मैंने अभी इस कोड को एक बटन पर लागू किया है जो एक RecyclerView के किसी आइटम में निहित है और पूरी तरह से काम करता है। मैं सिर्फ यह कहना चाहता था कि अगर आप getHitRect से पहले इसे जोड़ते हैं तो आप ** ACTION_DOWN ** पर नए रेक्ट से छुटकारा पा सकते हैं। ** Rect rect = new Rect(); getHitRect (rect); ** –

5

मैं अपने OnTouch में कुछ लॉगिंग जोड़ा गया है और पता चला कि MotionEvent.ACTION_CANCEL मारा जा रहा था। वह मेरे लिए काफी अच्छा है ...

+1

आपको 'मोशनइवेंट.एक्शन_CANCEL' कॉलबैक मिलेगा यदि आपका' व्यू' 'स्क्रॉल व्यू' जैसे माता-पिता 'व्यू ग्रुप' के भीतर निहित है, जो कि अपने बच्चों से चाल छूने में रूचि रखता है लेकिन अन्यथा कोई गारंटी नहीं है कि आपको अपने 'व्यू' में 'मोशनइवेंट.एक्शन_CANCEL' कॉलबैक मिलेगा। –

0
view.setClickable(true); 
view.setOnTouchListener(new OnTouchListener() { 
    @Override 
    public boolean onTouch(View v, MotionEvent event) { 
     if (!v.isPressed()) { 
      Log.e("onTouch", "Moved outside view!"); 
     } 
     return false; 
    } 
}); 

view.isPressedview.pointInView का उपयोग करता है और कुछ स्पर्श ढिलाई भी शामिल है। यदि आप ढलान नहीं चाहते हैं, तो बस तर्क को आंतरिक view.pointInView से कॉपी करें (जो सार्वजनिक है, लेकिन छुपा हुआ है, इसलिए यह आधिकारिक एपीआई का हिस्सा नहीं है और किसी भी समय गायब हो सकता है)।

view.setClickable(true); 
view.setOnTouchListener(new OnTouchListener() { 
    @Override 
    public boolean onTouch(View v, MotionEvent event) { 
     if (event.getAction() == MotionEvent.ACTION_DOWN) { 
      v.setTag(true); 
     } else { 
      boolean pointInView = event.getX() >= 0 && event.getY() >= 0 
        && event.getX() < (getRight() - getLeft()) 
        && event.getY() < (getBottom() - getTop()); 
      boolean eventInView = ((boolean) v.getTag()) && pointInView; 
      Log.e("onTouch", String.format("Dragging currently in view? %b", pointInView)); 
      Log.e("onTouch", String.format("Dragging always in view? %b", eventInView)); 
      v.setTag(eventInView); 
     } 
     return false; 
    } 
}); 
1

मैं ओ पी के रूप में यह एक ही समस्या थी जिससे मुझे पता है कि जब (1) एक विशेष View नीचे छुआ था और साथ ही चाहता था या तो (2) जब नीचे स्पर्श View या को जारी किया गया (3) जब डाउन टच View की सीमाओं के बाहर चले गए। मैंने View.OnTouchListener (SimpleTouchListener नामित) का एक सरल विस्तार बनाने के लिए इस धागे में विभिन्न उत्तरों को एक साथ लाया ताकि अन्य MotionEvent ऑब्जेक्ट के साथ घूमने की आवश्यकता न हो। कक्षा के लिए स्रोत here या इस उत्तर के नीचे पाया जा सकता है।

इस वर्ग का उपयोग करने के लिए बस View.setOnTouchListener(View.OnTouchListener) विधि के एकल पैरामीटर के रूप में सेट इस प्रकार है:

public abstract class SimpleTouchListener implements View.OnTouchListener { 

    /** 
    * Flag determining whether the down touch has stayed with the bounds of the view. 
    */ 
    private boolean touchStayedWithinViewBounds; 

    @Override 
    public boolean onTouch(View view, MotionEvent event) { 
     switch (event.getAction()) { 
      case MotionEvent.ACTION_DOWN: 
       touchStayedWithinViewBounds = true; 
       onDownTouchAction(); 
       return true; 

      case MotionEvent.ACTION_UP: 
       if (touchStayedWithinViewBounds) { 
        onUpTouchAction(); 
       } 
       return true; 

      case MotionEvent.ACTION_MOVE: 
       if (touchStayedWithinViewBounds 
         && !isMotionEventInsideView(view, event)) { 
        onCancelTouchAction(); 
        touchStayedWithinViewBounds = false; 
       } 
       return true; 

      case MotionEvent.ACTION_CANCEL: 
       onCancelTouchAction(); 
       return true; 

      default: 
       return false; 
     } 
    } 

    /** 
    * Method which is called when the {@link View} is touched down. 
    */ 
    public abstract void onDownTouchAction(); 

    /** 
    * Method which is called when the down touch is released on the {@link View}. 
    */ 
    public abstract void onUpTouchAction(); 

    /** 
    * Method which is called when the down touch is canceled, 
    * e.g. because the down touch moved outside the bounds of the {@link View}. 
    */ 
    public abstract void onCancelTouchAction(); 

    /** 
    * Determines whether the provided {@link MotionEvent} represents a touch event 
    * that occurred within the bounds of the provided {@link View}. 
    * 
    * @param view the {@link View} to which the {@link MotionEvent} has been dispatched. 
    * @param event the {@link MotionEvent} of interest. 
    * @return true iff the provided {@link MotionEvent} represents a touch event 
    * that occurred within the bounds of the provided {@link View}. 
    */ 
    private boolean isMotionEventInsideView(View view, MotionEvent event) { 
     Rect viewRect = new Rect(
       view.getLeft(), 
       view.getTop(), 
       view.getRight(), 
       view.getBottom() 
     ); 

     return viewRect.contains(
       view.getLeft() + (int) event.getX(), 
       view.getTop() + (int) event.getY() 
     ); 
    } 
} 
+1

यह एक अच्छा जवाब है, खासकर इस मुद्दे के साथ काम करने वाले लोगों के लिए जब ACTION_CANCEL इसे ट्रिगर नहीं कर रहा है। –

संबंधित मुद्दे