zoukankan      html  css  js  c++  java
  • Android学习系列(15)--App列表之游标ListView(索引ListView)

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

    也可以用在java后台书写:

    1. tweakedListView.setFastScrollEnabled(true);
    复制代码

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

    1. //启用FastScroller对象
    2. public void setFastScrollEnabled(boolean enabled) {
    3.     mFastScrollEnabled = enabled;
    4.     if (enabled) {
    5.         if (mFastScroller == null) {
    6.             mFastScroller = new FastScroller(getContext(), this);
    7.         }
    8.     } else {
    9.         if (mFastScroller != null) {
    10.             mFastScroller.stop();
    11.             mFastScroller = null;
    12.         }
    13.     }
    14. }
    复制代码

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

    1. /**
    2. * Cursor表示数据游标
    3. * sortedColumnIndex数据集合中的第几列
    4. * alphabet字母列表,用的最多的是"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
    5. **/
    6. public AlphabetIndexer(Cursor cursor, int sortedColumnIndex, CharSequence alphabet) {}
    复制代码

    用到3个方法:

    1. //这三个方法,实现了索引数据和列表数据的对应和定位
    2. public int getPositionForSection(int section) {}
    3. public int getSectionForPosition(int position) {}
    4. public Object[] getSections() {}
    复制代码

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

    1. /**
    2.     * 伪装一个Cursor供AlphabetIndexer作数据索引源
    3.     */
    4.    private class IndexCursor implements Cursor{
    5.        
    6.        private ListAdapter adapter;
    7.        private int position;
    8.        private Map<String, String> map;
    9.        
    10.        public IndexCursor(ListAdapter adapter){
    11.            this.adapter = adapter;
    12.        }
    13.        @Override
    14.        public int getCount() {return this.adapter.getCount();}
    15.        
    16.        /**
    17.         * 取得索引字母,这个方法非常重要,根据实际情况具体处理
    18.         */
    19.        @SuppressWarnings("unchecked")
    20.        @Override
    21.        public String getString(int columnIndex) {
    22.            map = (HashMap<String, String>)adapter.getItem(position);
    23.            return map.get(key).substring(0,1);
    24.        }
    25.        
    26.        @Override
    27.        public boolean moveToPosition(int position) {
    28.            if(position<-1||position>getCount()){
    29.                return false;
    30.            }
    31.            
    32.            this.position = position;
    33.            //如果不满意位置有点向上偏的话,下面这几行代码是修复定位索引值为顶部项值的问题
    34.            //if(position+2>getCount()){                
    35.            //    this.position = position;
    36.            //}else{
    37.            //   this.position = position + 2;
    38.            //}
    39.            return true;
    40.        }
    41.        
    42.        @Override
    43.        public void close() {}
    44.        @Override
    45.        public void copyStringToBuffer(int arg0, CharArrayBuffer arg1) {}
    46.        @Override
    47.        public void deactivate() {}
    48.        @Override
    49.        public byte[] getBlob(int arg0) {return null;}
    50.        @Override
    51.        public int getColumnCount() {return 0;}
    52.        @Override
    53.        public int getColumnIndex(String columnName) {return 0;}
    54.        @Override
    55.        public int getColumnIndexOrThrow(String columnName) throws IllegalArgumentException {return 0;}
    56.        @Override
    57.        public String getColumnName(int columnIndex) {return null;}
    58.        @Override
    59.        public String[] getColumnNames() {return null;}
    60.        @Override
    61.        public double getDouble(int columnIndex) {return 0;}
    62.        @Override
    63.        public Bundle getExtras() {return null;}
    64.        @Override
    65.        public float getFloat(int columnIndex) {return 0;}
    66.        @Override
    67.        public int getInt(int columnIndex) {return 0;}
    68.        @Override
    69.        public long getLong(int columnIndex) {return 0;}
    70.        @Override
    71.        public int getPosition() {return position;}
    72.        @Override
    73.        public short getShort(int columnIndex) {return 0;}
    74.        @Override
    75.        public boolean getWantsAllOnMoveCalls() {return false;}
    76.        @Override
    77.        public boolean isAfterLast() {return false;}
    78.        @Override
    79.        public boolean isBeforeFirst() {return false;}
    80.        @Override
    81.        public boolean isClosed() {return false;}
    82.        @Override
    83.        public boolean isFirst() {return false;}
    84.        @Override
    85.        public boolean isLast() {return false;}
    86.        @Override
    87.        public boolean isNull(int columnIndex) {return false;}
    88.        @Override
    89.        public boolean move(int offset) {return false;}
    90.        @Override
    91.        public boolean moveToFirst() {return false;}
    92.        @Override
    93.        public boolean moveToLast() {return false;}
    94.        @Override
    95.        public boolean moveToNext() {return false;}
    96.        @Override
    97.        public boolean moveToPrevious() {return false;}
    98.        @Override
    99.        public void registerContentObserver(ContentObserver observer) {}
    100.        @Override
    101.        public void registerDataSetObserver(DataSetObserver observer) {}
    102.        @Override
    103.        public boolean requery() {return false;}
    104.        @Override
    105.        public Bundle respond(Bundle extras) {return null;}
    106.        @Override
    107.        public void setNotificationUri(ContentResolver cr, Uri uri) {}
    108.        @Override
    109.        public void unregisterContentObserver(ContentObserver observer) {}
    110.        @Override
    111.        public void unregisterDataSetObserver(DataSetObserver observer) {}
    112.        
    113.    }
    复制代码
     这个类的实例就可作为AlphaIndexer的构造函数第一个参数数据游标。
    4.自定义Adapter的实现
          使用前面介绍的东西,我们来实现最终的IndexAdapter:
    1. class IndexAdapter extends SimpleAdapter implements SectionIndexer{
    2.     
    3.     private AlphabetIndexer alphabetIndexer;
    4.     
    5.     public IndexAdapter(Context context,List<? extends Map<String, ?>> data, int resource,String[] from, int[] to) {
    6.         super(context, data, resource, from, to);
    7.         //设置数据游标
    8.         //设置索引字母列表
    9.         alphabetIndexer = new AlphabetIndexer(new IndexCursor(this), 0, "ABCDEFGHIJKLMNOPQRSTUVWXYZ");
    10.     }
    11.     @Override
    12.     public Object[] getSections() {
    13.         return alphabetIndexer.getSections();
    14.     }
    15.     @Override
    16.     public int getPositionForSection(int section) {
    17.         return alphabetIndexer.getPositionForSection(section);
    18.     }
    19.     @Override
    20.     public int getSectionForPosition(int position) {
    21.         return alphabetIndexer.getSectionForPosition(position);
    22.     }
    23. }
    复制代码

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

    1. public List<Map<String, String>> getData(){
    2.     List<Map<String, String>> itemList = new ArrayList<Map<String, String>>();
    3.     String alphas = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
    4.     
    5.     Map<String, String> map = null;
    6.     for(char c:alphas.toCharArray()){
    7.         for(int i=0; i<10; i++){                
    8.             map = new HashMap<String, String>();
    9.             map.put("itemText", ""+c+i);
    10.             itemList.add(map);
    11.         }
    12.     }
    13.     return itemList;
    14. }
    复制代码

     子项的布局文件:

    1. <?xml version="1.0" encoding="utf-8"?>
    2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    3.     android:orientation="vertical"
    4.     android:layout_width="fill_parent"
    5.     android:layout_height="50dip"
    6.     android:gravity="center_vertical"
    7.     >
    8.     <TextView 
    9.         android:id="@+id/tweaked_item_text"
    10.         android:layout_width="fill_parent" 
    11.         android:layout_height="wrap_content" />
    12. </LinearLayout>
    复制代码

      使用并运行:

    1. protected void onCreate(Bundle savedInstanceState) {
    2.     super.onCreate(savedInstanceState);
    3.     setContentView(R.layout.tweake_list);
    4.     
    5.     tweakedListView = (ListView)findViewById(R.id.tweaked_list);
    6.     
    7.     //获取数据
    8.     List<Map<String, String>> itemList = getData();
    9.     ListAdapter adapter = new IndexAdapter(this, itemList, R.layout.tweake_list_item, new String[]{"itemText"}, new int[]{R.id.tweaked_item_text});
    10.     tweakedListView.setAdapter(adapter);
    11. }
    复制代码
    效果如下:
    <ignore_js_op>
    6.小结
          这种索引效果,在大数据量列表显示中非常的实用,是Android开发必备常识。
          本文只是一个简单的sample,实际工作中肯定会需要进一步扩展定义:
          (1).对于复杂类型的处理,可根据Map<String,?>扩展自定义实体类,再通过adapter转换使用即可。
          (2).对于索引字母列表,可动态设置,举个例子,你的列表只有ABCD四个字母,如果索引字母列表还是设置“ABCDEFGHIJKLMNOPQRSTUVWXYZ”就不合适了,会有个索引偏位的问题。
          (3).对于复杂界面的显示,可重写adapter的getView方法自定义视图。
  • 相关阅读:
    自定义simple_tag
    测试流程规范系列(2):测试计划
    测试流程规范系列(4):测试准入
    测试流程规范系列(3):测试用例
    测试流程规范系列(1):测试流程
    Jmeter系列(1):安装
    Jmeter系列(2):简介
    Jmeter系列(3):使用
    Jmeter系列(4):插件
    Jmeter系列(5):参数化
  • 原文地址:https://www.cnblogs.com/xiaochao1234/p/3608203.html
Copyright © 2011-2022 走看看