web-dev-qa-db-ja.com

Android確認ダイアログがtrueまたはfalseを返す

アラートダイアログで単純な値を返す簡単な方法はないようです。
このコード機能しませんanswer変数はリスナー内から設定できません。実際にはコンパイルすらしません)

public static boolean Confirm(Context context) {
    boolean answer;
    AlertDialog dialog = new AlertDialog.Builder(context).create();
    dialog.setTitle("Confirmation");
    dialog.setMessage("Choose Yes or No");
    dialog.setCancelable(false);
    dialog.setButton(DialogInterface.BUTTON_POSITIVE, "Yes", new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialog, int buttonId) {
            answer = true;
        }
    });
    dialog.setButton(DialogInterface.BUTTON_NEGATIVE, "No", new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialog, int buttonId) {
            answer = false;
        }
    });
    dialog.setIcon(Android.R.drawable.ic_dialog_alert);
    dialog.show();
    return answer;
}

注:メソッドが自己完結型であることが重要です。つまり、メソッドの外部の変数や構成に依存しないことが重要です。ただそれを呼んで、あなたの答えを得てください、真か偽か。

じゃあ何をすればいいの? trueまたはfalseを返すというこの単純な願いは、それに値するよりもはるかに複雑なようです。

また、setButtonメソッドの形式は次のとおりです。

dialog.setButton(int buttonId, String buttonText, Message msg)

しかし、それをどのように使用するか、メッセージはどこに送信され、誰に、どのハンドラーが使用されるかは明確ではありませんか?

12
ilomambo

さて、私は簡単な答えをすべて自分で見つけたので、私は自分自身に非常に満足していると言うつもりでした!
しかし、真実は、値を返す方法を見つけたとしても(以下に示します)です。 役に立たない

本当の問題は、同期ダイアログ、つまりdialog.show()の後にコードを再開する前にユーザーが応答するのを待つダイアログが必要だったことです。
Androidにはそのような獣はありません。すべてのダイアログは非同期であるため、dialog.show()はダイアログをあるキューに投稿するだけで(私は思う)、続行します。したがって、あなたは時間内にあなたの答えを得ることができません。

以下のすべての価値(何もない)については、ダイアログを構築するメソッド内で値を設定する方法を説明します。ダイアログのライフサイクルとは関係なく、この手法には他の用途があるかもしれません。




いくつかの関連情報を与えるために、私はあなたが置き換えるならばそれを言うでしょう

boolean answer;

final boolean answer;

リスナー内から変数にアクセスすることは可能ですが、finalとして宣言されているため、新しい値を割り当てることはできません。

ここにトリックがあります。
変数を次のように定義します。

final boolean[] answer = new boolean[1];

なぜこれが機能するのか、すでにご存知の方もいらっしゃいます。ここでの最後の変数はブール配列の単一要素ではなく、配列自体です。
これで、必要に応じて配列要素[0]を割り当てることができます。

dialog.setButton(DialogInterface.BUTTON_POSITIVE, "Yes", new DialogInterface.OnClickListener() {
    public void onClick(DialogInterface dialog, int buttonId) {
        answer[0] = true;
    }
});
. . .
return answer[0];

そして最後に、メソッドからそれを返すことができます。

10
ilomambo

私はこのフォーラムで同様の問題を投稿していますが、最終的に私は私の答えを得ます。その投稿での私の問題は、他のクラスまたはアクティビティでアクセスできる個別の確認ダイアログクラスを作成する方法です。そのため、その確認ダイアログクラスを使用すると、長いコーディングを記述する必要がありません。

これが私の答えです。

まず、DialogHandler.Javaを作成する必要があります

import Android.app.Activity;
import Android.app.AlertDialog;
import Android.content.Context;
import Android.content.DialogInterface;
import src.app.R;

public class DialogHandler {
    public Runnable ans_true = null;
    public Runnable ans_false = null;

    // Dialog. --------------------------------------------------------------

    public boolean Confirm(Activity act, String Title, String ConfirmText,
            String CancelBtn, String OkBtn, Runnable aProcedure, Runnable bProcedure) {
        ans_true = aProcedure;
        ans_false= bProcedure;
        AlertDialog dialog = new AlertDialog.Builder(act).create();
        dialog.setTitle(Title);
        dialog.setMessage(ConfirmText);
        dialog.setCancelable(false);
        dialog.setButton(DialogInterface.BUTTON_POSITIVE, OkBtn,
                new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int buttonId) {
                         ans_true.run();
                    }
                });
        dialog.setButton(DialogInterface.BUTTON_NEGATIVE, CancelBtn,
                new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int buttonId) {
                        ans_false.run();
                    }
                });
        dialog.setIcon(Android.R.drawable.ic_dialog_alert);
        dialog.show();
        return true;
    }
}

そしてこれは別のクラスでそれを呼び出す例です

public class YourActivity extends Activity {
    /** Called when the activity is first created. */

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        findViewById(R.id.button1).setOnClickListener(myclick);
    }

    public final Button.OnClickListener myclick = new Button.OnClickListener() {
        @Override
        public void onClick(View v) {
            doclick();
        }
    };

    public void doclick() {
        DialogHandler appdialog = new DialogHandler();
        appdialog.Confirm(this, "Message title", "Message content",
                "Cancel", "OK", aproc(), bproc());
    }

    public Runnable aproc(){
        return new Runnable() {
            public void run() {
                Log.d("Test", "This from A proc");
            }
          };
    }

    public Runnable bproc(){
        return new Runnable() {
            public void run() {
                Log.d("Test", "This from B proc");
            }
          };
    }


}
17
AsepRoro

できることは、インターフェイスを使用してAlertDialogsアクションをリッスンするアラートダイアログのリスナーを作成することです。

インターフェイスを作成します。

public class MyInterface {

    DialogReturn dialogReturn;

    public interface DialogReturn {

        void onDialogCompleted(boolean answer);
    }

    public void setListener(DialogReturn dialogReturn) {
        this.dialogReturn = dialogReturn;
    }

    public DialogReturn getListener() {
        return dialogReturn;

    }
}

これで、クラスにimplements MyInterface.DialogReturnを使用して作成したインターフェイスを実装するだけです。

次に、リスナーを設定して、以下に示すように機能させることができます。

public class Main extends Activity implements MyInterface.DialogReturn{

    MyInterface myInterface;
    MyInterface.DialogReturn dialogReturn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
                ....
        myInterface = new MyInterface();
        myInterface.setListener(this);
    }


   public void Confirm(Context context) {
        AlertDialog dialog = new AlertDialog.Builder(context).create();
        dialog.setTitle("Confirmation");
        dialog.setMessage("Choose Yes or No");
        dialog.setCancelable(false);
        dialog.setButton(DialogInterface.BUTTON_POSITIVE, "Yes", new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int buttonId) {
                myInterface.getListener().onDialogCompleted(true);
            }
        });
        dialog.setButton(DialogInterface.BUTTON_NEGATIVE, "No", new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int buttonId) {
                myInterface.getListener().onDialogCompleted(false);
            }
        });
        dialog.setIcon(Android.R.drawable.ic_dialog_alert);
        dialog.show();
         }


@Override
    public void onDialogCompleted(boolean answer) {
        Toast.makeText(Main.this, answer+"", Toast.LENGTH_LONG).show();
            if(answer)
            // do something
            else
            // do something
    }
}
3
Lalit Poptani

Andriodでは、ユーザーが「はい」または「いいえ」と言うのを待って、実行中のスレッドをブロックすることはお勧めできません。

確認を求めるために、AsynTaskを受け取るメソッドを定義できます。ユーザーが確認ボタンを押すと、メソッドはそのタスクを実行します。

例えば:

    //this method displays a confirm dialog. If 'yes' answer, runs 'yesTask', 
    //if 'no' answer, runs 'noTask'
    //notice than 'yesTask' and 'noTask' are AysncTask
    //'noTask' can be null, example: if you want to cancel when 'no answer'

    public static void confirm(Activity act, String title, String confirmText,
                       String noButtonText, String yesButtonText,
                       final AsyncTask<String, Void, Boolean> yesTask,
                       final AsyncTask<String, Void, Boolean> noTask) {

    AlertDialog dialog = new AlertDialog.Builder(act).create();
    dialog.setTitle(title);
    dialog.setMessage(confirmText);
    dialog.setCancelable(false);
    dialog.setButton(DialogInterface.BUTTON_POSITIVE, yesButtonText,
        new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int buttonId) {
                yesTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
            }
        });
    dialog.setButton(DialogInterface.BUTTON_NEGATIVE, noButtonText,
        new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int buttonId) {
                if(noTask!=null) {
                    noTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
                }

            }
        });
    dialog.setIcon(Android.R.drawable.ic_dialog_alert);
    dialog.show();
}

あなたはあなたの活動からそれを呼び出すことができます:

 YourTask yourTask =  new YourTask( ... );
 confirm( YourActivity.this, 
         "Confirm", 
         "Are you sure?", 
         "Cancel", 
         "Continue", 
         yourTask,
         null);

YourTaskクラスはAsyncTaskを拡張する必要があります

1
robcalvo

jDeferred を使用すると、入力を待ちたい場合に役立つことがわかりました。

これは基本的にインターフェイスを使用するのと同じですが、代わりに完了ハンドラーと失敗ハンドラーを作成します。考慮すべき単なる代替案:

new ConfirmationDialog(mContext)
        .showConfirmation("Are you sure?", "Yes", "No")
        .done(new DoneCallback<Void>() {
            @Override
            public void onDone(Void aVoid) {
                ....
            }
        })
        .fail(new FailCallback<Void>() {

            @Override
            public void onFail(Void aVoid) {
                ...
            }
        });

実装:

public class ConfirmationDialog {


    private final Context mContext;
    private final DeferredObject<Void, Void, Void> mDeferred = new DeferredObject<Void, Void, Void>();

    public ConfirmationDialog(Context context) {
        mContext = context;
    }

    public Promise<Void, Void, Void> showConfirmation(String message, String positiveButton, String negativeButton) {
        AlertDialog dialog = new AlertDialog.Builder(mContext).create();
        dialog.setTitle("Alert");
        dialog.setMessage(message);
        dialog.setCancelable(false);
        dialog.setButton(DialogInterface.BUTTON_POSITIVE, positiveButton, new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int buttonId) {
                mDeferred.resolve(null);
            }
        });
        dialog.setButton(DialogInterface.BUTTON_NEGATIVE, negativeButton, new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int buttonId) {
                mDeferred.reject(null);
            }
        });
        dialog.setIcon(Android.R.drawable.ic_dialog_alert);
        dialog.show();
        return mDeferred.promise();
    }

}
1
Peter Ajtai

また、ブロッキング確認ダイアログを使用するのに苦労していましたが、ついにBlockingQueueを使用してそれを行いました。

public static class BlockingConfirmDialog{

    private Activity context;

    BlockingQueue<Boolean> blockingQueue;

    public BlockingConfirmDialog(Activity activity) {
        super();
        this.context = activity;
        blockingQueue = new ArrayBlockingQueue<Boolean>(1);
    }

    public boolean confirm(final String title, final String message){

        context.runOnUiThread(new Runnable() {
            @Override
            public void run() {
                new AlertDialog.Builder(context)
                .setTitle(title)
                .setMessage(message)
                .setPositiveButton("Ok", new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int which) { 
                        blockingQueue.add(true);
                    }
                 })
                 .setNegativeButton("Cancel", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        blockingQueue.add(false);
                    }
                })
                 .show();
            }
        });

        try {
            return blockingQueue.take();
        } catch (InterruptedException e) {
            e.printStackTrace();
            return false;
        }

    }
}
0
Guillaume

私はすべての解決策を試しましたが、私の意見では、Runnableを使用した最初の解決策ははるかに簡単でクリーンです。 [キャンセル]ボタンリスナー、OnBAckPressed()およびonOptionsItemSelected()のダイアログをサポートします。

説明されているコードはans_false.run();を呼び出します。 BUTTON_POSITIVEおよびans_true.run()をクリックしたとき。 BUTTON_NEGATIVEをクリックしたとき。

この問題を修正するために使用したコードは次のとおりです。

public class MyDialogs {

// private constructor
public Runnable answerTrue = null;
public Runnable answerFalse = null;

// Dialog. --------------------------------------------------------------

public boolean confirm(Activity act, String Title, String ConfirmText,
                       String noBtn, String yesBtn, Runnable yesProc, Runnable noProc) {
    answerTrue = yesProc;
    answerFalse= noProc;
    AlertDialog.Builder alert = new AlertDialog.Builder(act);
    alert.setTitle(Title);
    alert.setMessage(ConfirmText);
    alert.setCancelable(false);
    alert.setPositiveButton(R.string.button_positive, new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialog, int id) {
            answerTrue.run();
        }
    });
    alert.setNegativeButton(R.string.button_negative, new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialog, int id) {
            answerFalse.run();
        }
    });
    alert.show().getButton(DialogInterface.BUTTON_NEGATIVE).requestFocus();
    return true;
}

}

0
mmarin1m

アクティビティでフィールド「answer」を宣言し、それに値を設定します。クラスのフィールドは内部クラスに表示されるので、それを行うことができます。

0
Vladimir Ivanov