人人网客户端,有一个侧滑效果不错,就是菜单(menu)和内容(content)可以实现侧滑。

首先,我们先来看一下最终的效果图,有图有真相,这样讲解起来,会更加的清楚。

程序运行时,我们刚开始看到的是内容布局,然后手指向右滑动,可以看到菜单布局出现,最后可以看到整个菜单布局,以及右侧的一部分内容布局。

界面(1)

界面(2)

界面(3)

下面先讲解一下原理:

在一个Activity的布局中有两部分,一部分是菜单(menu)布局,一部分是内容(content)布局。这两个布局横向排列,菜单布局在左,内容布局在右。初始化时,菜单布局向左偏移,以至于能够全部隐藏起来,这样内容布局就能够完全显示在Activity中。然后通过监听手指滑动事件,来改变菜单布局的左便宜距离,从而控制菜单布局的显示和隐藏。原理图如下:

此时,如果将菜单布局的左偏移值改为 0 时,如图:

好,原理到此为止,下面进行代码的编写。

新建RenRenSlideMenuDemo工程,然后写一下布局文件,创建或者打开layout目录下的activity_main.xml文件,加入如下代码:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="fill_parent"android:layout_height="fill_parent"android:orientation="horizontal"tools:context=".MainActivity" ><LinearLayoutandroid:id="@+id/menu"android:layout_width="fill_parent"android:layout_height="fill_parent"android:background="@drawable/menu" ></LinearLayout><LinearLayoutandroid:id="@+id/content"android:layout_width="fill_parent"android:layout_height="fill_parent"android:background="@drawable/content" ></LinearLayout></LinearLayout>

里面的两个LinearLayout布局,分别只包含一张背景图片,是为了让程序的演示更加的简单,达到真正理解原理。

创建或者打开MainActivity,加入如下代码:

package com.example.renrenslidemenudemo;import android.app.Activity;
import android.content.Context;
import android.os.AsyncTask;
import android.os.Bundle;
import android.view.MotionEvent;
import android.view.VelocityTracker;
import android.view.View;
import android.view.View.OnTouchListener;
import android.view.WindowManager;
import android.widget.LinearLayout;public class MainActivity extends Activity implements OnTouchListener {/*** 滚动显示和隐藏menu时,手指滑动需要达到的速度。*/public static final int SNAP_VELOCITY = 200;/*** 屏幕宽度值。*/private int screenWidth;/*** menu最多可以滑动到的左边缘。值由menu布局的宽度来定,marginLeft到达此值之后,不能再减少。*/private int leftEdge;/*** menu最多可以滑动到的右边缘。值恒为0,即marginLeft到达0之后,不能增加。*/private int rightEdge = 0;/*** menu完全显示时,留给content的宽度值。*/private int menuPadding = 80;/*** 主内容的布局。*/private View content;/*** menu的布局。*/private View menu;/*** menu布局的参数,通过此参数来更改leftMargin的值。*/private LinearLayout.LayoutParams menuParams;/*** 记录手指按下时的横坐标。*/private float xDown;/*** 记录手指移动时的横坐标。*/private float xMove;/*** 记录手机抬起时的横坐标。*/private float xUp;/*** menu当前是显示还是隐藏。只有完全显示或隐藏menu时才会更改此值,滑动过程中此值无效。*/private boolean isMenuVisible;/*** 用于计算手指滑动的速度。*/private VelocityTracker mVelocityTracker;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initValues();content.setOnTouchListener(this);}/*** 初始化一些关键性数据。包括获取屏幕的宽度,给content布局重新设置宽度,给menu布局重新设置宽度和偏移距离等。*/private void initValues() {WindowManager window = (WindowManager) getSystemService(Context.WINDOW_SERVICE);screenWidth = window.getDefaultDisplay().getWidth();content = findViewById(R.id.content);menu = findViewById(R.id.menu);menuParams = (LinearLayout.LayoutParams) menu.getLayoutParams();// 将menu的宽度设置为屏幕宽度减去menuPaddingmenuParams.width = screenWidth - menuPadding;// 左边缘的值赋值为menu宽度的负数leftEdge = -menuParams.width;// menu的leftMargin设置为左边缘的值,这样初始化时menu就变为不可见menuParams.leftMargin = leftEdge;// 将content的宽度设置为屏幕宽度content.getLayoutParams().width = screenWidth;}@Overridepublic boolean onTouch(View v, MotionEvent event) {createVelocityTracker(event);switch (event.getAction()) {case MotionEvent.ACTION_DOWN:// 手指按下时,记录按下时的横坐标xDown = event.getRawX();break;case MotionEvent.ACTION_MOVE:// 手指移动时,对比按下时的横坐标,计算出移动的距离,来调整menu的leftMargin值,从而显示和隐藏menuxMove = event.getRawX();int distanceX = (int) (xMove - xDown);if (isMenuVisible) {menuParams.leftMargin = distanceX;} else {menuParams.leftMargin = leftEdge + distanceX;}if (menuParams.leftMargin < leftEdge) {menuParams.leftMargin = leftEdge;} else if (menuParams.leftMargin > rightEdge) {menuParams.leftMargin = rightEdge;}menu.setLayoutParams(menuParams);break;case MotionEvent.ACTION_UP:// 手指抬起时,进行判断当前手势的意图,从而决定是滚动到menu界面,还是滚动到content界面xUp = event.getRawX();if (wantToShowMenu()) {if (shouldScrollToMenu()) {scrollToMenu();} else {scrollToContent();}} else if (wantToShowContent()) {if (shouldScrollToContent()) {scrollToContent();} else {scrollToMenu();}}recycleVelocityTracker();break;}return true;}/*** 判断当前手势的意图是不是想显示content。如果手指移动的距离是负数,且当前menu是可见的,则认为当前手势是想要显示content。* * @return 当前手势想显示content返回true,否则返回false。*/private boolean wantToShowContent() {return xUp - xDown < 0 && isMenuVisible;}/*** 判断当前手势的意图是不是想显示menu。如果手指移动的距离是正数,且当前menu是不可见的,则认为当前手势是想要显示menu。* * @return 当前手势想显示menu返回true,否则返回false。*/private boolean wantToShowMenu() {return xUp - xDown > 0 && !isMenuVisible;}/*** 判断是否应该滚动将menu展示出来。如果手指移动距离大于屏幕的1/2,或者手指移动速度大于SNAP_VELOCITY,* 就认为应该滚动将menu展示出来。* * @return 如果应该滚动将menu展示出来返回true,否则返回false。*/private boolean shouldScrollToMenu() {return xUp - xDown > screenWidth / 2 || getScrollVelocity() > SNAP_VELOCITY;}/*** 判断是否应该滚动将content展示出来。如果手指移动距离加上menuPadding大于屏幕的1/2,* 或者手指移动速度大于SNAP_VELOCITY, 就认为应该滚动将content展示出来。* * @return 如果应该滚动将content展示出来返回true,否则返回false。*/private boolean shouldScrollToContent() {return xDown - xUp + menuPadding > screenWidth / 2 || getScrollVelocity() > SNAP_VELOCITY;}/*** 将屏幕滚动到menu界面,滚动速度设定为30.*/private void scrollToMenu() {new ScrollTask().execute(30);}/*** 将屏幕滚动到content界面,滚动速度设定为-30.*/private void scrollToContent() {new ScrollTask().execute(-30);}/*** 创建VelocityTracker对象,并将触摸content界面的滑动事件加入到VelocityTracker当中。* * @param event*            content界面的滑动事件*/private void createVelocityTracker(MotionEvent event) {if (mVelocityTracker == null) {mVelocityTracker = VelocityTracker.obtain();}mVelocityTracker.addMovement(event);}/*** 获取手指在content界面滑动的速度。* * @return 滑动速度,以每秒钟移动了多少像素值为单位。*/private int getScrollVelocity() {mVelocityTracker.computeCurrentVelocity(1000);int velocity = (int) mVelocityTracker.getXVelocity();return Math.abs(velocity);}/*** 回收VelocityTracker对象。*/private void recycleVelocityTracker() {mVelocityTracker.recycle();mVelocityTracker = null;}class ScrollTask extends AsyncTask<Integer, Integer, Integer> {@Overrideprotected Integer doInBackground(Integer... speed) {int leftMargin = menuParams.leftMargin;// 根据传入的速度来滚动界面,当滚动到达左边界或右边界时,跳出循环。while (true) {leftMargin = leftMargin + speed[0];if (leftMargin > rightEdge) {leftMargin = rightEdge;break;}if (leftMargin < leftEdge) {leftMargin = leftEdge;break;}publishProgress(leftMargin);// 为了要有滚动效果产生,每次循环使线程睡眠20毫秒,这样肉眼才能够看到滚动动画。sleep(20);}if (speed[0] > 0) {isMenuVisible = true;} else {isMenuVisible = false;}return leftMargin;}@Overrideprotected void onProgressUpdate(Integer... leftMargin) {menuParams.leftMargin = leftMargin[0];menu.setLayoutParams(menuParams);}@Overrideprotected void onPostExecute(Integer leftMargin) {menuParams.leftMargin = leftMargin;menu.setLayoutParams(menuParams);}}/*** 使当前线程睡眠指定的毫秒数。* * @param millis*            指定当前线程睡眠多久,以毫秒为单位*/private void sleep(long millis) {try {Thread.sleep(millis);} catch (InterruptedException e) {e.printStackTrace();}}
}

下面对这个MainActivity解释一下:首先初始化时调用initValues方法,在里面将内容布局的宽度设定为屏幕宽度,菜单布局的宽度设定为屏幕宽度减去menuPadding值,这样能保证在显示菜单布局时,右边能显示一部分的内容布局。

之后给内容布局设置监听事件,这样当手指在内容布局上滑动的时候,就会触发onTouch事件。在onTouch事件里面,根据手指滑动的距离会改变菜单布局的左偏移量,从而控制菜单布局的显示和隐藏。当手指离开屏幕的时候,会判断应该滑动到菜单布局还是内容布局,判断依据是根据手指滑动的距离或者滑动的速度,细节可以看代码中的注释。

最后给出AndroidManifest.xml的代码,都是自动生成的:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"package="com.example.renrenslidemenudemo"android:versionCode="1"android:versionName="1.0" ><uses-sdkandroid:minSdkVersion="8"android:targetSdkVersion="8" /><applicationandroid:allowBackup="true"android:icon="@drawable/ic_launcher"android:label="@string/app_name"android:theme="@android:style/Theme.NoTitleBar" ><activityandroid:name="com.example.renrenslidemenudemo.MainActivity"android:label="@string/app_name" ><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity></application></manifest>

此时,已讲解完毕。

防人人网客户端侧滑效果,简单实现Android滑动菜单特效相关推荐

  1. Android滑动菜单框架完全解析,教你如何一分钟实现滑动菜单特效

    转载请注明出处:http://blog.csdn.net/sinyu890807/article/details/8744400 之前我向大家介绍了史上最简单的滑动菜单的实现方式,相信大家都还记得.如 ...

  2. JS+CSS防FLASH效果竖向可折叠的滑动菜单

    代码简介:很不错的效果,类似动画慢慢的折叠. 代码内容: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN&q ...

  3. css下拉菜单出现下划线,简单带下划线跟随效果的CSS3下拉菜单特效

    简要教程 这是一款使用纯CSS3制作的带下划线跟随效果的下拉菜单特效.该下拉菜单通过CSS3 transform和transition来制作下划线跟随效果和下拉菜单效果. 使用方法 HTML结构 该下 ...

  4. Android 滑动菜单框架--SwipeMenuListView框架完全解析

    SwipeMenuListView(滑动菜单) A swipe menu for ListView.--一个非常好的滑动菜单开源项目. Demo 一.简介 看了挺长时间的自定义View和事件分发,想找 ...

  5. html炫酷的导航栏效果,纯CSS3创意导航菜单特效

    这是一款非常有创意的纯CSS3导航菜单特效.该导航菜单主要通过CSS3 transform和transition方法实现效果,非常简单.该特效由進擊的燊提供. 使用方法 HTML结构 该导航菜单使用一 ...

  6. android滑动菜单图标,Android实现简单底部导航栏 Android仿微信滑动切换效果

    Android仿微信滑动切换最终实现效果: 大体思路: 1. 主要使用两个自定义View配合实现; 底部图标加文字为一个自定义view,底部导航栏为一个载体,根据需要来添加底部图标; 2. 底部导航栏 ...

  7. CSS3手机端侧滑菜单 4种滑动菜单特效

    详细内容请点击 这是一款基于CSS3的手机端侧滑菜单,一共有4种侧滑动画特效.这款CSS3菜单的特点是鼠标划过时即可以各种动画方式展开和隐藏菜单项,该动画方式由CSS3中的transition-del ...

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

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

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

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

最新文章

  1. 【原创】Github团队协作之Pull请求
  2. y7000p内存是一个16还是8+8_16层蜜瓜蛋糕,每日限量8件,只卖一个夏天!
  3. 实践作业三 结对项目
  4. 长时间使用s档有危害吗_空调长时间不清洗竟有这么多危害 你知道吗?
  5. laravel获取当前的url以及当前的基础域名方法汇总
  6. mac 强制退出程序_Mac OS系统如何强制退出应用程序
  7. ubuntu安装sasl失败 - 解决方法
  8. unittest框架怎么生成测试报告?
  9. Java多个PDF文件合并成一个PDF文件
  10. reset()方法 submit()方法
  11. 红米6.0系统如何无root激活xposed框架的教程
  12. 数据库时间慢了14个小时,Mybatis说,这个锅我不背~
  13. 一键平仓含挂单全商品版脚本.mq4
  14. java识别照片是彩色还是黑白照
  15. SAS系统学习之初探
  16. 带你轻松玩转“高颜值”3D图表
  17. android 查看cpu 工具6,Android 之CPU监控命令
  18. 山东大学程序设计思维与实践 四月模拟:TT与可怜的猫
  19. 关于《冬天时我喜欢靠近温暖的事》这首歌 (民谣在路上)
  20. android 调用系统照相机拍照后保存到系统相册

热门文章

  1. git detached HEAD 问题处理
  2. 用计算机找到自己的另一半,心理专家教你,如何利用心理学找到自己的另一半?...
  3. 浪涌保护器+电涌保护器+SPD的选用指南
  4. 华为云重大变革:CloudAI 升至华为第四大 BG ,火力全开
  5. 我像“小马过河”一样升级了我的开源系统
  6. 【愚公系列】2021年12月 二十三种设计模式(七)-桥接模式(Bridge Pattern)
  7. C#——向工具箱里添加控件的方法
  8. 4.5 游戏效果 - Gameplay Effects
  9. 日本电视连续剧《阿信》主题歌歌词—永远相信
  10. css:a:visited限制