百行代码打造高级联动特效

2024-02-29 08:08

本文主要是介绍百行代码打造高级联动特效,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前两天突然看到一个联动效果蛮不错的,虽然不知道具体什么地方会用到。不过也随手鲁了一个。效果如下图:
这里写图片描述

效果是不是挺好玩的~~。那么让我们接下来一步步的分析一下。

思路

首先,让我们想象一下如何实现?自定义view?自定义layout?还是什么?首先这是多层布局的嵌套。肯定会发生的就是事件拦截和分发。我们想像一下Coordinatorlayout+Appbarlayout+CollapsingToolbarLayout和我们现在的效果是不是差不多?只是上者是一个这个是多个而已。于是我的想法就是鲁一个behavior来实现此效果。

自定义layout

首先我们把需要的title和Recyclerview放入一个自定义布局里面。这是为了简单我们的代码。所以把功能放到自定义layout里面。代码如下:

 private void initial(Context context, AttributeSet attrs) {LayoutInflater.from(context).inflate(R.layout.cardlayout, this);TypedArray array = context.obtainStyledAttributes(attrs, R.styleable.SWCardLayout);TextView title = (TextView) findViewById(R.id.title);title.setText(array.getText(R.styleable.SWCardLayout_text));title.setBackgroundColor(array.getColor(R.styleable.SWCardLayout_backgroundcolor,Color.BLACK));RecyclerView recyclerView = (RecyclerView) findViewById(R.id.recycler);recyclerView.setLayoutManager(new LinearLayoutManager(context));NumberAdapter numberAdapter = new NumberAdapter(context);recyclerView.setAdapter(numberAdapter);array.recycle();}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {if (w != oldw || h != oldh) {titleheight = findViewById(R.id.title).getMeasuredHeight();}}public int getTitleheight() {return titleheight;}

这边代码我就不多说了。返回title的高度为了后面的计算。

随手鲁一个behavior

我们先自定义一个behavior放入我们刚刚写的自定义layout。代码如下:


public class CardBehavior extends Behavior<SWCardLayout> {private int childheight = 0;public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout, SWCardLayout child,View directTargetChild, View target, int nestedScrollAxes) {return (nestedScrollAxes & ViewCompat.SCROLL_AXIS_VERTICAL) != 0 && directTargetChild == child;}
}

至于nestedscrolling的这个方法我就不多少了。前面有几篇讲到这个的。
现在我们需要思考的是如何摆放他们的位置。其实behavior都帮我们处理好了。那就是onLayoutChild和OnmesureChild。控制他们的宽高和布局显示都在这2个方法里面。接下来我们看下这块的代码如何去写:

//控制子控件的onlayout和onmesure@Overridepublic boolean onLayoutChild(CoordinatorLayout parent, SWCardLayout child, int layoutDirection) {//按照默认的情况控制,会导致child重叠parent.onLayoutChild(child, layoutDirection);//控制顶部的偏移量(上一个距离顶部的边距+自身title高度)SWCardLayout frontChild = getFrontChild(parent, child);if (frontChild != null) {int offset = frontChild.getTop() + frontChild.getTitleheight();child.offsetTopAndBottom(offset);}childheight = child.getTop();return true;}@Overridepublic boolean onMeasureChild(CoordinatorLayout parent, SWCardLayout child, int parentWidthMeasureSpec, int widthUsed, int parentHeightMeasureSpec, int heightUsed) {//防止布局的重绘int offset = getChildOffset(parent, child);int height = View.MeasureSpec.getSize(parentHeightMeasureSpec) - offset;child.measure(parentWidthMeasureSpec, View.MeasureSpec.makeMeasureSpec(height, View.MeasureSpec.EXACTLY));return true;}

我们来分析下,先从onmesure说起。我们需要得到他的偏移量,也就是上一个的title的高度。如果是若干个布局,那就是若干的title的高度。计算title的偏移量的代码很简单,就是叠加title的高度:

    private int getChildOffset(CoordinatorLayout parent, SWCardLayout child) {int offset = 0;for (int i = 0; i < parent.getChildCount(); i++) {View view = parent.getChildAt(i);if (view != child) {if (view instanceof SWCardLayout) {offset += ((SWCardLayout) view).getTitleheight();}}}return offset;}

得到它的偏移量之后我们直接通过子view的onmesure布局扔进去。所以这个不是很难。

现在我们继续看onlayoutchild方法。我注释也写的比较明确了。如果单独通过layout放置的话,child会重叠在一起。那么计算方法就是计算前一个child顶部的距离加上他自身的title的高度。那么我们来看看如何得到前一个child。其他代码很简单,遍历父view就可以。代码如下:

 //得到前一个childprivate SWCardLayout getFrontChild(CoordinatorLayout parent, SWCardLayout child) {int index = parent.indexOfChild(child);for (int i = index - 1; i >= 0; i--) {View view = parent.getChildAt(i);if (view instanceof SWCardLayout) {return (SWCardLayout) view;}}return null;}

我们先来看下效果。
这里写图片描述

我们会发现现在是滚动不了的。那么既然显示都显示出来 了。还怕他滚不了。看老夫随手一段代码分分钟让他滚起来。

那么滚动的处理在那边呢。当然是在onnestprescroll里面。nestedscrolling的一个方法里面处理。这个我前面有介绍过。不了解nestedscrolling可以先翻翻之前的文章。

那么我们应该怎么处理呢?还是老套路,计算偏移量。我们需要先处理自己本身的滑动,在处理其他的滑动来实现联动效果。具体代码如下:

  //为了处理联动public void onNestedPreScroll(CoordinatorLayout coordinatorLayout, SWCardLayout child,View target, int dx, int dy, int[] consumed) {//先处理自己的int min = childheight;int max = childheight + child.getHeight() - child.getTitleheight();int top = child.getTop();int offset = Math.min(Math.max(top - dy, min), max) - top;child.offsetTopAndBottom(offset);consumed[1] = -offset;Log.i("----------", "onNestedPreScroll: " + consumed[1]);//再处理其他的if (consumed[1] == 0) {return;} else if (consumed[1] > 0) {//上滑SWCardLayout current = child;SWCardLayout cardLayout = getFrontChild(coordinatorLayout, child);while (cardLayout != null) {int layoutoffset = getHeightOffset(cardLayout, current);cardLayout.offsetTopAndBottom(-layoutoffset);current = cardLayout;cardLayout = getFrontChild(coordinatorLayout, current);}} else if (consumed[1] < 0) {//下滑SWCardLayout current = child;SWCardLayout cardLayout = getNextChild(coordinatorLayout, child);while (cardLayout != null) {int layoutoffset = getHeightOffset(current, cardLayout);cardLayout.offsetTopAndBottom(layoutoffset);current = cardLayout;cardLayout = getNextChild(coordinatorLayout, current);}}}

上滑的时候我们需要得到前一个的swcardlayout带着他滚动,下滑的时候我们需要得到下一个child进行滚动,得到下一个child的方法其实他之前得到前一个的方法是差不多的。代码如下:

private SWCardLayout getNextChild(CoordinatorLayout parent, SWCardLayout child) {int index = parent.indexOfChild(child);for (int i = index + 1; i < parent.getChildCount(); i++) {View view = parent.getChildAt(i);if (view instanceof SWCardLayout) {return (SWCardLayout) view;}}return null;}

那么就这么写完了。我们先来看下效果:
这里写图片描述

额。。。似乎翻车了。不过就看着图,我们知道应该是偏移量的问题搞错了。我们的偏移量必须要大于0应该。那么我们修改下代码。改成如下的:

else if (consumed[1] > 0) {//上滑SWCardLayout current = child;SWCardLayout cardLayout = getFrontChild(coordinatorLayout, child);while (cardLayout != null) {int layoutoffset = getHeightOffset(cardLayout, current);if (layoutoffset > 0) {cardLayout.offsetTopAndBottom(-layoutoffset);}current = cardLayout;cardLayout = getFrontChild(coordinatorLayout, current);}} else if (consumed[1] < 0) {//下滑SWCardLayout current = child;SWCardLayout cardLayout = getNextChild(coordinatorLayout, child);while (cardLayout != null) {int layoutoffset = getHeightOffset(current, cardLayout);if (layoutoffset > 0) {cardLayout.offsetTopAndBottom(layoutoffset);}current = cardLayout;cardLayout = getNextChild(coordinatorLayout, current);}}

判断下偏移量的值在做child的offset设置。我们在来看一下效果。
这里写图片描述

代码

一个成功的联动效果就这么出来了。最后我放出自定义behavior的整体代码:


public class CardBehavior extends Behavior<SWCardLayout> {private int childheight = 0;public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout, SWCardLayout child,View directTargetChild, View target, int nestedScrollAxes) {return (nestedScrollAxes & ViewCompat.SCROLL_AXIS_VERTICAL) != 0 && directTargetChild == child;}//为了处理联动public void onNestedPreScroll(CoordinatorLayout coordinatorLayout, SWCardLayout child,View target, int dx, int dy, int[] consumed) {//先处理自己的int min = childheight;int max = childheight + child.getHeight() - child.getTitleheight();int top = child.getTop();int offset = Math.min(Math.max(top - dy, min), max) - top;child.offsetTopAndBottom(offset);consumed[1] = -offset;Log.i("----------", "onNestedPreScroll: " + consumed[1]);//再处理其他的if (consumed[1] == 0) {return;} else if (consumed[1] > 0) {//上滑SWCardLayout current = child;SWCardLayout cardLayout = getFrontChild(coordinatorLayout, child);while (cardLayout != null) {int layoutoffset = getHeightOffset(cardLayout, current);if (layoutoffset > 0) {cardLayout.offsetTopAndBottom(-layoutoffset);}current = cardLayout;cardLayout = getFrontChild(coordinatorLayout, current);}} else if (consumed[1] < 0) {//下滑SWCardLayout current = child;SWCardLayout cardLayout = getNextChild(coordinatorLayout, child);while (cardLayout != null) {int layoutoffset = getHeightOffset(current, cardLayout);if (layoutoffset > 0) {cardLayout.offsetTopAndBottom(layoutoffset);}current = cardLayout;cardLayout = getNextChild(coordinatorLayout, current);}}}private SWCardLayout getNextChild(CoordinatorLayout parent, SWCardLayout child) {int index = parent.indexOfChild(child);for (int i = index + 1; i < parent.getChildCount(); i++) {View view = parent.getChildAt(i);if (view instanceof SWCardLayout) {return (SWCardLayout) view;}}return null;}private int getHeightOffset(SWCardLayout top, SWCardLayout bottom) {return top.getTop() + top.getTitleheight() - bottom.getTop();}//控制子控件的onlayout和onmesure@Overridepublic boolean onLayoutChild(CoordinatorLayout parent, SWCardLayout child, int layoutDirection) {//按照默认的情况控制,会导致child重叠parent.onLayoutChild(child, layoutDirection);//控制顶部的偏移量(上一个距离顶部的边距+自身title高度)SWCardLayout frontChild = getFrontChild(parent, child);if (frontChild != null) {int offset = frontChild.getTop() + frontChild.getTitleheight();child.offsetTopAndBottom(offset);}childheight = child.getTop();return true;}//得到前一个childprivate SWCardLayout getFrontChild(CoordinatorLayout parent, SWCardLayout child) {int index = parent.indexOfChild(child);for (int i = index - 1; i >= 0; i--) {View view = parent.getChildAt(i);if (view instanceof SWCardLayout) {return (SWCardLayout) view;}}return null;}@Overridepublic boolean onMeasureChild(CoordinatorLayout parent, SWCardLayout child, int parentWidthMeasureSpec, int widthUsed, int parentHeightMeasureSpec, int heightUsed) {//防止布局的重绘int offset = getChildOffset(parent, child);int height = View.MeasureSpec.getSize(parentHeightMeasureSpec) - offset;child.measure(parentWidthMeasureSpec, View.MeasureSpec.makeMeasureSpec(height, View.MeasureSpec.EXACTLY));return true;}private int getChildOffset(CoordinatorLayout parent, SWCardLayout child) {int offset = 0;for (int i = 0; i < parent.getChildCount(); i++) {View view = parent.getChildAt(i);if (view != child) {if (view instanceof SWCardLayout) {offset += ((SWCardLayout) view).getTitleheight();}}}return offset;}
}

当然那个SWCardLayout你们可以自己进行封装。想怎么玩怎么玩~。就是这么6。

这篇关于百行代码打造高级联动特效的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/758090

相关文章

python使用Akshare与Streamlit实现股票估值分析教程(图文代码)

《python使用Akshare与Streamlit实现股票估值分析教程(图文代码)》入职测试中的一道题,要求:从Akshare下载某一个股票近十年的财务报表包括,资产负债表,利润表,现金流量表,保存... 目录一、前言二、核心知识点梳理1、Akshare数据获取2、Pandas数据处理3、Matplotl

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

精选20个好玩又实用的的Python实战项目(有图文代码)

《精选20个好玩又实用的的Python实战项目(有图文代码)》文章介绍了20个实用Python项目,涵盖游戏开发、工具应用、图像处理、机器学习等,使用Tkinter、PIL、OpenCV、Kivy等库... 目录① 猜字游戏② 闹钟③ 骰子模拟器④ 二维码⑤ 语言检测⑥ 加密和解密⑦ URL缩短⑧ 音乐播放

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java

Python办公自动化实战之打造智能邮件发送工具

《Python办公自动化实战之打造智能邮件发送工具》在数字化办公场景中,邮件自动化是提升工作效率的关键技能,本文将演示如何使用Python的smtplib和email库构建一个支持图文混排,多附件,多... 目录前言一、基础配置:搭建邮件发送框架1.1 邮箱服务准备1.2 核心库导入1.3 基础发送函数二、

Python中你不知道的gzip高级用法分享

《Python中你不知道的gzip高级用法分享》在当今大数据时代,数据存储和传输成本已成为每个开发者必须考虑的问题,Python内置的gzip模块提供了一种简单高效的解决方案,下面小编就来和大家详细讲... 目录前言:为什么数据压缩如此重要1. gzip 模块基础介绍2. 基本压缩与解压缩操作2.1 压缩文

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示