首先我们看下面视图:

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

方法一:其实就是对GestureDetector手势的应用及布局文件的设计.

布局文件main.xml    采用RelativeLayout布局.

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" > <LinearLayout
android:id="@+id/layout_right"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_marginLeft="50dp"
android:orientation="vertical" > <AbsoluteLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:background="@color/grey21"
android:padding="10dp" > <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="设置"
android:textColor="@android:color/background_light"
android:textSize="20sp" />
</AbsoluteLayout> <ListView
android:id="@+id/lv_set"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_weight="" >
</ListView>
</LinearLayout> <LinearLayout
android:id="@+id/layout_left"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:background="@color/white"
android:orientation="vertical" > <RelativeLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:background="@drawable/nav_bg" > <ImageView
android:id="@+id/iv_set"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentRight="true"
android:layout_alignParentTop="true"
android:src="@drawable/nav_setting" /> <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="我"
android:textColor="@android:color/background_light"
android:textSize="20sp" />
</RelativeLayout> <ImageView
android:id="@+id/iv_set"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:scaleType="fitXY"
android:src="@drawable/bg_guide_5" />
</LinearLayout> </RelativeLayout>

ayout_right:这个大布局文件,layout_left:距离左边50dp像素.(我们要移动的是layout_left).

看到这个图我想大家都很清晰了吧,其实:我们就是把layout_left这个布局控件整理向左移动,至于移动多少,就要看layout_right有多宽了。layout_left移动到距离左边的边距就是layout_right的宽及-MAX_WIDTH.相信大家都理解.

布局文件就介绍到这里,下面看代码.

/***
* 初始化view
*/
void InitView() {
layout_left = (LinearLayout) findViewById(R.id.layout_left);
layout_right = (LinearLayout) findViewById(R.id.layout_right);
iv_set = (ImageView) findViewById(R.id.iv_set);
lv_set = (ListView) findViewById(R.id.lv_set);
lv_set.setAdapter(new ArrayAdapter<String>(this, R.layout.item,
R.id.tv_item, title));
lv_set.setOnItemClickListener(new OnItemClickListener() { @Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
Toast.makeText(MainActivity.this, title[position], ).show();
}
});
layout_left.setOnTouchListener(this);
iv_set.setOnTouchListener(this);
mGestureDetector = new GestureDetector(this);
// 禁用长按监听
mGestureDetector.setIsLongpressEnabled(false);
getMAX_WIDTH();
}

这里要对手势进行监听,我想大家都知道怎么做,在这里我要说明一个方法

/***
* 获取移动距离 移动的距离其实就是layout_left的宽度
*/
void getMAX_WIDTH() {
ViewTreeObserver viewTreeObserver = layout_left.getViewTreeObserver();
// 获取控件宽度
viewTreeObserver.addOnPreDrawListener(new OnPreDrawListener() {
@Override
public boolean onPreDraw() {
if (!hasMeasured) {
window_width = getWindowManager().getDefaultDisplay()
.getWidth();
RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_left
.getLayoutParams();
layoutParams.width = window_width;
layout_left.setLayoutParams(layoutParams);
MAX_WIDTH = layout_right.getWidth();
Log.v(TAG, "MAX_WIDTH=" + MAX_WIDTH + "width="
+ window_width);
hasMeasured = true;
}
return true;
}
}); }

在这里我们要获取屏幕的宽度,并将屏幕宽度设置给layout_left这个控件,为什么要这么做呢,因为如果不把该控件宽度写死的话,那么系统将认为layout_left会根据不同环境宽度自动适应,也就是说我们通过layout_left.getLayoutParams动态移动该控件的时候,该控件会伸缩而不是移动。描述的有点模糊,大家请看下面示意图就明白了.

我们不为layout_left定义死宽度效果:

getLayoutParams可以很清楚看到,layout_left被向左拉伸了,并不是我们要的效果.

还有一种解决办法就是我们在配置文件中直接把layout_left宽度写死,不过这样不利于开发,因为分辨率的问题.因此就用ViewTreeObserver进行对layout_left设置宽度.

ViewTreeObserver,这个类主要用于对布局文件的监听.强烈建议同学们参考这篇文章 android ViewTreeObserver详细讲解,相信让你对ViewTreeObserver有更一步的了解.

其他的就是对GestureDetector手势的应用,下面我把代码贴出来:

package com.jj.slidingmenu;

import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.GestureDetector;
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewTreeObserver;
import android.view.ViewTreeObserver.OnPreDrawListener;
import android.view.Window;
import android.view.View.OnTouchListener;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.ListView;
import android.widget.RelativeLayout;
import android.widget.Toast;
import android.widget.LinearLayout.LayoutParams; /***
* 滑动菜单
*
* @author jjhappyforever...
*
*/
public class MainActivity extends Activity implements OnTouchListener,
GestureDetector.OnGestureListener {
private boolean hasMeasured = false;// 是否Measured.
private LinearLayout layout_left;
private LinearLayout layout_right;
private ImageView iv_set;
private ListView lv_set; /** 每次自动展开/收缩的范围 */
private int MAX_WIDTH = ;
/** 每次自动展开/收缩的速度 */
private final static int SPEED = ; private GestureDetector mGestureDetector;// 手势
private boolean isScrolling = false;
private float mScrollX; // 滑块滑动距离
private int window_width;// 屏幕的宽度 private String TAG = "jj"; private String title[] = { "待发送队列", "同步分享设置", "编辑我的资料", "找朋友", "告诉朋友",
"节省流量", "推送设置", "版本更新", "意见反馈", "积分兑换", "精品应用", "常见问题", "退出当前帐号" }; /***
* 初始化view
*/
void InitView() {
layout_left = (LinearLayout) findViewById(R.id.layout_left);
layout_right = (LinearLayout) findViewById(R.id.layout_right);
iv_set = (ImageView) findViewById(R.id.iv_set);
lv_set = (ListView) findViewById(R.id.lv_set);
lv_set.setAdapter(new ArrayAdapter<String>(this, R.layout.item,
R.id.tv_item, title));
lv_set.setOnItemClickListener(new OnItemClickListener() { @Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
Toast.makeText(MainActivity.this, title[position], ).show();
}
});
layout_left.setOnTouchListener(this);
iv_set.setOnTouchListener(this);
mGestureDetector = new GestureDetector(this);
// 禁用长按监听
mGestureDetector.setIsLongpressEnabled(false);
getMAX_WIDTH();
} /***
* 获取移动距离 移动的距离其实就是layout_left的宽度
*/
void getMAX_WIDTH() {
ViewTreeObserver viewTreeObserver = layout_left.getViewTreeObserver();
// 获取控件宽度
viewTreeObserver.addOnPreDrawListener(new OnPreDrawListener() {
@Override
public boolean onPreDraw() {
if (!hasMeasured) {
window_width = getWindowManager().getDefaultDisplay()
.getWidth();
RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_left
.getLayoutParams();
// layoutParams.width = window_width;
layout_left.setLayoutParams(layoutParams);
MAX_WIDTH = layout_right.getWidth();
Log.v(TAG, "MAX_WIDTH=" + MAX_WIDTH + "width="
+ window_width);
hasMeasured = true;
}
return true;
}
}); } @Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.main);
InitView(); } // 返回键
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
if (KeyEvent.KEYCODE_BACK == keyCode && event.getRepeatCount() == ) {
RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_left
.getLayoutParams();
if (layoutParams.leftMargin < ) {
new AsynMove().execute(SPEED);
return false;
}
} return super.onKeyDown(keyCode, event);
} @Override
public boolean onTouch(View v, MotionEvent event) {
// 松开的时候要判断,如果不到半屏幕位子则缩回去,
if (MotionEvent.ACTION_UP == event.getAction() && isScrolling == true) {
RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_left
.getLayoutParams();
// 缩回去
if (layoutParams.leftMargin < -window_width / ) {
new AsynMove().execute(-SPEED);
} else {
new AsynMove().execute(SPEED);
}
} return mGestureDetector.onTouchEvent(event);
} @Override
public boolean onDown(MotionEvent e) {
mScrollX = ;
isScrolling = false;
// 将之改为true,不然事件不会向下传递.
return true;
} @Override
public void onShowPress(MotionEvent e) { } /***
* 点击松开执行
*/
@Override
public boolean onSingleTapUp(MotionEvent e) {
RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_left
.getLayoutParams();
// 左移动
if (layoutParams.leftMargin >= ) {
new AsynMove().execute(-SPEED);
} else {
// 右移动
new AsynMove().execute(SPEED);
} return true;
} /***
* e1 是起点,e2是终点,如果distanceX=e1.x-e2.x>0说明向左滑动。反之亦如此.
*/
@Override
public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX,
float distanceY) {
isScrolling = true;
mScrollX += distanceX;// distanceX:向左为正,右为负
RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_left
.getLayoutParams();
layoutParams.leftMargin -= mScrollX;
if (layoutParams.leftMargin >= ) {
isScrolling = false;// 拖过头了不需要再执行AsynMove了
layoutParams.leftMargin = ; } else if (layoutParams.leftMargin <= -MAX_WIDTH) {
// 拖过头了不需要再执行AsynMove了
isScrolling = false;
layoutParams.leftMargin = -MAX_WIDTH;
}
layout_left.setLayoutParams(layoutParams);
return false;
} @Override
public void onLongPress(MotionEvent e) { } @Override
public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
float velocityY) {
return false;
} class AsynMove extends AsyncTask<Integer, Integer, Void> { @Override
protected Void doInBackground(Integer... params) {
int times = ;
if (MAX_WIDTH % Math.abs(params[]) == )// 整除
times = MAX_WIDTH / Math.abs(params[]);
else
times = MAX_WIDTH / Math.abs(params[]) + ;// 有余数 for (int i = ; i < times; i++) {
publishProgress(params[]);
try {
Thread.sleep(Math.abs(params[]));
} catch (InterruptedException e) {
e.printStackTrace();
}
} return null;
} /**
* update UI
*/
@Override
protected void onProgressUpdate(Integer... values) {
RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_left
.getLayoutParams();
// 右移动
if (values[] > ) {
layoutParams.leftMargin = Math.min(layoutParams.leftMargin
+ values[], );
Log.v(TAG, "移动右" + layoutParams.rightMargin);
} else {
// 左移动
layoutParams.leftMargin = Math.max(layoutParams.leftMargin
+ values[], -MAX_WIDTH);
Log.v(TAG, "移动左" + layoutParams.rightMargin);
}
layout_left.setLayoutParams(layoutParams); } } }

上面代码注释已经很明确,相信大家都看的明白,我就不过多解释了。

效果图:截屏出来有点卡,不过在手机虚拟机上是不卡的.

Android 滑动菜单SlidingMenu的更多相关文章

  1. android 滑动菜单SlidingMenu的实现

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

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

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

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

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

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

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

  5. 如何使用滑动菜单SlidingMenu?

    左側滑: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvanVuaHVhaG91c2U=/font/5a6L5L2T/fontsize/400/fill/I ...

  6. 它们的定义android滑动菜单

    在这里实现了两个滑动菜单效果,的拖放内容的第一部分,菜单拖出像这样的效果感觉,另一种是拖动内容.后面的内容固定菜单.我感觉有层次感的效果,如下面 第一种效果的代码实现例如以下: package com ...

  7. android ——滑动菜单

    一.DrawerLayout是一个拥有两个子控件的布局,第一个子控件是主屏幕中显示的内容,第二个子控件是滑动菜单中显示的内容: <android.support.v4.widget.Drawer ...

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

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

  9. Android滑动菜单使用(MenuDrawer和SlidingMenu)

    项目地址: https://github.com/gokhanakkurt/android-menudrawer   https://github.com/jfeinstein10/SlidingMe ...

随机推荐

  1. Android通过tcpdump抓包(wifi, 2g, 3g都可以)

    http://blog.csdn.net/deng529828/article/details/20646197 1. 手机要有root权限 2. 下载tcpdump   http://www.str ...

  2. IOS-用动画组制作花瓣掉落效果(另附iOS动画图表)

    重要的两个方法:1.动画的数组:animations 2.启动的时间 beginTime 注意:动画组设置了持续时间(duration)可能会导致动画组里面的持续时间不管用 代码如下: #import ...

  3. 开启或关闭SQLSERVER服务的bat文件

    界面如下: 因为电脑的SQLSERVER服务没有自己启动,有时候又需要关闭SQLSERVER服务,就自己在网上找了下教程. 源码如下: @echo offchoice /t 3 /c yn /d y ...

  4. C#上传图片和生成缩略图以及图片预览

    因工作需要,上传图片要增加MIME类型验证和生成较小尺寸的图片用于浏览.根据网上代码加以修改做出如下效果图: 前台代码如下: <html xmlns="http://www.w3.or ...

  5. 一 JavaScript应用开发实践指南

    渐进增强模型 总述: 结构层(Structure),表现层(presentation)与行为层(behavior). HTML,CSS,JavaScript. 只有HTML 的情况下也可以显示内容,C ...

  6. 第一个 MIC shared_memory 程序

    设置Intel编译器的运行环境 在terminal中执行编译器的环境脚本 compilervars.sh: source <install-dir>/bin/compilervars.sh ...

  7. 在JQuery和Js中,如何让ajax执行完后再继续往下执行 async

    async (默认: true) 默认设置下,所有请求均为异步请求.如果需要发送同步请求,请将此选项设置为 false.注意,同步请求将锁住浏览器,用户其它操作必须等待请求完成才可以执行. var t ...

  8. node.js 安装express 提示 command is not found

    肯定有遇到这样的问题. 在执行express -e app时,出现 command is not found 此时需要执行 : $ npm install -g express-generator 你 ...

  9. C# 读取快捷方式指向的文件

    C# 读取快捷方式指向的文件 [Flags()] public enum SLR_FLAGS { SLR_NO_UI = 0x1, SLR_ANY_MATCH = 0x2, SLR_UPDATE = ...

  10. asp.net mvc上传头像加剪裁功能介绍

    正好项目用到上传+剪裁功能,发上来便于以后使用. 我不能告诉你们其实是从博客园扒的前台代码,哈哈. 前端是jquery+fineuploader+jquery.Jcrop 后台是asp.net mvc ...