アプリケーションにカスタムダイアログを実装しました。ユーザーがダイアログの外側をクリックすると、ダイアログが閉じられるように実装します。これのために何をしなければなりませんか?
アプリケーションにカスタムダイアログを実装しました。ユーザーがダイアログの外側をクリックすると、ダイアログが閉じられるように実装します。これのために何をしなければなりませんか?
回答:
dialog.setCanceledOnTouchOutside(true);
ダイアログの外側をタッチすると、ダイアログを閉じるを使用できます。
何かのようなもの、
Dialog dialog = new Dialog(context)
dialog.setCanceledOnTouchOutside(true);
または、ダイアログが非モデルの場合、
1- FLAG_NOT_TOUCH_MODAL
ダイアログのウィンドウ属性にフラグを設定します
Window window = this.getWindow();
window.setFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL,
WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);
2-Windowsプロパティに別のフラグを追加します。FLAG_WATCH_OUTSIDE_TOUCH
これは、ダイアログが表示領域外でタッチイベントを受け取るためのものです。
3- onTouchEvent()
ダイアログをオーバーライドし、アクションタイプを確認します。アクションタイプが ' MotionEvent.ACTION_OUTSIDE
'の場合、ユーザーはダイアログ領域の外で対話しています。したがって、この場合、ダイアログを無視するか、何を実行するかを決定できます。プレーンプリントを表示しますか?
public boolean onTouchEvent(MotionEvent event)
{
if(event.getAction() == MotionEvent.ACTION_OUTSIDE){
System.out.println("TOuch outside the dialog ******************** ");
this.dismiss();
}
return false;
}
詳細については、タッチポイントに基づいてカスタムダイアログを閉じる方法をご覧ください。および ダイアログ領域の外側に触れたときに非モーダルダイアログを閉じる方法
単に使う
dialog.setCanceledOnTouchOutside(true);
このonTouchEventの実装を使用できます。これは、(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 );
}
出典:http : //blog.twimager.com/2010/08/closing-activity-by-touching-outside.html
または、スタイルxmlで定義されたテーマを使用してダイアログをカスタマイズする場合は、次の行をテーマに追加します。
<item name="android:windowCloseOnTouchOutside">true</item>
dialog.setCanceledOnTouchOutside(true);
素晴らしく機能します。
このメソッドは、クリックイベントを取得する灰色の領域の下のアクティビティを完全に回避する必要があります。
この行がある場合は削除します。
window.setFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL, WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);
作成したあなたの活動にこれを置きます
getWindow().setFlags(LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH, LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH);
次にこれでタッチイベントをオーバーライドします
@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);
}
このコードは、ダイアログボックスでhidesoftinputをクリックするとき、およびユーザーがダイアログボックスの外側をクリックして、softinputとダイアログボックスの両方が閉じるときに使用します。
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;
}
};
別の解決策、このコードはAndroidのソースコードから取得したWindow
ものです。これらの2つのメソッドをダイアログのソースコードに追加するだけです。
@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));
}
このソリューションにはこの問題はありません。
下のアクティビティがタッチイベントにも反応することを除いて、これはうまく機能します。これを防ぐ方法はありますか?– howettl
以下は私のために働いています:
myDialog.setCanceledOnTouchOutside(true);
これがコードです
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;
}
});
これを試してください。外をタッチするとキーボードを隠すことができます