ListView小知识整理:滑动背景、Item间隙等

在Android中,ListView是最常用的一个控件,在做UI设计的时候,很多人希望能够改变一下它的背景,使他能够符合整体的UI设计,改变背景很简单只需要准备一张图片,然后指定属性 android:background="@drawable/bg",不过不要高兴地太早,当你这么做以后,发现背景是变了,但是当你拖动,或者点击 list空白位置的时候发现ListItem都变成黑色的了,破坏了整体效果,如下图所示:

这是为什么呢?

这个要从Listview的效果说起,默认的ListItem背景是透明的,而ListView的背景是固定不变的,所以在滚动条滚动的过程中如果实时地 去将当前每个Item的显示内容跟背景进行混合运算,所以android系统为了优化这个过程用,就使用了一个叫做 android:cacheColorHint的属性,在黑色主题下默认的颜色值是#191919,所以就出现了刚才的画面,有一半是黑色的.

那怎么办呢?

如果你只是换背景的颜色的话,可以直接指定android:cacheColorHint为你所要的颜色,如果你是用图片做背景的话,那也只要将 android:cacheColorHint指定为透明(#00000000)就可以了,当然为了美化是要牺牲一些效率的。最后就不回出现上面所说的你 不想要的结果了!

自定义ListView行间的分割线

在Android平台中系统控件提供了灵活的自定义选项,所有基于ListView或者说AbsListView实现的widget控件均可以通过下面的方法设置行间距的分割线,分割线可以自定义颜色、或图片。

在ListView中我们使用属性 android:divider="#FF0000" 定义分隔符为红色,当然这里值可以指向一个drawable图片对象,如果使用了图片可能高度大于系统默认的像素,可以自己设置高度比如6个像素 android:dividerHeight="6px" ,当然在Java中ListView也有相关方法可以设置。

点击Item时无背景颜色变化

在xml文件中的ListView控件中加入如下属性:

 
 
  1. android:listSelector="@drawable/timer_list_selector"  

在drawable中定义timer_list_selector的属性值

timer_list_selector.xml中定义如下:

 
 
  1. <?xml version="1.0" encoding="utf-8"?> 
  2. <selector xmlns:android="http://schemas.android.com/apk/res/android"> 
  3. <item android:state_selected="true" android:drawable="@android :color/transparent" /> 
  4. </selector>  

在values文件夹下的colors.xml中定义transparent如下:

 
 
  1. <color name="transparent">#50000000</color>  

设置Item之间无间隙

在xml文件中ListView控件中加入如下属性:

 
 
  1. android:divider="#00000000"  

或者在javaCode中如下定义:

 
 
  1. listView.setDividerHeight(0);  

自定义的BaseAdapter中调用notifyDataSetChanged()方法会重新调用BaseAdapter的getView()方法。

listview中在设置了背景之后,会有些问题:

1.、listview在拖动的时候背景图片消失变成黑色背景。等到拖动完毕我们自己的背景图片才显示出来。

2 、listview的上边和下边有黑色的阴影。

3、lsitview的每一项之间需要设置一个图片做为间隔。

针对以上问题 在listview的xml文件中设置一下语句。

问题1 有如下代码结解决 android:scrollingCache="false"

问题2 用如下代码解决:android:fadingEdge="none" 

问题3 用如下代码解决: android:divider="@drawable/list_driver" 其中 @drawable/list_driver 是一个图片资源

总体如下:

 
 
  1. <ListView 
  2. android:id="@+id/myListView01" 
  3. android:layout_width="fill_parent" 
  4. android:layout_height="287dip" 
  5. android:fadingEdge="none"  
  6. android:scrollingCache="false" 
  7. android:divider="@drawable/list_driver" 
  8. android:background="@drawable/list"> 
  9. </ListView>  

____________________________________________________________________________________________________________________________________________________

android ListView 的scrollState

1.  SCROLL_STATE_IDLE 滑动后静止

2.  SCROLL_STATE_FLING 手指离开屏幕后,惯性滑动

3.  SCROLL_STATE_TOUCH_SCROLL 手指在屏幕上滑动


一般执行顺序为:3---》2---》1


问题:ListView接收到MotionEvent.ACTION_CANCEL事件时,有可能导致无法触发SCROLL_STATE_IDLE状态。

解决方法:主动调用onScrollStateChanged触发SCROLL_STATE_IDLE状态


____________________________________________________________________________________________________________________________________________________

精确监听AbsListView滚动至底部


  用户使用android客户端时,当ListView滚动至底部,可以由一个按钮来提示用户是否读下一页,那么如果使用GridView呢?现在很多WEB 2.0上的体验就是当底部时自动读取下一页数据,GridView(ListView也可)可以采用这种方法。网上已经有很多文章介绍了如何判断ListView是否滚动至底部,原理是 AbsListView.getLastVisiblePosition() = (AbsListView.getCount() - 1) 即到底,如果往上拖一点,用户看起来已经离开底部,但实际上 AbsListView.getLastVisiblePosition() == (AbsListView.getCount() - 1) 依然成立,会导致误判断。本文在它们基础上加以改进,做到更精确地监听是否滚动至底部。先来看看本文程序运行的效果:

 

本文在 Android入门第八篇之GridView(九宫图)基础上加入滚到事件判断,文件名为AutoLoadListener.java,原理是在AbsListView.getLastVisiblePosition() = =(AbsListView.getCount() - 1) 时,保存最后一个Item的绝对坐标,如果两次获取的绝对Y值都一样,即到底然后执行回调函数......源码如下:

  1. <span style="font-family:Comic Sans MS;">package com.testScroll;  
  2.   
  3. import android.util.Log;  
  4. import android.view.View;  
  5. import android.widget.AbsListView;  
  6. import android.widget.Toast;  
  7. import android.widget.AbsListView.OnScrollListener;  
  8. /** 
  9.  * 滚动至列表底部,读取下一页数据 
  10.  */  
  11. public class AutoLoadListener implements OnScrollListener{  
  12.   
  13.     public interface AutoLoadCallBack {  
  14.         void execute(String url);  
  15.     }  
  16.   
  17.     private int getLastVisiblePosition = 0,lastVisiblePositionY=0;  
  18.     private AutoLoadCallBack  mCallback;  
  19.     public AutoLoadListener(AutoLoadCallBack callback)  
  20.     {  
  21.         this.mCallback = callback;  
  22.     }  
  23.       
  24.     public void onScrollStateChanged(AbsListView view, int scrollState) {  
  25.   
  26.         if (scrollState == OnScrollListener.SCROLL_STATE_IDLE) {  
  27.             //滚动到底部  
  28.             if (view.getLastVisiblePosition() == (view.getCount() - 1)) {  
  29.                 View v=(View) view.getChildAt(view.getChildCount()-1);  
  30.                 int[] location = new  int[2] ;  
  31.                 v.getLocationOnScreen(location);//获取在整个屏幕内的绝对坐标  
  32.                 int y=location [1];  
  33.   
  34.                 Log.e("x"+location[0],"y"+location[1]);  
  35.                 if (view.getLastVisiblePosition()!=getLastVisiblePosition  
  36.                         && lastVisiblePositionY!=y)//第一次拖至底部  
  37.                 {  
  38.                     Toast.makeText(view.getContext(), "再次拖至底部,即可翻页",500).show();  
  39.                     getLastVisiblePosition=view.getLastVisiblePosition();  
  40.                     lastVisiblePositionY=y;  
  41.                     return;  
  42.                 }  
  43.                 else if (view.getLastVisiblePosition()==getLastVisiblePosition  
  44.                         && lastVisiblePositionY==y)//第二次拖至底部  
  45.                 {  
  46.                     mCallback.execute(">>>>>拖至底部");  
  47.                 }  
  48.             }  
  49.               
  50.             //未滚动到底部,第二次拖至底部都初始化  
  51.             getLastVisiblePosition=0;     
  52.             lastVisiblePositionY=0;  
  53.         }  
  54.     }  
  55.   
  56.     public void onScroll(AbsListView arg0, int arg1, int arg2, int arg3) {  
  57.           
  58.     }  
  59. }  
  60. </span>  

主程序为testScroll.java,源码如下:

  1. <span style="font-family:Comic Sans MS;">package com.testScroll;  
  2.   
  3. import java.util.ArrayList;  
  4. import java.util.HashMap;  
  5.   
  6. import com.testScroll.AutoLoadListener.AutoLoadCallBack;  
  7.   
  8.   
  9. import android.app.Activity;  
  10. import android.os.Bundle;  
  11. import android.view.View;  
  12. import android.widget.AdapterView;  
  13. import android.widget.AdapterView.OnItemClickListener;  
  14. import android.widget.GridView;  
  15. import android.widget.SimpleAdapter;  
  16. import android.widget.Toast;  
  17.   
  18. public class testScroll extends Activity {  
  19.     /** Called when the activity is first created. */  
  20.     @Override  
  21.     public void onCreate(Bundle savedInstanceState) {  
  22.         super.onCreate(savedInstanceState);  
  23.         setContentView(R.layout.main);  
  24.         setTitle("精确监听AbsListView滚动至底部----hellogv");  
  25.         GridView gridview = (GridView) findViewById(R.id.gridview);  
  26.   
  27.         // 生成动态数组,并且转入数据  
  28.         ArrayList<HashMap<String, Object>> lstImageItem = new ArrayList<HashMap<String, Object>>();  
  29.         for (int i = 0; i < 30; i++) {  
  30.             HashMap<String, Object> map = new HashMap<String, Object>();  
  31.             map.put("ItemImage", R.drawable.icon);// 添加图像资源的ID  
  32.             map.put("ItemText""NO." + String.valueOf(i));// 按序号做ItemText  
  33.             lstImageItem.add(map);  
  34.         }  
  35.         // 生成适配器的ImageItem <====> 动态数组的元素,两者一一对应  
  36.         SimpleAdapter saImageItems = new SimpleAdapter(this// 没什么解释  
  37.                 lstImageItem,// 数据来源  
  38.                 R.layout.night_item,// night_item的XML实现  
  39.   
  40.                 // 动态数组与ImageItem对应的子项  
  41.                 new String[] { "ItemImage""ItemText" },  
  42.   
  43.                 // ImageItem的XML文件里面的一个ImageView,两个TextView ID  
  44.                 new int[] { R.id.ItemImage, R.id.ItemText });  
  45.            
  46.         //添加自动读页的事件  
  47.         AutoLoadListener autoLoadListener =new AutoLoadListener(callBack);  
  48.         gridview.setOnScrollListener(autoLoadListener);  
  49.         // 添加并且显示  
  50.         gridview.setAdapter(saImageItems);  
  51.         // 添加消息处理  
  52.         gridview.setOnItemClickListener(new ItemClickListener());  
  53.     }  
  54.   
  55.     AutoLoadCallBack callBack=new AutoLoadCallBack(){  
  56.   
  57.         public void execute(String url) {  
  58.             Toast.makeText(testScroll.this, url, 500).show();  
  59.         }  
  60.           
  61.     };  
  62.     // 当AdapterView被单击(触摸屏或者键盘),则返回的Item单击事件  
  63.     class ItemClickListener implements OnItemClickListener {  
  64.         public void onItemClick(AdapterView<?> arg0,// The AdapterView where the  
  65.                                                     // click happened  
  66.                 View arg1,// The view within the AdapterView that was clicked  
  67.                 int arg2,// The position of the view in the adapter  
  68.                 long arg3// The row id of the item that was clicked  
  69.         ) {  
  70.             // 在本例中arg2=arg3  
  71.             HashMap<String, Object> item = (HashMap<String, Object>) arg0  
  72.                     .getItemAtPosition(arg2);  
  73.             // 显示所选Item的ItemText  
  74.             setTitle((String) item.get("ItemText"));  
  75.         }  
  76.   
  77.     }  
  78. }  
  79. </span> 






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值