如何在对话框(Dialog)和活动(Activity)之间传递数值?

32

我正在通过对话框向用户请求输入:

package com.android.cancertrials;

import android.app.Dialog;
import android.content.Context;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

public class CustomDialog extends Dialog  {


    private String name;
//    private ReadyListener readyListener;
     public static EditText etName;
     public String zip;

    public CustomDialog(Context context, String name) {
        super(context);
        this.name = name;
//        this.readyListener = readyListener;
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.mycustomdialog);
        setTitle("Enter the Zip Code ");
        Button buttonOK = (Button) findViewById(R.id.ok);
        buttonOK.setOnClickListener(new OKListener());
        etName = (EditText) findViewById(R.id.EditZip);
    }

    private class OKListener implements android.view.View.OnClickListener {
        @Override
        public void onClick(View v) {
//            readyListener.ready(String.valueOf(etName.getText()));
            CustomDialog.this.dismiss();
        }
    }


}
当用户点击“确定”时,我如何将在文本框中输入的值传回启动它的 Activity 中的成员变量?

3
你需要创建一个扩展对话框的类吗?看起来你并没有做任何需要子类的特殊操作。 - Falmarri
我猜不是吧?你能给我一个使用标准AlertDialog的带有EditText的示例吗?我该如何获取这些信息并传递给调用活动? - Sheehan Alam
3个回答

85
你可以用不同的方法来实现这个... 实际上,如果你的对话框只有一个“确定”按钮以关闭它,为什么不使用 AlertDialog.Builder 类创建自定义对话框,而不是子类化 Dialog 呢?
无论如何...假设你有充分的理由按照你的方式来做。在这种情况下,我会使用观察者模式。像这样:
public class CustomDialog extends Dialog  {


    private String name;
     public static EditText etName;
     public String zip;
    OnMyDialogResult mDialogResult; // the callback

    public CustomDialog(Context context, String name) {
        super(context);
        this.name = name;
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        // same you have
    }

    private class OKListener implements android.view.View.OnClickListener {
        @Override
        public void onClick(View v) {
            if( mDialogResult != null ){
                mDialogResult.finish(String.valueOf(etName.getText()));
            }
            CustomDialog.this.dismiss();
        }
    }

    public void setDialogResult(OnMyDialogResult dialogResult){
        mDialogResult = dialogResult;
    }

    public interface OnMyDialogResult{
       void finish(String result);
    }
}

在您的活动中:

CustomDialog dialog;
// initialization stuff, blah blah
dialog.setDialogResult(new OnMyDialogResult(){
    public void finish(String result){
        // now you can use the 'result' on your activity
    }
});

从你的代码来看,似乎你已经尝试过类似的东西。

编辑:采用简单方法

你仍然可以使用你的mycustomdialog布局。以下是如何使用AlertDialog.Builder

LayoutInflater inflater = LayoutInflater.from(YourActivity.this);
final View yourCustomView = inflater.inflate(R.layout.mycustomdialog, null);

final TextView etName = (EditText) yourCustomView.findViewById(R.id.EditZip);
AlertDialog dialog = new AlertDialog.Builder(YourActivity.this)
    .setTitle("Enter the Zip Code")
    .setView(yourCustomView)
    .setPositiveButton("OK", new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialog, int whichButton) {
            mSomeVariableYouHaveOnYourActivity = etName.getText().toString();
        }
    })
    .setNegativeButton("Cancel", null).create();
dialog.show();

3
感谢您提供详细的帖子。为了学习和简化起见,我该如何使用AlertDialog.Builder执行同样的操作呢? - Sheehan Alam
允许10分钟...我会这样写。 - Cristian
另外,在你的第一个例子中,我看不到我可以提取etName.getText()的地方?那本质上就是我的“结果”。 - Sheehan Alam
仔细看...搜索这一行:mDialogResult.finish(String.valueOf(etName.getText()));。观察者模式不是很棒吗?我喜欢它! - Cristian
非常棒。谢谢Cristian,我今天早上学到了很多东西!两种方法都很好,为了简单和更少的代码行数,我会实现Builder。 - Sheehan Alam
显示剩余3条评论

1
我通过向[activity]广播意图(intent)来实现这一点。
首先将活动(activity)传递到函数中:
public class DialogFactory {

    public static AlertDialog addSomeDialog(Activity activity) {
        builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
                  if (SOMETHING_IS_TRUE) {

                      // prepare your parameters that need to be sent back to activity
                      Intent intent = new Intent(IntentAction.INTENT_ADD_TASK);
                      intent.putExtra(IntentConst.PARAM_A, aInput);
                      intent.putExtra(IntentConst.PARAM_B, bInput);
                      activity.sendBroadcast(intent);

                      Toast.makeText(activity, "Something is TRUE!", Toast.LENGTH_SHORT).show();
                  } else {
                      Toast.makeText(activity, "Something NOT TRUE!", Toast.LENGTH_SHORT).show();
                  }
            }
        });
    }
}

当您的活动中某个选项菜单或按钮被点击时,请调用上述函数。
接下来,在活动中使用BroadcastReceiver准备接收意图:
private BroadcastReceiver mReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        if (intent.getAction() == IntentAction.INTENT_ADD_TASK) {
             // Do whatever you want to refresh the layout or anything in the activity
             // or even ask fragments inside to act upon it.
             .....
        }
    }
};

不要忘记注册和注销接收器:
@Override
protected void onPause() {
    unregisterReceiver(mReceiver);
    super.onPause();
}

@Override
protected void onResume() {
    super.onResume();
    registerReceiver(mReceiver, new IntentFilter(IntentAction.INTENT_ADD_TASK));
}

0
您可以在onDismiss监听器中像这样获取您对话框类的任何公共变量的值。
SaveRecordedFileDialogClass cdd = new SaveRecordedFileDialogClass(this,extraVariable);
cdd.setOnDismissListener(new 
DialogInterface.OnDismissListener() {
        @Override
        public void onDismiss(DialogInterface dialog) {
            myPublicActivityVar=cdd.anyTypevariable;
        }
});
cdd.show();

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接