`
lifeijiyuan
  • 浏览: 40364 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

android progressbar

阅读更多
===================================================================

Android 在Java代码中设置style属性--使用代码创建ProgressBar对象
在andriod开发中,很大一部分都要与资源打交道,比如说:图片,布局文件,字符串,样式等等。这给我们想要开发一些公共的组件带来很大的困难,因为公共的组件可能更愿意以jar包的形式出现。但是java的jar包中只允许出现java代码而不能出现资源。

当我们想要以jar包的形式提供我们自己开发的公共组件时,我们就需要把以代码的形式创建资源。

下面提供一个使用全Java代码的形式创建一个ProgressBar。

ProgressBar默认的样式是一个圈圈,我们要想其显示为进度条的样式可以在布局文件中使用如下代码:

<ProgressBar android:layout_width="fill_parent"

android:layout_height="wrap_content"

style="?android:attr/progressBarStyleHorizontal"  />

上面的关键代码是红色的部分,这部分的代码就是使得ProgressBar由转圈圈的样式变成进度条的样式。使用这种方式创建的ProgressBar不能包含在jar包中。

同样我们也可以使用纯代码的形式创建ProgressBar对象,如下:

...

ProgressBar progressBar = new ProgressBar(context);

LineanerLayout layout = new LinearLayout(context);

layout.addView(progressBar, new LayoutParam(LayoutParam.FILL_PARENT, LayoutParam.FILL_PARENT));

....

这样就使用纯代码的方式创建了一个ProgressBar对象,但是他还只是默认的样式一个不停的转的圈圈。

这时我们可能都会想到好像没有设置样式。我们可以把之前的那个样式设进去,但是我们找遍API发现View并没有提供任何给我们设置样式的方法。

其实样式就是通过一种方式给一个View或一组View设置一些共同的属性值,所以不可能能使用代码来设置。

我们可以看下progressBarStyleHorizontal样式中给View设置了哪些属性,我们找到framework下的res目录下的values/Theme.xml文件,搜索progressBarStyleHorizontal会发现如下行:

<item name="progressBarStyleHorizontal">@android:style/Widget.ProgressBar.Horizontal</item>

该主题对应的Widget样式是Widget.ProgressBar.Horizontal,我们在同样的的目录下打开style.xml文件,搜索该样式,可以找到如下代码:

  <style name="Widget.ProgressBar.Horizontal">

        <item name="android:indeterminateOnly">false</item>

        <item name="android:progressDrawable">@android:drawable/progress_horizontal</item>

        <item name="android:indeterminateDrawable">@android:drawable/progress_indeterminate_horizontal</item>

        <item name="android:minHeight">20dip</item>

        <item name="android:maxHeight">20dip</item>

    </style>

也就是progressBarStyleHorizontal样式实际上就是设置了如上的属性,我们直接在布局文件中把如上的值设进去,代码看起来如下:

<ProgressBar android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:indeterminateOnly="false"

android:progressDrawable="@android:drawable/progress_horizontal"

android:indeterminateDrawable="@android:drawable/progress_indeterminate_horizontal"

android:minHeight="20dip"

android:maxHeight="20dip" />

这时运行我们的程序,发现ProgressBar已从圈圈变成进度条的样式。这时我们可以在代码中把这些属性设成布局文件中的值,纯Java代码看起来应该如下面的那样:

ProgressBar progressBar = new ProgressBar(this);

        progressBar.setIndeterminate(false);

        progressBar.setProgressDrawable(getResources().getDrawable(android.R.drawable.progress_horizontal));

        progressBar.setIndeterminateDrawable(getResources().getDrawable(android.R.drawable.progress_indeterminate_horizontal));

        progressBar.setMinimumHeight(20);

       

        LinearLayout layout = new LinearLayout(this);

        layout.addView(progressBar, new LayoutParams(LayoutParams.FILL_PARENT, LayoutParams.WRAP_CONTENT));

        setContentView(layout);

这时我们发现ProgressBar确实变成了横条,但并没有显示成进度条的样子,我们仔细对比一下纯Java代码和xml布局文件之间差异,我们发现android:indeterminateOnly="false"和 progressBar.setIndeterminate(false);并不完全一样布局文件的属性有一个Only结尾但代码中并没有,我们查找Api发现并没有setIndeterminateOnly这样的一个方法。

我们打开ProgressBar的源代码,找到.setIndeterminate(false) 方法,方法的代码如下:



public synchronized void setIndeterminate(boolean indeterminate) {

        if ((!mOnlyIndeterminate || !mIndeterminate) && indeterminate != mIndeterminate) {

            mIndeterminate = indeterminate;



            if (indeterminate) {

                // swap between indeterminate and regular backgrounds

                mCurrentDrawable = mIndeterminateDrawable;

                startAnimation();

            } else {

                mCurrentDrawable = mProgressDrawable;

                stopAnimation();

            }

        }

    }

我们这时候可以发现Indeterminate和IndeterminateOnly并不是同一个东西,这时我们应该想的到,只要我们把IndeterminateOnly的值变成false就可以使ProgressBar变成进度条的样式,我们查找所有的代码,发现并没有提供相应的公开方法来修改该属性的值。

也就是说,我们讨论了那么久发现根本就无法通过纯代码的形式来创建一个进度条样式的ProgressBar.



但是。。。

不就是改变一个类的私有变量的值嘛,Java的封装性其实并没有我想的那么好,我们完全可以通过反射机制来修改一个对象的私有变量的值,由于该文章并不是讨论反射的的文章,所以这里只给出通过反射来修改私有变量值的代码,但并不作详细的说明:

我们创建一个新的类,叫BeanUtils.java

类得内容看其来如下:



public class BeanUtils {





    private BeanUtils() {

    }





    /**

     * 直接设置对象属性值,无视private/protected修饰符,不经过setter函数.

     */

    public static void setFieldValue(final Object object, final String fieldName, final Object value) {

        Field field = getDeclaredField(object, fieldName);



        if (field == null)

            throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + object + "]");



        makeAccessible(field);



        try {

            field.set(object, value);

        } catch (IllegalAccessException e) {

        Log.e("zbkc", "", e);

        }

    }



    /**

     * 循环向上转型,获取对象的DeclaredField.

     */

    protected static Field getDeclaredField(final Object object, final String fieldName) {

        return getDeclaredField(object.getClass(), fieldName);

    }



    /**

     * 循环向上转型,获取类的DeclaredField.

     */

    @SuppressWarnings("unchecked")

    protected static Field getDeclaredField(final Class clazz, final String fieldName) {

        for (Class superClass = clazz; superClass != Object.class; superClass = superClass.getSuperclass()) {

            try {

                return superClass.getDeclaredField(fieldName);

            } catch (NoSuchFieldException e) {

                // Field不在当前类定义,继续向上转型

            }

        }

        return null;

    }



    /**

     * 强制转换fileld可访问.

     */

    protected static void makeAccessible(Field field) {

        if (!Modifier.isPublic(field.getModifiers()) || !Modifier.isPublic(field.getDeclaringClass().getModifiers())) {

            field.setAccessible(true);

        }

    }

}


该工具提供一个共有的方法:public static void setFieldValue(final Object object, final String fieldName, final Object value)来修改一个对象的私有变量的值:

这时我们的ProgressBar代码看起来应该如下:

   ProgressBar progressBar = new ProgressBar(this);

        BeanUtils.setFieldValue(progressBar, "mIndeterminateOnly", new Boolean(false));

        progressBar.setIndeterminate(false);

        progressBar.setProgressDrawable(getResources().getDrawable(android.R.drawable.progress_horizontal));

        progressBar.setIndeterminateDrawable(getResources().getDrawable(android.R.drawable.progress_indeterminate_horizontal));

        progressBar.setMinimumHeight(20);

       

       

        LinearLayout layout = new LinearLayout(this);

        layout.addView(progressBar, new LayoutParams(LayoutParams.FILL_PARENT, LayoutParams.WRAP_CONTENT));

        setContentView(layout);



到此为止我们终于使用纯java代码的方式创建了一个ProgressBar的进度条样式。

++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

android中ProgressBar的使用

范例说明
Android的Widget,有许多是为了与User交互而特别设计的,但也有部分是作为程序提示、显示程序运行状态的Widget。现在介绍的范例,与前一章介绍过的ProgressDialog对话框的应用目的相似,但由于前章介绍的ProgressDialog是继承自Android.app.ProgressDialog所设计的互动对话窗口,在应用时,必须新建ProgressDialog对象,在运行时会弹出“对话框”作为提醒,此时应用程序后台失去焦点,直到进程结束后,才会将控制权交给应用程序,如果在Activity当中不希望后台失焦,又希望提示User有某后台程序正处于忙碌阶段,此时,ProgressBar就会派上用场了。

Android提供的ProgressBar Widget控件与ProgressDialog应用目标不同,在程序一开始即可在main.xml Layout当中布局,先将部署在Layout里的ProgressBar的属性设为隐藏(一开始看不见),而后使用进程来“假装”程序忙碌中,但不同的是,可在进程当中取得运行时的进度,在“运行”的过程中,将运行进度通过TextView显示出来。本范例除了学习ProgressBar Widget的显示及使用之外,另一个学习关键则是Handler的使用,因为新起的进程无法访问Activity里的Widget,也无法将运行状态外送出来,所以需要通过Handler及Message对象,将进程里的状态往外传递,最后由Activity的Handler事件接收取得运行的状态。


范例程序
src/irdc.ex04_17/EX04_17.java
为了让Thread运行过程中,可以不断地将信息往Activity传递,所以用了Android.os.Handler对象及Android.os.Message对象,且在类成员变量中声明了两个整数:GUI_STOP_NOTIFIER与GUI_THREADING_NOTIFIER,这两个整数将作为信息传递出来时的信号标识,前者为当Thread需要喊停的时候处理,后者为进程正在运行过程中所需处理的标识。

程序中设计了一个按钮,此按钮的工作是让原本部署在main.xml里的ProgressBar显示出来(原来是设置为Android:visibility="gone"),而因为默认在main.xml中没有指定它的indeterm- inate属性,所以即便在程序中强制调用了ProgressBar的setIndeterminate()方法,也无法改变ProgressBar.getProgress()的值,这个值将永远为0。因此,笔者想要使用循环图片动画作为运行过程中的动画,并用了一个Counter(整数)来递增,表示运行的百分比。

public class EX04_17 extends Activity

{

  private TextView mTextView01;

  private Button mButton01;

  private ProgressBar mProgressBar01;

  public int intCounter=0;

  protected static final int GUI_STOP_NOTIFIER = 0x108;

  protected static final int GUI_THREADING_NOTIFIER = 0x109;

  @Override

  public void onCreate(Bundle savedInstanceState)

  {

    super.onCreate(savedInstanceState);

    setContentView(R.layout.main);

    mButton01 = (Button)findViewById(R.id.myButton1);

    mTextView01 = (TextView)findViewById(R.id.myTextView1);

    mProgressBar01 = (ProgressBar)findViewById(R.id.myProgressBar1);

    mProgressBar01.setIndeterminate(false);

    mButton01.setOnClickListener(new Button.OnClickListener()

    {

      @Override

      public void onClick(View v)

      {

        // TODO Auto-generated method stub

        mTextView01.setText(R.string.str_progress_start);

        mProgressBar01.setVisibility(View.VISIBLE);

        mProgressBar01.setMax(100);

        mProgressBar01.setProgress(0);

        new Thread(new Runnable()

        {

          public void run()

          {

            for (int i=0;i<10;i++)

            {

              try

              {

                intCounter = (i+1)*20;

                Thread.sleep(1000);

                if(i==4)

                {

                  Message m = new Message();

                  m.what = EX04_17.GUI_STOP_NOTIFIER;

                  EX04_17.this.myMessageHandler.sendMessage(m);

                  break;

                }

                else

                {

                  Message m = new Message();

                  m.what = EX04_17.GUI_THREADING_NOTIFIER;

                  EX04_17.this.myMessageHandler.sendMessage(m);

                }

              }

              catch(Exception e)

              {

                e.printStackTrace();

              }

            }

          }

        }).start();

      }

    });

  }

  Handler myMessageHandler = new Handler()

  {

    // @Override

    public void handleMessage(Message msg)

    {

      switch (msg.what)

      {

        case EX04_17.GUI_STOP_NOTIFIER:

          mTextView01.setText(R.string.str_progress_done);

          mProgressBar01.setVisibility(View.GONE);

          Thread.currentThread().interrupt();

          break;

        case EX04_17.GUI_THREADING_NOTIFIER:

          if(!Thread.currentThread().isInterrupted())

          {

            mProgressBar01.setProgress(intCounter);

            mTextView01.setText

            (

              getResources().getText(R.string.str_progress_start)+

              "("+Integer.toString(intCounter)+"%)\n"+

              "Progress:"+

              Integer.toString(mProgressBar01.getProgress())+

              "\n"+"Indeterminate:"+

              Boolean.toString(mProgressBar01.isIndeterminate())

            );

          }

          break;

      }

      super.handleMessage(msg);

    }

  };

}

扩展学习
范例程序中,调用mProgressBar01.setIndeterminate(false),不显示背景进度Bar,若设置为mProgressBar01.setIndeterminate(true),也无法让默认的ProgressBar图片(转圈圈)有正确的进度提示,理由是默认的ProgressBar不支持indeterminate mode循环图片方式,所以即便setIn- determinate(true)也无法正确显示进度。在本程序中,为刻意写出作为对照练习,一般在未知“进度”的情况下,可改用文字的方式显示进度百分比,通过调用mProgressBar01.getProgress()取得运行进度值,显示在文字中。请将Layout里的ProgressBar Widget定义中,加上一个android: indeterminateOnly属性,指定其值为false,不显示后台进度Bar,如下所示:

<ProgressBar

  android:id="@+id/myProgressBar1"

  android:layout_width="wrap_content"

  android:layout_height="wrap_content"

  android:max="100"

  android:progress="0"

  android:orientation="horizontal"

  android:progressBarStyle=

  "@android:style/Widget.ProgressBar.Horizontal"

  android:indeterminateOnly="false"

  android:visibility="gone"

/>

ProgressBar除了上述关于Android:progressBarStyle的属性设置之外,笔者也调查了在线Android的源代码(http://source.android.com),一些原本Android所使用的progressBarStyleHori- zontal属性,除默认“圆形”的图片之外,还有其他的主题及方形图片Drawable模式可以使用。

<resources>

  <declare-styleable name="Theme">

  <!-- snip -->

  <attr name="progressBarStyleHorizontal" format="reference" />

</resources>

接下来看看,这段主题中的属性名称progressBarStyleHorizontal定义在frameworks/base/ core/res/res/values/ styles.xml里,如下所示:

<resources>

  <style name="Widget.ProgressBar.Horizontal">

    <item name="android:indeterminateOnly">false</item>

    <item name="android:progressDrawable">

      @android:drawable/progress_horizontal

    </item>

    <item name="android:indeterminateDrawable">

      @android:drawable/progress_indeterminate_horizontal

    </item>

    <item name="android:minHeight">20dip</item>

    <item name="android:maxHeight">20dip</item>

  </style>

</resources>

由此可见,如果想让Android使用其他样式的ProgressBar,可以在原本的Layout(main.xml)里添加以下两项属性,以观察运行过程中的图片变化。

android:progressDrawable="@android:drawable/progress_horizontal"

android:indeterminateDrawable=

"@android:drawable/progress_indeterminate_horizontal"
分享到:
评论
1 楼 accelater 2012-01-12  
[/u][u][u][/u]

相关推荐

    AndroidProgressBar_Android ProgressBar进度条的几乎全部的用法源码集.rar

    本资源"AndroidProgressBar_Android ProgressBar进度条的几乎全部的用法源码集.rar"提供了关于Android ProgressBar的多种实现方式和应用场景的源代码,非常适合开发者学习和参考。 首先,我们来详细探讨一下Android...

    android ProgressBar 全部的用法

    在Android开发中,ProgressBar是一个非常重要的组件,它用于表示任务的进度或操作的等待状态。ProgressBar提供了多种样式和用法,适用于不同的界面需求。本文将详细介绍Android ProgressBar的使用方法,帮助开发者更...

    类似iOS菊花效果的android progressbar

    在Android开发中,ProgressBar是用于显示进度或加载状态的一个组件,它可以给用户带来良好的交互体验。iOS系统中的菊花效果(也称为活动指示器)是一种典型的加载动画,它以旋转的圆环形展示加载过程。本篇文章将...

    android ProgressBar进度条等待demo

    在Android开发中,ProgressBar是一个非常重要的组件,它用于显示任务的进度或表示一个正在执行的操作。这个"android ProgressBar进度条等待demo"是一个示例项目,旨在帮助开发者理解如何在应用程序中实现不同类型的...

    android Progressbar

    在Android开发中,Progressbar是一个重要的UI组件,用于显示进度或加载状态,用户通常会在数据加载、文件上传或下载等耗时操作时看到它。在本案例中,"android progressBar test"很可能是一个示例项目,用于展示如何...

    Android ProgressBar 简单进度条的实现及示例下载.rar

    此为Android ProgressBar 简单进度条的实现及示例下载源码,基本思路是:将模拟任务分成大小为100个的数组,任务完成度--默认为0,状态完成度默认也为0,然后构造一个线程用于显示消息,然后定义dowork()函数,完成...

    android progressbar使用系列集合

    在Android开发中,ProgressBar是一个非常常见的组件,用于展示任务的进度或者表示正在执行的操作。本篇文章将深入探讨如何自定义ProgressBar以满足特定的设计需求,尤其是根据系统的风格进行定制,包括改变默认背景...

    android progressbar 垂直布局

    在Android开发中,ProgressBar是用于显示进度或加载状态的一个组件,通常我们看到的都是水平方向的条形或圆形进度条。然而,有时候根据界面设计的需求,我们可能需要将ProgressBar设置为垂直布局,这就是"android ...

    Android应用源码之ProgressBar 几乎全部的用法

    在Android开发中,ProgressBar是一个非常重要的组件,它用于显示进度或等待状态,为用户提供反馈,表明应用程序正在进行后台处理。ProgressBar提供了多种样式和用法,包括水平、垂直、圆形以及自定义样式。本教程将...

    android ProgressBar各种用法

    ### Android ProgressBar 各种用法详解 #### 一、引言 `ProgressBar` 是 Android 开发中最常用的 UI 控件之一,用于表示任务处理进度或等待时间的状态。它可以帮助用户了解应用正在后台执行某些操作,避免因为等待...

    android ProgressBar的应用demo

    在Android开发中,ProgressBar是一个非常重要的控件,用于显示任务的进度,比如数据加载、文件下载等操作。这个“android ProgressBar的应用demo”是为开发者提供一个直观的学习和实践平台,帮助理解如何在实际项目...

    自定义Android ProgressBar ics4.0加载样式

    在Android开发中,ProgressBar是一个非常重要的组件,它用于显示进度或加载状态,为用户提供反馈。在ICS(Ice Cream Sandwich)4.0版本中,ProgressBar引入了一些新的视觉样式,特别是加载转圈的效果,使得UI看起来...

    android progressBar操作(基于eclipse)

    在Android开发中,ProgressBar是一个非常重要的组件,它用于表示任务的进度或操作的等待状态。在Eclipse这个经典的Android开发环境中,对ProgressBar的使用和操作是开发者必须掌握的基础技能之一。下面我们将深入...

    android ProgressBar 示例二

    在Android开发中,ProgressBar是一个非常重要的组件,它用于显示进度或加载状态,为用户提供反馈,让用户知道应用程序正在执行后台操作。本示例将深入探讨如何在Android应用中使用ProgressBar,并提供一个具体的实例...

    Android ProgressBar进度条Demo源码.rar

    在Android开发中,ProgressBar是一个非常重要的组件,它用于显示任务的进度,比如数据加载、文件下载等操作。这个"Android ProgressBar进度条Demo源码.rar"压缩包提供的就是一个关于如何在Android应用中使用...

    Android progressbar

    在Android开发中,ProgressBar是用户界面(UI)中不可或缺的组件,它用于向用户显示某个操作的进度,比如数据加载、文件下载等。当需要在应用程序中实现一个对话框(Dialog)形式的进度条时,我们可以使用Android的...

    Android ProgressBar 几乎全部的用法.rar

    在Android开发中,ProgressBar是一个非常重要的组件,它用于显示进度状态,可以是确定性的,比如文件下载或上传的百分比,也可以是不确定性的,比如在后台执行任务时展示加载状态。这个压缩包“Android ProgressBar ...

    Android ProgressBar 几乎全部的用法.zip

    在Android开发中,ProgressBar是一个非常重要的组件,它用于表示一个操作的进度或者提供用户等待的反馈。这个"Android ProgressBar 几乎全部的用法.zip"压缩包可能包含了关于ProgressBar的各种用法,包括基本的样式...

    AndroidProgressBar-几乎全部的用法.zip

    这个资源包"AndroidProgressBar-几乎全部的用法.zip"显然是为了帮助开发者全面了解如何在Android应用中有效地使用ProgressBar。以下是对这个主题的详细讲解: 1. **ProgressBar类型**: - **Circular ProgressBar*...

Global site tag (gtag.js) - Google Analytics