Android中的GridView详解

GridView的学习

我们将通过两个例子学习GridView。Grid和Table有一点点类似。我们将在例子中逐步描绘如何编写一个Grid的Activity

例子一:继承ArrayAdapter作为自定义adapter

1、编写Android XML

<?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="fill_parent">
  <TextView android:id="@+id/selection4"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content" />

  <GridView android:id="@+id/grid"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"

    android:verticalSpacing="35px" <!-- grid元素之间的竖直间隔 -->
    android:horizontalSpacing="5px"  <!--grid元素之间的水平间隔  -->
    android:numColumns="auto_fit" <!--表示有多少列,如果设置为auto_fit,将根据columnWidth和Spacing来自动计算 -->
    android:columnWidth="100px" <!-- 一般建议采用有像素密度无关的dip或者dp来表示-->
    android:stretchMode="columnWidth" <!--如何填满空余的位置,模拟器采用WVGA800*480,每排4列,有4*100+5*3=415,还余65px的空间,如果是 columnWidth,则这剩余的65将分摊给4列,每列增加16/17px。如果采用SpacingWidth,则分摊给3个间隔空隙 -->
    android:gravity="center"  />  
</LinearLayout>

2、编写代码。和其他selected widget,我们之前学习的ListView和Spinner的方式,通过setAdapter()来提供数据和子View显示风格,通过触发 setOnItemSelectedListener()注册选择listner。在这里处理选择之外,我们增加一个Click的触发,可以比较一下此两 的差异。此次我们不再使用Android自带的格式,而设置我们自己的UI风格。

public class Chapter7Test4 extends Activity  implements  OnItemSelectedListener,OnItemClickListener{
    private TextView selection = null;
    private String[] items={"lorem", "ipsum", "dolor", "sit", "amet", "hello", "me", "elit", "morbi", "vel", "ligula", "vitae", "arcu", "aliquet", "mollis", "etiam", "vel", "erat", "placerat", "ante","hi", "sodales", "test", "augue", "purus"};


    @Override

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.chapter_7_test4);
        

        selection = (TextView)findViewById(R.id.selection4);
        GridView grid= (GridView)findViewById(R.id.grid);
        //步骤1:设置ArrayAdapter,可以采用android自带的格式,也可以自定义,这里我们将自己定义。
        grid.setAdapter( new FunnyLookingAdapter(this, android.R.layout.simple_list_item_1,items));
        //grid.setAdapter(new ArrayAdapter (this,android.R.layout.simple_list_item_1,items)); 
        //步骤2:设置元素被选择以及被点击的回调触发处理
        grid.setOnItemSelectedListener(this);
        grid.setOnItemClickListener(this);
    }
  //步骤3:编写CallBack触发函数
    @Override /* 这是OnItemSelectedListener接口*/
    public void onItemSelected(AdapterView arg0, View arg1, int arg2, long arg3) {
        selection.setText(items[arg2]);
    }

    @Override /* 这是OnItemSelectedListener接口*/
    public void onNothingSelected(AdapterView arg0) {
        selection.setText("");
    }

    @Override /* 这是OnItemClickListener接口*/
    public void onItemClick(AdapterView arg0, View arg1, int arg2, long arg3) {
        selection.setText("Clicked: " + items[arg2]);
    }

   //步骤4:编写自定义的adapter,继承ArrayAdapter 
    private class FunnyLookingAdapter extends ArrayAdapter {
        private Context context;
        private String[] theItems;

        //步骤4.1:编写adapter的构造函数
        FunnyLookingAdapter( Context context, int resource, String[] items){
                super(context,resource,items);
                this.context = context;
                theItems = items;
        }
        
        //步骤4.2:重写getView(),对每个单元的内容以及UI格式进行描述
        /*如果我们不使用TextView,则我们必须通过getView()对每一个gridview单元进行描述。这些单元可以是Button,ImageView,在这里我们使用Button和TextView分别作测试 重override getView(int, View, ViewGroup),返回任何我们所希望的view。*/
        public View   getView  (int position, View  convertView, ViewGroup  parent){
            TextView label = (TextView)convertView;
            //我们测试发现,除第一个convertView外,其余的都是NULL,因此如果没有view,我们需要创建
            if(convertView == null){
                convertView = new TextView(context);
                label = (TextView)convertView;
            }
            
            label.setText(position + ": "  + theItems[position]);            
            return convertView;
        }
    }// End of class FunnyLookingAdapter

}

左图是使用android自带的粗体格式,即被注释掉的setAdapter,图二为例子源代码示例,图右将 FunnyLookingAdapter中的getView()用Button代替TextView,这是发现有一个有趣的现象,Button是检测 Click动作,而我们在类中通过setItemClickListener中设置了对GridView中的item检测Click的动作,这两个是重叠 的,而将优先监听Button的Click,即我们定制的GridView中将得不到触发,这需特别注意。

例子二:继承BaseAdapter作为自定义adapter

这是来自Totorial的例子。在这里GridView里面的元素是ImageView。Android XML的文件很简单:



<GridView xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/gridview"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent"

  android:columnWidth="90dp"
  android:numColumns="auto_fit"
  android:verticalSpacing="10dp"
  android:horizontalSpacing="10dp"
  android:stretchMode="columnWidth"
  android:gravity="center" >

我们在来看看Java code。

1)我们将图片,copy至res/drawable-hdpi/中。

2)设置GridView的adapter,并设置点击触发函数,点击后采用Toast显示点击的序号。

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.chapter_7_test5);

        
        GridView gridview = (GridView)findViewById(R.id.gridview);
        gridview.setAdapter(new ImageAdapter(this));
        
        gridview.setOnItemClickListener(new OnItemClickListener(){
            public void onItemClick(AdapterView parent, View v, int position, long id){
                Toast.makeText(this, ""+position, Toast.LENGTH_SHORT).show();
            }    
        });
    }

3)设置我们自子的ImageAdapter,继承BasAdapter。

    public class ImageAdapter extends BaseAdapter{
        private Context context = null;
        // references to our images
        private Integer[] mThumbIds = {R.drawable.sample_2, R.drawable.sample_3,R.drawable.sample_4, R.drawable.sample_5,R.drawable.sample_6,...// 这里是文件的对应的Id,不在具体列出};
        //步骤1: 构造函数       
        public ImageAdapter(Context context){
            this.context = context;
        }
        //步骤2:BaseAdapter需要重构四个方法getCount(),getItem(),getItemId(int position),getView()
        //步骤2.1:getCount() 表示How many items are in the data set represented by this Adapter.
        public int getCount() {
            return mThumbIds.length;
        }
        //步骤2.2:getItem()根据需要position获得布放在GridView的对象。在这个例子中,我们不需要处理里面的对象,可以设为null
        public Object getItem(int position) {
            return null;
        }
        //步骤2.3:getItemId()获得row id(Get the row id associated with the specified position in the list),由于我们也不需要,简单的设为0
        public long getItemId(int position) {
            return 0;
        }
        //步骤2.4:获得GridView里面的ViewGet a View that displays the data at the specified position in the data set. 和第一个例子一样,传递的第二个函数可能为null,必须进行处理。
        public View getView(int position, View convertView, ViewGroup parent) {
            ImageView imageView = null;
            if(convertView == null){
                imageView = new ImageView(context);
                // 设置View的height和width:这样保证无论image原来的尺寸,每个图像将重新适合这个指定的尺寸。
                imageView.setLayoutParams(new GridView.LayoutParams(85,85));
                /* ImageView.ScaleType.CENTER 但不执行缩放比例
                 * ImageView.ScaleType.CENTER_CROP 按比例统一缩放图片(保持图片的尺寸比例)便于图片的两维(宽度和高度)等于或大于相应的视图维度
                 * ImageView.ScaleType.CENTER_INSIDE 按比例统一缩放图片(保持图片的尺寸比例)便于图片的两维(宽度和高度)等于或小于相应的视图维度 */

                imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
                imageView.setPadding(8,8,8,8);
            }else{
                imageView = (ImageView)convertView;
            }
            imageView.setImageResource(mThumbIds[position]);
            return imageView;
        }
        
    }

我们设置了几种scaleType,下面左图是ImageView.ScaleType.CENTER_CROP,中图是ImageView.ScaleType.CENTER_INSIDE,右图是ImageView.ScaleType.CENTER

Android中的GridView详解_第1张图片




1.GridView实例展示一

本文主要实现如下功能:登陆界面经过通信线程鉴权后返回各种权限,权限对应各种功能模块,打算将各种功能模块以GridView方式显示给用户,GridView中放置各种功能图标,如果权限不足则置灰。

 

界面参考如下图片(软件来自电信内部掌上办公平台):

 

登陆界面在这里不再赘述,本文主要关心以面向对象的方式来构建GridView,以及自定义ListAdapter的使用来个性化GridView中Item的显示方式。

 

首先,本Activity是一个fullparent的GridView,其XML代码没什么技术含量,拷贝如下:

 

[xhtml]  view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.  android:id="@+id/main_table_layout" android:layout_height="fill_parent"  
  4.  android:orientation="vertical" android:layout_width="fill_parent"  
  5.  android:background="@drawable/background">  
  6. <GridView xmlns:android="http://schemas.android.com/apk/res/android"  
  7.   android:id="@+id/grid_view"  
  8.   android:layout_width="fill_parent"  
  9.   android:layout_height="fill_parent"  
  10.   android:numColumns="auto_fit"  
  11.   android:horizontalSpacing="20dp"  
  12.   android:verticalSpacing="0dp"  
  13.   android:columnWidth="50dp"  
  14.   android:stretchMode="columnWidth"  
  15.   android:gravity="center"></GridView>  
  16. </LinearLayout>  
  17.   
  18. <!-- android:numColumns="auto_fit" ,GridView的列数设置为自动 -->  
  19. <!-- android:columnWidth="90dp",每列的宽度,也就是Item的宽度 -->  
  20. <!-- android:stretchMode="columnWidth",缩放与列宽大小同步 -->  
  21. <!-- android:verticalSpacing="10dp",两行之间的边距,如:行一(NO.0~NO.2)与行二(NO.3~NO.5)间距为10dp -->  
  22. <!-- android:horizontalSpacing="10dp",两列之间的边距。  -->  

 

接下来要设计GridView中的元素,也就是本例中的功能按钮,每个按钮的图标不同,显示的名字也不同,最重要的是其onClick事件也不同。很多朋友处理这类问题的时候会编写大量的if和switch语句,我一贯的宗旨是尽量少用甚至不用switch分支,以面向对象的方式提高代码的可复用、可扩展、可读等性能。

 

闲话少说,开始设计GridView中的Item样式,编写元素布局文件menu_item.xml:

 

[c-sharp]  view plain copy
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     android:layout_width="fill_parent" android:layout_height="fill_parent">  
  3.   
  4.     <ImageView android:layout_width="wrap_content" android:id="@+id/item_image"  
  5.         android:layout_height="wrap_content" android:layout_centerHorizontal="true"/>  
  6.   
  7.     <TextView android:layout_width="wrap_content"  
  8.         android:layout_height="wrap_content" android:layout_below="@id/item_image"  
  9.         android:id="@+id/item_text" android:layout_centerHorizontal="true"  
  10.         android:textColor="#000000" android:textStyle="bold"/>  
  11.   
  12. </RelativeLayout>   

可以看到,很简单的一个View组合,上面是一个居中摆放的ImageView,下面是一个居中摆放的TextView,这就是功能按钮的通用布局,至于每个按钮的具体设计,这边还不用考虑。

 

下面进行面向对象的分析,构造Function抽象父类,将所有的功能按钮都看成是一个Function的一个子类,不同的功能按钮继承Function,在各自的构造函数中对该功能的图标,文字进行定义,与此同时实现该功能的点击事件处理。

 

Function父类的设计如下:

[java]  view plain copy
  1. public abstract class Function {  
  2.     protected int functionImageRscId;  
  3.     protected int functionTextRscId;  
  4.     protected Context context = null;  
  5.   
  6.     public Function(Context context) {  
  7.         // TODO Auto-generated constructor stub  
  8.         this.context = context; // 获取上下文  
  9.     }  
  10.   
  11.     public abstract OnClickListener getOnClickListener();   
  12. // 抽象函数,强制子类去继承它,并实现各自的功能  
  13.   
  14.     public int getFunctionImageRscId() {  
  15.         return functionImageRscId;  
  16.     }  
  17.   
  18.     public int getFunctionTextRscId() {  
  19.         return functionTextRscId;  
  20.     }  
  21. }  

初步实现一个功能按钮,我这边是一个叫“释放号码”的功能,代码如下:

[java]  view plain copy
  1. public class FreeNumberFunction extends Function {  
  2.   
  3.     public FreeNumberFunction(Context context) {  
  4.         super(context);  
  5.         // TODO Auto-generated constructor stub  
  6.         this.functionImageRscId=R.drawable.free_number_function_a;//本功能的图标资源  
  7.         this.functionTextRscId=R.string.free_number_function;//功能名字的字符串资源  
  8.     }  
  9.   
  10.     public OnClickListener getOnClickListener() {  
  11.         // TODO Auto-generated method stub  
  12.         return new OnClickListener() {  //测试的响应事件         
  13.             public void onClick(View v) {  
  14.                 new AlertDialog.Builder(context)  
  15.                 .setTitle("释放号码")  
  16.                 .setMessage("打开释放号码的界面")  
  17.                 .show();                  
  18.             }  
  19.         };  
  20.     }     
  21. }  

在实现一个功能按钮,用做比较,名叫“经营报表”的功能按钮,代码如下:

[c-sharp]  view plain copy
  1. public class ReportFunction extends Function {  
  2.   
  3.     public ReportFunction(Context context) {  
  4.         super(context);  
  5.         this.functionImageRscId=R.drawable.report_function_a;  
  6.         this.functionTextRscId=R.string.report_function;  
  7.     }  
  8.   
  9.     @Override  
  10.     public OnClickListener getOnClickListener() {  
  11.         // TODO Auto-generated method stub  
  12.         return new OnClickListener() {            
  13.             public void onClick(View v) {  
  14.                 new AlertDialog.Builder(context)  
  15.                 .setTitle("经营分析")  
  16.                 .setMessage("MessageTest")  
  17.                 .show();                  
  18.             }  
  19.         };  
  20.     }  
  21.   
  22. }  

这样就基本以面向对象的方式,设计好功能按钮的框架了,以后如果项目需要新的功能添加,则仅需添加一个类,继承Function父类,做好相关的png和功能名资源,设置好onClick事件就OK了,避免了大量的switch语句去响应GridView的那个什么onSelectItem事件,还要在里面提取各种序列,应用程序改变功能按钮显示的顺序都变得很麻烦。

有了这样的结构,下面一部就是让GridView认识咱们的Function,这里需要用到自定义Adapter适配器,GridView需要一个ListAdapter来解析子元素,系统提供一个自带的SimpleListAdapter,很是不好用,仅可以传输有限的数据和统一化的显示方法,这里需要让GridView认识Function就必须自己写适配器,适配器继承一个叫BaseAdaper的类即可,适配器名为FunctionsAdapter,代码如下:

[java]  view plain copy
  1. public class FunctionsAdapter extends BaseAdapter {  
  2.     private Context context = null;// 上下文  
  3.     private ArrayList<Function> list = null;// 数据源  
  4.   
  5.     private ImageView functionImage = null;  
  6.     private TextView functionText = null;  
  7.   
  8.     // 适配器构造函数  
  9.     public FunctionsAdapter(Context c, ArrayList<Function> list) {  
  10.         this.context = c;//c是上下文,在UI编程中,一般这个参数都是必要的  
  11.         this.list = list;//list中是一个Function数组,存放了所有要显示的Function  
  12.     }  
  13.   
  14.     // 下面三个是实现抽象函数,可以无视  
  15.     public int getCount() {  
  16.         return list.size();  
  17.     }  
  18.     public Object getItem(int position) {  
  19.         return list.get(position);  
  20.     }  
  21.     public long getItemId(int position) {  
  22.         return position;  
  23.     }  
  24.   
  25.     // 根据参数,个性化自己的View  
  26.     public View getView(int position, View convertView, ViewGroup parent) {  
  27.         convertView = LayoutInflater.from(context).inflate(R.layout.menu_item,  
  28.                 null);// 通过上下文获取一开始定义的menu_item布局,以这个布局文件为样式造出一个自定义View  
  29.   
  30.         functionImage = (ImageView) convertView.findViewById(R.id.item_image);//获取布局文件里的ImageView  
  31.         functionText = (TextView) convertView.findViewById(R.id.item_text);//获取布局文件里的TextView  
  32.   
  33.         functionImage.setImageResource(list.get(position)  
  34.                 .getFunctionImageRscId());//对功能按钮的图标进行赋值  
  35.         functionText.setText(list.get(position).getFunctionTextRscId());//对功能按钮的名字进行赋值  
  36.   
  37.         convertView.setOnClickListener(list.get(position).getOnClickListener());  
  38.         //响应不同Function的onClick事件,个人认为这行代码最精妙,仅仅一行代码省去了多少switch和case  
  39.   
  40.         return convertView;//最终返回一个View,这一个View就是一整个功能按钮,而且是个性化的功能按钮  
  41.     }  
  42. }  

编写自定义适配器没有想象中恐怖,继承BaseAdapter后,Eclipse会自动要求你实现这么几个函数,首先是构造函数,这个你只要传个上下文context和数据源过来就行,context就是上下文,这个在UI编程中一直要传递,Activity类的this对象就行了,数据源我们这里是一个Function类型的数组,包含了所有要在界面上显示的Function,当然你在构造的时候都必须用Function的子类来构造。除了构造函数以外,getCount、getItem、getItemId这三个函数也要你重写,看我的代码就知道纯粹是划水性质的,不再赘述,最重要的是一个getView的函数,这个函数将返回一个View给GridView作为Item使用,相当于迭代生成所有的GridView中要显示的元素,注释写的很详细了,大家看着应该能明白个中含义。

最终在Activity中生成整个GridView的代码,如下:

[java]  view plain copy
  1. public class MainTable extends Activity {  
  2.   
  3.     @Override  
  4.     protected void onCreate(Bundle savedInstanceState) {  
  5.         // TODO Auto-generated method stub  
  6.         super.onCreate(savedInstanceState);  
  7.         this.requestWindowFeature(Window.FEATURE_NO_TITLE);//无title  
  8.         this.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_NOSENSOR);// 禁止竖屏  
  9.         setContentView(R.layout.main_table);  
  10.         super.onCreate(savedInstanceState);  
  11.   
  12.         GridView gridview = (GridView) findViewById(R.id.grid_view);  
  13.           
  14.         ArrayList<Function> meumList=new ArrayList<Function>();  
  15.         meumList.add(new FreeNumberFunction(this));  
  16.         meumList.add(new ReportFunction(this));  
  17.           
  18.         FunctionsAdapter fa=new FunctionsAdapter(this,meumList);  
  19.         gridview.setAdapter(fa);  
  20.   
  21.     }  
  22. }  

显示界面如以下个图所示:

Android中的GridView详解_第2张图片    Android中的GridView详解_第3张图片    

 

 2.有趣味的GridView


工作这么久以来,都是以解决需求为目标。渐渐发现这种学习方式不好,学到的知识能马上解决问题,但没有经过梳理归纳。故想系统总结下一些有趣味的知识点。在这篇博客中想以一个例子系统讲解下GridView控件涉及到的方方面面,比如监听,背景图的设置等。

  1.控件属性介绍

     android:listSelector="@drawable/bg"

        //该属性很重要,如果不设置的话,GridView控件会自带自己的选中样式(黄色边框),如图所示:

      Android中的GridView详解_第4张图片

      做项目一般是不需要这个的。所以解决办法是将GridView的android:listSelector属性设置为和Activity背景相同即可。
    android:verticalSpacing="10dp"  //行宽
    android:horizontalSpacing="10dip"  //列宽
    android:numColumns="3"  //列数

    android:stretchMode="columnWidth"  //列可扩展
      以上列数为3,若想在高分辨率中列数设置为4怎么办呢。

    首先,需要在属性中设置列为可扩展android:stretchMode="columnWidth"。然后在代码中做个判断:

    if(Metrics.heightPixels == 1280 && Metrics.widthPixels == 800){
               gridview.setNumColumns(4);
          }这样GridView的列数是可以随分辨率不同而做更多的设置。

  2.例子一:设置个边框,并且右上角设置个CheckBox来美化选中状态(左图→右图)。

               Android中的GridView详解_第5张图片→→→Android中的GridView详解_第6张图片

   这个Item的xml设计中,我是用两个ImageView控件来实现的.

   第一个ImageView 要设置一个属性:android:layout_margin="4dip"。上下左右留出4dip的具体来显示边框的效果。并且在相应的Adapter中做个判断,选中的话背景图片颜色为#87CEFA,未选中时为3FFFFFF:

  if(bSelect){ //bSelect表示是否选中
        arg1.setBackgroundColor(Color.parseColor("#87CEFA"));
    }else{
        arg1.setBackgroundColor(Color.parseColor("#FFFFFF"));

  右上角的看起来像复选框,其实我也是用ImageView控件来实现的。界面设计很重要,距离要对好。我设置的该控件离上边距和右边距分别为2dip的距离。同样的只需要修改该ImageView所绑定的图片就可以。

  if(bSelect){
     viewclass.checkboxImage.setImageBitmap(bmpwater_sel); //bmpwater_sel表示左图右上角的图片
    }else{
     viewclass.checkboxImage.setImageBitmap(bmpwater_unsel); //bmpwater_unsel表示右图右上角的图片
      }

  很简单吧。你们可以试试。

  3.例子二:打开一个GridView控件的Activity时,如果打开的图片过多,如何让图片异步加载。如图所示:

    Android中的GridView详解_第7张图片  → Android中的GridView详解_第8张图片

  首先在界面定义中要定义一个全屏显示的加载圈效果,叫做mLoading。程序刚启动时显示图片加载结束后调用函数mLoading.setVisibility(View.GONE);让它不显示。我是用一个函数来调用它,这样更好的体现了封装性,如下:

   private void showLoading(boolean bShow){
      if(mLoading == null) return;
      mLoading.setVisibility(bShow ? View.VISIBLE : View.GONE);
   }

  xml中该加载框的设计如下:  

1
2
3
4
5
6
7
8
9
10
11
12
13
<span style= "font-size: 13px;" ><RelativeLayout android:id= "@+id/video_tip_layout"
     android:layout_width= "fill_parent"  android:layout_height= "fill_parent" >
<ProgressBar style= "?android:attr/progressBarStyleLarge"
     android:layout_width= "wrap_content"  android:id= "@+id/video_tip_progressbar"
     android:layout_height= "wrap_content"  android:layout_centerVertical= "true"
     android:layout_centerHorizontal= "true"  />
<TextView android:textAppearance= "?android:attr/textAppearanceMedium"
     android:id= "@+id/video_tip_text"  android:text= "加载中..."
     android:layout_width= "wrap_content"  android:layout_height= "wrap_content"
     android:layout_below= "@+id/video_tip_progressbar"
     android:textColor= "#808080"
     android:layout_centerHorizontal= "true" ></TextView>
</span>

  ①为了不卡主线程,实现这一的效果要多开一个线程。在该线程中调用函数InitList(),在该函数中将图片加载到ArrayList中去(花时间最多的就是对多图片的处理,这个放在子线程中可以不卡主线程,如果直接在主线程中做,会让画面出现2-3秒的黑屏停顿)。

  像这样,在onCreate方法中开一个子线程:

  new Thread(new Runnable() {
     @Override
     public void run() {
        InitList();
     }
    }).start();

  ②方法InitList()中要做两件事情。一是加载图片,二是通知Handler图片加载完成(因为子线程不可以操作UI控件,只能通过Handler的方式)。

  private void InitList()
  {

    XXXX; //加载图片代码
      mMainHandler.sendMessage(MainHandler.MSG_FINISH); //发送消息
    }

  ③当收到InitList发送的消息时就让上面所提的mLoading不显示。同时GridView绑定adapter的代码也在这个里面写。因为图片加载完成后再绑定adapter,这样图片才能全部显示,否则只能显示一部分。

     case MSG_FINISH:
        showLoading(false); //让加载框不再显示
        gridview.setAdapter(ia); //绑定数据
       break;
     default:
       break;


 

3.GridView中的GridAdapter

说到GridView Listview一定会提到他的adapter,现整理如下:

main.xml布局里面:

[html]  view plain copy
  1. <GridView  
  2.     android:id="@+id/grid"  
  3.     android:layout_width="fill_parent"  
  4.     android:layout_height="fill_parent"  
  5.     android:columnWidth="70dp"  
  6.     android:numColumns="auto_fit"   
  7.     android:verticalSpacing="10dp"   
  8.     android:horizontalSpacing="10dp"  
  9.     android:stretchMode="columnWidth"  
  10.     android:gravity="center"  
  11.     />  

gridview_item.xml:

[html]  view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:layout_height="wrap_content"   
  4.     android:layout_width="wrap_content"  
  5.     android:orientation="vertical"  
  6.     android:gravity="center"  
  7.     android:layout_gravity="center"  
  8.     >  
  9.     <LinearLayout   
  10.         android:layout_height="wrap_content"   
  11.         android:layout_width="wrap_content"  
  12.         android:orientation="vertical"  
  13.         android:gravity="center"  
  14.         android:layout_gravity="center">  
  15.         <ImageView   
  16.             android:id="@+id/grid_icon"  
  17.             android:layout_height="80dip"   
  18.             android:layout_width="80dip"  
  19.             android:gravity="center"  
  20.             android:layout_gravity="center"   
  21.             >  
  22.         </ImageView>  
  23.         <TextView   
  24.             android:id="@+id/grid_name"  
  25.             android:layout_width="wrap_content"  
  26.             android:layout_height="wrap_content"  
  27.             android:singleLine="true"  
  28.             android:gravity="center"  
  29.             android:layout_gravity="center"  
  30.             android:textSize="13px"  
  31.             android:layout_marginTop="4dip"  
  32.             >  
  33.         </TextView>  
  34.     </LinearLayout>   
  35. </FrameLayout>    

java 代码:

[java]  view plain copy
  1. GridView grid = (GridView) findViewById(R.id.grid);  
  2. grid.setAdapter(new GridAdapter(this, mNameList, mDrawableList));  

GridAdapter.java类:

[java]  view plain copy
  1. package com.android.test;  
  2.   
  3. import java.util.ArrayList;  
  4.   
  5. import android.content.Context;  
  6. import android.graphics.drawable.Drawable;  
  7. import android.view.Gravity;  
  8. import android.view.LayoutInflater;  
  9. import android.view.View;  
  10. import android.view.ViewGroup;  
  11. import android.widget.BaseAdapter;  
  12. import android.widget.ImageView;  
  13. import android.widget.LinearLayout;  
  14. import android.widget.TextView;  
  15.   
  16. public class GridAdapter extends BaseAdapter {  
  17.     private ArrayList<String> mNameList = new ArrayList<String>();  
  18.     private ArrayList<Drawable> mDrawableList = new ArrayList<Drawable>();  
  19.     private LayoutInflater mInflater;  
  20.     private Context mContext;  
  21.     LinearLayout.LayoutParams params;  
  22.   
  23.     public GridAdapter(Context context, ArrayList<String> nameList, ArrayList<Drawable> drawableList) {  
  24.         mNameList = nameList;  
  25.         mDrawableList = drawableList;  
  26.         mContext = context;  
  27.         mInflater = LayoutInflater.from(context);  
  28.           
  29.         params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT);  
  30.         params.gravity = Gravity.CENTER;  
  31.     }  
  32.   
  33.     public int getCount() {  
  34.         return mNameList.size();  
  35.     }  
  36.   
  37.     public Object getItem(int position) {  
  38.         return mNameList.get(position);  
  39.     }  
  40.   
  41.     public long getItemId(int position) {  
  42.         return position;  
  43.     }  
  44.   
  45.     public View getView(int position, View convertView, ViewGroup parent) {  
  46.         ItemViewTag viewTag;  
  47.           
  48.         if (convertView == null)  
  49.         {  
  50.             convertView = mInflater.inflate(R.layout.gridview_item, null);  
  51.               
  52.             // construct an item tag  
  53.             viewTag = new ItemViewTag((ImageView) convertView.findViewById(R.id.grid_icon), (TextView) convertView.findViewById(R.id.grid_name));  
  54.             convertView.setTag(viewTag);  
  55.         } else  
  56.         {  
  57.             viewTag = (ItemViewTag) convertView.getTag();  
  58.         }  
  59.           
  60.         // set name  
  61.         viewTag.mName.setText(mNameList.get(position));  
  62.           
  63.         // set icon  
  64.         viewTag.mIcon.setBackgroundDrawable(mDrawableList.get(position));  
  65.         viewTag.mIcon.setLayoutParams(params);  
  66.         return convertView;  
  67.     }  
  68.       
  69.     class ItemViewTag  
  70.     {  
  71.         protected ImageView mIcon;  
  72.         protected TextView mName;  
  73.           
  74.         /** 
  75.          * The constructor to construct a navigation view tag 
  76.          *  
  77.          * @param name 
  78.          *            the name view of the item 
  79.          * @param size 
  80.          *            the size view of the item 
  81.          * @param icon 
  82.          *            the icon view of the item 
  83.          */  
  84.         public ItemViewTag(ImageView icon, TextView name)  
  85.         {  
  86.             this.mName = name;  
  87.             this.mIcon = icon;  
  88.         }  
  89.     }  
  90.   
  91. }  


到此,adapter的使用就算结束,效果图如下:




4.Android中GridView的两种不同实现方式


1.利用SimpleAdapter适配器实现。

这里以每一个网格中添加一张图片和相应的文字说明为例:

main.xml

[html]  view plain copy
  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="fill_parent">  
  6.     <GridView android:id="@+id/gridView"  
  7.         android:numColumns="4"  
  8.         android:horizontalSpacing="10px"  
  9.         android:verticalSpacing="10px"  
  10.         android:layout_width="fill_parent"  
  11.         android:layout_height="fill_parent"  
  12.         />  
  13. </LinearLayout>  


因为除了添加图片之外,还相应的添加说明性的文字,所以这里需要为每网格设置一个布局item.xml

[html]  view plain copy
  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:gravity="center_horizontal"  
  5.     android:layout_width="fill_parent"  
  6.     android:layout_height="fill_parent">  
  7.     <ImageView android:id="@+id/imageView"  
  8.         android:layout_weight="4"  
  9.         android:scaleType="fitCenter"  
  10.         android:layout_width="wrap_content"  
  11.         android:layout_height="wrap_content"  
  12.         />  
  13.     <TextView android:id="@+id/textView"  
  14.         android:layout_weight="1"  
  15.         android:gravity="center"  
  16.         android:layout_width="wrap_content"  
  17.         android:layout_height="wrap_content"  
  18.         android:textSize="24px"  
  19.         />  
  20. </LinearLayout>  

MainActivity.java中的代码:

[html]  view plain copy
  1. public class MainActivity extends Activity {  
  2.       
  3.     private int []imageId = new int[]{R.drawable.img01, R.drawable.img02, R.drawable.img03, R.drawable.img04,   
  4.             R.drawable.img05, R.drawable.img06, R.drawable.img07, R.drawable.img08, R.drawable.img09,   
  5.             R.drawable.img10, R.drawable.img11, R.drawable.img12};  
  6.     private String[] title = new String[]{"花开富贵", "海天一色", "日出", "天路", "一枝独秀", "云", "独占鳌头",   
  7.             "蒲公英花", "花团锦簇", "争奇斗艳", "和谐", "林间小路"};  
  8.     @Override  
  9.     protected void onCreate(Bundle savedInstanceState) {  
  10.         super.onCreate(savedInstanceState);  
  11.         setContentView(R.layout.activity_main);   
  12.           
  13.           
  14.         GridView gridView = (GridView)findViewById(R.id.gridView);  
  15.         List<Map<String, Object>>list = new ArrayList<Map<String, Object>>();  
  16.         for(int i = 0; i < imageId.length; i++)  
  17.         {  
  18.             Map<String, Object> map = new HashMap<String, Object>();  
  19.             map.put("image", imageId[i]);  
  20.             map.put("title", title[i]);  
  21.             list.add(map);  
  22.         }  
  23.         SimpleAdapter adapter = new SimpleAdapter(MainActivity.this, list, R.layout.item, new String[]{"image", "title"}, new int[]{R.id.imageView, R.id.textView});  
  24.         gridView.setAdapter(adapter);  
  25.     }  

Android中的GridView详解_第9张图片

2.利用BaseAdapter实现添加图片:

这里我们分两种类型进行讲解:

@1:如果只是涉及添加图像,没有相应的文字说明,这样的话BaseAdapter的实现方式就比较简单。

这里

main.xml

[html]  view plain copy
  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="fill_parent">  
  6.     <GridView android:id="@+id/gridView"  
  7.         android:numColumns="4"  
  8.         android:horizontalSpacing="10px"  
  9.         android:verticalSpacing="10px"  
  10.         android:layout_width="fill_parent"  
  11.         android:layout_height="fill_parent"  
  12.         />  
  13. </LinearLayout>  

此时不再需要item.xml.

MainActivity.java中的代码:

BaseAdapter需要重写getView()(最重要的)、getItem()、getItemId()、getCount()这四个方法。

还有就是convertView是一个缓存View。

通过缓存convertView,这种利用缓存contentView的方式可以判断如果缓存中不存在View才创建View,如果已经存在可以利用缓存中的View,提升了性能

[java]  view plain copy
  1. public class MainActivity extends Activity {  
  2.       
  3.     private int []imageId = new int[]{R.drawable.img01, R.drawable.img02, R.drawable.img03, R.drawable.img04,   
  4.             R.drawable.img05, R.drawable.img06, R.drawable.img07, R.drawable.img08, R.drawable.img09,   
  5.             R.drawable.img10, R.drawable.img11, R.drawable.img12, };  
  6.     //private String[] title = new String[]{"花开富贵", "海天一色", "日出", "天路", "一枝独秀", "云", "独占鳌头",   
  7.             //"蒲公英花", "花团锦簇", "争奇斗艳", "和谐", "林间小路"};  
  8.     @Override  
  9.     protected void onCreate(Bundle savedInstanceState) {  
  10.         super.onCreate(savedInstanceState);  
  11.         setContentView(R.layout.activity_main);   
  12.           
  13.         GridView gridView = (GridView)findViewById(R.id.gridView);  
  14.         BaseAdapter adapter = new BaseAdapter()  
  15.         {  
  16.             @Override  
  17.             public int getCount() {  
  18.                 // TODO 自动生成的方法存根  
  19.                 return imageId.length;  
  20.             }  
  21.             @Override  
  22.             public Object getItem(int position) {  
  23.                 // TODO 自动生成的方法存根  
  24.                 return position;  
  25.             }  
  26.             @Override  
  27.             public long getItemId(int position) {  
  28.                 // TODO 自动生成的方法存根  
  29.                 return position;  
  30.             }  
  31.             @Override  
  32.             public View getView(int position, View convertView, ViewGroup parent) {  
  33.                 // TODO 自动生成的方法存根  
  34.                 ImageView imageView;  
  35.                 if(convertView == null)  
  36.                 {  
  37.                     imageView = new ImageView(MainActivity.this);  
  38.                     imageView.setScaleType(ImageView.ScaleType.FIT_CENTER);  
  39.                     imageView.setLayoutParams(new GridView.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));  
  40.                 }  
  41.                 else  
  42.                     imageView = (ImageView)convertView;  
  43.                 imageView.setImageResource(imageId[position]);  
  44.                 return imageView;  
  45.             }  
  46.         };  
  47.         gridView.setAdapter(adapter);  
  48.           
  49.     }  

3.利用BaseAdapter实现添加图片与文字说明。

这里不仅要添加图片还要添加相应的文字说明,相对上面的方法2比较复杂。但这确是BaseAdpter的最经典的用法。这里涉及到效率问题,因而又有三种实现方式。

1.实现最简单,但效率最低。

这种方式没有采用convertView缓存机制。

其中LayoutInflater作用是将布局文件实例化为View。

main.xml与item.xml不变。

MainActivity.java中的代码如下:

[java]  view plain copy
  1. public class MainActivity extends Activity {  
  2.       
  3.     private int []imageId = new int[]{R.drawable.img01, R.drawable.img02, R.drawable.img03, R.drawable.img04,   
  4.             R.drawable.img05, R.drawable.img06, R.drawable.img07, R.drawable.img08, R.drawable.img09,   
  5.             R.drawable.img10, R.drawable.img11, R.drawable.img12, };  
  6.     private String[] title = new String[]{"花开富贵""海天一色""日出""天路""一枝独秀""云""独占鳌头",   
  7.             "蒲公英花""花团锦簇""争奇斗艳""和谐""林间小路"};  
  8.     @Override  
  9.     protected void onCreate(Bundle savedInstanceState) {  
  10.         super.onCreate(savedInstanceState);  
  11.         setContentView(R.layout.activity_main);   
  12.           
  13.   
  14.         final LayoutInflater mInflater = LayoutInflater.from(this);  
  15.         GridView gridView = (GridView)findViewById(R.id.gridView);  
  16.         BaseAdapter adapter = new BaseAdapter()  
  17.         {  
  18.             @Override  
  19.             public int getCount() {  
  20.                 // TODO 自动生成的方法存根  
  21.                 return imageId.length;  
  22.             }  
  23.             @Override  
  24.             public Object getItem(int position) {  
  25.                 // TODO 自动生成的方法存根  
  26.                 return position;  
  27.             }  
  28.             @Override  
  29.             public long getItemId(int position) {  
  30.                 // TODO 自动生成的方法存根  
  31.                 return position;  
  32.             }  
  33.             @Override  
  34.             public View getView(int position, View convertView, ViewGroup parent) {  
  35.                 // TODO 自动生成的方法存根  
  36.                 View view = mInflater.inflate(R.layout.item, null);  
  37.                 ImageView imageView = (ImageView)view.findViewById(R.id.imageView);  
  38.                 TextView textView = (TextView)view.findViewById(R.id.textView);  
  39.                 imageView.setImageResource(imageId[position]);  
  40.                 textView.setText(title[position]);  
  41.                 return view;  
  42.             }  
  43.               
  44.         };  
  45.         gridView.setAdapter(adapter);  
  46.           
  47.     }  

2.采用缓存,效率提高将近200%

除了getView()不同之外,其他和方法1完全相同。

[html]  view plain copy
  1. @Override  
  2.             public View getView(int position, View convertView, ViewGroup parent) {  
  3.                 // TODO 自动生成的方法存根  
  4.                 if(convertView == null)  
  5.                 {  
  6.                     convertView = mInflater.inflate(R.layout.item, null);  
  7.                 }  
  8.                 ImageView imageView = (ImageView)convertView.findViewById(R.id.imageView);  
  9.                 TextView textView = (TextView)convertView.findViewById(R.id.textView);  
  10.                 imageView.setImageResource(imageId[position]);  
  11.                 textView.setText(title[position]);  
  12.                 return convertView;  
  13.             }  
  14.               
  15.         };  

3.设置标签。效率再次提升50%。

[html]  view plain copy
  1. class ViewHolder  
  2. {  
  3.     private ImageView imageView;  
  4.     private TextView textView;  
  5. }  
  6. @Override  
  7. public View getView(int position, View convertView, ViewGroup parent) {  
  8.     // TODO 自动生成的方法存根  
  9.     ViewHolder holder;  
  10.     if(convertView == null)  
  11.     {  
  12.         convertView = mInflater.inflate(R.layout.item, null);  
  13.         holder = new ViewHolder();  
  14.         holder.imageView = (ImageView)convertView.findViewById(R.id.imageView);  
  15.         holder.textView = (TextView)convertView.findViewById(R.id.textView);  
  16.         convertView.setTag(holder);  
  17.     }  
  18.     holder = (ViewHolder)convertView.getTag();  
  19.       
  20.     holder.imageView.setImageResource(imageId[position]);  
  21.     holder.textView.setText(title[position]);  
  22.     return convertView;  
  23. }  


5.Android PullToRefresh (ListView GridView 下拉刷新) 使用详解


本篇博客详细给大家介绍下ListView和GridView利用pull-to-rerfesh 实现下拉刷新和上拉加载更多。

1、ListView下拉刷新快速入门

pull-to-refresh对ListView进行了封装,叫做:PullToRefreshListView,用法和listview没什么区别,下面看demo.

布局文件:

[html]  view plain copy
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent" >  
  5.   
  6.     <com.handmark.pulltorefresh.library.PullToRefreshListView  
  7.         xmlns:ptr="http://schemas.android.com/apk/res-auto"  
  8.         android:id="@+id/pull_refresh_list"  
  9.         android:layout_width="fill_parent"  
  10.         android:layout_height="fill_parent"  
  11.         android:cacheColorHint="#00000000"  
  12.         android:divider="#19000000"  
  13.         android:dividerHeight="4dp"  
  14.         android:fadingEdge="none"  
  15.         android:fastScrollEnabled="false"  
  16.         android:footerDividersEnabled="false"  
  17.         android:headerDividersEnabled="false"  
  18.         android:smoothScrollbar="true" >  
  19.     </com.handmark.pulltorefresh.library.PullToRefreshListView>  
  20.   
  21. </RelativeLayout>  

声明了一个PullToRefreshListView,里面所有的属性都是ListView的,没有任何其他属性,当然了PullToRefreshListView也提供了很多配置的属性,后面会详细介绍。

Activity的代码:

[java]  view plain copy
  1. package com.example.zhy_pulltorefreash_chenyoca;  
  2.   
  3. import java.util.LinkedList;  
  4.   
  5. import android.app.Activity;  
  6. import android.os.AsyncTask;  
  7. import android.os.Bundle;  
  8. import android.text.format.DateUtils;  
  9. import android.widget.ArrayAdapter;  
  10. import android.widget.ListView;  
  11.   
  12. import com.handmark.pulltorefresh.library.PullToRefreshBase;  
  13. import com.handmark.pulltorefresh.library.PullToRefreshBase.OnRefreshListener;  
  14. import com.handmark.pulltorefresh.library.PullToRefreshListView;  
  15.   
  16. public class PullToRefreshListActivity extends Activity  
  17. {  
  18.   
  19.     private LinkedList<String> mListItems;  
  20.     /** 
  21.      * 上拉刷新的控件 
  22.      */  
  23.     private PullToRefreshListView mPullRefreshListView;  
  24.   
  25.     private ArrayAdapter<String> mAdapter;  
  26.   
  27.     private int mItemCount = 9;  
  28.   
  29.     @Override  
  30.     protected void onCreate(Bundle savedInstanceState)  
  31.     {  
  32.         super.onCreate(savedInstanceState);  
  33.         setContentView(R.layout.activity_main);  
  34.         // 得到控件  
  35.         mPullRefreshListView = (PullToRefreshListView) findViewById(R.id.pull_refresh_list);  
  36.         //初始化数据  
  37.         initDatas();  
  38.         //设置适配器  
  39.         mAdapter = new ArrayAdapter<String>(this,  
  40.                 android.R.layout.simple_list_item_1, mListItems);  
  41.         mPullRefreshListView.setAdapter(mAdapter);  
  42.         // 设置监听事件  
  43.         mPullRefreshListView  
  44.                 .setOnRefreshListener(new OnRefreshListener<ListView>()  
  45.                 {  
  46.                     @Override  
  47.                     public void onRefresh(  
  48.                             PullToRefreshBase<ListView> refreshView)  
  49.                     {  
  50.                         String label = DateUtils.formatDateTime(  
  51.                                 getApplicationContext(),  
  52.                                 System.currentTimeMillis(),  
  53.                                 DateUtils.FORMAT_SHOW_TIME  
  54.                                         | DateUtils.FORMAT_SHOW_DATE  
  55.                                         | DateUtils.FORMAT_ABBREV_ALL);  
  56.                         // 显示最后更新的时间  
  57.                         refreshView.getLoadingLayoutProxy()  
  58.                                 .setLastUpdatedLabel(label);  
  59.   
  60.                         // 模拟加载任务  
  61.                         new GetDataTask().execute();  
  62.                     }  
  63.                 });  
  64.   
  65.     }  
  66.   
  67.     private void initDatas()  
  68.     {  
  69.         // 初始化数据和数据源  
  70.         mListItems = new LinkedList<String>();  
  71.   
  72.         for (int i = 0; i < mItemCount; i++)  
  73.         {  
  74.             mListItems.add("" + i);  
  75.         }  
  76.     }  
  77.   
  78.     private class GetDataTask extends AsyncTask<Void, Void, String>  
  79.     {  
  80.   
  81.         @Override  
  82.         protected String doInBackground(Void... params)  
  83.         {  
  84.             try  
  85.             {  
  86.                 Thread.sleep(2000);  
  87.             } catch (InterruptedException e)  
  88.             {  
  89.             }  
  90.             return "" + (mItemCount++);  
  91.         }  
  92.   
  93.         @Override  
  94.         protected void onPostExecute(String result)  
  95.         {  
  96.             mListItems.add(result);  
  97.             mAdapter.notifyDataSetChanged();  
  98.             // Call onRefreshComplete when the list has been refreshed.  
  99.             mPullRefreshListView.onRefreshComplete();  
  100.         }  
  101.     }  
  102.   
  103. }  

代码极其简单,得到PullToRefreshListView控件,然后像ListView一样设置数据集。当然了,我们有下拉刷新,所以必须设置下拉刷新的回调:

setOnRefreshListener(new OnRefreshListener<ListView>(){}

我们在回调中模拟了一个异步任务,加载了一个Item。

效果图:

Android中的GridView详解_第10张图片

下拉时,执行我们的GetDataTask任务,任务执行完成后在onPostExecute中  调用mPullRefreshListView.onRefreshComplete();完成刷新。

是不是分分钟实现下拉刷新。当然了,你可能会有疑问,下拉刷新的指示器上的文字可以自定义吗?那个图片可以换成箭头吗?说好的上拉加载更多呢?后面会一一添加~

2、添加上拉加载更多

如过希望实现上拉加载更多,那么首先需要在布局文件的声明属性中添加一个属性,用于指定目前的下拉模式:

[html]  view plain copy
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent" >  
  5.   
  6.     <com.handmark.pulltorefresh.library.PullToRefreshListView  
  7.         xmlns:ptr="http://schemas.android.com/apk/res-auto"  
  8.         android:id="@+id/pull_refresh_list"  
  9.         android:layout_width="fill_parent"  
  10.         android:layout_height="fill_parent"  
  11.         android:cacheColorHint="#00000000"  
  12.         android:divider="#19000000"  
  13.         android:dividerHeight="4dp"  
  14.         android:fadingEdge="none"  
  15.         android:fastScrollEnabled="false"  
  16.         android:footerDividersEnabled="false"  
  17.         android:headerDividersEnabled="false"  
  18.         android:smoothScrollbar="true"  
  19.         ptr:ptrMode="both" >  
  20.     </com.handmark.pulltorefresh.library.PullToRefreshListView>  
  21.   
  22. </RelativeLayout>  
我们添加了一个属性:ptr:ptrMode="both" ,意思:上拉和下拉都支持。

可选值为:disabled(禁用下拉刷新),pullFromStart(仅支持下拉刷新),pullFromEnd(仅支持上拉刷新),both(二者都支持),manualOnly(只允许手动触发)

当然了,如果你不喜欢在布局文件中指定,完全可以使用代码设置,在onCreate里面写:mPullRefreshListView.setMode(Mode.BOTH);//设置你需要的模式

设置了模式为双向都支持,当然必须为上拉和下拉分别设置回调,请看下面的代码:

[java]  view plain copy
  1. package com.example.zhy_pulltorefreash_chenyoca;  
  2.   
  3. import java.util.LinkedList;  
  4.   
  5. import android.app.Activity;  
  6. import android.os.AsyncTask;  
  7. import android.os.Bundle;  
  8. import android.text.format.DateUtils;  
  9. import android.util.Log;  
  10. import android.widget.ArrayAdapter;  
  11. import android.widget.ListView;  
  12.   
  13. import com.handmark.pulltorefresh.library.PullToRefreshBase;  
  14. import com.handmark.pulltorefresh.library.PullToRefreshBase.Mode;  
  15. import com.handmark.pulltorefresh.library.PullToRefreshBase.OnRefreshListener;  
  16. import com.handmark.pulltorefresh.library.PullToRefreshBase.OnRefreshListener2;  
  17. import com.handmark.pulltorefresh.library.PullToRefreshListView;  
  18.   
  19. public class PullToRefreshListActivity extends Activity  
  20. {  
  21.   
  22.     private LinkedList<String> mListItems;  
  23.     /** 
  24.      * 上拉刷新的控件 
  25.      */  
  26.     private PullToRefreshListView mPullRefreshListView;  
  27.   
  28.     private ArrayAdapter<String> mAdapter;  
  29.   
  30.     private int mItemCount = 9;  
  31.   
  32.     @Override  
  33.     protected void onCreate(Bundle savedInstanceState)  
  34.     {  
  35.         super.onCreate(savedInstanceState);  
  36.         setContentView(R.layout.activity_main);  
  37.         // 得到控件  
  38.         mPullRefreshListView = (PullToRefreshListView) findViewById(R.id.pull_refresh_list);  
  39.         mPullRefreshListView.setMode(Mode.BOTH);  
  40.         // 初始化数据  
  41.         initDatas();  
  42.         // 设置适配器  
  43.         mAdapter = new ArrayAdapter<String>(this,  
  44.                 android.R.layout.simple_list_item_1, mListItems);  
  45.         mPullRefreshListView.setAdapter(mAdapter);  
  46.   
  47.         mPullRefreshListView  
  48.                 .setOnRefreshListener(new OnRefreshListener2<ListView>()  
  49.                 {  
  50.                     @Override  
  51.                     public void onPullDownToRefresh(  
  52.                             PullToRefreshBase<ListView> refreshView)  
  53.                     {  
  54.                         Log.e("TAG""onPullDownToRefresh");  
  55.                         //这里写下拉刷新的任务  
  56.                         new GetDataTask().execute();  
  57.                     }  
  58.   
  59.                     @Override  
  60.                     public void onPullUpToRefresh(  
  61.                             PullToRefreshBase<ListView> refreshView)  
  62.                     {  
  63.                         Log.e("TAG""onPullUpToRefresh");  
  64.                         //这里写上拉加载更多的任务  
  65.                         new GetDataTask().execute();  
  66.                     }  
  67.                 });  
  68.   
  69.     }  
  70.   
  71.     private void initDatas()  
  72.     {  
  73.         // 初始化数据和数据源  
  74.         mListItems = new LinkedList<String>();  
  75.   
  76.         for (int i = 0; i < mItemCount; i++)  
  77.         {  
  78.             mListItems.add("" + i);  
  79.         }  
  80.     }  
  81.   
  82.     private class GetDataTask extends AsyncTask<Void, Void, String>  
  83.     {  
  84.   
  85.         @Override  
  86.         protected String doInBackground(Void... params)  
  87.         {  
  88.             try  
  89.             {  
  90.                 Thread.sleep(2000);  
  91.             } catch (InterruptedException e)  
  92.             {  
  93.             }  
  94.             return "" + (mItemCount++);  
  95.         }  
  96.   
  97.         @Override  
  98.         protected void onPostExecute(String result)  
  99.         {  
  100.             mListItems.add(result);  
  101.             mAdapter.notifyDataSetChanged();  
  102.             // Call onRefreshComplete when the list has been refreshed.  
  103.             mPullRefreshListView.onRefreshComplete();  
  104.         }  
  105.     }  

和第一段的代码只有一个地方有区别,可能很难发现:
mPullRefreshListView.setOnRefreshListener(new OnRefreshListener2<ListView>(){});注意这里的接口类型是OnRefreshListener2,多了个2,和上面的不一样,这个接口包含两个方法,一个上拉回调,一个下拉回调。好了,这样我们就成功添加了上拉与下拉,并且分别可以控制其回调代码。

效果图:

Android中的GridView详解_第11张图片

咋样,是不是也很简单~注:如果你的上拉和下拉需求是执行一样的代码,那么你可以继续注册OnRefreshListener接口,上拉和下拉都会执行同一个方法。

接下来介绍如何使用带下拉刷新和加载更多的的GridView和自定义样式~

3、带下拉和上拉的GridView ( PullToRefreshGridView )

同样的pull-to-refresh把GridView封装为:PullToRefreshGridView 。用法和PullToRefreshListView一摸一样~

首先看主布局文件:

[html]  view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:layout_width="fill_parent"  
  4.     android:layout_height="fill_parent"  
  5.     android:orientation="vertical" >  
  6.   
  7.     <!-- The PullToRefreshGridView replaces a standard GridView widget. -->  
  8.   
  9.     <com.handmark.pulltorefresh.library.PullToRefreshGridView  
  10.         xmlns:ptr="http://schemas.android.com/apk/res-auto"  
  11.         android:id="@+id/pull_refresh_grid"  
  12.         android:layout_width="fill_parent"  
  13.         android:layout_height="fill_parent"  
  14.         android:columnWidth="100dp"  
  15.         android:gravity="center_horizontal"  
  16.         android:horizontalSpacing="1dp"  
  17.         android:numColumns="auto_fit"  
  18.         android:stretchMode="columnWidth"  
  19.         android:verticalSpacing="1dp"  
  20.         ptr:ptrDrawable="@drawable/ic_launcher"  
  21.         ptr:ptrMode="both" />  
  22.   
  23. </LinearLayout>  
PullToRefreshGridView 的item的布局文件:

[html]  view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <TextView xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:id="@+id/id_grid_item_text"  
  4.     android:layout_width="100dp"  
  5.     android:gravity="center"  
  6.     android:textColor="#ffffff"  
  7.     android:textSize="16sp"  
  8.     android:background="#000000"  
  9.     android:layout_height="100dp" />  

接下来就是Activity的代码了:

[java]  view plain copy
  1. public class PullToRefreshGridActivity extends Activity  
  2. {  
  3.     private LinkedList<String> mListItems;  
  4.     private PullToRefreshGridView mPullRefreshListView;  
  5.     private ArrayAdapter<String> mAdapter;  
  6.   
  7.     private int mItemCount = 10;  
  8.   
  9.     @Override  
  10.     protected void onCreate(Bundle savedInstanceState)  
  11.     {  
  12.         super.onCreate(savedInstanceState);  
  13.         setContentView(R.layout.activity_ptr_grid);  
  14.         // 得到控件  
  15.         mPullRefreshListView = (PullToRefreshGridView) findViewById(R.id.pull_refresh_grid);  
  16.   
  17.         // 初始化数据和数据源  
  18.         initDatas();  
  19.   
  20.         mAdapter = new ArrayAdapter<String>(this, R.layout.grid_item,  
  21.                 R.id.id_grid_item_text, mListItems);  
  22.         mPullRefreshListView.setAdapter(mAdapter);  
  23.   
  24.         mPullRefreshListView  
  25.                 .setOnRefreshListener(new OnRefreshListener2<GridView>()  
  26.                 {  
  27.   
  28.                     @Override  
  29.                     public void onPullDownToRefresh(  
  30.                             PullToRefreshBase<GridView> refreshView)  
  31.                     {  
  32.                         Log.e("TAG""onPullDownToRefresh"); // Do work to  
  33.                         String label = DateUtils.formatDateTime(  
  34.                                 getApplicationContext(),  
  35.                                 System.currentTimeMillis(),  
  36.                                 DateUtils.FORMAT_SHOW_TIME  
  37.                                         | DateUtils.FORMAT_SHOW_DATE  
  38.                                         | DateUtils.FORMAT_ABBREV_ALL);  
  39.   
  40.                         // Update the LastUpdatedLabel  
  41.                         refreshView.getLoadingLayoutProxy()  
  42.                                 .setLastUpdatedLabel(label);  
  43.                           
  44.                         new GetDataTask().execute();  
  45.                     }  
  46.   
  47.                     @Override  
  48.                     public void onPullUpToRefresh(  
  49.                             PullToRefreshBase<GridView> refreshView)  
  50.                     {  
  51.                         Log.e("TAG""onPullUpToRefresh"); // Do work to refresh  
  52.                                                             // the list here.  
  53.                         new GetDataTask().execute();  
  54.                     }  
  55.                 });  
  56.     }  
  57.   
  58.     private void initDatas()  
  59.     {  
  60.         mListItems = new LinkedList<String>();  
  61.   
  62.         for (int i = 0; i < mItemCount; i++)  
  63.         {  
  64.             mListItems.add(i + "");  
  65.         }  
  66.     }  
  67.   
  68.     private class GetDataTask extends AsyncTask<Void, Void, Void>  
  69.     {  
  70.   
  71.         @Override  
  72.         protected Void doInBackground(Void... params)  
  73.         {  
  74.             try  
  75.             {  
  76.                 Thread.sleep(2000);  
  77.             } catch (InterruptedException e)  
  78.             {  
  79.             }  
  80.             return null;  
  81.         }  
  82.   
  83.         @Override  
  84.         protected void onPostExecute(Void result)  
  85.         {  
  86.             mListItems.add("" + mItemCount++);  
  87.             mAdapter.notifyDataSetChanged();  
  88.             // Call onRefreshComplete when the list has been refreshed.  
  89.             mPullRefreshListView.onRefreshComplete();  
  90.         }  
  91.     }  

基本上上例没有任何区别,直接看效果图吧:

Android中的GridView详解_第12张图片

效果还是不错的,如果你比较细心会发现,那个下拉刷新的转圈的图片咋变成机器人了,那是因为我在布局文件里面设置了:

[html]  view plain copy
  1. <com.handmark.pulltorefresh.library.PullToRefreshGridView  
  2.        ptr:ptrDrawable="@drawable/ic_launcher"  
  3.        ...  
  4.        />  

当然了这是旋转的效果,一般常用的还有,一个箭头倒置的效果,其实也很简单,一个属性:

ptr:ptrAnimationStyle="flip"

去掉 ptr:ptrDrawable="@drawable/ic_launcher"这个属性,如果你希望用下图默认的箭头,你也可以自定义。

添加后,箭头就是这个样子:

Android中的GridView详解_第13张图片

ptr:ptrAnimationStyle的取值:flip(翻转动画), rotate(旋转动画) 。 

ptr:ptrDrawable则就是设置图标了。


4、自定义下拉指示器文本内容等效果

可以在初始化完成mPullRefreshListView后,通过mPullRefreshListView.getLoadingLayoutProxy()可以得到一个ILoadingLayout对象,这个对象可以设置各种指示器中的样式、文本等。

[java]  view plain copy
  1. ILoadingLayout startLabels = mPullRefreshListView  
  2.                 .getLoadingLayoutProxy();  
  3.         startLabels.setPullLabel("你可劲拉,拉...");// 刚下拉时,显示的提示  
  4.         startLabels.setRefreshingLabel("好嘞,正在刷新...");// 刷新时  
  5.         startLabels.setReleaseLabel("你敢放,我就敢刷新...");// 下来达到一定距离时,显示的提示  

如果你比较细心,会发现,前面我们设置上次刷新时间已经用到了:

// Update the LastUpdatedLabel
refreshView.getLoadingLayoutProxy().setLastUpdatedLabel(label);

现在的效果是:



默认是上拉和下拉的字同时改变的,如果我希望单独改变呢?

[java]  view plain copy
  1. private void initIndicator()  
  2.     {  
  3.         ILoadingLayout startLabels = mPullRefreshListView  
  4.                 .getLoadingLayoutProxy(truefalse);  
  5.         startLabels.setPullLabel("你可劲拉,拉...");// 刚下拉时,显示的提示  
  6.         startLabels.setRefreshingLabel("好嘞,正在刷新...");// 刷新时  
  7.         startLabels.setReleaseLabel("你敢放,我就敢刷新...");// 下来达到一定距离时,显示的提示  
  8.   
  9.         ILoadingLayout endLabels = mPullRefreshListView.getLoadingLayoutProxy(  
  10.                 falsetrue);  
  11.         endLabels.setPullLabel("你可劲拉,拉2...");// 刚下拉时,显示的提示  
  12.         endLabels.setRefreshingLabel("好嘞,正在刷新2...");// 刷新时  
  13.         endLabels.setReleaseLabel("你敢放,我就敢刷新2...");// 下来达到一定距离时,显示的提示  
  14.     }  

mPullRefreshListView.getLoadingLayoutProxy(true, false);接收两个参数,为true,false返回设置下拉的ILoadingLayout;为false,true返回设置上拉的。

5、常用的一些属性

当然了,pull-to-refresh在xml中还能定义一些属性:

ptrMode,ptrDrawable,ptrAnimationStyle这三个上面已经介绍过。

ptrRefreshableViewBackground 设置整个mPullRefreshListView的背景色

ptrHeaderBackground 设置下拉Header或者上拉Footer的背景色

ptrHeaderTextColor 用于设置Header与Footer中文本的颜色

ptrHeaderSubTextColor 用于设置Header与Footer中上次刷新时间的颜色

ptrShowIndicator如果为true会在mPullRefreshListView中出现icon,右上角和右下角,挺有意思的。

ptrHeaderTextAppearance , ptrSubHeaderTextAppearance分别设置拉Header或者上拉Footer中字体的类型颜色等等。

ptrRotateDrawableWhilePulling当动画设置为rotate时,下拉是是否旋转。

ptrScrollingWhileRefreshingEnabled刷新的时候,是否允许ListView或GridView滚动。觉得为true比较好。

ptrListViewExtrasEnabled 决定了Header,Footer以何种方式加入mPullRefreshListView,true为headView方式加入,就是滚动时刷新头部会一起滚动。

最后2个其实对于用户体验还是挺重要的,如果设置的时候考虑下~。其他的属性自己选择就好。

注:上述属性很多都可以代码控制,如果有需要可以直接mPullRefreshListView.set属性名 查看

以上为pull-to-refresh所有支持的属性~~

定义了一堆的效果:

Android中的GridView详解_第14张图片

右上、右下那个图标就是ptrShowIndicator。好了,大家可以按照自己的需求对着上面的属性解释配置。


在github上下载的例子,是依赖3个项目的,一个基本的library_pullToRefresh,一个PullToRefreshViewPager,一个PullToRefreshListFragment ;

上面介绍的例子只依赖library_pullToRefresh,其他两个依赖不用导入。


你可能感兴趣的:(android,GridView)