Android实战技巧:Dialog (转)

转:http://blog.csdn.net/hitlion2008/article/details/7567549#t0

Dialog是任何系统都必须有的一个控件,作为辅助窗口,用于显示一些消息,或请求用户采取一引起操作等。

在Android中也不例外,基本使用可能参看文档。

使用时的注意事项

1. BACK键能取消掉对话框(dismiss),但是却不会触发其onOkey和onCancel回调接口,所以如果你的对话框会改某些状态,一定要注意还有第三种方式取消对话框。

2. 尽量少用模态对话框(Model
dialog),如果Dialog.setCancellable(false),就变成了一个模态对话框,除了程序内部把其Dismiss,否则按什么
键都无法将其取消。这是极差的用户体验,对话框本身就是一种干扰,再无法取消会把用户搞疯的。所以除非特别有必要,也即当执行某个操作时不希望被打破,才
可以使用模态对话框。

3. 尽量少用对话框,它对用户是一种干扰,除非需要用户做操作,或者做出选择。通常的一般性的通知用Toast或者Notification就足够了。

4.
不要使用对话框风格的Activity,也即把Activity变成一个对话框。因为这样是自已定义的布局,与系统Dialog的风格可能会不一致。最严
重的是当系统风格发生变化,Dialog的子类会变化,但Activity式的对话框就不会变化。可以在ICS中找一找Activity对话框,你会发现
其OK是在左边,而ICS中系统Dialog的OK都是在右边的。

5. 尽量保证Dialog对象活在Activity的生命周期之内,也即至多是在onCreate()和onDestroy()之间。

6.
要想到和测试到Activity在其Dialog.dismiss()之前死掉的情况。因为Activity必须依附于某个正在显示的Activity实
例,当显示和取消的时候其Activity实例必须存在,否则就会有"IllegalArgumentException: View not
attached to window manager"。

05-15 02:45:26.320: E/AndroidRuntime(1161): java.lang.IllegalArgumentException: View not attached to window manager
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.view.WindowManagerImpl.findViewLocked(WindowManagerImpl.java:355)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.view.WindowManagerImpl.removeView(WindowManagerImpl.java:200)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.view.Window$LocalWindowManager.removeView(Window.java:432)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.app.Dialog.dismissDialog(Dialog.java:278)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.app.Dialog.access$000(Dialog.java:71)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.app.Dialog$1.run(Dialog.java:111)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.app.Dialog.dismiss(Dialog.java:268)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at com.hilton.effectiveandroid.app.DialogDemo$1.handleMessage(DialogDemo.java:26)
05-15 02:45:26.320: E/AndroidRuntime(1161):     at android.os.Handler.dispatchMessage(Handler.java:99)

7. Dialog.show()必须在主线程里调用,但Dialog.dismiss()却可以在任何线程中调用。

三种使用方式比较

1. 直接创建一个局部的Dialog对象

优点是变量是局部的容易理解和维护。缺点是Dialog对象难以控制,容易引发RuntimeException。

2. 把Dialog对象变成Activity的域

优点是Dialog对象可以重复利用,且Activity可以控制以保证Dialog不会在Activity生命周期外显示。是推荐的使用方式。

3. 用Activity的方法onCreateDialog(), showDialog()和dismissDialog()

优点是Frameworks会帮忙照看Dialog,在大多数情况下这是推荐的做法。但是对于Activity提前死掉的情况,此方法必有RuntimeException,且无法回避。

实例

public class DialogDemo extends Activity {
    private static final int DISMISS_DIALOG = 1;

    private ProgressDialog mBetterDialog;

    private Handler mMainHandler = new Handler() {
	@Override
	public void handleMessage(Message msg) {
	    switch (msg.what) {
	    case DISMISS_DIALOG:
		Dialog dialog = (Dialog) msg.obj;
		dialog.dismiss();
		break;
	    default:
		break;
	    }
	}
    };
    @Override
    protected void onCreate(Bundle savedInstanceState) {
	super.onCreate(savedInstanceState);
	setContentView(R.layout.dialog_demo);

	final Button sucking = (Button) findViewById(R.id.sucking);
	sucking.setOnClickListener(new View.OnClickListener() {
	    public void onClick(View v) {
		final Activity activity = DialogDemo.this;
		final ProgressDialog dialog = new ProgressDialog(activity);
		dialog.setTitle("Worst dialogging");
		dialog.setMessage("This is the worst dialogging scheme, NEVER use it. This dialog is easy to " +
			"run out of its attached activity, yielding WindowManager#BadTokenException if the activity is gone when dismissing");
		dialog.setIndeterminate(true);
		dialog.setCancelable(true);
		// You MUST do the show in main thread anyway
		dialog.show();
		new Thread(new Runnable() {
		    public void run() {
			SystemClock.sleep(10000);
			/*
			 * IllegalArgumentException: View not attached to window manager
			 * If the activity showing the dialog was killed before dismiss() out of rotation or locale changed,
			 * the dialog will gone with activity, but when dismiss() yields "IllegalArgumentException: View not attached to
			 * window manager".
			 * Checking isShowing() won‘t help.
			 * Checking activity.isFinishing() won‘t help, either.
			 * Dismiss it in main thread also won‘t give any help.
			 */
			// THIS WON‘t WORK
//			if (dialog.isShowing()) {
//			    dialog.dismiss();
//			}
//			if (!activity.isFinishing()) {
//			    dialog.dismiss();
//			}
			Message msg = Message.obtain();
			msg.what = DISMISS_DIALOG;
			msg.obj = dialog;
			mMainHandler.sendMessage(msg);
		    }
		}).start();
	    }
	});

	final Button better = (Button) findViewById(R.id.better);
	better.setOnClickListener(new View.OnClickListener() {
	    public void onClick(View v) {
		mBetterDialog = new ProgressDialog(DialogDemo.this);
		mBetterDialog.setTitle("Better dialogging");
		mBetterDialog.setMessage("This dialogging can be used. The dialog object is a field of its activity, so activity can" +
				" control it to make sure dialog only lives within activity lifecircle");
		mBetterDialog.setIndeterminate(true);
		mBetterDialog.setCancelable(true);
		// You MUST do the show in main thread anyway
		mBetterDialog.show();
		new Thread(new Runnable() {
		    public void run() {
			SystemClock.sleep(10000);
			/*
			 * This is much better, mBetterDialog is a field of its activity, so activity can take care of it in order
			 * to make sure dialog only live within activity‘s life circle, to avoid any unexpected exceptions.
			 */
			// THIS really works
    			if (mBetterDialog != null && mBetterDialog.isShowing()) {
    			    mBetterDialog.dismiss();
    			}
		    }
		}).start();
	    }
	});

	final Button optional = (Button) findViewById(R.id.optional);
	optional.setOnClickListener(new View.OnClickListener() {
	    @SuppressWarnings("deprecation")
	    public void onClick(View v) {
		showDialog(0);
		new Thread(new Runnable() {
		    public void run() {
			SystemClock.sleep(10000);
			/*
			 * This way works best for most of time, except if activity died before dismissing, exception must be
			 * thrown: "IllegalArgumentException: View not attached to window manager".
			 * Although activity takes care of its belonging dialog, there is no way to operate it manually any more.
			 * First you do not have reference to dialog object and second, any manual operation only interferences
			 * and breaks state maintained by frameworks.
			 */
			dismissDialog(0);
		    }
		}).start();
	    }
	});
    }

    @Override
    protected Dialog onCreateDialog(int id) {
	ProgressDialog d = new ProgressDialog(this);
	d.setTitle("Optional dialogging");
	d.setMessage("This dialogging scheme works best for most times, the dialogs are all taken care of by activitys and frameworks" +
			". Except for activity being killed during dialog showing");
	d.setIndeterminate(true);
	d.setCancelable(true);
	return d;
    }

    @Override
    protected void onDestroy() {
	super.onDestroy();
	// Activity is dying, all its belonging dialogs should be dismissed, of course.
	if (mBetterDialog != null && mBetterDialog.isShowing()) {
	    mBetterDialog.dismiss();
	    mBetterDialog = null;
	}
	// For dialogs showed via showDialog(int), no way to stop it in onDestroy()
//	dismissDialog(0); // cause "IllegalArgumentException: no dialog with id 0 was ever shown via Activity#showDialog"
			    // This is because Activity has to manage its dialog during onPause() and onResume() to restore
	                  // dialogs‘ state. So if you manually dismiss it in onDestroy(), it will cause JE.

//	removeDialog(0);// cause "IllegalArgumentException: no dialog with id 0 was ever shown via Activity#showDialog", when
			// dismissing in thread.
	              // This is because Activity has to manage its dialog during onPause() and onResume() to restore
                     // dialogs‘ state. So if you manually dismiss it in onDestroy(), it will cause JE.
    }
}
时间: 2024-12-29 23:12:58

Android实战技巧:Dialog (转)的相关文章

Android实战技巧之四十三:终止一个线程引起的

这是一道老牌面试题.通常面试官会问你对Java线程的了解,然后再问此问题. 从理论到实践,这是一条好路子. 线程是操作系统实现多任务的一种方式,可以理解为线程是一个任务的执行单元.比如Android系统中每个App都会有自己的主线程,同时还可以创建worker thread"并行"为我们工作. Java中创建新线程的方法 Java对线程(Thread)提供了语言级的支持(依托虚拟机吧).java.lang包下有Thread类和Runnable接口,都可以替你完成创建新线程的工作. 1.

Android实战技巧:深入解析AsyncTask

AsyncTask的介绍及基本使用方法 关于AsyncTask的介绍和基本使用方法可以参考官方文档和Android实战技巧:多线程AsyncTask这里就不重复. AsyncTask引发的一个问题 上周遇到了一个极其诡异的问题,一个小功能从网络上下载一个图片,然后放到ImageView中,是用AsyncTask来实现的,本身逻辑也很简单,仅是在doInBackground中用HTTP请求把图片的输入流取出,然后用BitmapFactory去解析,然后再把得到的Bitmap放到ImageView中

Android实战技巧:ViewStub的应用

Android实战技巧:ViewStub的应用

Android实战技巧之二十七:Maven编译开源二维码扫描项目zxing

拥有自己的手机软件工具箱是件非常有意义的事情.就目前国内Android的生态环境来说,混乱的不能再乱了.由于我们登录不了官网App商店,下软件就只好在国内五花八门的软件市场下载.由于这些市场的监管不力,什么样的软件都有,就拿二维码扫描软件来说,好多都带那种狗皮膏药一样的广告插件,真是特别讨厌. 在开源世界中有很多优秀的软件,其中zxing就是非常好的Android扫碼工具软件.我们可以拿来即用还可以学习内部机制,然后做些定制化个性化.既可以自己享用,又可以跟大家分享.真是不错. zxing在gi

Android实战技巧之十一:Android Studio和Gradle

经过两个多月的AS体验,我认为是时候将Android的开发环境迁移到AS上了.目前最新版本是1.0.2,除了UI控件拖拽偶尔崩溃的问题(Ubuntu),其他功能用来还是十分流畅和高效.打动我的有如下几个特色: 智能感知体验特好,堪比VS 布局预览,手写布局后预览页面即时显示,便于布局调整和优化 编辑速度飞快流畅,毫无eclipse的卡顿 布局或源码中有图标和颜色的预览,十分直观 调试时体验极佳 集成了Terminal,喜欢命令行操作的伙伴不用额外启动终端了. 总之一句话,就是用起来特别爽! An

Android实战技巧之二十四:横竖屏切换

这几年一直在做手机上和电视盒的App,几乎没有考虑过横竖屏切换的问题.电视盒好说,横屏不变,你要是给它设计个竖屏人家也没机会使:而手机上的应用就不好说了,有些界面你设计了横竖屏兼容可能是为了表示你的功能强大.但是按照惯例,或许也是设计师图省事,我们只是做一个方案.就像目前主流的App都只有竖屏一个模式,比如微信.京东和招商银行.我截了几张图表示一下. 但是像地图之类的应用,也许横屏会显示的更友好一些.请看腾讯地图的设计如下: 细心的你会发现,地图的横竖屏的样式几乎是一样的布局,调整起来还是比较容

Android实战技巧之三十:人脸检测-静态

最近微软的how-old.net把人脸识别技术又大大的火了一把.通过大数据和复杂的算法,能够神奇的预测出照片中人物的性别和年龄.虽然错误率也不低,但是大家都抱着玩一玩乐一乐的心态把照片传上去让机器来鉴定一下自己的颜龄. 人脸识别算法是高深复杂的,面对着计算机视觉的种种数学公式,我就已经投降了.先来简单的玩玩人脸检测吧.Android早已提供了FaceDetector类,今天就来看看如何使用这个类人脸检测吧. 流程: 1.打开文件夹选择照片 2.将照片加载到bitmap中并缩放到设置的宽高 3.用

Android实战技巧之八:Ubuntu下切换JDK版本【转】

本文转载自:http://blog.csdn.net/lincyang/article/details/42024565 Android L之后推荐使用JDK7编译程序,这是自然发展规律,就像是4年前编译android 1.6需要使用JDK5一样. 多版本JDK是可以共存的,只需要使用update-alternatives工具就可以随时将它们切换.下面描述安装openjdk和Oracle jdk(对不住了sun)以及切换版本的过程. 一.安装openjdk7 [plain] view plain

Android实战技巧之十二:Android Studio导入第三方类库、jar包和so库

第三方类库源码 将一网友的XMPP代码从ADT转到AS时,发现其使用了第三方类库,源码放在了lib下,直接在AS中Import project,第三方类库并没有自动导入进来,看来需要自己动手了. 项目的目录结构如下: XMPP$ ls app build.gradle gradlew import-summary.txt XMPP.iml build gradle gradlew.bat local.properties settings.gradle 1 2 3 1 2 3 将第三方源码qqE