android群英传 自定义滑动view,Android群英传学习之路-View的滑动

滑动效果是如何产生的?

滑动一个View从本质上来说就是移动一个View,改变其当前的坐标。所以,想要滑动一个View,就必须监听该View的触摸事件,并且根据事件的坐标,不断的改变View的坐标从而实现View的滑动。

Android中的坐标体系

Android中的坐标体系分为两种:

Android坐标系

视图坐标系

Android坐标系

BeuoVGffvDpd72jM.jpg

Android坐标系

从图中我们可以看的出来**在Android坐标系中以屏幕的左上方作为坐标系的原点,从原点向右为X轴的正方向,向下为Y轴的正方向**。在触控事件中使用**getRawX()、getRawY()**来获得Android坐标系的坐标

视图坐标系

puR17miMYRJtL4KE.png

视图坐标系

视图坐标系主要描述的是该View和该View在父视图中的位置关系,和Android坐标系一样,也是以原点右方为X正方向,以原点下方为Y轴正方向,不过这里不是以屏幕的左上角为原点,而是**以父视图的左上角为原点**。在触控事件中使用**getX()、getY()**来获得视图坐标系的坐标

触控事件

在上一篇文章中我们使用到了MotionEvent,这个就是触控事件的封装,首先看一下MotionEvent中封装的事件常量

public static final int ACTION_DOWN = 0; //单点触摸按下动作

public static final int ACTION_UP = 1; //单点触摸离开动作

public static final int ACTION_MOVE = 2; //单点触摸移动动作

public static final int ACTION_CANCEL = 3; //触摸动作取消

public staiic final int ACTION_OUTSIDE = 4; //触摸动作超出边界

public static final int ACTION_POINTER_DOWN = 5; //多点触摸按下动作

public static final int ACTION_POINTER_UP = 6; 多点离开动作

通常情况下我们会在onTouchEvent(MotionEvent event)方法中通过event.getAction()来获取触控事件的类型。知道Android中的坐标系和触控事件之后我们就可以来实现View的滑动了

滑动实现

实现滑动有好多种方法,但是其思想都是一样的。当触摸到View时,记录当前触摸点的坐标,然后当手指移动时,获取到相对于前一个点的偏移量,这样我们就可以通过这个偏移量来进行修改View的坐标,这样不断的重复,就能实现View的滑动

举个例子

我们自定义一个View,让他来跟随我们手指的移动来移动,如图:

4nCHXGhBZgbwlLGh.gif

layout方法

我们知道在View进行绘制的时候会通过onLayout()方法来设置View的显示位置,这里我们通过修改View的left, top, right, bottom四个值来控制View的坐标

public boolean onTouchEvent(MotionEvent event) {

int x = (int) event.getX();

int y = (int) event.getY();

switch (event.getAction()) {

case MotionEvent.ACTION_DOWN:

//记录触摸点的坐标

lastX = x;

lastY = y;

break;

case MotionEvent.ACTION_MOVE:

//计算偏移量

int offsetX = x - lastX;

int offsetY = y - lastY;

//在当前位置基础加上偏移量

layout(

getLeft() + offsetX,

getTop() + offsetY,

getRight() + offsetX,

getBottom() + offsetY

);

break;

}

return true;

}

offsetLeftAndRight()和offsetTopAndBottom

这个方法就是相当于系统提供的一个对左右、上下移动的API的封装,计算好偏移量后,调用方法即可:

offsetLeftAndRight(offsetX);

offsetTopAndBottom(offsetY);

LayoutParams

LayoutParams保存了View的布局参数,所以我们可以改变LayoutParams来动态的改变布局的位置来达到滑动的效果。在程序中使用getLayoutParams()来获取当前View的LayoutParams,这里就和上面一样,获取到偏移量后通过setLayoutParams来设置该View的LayoutParams:

RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams) getLayoutParams();

params.leftMargin = getLeft() + offsetX;

params.topMargin = getTop() + offsetY;

setLayoutParams(params);

需要注意一点,通过getLayoutParams()获取LayoutParams的时候,要根据View的父布局的类型来设置,当然前提还要有一个父布局。还有就是我们可以使用ViewGroup.MarginLayoutParams,这样就不用考虑父布局是什么了

scrollTo、scrollBy

在View中,系统还提供了scrollTo、scrollBy两种方式来改变一个View的位置。

其中scrollTo表示移动到一个具体的坐标点,scrollBy表示移动的偏移量,其实在scrollBy内部也是调用的scrollTo。

和前面的方式一样,获取到偏移量后调用scrollBy来移动view,可是当我们运行程序,拖动View,发现View并没有移动!这是因为scrollTo、scrollBy方法移动的是View的内容,如果在ViewGroup中使用scrollTo、scrollBy的滑,那么移动的就是View了,但是如果在View中使用,那么移动的将是View里面的内容,比如TextView,那么移动的就是文本了。

所以通过上面的分析,我们把代码改为如下:

((View)getParent()).scrollBy(offsetX, offsetY);

再次运行程序,我们会发现动是动了,但是在乱动。因为这里我们其实是在相反的方向移动(这里由于篇幅限制,感兴趣的同学自行查找一下资料),我们把偏移量改为负的之后就会发现滑动正常了。

Scroller

首先来想一个场景:我们需要让一个Button向右面移动100个像素,如果我们用scrollTo/scrollBy的话,那么该View就会很突兀的移动到该点,没有过程,而Scroller则是可以实现平滑的过渡效果的

下面我们把上面跟随手指滑动的例子改一下,我们在松开手指后,View自动回到原点(屏幕左上角)。使用Scroller需要三个步骤

初始化

通过构造方法即可 Scroller mScroller = new Scroller(context);

重写computeScroll()方法,实现滑动

startScroll()开启滑动

代码如下:

case MotionEvent.ACTION_UP:

View viewGroup = (View) getParent();

mScroller.startScroll(

viewGroup.getScrollX(),

viewGroup.getScrollY(),

-viewGroup.getScrollX(),

-viewGroup.getScrollY()

);

invalidate();

break;

----------------------------------------------------------------

@Overrid

public void computeScroll() {

super.computeScroll();

if (mScroller.computeScrollOffset()) {

((View) getParent()).scrollTo(mScroller.getCurrX(), mScroller.getCurrY());

invalidate();

}

}

当我们手指抬起来的时候开启Scroller滑动,获取到当前view的移动距离,然后我们把需要移动的偏移量设置为相反数就ok,最后调用invalidate来通知重绘,从而调用computeScroll()方法。运行程序如下:

zIn1P7eOLR4hzpD1.gif

现在就把这几种View的滑动都说完了,自己动手写一下,感触良多。

最后

爱生活,爱小丽,爱Android

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

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

相关文章

艹,正则表达式!

大家好,我是球哥,转发良许同学的一篇文章。不知道大家有没有被正则表达式支配过的恐惧?看着一行火星文一样的表达式,虽然每一个字符都认识,但放在一起直接就让人蒙圈了~你是不是也有这样的操作,比如你需要使…

Linux Centos安装步骤

在虚拟机VMware10上安装Centos系统步骤 (1)首先需要创建一个虚拟机,选择操作系统,进行一系列的硬件配置,选择Centos系统镜像 虚拟机创建完成 虚拟机创建完成后,重新启动虚拟机,就会自动开始安装CentOS系统的…

通信PK电子,谁牛?

工程师很多都有强迫症,有的是拆解强迫症,有的是排线布线强迫症,至今还记得第一堂焊电路板的课上,有同学把跳线排排焊,期末考获得了光荣的A。论起排线布线的手艺,早前,德国弱电工程师的完美布线让…

[机器视觉] SIFT特征-尺度不变特征理解

SIFT特征-尺度不变特征理解 简介 SIFT,即尺度不变特征变换(Scale-invariant feature transform,SIFT),是用于图像处理领域的一种描述。这种描述具有尺度不变性,可在图像中检测出关键点,是一种局…

harmonyos手机开发者beta,HarmonyOS 手机应用开发者 Beta 版到来,对开发者意味着什么...

北京时间12月16日,HarmonyOS手机开发者Beta活动在北京国贸开启,这场活动现场的人潮涌动也足以说明一切——用户和开发者对HarmonyOS手机应用开发者Beta版期待已久。作者 | 贾凯强头图 | 华为官方提供打破手机单设备限制,为万物互联而生在活动…

你确定不反编译 likely 看看?

大家好,我是写代码的篮球球痴,下面推荐飞哥的一篇文章,觉得对大家很有帮助,希望大家先看看再扔到收藏夹吃灰,转发不转发看大爷们心情咯!今天我给大家分享一个内核中常用的提升性能的小技巧。理解了它对你一…

看完这篇文章,会让你对电容的作用更明白!

电容是电路设计中最为普通常用的器件,是无源元件之一,有源器件简单地说就是需能(电)源的器件叫有源器件,无需能(电)源的器件就是无源器件。电容的作用和用途一般都有好多种,如:在旁路、去耦、滤波、储能方面的作用&…

华为mate40pro 升级鸿蒙,华为Mate40Pro再次曝光,麒麟9000+可升级到鸿蒙OS,这才是华为...

原标题:华为Mate40Pro再次曝光,麒麟9000可升级到鸿蒙OS,这才是华为在如今的手机市场,创新和差异化的设计已经成为消费者最在意的地方,所以加大技术创新和研发投入才是手机厂商们谋求发展的唯一道路。华为手机是一个大家…

10年深圳,回去了

十年前,2012年,我们毕业。毕业后,我们很多人都选择来深圳,那时候的想法很简单,深圳有我们想要的工作,我们很多人都是村里来的孩子,没有背景,家里也没有矿。考不上公务员也不想去做老…

android studio导入eclipse项目各种问题,Android | 导入Eclipse项目到Android studio的问题解决全过程...

D老师今天给了一个Android项目,但是他当时(2015年)用的是eclipse写的而我用的是Android studio,所以在导入项目的过程中出现了一些常见的问题,在这里总结一下解决的问题的过程。1.eclipse项目导入到Android studio的过程:File->…

蜡烛节能车模

❝简 介: 本文给出了几款使用蜡烛驱动的车模。关键词: 蜡烛,车模01 蜡烛车模1.1 蜡烛光电池点上蜡烛就能跑,DIY制作玩具汽车小模型The Energy of the Candle. Car Model[4]▲ 图1.2 蜡烛驱动的智能车模▲ 图1.1.3 光电池车模▲ …

Xen Desktop测试报告

Xen Desktop测试报告 (九叔最近写的一个测试报告,仅供参考,可能结论不适合你们公司) 目的 测试目的:本次测试周期耗时两周(2011.11.14-2011.11.24),主要为了理解思杰的桌面虚拟化&am…

Linux基础提高_系统性能相关命令

w 看系统的负载信息 用于显示已经登陆系统的用户列表,并显示用户正在执行的指令 uptime [rootlocalhost]#uptime 17:26:07 up 9:02, 3 users, load average: 0.00, 0.00, 0.00 能够打印系统总共运行了多长时间和系统的平均负载。uptime命令可以显示的信息显示依次为…

去广州见了我大学老师标哥

标哥是我的大学老师。标哥也是我们很多学生公认上课非常好的老师,最近标哥突然想见见我这个年轻仔,让我去广州见见他,当然,并不是说他不能来深圳,见面的时候标哥特意给我们解释了一遍,我肯定是能理解他的良…

说好的薪资统计

本来是不想把最后的统计发出来的,但是后台很多同学都想看看嵌入式的薪资,我自己也答应了会发出来,所以趁着这个假期再发一次,也让大家做个参考。不想发的原因是因为担心有别有用心的人拿这个薪资来做比较,多少会让大家…

STL面试题

http://blog.csdn.NET/tianya_team/article/details/50753759 1.说说std::vector的底层(存储)机制。 vector就是一个动态数组,里面有一个指针指向一片连续的内存空间,当空间不够装下数据时,会自动申请另一片更大的空间…

实现IButtonControl接口时, PerformClick()方法的代码怎么写,请高手指导!

在做WinForm时,想做一个像Asp.Net里的ImageButton的控件,于是自己写了一个自定义控件,名为:ImageButton,继承PictureBox类,并实现IButtonControl接口,但是实现IButtonControl接口时必须重写IBut…

【转载】 Asp.net Mvc 3当然也不会例

最快的学习方法:找到官方的教程,最好是类似Msdn演练或者冠以Step By Step的,跟着做一遍,做的过程中可以拷贝少量教程中的代码,确保每个步骤、每行代码含义都能够明白,中途随手记录一下,每个步骤…

什么是BSP工程师?

文章转自https://luomuxiaoxiao.com/?p170一、嵌入式系统要明白什么是嵌入式软件工程师,我们先从嵌入式系统(嵌入式设备)说起。维基百科上对嵌入式系统的定义如下:嵌入式系统(Embedded System)&#xff0c…

《STL源码剖析》相关面试题总结

一、STL简介 STL提供六大组件,彼此可以组合套用: 容器 容器就是各种数据结构,我就不多说,看看下面这张图回忆一下就好了,从实现角度看,STL容器是一种class template。 算法 各种常见算法,如sor…