欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

解析android中ProgressBar的用法

程序员文章站 2023-10-19 14:32:53
范例说明android的widget,有许多是为了与user交互而特别设计的,但也有部分是作为程序提示、显示程序运行状态的widget。现在介绍的范例,与前一章介绍过的pr...

范例说明
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(整数)来递增,表示运行的百分比。

复制代码 代码如下:

/* import程序略 */

public class ex04_17 extends activity
{
  private textview mtextview01;
  private button mbutton01;
  private progressbar mprogressbar01;
  public int intcounter=0;

  /* 自定义handler信息代码,用以作为标识事件处理 */
  protected static final int gui_stop_notifier = 0x108;
  protected static final int gui_threading_notifier = 0x109;

  /** called when the activity is first created. */
  @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);

    /* 设置progressbar widget对象 */
    mprogressbar01 = (progressbar)findviewbyid(r.id.myprogressbar1);

    /* 调用setindeterminate方法赋值indeterminate模式为false */
    mprogressbar01.setindeterminate(false);

    /* 当单击按钮后,开始进程工作 */
    mbutton01.setonclicklistener(new button.onclicklistener()
    {
      @override
      public void onclick(view v)
      {
        // todo auto-generated method stub

        /* 单击按钮让progressbar显示 */
        mtextview01.settext(r.string.str_progress_start);

        /* 将隐藏的progressbar显示出来 */
        mprogressbar01.setvisibility(view.visible);

        /* 指定progress为最多100 */
        mprogressbar01.setmax(100);

        /* 初始progress为0 */
        mprogressbar01.setprogress(0);

        /* 开始一个进程 */
        new thread(new runnable()
        {
          public void run()
          {
            /* 默认0至9,共运行10次的循环语句 */
            for (int i=0;i<10;i++)
            {
              try
              {
                /* 成员变量,用以识别加载进度 */
                intcounter = (i+1)*20;
                /* 每运行一次循环,即暂停1秒 */
                thread.sleep(1000);

                /* 当thread运行5秒后显示运行结束 */
                if(i==4)
                {
                  /* 以message对象,传递参数给handler */
                  message m = new message();

                  /* 以what属性指定user自定义 */
                  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构建之后,会监听传来的信息代码 */
  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);

          /* 设置progressbar widget为隐藏 */
          mprogressbar01.setvisibility(view.gone);
          thread.currentthread().interrupt();
          break;

        /* 当取得标识为持续在进程当中时所取得的信息 */
        case ex04_17.gui_threading_notifier:
          if(!thread.currentthread().isinterrupted())
          {
            mprogressbar01.setprogress(intcounter);
            /* 将显示进度显示于textview当中 */
            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"