ListView实现顶部和底部内容指示器的方法

顶部指示器?
这是什么?
好吧,我承认这是我自己想出来的词,因为我不知道它有什么学名,究竟是什么呢?看下这个图就知道了。

这是我们的美工MM画的,偶的神呐,这虽然很漂亮,不过也让人头疼,这个箭头应该在滚到顶部的时候消失,滚下来的时候(即有条目隐藏的时候)才显示,类似的底部指示器也要有这样的效果。事实上默认的ListView和ScrollView都已经有了类似的效果,在顶部或底部还有更多内容时,会有部分渐变虚化的效果,不过美工已经设计了这样的效果,那么我们就来做吧。
出于省事的目的,本教程中的例子会基于上一篇教程来修改,主要是添加1个继承自ListView的类,以及修改布局定义文件。
ArrowListView控件的编写

package net.learningandroid.lib.view;

import net.learningandroid.lib.R;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.drawable.BitmapDrawable;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;
import android.widget.ListView;

/**
 * 支持上下箭头的ListView
 *
 * <a class="referer" href="http://my.oschina.net/arthor" target="_blank">@author</a> Mr. Lu
 */
public class ArrowListView extends ListView {

  private final float scale = getContext().getResources().getDisplayMetrics().density;
  private float topArrowPadding;
  private float bottomArrowPadding;

  private static float DEFAULT_TOP_PADDING_DP = 2.0f;
  private static float DEFAULT_BOTTOM_PADDING_DP = 2.0f;

  public ArrowListView(Context context, AttributeSet attrs) {
    super(context, attrs);

    String strTopArrowPadding = attrs.getAttributeValue(null,
        "topArrowPadding");
    String strBottomArrowPadding = attrs.getAttributeValue(null,
        "bottomArrowPadding");

    topArrowPadding = convertDisplayUom(strTopArrowPadding,
        DEFAULT_TOP_PADDING_DP);
    bottomArrowPadding = convertDisplayUom(strBottomArrowPadding,
        DEFAULT_BOTTOM_PADDING_DP);

    Log.v("ArrowListView", String.valueOf(topArrowPadding));
  }

  /**
   * 单位转换
   */
  private float convertDisplayUom(String sour, float defaultValue) {
    try {
      if (sour.toLowerCase().endsWith("px")) {
        return Float.parseFloat(sour.toLowerCase().replace("px", ""));
      } else if (sour.toLowerCase().endsWith("dp")) {
        return Integer.parseInt(sour.toLowerCase().replace("dp",
            ""))
            * scale + 0.5f;
      }
    } catch (Exception e) {
    }

    return (defaultValue * scale + 0.5f);
  }

  /**
   * onDraw方法,根据ListView滚动位置绘出箭头.
   */
  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    Paint paint = new Paint();

    // 取得箭头的图片,此处是固定图片,其实上可以做成配置方式
    Bitmap topPic = ((BitmapDrawable) getResources().getDrawable(
        R.drawable.arrow_up)).getBitmap();
    Bitmap bottomPic = ((BitmapDrawable) getResources().getDrawable(
        R.drawable.arrow_down)).getBitmap();

    // 取得ListView的绘制区域大小
    Rect r = new Rect();
    this.getDrawingRect(r);

    // 计算箭头的绘制位置
    float top = r.top + topArrowPadding;
    float bottom = r.bottom - bottomArrowPadding - bottomPic.getHeight();
    float left = r.left + (r.right - r.left - topPic.getWidth()) / 2;

    // 计算是否需要绘制
    boolean drawTop = false;
    boolean drawBottom = false;

    if (this.getChildCount() > 0) {
      Rect rTop = new Rect();
      this.getChildAt(0).getLocalVisibleRect(rTop);
      Rect rBottom = new Rect();
      View lastChild = this.getChildAt(this.getChildCount() - 1);
      lastChild.getLocalVisibleRect(rBottom);

      drawTop = (this.getFirstVisiblePosition() > 0 || this
              .getFirstVisiblePosition() == 0
              && rTop.top > 0);
      drawBottom = (this.getLastVisiblePosition() < this.getAdapter()
              .getCount() - 1 || this.getLastVisiblePosition() == this
              .getAdapter().getCount() - 1
              && rBottom.bottom < lastChild.getHeight());
    }
    // 绘出箭头
    if (drawTop) {
      canvas.drawBitmap(topPic, left, top, paint);
    }

    if (drawBottom) {
      canvas.drawBitmap(bottomPic, left, bottom, paint);
    }
  }
}

就要点解释一下上面这段代码:
注意构造方法,我们必须继承public ArrowListView(Context context, AttributeSet attrs),这样才可以让这个类在xml定义文件中使用。
还要注意到,这里用了attrs.getAttributeValue来读取XML定义文件中的属性,其实有更好的方法,容我下次再讲解,这里先偷个懒。
convertDisplayUom方法是用来将dp转换成px的,可以看到由于我们用了getAttributeValue的方式,所以需要手动将String转成Float,很麻烦。
最后就是onDraw啦,计算出画箭头的位置,画出来就行了。
接下来就是布局文件的编写了

ArrowListView在XML文件中的使用

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="fill_parent" android:layout_height="fill_parent"

  android:orientation="vertical"
  >
  <TextView
    android:text="Arrow List View Sample"
    android:layout_width="fill_parent" android:layout_height="wrap_content"

  />
  <net.learningandroid.lib.view.ArrowListView
    android:id="@+id/arrowListView"
    android:layout_width="fill_parent" android:layout_height="wrap_content"
    android:paddingTop="15dp" android:paddingBottom="20dp"

    android:layout_margin="10dp"
    android:background="@drawable/float_panel"
    android:layout_weight="1"
    android:cacheColorHint="#FFEDEDED" android:divider="#00EDEDED"

    topArrowPadding="5dp" bottomArrowPadding="10dp"
  />
</LinearLayout>

这里需要注意的是自定义控件和其中的属性的写法,不再是ListView了,而是你自己编写的控件类的类名。其它的内容就是定义padding,background,以及取消了分隔线的显示。
用这个布局文件替代上一篇教程中的布局文件,但Adapter的定义不变,因为ArrowListView是继承自ListView的,所以原先的Adapter的使用是一样的。

最后我们来看下效果:

如何?只需要小心的调整ListView的Padding和ArrowPadding的值就可以控制箭头出现的位置,如果需要控制更多,比如更换图片,或者当顶部无内容时让箭头变暗、有内容时变亮,都可以用同样的方法。
并且,如果修改了Attribute的读取方法之后,还可以通过xml文件来指定箭头的图片。

(0)

相关推荐

  • Android之IphoneTreeView带组指示器的ExpandableListView效果

    之前实现过一次这种效果的ExpandableListView:http://www.jb51.net/article/38482.htm,带效果比较挫,最近,在参考联系人源码PinnedHeaderListView,以及网上各位大侠的源码,封装了一个效果最好,而且使用最简单的IphoneTreeView,下面先看看效果图:  首先让我们看看封装得比较完善的IphoneTreeView: 复制代码 代码如下: public class IphoneTreeView extends Expandab

  • Android自定义ViewPager指示器

    本文实例为大家分享了Android ViewPager指示器的制作方法,供大家参考,具体内容如下 1.概述 ViewPageIndicator这个开源框架大家都接触过,个人感觉还不错就是用起来比较麻烦,需要这里配置那里配置效果定制起来也不方便.我第一次使用的时候就一直出不来效果,后来找了很久发现是activity的主题没有配置好.今天我们自己来造个轮子实现一把,其中用到了Adapter模式,如果不清楚这个模式的请看我的Android源码设计模式分析:http://www.jb51.net/art

  • jquery动画1.加载指示器

    该系列文章是我阅读<jQuery 1.4 Animation Techniques>后的总结,有兴趣的朋友可以去看原版书籍. 动画效果对于丰富网站的交互体验起到了不可或缺的作用,他可以让你的站点更酷,更吸引人. 什么时候使用动画: 1.当显示或隐藏页面.弹出框或内容区域的时候: 2.当一些内容从页面的一个地方移动到其他地方的时候: 3.当页面中的一些内容,应用户的操作而产生状态改变的时候: 4.当一些内容在几种状态间转变的时候: 5.引导用户去执行某些操作,或者引起他们对某些很重要信息注意的时

  • JQuery分屏指示器图片轮换效果实例

    本文实例讲述了JQuery分屏指示器图片轮换效果实现方法.分享给大家供大家参考.具体分析如下: 在Web App大行其道的今天,分屏指示器用得非常广泛,从Android.到腾讯的Web OS等等.分屏指示器给人很好的用户体验,下面就实现一个分屏指示器,用于实现图片的简单轮换效果,仅抛砖引玉- 代码如下: <script type="text/javascript"> var curr = 0, next = 0, count = 0; $(document).ready(f

  • Android中自定义控件之液位指示器

    由于安卓应用很广泛,在工业中也常有一些应用,比如可以用安卓来去工业中的一些数据进行实现的监测,显示,同时可以做一些自动化控制,当然在这里,我不是做这些自动化控制方面的研究,只是做一个控件,液位指示,其实就是继承自progressbar,然后重新写一测量与绘制,算是对自定义控件进行一下复习. 我们要做的最终就是下面这个效果: 在这里,我们要做到对这个指示器的以下属性可设置: 容器壁的厚度.容器壁的颜色.容器中液体的宽度.液体总高度.液体当前高度的颜色显示.液体未达到颜色显示.当前高度的文字指示.指

  • Android实现仿网易新闻的顶部导航指示器

    我们知道,页面导航器(Navigator)在几乎所有的项目中都会用到,平时大多数时候为了节省时间,都会直接在github上面拿别人的开源项目来用,最近自己在复习自定义View,就尝试封装了一下,源码参考项目PagerSlidingTabStrip 大家先来看一下效果图 基于文字的页面导航器 基于图片的页面导航器 使用方法 主要步骤分为三步 1)在xml文件里面 <com.xujun.viewpagertabindicator.TabPagerIndicator android:id="@+

  • Android之带group指示器的ExpandableListView(自写)

    我们都知道Android缺省的ExpandableListView的group header无法固定在界面上,当向下滚动后,不能对当前显示的那些child 指示出它们归属于哪个group,在网上搜了很多关于仿手机QQ好友分组效果的ExpandableListView,发现都不尽如意,于是乎在别人的基础上改进了一点点,其实原理还是差不多的,只是增加了往上挤出去的动画效果,而且更加简单,只不过还是没有完全到达跟QQ一样的效果,希望有高手能实现更加逼真的效果,下面我们先看看效果图:  我这里没有把Ex

  • Android应用中仿今日头条App制作ViewPager指示器

    一.概述 顶部ViewPager指示器的字体变色,该效果图是这样的: 大概是今天头条的app,神奇的地方就在于,切换ViewPager页面的时候,顶部指示器改成了字体颜色的变化,个人觉得还是不错的. 那么核心的地方就是做一个支持字体这样逐渐染色就可以了,我大概想了32s,扫描了一些可能实现的方案,最终定位了一个靠谱的,下面我就带大家开始实现的征程. 实现之前贴一下我们的效果图: 1.简单使用 效果如上图了,关于颜失色的改变我添加了两个方向,一个是左方向,一个是有方向. 单纯的使用,可能觉得没什么

  • 仅需几行代码实现方便易用的状态栏指示器

    我们在使用微博的时候经常会遇到状态指示器,想知道它是怎么做的吗?本篇文章就是通过几行代码实现方便易用的状态栏指示器. 微博项目的微博数提醒框 /** 提醒最新微博数量 */ - (void)showNewStatusCount:(NSInteger)count { if (count) { [[XZMStatusBarHUD sharedXZMStatusBarHUD] showNormal:[NSString stringWithFormat:@"有%ld条新的微博" ,count]

  • Android应用中使用ViewPager和ViewPager指示器来制作Tab标签

    一.ViewPageIndicator开源框架的基本用法 我们先得去Github上面下载这个库,下载地址:https://github.com/JakeWharton/Android-ViewPagerIndicator,下载下来之后你可以运行例子,来看看我们需要什么样的效果,然后在此基础上改成我们自己想要的效果 1.如何使用开源框架 第1步:improt library项目 第2步:导入library进我们自己新建的项目 从Github上Download下来这个zip包之后,里面会有一个lib

随机推荐