自定义ProgressDialog

ProgressDialog

ProgressDialog

项目当中遇到这样一个应用场景:执行某个操作需要耗时15秒以上,依照惯例,这就要使用到进度条一类的UI控件,以安抚用户等待的烦躁心情。Android Framework已经提供了ProgressDialog,可以很好的解决这个问题。

ProgressDialog实际上是AlertDialog的子类,其有着两种不同的表现形式。第一种是针对没有明确的进度,不知道当前完成了多少的情况,此时使用一个转动的圆环来展现;第二种是针对有了明确的总进度,并知道当前的完成比例等信息,此时使用的是一个横条来展现。根据项目方案,我们的效果类似第一种情形。

不过我所处的项目情况比较特殊,因为由于设备的特性导致不能频繁刷新屏幕,–刷新一次屏幕的时间大于2s。因此凡是动画之类的效果统统禁用,原生的ProgressDialog也就不能使用了,只能另想解决办法。

解决问题的办法之一就是修改ProgressDialog的默认实现,使其不要频繁刷新。修改之前要先清楚其是怎么实现的。通过源代码可以知道,ProgressDiglog是通过在themes.xml定义progressBarStyle,进一步从styles.xml中找到Widget.ProgressBar的样式定义:

<style name="Widget.ProgressBar">
    <item name="android:indeterminateOnly">true</item>
    <item name="android:indeterminateDrawable">@android:drawable/progress_medium</item>
    <item name="android:indeterminateBehavior">repeat</item>
</style>

如果没猜错的话,接下来要做事情就变成了修改android:indeterminateDrawable对应的drawable。

先向drawable文件夹下增加两个图片progress_round_1.png和progress_round_2.png,然后增加一个XML文件progress_round.xml。

<?xml version="1.0" encoding="utf-8"?>
<animation-list
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:oneshot="false">
    <item android:drawable="@drawable/progress_round_1" android:duration="2000" />
    <item android:drawable="@drawable/progress_round_2" android:duration="2000" />
</animation-list>

这样不断转动的圆环变成了自己想要的图片。

接着增加相应的style:

<style name="ProgressRound">
    <item name="android:indeterminateDrawable">@drawable/progress_round</item>
</style>

然后增加相应的theme:

<style name="OppoTheme" parent="@android:style/Theme.Dialog.Alert">
    <item name="android:progressBarStyle">@style/ProgressRound</item>
</style>

最后来到生成Dialog的地方:

    @Override
    protected Dialog onCreateDialog(int id) {
    	Log.d(TAG, "onCreateDialog");
 
        switch (id) {
            case PRODIALOG_KEY: 
            	mProDialog = new ProgressDialog(this, R.style.OppoTheme);
            	mProDialog.setIndeterminate(true);
            	mProDialog.setCancelable(true);
                return mProDialog;
            default:
            	return null;
        }
    }

13 Comments

Frank zhang28 12 月, 2009 at 6:52 下午

请问是在源码中修改么?

Xu Haojie29 12 月, 2009 at 12:25 上午

不,我是在应用中“覆盖”掉了系统默认的表现方式,并不涉及android源码修改。

Frank zhang29 12 月, 2009 at 9:47 上午

我照你这个方法做了,几乎是完全copy过去的,但显示效果依然是默认的效果,不知是怎么回事?

Frank zhang29 12 月, 2009 at 9:48 上午

需要重写ProgressDialog么?应该不需要吧,只是重新自定义了一个样式而已,好像。

Xu Haojie30 12 月, 2009 at 12:30 上午

对,不需要重写,只是定义了一个样式

Frank zhang29 12 月, 2009 at 9:50 上午

能否把过程发给我参考下,0728zhj#163.com,万分感谢!

Xu Haojie30 12 月, 2009 at 12:34 上午

非常遗憾,由于一些制度原因,不能提供更为详细的代码,望谅解。

Frank zhang29 12 月, 2009 at 11:44 上午

是不是还需要在配置文件里配置上面东西?

Xu Haojie30 12 月, 2009 at 12:39 上午

由于现在手头上没有代码,回忆了一下,这个style应该是要加载AndroidManifest.xml中对应的activity的节点上,试试看。

张井路21 6 月, 2011 at 4:59 下午

按照你的方法做,确实不行啊?请有时间,修改修改博客吧

Xu Haojie21 6 月, 2011 at 7:43 下午

谢谢你的提醒,时间久远了,博文难免陈旧,上面的内容仅供参考,毕竟尽信书不如无书。

Ming31 8 月, 2011 at 9:23 上午

@drawable/progress_round

@style/ProgressRound

请问这两节点是加在哪?

tigoss3 11 月, 2011 at 4:29 下午

太复杂,有很简单的办法

Leave a comment

Your comment