App列表之游标ListView(索引ListView)

游标ListView,提供索引标签,使用户能够快速定位列表项。
      也可以叫索引ListView,有的人称也为Tweaked ListView,可能更形象些吧。
      一看图啥都懂了:

1.游标(Fast scroll thumb)
      就是右边的那个拖动的方块,这个非常的简单:


1

2

3

4

5

<ListView

    android:id="@+id/tweaked_list"

    android:layout_width="fill_parent"

    android:layout_height="wrap_content"

    android:fastScrollEnabled="true"/>

  也可以用在java后台书写:


1

tweakedListView.setFastScrollEnabled(true);

  在数据量有一定大的时候,滑动列表,就会出现右边的所谓的"游标"了。
      简单,这也是我为什么私下里喜欢自己写控件,但是工作中却喜欢用通用控件。
      我们看下源代码,其实就是启用FastScroller对象:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

//启用FastScroller对象

public void setFastScrollEnabled(boolean enabled) {

    mFastScrollEnabled = enabled;

    if (enabled) {

        if (mFastScroller == null) {

            mFastScroller = new FastScroller(getContext(), this);

        }

    else {

        if (mFastScroller != null) {

            mFastScroller.stop();

            mFastScroller = null;

        }

    }

}

2.字母索引
     在Android学习系列(10)--App列表之拖拽ListView(上)中我们使用了一种WindowManager在ListView中添加一些自定义影像,这种方法我觉得一定是可行的。
   但是,android系统给我们提供了一个更简单的方法:使用AlphabetIndexer。
   AlphabetIndexer,实现了SectionIndexer接口,是adapter的一个辅助类,辅助实现在快滑时,显示索引字母。
   使用字母索引的话,必须保证数据列表是按字母顺序排序,以便AlphabetIndexerh采用二分查找法快速定位。


1

2

3

4

5

6

/**

* Cursor表示数据游标

* sortedColumnIndex数据集合中的第几列

* alphabet字母列表,用的最多的是"ABCDEFGHIJKLMNOPQRSTUVWXYZ"

**/

public AlphabetIndexer(Cursor cursor, int sortedColumnIndex, CharSequence alphabet) {}

  用到3个方法:


1

2

3

4

//这三个方法,实现了索引数据和列表数据的对应和定位

public int getPositionForSection(int section) {}

public int getSectionForPosition(int position) {}

public Object[] getSections() {}

3.游标Cursor的实现
     Cursor接口的实现,有两种选择:
     (1).直接使用数据库查询返回的cursor
     (2).自定义实现Cursor接口的新类
     第一种方式很简单,查询一下数据库返回Cursor即可。
     这里我们以第二种方式实践,伪装一个Cursor,主要是实现3个方法:
     (1).getCount()
     (2). moveToPosition()
     (3). getString()


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

/**

    * 伪装一个Cursor供AlphabetIndexer作数据索引源

    */

   private class IndexCursor implements Cursor{

       

       private ListAdapter adapter;

       private int position;

       private Map<String, String> map;

       

       public IndexCursor(ListAdapter adapter){

           this.adapter = adapter;

       }

       @Override

       public int getCount() {return this.adapter.getCount();}

       

       /**

        * 取得索引字母,这个方法非常重要,根据实际情况具体处理

        */

       @SuppressWarnings("unchecked")

       @Override

       public String getString(int columnIndex) {

           map = (HashMap<String, String>)adapter.getItem(position);

           return map.get(key).substring(0,1);

       }

       

       @Override

       public boolean moveToPosition(int position) {

           if(position<-1||position>getCount()){

               return false;

           }

           

           this.position = position;

           //如果不满意位置有点向上偏的话,下面这几行代码是修复定位索引值为顶部项值的问题

           //if(position+2>getCount()){               

           //    this.position = position;

           //}else{

           //   this.position = position + 2;

           //}

           return true;

       }

       

       @Override

       public void close() {}

       @Override

       public void copyStringToBuffer(int arg0, CharArrayBuffer arg1) {}

       @Override

       public void deactivate() {}

       @Override

       public byte[] getBlob(int arg0) {return null;}

       @Override

       public int getColumnCount() {return 0;}

       @Override

       public int getColumnIndex(String columnName) {return 0;}

       @Override

       public int getColumnIndexOrThrow(String columnName) throws IllegalArgumentException {return 0;}

       @Override

       public String getColumnName(int columnIndex) {return null;}

       @Override

       public String[] getColumnNames() {return null;}

       @Override

       public double getDouble(int columnIndex) {return 0;}

       @Override

       public Bundle getExtras() {return null;}

       @Override

       public float getFloat(int columnIndex) {return 0;}

       @Override

       public int getInt(int columnIndex) {return 0;}

       @Override

       public long getLong(int columnIndex) {return 0;}

       @Override

       public int getPosition() {return position;}

       @Override

       public short getShort(int columnIndex) {return 0;}

       @Override

       public boolean getWantsAllOnMoveCalls() {return false;}

       @Override

       public boolean isAfterLast() {return false;}

       @Override

       public boolean isBeforeFirst() {return false;}

       @Override

       public boolean isClosed() {return false;}

       @Override

       public boolean isFirst() {return false;}

       @Override

       public boolean isLast() {return false;}

       @Override

       public boolean isNull(int columnIndex) {return false;}

       @Override

       public boolean move(int offset) {return false;}

       @Override

       public boolean moveToFirst() {return false;}

       @Override

       public boolean moveToLast() {return false;}

       @Override

       public boolean moveToNext() {return false;}

       @Override

       public boolean moveToPrevious() {return false;}

       @Override

       public void registerContentObserver(ContentObserver observer) {}

       @Override

       public void registerDataSetObserver(DataSetObserver observer) {}

       @Override

       public boolean requery() {return false;}

       @Override

       public Bundle respond(Bundle extras) {return null;}

       @Override

       public void setNotificationUri(ContentResolver cr, Uri uri) {}

       @Override

       public void unregisterContentObserver(ContentObserver observer) {}

       @Override

       public void unregisterDataSetObserver(DataSetObserver observer) {}

       

   }

  这个类的实例就可作为AlphaIndexer的构造函数第一个参数数据游标。

4.自定义Adapter的实现
      使用前面介绍的东西,我们来实现最终的IndexAdapter:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

class IndexAdapter extends SimpleAdapter implements SectionIndexer{

    

    private AlphabetIndexer alphabetIndexer;

    

    public IndexAdapter(Context context,List<? extends Map<String, ?>> data, int resource,String[] from, int[] to) {

        super(context, data, resource, from, to);

        //设置数据游标

        //设置索引字母列表

        alphabetIndexer = new AlphabetIndexer(new IndexCursor(this), 0"ABCDEFGHIJKLMNOPQRSTUVWXYZ");

    }

    @Override

    public Object[] getSections() {

        return alphabetIndexer.getSections();

    }

    @Override

    public int getPositionForSection(int section) {

        return alphabetIndexer.getPositionForSection(section);

    }

    @Override

    public int getSectionForPosition(int position) {

        return alphabetIndexer.getSectionForPosition(position);

    }

}

5.跑起来
     提供样本数据如下:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

public List<Map<String, String>> getData(){

    List<Map<String, String>> itemList = new ArrayList<Map<String, String>>();

    String alphas = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";

    

    Map<String, String> map = null;

    for(char c:alphas.toCharArray()){

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

            map = new HashMap<String, String>();

            map.put("itemText"""+c+i);

            itemList.add(map);

        }

    }

    return itemList;

}

  子项的布局文件:


1

2

3

4

5

6

7

8

9

10

11

12

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    android:orientation="vertical"

    android:layout_width="fill_parent"

    android:layout_height="50dip"

    android:gravity="center_vertical"

    >

    <TextView

        android:id="@+id/tweaked_item_text"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content" />

</LinearLayout>

  使用并运行:


1

2

3

4

5

6

7

8

9

10

11

protected void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);

    setContentView(R.layout.tweake_list);

    

    tweakedListView = (ListView)findViewById(R.id.tweaked_list);

    

    //获取数据

    List<Map<String, String>> itemList = getData();

    ListAdapter adapter = new IndexAdapter(this, itemList, R.layout.tweake_list_item, new String[]{"itemText"}, new int[]{R.id.tweaked_item_text});

    tweakedListView.setAdapter(adapter);

}

  效果如下:

6.小结
      这种索引效果,在大数据量列表显示中非常的实用,是android开发必备常识。
      本文只是一个简单的sample,实际工作中肯定会需要进一步扩展定义:
      (1).对于复杂类型的处理,可根据Map<String,?>扩展自定义实体类,再通过adapter转换使用即可。
      (2).对于索引字母列表,可动态设置,举个例子,你的列表只有ABCD四个字母,如果索引字母列表还是设置“ABCDEFGHIJKLMNOPQRSTUVWXYZ”就不合适了,会有个索引偏位的问题。
      (3).对于复杂界面的显示,可重写adapter的getView方法自定义视图。

时间: 2024-10-11 23:57:37

App列表之游标ListView(索引ListView)的相关文章

Android学习系列(15)--App列表之游标ListView(索引ListView)

游标ListView,提供索引标签,使用户能够快速定位列表项.      也可以叫索引ListView,有的人称也为Tweaked ListView,可能更形象些吧.      一看图啥都懂了: 1.游标(Fast scroll thumb)      就是右边的那个拖动的方块,这个非常的简单: 1 2 3 4 5 <ListView     android:id="@+id/tweaked_list"     android:layout_width="fill_pa

Android学习系列(17)--App列表之圆角ListView(续)

http://www.cnblogs.com/qianxudetianxia/archive/2011/09/19/2068760.html 本来这篇文章想并到上篇Android学习系列(16)--App列表之圆角ListView中的,但是若是如此就让大家错过一篇新的好的文章,着实可惜.上篇中我们使用shape,corners,gradient实现了一个渐变的圆角效果,但是在完文之后的实践中,我发现有时效果不甚满意,选中和放手的事件监听没有去正确的判断,然后渐变效果也比较单一,性能也觉得不是很快

WPF自定义控件与样式(7)-列表控件DataGrid与ListView自定义样式

一.前言 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要内容: DataGrid自定义样式: ListView自定义样式: 二.DataGrid自定义样式 DataGrid是常用的数据列表显示控件,先看看实现的效果(动态图,有点大): DataGrid控件样式结构包括以下几个部分: 列头header样式 调整列头宽度的列分割线样式 行样式 行头调整高度样式 行头部样式

App列表之下拉刷新

Android的ListView是应用最广的一个组件,功能强大,扩展性灵活(不局限于ListView本身一个类),前面的文章有介绍分组,拖拽,3D立体,游标,圆角,而今天我们要介绍的是另外一个扩展ListView:下拉刷新的ListView.    下拉刷新界面最初流行于iphone应用界面,如图: 然后在Android中也逐渐被应用,比如微博,资讯类.    所以,今天要实现的结果应该也是类似的,先贴出最终完成效果,如下图,接下来我们一步一步实现. 1. 流程分析    下拉刷新最主要的流程是

[iOS基础控件 - 5.5] 代理设计模式 (基于”APP列表&quot;练习)

A.概述 在"[iOS基础控件 - 4.4] APP列表 进一步封装,初见MVC模式”上进一步改进,给“下载”按钮加上效果.功能 1.按钮点击后,显示为“已下载”,并且不可以再按 2.在屏幕中间弹出一个消息框,通知消息“xx已经被安装”,慢慢消失 3.消息框样式为圆角半透明 B.不使用代理模式,使用app空间组和主View之间的父子View关系 1.在主View中创建一个消息框 主View控制器:ViewController.m 1 // 创建下载成功消息框 2 CGFloat labelWid

四种方法解决scrollview嵌套listview,listview高度确定问题

以下文章转自@安卓泡面 在工作中,曾多次碰到ScrollView嵌套ListView的问题,网上的解决方法有很多种,但是杂而不全.我试过很多种方法,它们各有利弊. 在这里我将会从使用ScrollView嵌套ListView结构的原因.这个结构碰到的问题.几种解决方案和优缺点比较,这4个方面来为大家阐述.分析.总结. 实际上不光是ListView,其他继承自AbsListView的类也适用,包括ExpandableListView.GridView等等,为了方便说明,以下均用ListView来代表

Android ListView嵌套ListView的实现方式

首先刚到北京一个月,产品经理让做一个类似于商城的东东,起初感觉没什么难度,(不就一个电子商务app嘛,以前也做过啊),但是当看到有需求是这样的 然后就开始做,起初太懒了,就在网上找,找到了一个ListView嵌套ListView的一哥们的讲解的大致思路的,然后根据那哥们的思路自己写了一个demo,感觉效果还挺好,不卡, 第一种实现方式:这种方式有个问题就像我项目中的问题,子列中的值如果是加减变化的,对应的每个父类的item的总价格会动态变化的话用此方式就会出现一定的问题,如果不需要实现像我项目图

使用朴素贝叶斯算法,通过用户安装的APP列表来推测用户的性别

从本质上来说,这是一个分类问题,类似于通过邮件内容来推测垃圾邮件,通过用户的相关信息来推测用户是否会拖欠贷款,而通过用户的APP安装列表来推测用户的性别也是一个类似的问题. 对于贝叶斯算法来说,我们首先需要一个训练集数据,这个训练集是一个已经打好标签的数据.而要对一堆的数据打标签,人工来做是不太合适的,在这里需要结合人类与计算机各自的优势,来半自动化的识别出比较明显的有性别倾向的APP的列表,再通过这些(有性别标识的)APP在用户的安装列表中所占的比例来判定出性别特征比较明显的用户,并为这些用户

ListView嵌套ListView优化

在做业务时候,一个ListView显示多种数据类型,我们想到的方法是ListView在嵌套一个ListView,对于子ListView 占父ListView的一行,就解决了问题,但是这样的逻辑是不是有点奇怪呢? 第一,父ListView的长度size需要加1,因为增加了一行放子listView public int getCount() { return datas.size()+1; } 第二,需要判断在什么地方插入子listView ,就要用到getViewTypeCount()和getIt