近来学习了极客学院有关于界面左右滑动的视频,就在这里写个博客,巩固一下知识点,以免忘了。

这里主要介绍界面是如何左右滑动的:

1.首先我们应该设置好将要滑动的三个界面,即leftMenu、middleMenu、rightMenu三个布局,并且放置好它们的位置,这段大家自己在源码中看

2.当位置放好后,就可以开始关于滑动方面的代码。

页面的滑动是通过点的坐标变化距离来进行来实现的。首先我们定义了20dp来确保最小下限滑动的距离,来确定是否进行了滑动;接着就可以进行判断页面的滑动方向,ACTION_DOWN、ACTION_MOVE、ACTION-UP分别对应了手指点击时的按下,移动,抬起时的事件。当按下时,我们获取此时点击点的坐标,随后我们实时获取活动过程中点的滑动坐标,后者减去前者就得到了滑动的距离。当滑动的距离大于TEST_DIS时,如果此时在左右滑动的距离大于在上下滑动距离,就置isLeftRightFragment为true,用于下面的判断。

private Point point = new  Point();
private boolean isLeftRightFragment;
//设置比较值20;当移动小于20dp时,默认没有移动
private static final int TEST_DIS = 20;
private void getTypeEvent(MotionEvent ev) {
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_DOWN:
point.x = (int) ev.getX();
point.y = (int) ev.getY();
break;
case MotionEvent.ACTION_MOVE:
int dX = Math.abs((int)ev.getX() - point.x);
int dY = Math.abs((int)ev.getY() - point.y);
if (dX>TEST_DIS&&dX>dY) { //左右滑动
isLeftRightFragment = true;
isTestCompete = true;
point.x = (int) ev.getX();
point.y = (int) ev.getY();
}else if (dY>TEST_DIS&&dY>dX) { //上下滑动
isLeftRightFragment = false;
isTestCompete = true;
point.x = (int) ev.getX();
point.y = (int) ev.getY();
} break;
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
break;
default:
break;
}
}

3.这部分一些地方我也不太明白,大家看我写的注释吧,没有的地方我也不太懂(比如回调的方法是干嘛的,Action_move下面的两行代码,知道的给我留言啊)。

    public boolean isTestCompete;
public int finalX=0;
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
if (!isTestCompete) {
getTypeEvent(ev);
return true;
}
if (isLeftRightFragment) {
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_MOVE:
int curScrollX = getScrollX(); //获取滑动的距离,向左为负,向右为正
int dis_x = (int) (ev.getX() - point.x);
int expectX = -dis_x + curScrollX;
if (expectX<0) {
          //向左滑动
finalX = Math.max(expectX, -leftMenu.getMeasuredWidth());
}else {
          //向右滑动
finalX = Math.min(expectX, rightMenu.getMeasuredWidth());
}
          //跟随点的滑动,页面滑动
scrollTo(finalX, 0);
point.x= (int) ev.getX();
break; case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
curScrollX = getScrollX();
if (Math.abs(curScrollX) > leftMenu.getMeasuredWidth() >> 1) {
//当滑动的距离大于外布局宽度的一半时
if (curScrollX < 0) {
//向左滑动,前两个参数是开始时坐标,中间是将要移动的距离,200是动画的时间单位毫秒
mScroller.startScroll(curScrollX, 0, -leftMenu.getMeasuredWidth()-curScrollX, 0 ,200);
}else {
//向右滑动
mScroller.startScroll(curScrollX, 0, leftMenu.getMeasuredWidth()-curScrollX, 0,200);
}
}else {
//小于宽度的一半,自动返回原位
mScroller.startScroll(curScrollX, 0, -curScrollX, 0,200);
}
invalidate(); //view的重绘
isLeftRightFragment = false;
isTestCompete = false;
break;
}
}else {
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_UP:
isLeftRightFragment = false;
isTestCompete = false;
break;
default:
break;
}
}
return super.dispatchTouchEvent(ev);
} //回调的方法
@Override
public void computeScroll() {
super.computeScroll();
if (!mScroller.computeScrollOffset()) {
return;
}
int tempX = mScroller.getCurrX();
scrollTo(tempX, 0);
}

4.三完成后,就可以实现页面滑动了,第四步是在页面滑动时,中间的那个布局会变暗。思路是在定义一个布局middleMask,将它的位置和颜色设置完毕。初始化中设置透明度middleMask.setAlpha(0);

再在重写的ScrollTo方法中设置middleMask的透明度跟随移动渐变。

@Override
public void scrollTo(int x, int y) { super.scrollTo(x, y);
     // 设置middlemask的透明度随移动距离渐变
int curX = Math.abs(getScrollX());
//scale的范围为0-1
float scale = curX/ (float)leftMenu.getMeasuredWidth();
middleMask.setAlpha(scale);
}

源码:

MainActivity:

public class MainActivity extends ActionBarActivity {

    private MenuUI menuUI;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
menuUI=new MenuUI(this);
setContentView(menuUI);
}
}

MenuUI:

public class MenuUI extends RelativeLayout{
private Context context;
private FrameLayout leftMenu;
private FrameLayout middleMenu;
private FrameLayout rightMenu;
private FrameLayout middleMask;
private Scroller mScroller; public MenuUI(Context context) {
super(context);
initView(context); } public MenuUI(Context context, AttributeSet attrs) {
super(context, attrs);
initView(context);
}
//初始化
private void initView(Context context){
this.context=context;
mScroller = new Scroller(context, new DecelerateInterpolator()); //第二个参数为渲染器
leftMenu = new FrameLayout(context);
middleMenu = new FrameLayout(context);
rightMenu = new FrameLayout(context);
middleMask = new FrameLayout(context);
leftMenu.setBackgroundColor(Color.YELLOW);
middleMenu.setBackgroundColor(Color.GREEN);
rightMenu.setBackgroundColor(Color.YELLOW);
middleMask.setBackgroundColor(0x88000000);
addView(leftMenu);
addView(middleMenu);
addView(rightMenu);
addView(middleMask);
//设置透明度
middleMask.setAlpha(0);
} @Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
middleMenu.measure(widthMeasureSpec, heightMeasureSpec);
middleMask.measure(widthMeasureSpec, heightMeasureSpec);
//得到菜单的宽的大小
int realWidth = MeasureSpec.getSize(widthMeasureSpec);
//令tempRealtempwidth等于菜单0.8倍的宽,用于设置左右菜单的宽度
int tempRealWidth = MeasureSpec.makeMeasureSpec((int)(realWidth*0.8f),
MeasureSpec.EXACTLY);
//设置左右菜单的大小
leftMenu.measure(tempRealWidth, heightMeasureSpec);
rightMenu.measure(tempRealWidth, heightMeasureSpec);
} public float onMiddleMask(){
System.out.println("透明度:"+middleMask.getAlpha());
return middleMask.getAlpha();
} @Override
public void scrollTo(int x, int y) {

super.scrollTo(x, y);
     // 设置middlemask的透明度随移动距离渐变
int curX = Math.abs(getScrollX());
//scale的范围为0-1
float scale = curX/ (float)leftMenu.getMeasuredWidth();
middleMask.setAlpha(scale);
} @Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
//设置各个布局菜单的位置
super.onLayout(changed, l, t, r, b);
//将中间界面的位置设置为l,t,r,b
middleMenu.layout(l, t, r, b);
middleMask.layout(l, t, r, b);
//根据middleMenu的位置确定其他菜单的位置
leftMenu.layout(l-leftMenu.getMeasuredWidth(), t, r-middleMenu.getMeasuredWidth(), b);
rightMenu.layout(l+middleMenu.getMeasuredWidth(), t, l+rightMenu.getMeasuredWidth()+middleMenu.getMeasuredWidth(), b);
} public boolean isTestCompete;
public int finalX=0;
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
if (!isTestCompete) {
getTypeEvent(ev);
return true;
}
if (isLeftRightFragment) {
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_MOVE:
int curScrollX = getScrollX(); //获取滑动的距离,向左为负,向右为正
int dis_x = (int) (ev.getX() - point.x);
int expectX = -dis_x + curScrollX;
if (expectX<0) {
finalX = Math.max(expectX, -leftMenu.getMeasuredWidth());
}else {
finalX = Math.min(expectX, rightMenu.getMeasuredWidth());
}
scrollTo(finalX, 0);
point.x= (int) ev.getX();
break; case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
curScrollX = getScrollX();
if (Math.abs(curScrollX) > leftMenu.getMeasuredWidth() >> 1) {
//当滑动的距离大于外布局宽度的一半时
if (curScrollX < 0) {
//向左滑动
mScroller.startScroll(curScrollX, 0, -leftMenu.getMeasuredWidth()-curScrollX, 0 ,200);
}else {
//向右滑动
mScroller.startScroll(curScrollX, 0, leftMenu.getMeasuredWidth()-curScrollX, 0,200);
}
}else {
//小于宽度的一半,自动返回原位
mScroller.startScroll(curScrollX, 0, -curScrollX, 0,200);
}
invalidate(); //view的重绘
isLeftRightFragment = false;
isTestCompete = false;
break;
}
}else {
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_UP:
isLeftRightFragment = false;
isTestCompete = false;
break;
default:
break;
}
}
return super.dispatchTouchEvent(ev);
} //回调的方法
@Override
public void computeScroll() {
super.computeScroll();
if (!mScroller.computeScrollOffset()) {
return;
}
int tempX = mScroller.getCurrX();
scrollTo(tempX, 0);
}


private Point point = new Point();
private boolean isLeftRightFragment;
private static final int TEST_DIS = 20;
private void getTypeEvent(MotionEvent ev) {
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_DOWN:
point.x = (int) ev.getX();
point.y = (int) ev.getY();
break;
case MotionEvent.ACTION_MOVE:
int dX = Math.abs((int)ev.getX() - point.x);
int dY = Math.abs((int)ev.getY() - point.y);
if (dX>TEST_DIS&&dX>dY) { //左右滑动
isLeftRightFragment = true;
isTestCompete = true;
point.x = (int) ev.getX();
point.y = (int) ev.getY();
}else if (dY>TEST_DIS&&dY>dX) { //上下滑动
isLeftRightFragment = false;
isTestCompete = true;
point.x = (int) ev.getX();
point.y = (int) ev.getY();
} break;
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
break;
default:
break;
}
}
}

Android之 左右滑动菜单的更多相关文章

  1. Android 侧滑(双向滑动菜单)效果

    下面看看我们如何使用它,达到我们想要的效果 public class MainActivity extends Activity { /** * 双向滑动菜单布局 */ private SliderM ...

  2. 超酷的Android 侧滑(双向滑动菜单)效果

    下面看看我们如何使用它,达到我们想要的效果 public class MainActivity extends Activity { /** * 双向滑动菜单布局 */ private SliderM ...

  3. android实现左右滑动菜单

    直接看效果图:       主要实现代码: package com.way.view; import android.content.Context; import android.media.Den ...

  4. Android 3D滑动菜单完全解析,实现推拉门式的立体特效

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/10471245 在上一篇文章中,我们学习了Camera的基本用法,并借助它们编写了一 ...

  5. Android双向滑动菜单完全解析,教你如何一分钟实现双向滑动特效

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/9671609 记得在很早之前,我写了一篇关于Android滑动菜单的文章,其中有一个 ...

  6. Android 学习笔记之AndBase框架学习(七) SlidingMenu滑动菜单的实现

    PS:努力的往前飞..再累也无所谓.. 学习内容: 1.使用SlidingMenu实现滑动菜单..   SlidingMenu滑动菜单..滑动菜单在绝大多数app中也是存在的..非常的实用..Gith ...

  7. android 自定义ViewGroup和对view进行切图动画实现滑动菜单SlidingMenu

    示意图就不展示了,和上一节的一样,滑动菜单SlidingMenu效果如何大家都比较熟悉,在这里我简单说明一下用自定义ViewGroup来实现. 实现方法:我们自定义一个ViewGroup实现左右滑动, ...

  8. Android 滑动菜单SlidingMenu

    首先我们看下面视图: 这种效果大家都不陌生,网上好多都说是仿人人网的,估计人家牛逼出来的早吧,我也参考了一一些例子,实现起来有三种方法,我下面简单介绍下: 方法一:其实就是对GestureDetect ...

  9. android 自定义ViewGroup和对view进行切图动画实现滑动菜单SlidingMenu[转]

    http://blog.csdn.net/jj120522/article/details/8095852 示意图就不展示了,和上一节的一样,滑动菜单SlidingMenu效果如何大家都比较熟悉,在这 ...

随机推荐

  1. java中的vo 、dto 、dao--转

    原文地址:http://yinchunjian.iteye.com/blog/758196 O是跟数据库里表的映射,一个表对应一个VO DAO是用VO来访问真实的表,对数据库的操作都在DAO中完成 B ...

  2. Network - 工具列表

    Tcpdump homepage - tcpdump wiki - tcpdump Wireshark homepage - wireshark wiki - wireshark Fiddler ho ...

  3. 远程方法调用(RMI)原理与示例

    RMI介绍 远程方法调用(RMI)顾名思义是一台机器上的程序调用另一台机器上的方法.这样可以大致知道RMI是用来干什么的,但是这种理解还不太确切.RMI是Java支撑分布式系统的基石,例如著名的EJB ...

  4. Linux网络管理

    关于OSI七层模型.TCP五层模型.TCP的三次握手.HTTP协议.DNS解析等相关的网络基础知识请参考我整理的一篇博客:http://www.cnblogs.com/wxisme/p/4699049 ...

  5. 一个ListView布局的不断演化

    刚出来工作,就负责一个APP的某块功能的编写,该功能就是类似微博那样的界面.微博界面的编写实际上是非常复杂的,虽然它只是一个ListView,但要想让这个ListView滑得动,是的,在一些配置低的手 ...

  6. SQL Server 2016里TempDb的提升

    几个星期前,SQL Server 2016的最新CTP版本已经发布了:CTP 2.4(目前已经是CTP 3.0).这个预览版相比以前的CTP包含了很多不同的提升.在这篇文章里我会谈下对于SQL Ser ...

  7. Redis设计与实现-附加功能

    发布与订阅 redis订阅分为精准的频道订阅与模糊匹配的模式订阅: redis将所有频道的订阅关系都保存在服务器状态pubsub_channels字典里,键是频道名,值是一个记录所有订阅此频道的客户端 ...

  8. Cookie 和 Session 的区别

    [[ from 老生常谈session,cookie的区别,安全性 ]] 一,为什么session,cookie经常会有人提到 做web开发的人基本上都会用session和cookie,但是仅仅只是会 ...

  9. weblogic 12C 数据源配置出错的解决办法

    驱动程序类名称:   11G 10.3.6与12G数据源配置有很大区别,整个一天才搞明白.   如有疑问可留言:http://www.cnblogs.com/endv/p/4110798.html 配 ...

  10. Linq之select子句

    在Linq中select子句用来指定查询结果的类型和表现形式.Linq查询要么以select子句结尾,要么以group子句结尾. List<UserBaseInfo> users = ne ...