GridView上拉更多和下拉刷新的原理分析:
一、要解决的问题:
1:GridView和ListView不一样listView直接提供了addFootView和addHeadView两个方法,GridView是没有这两个方法的.
猜想解决方案:GridView既然是不能直接添加HeadView和FootView的那么可以借助于组合控件.
2:猜想产生的问题
如果借助于组合控件的话那么GridView的滑动和FootView的滑动是独立的,没有相关联(也就是说,GridView的滑动与否和FootVew本身没有任何的联系)
猜想解决方案:如果我们借助于,ScrollView来进行滑动的话那么在ScrollView的底部有了这个FootView的话那么是可以滑动的
3:猜想产生的问题
ScrollView和GridView不一样,ScrollView是不能直接给添加适配器的
猜想解决方案
使用ScrollView中嵌套一个布局不居中包括 GridView和一个底部的FootView
二、ScrollView中嵌套GridView的时候会造成显示不全的问题,需要解决ScrollView计算孩子高度的问题
解决方案:可以重写GridView中的onMeasure()方法,然后给定孩子高度的最大值,然后再设计一个测量的模式.
protected voidonMeasure(intwidthMeasureSpec,intheightMeasureSpec) {
intexpectHeight=MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE>>2,MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
三、需要解决什么时候滑动到了底部和顶部
解决方案:可以通过计算ScrollView孩子的高度,以及当前孩子滑动的顶部的相对的Y的偏移量 getScrollY和当前窗体的高度getHeight.如果getScrollY+getHeight>=孩子的高度,说明滑动到底部了
四、解决问题:ScrollView知道了什么时候滑动到底部,如何将上级的状态
解决方案:直接通过回调的方法将状态在传递到上级页面
五、整合所有的对象
源代码如下:
GridView
public classMyGridViewextendsGridView {
publicMyGridView(Context context, AttributeSet attrs,intdefStyleAttr) {
super(context, attrs, defStyleAttr);
}
publicMyGridView(Context context) {
super(context);
}
publicMyGridView(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
protected voidonMeasure(intwidthMeasureSpec,intheightMeasureSpec) {
intheighMeasureSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, heighMeasureSpec);
}
}
MyScrollViewpublic classMyScrollViewextendsScrollView{
private static finalString LOAD="load"; //表示的是加载数据的状态
private static finalString REFRESH="refresh"; //表示的是需要刷新页面的状态
publicMyScrollView(Context context, AttributeSet attrs) {
super(context, attrs);
}
interfaceICallBack{
public voidnotifactionAbovePage(String tag);
}
ICallBack mICallBack=null;
public voidsetOnCallBackLister(ICallBack mICallBack){
this.mICallBack=mICallBack;
}
@Override
protected voidonScrollChanged(intl,intt,intoldl,intoldt) {
super.onScrollChanged(l, t, oldl, oldt);
if(getChildAt(0)!=null){ //有这个孩子才有意义的
View childView=getChildAt(0);
//第二步:通过孩子获取孩子本身的高度
intchildMeasuredHeight = childView.getMeasuredHeight();
//第三步:获取屏幕顶部的相对y坐标
intscrollY = getScrollY(); //屏幕顶部相对的y坐标
//第四步:获取当前屏幕的高度
intscreenHeight=getHeight();
if(scrollY+screenHeight==childMeasuredHeight){ //说明滑动到底部了?
//这个地方是有遗留的问题的最后还需要处理
if(PullUpLoadView.getDownLoadingOrNot()){
return;
}
mICallBack.notifactionAbovePage(LOAD);
}
}
}
}
PullUpLoadViewpublic classPullUpLoadViewextendsLinearLayout{
private static finalString LOAD="load"; //表示的是加载数据的状态
private static finalString REFRESH="refresh"; //表示的是需要刷新页面的状态
MyGridView mMyGridView;
LinearLayout footView;
MyScrollView mMyScrollView;
static booleandownLoadingOrNot=false; //判断当前是否正在加载数据
publicPullUpLoadView(Context context, AttributeSet attrs) {
super(context, attrs);
initView();
}
/**
* 初始化数据用的
*/
private voidinitView() {
//初始化布局加载器
LayoutInflater mLayoutInflater= (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View view=mLayoutInflater.inflate(R.layout.scroll_grid_footview,this); //这句话的意思是把当前的布局绑定给谁
findView(view);
setCallBack();
}
/**
* 写一个方法来实现回调
*/
private voidsetCallBack() {
mMyScrollView.setOnCallBackLister(newOnCallBack());
}
/**
* 定义一个接口回调到Activity页面
*/
public interfacePullCallBack{
public voidload();
public voidrefresh();
}
PullCallBack mPullCallBack=null;
public voidsetPullCallBacklistener(PullCallBack mPullCallBack ){
this.mPullCallBack=mPullCallBack;
}
/**
* 上一个页面回调的状态
*/
private classOnCallBackimplementsMyScrollView.ICallBack{
@Override
public voidnotifactionAbovePage(String tag) {
if(tag.equals(LOAD)){ //说明需要加载数据
//在这个地方我们书不是需要将这个状态发送到Activity这个页面去
mPullCallBack.load();
handler.sendEmptyMessage(100);
downLoadingOrNot=true; //表示的额是正在加载
}else{ //说明需要刷新数据
}
}
}
//刷新UI
Handler handler=newHandler(){
@Override
public voidhandleMessage(Message msg) {
switch(msg.what){
case100: //表示的是需要显示底部的View
setFootViewVisible();
break;
case101: //表示的是需要隐藏底部的View
setFootViewHide();
downLoadingOrNot=false;
break;
}
}
};
/**
* 获取是否正在加载数据的这个状态
*@return
*/
public static booleangetDownLoadingOrNot(){
returndownLoadingOrNot;
}
/**
* 说明数据已经加载完成
* 这个方法是Activity调用的
*/
public voiddataDone(){
handler.sendEmptyMessage(101);
}
/**
* 找下那个View
*@paramview
*/
private voidfindView(View view) {
mMyGridView= (MyGridView) findViewById(R.id.gridView);
footView= (LinearLayout) findViewById(R.id.footView);
mMyScrollView= (MyScrollView) findViewById(R.id.myScrollView);
}
/**
* 返回GridView
*@return
*/
publicMyGridView getMyGridView(){
returnmMyGridView;
}
/**
* 设置footView影藏
*/
public voidsetFootViewHide(){
footView.setVisibility(View.GONE);
}
/**
* 设置FootView显示
*/
public voidsetFootViewVisible(){
footView.setVisibility(View.VISIBLE);
}
}
xml version="1.0"encoding="utf-8"?>
xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent"> android:id="@+id/myScrollView" android:layout_width="match_parent" android:layout_height="match_parent"> android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > android:id="@+id/gridView" android:layout_width="match_parent" android:layout_height="match_parent" android:numColumns="3" android:horizontalSpacing="10dp" android:verticalSpacing="10dp" /> android:layout_width="match_parent" android:layout_height="50dp" android:background="#ff0077" android:gravity="center" android:id="@+id/footView" > android:layout_width="wrap_content" android:layout_height="wrap_content" style="@android:style/Widget.ProgressBar.Small" /> android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="玩命加载中..." /> LinearLayout> LinearLayout> com.example.mylibrary.MyScrollView> RelativeLayout>