`

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的进度条样式。

 

分享到:
评论

相关推荐

    纯Java代码创建ProgressBar,不使用xml

    在Java代码中创建ProgressBar,我们首先需要实例化一个ProgressBar对象,并将其添加到视图层次结构中。以下是一个简单的示例: ```java ProgressBar progressBar = new ProgressBar(context); // 设置ProgressBar的...

    Android开发之ProgressBar及其style样式的使用.zip

    本教程将详细讲解如何在Android应用中使用ProgressBar及其style样式,特别关注使用Kotlin进行实现。 一、ProgressBar的基本使用 ProgressBar在Android中分为两种类型:垂直和水平。默认情况下,它是水平的,可以...

    使用ProgressBar实现进度条,设置进度条显示风格。以及自定义进度条样式

    在Android开发中,ProgressBar是一个非常常见的组件,用于展示任务的执行进度,比如文件...在ProgressBar-master这个项目中,可能包含了更多关于自定义进度条样式的示例代码,可以参考学习以提升你的Android开发技能。

    Android代码-circular-music-progressbar

    Android Circular Music ProgressBar Description This Circular Progress bar is a designed and made for music players that need beautiful music progress bar. Choosed for best of the day at uplabs.com ...

    Android之自定义ProgressBar

    在Android开发中,ProgressBar是用户界面中非常常见的一种组件,用于表示某个操作的进度或等待状态。本教程将深入探讨如何在Android中自定义ProgressBar,让其满足特定的设计需求,提升应用的用户体验。 首先,我们...

    Android代码-一个可以定制进度动画的ProgressBar

    在Android开发中,ProgressBar是用于显示进度或等待状态的基本组件,它为用户提供了可视化的反馈。本项目"AnimProgressBar"专注于提供一个可自定义动画效果的ProgressBar,这为开发者提供了更多的设计自由度,以增强...

    Android代码-ProgressWheel支持进度显示的圆形ProgressBar.zip

    在压缩包中,"JavaApk源码说明.txt"可能是一个包含源代码解读和使用指南的文本文件,通常会讲解如何将ProgressWheel集成到项目中,包括添加依赖、布局文件中的使用方法以及如何在代码中控制进度更新。开发者可以通过...

    Android 动态生成 步骤 ProgressBar

    在Activity或Fragment的Java/Kotlin代码中,先实例化一个ProgressBar对象。例如: ```java ProgressBar progressBar = new ProgressBar(context); ``` 然后设置其属性,如样式、颜色、大小等。如果需要自定义...

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

    ProgressBar的基本使用非常简单,只需在布局文件中添加一个ProgressBar标签,设置其属性,如id、style、max等。例如: ```xml &lt;ProgressBar android:id="@+id/progressBar" style="?android:attr/...

    Android代码-android-square-progressbar

    If you're using the Eclipse environment then check out the legacy repository here: android-square-progressbar-legacy. You can find my blog post about the newest version here: halcyon.ch - android-...

    Android中使用ProgressBar实现进度条

    本篇文章将深入探讨如何在Android中使用ProgressBar以及自定义其样式。 首先,ProgressBar的基本用法是在XML布局文件中声明它。以下是一个简单的例子: ```xml &lt;ProgressBar android:id="@+id/progressBar" ...

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

    2. **Java代码控制**:在Java代码中,可以通过`ProgressBar`对象来控制进度条的显示和隐藏,以及设置进度值: ```java ProgressBar progressBar = findViewById(R.id.progressBar); progressBar.setVisibility(View....

    android Progressbar

    在Java代码中,可以通过获取ProgressBar的引用并调用`setProgress()`方法来设置或更新进度: ```java ProgressBar progressBar = findViewById(R.id.progressBar); progressBar.setProgress(50); // 设置进度为50% ...

    android中自定义progressbar组件

    总结来说,自定义一个基于逐帧动画的ProgressBar组件,涉及到的主要知识点有:ProgressBar组件的使用、动画资源的创建、XML布局文件的编写以及在Java代码中的动画控制。通过这样的实践,开发者可以更好地理解Android...

    Android代码-progressBar

    主要是分段时progressbar的颜色不一样 Canvas:https://developer.android.com/reference/android/graphics/Canvas.html mProgressbarView.setProgress(30,true); 和 mProgressbarView.setProgress(50); 同一个控件...

    html5-canvas-border-pixel-progressbar

    在"html5-canvas-border-pixel-progressbar"这个项目中,关键在于利用Canvas的`strokeStyle`属性来设置边框颜色,并使用`beginPath()`、`moveTo()`、`lineTo()`、`stroke()`等方法来绘制边框。这些方法配合起来可以...

    Android progressBar的简单使用例子

    在Android开发中,ProgressBar是一个非常重要的组件,它用于显示进度状态,可以是确定性的,...这个例子展示了如何在Android应用中创建和控制一个基本的ProgressBar,通过查看源代码,你可以更深入地理解其工作原理。

    Android代码-仿知乎的横线直线progressbar.rar

    【标题】"Android代码-仿知乎的横线直线progressbar.rar" 涉及的是在Android平台上开发一个类似于知乎应用中的进度条组件。这个组件通常用于显示任务或数据加载的进度,给用户一种视觉反馈,表明应用程序正在执行...

    android progressbar使用系列集合

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

    Android代码-带文字的ProgressBar.zip

    7. **示例代码**:在项目的主目录下,可能会有一个示例代码的Java或Kotlin文件,展示了如何在代码中使用这个自定义的带文字的ProgressBar,包括初始化、设置进度和文字内容等操作。 理解并掌握这种自定义...

Global site tag (gtag.js) - Google Analytics