上一篇《Android自己定义组件系列【6】——进阶实践(3)》中补充了关于Android中事件分发的过程知识。这一篇我们接着来分析任老师的《可下拉的PinnedHeaderExpandableListView的实现》。

一、StickyLayout中的OnGiveUpTouchEventListener接口的作用是什么?

    public interface OnGiveUpTouchEventListener {
public boolean giveUpTouchEvent(MotionEvent event);
}

在StickyLayout中还提供了设置监听的方法例如以下:

    public void setOnGiveUpTouchEventListener(OnGiveUpTouchEventListener l) {
mGiveUpTouchEventListener = l;
}

这样的方式事实上是一种钩子方法。在OnGiveUpTouchEventListener中定义了一个抽象方法(未详细实现)giveUpTouchEvent.,然后通过MainActivity继承OnGiveUpTouchEventListener接口来实现详细逻辑。

    @Override
public boolean giveUpTouchEvent(MotionEvent event) {
if (expandableListView.getFirstVisiblePosition() == 0) {
View view = expandableListView.getChildAt(0);
if (view != null && view.getTop() >= 0) {
return true;
}
}
return false;
}

这种方法中的逻辑:取到ExpandableListView中的第一个可见项。假设是它的子View中的第一个则说明如今首先应该滑动上面的Header部分(让其展开)。

这里返回的true和false有什么不同呢?向下看

    @Override
public boolean onInterceptTouchEvent(MotionEvent event) {
int intercepted = 0;
int x = (int) event.getX();
int y = (int) event.getY(); switch (event.getAction()) {
case MotionEvent.ACTION_DOWN: {
mLastXIntercept = x;
mLastYIntercept = y;
mLastX = x;
mLastY = y;
intercepted = 0;
break;
}
case MotionEvent.ACTION_MOVE: {
int deltaX = x - mLastXIntercept;
int deltaY = y - mLastYIntercept;
if (mStatus == STATUS_EXPANDED && deltaY <= -mTouchSlop) {
intercepted = 1;
} else if (mGiveUpTouchEventListener != null) {
if (mGiveUpTouchEventListener.giveUpTouchEvent(event) && deltaY >= mTouchSlop) {
intercepted = 1;
}
}
break;
}
case MotionEvent.ACTION_UP: {
intercepted = 0;
mLastXIntercept = mLastYIntercept = 0;
break;
}
default:
break;
} Log.d(TAG, "intercepted=" + intercepted);
return intercepted != 0;
}

在StickyLayout类中的事件拦截方法的ACTION_MOVE中有这么几句代码:

            if (mStatus == STATUS_EXPANDED && deltaY <= -mTouchSlop) {
intercepted = 1;
} else if (mGiveUpTouchEventListener != null) {
if (mGiveUpTouchEventListener.giveUpTouchEvent(event) && deltaY >= mTouchSlop) {
intercepted = 1;
}
}

如今应该明确了吧,呵呵。STATUS_EXPANDED是一个状态值。意思是如今Header部分是展开的,假设Header部分是收起的则会推断giveUpTouchEvent的返回值。假设giveUpTouchEvent返回true说明列表所有被拉下来了,此时应该将Header部分展开。假设返回false则应该下滑列表而不是展开Header部分。

二、PinnedHeaderExpandableListView对OnScrollLister的实现

    @Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
if (mHeaderView != null && scrollState == SCROLL_STATE_IDLE) {
int firstVisiblePos = getFirstVisiblePosition();
if (firstVisiblePos == 0) {
mHeaderView.layout(0, 0, mHeaderWidth, mHeaderHeight);
}
}
if (mScrollListener != null) {
mScrollListener.onScrollStateChanged(view, scrollState);
}
} @Override
public void onScroll(AbsListView view, int firstVisibleItem,
int visibleItemCount, int totalItemCount) {
if (totalItemCount > 0) {
refreshHeader();
}
if (mScrollListener != null) {
mScrollListener.onScroll(view, firstVisibleItem, visibleItemCount, totalItemCount);
}
}

OnScrollListener是ListView的滚动事件。

在onScrollStateChanged(AbsListView view, int scrollState)中,scrollState有三种状态。各自是:

1、SCROLL_STATE_FLING:開始滚动

2、SCROLL_STATE_TOUCH_SCROLL:正在滚动

3、SCROLL_STATE_IDLE:已经停止

onScroll()方法在列表滚动时一直回调。知道滚动停止才停止回调,另外单击时也回调一次。而OnScrollStateChanged的意思是上面三种状态改变时回调。回调顺序例如以下:

  1. 第1次:scrollState = SCROLL_STATE_TOUCH_SCROLL(1) 正在滚动
  2. 第2次:scrollState = SCROLL_STATE_FLING(2) 手指做了抛的动作(手指离开屏幕前,用力滑了一下)
  3. 第3次:scrollState = SCROLL_STATE_IDLE(0) 停止滚动
  4. 上面的OnScrollStateChanged方法中在滚动停止后再次布局(绘制)了列表的头。在onScroll方法中主要是为了调用前面提到的refreshHeader()方法去回调并刷新列表头。

  5. 四、怎样处理以下列表头将上面列表头顶上去?
  6.     protected void refreshHeader() {
    if (mHeaderView == null) {
    return;
    }
    int firstVisiblePos = getFirstVisiblePosition();
    int pos = firstVisiblePos + 1;
    int firstVisibleGroupPos = getPackedPositionGroup(getExpandableListPosition(firstVisiblePos));
    int group = getPackedPositionGroup(getExpandableListPosition(pos)); if (group == firstVisibleGroupPos + 1) {
    View view = getChildAt(1);
    if (view.getTop() <= mHeaderHeight) {
    int delta = mHeaderHeight - view.getTop();
    mHeaderView.layout(0, -delta, mHeaderWidth, mHeaderHeight - delta);
    }
    } else {
    mHeaderView.layout(0, 0, mHeaderWidth, mHeaderHeight);
    } if (mHeaderUpdateListener != null) {
    mHeaderUpdateListener.updatePinnedHeader(firstVisibleGroupPos);
    }
    }

    refreshHeader()方法中能够看到,先推断上面的是不是多个列表头假设是则又一次设置以下列表头的位置到“标准位置”,这样就感觉有一种顶上去的感觉了。

  7. 三、怎样展开收缩列表
  8.     @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
    int x = (int) ev.getX();
    int y = (int) ev.getY();
    Log.d(TAG, "dispatchTouchEvent");
    int pos = pointToPosition(x, y);
    if (y >= mHeaderView.getTop() && y <= mHeaderView.getBottom()) {
    if (ev.getAction() == MotionEvent.ACTION_DOWN) {
    mActionDownHappened = true;
    } else if (ev.getAction() == MotionEvent.ACTION_UP) {
    int groupPosition = getPackedPositionGroup(getExpandableListPosition(pos));
    if (groupPosition != INVALID_POSITION && mActionDownHappened) {
    if (isGroupExpanded(groupPosition)) {
    collapseGroup(groupPosition);
    } else {
    expandGroup(groupPosition);
    }
    mActionDownHappened = false;
    } }
    return true;
    } return super.dispatchTouchEvent(ev);
    }

    能够看到PinnedHeaderExpandableListView中的事件分发函数中有例如以下代码:

  9. if (y >= mHeaderView.getTop() && y <= mHeaderView.getBottom()) 

    限制高度(在列表头位置)

  10. if (isGroupExpanded(groupPosition)) {
    collapseGroup(groupPosition);
    } else {
    expandGroup(groupPosition);
    }

    假设点击则展开或闭合。

先分析到这里吧,差点儿相同完了。。。

。。。

Android自己定义组件系列【7】——进阶实践(4)的更多相关文章

  1. Android自己定义组件系列【6】——进阶实践(3)

    上一篇<Android自己定义组件系列[5]--进阶实践(2)>继续对任老师的<可下拉的PinnedHeaderExpandableListView的实现>进行了分析,这一篇计 ...

  2. Android自己定义组件系列【5】——进阶实践(2)

    上一篇<Android自己定义组件系列[5]--进阶实践(1)>中对任老师的<可下拉的PinnedHeaderExpandableListView的实现>前一部分进行了实现,这 ...

  3. Android自己定义组件系列【4】——自己定义ViewGroup实现双側滑动

    在上一篇文章<Android自己定义组件系列[3]--自己定义ViewGroup实现側滑>中实现了仿Facebook和人人网的側滑效果,这一篇我们将接着上一篇来实现双面滑动的效果. 1.布 ...

  4. Android自己定义组件系列【3】——自己定义ViewGroup实现側滑

    有关自己定义ViewGroup的文章已经非常多了,我为什么写这篇文章,对于刚開始学习的人或者对自己定义组件比較生疏的朋友尽管能够拿来主义的用了,可是要一步一步的实现和了解当中的过程和原理才干真真脱离别 ...

  5. Android自己定义组件系列【5】——高级实践(1)

    在接下来的几篇文章将任老师的博文<您可以下拉PinnedHeaderExpandableListView实现>骤来具体实现.来学习一下大神的代码并记录一下. 原文出处:http://blo ...

  6. Android自己定义组件系列【1】——自己定义View及ViewGroup

    View类是ViewGroup的父类,ViewGroup具有View的全部特性.ViewGroup主要用来充当View的容器.将当中的View作为自己孩子,并对其进行管理.当然孩子也能够是ViewGr ...

  7. Android自己定义组件系列【2】——Scroller类

    在上一篇中介绍了View类的scrollTo和scrollBy两个方法,对这两个方法不太了解的朋友能够先看<自己定义View及ViewGroup> scrollTo和scrollBy尽管实 ...

  8. Android自己定义组件系列【9】——Canvas绘制折线图

    有时候我们在项目中会遇到使用折线图等图形,Android的开源项目中为我们提供了非常多插件,可是非常多时候我们须要依据详细项目自己定义这些图表,这一篇文章我们一起来看看怎样在Android中使用Can ...

  9. Android自己定义组件系列【8】——面膜文字动画

    我们掩盖文字动画Flash中非经货共同体共同,由于Android应用程序开发人员做你想要做这个动画在应用程序中去?本文中,我们看的是如何自己的定义ImageView来实现让一张文字图片实现文字的遮罩闪 ...

随机推荐

  1. find命令笔记

    find 命令: 文件查找:locate:    非实时,模糊匹配,查找是根据全系统文件数据库进行的:# updatedb, 手动生成文件数据库速度快 find:    实时    精确    支持众 ...

  2. wget命令1(转载)

    Linux系统中的wget是一个下载文件的工具,它用在命令行下.对于Linux用户是必不可少的工具,我们经常要下载一些软件或从远程服务器恢复备份到本地服务器.wget支持HTTP,HTTPS和FTP协 ...

  3. vue.js的devtools安装

    安装 1.github下载地址:https://github.com/vuejs/vue-devtools 2.下载好后进入vue-devtools-master工程  执行npm install - ...

  4. Ubuntu下嵌入式Qt开发环境配置全攻略

    http://qpcwth.blog.163.com/blog/static/20993024620139151424822/ 在安装的过称中,出现一些问题,注意试想: 1.本次开发环境的配置,是基于 ...

  5. angular在ie8下的一个bug

    昨天拿项目在ie8下测试,发现不少bug,其中有一个bug让我很不解,报了一个thead开头的bug,因为已经切回到linux下了,我就不报具体是什么bug了,鼓捣了半天,发现引用angular的应用 ...

  6. Java - 注解 (Annotation)

    Java - 注解 (Annotation)   一.基本的 Annotation     > 使用 Annotation 时要在其前面增加 @符号,并把该 Annotation 当成一个修饰符 ...

  7. 在任务计划程序用Bat命令执行exe程序

    @echo off :open choice /c:123 /m "1:创建,2:终止,3:删除" if errorlevel 3 goto delete if errorleve ...

  8. 《JavaScript 闯关记》之对象

    对象是 JavaScript 的数据类型.它将很多值(原始值或者其他对象)聚合在一起,可通过名字访问这些值,因此我们可以把它看成是从字符串到值的映射.对象是动态的,可以随时新增和删除自有属性.对象除了 ...

  9. background小结

    CSS背景属性Background详解 本文详解了CSS的背景属性Background,包括CSS3中新增的背景属性.如果你是个CSS初学者,还可以查看之前介绍的CSS浮动属性和CSS透明属性详解. ...

  10. 自学Android的第一个小程序(小布局、button点击事件、toast弹出)

    因为上班,学习时间有限,昨晚才根据教程写了一个小程序,今天忙里偷闲写一下如何实现的,来加深一下印象. 首先创建一个Android项目, 通过activity_xxx.xml布局文件来添加组件来达到自己 ...