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" />

<GridViewandroid: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 implementsOnItemSelectedListener,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( newFunnyLookingAdapter(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(AdapterViewarg0, View arg1, int arg2, long arg3) {
selection.setText(items[arg2]);
}

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

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

//步骤4:编写自定义的adapter,继承ArrayAdapter
private classFunnyLookingAdapter 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 ViewgetView (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的文件很简单:



<GridViewxmlns: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(AdapterViewparent, View v, int position, long id){
Toast.makeText(this, ""+position, Toast.LENGTH_SHORT).show();
}
});
}

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

public classImageAdapter 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(intposition, ViewconvertView, ViewGroupparent){
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. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns: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. <GridViewxmlns: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. <!--android:numColumns="auto_fit",GridView的列数设置为自动-->
  18. <!--android:columnWidth="90dp",每列的宽度,也就是Item的宽度-->
  19. <!--android:stretchMode="columnWidth",缩放与列宽大小同步-->
  20. <!--android:verticalSpacing="10dp",两行之间的边距,如:行一(NO.0~NO.2)与行二(NO.3~NO.5)间距为10dp-->
  21. <!--android:horizontalSpacing="10dp",两列之间的边距。-->

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

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

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

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

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

Function父类的设计如下:

[java] view plain copy
  1. publicabstractclassFunction{
  2. protectedintfunctionImageRscId;
  3. protectedintfunctionTextRscId;
  4. protectedContextcontext=null;
  5. publicFunction(Contextcontext){
  6. //TODOAuto-generatedconstructorstub
  7. this.context=context;//获取上下文
  8. }
  9. publicabstractOnClickListenergetOnClickListener();
  10. //抽象函数,强制子类去继承它,并实现各自的功能
  11. publicintgetFunctionImageRscId(){
  12. returnfunctionImageRscId;
  13. }
  14. publicintgetFunctionTextRscId(){
  15. returnfunctionTextRscId;
  16. }
  17. }

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

[java] view plain copy
  1. publicclassFreeNumberFunctionextendsFunction{
  2. publicFreeNumberFunction(Contextcontext){
  3. super(context);
  4. //TODOAuto-generatedconstructorstub
  5. this.functionImageRscId=R.drawable.free_number_function_a;//本功能的图标资源
  6. this.functionTextRscId=R.string.free_number_function;//功能名字的字符串资源
  7. }
  8. publicOnClickListenergetOnClickListener(){
  9. //TODOAuto-generatedmethodstub
  10. returnnewOnClickListener(){//测试的响应事件
  11. publicvoidonClick(Viewv){
  12. newAlertDialog.Builder(context)
  13. .setTitle("释放号码")
  14. .setMessage("打开释放号码的界面")
  15. .show();
  16. }
  17. };
  18. }
  19. }

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

[c-sharp] view plain copy
  1. publicclassReportFunctionextendsFunction{
  2. publicReportFunction(Contextcontext){
  3. super(context);
  4. this.functionImageRscId=R.drawable.report_function_a;
  5. this.functionTextRscId=R.string.report_function;
  6. }
  7. @Override
  8. publicOnClickListenergetOnClickListener(){
  9. //TODOAuto-generatedmethodstub
  10. returnnewOnClickListener(){
  11. publicvoidonClick(Viewv){
  12. newAlertDialog.Builder(context)
  13. .setTitle("经营分析")
  14. .setMessage("MessageTest")
  15. .show();
  16. }
  17. };
  18. }
  19. }

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

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

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

编写自定义适配器没有想象中恐怖,继承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. publicclassMainTableextendsActivity{
  2. @Override
  3. protectedvoidonCreate(BundlesavedInstanceState){
  4. //TODOAuto-generatedmethodstub
  5. super.onCreate(savedInstanceState);
  6. this.requestWindowFeature(Window.FEATURE_NO_TITLE);//无title
  7. this.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_NOSENSOR);//禁止竖屏
  8. setContentView(R.layout.main_table);
  9. super.onCreate(savedInstanceState);
  10. GridViewgridview=(GridView)findViewById(R.id.grid_view);
  11. ArrayList<Function>meumList=newArrayList<Function>();
  12. meumList.add(newFreeNumberFunction(this));
  13. meumList.add(newReportFunction(this));
  14. FunctionsAdapterfa=newFunctionsAdapter(this,meumList);
  15. gridview.setAdapter(fa);
  16. }
  17. }

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

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. <?xmlversion="1.0"encoding="utf-8"?>
  2. <FrameLayoutxmlns: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. GridViewgrid=(GridView)findViewById(R.id.grid);
  2. grid.setAdapter(newGridAdapter(this,mNameList,mDrawableList));

GridAdapter.java类:

[java] view plain copy
  1. packagecom.android.test;
  2. importjava.util.ArrayList;
  3. importandroid.content.Context;
  4. importandroid.graphics.drawable.Drawable;
  5. importandroid.view.Gravity;
  6. importandroid.view.LayoutInflater;
  7. importandroid.view.View;
  8. importandroid.view.ViewGroup;
  9. importandroid.widget.BaseAdapter;
  10. importandroid.widget.ImageView;
  11. importandroid.widget.LinearLayout;
  12. importandroid.widget.TextView;
  13. publicclassGridAdapterextendsBaseAdapter{
  14. privateArrayList<String>mNameList=newArrayList<String>();
  15. privateArrayList<Drawable>mDrawableList=newArrayList<Drawable>();
  16. privateLayoutInflatermInflater;
  17. privateContextmContext;
  18. LinearLayout.LayoutParamsparams;
  19. publicGridAdapter(Contextcontext,ArrayList<String>nameList,ArrayList<Drawable>drawableList){
  20. mNameList=nameList;
  21. mDrawableList=drawableList;
  22. mContext=context;
  23. mInflater=LayoutInflater.from(context);
  24. params=newLinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT,LinearLayout.LayoutParams.WRAP_CONTENT);
  25. params.gravity=Gravity.CENTER;
  26. }
  27. publicintgetCount(){
  28. returnmNameList.size();
  29. }
  30. publicObjectgetItem(intposition){
  31. returnmNameList.get(position);
  32. }
  33. publiclonggetItemId(intposition){
  34. returnposition;
  35. }
  36. publicViewgetView(intposition,ViewconvertView,ViewGroupparent){
  37. ItemViewTagviewTag;
  38. if(convertView==null)
  39. {
  40. convertView=mInflater.inflate(R.layout.gridview_item,null);
  41. //constructanitemtag
  42. viewTag=newItemViewTag((ImageView)convertView.findViewById(R.id.grid_icon),(TextView)convertView.findViewById(R.id.grid_name));
  43. convertView.setTag(viewTag);
  44. }else
  45. {
  46. viewTag=(ItemViewTag)convertView.getTag();
  47. }
  48. //setname
  49. viewTag.mName.setText(mNameList.get(position));
  50. //seticon
  51. viewTag.mIcon.setBackgroundDrawable(mDrawableList.get(position));
  52. viewTag.mIcon.setLayoutParams(params);
  53. returnconvertView;
  54. }
  55. classItemViewTag
  56. {
  57. protectedImageViewmIcon;
  58. protectedTextViewmName;
  59. /**
  60. *Theconstructortoconstructanavigationviewtag
  61. *
  62. *@paramname
  63. *thenameviewoftheitem
  64. *@paramsize
  65. *thesizeviewoftheitem
  66. *@paramicon
  67. *theiconviewoftheitem
  68. */
  69. publicItemViewTag(ImageViewicon,TextViewname)
  70. {
  71. this.mName=name;
  72. this.mIcon=icon;
  73. }
  74. }
  75. }


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




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


1.利用SimpleAdapter适配器实现。

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

main.xml

[html] view plain copy
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns: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. <GridViewandroid: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. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns: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. <ImageViewandroid: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. <TextViewandroid: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. publicclassMainActivityextendsActivity{
  2. privateint[]imageId=newint[]{R.drawable.img01,R.drawable.img02,R.drawable.img03,R.drawable.img04,
  3. R.drawable.img05,R.drawable.img06,R.drawable.img07,R.drawable.img08,R.drawable.img09,
  4. R.drawable.img10,R.drawable.img11,R.drawable.img12};
  5. privateString[]title=newString[]{"花开富贵","海天一色","日出","天路","一枝独秀","云","独占鳌头",
  6. "蒲公英花","花团锦簇","争奇斗艳","和谐","林间小路"};
  7. @Override
  8. protectedvoidonCreate(BundlesavedInstanceState){
  9. super.onCreate(savedInstanceState);
  10. setContentView(R.layout.activity_main);
  11. GridViewgridView=(GridView)findViewById(R.id.gridView);
  12. List<Map<String,Object>>list=newArrayList<Map<String,Object>>();
  13. for(inti=0;i<imageId.length;i++)
  14. {
  15. Map<String,Object>map=newHashMap<String,Object>();
  16. map.put("image",imageId[i]);
  17. map.put("title",title[i]);
  18. list.add(map);
  19. }
  20. SimpleAdapteradapter=newSimpleAdapter(MainActivity.this,list,R.layout.item,newString[]{"image","title"},newint[]{R.id.imageView,R.id.textView});
  21. gridView.setAdapter(adapter);
  22. }

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

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

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

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

这里

main.xml

[html] view plain copy
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns: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. <GridViewandroid: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. publicclassMainActivityextendsActivity{
  2. privateint[]imageId=newint[]{R.drawable.img01,R.drawable.img02,R.drawable.img03,R.drawable.img04,
  3. R.drawable.img05,R.drawable.img06,R.drawable.img07,R.drawable.img08,R.drawable.img09,
  4. R.drawable.img10,R.drawable.img11,R.drawable.img12,};
  5. //privateString[]title=newString[]{"花开富贵","海天一色","日出","天路","一枝独秀","云","独占鳌头",
  6. //"蒲公英花","花团锦簇","争奇斗艳","和谐","林间小路"};
  7. @Override
  8. protectedvoidonCreate(BundlesavedInstanceState){
  9. super.onCreate(savedInstanceState);
  10. setContentView(R.layout.activity_main);
  11. GridViewgridView=(GridView)findViewById(R.id.gridView);
  12. BaseAdapteradapter=newBaseAdapter()
  13. {
  14. @Override
  15. publicintgetCount(){
  16. //TODO自动生成的方法存根
  17. returnimageId.length;
  18. }
  19. @Override
  20. publicObjectgetItem(intposition){
  21. //TODO自动生成的方法存根
  22. returnposition;
  23. }
  24. @Override
  25. publiclonggetItemId(intposition){
  26. //TODO自动生成的方法存根
  27. returnposition;
  28. }
  29. @Override
  30. publicViewgetView(intposition,ViewconvertView,ViewGroupparent){
  31. //TODO自动生成的方法存根
  32. ImageViewimageView;
  33. if(convertView==null)
  34. {
  35. imageView=newImageView(MainActivity.this);
  36. imageView.setScaleType(ImageView.ScaleType.FIT_CENTER);
  37. imageView.setLayoutParams(newGridView.LayoutParams(LayoutParams.WRAP_CONTENT,LayoutParams.WRAP_CONTENT));
  38. }
  39. else
  40. imageView=(ImageView)convertView;
  41. imageView.setImageResource(imageId[position]);
  42. returnimageView;
  43. }
  44. };
  45. gridView.setAdapter(adapter);
  46. }

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

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

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

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

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

main.xml与item.xml不变。

MainActivity.java中的代码如下:

[java] view plain copy
  1. publicclassMainActivityextendsActivity{
  2. privateint[]imageId=newint[]{R.drawable.img01,R.drawable.img02,R.drawable.img03,R.drawable.img04,
  3. R.drawable.img05,R.drawable.img06,R.drawable.img07,R.drawable.img08,R.drawable.img09,
  4. R.drawable.img10,R.drawable.img11,R.drawable.img12,};
  5. privateString[]title=newString[]{"花开富贵","海天一色","日出","天路","一枝独秀","云","独占鳌头",
  6. "蒲公英花","花团锦簇","争奇斗艳","和谐","林间小路"};
  7. @Override
  8. protectedvoidonCreate(BundlesavedInstanceState){
  9. super.onCreate(savedInstanceState);
  10. setContentView(R.layout.activity_main);
  11. finalLayoutInflatermInflater=LayoutInflater.from(this);
  12. GridViewgridView=(GridView)findViewById(R.id.gridView);
  13. BaseAdapteradapter=newBaseAdapter()
  14. {
  15. @Override
  16. publicintgetCount(){
  17. //TODO自动生成的方法存根
  18. returnimageId.length;
  19. }
  20. @Override
  21. publicObjectgetItem(intposition){
  22. //TODO自动生成的方法存根
  23. returnposition;
  24. }
  25. @Override
  26. publiclonggetItemId(intposition){
  27. //TODO自动生成的方法存根
  28. returnposition;
  29. }
  30. @Override
  31. publicViewgetView(intposition,ViewconvertView,ViewGroupparent){
  32. //TODO自动生成的方法存根
  33. Viewview=mInflater.inflate(R.layout.item,null);
  34. ImageViewimageView=(ImageView)view.findViewById(R.id.imageView);
  35. TextViewtextView=(TextView)view.findViewById(R.id.textView);
  36. imageView.setImageResource(imageId[position]);
  37. textView.setText(title[position]);
  38. returnview;
  39. }
  40. };
  41. gridView.setAdapter(adapter);
  42. }

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

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

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

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

[html] view plain copy
  1. classViewHolder
  2. {
  3. privateImageViewimageView;
  4. privateTextViewtextView;
  5. }
  6. @Override
  7. publicViewgetView(intposition,ViewconvertView,ViewGroupparent){
  8. //TODO自动生成的方法存根
  9. ViewHolderholder;
  10. if(convertView==null)
  11. {
  12. convertView=mInflater.inflate(R.layout.item,null);
  13. holder=newViewHolder();
  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. holder.imageView.setImageResource(imageId[position]);
  20. holder.textView.setText(title[position]);
  21. returnconvertView;
  22. }


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


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

1、ListView下拉刷新快速入门

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

布局文件:

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

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

Activity的代码:

[java] view plain copy
  1. packagecom.example.zhy_pulltorefreash_chenyoca;
  2. importjava.util.LinkedList;
  3. importandroid.app.Activity;
  4. importandroid.os.AsyncTask;
  5. importandroid.os.Bundle;
  6. importandroid.text.format.DateUtils;
  7. importandroid.widget.ArrayAdapter;
  8. importandroid.widget.ListView;
  9. importcom.handmark.pulltorefresh.library.PullToRefreshBase;
  10. importcom.handmark.pulltorefresh.library.PullToRefreshBase.OnRefreshListener;
  11. importcom.handmark.pulltorefresh.library.PullToRefreshListView;
  12. publicclassPullToRefreshListActivityextendsActivity
  13. {
  14. privateLinkedList<String>mListItems;
  15. /**
  16. *上拉刷新的控件
  17. */
  18. privatePullToRefreshListViewmPullRefreshListView;
  19. privateArrayAdapter<String>mAdapter;
  20. privateintmItemCount=9;
  21. @Override
  22. protectedvoidonCreate(BundlesavedInstanceState)
  23. {
  24. super.onCreate(savedInstanceState);
  25. setContentView(R.layout.activity_main);
  26. //得到控件
  27. mPullRefreshListView=(PullToRefreshListView)findViewById(R.id.pull_refresh_list);
  28. //初始化数据
  29. initDatas();
  30. //设置适配器
  31. mAdapter=newArrayAdapter<String>(this,
  32. android.R.layout.simple_list_item_1,mListItems);
  33. mPullRefreshListView.setAdapter(mAdapter);
  34. //设置监听事件
  35. mPullRefreshListView
  36. .setOnRefreshListener(newOnRefreshListener<ListView>()
  37. {
  38. @Override
  39. publicvoidonRefresh(
  40. PullToRefreshBase<ListView>refreshView)
  41. {
  42. Stringlabel=DateUtils.formatDateTime(
  43. getApplicationContext(),
  44. System.currentTimeMillis(),
  45. DateUtils.FORMAT_SHOW_TIME
  46. |DateUtils.FORMAT_SHOW_DATE
  47. |DateUtils.FORMAT_ABBREV_ALL);
  48. //显示最后更新的时间
  49. refreshView.getLoadingLayoutProxy()
  50. .setLastUpdatedLabel(label);
  51. //模拟加载任务
  52. newGetDataTask().execute();
  53. }
  54. });
  55. }
  56. privatevoidinitDatas()
  57. {
  58. //初始化数据和数据源
  59. mListItems=newLinkedList<String>();
  60. for(inti=0;i<mItemCount;i++)
  61. {
  62. mListItems.add(""+i);
  63. }
  64. }
  65. privateclassGetDataTaskextendsAsyncTask<Void,Void,String>
  66. {
  67. @Override
  68. protectedStringdoInBackground(Void...params)
  69. {
  70. try
  71. {
  72. Thread.sleep(2000);
  73. }catch(InterruptedExceptione)
  74. {
  75. }
  76. return""+(mItemCount++);
  77. }
  78. @Override
  79. protectedvoidonPostExecute(Stringresult)
  80. {
  81. mListItems.add(result);
  82. mAdapter.notifyDataSetChanged();
  83. //CallonRefreshCompletewhenthelisthasbeenrefreshed.
  84. mPullRefreshListView.onRefreshComplete();
  85. }
  86. }
  87. }

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

setOnRefreshListener(new OnRefreshListener<ListView>(){}

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

效果图:

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

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

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

2、添加上拉加载更多

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

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

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

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

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

[java] view plain copy
  1. packagecom.example.zhy_pulltorefreash_chenyoca;
  2. importjava.util.LinkedList;
  3. importandroid.app.Activity;
  4. importandroid.os.AsyncTask;
  5. importandroid.os.Bundle;
  6. importandroid.text.format.DateUtils;
  7. importandroid.util.Log;
  8. importandroid.widget.ArrayAdapter;
  9. importandroid.widget.ListView;
  10. importcom.handmark.pulltorefresh.library.PullToRefreshBase;
  11. importcom.handmark.pulltorefresh.library.PullToRefreshBase.Mode;
  12. importcom.handmark.pulltorefresh.library.PullToRefreshBase.OnRefreshListener;
  13. importcom.handmark.pulltorefresh.library.PullToRefreshBase.OnRefreshListener2;
  14. importcom.handmark.pulltorefresh.library.PullToRefreshListView;
  15. publicclassPullToRefreshListActivityextendsActivity
  16. {
  17. privateLinkedList<String>mListItems;
  18. /**
  19. *上拉刷新的控件
  20. */
  21. privatePullToRefreshListViewmPullRefreshListView;
  22. privateArrayAdapter<String>mAdapter;
  23. privateintmItemCount=9;
  24. @Override
  25. protectedvoidonCreate(BundlesavedInstanceState)
  26. {
  27. super.onCreate(savedInstanceState);
  28. setContentView(R.layout.activity_main);
  29. //得到控件
  30. mPullRefreshListView=(PullToRefreshListView)findViewById(R.id.pull_refresh_list);
  31. mPullRefreshListView.setMode(Mode.BOTH);
  32. //初始化数据
  33. initDatas();
  34. //设置适配器
  35. mAdapter=newArrayAdapter<String>(this,
  36. android.R.layout.simple_list_item_1,mListItems);
  37. mPullRefreshListView.setAdapter(mAdapter);
  38. mPullRefreshListView
  39. .setOnRefreshListener(newOnRefreshListener2<ListView>()
  40. {
  41. @Override
  42. publicvoidonPullDownToRefresh(
  43. PullToRefreshBase<ListView>refreshView)
  44. {
  45. Log.e("TAG","onPullDownToRefresh");
  46. //这里写下拉刷新的任务
  47. newGetDataTask().execute();
  48. }
  49. @Override
  50. publicvoidonPullUpToRefresh(
  51. PullToRefreshBase<ListView>refreshView)
  52. {
  53. Log.e("TAG","onPullUpToRefresh");
  54. //这里写上拉加载更多的任务
  55. newGetDataTask().execute();
  56. }
  57. });
  58. }
  59. privatevoidinitDatas()
  60. {
  61. //初始化数据和数据源
  62. mListItems=newLinkedList<String>();
  63. for(inti=0;i<mItemCount;i++)
  64. {
  65. mListItems.add(""+i);
  66. }
  67. }
  68. privateclassGetDataTaskextendsAsyncTask<Void,Void,String>
  69. {
  70. @Override
  71. protectedStringdoInBackground(Void...params)
  72. {
  73. try
  74. {
  75. Thread.sleep(2000);
  76. }catch(InterruptedExceptione)
  77. {
  78. }
  79. return""+(mItemCount++);
  80. }
  81. @Override
  82. protectedvoidonPostExecute(Stringresult)
  83. {
  84. mListItems.add(result);
  85. mAdapter.notifyDataSetChanged();
  86. //CallonRefreshCompletewhenthelisthasbeenrefreshed.
  87. mPullRefreshListView.onRefreshComplete();
  88. }
  89. }

和第一段的代码只有一个地方有区别,可能很难发现:
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. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns: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. <!--ThePullToRefreshGridViewreplacesastandardGridViewwidget.-->
  7. <com.handmark.pulltorefresh.library.PullToRefreshGridView
  8. xmlns:ptr="http://schemas.android.com/apk/res-auto"
  9. android:id="@+id/pull_refresh_grid"
  10. android:layout_width="fill_parent"
  11. android:layout_height="fill_parent"
  12. android:columnWidth="100dp"
  13. android:gravity="center_horizontal"
  14. android:horizontalSpacing="1dp"
  15. android:numColumns="auto_fit"
  16. android:stretchMode="columnWidth"
  17. android:verticalSpacing="1dp"
  18. ptr:ptrDrawable="@drawable/ic_launcher"
  19. ptr:ptrMode="both"/>
  20. </LinearLayout>
PullToRefreshGridView 的item的布局文件:

[html] view plain copy
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <TextViewxmlns: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. publicclassPullToRefreshGridActivityextendsActivity
  2. {
  3. privateLinkedList<String>mListItems;
  4. privatePullToRefreshGridViewmPullRefreshListView;
  5. privateArrayAdapter<String>mAdapter;
  6. privateintmItemCount=10;
  7. @Override
  8. protectedvoidonCreate(BundlesavedInstanceState)
  9. {
  10. super.onCreate(savedInstanceState);
  11. setContentView(R.layout.activity_ptr_grid);
  12. //得到控件
  13. mPullRefreshListView=(PullToRefreshGridView)findViewById(R.id.pull_refresh_grid);
  14. //初始化数据和数据源
  15. initDatas();
  16. mAdapter=newArrayAdapter<String>(this,R.layout.grid_item,
  17. R.id.id_grid_item_text,mListItems);
  18. mPullRefreshListView.setAdapter(mAdapter);
  19. mPullRefreshListView
  20. .setOnRefreshListener(newOnRefreshListener2<GridView>()
  21. {
  22. @Override
  23. publicvoidonPullDownToRefresh(
  24. PullToRefreshBase<GridView>refreshView)
  25. {
  26. Log.e("TAG","onPullDownToRefresh");//Doworkto
  27. Stringlabel=DateUtils.formatDateTime(
  28. getApplicationContext(),
  29. System.currentTimeMillis(),
  30. DateUtils.FORMAT_SHOW_TIME
  31. |DateUtils.FORMAT_SHOW_DATE
  32. |DateUtils.FORMAT_ABBREV_ALL);
  33. //UpdatetheLastUpdatedLabel
  34. refreshView.getLoadingLayoutProxy()
  35. .setLastUpdatedLabel(label);
  36. newGetDataTask().execute();
  37. }
  38. @Override
  39. publicvoidonPullUpToRefresh(
  40. PullToRefreshBase<GridView>refreshView)
  41. {
  42. Log.e("TAG","onPullUpToRefresh");//Doworktorefresh
  43. //thelisthere.
  44. newGetDataTask().execute();
  45. }
  46. });
  47. }
  48. privatevoidinitDatas()
  49. {
  50. mListItems=newLinkedList<String>();
  51. for(inti=0;i<mItemCount;i++)
  52. {
  53. mListItems.add(i+"");
  54. }
  55. }
  56. privateclassGetDataTaskextendsAsyncTask<Void,Void,Void>
  57. {
  58. @Override
  59. protectedVoiddoInBackground(Void...params)
  60. {
  61. try
  62. {
  63. Thread.sleep(2000);
  64. }catch(InterruptedExceptione)
  65. {
  66. }
  67. returnnull;
  68. }
  69. @Override
  70. protectedvoidonPostExecute(Voidresult)
  71. {
  72. mListItems.add(""+mItemCount++);
  73. mAdapter.notifyDataSetChanged();
  74. //CallonRefreshCompletewhenthelisthasbeenrefreshed.
  75. mPullRefreshListView.onRefreshComplete();
  76. }
  77. }

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

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. ILoadingLayoutstartLabels=mPullRefreshListView
  2. .getLoadingLayoutProxy();
  3. startLabels.setPullLabel("你可劲拉,拉...");//刚下拉时,显示的提示
  4. startLabels.setRefreshingLabel("好嘞,正在刷新...");//刷新时
  5. startLabels.setReleaseLabel("你敢放,我就敢刷新...");//下来达到一定距离时,显示的提示

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

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

现在的效果是:



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

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

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,其他两个依赖不用导入。


更多相关文章

  1. Android:ImageView如何显示网络图片
  2. Android 在 xml中定义图片
  3. 推送功能,(服务器向android客户端推送信息,通知显示在应用栏)(2018-0
  4. Android中.9.png图片的使用过程和原理
  5. Android获取运营商代码
  6. [小代码]在Android和PHP之间的加密/解密,AES加密
  7. android 使用xml selector设置按钮点击效果图片
  8. Android关于SD卡的读写操作及固定图片大小

随机推荐

  1. Android(安卓)studio 页面布局无法显示问
  2. android 文档路径获取
  3. Android软键盘用法实例
  4. Ubuntu amd64 running Android(安卓)SDKs
  5. android 代码 退出程序
  6. Android中Parcelable和Serializable接口
  7. API-TypedArray
  8. Android全局处理异常
  9. android 幕滑动效果
  10. Android(安卓)– 加载图片本缓存到内存与