Android开发中RecyclerView模仿探探左右滑动布局功能
程序员文章站
2024-02-20 11:12:40
我在此基础上优化了部分代码, 添加了滑动回调, 可自定义性更强. 并且添加了点击按钮左右滑动的功能.
据说无图都不敢发文章了.
看图:
1:这种功能, 首先需要自...
我在此基础上优化了部分代码, 添加了滑动回调, 可自定义性更强. 并且添加了点击按钮左右滑动的功能.
据说无图都不敢发文章了.
看图:
1:这种功能, 首先需要自己管理布局
继承 recyclerview.layoutmanager
, 显示自己管理布局, 比如最多显示4个view, 并且都是居中显示.
底部的view还需要进行缩放,平移操作.
public class overlaycardlayoutmanager extends recyclerview.layoutmanager { private static final string tag = "swipecard"; public static int max_show_count = 4; public static float scale_gap = 0.05f; public static int trans_y_gap; public overlaycardlayoutmanager(context context) { //平移时, 需要用到的参考值 trans_y_gap = (int) (20 * context.getresources().getdisplaymetrics().density); } @override public recyclerview.layoutparams generatedefaultlayoutparams() { //必须要实现的方法 return new recyclerview.layoutparams(viewgroup.layoutparams.wrap_content, viewgroup.layoutparams.wrap_content); } @override public void onlayoutchildren(recyclerview.recycler recycler, recyclerview.state state) { //在这个方法中进行view的布局操作.此方法会被调用多次. detachandscrapattachedviews(recycler); int itemcount = getitemcount(); if (itemcount < 1) { return; } //top-3view的position int bottomposition; //边界处理 if (itemcount < max_show_count) { bottomposition = 0; } else { bottomposition = itemcount - max_show_count; } //从可见的最底层view开始layout,依次层叠上去 for (int position = bottomposition; position < itemcount; position++) { //1:重recycler的缓存机制中拿到一个view view view = recycler.getviewforposition(position); //2:和自定义viewgroup一样, 需要先addview addview(view); //3:和自定义viewgroup一样, 也需要测量view的大小 measurechildwithmargins(view, 0, 0); int widthspace = getwidth() - getdecoratedmeasuredwidth(view); int heightspace = getheight() - getdecoratedmeasuredheight(view); //4:和自定义viewgroup的onlayout一样, 需要layout view.对view进行布局 //我们在布局时,将childview居中处理,这里也可以改为只水平居中 layoutdecoratedwithmargins(view, widthspace / 2, heightspace / 2, widthspace / 2 + getdecoratedmeasuredwidth(view), heightspace / 2 + getdecoratedmeasuredheight(view)); /** * topview的scale 为1,translationy 0 * 每一级scale相差0.05f,translationy相差7dp左右 * * 观察人人影视的ui,拖动时,topview被拖动,scale不变,一直为1. * top-1view 的scale慢慢变化至1,translation也慢慢恢复0 * top-2view的scale慢慢变化至 top-1view的scale,translation 也慢慢变化只top-1view的translation * top-3view的scale要变化,translation岿然不动 */ //第几层,举例子,count =7, 最后一个topview(6)是第0层, int level = itemcount - position - 1; //如果不需要缩放平移, 那么下面的代码可以注释掉... //除了顶层不需要缩小和位移 if (level > 0 /*&& level < mshowcount - 1*/) { //每一层都需要x方向的缩小 view.setscalex(1 - scale_gap * level); //前n层,依次向下位移和y方向的缩小 if (level < max_show_count - 1) { view.settranslationy(trans_y_gap * level); view.setscaley(1 - scale_gap * level); } else {//第n层在 向下位移和y方向的缩小的成都与 n-1层保持一致 view.settranslationy(trans_y_gap * (level - 1)); view.setscaley(1 - scale_gap * (level - 1)); } } } } }
2:布局好了之后, 就需要监听鼠标事件了
谷歌官方提供了一个itemtouchhelper工具类, 对滑动进行了惨无人道的优越封装, 傻x都能用…
使用方法: new itemtouchhelper(callback).attachtorecyclerview(recyclerview);
就这么简单,
接下来的操作, 都在回调callback里面进行.
public class renrencallback extends itemtouchhelper.simplecallback { private static final string tag = "renren"; private static final int max_rotation = 15; onswipelistener mswipelistener; boolean isswipeanim = false; public renrencallback() { //第一个参数决定可以拖动排序的方向, 这里由于不需要拖动排序,所以传0 //第二个参数决定可以支持滑动的方向,这里设置了上下左右都可以滑动. super(0, itemtouchhelper.down | itemtouchhelper.up | itemtouchhelper.left | itemtouchhelper.right); } public void setswipelistener(onswipelistener swipelistener) { mswipelistener = swipelistener; } //水平方向是否可以被回收掉的阈值 public float getthreshold(recyclerview recyclerview, recyclerview.viewholder viewholder) { //2016 12 26 考虑 探探垂直上下方向滑动,不删除卡片,这里参照源码写死0.5f return recyclerview.getwidth() * /*getswipethreshold(viewholder)*/ 0.5f; } @override public boolean onmove(recyclerview recyclerview, recyclerview.viewholder viewholder, recyclerview.viewholder target) { //由于不支持滑动排序, 所以不需要处理此方法 return false; } @override public void onswiped(recyclerview.viewholder viewholder, int direction) { //当view需要滑动的时候,会回调此方法 //但是这个方法只是告诉你view需要滑动, 并不是对view和adapter进行额外的操作, //所以, 如果你需要实现滑动删除, 那么需要在此方法中remove item等. //我们这里需要对滑动过后的view,进行恢复操作. viewholder.itemview.setrotation(0);//恢复最后一次的旋转状态 if (mswipelistener != null) { mswipelistener.onswipeto(viewholder, 0); } notifylistener(viewholder.getadapterposition(), direction); } private void notifylistener(int position, int direction) { log.w(tag, "onswiped: " + position + " " + direction); if (mswipelistener != null) { mswipelistener.onswiped(position, direction); } } @override public float getswipethreshold(recyclerview.viewholder viewholder) { //滑动的比例达到多少之后, 视为滑动 return 0.3f; } @override public void onchilddraw(canvas c, recyclerview recyclerview, recyclerview.viewholder viewholder, float dx, float dy, int actionstate, boolean iscurrentlyactive) { super.onchilddraw(c, recyclerview, viewholder, dx, dy, actionstate, iscurrentlyactive); //当你在滑动的过程中, 此方法一直会被回调, 就跟ontouch事件一样... //先根据滑动的dx dy 算出现在动画的比例系数fraction float swipevalue = (float) math.sqrt(dx * dx + dy * dy); final float threshold = getthreshold(recyclerview, viewholder); float fraction = swipevalue / threshold; //边界修正 最大为1 if (fraction > 1) { fraction = 1; } else if (fraction < -1) { fraction = -1; } //对每个childview进行缩放 位移 int childcount = recyclerview.getchildcount(); for (int i = 0; i < childcount; i++) { view child = recyclerview.getchildat(i); //第几层,举例子,count =7, 最后一个topview(6)是第0层, int level = childcount - i - 1; if (level > 0) { child.setscalex(1 - scale_gap * level + fraction * scale_gap); if (level < max_show_count - 1) { child.setscaley(1 - scale_gap * level + fraction * scale_gap); child.settranslationy(trans_y_gap * level - fraction * trans_y_gap); } else { //child.settranslationy((float) (mtranslationygap * (level - 1) - fraction * mtranslationygap)); } } else { //最上层 //rotate if (dx < -50) { child.setrotation(-fraction * max_rotation); } else if (dx > 50) { child.setrotation(fraction * max_rotation); } else { child.setrotation(0); } if (mswipelistener != null) { recyclerview.layoutparams params = (recyclerview.layoutparams) child.getlayoutparams(); final int adapterposition = params.getviewadapterposition(); mswipelistener.onswipeto(recyclerview.findviewholderforadapterposition(adapterposition), dx); } } } } //扩展实现:点击按钮实现左滑效果 public void toleft(recyclerview recyclerview) { if (check(recyclerview)) { animto(recyclerview, false); } } //扩展实现:点击按钮实现右滑效果 public void toright(recyclerview recyclerview) { if (check(recyclerview)) { animto(recyclerview, true); } } private void animto(final recyclerview recyclerview, boolean right) { final int position = recyclerview.getadapter().getitemcount() - 1; final view view = recyclerview.findviewholderforadapterposition(position).itemview; translateanimation translateanimation = new translateanimation(animation.relative_to_self, 0, animation.relative_to_self, right ? 1f : -1f, animation.relative_to_self, 0f, animation.relative_to_self, 1.3f); translateanimation.setfillafter(true); translateanimation.setduration(300); translateanimation.setinterpolator(new decelerateinterpolator()); translateanimation.setanimationlistener(new animation.animationlistener() { @override public void onanimationstart(animation animation) { } @override public void onanimationend(animation animation) { isswipeanim = false; recyclerview.removeview(view); notifylistener(position, x > view.getmeasuredwidth() / 2 ? itemtouchhelper.right : itemtouchhelper.left); } @override public void onanimationrepeat(animation animation) { } }); view.startanimation(translateanimation); } private boolean check(recyclerview recyclerview) { if (isswipeanim) { return false; } if (recyclerview == null || recyclerview.getadapter() == null) { return false; } if (recyclerview.getadapter().getitemcount() == 0) { return false; } isswipeanim = true; return true; } public interface onswipelistener { /** * @param direction {@link itemtouchhelper#left} / {@link itemtouchhelper#right} * {@link itemtouchhelper#up} or {@link itemtouchhelper#down}). */ void onswiped(int adapterposition, int direction); /** * 最上层view滑动时回调. * * @param viewholder 最上层的viewholder * @param offset 距离原始位置的偏移量 */ void onswipeto(recyclerview.viewholder viewholder, float offset); } public static class simpleswipecallback implements onswipelistener { /** * {@inheritdoc} */ @override public void onswiped(int adapterposition, int direction) { } /** * {@inheritdoc} */ @override public void onswipeto(recyclerview.viewholder viewholder, float offset) { } } }
看起来不难, 但是真正做的时候, 要处理的地方很多,
并且有些地方要思考很久, 才能实现效果.
总之,做了你才会发现1+1=2的魅力, just do it.
开源地址: https://github.com/angcyo/recyclerlayoutmanager
好了,以上所示是小编给大家分享的android开发中recyclerview模仿探探左右滑动布局功能,希望对大家有所帮助
上一篇: Python中with语句的理解
下一篇: shell脚本---石头剪刀布