Zaimplementowałem niestandardowe okno dialogowe dla mojej aplikacji. Chcę zaimplementować, że gdy użytkownik kliknie poza oknem dialogowym, okno dialogowe zostanie zamknięte. Co mam z tym zrobić?
Zaimplementowałem niestandardowe okno dialogowe dla mojej aplikacji. Chcę zaimplementować, że gdy użytkownik kliknie poza oknem dialogowym, okno dialogowe zostanie zamknięte. Co mam z tym zrobić?
Odpowiedzi:
Możesz użyć polecenia, dialog.setCanceledOnTouchOutside(true);
które zamknie okno dialogowe, jeśli dotkniesz poza nim.
Coś jak,
Dialog dialog = new Dialog(context)
dialog.setCanceledOnTouchOutside(true);
Lub jeśli Twój Dialog jest inny niż model,
1 - Ustaw flagę - FLAG_NOT_TOUCH_MODAL
dla atrybutu okna dialogowego
Window window = this.getWindow();
window.setFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL,
WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);
2 - Dodaj kolejną flagę do właściwości okna, FLAG_WATCH_OUTSIDE_TOUCH
- ta opcja służy do odbierania przez okno dialogowe zdarzenia dotyku poza jego widocznym obszarem.
3 - Zastąp onTouchEvent()
okno dialogowe i sprawdź typ akcji. jeśli typ akcji to „ MotionEvent.ACTION_OUTSIDE
” oznacza, że użytkownik wchodzi w interakcję poza obszarem dialogu. W takim przypadku możesz zamknąć okno dialogowe lub zdecydować, co chcesz wykonać. wyświetlić zwykły druk?
public boolean onTouchEvent(MotionEvent event)
{
if(event.getAction() == MotionEvent.ACTION_OUTSIDE){
System.out.println("TOuch outside the dialog ******************** ");
this.dismiss();
}
return false;
}
Aby uzyskać więcej informacji, zobacz Jak zamknąć niestandardowe okno dialogowe oparte na punktach dotykowych? i jak zamknąć okno dialogowe niemodalne po dotknięciu poza obszarem dialogu
Po prostu użyj
dialog.setCanceledOnTouchOutside(true);
Możesz użyć tej implementacji onTouchEvent. Zapobiega reagowaniu pod działaniem na zdarzenie dotykowe (jak wspomniano w Howettl).
@Override
public boolean onTouchEvent ( MotionEvent event ) {
// I only care if the event is an UP action
if ( event.getAction () == MotionEvent.ACTION_UP ) {
// create a rect for storing the window rect
Rect r = new Rect ( 0, 0, 0, 0 );
// retrieve the windows rect
this.getWindow ().getDecorView ().getHitRect ( r );
// check if the event position is inside the window rect
boolean intersects = r.contains ( (int) event.getX (), (int) event.getY () );
// if the event is not inside then we can close the activity
if ( !intersects ) {
// close the activity
this.finish ();
// notify that we consumed this event
return true;
}
}
// let the system handle the event
return super.onTouchEvent ( event );
}
Źródło: http://blog.twimager.com/2010/08/closing-activity-by-touching-outside.html
Lub, jeśli dostosowujesz okno dialogowe za pomocą motywu zdefiniowanego w twoim stylu xml, umieść ten wiersz w swoim motywie:
<item name="android:windowCloseOnTouchOutside">true</item>
dialog.setCanceledOnTouchOutside(true);
działa cudownie.
Ta metoda powinna całkowicie unikać działań poniżej szarego obszaru pobierających zdarzenia kliknięcia.
Usuń tę linię, jeśli ją masz:
window.setFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL, WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);
Umieść to na utworzonej aktywności
getWindow().setFlags(LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH, LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH);
następnie zastąp zdarzenie dotyku tym
@Override
public boolean onTouchEvent(MotionEvent ev)
{
if(MotionEvent.ACTION_DOWN == ev.getAction())
{
Rect dialogBounds = new Rect();
getWindow().getDecorView().getHitRect(dialogBounds);
if (!dialogBounds.contains((int) ev.getX(), (int) ev.getY())) {
// You have clicked the grey area
displayYourDialog();
return false; // stop activity closing
}
}
// Touch events inside are fine.
return super.onTouchEvent(ev);
}
Ten kod jest używany do kliknięcia okna dialogowego, które powoduje ukrycie wejścia i gdy użytkownik kliknie zewnętrzną stronę okna dialogowego, gdy zarówno programowe wejście, jak i okno dialogowe są zamknięte.
dialog = new Dialog(act) {
@Override
public boolean onTouchEvent(MotionEvent event) {
// Tap anywhere to close dialog.
Rect dialogBounds = new Rect();
getWindow().getDecorView().getHitRect(dialogBounds);
if (!dialogBounds.contains((int) event.getX(),
(int) event.getY())) {
// You have clicked the grey area
InputMethodManager inputMethodManager = (InputMethodManager) act
.getSystemService(act.INPUT_METHOD_SERVICE);
inputMethodManager.hideSoftInputFromWindow(dialog
.getCurrentFocus().getWindowToken(), 0);
dialog.dismiss();
// stop activity closing
} else {
InputMethodManager inputMethodManager = (InputMethodManager) act
.getSystemService(act.INPUT_METHOD_SERVICE);
inputMethodManager.hideSoftInputFromWindow(dialog
.getCurrentFocus().getWindowToken(), 0);
}
return true;
}
};
Inne rozwiązanie, ten kod został wzięty z kodu źródłowego Androida Window
Powinieneś po prostu dodać te dwie metody do kodu źródłowego okna dialogowego.
@Override
public boolean onTouchEvent(MotionEvent event) {
if (isShowing() && (event.getAction() == MotionEvent.ACTION_DOWN
&& isOutOfBounds(getContext(), event) && getWindow().peekDecorView() != null)) {
hide();
}
return false;
}
private boolean isOutOfBounds(Context context, MotionEvent event) {
final int x = (int) event.getX();
final int y = (int) event.getY();
final int slop = ViewConfiguration.get(context).getScaledWindowTouchSlop();
final View decorView = getWindow().getDecorView();
return (x < -slop) || (y < -slop)
|| (x > (decorView.getWidth()+slop))
|| (y > (decorView.getHeight()+slop));
}
To rozwiązanie nie ma tego problemu:
Działa to świetnie, z wyjątkiem tego, że aktywność pod spodem również reaguje na zdarzenie dotykowe. Czy jest jakiś sposób, aby temu zapobiec? - howettl
Pomogło mi:
myDialog.setCanceledOnTouchOutside(true);
Oto kod
dialog.getWindow().getDecorView().setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent ev) {
if(MotionEvent.ACTION_DOWN == ev.getAction())
{
Rect dialogBounds = new Rect();
dialog. getWindow().getDecorView().getHitRect(dialogBounds);
if (!dialogBounds.contains((int) ev.getX(), (int) ev.getY())) {
// You have clicked the grey area
UiUtils.hideKeyboard2(getActivity());
return false; // stop activity closing
}
}
getActivity().dispatchTouchEvent(ev);
return false;
}
});
Spróbuj tego . możesz ukryć klawiaturę, gdy dotkniesz na zewnątrz