`

当滑动到顶部和底部时,实现Item的分离效果的ListView

阅读更多
拉动ListView,Item之间的间距会变大,释放后恢复原样;
package cn.tangdada.tangbang.widget;

import android.annotation.TargetApi;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.drawable.BitmapDrawable;
import android.os.Build;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.animation.AccelerateInterpolator;
import android.widget.AbsListView;
import android.widget.ListView;
import cn.tangdada.tangbang.R;

import com.nineoldandroids.view.ViewPropertyAnimator;

/**
 * 当滑动到顶部和底部时,实现Item的分离效果。
 * 
 * @author pythoner
 * 
 */
@TargetApi(Build.VERSION_CODES.HONEYCOMB)
public class PullSeparateListView extends ListView
{
    /**
     * 最大滑动距离
     */
    private static final float MAX_DELTAY = 100;

    /**
     * 分离后恢复的动画时长
     */
    private static final long SEPARATE_RECOVER_DURATION = 200;

    /**
     * 摩擦系数
     */
    private static final float FACTOR = 0.25f;

    /**
     * 按下x的缩放比例
     */
    private static final float SCALEX = 1.0f;

    /**
     * 按下y的缩放比例
     */
    private static final float SCALEY = 1.0f;

    /**
     * 展开全部
     */
    private boolean separateAll;

    /**
     * 到达边界时,滑动的起始位置
     */
    private float startY;

    /**
     * 按下时的View
     */
    private View downView;

    private int touchSlop;

    private boolean separate = false;

    private boolean showDownAnim;

    /**
     * 原始按下位置(在所有Item中的位置)
     */
    private int originDownPosition;

    /**
     * 按下的位置(在屏幕中的位置)
     */
    private int downPosition;

    /**
     * 上次滑动的位置,用于判断方向
     */
    private float preY;

    private float deltaY;

    private boolean reachTop, reachBottom, move;

    private OnScrollListener mScrollListener;

    public PullSeparateListView(Context context, AttributeSet attrs)
    {
        super(context, attrs);
        TypedArray t = context.obtainStyledAttributes(attrs, R.styleable.PullSeparateListView);
        separateAll = t.getBoolean(R.styleable.PullSeparateListView_separate_all, false);
        showDownAnim = t.getBoolean(R.styleable.PullSeparateListView_showDownAnim, true);
        t.recycle();
        init();
    }

    public PullSeparateListView(Context context, AttributeSet attrs, int defStyle)
    {
        super(context, attrs, defStyle);
        init();
    }

    public PullSeparateListView(Context context)
    {
        super(context);
        init();
    }

    @SuppressWarnings("deprecation")
    private void init()
    {
        // 不知道怎么让divider和selector和Item一起移动,所以去除,需要自己加分割线
        this.setDivider(null);
        this.setSelector(new BitmapDrawable());

        touchSlop = ViewConfiguration.get(getContext()).getScaledTouchSlop();
        super.setOnScrollListener(listener);
    }

    /**
     * 是否全部分离
     * 
     * @param separateAll 如果为true,那么全部都会分离。否则的话,如果是顶部下拉,只有点击位置之前的Item会分离</br> 如果是底部上拉,则只有点击位置之后的item会分离。默认为false
     */
    public void setSeparateAll(boolean separateAll)
    {
        this.separateAll = separateAll;
    }

    public boolean isSeparateAll()
    {
        return separateAll;
    }

    /**
     * 设置是否显示按下的Item的动画效果
     * 
     * @param showDownAnim 默认为true
     */
    public void setShowDownAnim(boolean showDownAnim)
    {
        this.showDownAnim = showDownAnim;
    }

    public boolean isShowDownAnim()
    {
        return showDownAnim;
    }

    public void setOnScrollListener(OnScrollListener l)
    {
        mScrollListener = l;
    }

    // 核心代码
    @Override
    public boolean dispatchTouchEvent(MotionEvent ev)
    {
        float currentY = ev.getY();
        switch (ev.getAction())
        {
            case MotionEvent.ACTION_DOWN:
                float downX = ev.getX();
                float downY = ev.getY();
                // 记录按下位置,当isSeparateAll()返回false时,会用到
                originDownPosition = pointToPosition((int) downX, (int) downY);
                downPosition = originDownPosition - getFirstVisiblePosition();
                if (showDownAnim)
                {
                    performDownAnim(downPosition);
                }
                break;
            case MotionEvent.ACTION_MOVE:
                // 记录到达顶部或底部时手指的位置
                if (!separate)
                {
                    startY = currentY;
                }
                deltaY = currentY - startY;

                // 到达顶部
                if (reachTop)
                {
                    if (!separateFromTop(currentY))
                    {
                        return super.dispatchTouchEvent(ev);
                    }
                    return false;
                }
                // 到达底部
                if (reachBottom)
                {
                    if (!separateFromBottom(currentY))
                    {
                        return super.dispatchTouchEvent(ev);
                    }
                    return false;
                }
                preY = currentY;
                break;
            case MotionEvent.ACTION_CANCEL:
            case MotionEvent.ACTION_UP:
                preY = 0;
                recoverDownView();
                if (separate)
                {
                    separate = false;
                    recoverSeparate();
                    // 移动,不响应点击事件
                    if (move)
                    {
                        move = false;
                        return false;
                    }
                }
                break;
        }
        return super.dispatchTouchEvent(ev);
    }

    private boolean separateFromTop(float currentY)
    {
        // 不能放在外部,否则在顶部滑动没有Fling效果
        if (deltaY > touchSlop)
        {
            move = true;
        }
        separate = true;
        // 超过滑动允许的最大距离,则将起始位置向下移
        if (deltaY > MAX_DELTAY)
        {
            startY = currentY - MAX_DELTAY;
            // 超过最大距离时,出现overScroll效果//有问题
            // return super.dispatchTouchEvent(ev);
        }
        else if (deltaY < 0)
        { // 为负值时(说明反方向超过了起始位置startY)归0
            deltaY = 0;
            separate = false;
        }

        if (deltaY <= MAX_DELTAY)
        {
            for (int index = 0; index < getChildCount(); index++)
            {
                View child = getChildAt(index);
                int multiple = index;
                if (!separateAll)
                {
                    if (index > downPosition)
                    {
                        multiple = Math.max(1, downPosition);
                    }
                }
                float distance = multiple * deltaY * FACTOR;
                child.setTranslationY(distance);
            }
            // 向分离方向的反方向滑动,但位置还未复原时
            if (deltaY != 0 && currentY - preY < 0)
            {
                return true;
            }
            // deltaY=0,说明位置已经复原,然后交给父类处理
        }
        if (deltaY == 0)
        {
            return false;
        }
        return true;
    }

    private boolean separateFromBottom(float currentY)
    {
        if (Math.abs(deltaY) > touchSlop)
        {
            move = true;
        }
        separate = true;
        // 超过滑动允许的最大距离,则将起始位置向上移
        if (Math.abs(deltaY) > MAX_DELTAY)
        {
            startY = currentY + MAX_DELTAY;
            // 超过最大距离时,出现overScroll效果
            // return super.dispatchTouchEvent(ev);
        }
        else if (deltaY > 0)
        { // 为正值时(说明反方向移动超过起始位置startY),归0
            deltaY = 0;
            separate = false;
        }
        if (Math.abs(deltaY) <= MAX_DELTAY)
        {
            int visibleCount = getChildCount();
            for (int inedex = 0; inedex < visibleCount; inedex++)
            {
                View child = getChildAt(inedex);
                int multiple = visibleCount - inedex - 1;
                if (!separateAll)
                {
                    if (inedex < downPosition)
                    {
                        multiple = Math.max(1, visibleCount - downPosition - 1);
                    }
                }
                float distance = multiple * deltaY * FACTOR;
                child.setTranslationY(distance);
            }
            // 向分离方向的反方向滑动,但位置还未复原时
            if (deltaY != 0 && currentY - preY > 0)
            {
                return true;
            }
            // deltaY=0,说明位置已经复原,然后交给父类处理
            if (deltaY == 0)
            {
                return false;
            }
        }
        return true;
    }

    /**
     * 恢复
     */
    private void recoverSeparate()
    {
        for (int i = 0; i < getChildCount(); i++)
        {
            View child = getChildAt(i);
            ViewPropertyAnimator.animate(child).translationY(0).setDuration(SEPARATE_RECOVER_DURATION).setInterpolator(new AccelerateInterpolator());
        }
    }

    /**
     * 按下的动画
     * 
     * @param downPosition 在屏幕中的位置
     */
    private void performDownAnim(int downPosition)
    {
        downView = getChildAt(downPosition);
        if (downView != null)
        {
            ViewPropertyAnimator.animate(downView).scaleX(SCALEX).scaleY(SCALEY).setDuration(50).setInterpolator(new AccelerateInterpolator());
        }
    }

    /**
     * 恢复点击的View
     */
    private void recoverDownView()
    {
        if (showDownAnim && downView != null)
        {
            ViewPropertyAnimator.animate(downView).scaleX(1f).scaleY(1f).setDuration(separate ? SEPARATE_RECOVER_DURATION : 100)
                    .setInterpolator(new AccelerateInterpolator());
        }
    }

    private OnScrollListener listener = new OnScrollListener()
    {
        @Override
        public void onScrollStateChanged(AbsListView view, int scrollState)
        {
            if (mScrollListener != null)
            {
                mScrollListener.onScrollStateChanged(view, scrollState);
            }
        }

        @Override
        public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount)
        {
            if (mScrollListener != null)
            {
                mScrollListener.onScroll(view, firstVisibleItem, visibleItemCount, totalItemCount);
            }

            // 是否到达顶部
            if (firstVisibleItem == 0)
            {
                View firstView = getChildAt(firstVisibleItem);
                if (firstView != null && (firstView.getTop() + getPaddingTop()) >= 0)
                {
                    downPosition = originDownPosition;
                    reachTop = true;
                }
                else
                {
                    reachTop = false;
                }
            }
            else
            {
                reachTop = false;
            }
            // 是否到达底部
            if (firstVisibleItem + visibleItemCount == getCount())
            {
                View lastView = getChildAt(visibleItemCount - 1);
                if (lastView != null && (lastView.getBottom() + getPaddingBottom()) <= getHeight() && getCount() > getChildCount())
                {
                    downPosition = originDownPosition - firstVisibleItem;
                    reachBottom = true;
                }
                else
                {
                    reachBottom = false;
                }
            }
            else
            {
                reachBottom = false;
            }
        }
    };

    /**
     * 是否到达顶部
     * 
     * @return
     */
    @Deprecated
    protected boolean isReachTopBound()
    {
        int firstVisPos = getFirstVisiblePosition();
        if (firstVisPos == 0)
        {
            View firstView = getChildAt(firstVisPos);
            if (firstView != null && firstView.getTop() >= 0)
            {
                return true;
            }
            else
            {
                return false;
            }
        }
        return false;
    }

    /**
     * 是否到达底部
     * 
     * @return
     */
    @Deprecated
    protected boolean isReachBottomBound()
    {
        int lastVisPos = getLastVisiblePosition();
        if (lastVisPos == getCount() - 1)
        {
            View lastView = getChildAt(getChildCount() - 1);
            if (lastView != null && lastView.getBottom() <= getHeight() && getCount() > getChildCount())
            {
                return true;
            }
            else
            {
                return false;
            }
        }
        return false;
    }

}



自定义属性:
<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="PullSeparateListView">
        <attr name="separate_all" format="boolean"/>
        <attr name="showDownAnim" format="boolean"/>
    </declare-styleable>
</resources>


高仿墨迹天气下拉拉伸图片,释放后返回效果
http://blog.csdn.net/wu928320442/article/details/44198157


ListView 滚动到边缘时,item产生缩放效果
http://www.jcodecraeer.com/a/opensource/2015/1204/3748.html
分享到:
评论
1 楼 qtfwd 2015-03-31  
lZ 有没有完整代码?

相关推荐

    解决scrollView和listview滚动冲突,实现listview滑动到顶端和底部之后还能拖拽一定的距离,松开后返回

    接下来,我们要实现ListView滑动到底部或顶部后还能继续拖拽一段距离,松手后自动回弹的效果。这可以通过监听ListView的滑动状态并结合Scroller类来实现。以下是一般步骤: 1. **监听滑动状态**:通过重写ListView...

    listview的item滑动效果

    本篇文章将深入探讨如何在ListView的Item上实现滑动效果,以此提升用户体验。 首先,我们需要了解ListView的工作原理。ListView采用滚动复用机制,即只有当前屏幕可见的Item会被实例化,当Item滑出屏幕时,会被回收...

    ListView滑动到底部时自动加载新的内容

    总之,实现ListView滑动到底部自动加载新内容的功能,需要监听滚动事件、处理数据加载和更新UI,同时考虑用户体验和性能优化。随着Android SDK的发展,开发者可以选择更先进的组件和工具来实现这一功能,如...

    ListView 滑动删除 item

    总结来说,实现ListView的滑动删除功能涉及到自定义Adapter、监听滑动事件、处理删除逻辑、提供视觉反馈以及可能的撤销机制。这个过程需要开发者具备扎实的Android基础,理解ListView的工作原理,以及熟悉第三方库的...

    Android 实现ListView滚动到底部自动加载数据

    当用户滚动到ListView的底部时,自动加载更多数据的功能通常被称为“无限滚动”或“下拉加载更多”。这种功能可以提升用户体验,避免用户手动翻页。下面我们将详细讨论如何在Android中实现这个功能。 首先,你需要...

    ListView滑动删除item

    当用户在ListView的Item上进行滑动操作时,我们需要捕捉这一手势并作出响应。这通常通过设置OnTouchListener或使用第三方库如SwipeRefreshLayout、SwipeMenuListView等来实现。 二、自定义适配器 1. 创建自定义的...

    仿微信的ListView item可左右滑动效果

    这个效果的实现主要涉及到自定义适配器(Adapter)和滑动监听器(SwipeListener)的运用。 首先,我们来理解“仿微信的ListView item可左右滑动效果”的核心原理。这个功能通常通过以下步骤实现: 1. **自定义...

    android listview item 左右滑动

    本篇文章将详细介绍如何在Android中为ListView的item实现左右滑动效果,特别是针对"滑动删除"的实现方法。 首先,我们要了解Android中的SwipeListView。SwipeListView是基于ListView的一个扩展,它提供了滑动触发...

    Android ListView 中item的左右滑动动画效果的实现

    公司有个项目要用到类似手机QQ聊天记录列表ListView左右滑动后改变item的效果,网上没找到好的代码,偶然在安卓巴士的开源站http://d.apkbus.com/里面找到了SwipeToDismiss的源码...改成自己的显示另一个view的效果就行...

    listview item特效之滑动删除Item

    首先,要实现滑动删除效果,我们需要对ListView的每个Item进行自定义。自定义ListView项通常涉及到创建一个新的布局文件,其中包含需要展示的元素,并在Adapter中使用这个布局。Adapter是连接ListView和数据源的关键...

    Android中判断listview是否滑动到顶部和底部的实现方法

    这样的实现方式可以确保在ListView真正滚动到顶部或底部时才触发相应的操作,避免了因为部分item显示而误触发的情况。同时,这种方法也适用于其他基于ListView的组件,如RecyclerView。 需要注意的是,为了提高性能...

    Listview 滑动 贴合效果

    分析和学习这个示例,可以帮助你更好地理解和实现滑动贴合效果。在实际开发中,可以根据具体需求进行适当的修改和定制,以满足不同场景的应用。 总之,实现ListView的滑动贴合效果是一个结合了事件监听、视图操作、...

    Listview左右滑动删除item

    总的来说,实现"Listview左右滑动删除item"涉及到了Android手势检测、视图拖动、数据绑定、动画和UI交互等多个方面,需要开发者具备扎实的Android基础和良好的编程习惯。通过以上步骤,你可以为你的应用创建一个高效...

    ListView像左滑动Item显示删除按钮

    总的来说,实现ListView中Item左滑显示删除按钮涉及多个步骤,包括自定义ListView、滑动布局设计、适配器逻辑以及动画效果的添加。使用第三方库可以简化这个过程,但自定义实现可以更好地满足特定需求。在实际开发中...

    自定义控件ListView上下滑动效果(开源)

    本知识点主要聚焦于如何创建一个自定义的ListView,实现上下滑动时的特殊效果。开源项目“ZrcListView”提供了一个很好的示例。 1. **自定义ListView**: 自定义ListView主要是通过继承Android提供的AbsListView或者...

    android listview item 左右滑动删除

    总结来说,实现Android ListView Item的左右滑动删除涉及自定义Item布局、滑动手势检测、Adapter数据更新、动画效果、性能优化等多个方面。通过合理的代码设计和第三方库的利用,可以轻松地为应用增添这一实用的交互...

    listview 滑动删除Item

    6. **状态保存与恢复**: 由于ListView会复用Item,所以必须保存和恢复每个Item的滑动状态。可以在getView()方法中根据数据源的状态设置SwipeableRow的初始显示。 7. **性能优化**: 考虑到性能问题,滑动删除功能应...

    Android 自定义ListView实现底部分页刷新与顶部下拉刷新

    Android 自定义ListView实现底部分页刷新与顶部下拉刷新 一.ListView 底部分页加载 整个底部分页加载,主要分一下几步: 1.加载底部自定义View; 2.响应OnScrollListener监听事件,onScroll方法记录最后可见的View ...

    基于ListView实现头部、底部视差效果

    在本教程中,我们将探讨如何在ListView中实现头部和底部的视差效果。 首先,我们需要创建一个自定义的ListView,这可以通过继承AbsListView或者直接使用ListView并重写其滚动方法来实现。关键在于监听ListView的...

    Android-滑动listview标题置顶listview吸顶效果

    这个效果通常是指当用户向上滑动ListView时,某些特定视图(如广告或者搜索框)会固定在屏幕顶部,直到用户再次滑动下去才隐藏。实现这个效果的方法有多种,一种是使用布局嵌套,将吸顶视图放在ListView之上,通过...

Global site tag (gtag.js) - Google Analytics