Dialog确认按钮不dismiss

本文介绍在使用AlertDialog时如何根据业务需求自定义对话框关闭逻辑,提供了两种方法:一是使用自定义按钮并添加点击事件;二是利用反射修改对话框的状态。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在使用AlertDialog的时候,经常会遇到这样的需求:点击【确认】之后,如果dialog内的输入内容有误,则dialog不dismiss。但实际情况是,无论点击【确认】还是【取消】,dialog都会执行dismiss操作。

查看源码如下:

View.OnClickListener mButtonHandler = new View.OnClickListener() {  
        public void onClick(View v) {  
            Message m = null;  
            if (v == mButtonPositive && mButtonPositiveMessage != null){  
                m = Message.obtain(mButtonPositiveMessage);  
            } else if (v == mButtonNegative && mButtonNegativeMessage != null) {  
                m = Message.obtain(mButtonNegativeMessage);  
            } else if (v == mButtonNeutral && mButtonNeutralMessage != null) {  
                m = Message.obtain(mButtonNeutralMessage);  
            }  
            if (m != null) {  
                m.sendToTarget();  
            }  

            // Post a message so we dismiss after the above handlers are executed  
            mHandler.obtainMessage(ButtonHandler.MSG_DISMISS_DIALOG, mDialogInterface).sendToTarget();  
        }  
    };  

上面的源码为AlertController中自定义的button点击监听事件,从代码中可以看到,点击按钮一共执行了两次handler的send操作,一次是按钮的点击send:m.sendToTarget(),执行按钮的点击事件,紧接着又执行了一条handler的send操作:mHandler.obtainMessage(ButtonHandler.MSG_DISMISS_DIALOG, mDialogInterface).sendToTarget();操作。因此我们得到,不论你点击dialog的哪个按钮,最后都会执行dialog的dismiss操作。

因此,基于上面的需求,我们需要修改点击事件的执行过程。通过查找资料,一共找到两种方式解决这一问题:

方法一:使用自定义的按钮,不使用dialog的setNegativeButton和setPositiveButton,并为自定义的按钮添加点击事件,在自定义的按钮点击监听事件中,对dialog的dismiss进行管理;

方法二:更多时候,我们的dialog实现一个很简单的功能,并不用大费周章的自定义一个button。所以我们可以使用反射的方式来实现上面的需求,具体实现代码如下:

 final EditText et = new EditText(mContext);
        et.setInputType(InputType.TYPE_CLASS_NUMBER);
        et.setText(PreferenceUtil.getInstance().getDefaultInt(ConstValue.TEST_DATA_LENGTH, 2000) + "");
        et.setTextColor(mContext.getResources().getColor(R.color.color_black));

        TextView tv = new TextView(mContext);
        tv.setText("Title测试");
        tv.setTextColor(mContext.getResources().getColor(R.color.red_normal));
        tv.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 25);
        new AlertDialog.Builder(mContext)
                .setCustomTitle(tv)
                .setView(et)
                .setPositiveButton("确定", new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int which) {
                        String input = et.getText().toString();
                        if (StringUtil.isEmpty(input)) { // 不满足条件式时,提示,并不dismiss该dialog
                            CommonUtil.customToast(mContext, "数据包长度不能为空", Toast.LENGTH_SHORT, Gravity.TOP);
                            try {
                                Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
                                field.setAccessible(true);
                                field.set(dialog, false);
                            } catch (Exception e) {
                                e.printStackTrace();
                            }
                        } else {
                            int length = Integer.valueOf(input);
                            if (length < 500) {
                                CommonUtil.customToast(mContext, "数据包长度至少500", Toast.LENGTH_SHORT, Gravity.TOP);
                                try {
                                    Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
                                    field.setAccessible(true);
                                    field.set(dialog, false);
                                } catch (Exception e) {
                                    e.printStackTrace();
                                }
                            } else { // 满足条件式时,使该dialog
可以dismiss                                PreferenceUtil.getInstance().persist(ConstValue.TEST_DATA_LENGTH, length);
                                try {
                                    Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
                                    field.setAccessible(true);
                                    field.set(dialog, true);
                                } catch (Exception e) {
                                    e.printStackTrace();
                                }
                            }
                        }
                    }
                })
                .setNegativeButton("取消", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        try {
                            Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
                            field.setAccessible(true);
                            field.set(dialog, true);
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }
                })
                .show();

这样就可以实现我们上面的需求了。

注:在开始实现反射这样的方法时,报了这样的错误:

java.lang.NoSuchFieldException: No field mShowing in class Landroid/support/v7/app/AppCompatDialog; (declaration of 'android.support.v7.app.AppCompatDialog' appears in /data/app/com.rencarehealth.mirhythm-1/split_lib_dependencies_apk.apk:classes19.dex)

从错误描述可知,mShowing名字的field并不存在AppCompatDialog的class中,所以需要将你代码中引入的Dialog类修改为非support_v7中的Dialog。

<think>好的,我现在需要回答用户关于Androiddialog.dismiss()和dialog.cancel()的区别及用法的问题。首先,我得回忆一下这两个方法的区别。记得dismiss()是直接关闭对话框,而cancel()在关闭前会触发取消事件,比如调用OnCancelListener。但具体的细节可能需要进一步确认。 我应该先查阅Android的官方文档,确认这两个方法的具体行为。根据文档,dismiss()方法仅仅是移除对话框,会触发任何取消事件。而cancel()方法在调用时,除了移除对话框,还会触发OnCancelListener,如果有设置的话。此外,如果对话框是通过show()方法显示的,cancel()方法可能还会发送一个取消的结果,这在某些情况下可能影响后续的逻辑。接下来,我需要考虑如何用示例代码来说明两者的用法。例如,创建一个对话框并设置OnCancelListener,当调用cancel()时,这个监听器会被触发,而dismiss()则会。同时,可能需要说明在哪些场景下使用哪个方法更合适,比如当需要处理取消操作时使用cancel(),否则使用dismiss()。另外,用户可能还想知道这两个方法在实际开发中的常见错误或注意事项。比如,重复调用dismiss()是否安全,或者是否需要在取消后做一些资源清理工作。这时候需要指出,通常多次调用dismiss()会有问题,但最好通过判断对话框是否还在显示来避免异常。 还要考虑是否有相关的引用资料可以支持回答。例如,引用中提到的BootstrapDialog的实例操作,虽然来自同的框架,但概念上可能类似,可以辅助说明。但主要还是要依据Android本身的文档和常见实践。 最后,组织回答的结构,先解释区别,再给出示例代码,然后说明使用场景和注意事项,确保内容清晰易懂。同时,生成相关问题,帮助用户进一步探索,比如性能影响或生命周期问题。</think>### dialog.dismiss() 与 dialog.cancel() 的区别及用法 在 Android 中,`dialog.dismiss()` 和 `dialog.cancel()` 都用于关闭对话框,但行为细节和适用场景同: --- #### **1. 核心区别** | 方法 | 行为 | 触发监听器 | |-----------------|----------------------------------------------------------------------|--------------------------| | `dismiss()` | 直接关闭对话框,**触发**与取消相关的逻辑 | 无 | | `cancel()` | 关闭对话框,**触发**与取消相关的逻辑(如 `OnCancelListener`) | 触发 `OnCancelListener` | --- #### **2. 代码示例** ```java // 创建对话框并设置监听器 AlertDialog dialog = new AlertDialog.Builder(context) .setTitle("示例对话框") .setPositiveButton("确定", null) .setOnCancelListener(dialogInterface -> { Log.d("TAG", "对话框被取消"); // 仅通过 cancel() 触发 }) .create(); // 场景1:调用 dismiss() dialog.dismiss(); // 直接关闭,无日志输出 // 场景2:调用 cancel() dialog.cancel(); // 关闭并输出日志:"对话框被取消" ``` --- #### **3. 使用场景** - **`dismiss()`** - 适用于**明确关闭对话框**(如用户点击“确定”按钮)。 - 需要避免触发取消逻辑时使用。 ```java positiveButton.setOnClickListener(v -> dialog.dismiss()); ``` - **`cancel()`** - 适用于**用户主动取消操作**(如点击返回键或外部区域)。 - 需要清理取消相关资源时使用。 ```java dialog.setOnKeyListener((d, keyCode, event) -> { if (keyCode == KeyEvent.KEYCODE_BACK) { dialog.cancel(); // 触发 OnCancelListener return true; } return false; }); ``` --- #### **4. 注意事项** - **重复调用安全**:多次调用 `dismiss()` 是安全的,但建议先检查 `dialog.isShowing()`。 - **生命周期关联**:若对话框依赖 Activity 上下文,需在 `onDestroy()` 中调用 `dismiss()` 避免内存泄漏。 - **异步操作**:在异步任务完成后调用 `dismiss()`,需通过 `runOnUiThread()` 确保线程安全。 --- #### **5. 底层机制** - `cancel()` 内部会调用 `dismiss()`,但增加了事件触发逻辑[^3]。 - 若对话框通过 `show()` 显示,`cancel()` 可能发送 `RESULT_CANCELED` 到关联的 `DialogInterface.OnCancelListener`[^3]。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值