使用ViewDragHelper打造属于自己的DragLayout(抽屉开关 )

news2024/11/26 12:26:36

</com.xujun.drawerLayout.drag.DragLayout>

在代码中若想为其设置监听器,

分别可以监听打开的 时候,关闭的时候,拖动的时候,可以在里面做相应的处理,同时我还加入了 自定义属性可以通过 app:range=”480”或者setRange()方法,即可设置打开抽屉的范围。

mDragLayout.setDragStatusListener(new OnDragStatusChangeListener() {

@Override

public void onOpen() {

Utils.showToast(MainActivity.this, “onOpen”);

// 左面板ListView随机设置一个条目

Random random = new Random();

Log.i(TAG, “onOpen:=” +mDragLayout.getRange());

int nextInt = random.nextInt(50);

mLeftList.smoothScrollToPosition(nextInt);

}

@Override

public void onDraging(float percent) {

Log.d(TAG, "onDraging: " + percent);// 0 -> 1

// 更新图标的透明度

// 1.0 -> 0.0

ViewHelper.setAlpha(mHeaderImage, 1 - percent);

}

@Override

public void onClose() {

Utils.showToast(MainActivity.this, “onClose”);

// 让图标晃动

ObjectAnimator mAnim = ObjectAnimator.ofFloat(mHeaderImage, “translationX”, 15.0f);

mAnim.setInterpolator(new CycleInterpolator(4));

mAnim.setDuration(500);

mAnim.start();

}

});


实现方式


关于ViewDragHelper的一些 讨论

DrawLayout在网上的 实现方式很多,千奇百怪,有一些是直接监听 onTouchEvent事件,处理Activon_Move,Action_Down,Action_up等动作,这样实现的话稍微有点复杂。本篇博客是使用ViewDragHelper来 处理触摸事件和拖拽事件的的,ViewDragHelper是2013Google IO大会推出的,目的是为了给开发者提供一个处理触摸事件,节省开发者的时间。

关于Google官方 关于ViewDragHelper的解释,简单来说就是处理ViewGroup的 触摸事件和拖拽事件

ViewDragHelper is a utility class for writing custom ViewGroups. It offers a number of useful operations and state tracking for allowing a user to drag and reposition views within their parent ViewGroup.

实现思路

  • 1) 我是通过继承FrameLayout来实现的,相比较于继承ViewGroup来实现,这样有一个好处就是省去了自己重写 onMeasure (),onLayout ()方法

  • 2)在构造方法里面初始化mDragHelper,mSensitivity代表打开抽屉的 难易程度,是Float类型,至于mCallback是什么,下面会详细讲,这里先不着急。

public DragLayout(Context context, AttributeSet attrs, int defStyle) {

super(context, attrs, defStyle);

initAttars(context, attrs);

// a.初始化 (通过静态方法)

mDragHelper = ViewDragHelper.create(this, mSensitivity, mCallback);

}

  • 3)重写 onInterceptTouchEvent和onTouchevent 方法 ,将事件交给

// b.传递触摸事件

@Override

public boolean onInterceptTouchEvent(MotionEvent ev) {

// 传递给mDragHelper

return mDragHelper.shouldInterceptTouchEvent(ev);

}

/***

  • 将事件交给mDragHelper处理

  • @param event

  • @return

*/

@Override

public boolean onTouchEvent(MotionEvent event) {

try {

mDragHelper.processTouchEvent(event);

} catch (Exception e) {

e.printStackTrace();

}

// 返回true, 持续接受事件

return true;

}

  • 4)重写onFinishInflate方法,在里面拿到 我们的侧滑菜单mLeftContent和主菜单mMainContent

@Override

protected void onFinishInflate() {

super.onFinishInflate();

// 容错性检查 (至少有俩子View, 子View必须是ViewGroup的子类)

if (getChildCount() < 2) {

throw new IllegalStateException("布局至少有俩孩子. Your ViewGroup must have 2 children at " +

“least.”);

}

if (!(getChildAt(0) instanceof ViewGroup && getChildAt(1) instanceof ViewGroup)) {

throw new IllegalArgumentException("子View必须是ViewGroup的子类. Your children must be an " +

“instance of ViewGroup”);

}

mLeftContent = (ViewGroup) getChildAt(0);

mMainContent = (ViewGroup) getChildAt(1);

}


下面我们一起来看一下这个mCallBack是什么东西

看之前我们需要了解Status和OnDragStatusChangeListener这两个东西。
  • Status代表DrawLayout 当前的状态,是否是打开,关闭还是拖拽。

  • OnDragStatusChangeListener是个监听器,在DrawLayout状态改变的时候会回调相关的方法,方便与外界进行通讯。

  • 我们可以通过 setDragStatusListener(OnDragStatusChangeListener mListener);这个方法设置监听

/**

  • 状态枚举

*/

public static enum Status {

Close, Open, Draging;

}

/**

  • 抽屉开关的监听器

*/

public interface OnDragStatusChangeListener {

void onClose();

void onOpen();

void onDraging(float percent);

}

接下来我们来看ViewDragHelper.Callback几个主要的方法


tryCaptureView(View child, int pointerId)

Called when the user’s input indicates that they want to capture the given child view with the pointer indicated by pointerId.

onViewCaptured(View capturedChild, int activePointerId)

Called when a child view is captured for dragging or settling.

getViewHorizontalDragRange(View child)

Return the magnitude of a draggable child view’s horizontal range of motion in pixels.

clampViewPositionHorizontal(View child, int left, int dx)

Restrict the motion of the dragged child view along the horizontal axis.

onViewPositionChanged(View changedView, int left, int top, int dx, int dy)

Called when the captured view’s position changes as the result of a drag or settle.

onViewReleased(View releasedChild, float xvel, float yvel)

Called when the child view is no longer being actively dragged.

谷歌官方的连接;https://developer.android.com/reference/android/support/v4/widget/ViewDragHelper.Callback.html


下面的代码有关于这几个方法的中文解释,这里就不详细讲解了

ViewDragHelper.Callback mCallback = new ViewDragHelper.Callback() {

// d. 重写事件

// 1. 根据返回结果决定当前child是否可以拖拽

// child 当前被拖拽的View

// pointerId 区分多点触摸的id

@Override

public boolean tryCaptureView(View child, int pointerId) {

Log.d(TAG, "tryCaptureView: " + child);

return mToogle;

}

@Override

public void onViewCaptured(View capturedChild, int activePointerId) {

Log.d(TAG, "onViewCaptured: " + capturedChild);

// 当capturedChild被捕获时,调用.

super.onViewCaptured(capturedChild, activePointerId);

}

@Override

public int getViewHorizontalDragRange(View child) {

// 返回拖拽的范围, 不对拖拽进行真正的限制. 仅仅决定了动画执行速度

Log.i(TAG, “getViewHorizontalDragRange:mRange=” +mRange);

return mRange;

}

// 2. 根据建议值 修正将要移动到的(横向)位置 (重要)

// 此时没有发生真正的移动

public int clampViewPositionHorizontal(View child, int left, int dx) {

// child: 当前拖拽的View

// left 新的位置的建议值, dx 位置变化量

// left = oldLeft + dx;

Log.d(TAG, "clampViewPositionHorizontal: "

  • "oldLeft: " + child.getLeft() + " dx: " + dx + " left: " + left);

if (child == mMainContent) {

left = fixLeft(left);

}

return left;

}

// 3. 当View位置改变的时候, 处理要做的事情 (更新状态, 伴随动画, 重绘界面)

// 此时,View已经发生了位置的改变

@Override

public void onViewPositionChanged(View c​
hangedView, int left, int top,

int dx, int dy) {

// changedView 改变位置的View

// left 新的左边值

// dx 水平方向变化量

super.onViewPositionChanged(changedView, left, top, dx, dy);

Log.d(TAG, "onViewPositionChanged: " + "left: " + left + " dx: " + dx);

int newLeft = left;

if (changedView == mLeftContent) {

// 把当前变化量传递给mMainContent

newLeft = mMainContent.getLeft() + dx;

}

// 进行修正

newLeft = fixLeft(newLeft);

if (changedView == mLeftContent) {

// 当左面板移动之后, 再强制放回去.

mLeftContent.layout(0, 0, 0 + mWidth, 0 + mHeight);

mMainContent.layout(newLeft, 0, newLeft + mWidth, 0 + mHeight);

}

// 更新状态,执行动画

dispatchDragEvent(newLeft);

// 为了兼容低版本, 每次修改值之后, 进行重绘

invalidate();

}

// 4. 当View被释放的时候, 处理的事情(执行动画)

@Override

public void onViewReleased(View releasedChild, float xvel, float yvel) {

// View releasedChild 被释放的子View

// float xvel 水平方向的速度, 向右为+

// float yvel 竖直方向的速度, 向下为+

Log.d(TAG, "onViewReleased: " + "xvel: " + xvel + " yvel: " + yvel);

super.onViewReleased(releasedChild, xvel, yvel);

// 判断执行 关闭/开启

// 先考虑所有开启的情况,剩下的就都是关闭的情况

if (xvel == 0 && mMainContent.getLeft() > mRange / 2.0f) {

open();

} else if (xvel > 0) {

open();

} else {

close();

}

}

@Override

public void onViewDragStateChanged(int state) {

// TODO Auto-generated method stub

super.onViewDragStateChanged(state);

}

};

其实主要思路就是

  • 1)在方法public boolean tryCaptureView(View child, int pointerId)处理那些child可以被捕捉,这里我们返回true表示所有的都可以被捕捉

  • 2)在public int clampViewPositionHorizontal(View child, int left, int dx)方法中根据child返回将要移动的水平位置的偏移量

  • 3)在 void onViewPositionChanged(View changedView, int left, int top, int dx, int dy)方法中处理要做的事情 包括更新状态, 伴随动画, 重绘界面等

public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {

// 进行修正

newLeft = fixLeft(newLeft);

if (changedView == mLeftContent) {

// 当左面板移动之后, 再强制放回去.

mLeftContent.layout(0, 0, 0 + mWidth, 0 + mHeight);

mMainContent.layout(newLeft, 0, newLeft + mWidth, 0 + mHeight);

if (changedView == mLeftContent) {

// 把当前变化量传递给mMainContent

newLeft = mMainContent.getLeft() + dx;

}

}

// 更新状态,执行动画

dispatchDragEvent(newLeft);

// 为了兼容低版本, 每次修改值之后, 进行重绘

invalidate();

}

protected void dispatchDragEvent(int newLeft) {

float percent = newLeft * 1.0f / mRange;

//0.0f -> 1.0f

Log.d(TAG, "percent: " + percent);

if (mListener != null) {

mListener.onDraging(percent);

}

// 更新状态, 执行回调

Status preStatus = mStatus;

mStatus = updateStatus(percent);

if (mStatus != preStatus) {

// 状态发生变化

if (mStatus == Status.Close) {

// 当前变为关闭状态

if (mListener != null) {

mListener.onClose();

}

} else if (mStatus == Status.Open) {

if (mListener != null) {

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取
" + percent);

if (mListener != null) {

mListener.onDraging(percent);

}

// 更新状态, 执行回调

Status preStatus = mStatus;

mStatus = updateStatus(percent);

if (mStatus != preStatus) {

// 状态发生变化

if (mStatus == Status.Close) {

// 当前变为关闭状态

if (mListener != null) {

mListener.onClose();

}

} else if (mStatus == Status.Open) {

if (mListener != null) {

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

[外链图片转存中…(img-ltiQLwvx-1719098040066)]一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1852206.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于JSP技术的家用电器销售网站

开头语&#xff1a;你好呀&#xff0c;我是计算机学长猫哥&#xff01;如果有相关需求&#xff0c;文末可以找到我的联系方式。 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;JSPJava 工具&#xff1a;ECLIPSE、MySQL数据库管理工具、Tomcat 系统展…

MVVM架构详解:前端开发的理想选择

目录 前言1. MVVM架构概述1.1 MVVM架构的定义1.2 MVVM与MVC的区别 2. MVVM架构的核心组件2.1 模型&#xff08;Model&#xff09;2.2 视图&#xff08;View&#xff09;2.3 视图模型&#xff08;ViewModel&#xff09; 3. MVVM架构的优势3.1 分离关注点3.2 提高代码可测试性3.3…

听说你还不会用Dagger2?Dagger2 For Android最佳实践教程

Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Log.d(TAG,chef.cook()); } } 可以看到&#xff0c;在使用Dagger2的时候&#xff0c;使用者的代码会变得非常简洁。但是&#…

Windows10中端口被占用处理方法

前言 在Windows 10中&#xff0c;查看端口被占用情况的方法主要依赖于命令行工具netstat。以下是详细步骤&#xff0c;以及必要的解释和归纳&#xff1a; 打开命令提示符 方法1&#xff1a;使用快捷键Win R&#xff0c;打开“运行”对话框&#xff0c;输入cmd&#xff0c;然…

2024-06-23 编译原理实验3——语义分析

文章目录 一、实验要求二、实验设计三、实验结果四、附完整代码 补录与分享本科实验&#xff0c;以示纪念。 一、实验要求 基于前面的实验&#xff0c;编写一个程序对使用 C—语言书写的源代码进行语义分析&#xff0c;输出语义分析中发现的错误&#xff08;涉及 17 种错误类…

异地局域网纯软件组网如何设置?

在现代社会中&#xff0c;随着企业的不断扩张和分布&#xff0c;异地办公成为一种常见的工作模式。随之而来的是&#xff0c;如何实现异地局域网的组网设置成为了一个挑战。在这种情况下&#xff0c;采用纯软件组网方案是一种有效的解决方案。本文将介绍异地局域网纯软件组网设…

Redis数据库的删除和安装

Redis数据库的删除和安装 1、删除Redis数据库2、下载Redis数据库 1、删除Redis数据库 没有下载过的&#xff0c;可以直接跳到下面的安装过程↓ 我们电脑中如果有下载过Redis数据库&#xff0c;要更换版本的话&#xff0c;其实Redis数据库的删除是比较简单的&#xff0c;打开我…

字节大神强推千页PDF学习笔记,弱化学历问题,已拿意向书字节提前批移动端!

主要问java&#xff0c;以及虚拟机&#xff0c;问了一点android 1.实习项目有关的介绍以及问题回答 2.反射与代理的区别&#xff0c;动态代理&#xff0c;静态代理&#xff0c;二者的区别&#xff0c;以及代理模式的UML图 3.字节码技术 4.虚拟机的双亲委派&#xff0c;以及好…

1Panel应用推荐:Bitwarden开源密码管理器

1Panel&#xff08;github.com/1Panel-dev/1Panel&#xff09;是一款现代化、开源的Linux服务器运维管理面板&#xff0c;它致力于通过开源的方式&#xff0c;帮助用户简化建站与运维管理流程。为了方便广大用户快捷安装部署相关软件应用&#xff0c;1Panel特别开通应用商店&am…

贪心算法—

贪心算法是一种在每一步选择中都采取在当前状态下最好或最优&#xff08;即最有利&#xff09;的选择&#xff0c;从而希望导致结果是全局最好或最优的算法。这种算法并不总是能找到全局最优解&#xff0c;但在某些问题上能提供足够好的解决方案。贪心算法的关键特性包括&#…

判断题无答案22届期末复习

判断: 1-3.结构体变量不能进行整体输入输出。 1-4.不同类型的结构变量之间也可以直接赋值。 1-5假设结构指针p已定义并正确赋值,其指向的结构变量有一个成员是int型的num,则语句 (*p).num = 100; 等价于p->num=1…

1panel OpenResty 设置网站重定向

当我们部署网站时需要&#xff0c;输入"cheshi.com"域名回车&#xff0c;希望他自动跳转https://cheshi.com/indx/&#xff0c;而不是直接跳转https://cheshi.com时可以利用重定向来实现&#xff0c; 这里演示的是 1panel 如何设置&#xff08;nginx 貌似也是这样配…

【大疆pocket3】到手后5个必改初始设置关键点(下)

【大疆pocket3】到手后5个必改初始设置关键点&#xff08;下&#xff09; 一&#xff0c;简介二&#xff0c;必改关键点2.1 数字变焦2.2 慢动作拍摄2.3 神奇的小摇杆2.4 云台模式使用方法&#xff08;默认增稳模式和俯仰角锁定的差异化以及使用场景&#xff09;2.5 云台转向速度…

[HBM] HBM 国产进程, 国产HBM首次研发成功 (202406)

依公知及经验整理&#xff0c;原创保护&#xff0c;禁止转载。 专栏 《深入理解DDR》 AI 的火热浪潮带火了高带宽内存的需求&#xff0c;HBM已是存储市场耀眼的明星。目前市场上还没有国产HBM, 什么时候可以看到国产希望呢&#xff1f; 或许现在可以看到曙光了。 1. 设计端 1…

Linux查看公网IP的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

使用Android Studio导入源码

2-1 基础准备工作 首先你得安装配置了Android Studio&#xff0c;具体不明白的参考《Android Studio入门到精通 》。 接着你得下载好了源码Code&#xff0c;至于如何下载这里不再说明&#xff0c;比较简单&#xff0c;上官网查看就行了。 其次你需要保证源码已经被编译生成了…

压力测试

1.什么是压力测试 压力测试考察当前软硬件环境下系统所能承受的最大负荷并帮助找出系统瓶颈所在。压测都是为了系统在线上的处理能力和稳定性维持在一个标准范围内&#xff0c;做到心中有数 使用压力测试&#xff0c;我们有希望找到很多种用其他测试方法更难发现的错误&#…

写一个坏越的个人天地(一)

好久没写什么大点的项目了,今天想着写一个个人博客好了。I did it! 做个人天地。肯定得有个主题色吧。整个下拉界面,先准备三个色系吧 <el-header class="title"><el-dropdown @command="handleCommand"><span class="el-dropdown-…

DEV-C++与EasyX图形库

&#x1f3ae;&#x1f50a;本文代码适合编译环境&#xff1a;DEV-C&#x1f4bb; ✨&#x1f9e8;温馨提示&#xff1a;此文乃作者心血&#xff0c;如要转载请标注版权&#xff0c;否则视为抄袭&#xff01;&#x1f389;&#x1f3a0; 今天就算是我们Easyx教程的第一篇博文…

使用python下载图片且批量将图片插入word文档

最近有一个小的功能实现&#xff0c;从小某书上下载指定帖子的图片们&#xff0c;然后批量插入到word文档中&#xff0c;便于打印。于是有了以上需求。 一、下载图片 1、首先获取图片们的链接img_urls 首先&#xff0c;获取到的指定帖子的所有信息可以存入一个json文件中&am…