SlidingMenu现在貌似成为了应用程序的标配了。越来越多的软件均使用了SlidingMenu,如人人(下图):
对于这个功能已经有开源项目实现了,其代码在github有,网址为https://github.com/jfeinstein10/SlidingMenu,我也下下来并按官方教程实验了一番,但总有错,没办法只好转战其他方法。
当然,首先是去网上扫荡一下,这里拜读了一篇博文,android 滑动菜单SlidingMenu的实现,它是向右滑动,而我则在其基础上修改了部分代码实现了向左滑动,具体效果如下:
![]() |
![]() |
未滑动或点击设置按钮 | 向右滑动或点击设置按钮 |
<?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" > <LinearLayout android:id="@+id/layout_left" android:layout_width="wrap_content" android:layout_height="fill_parent" android:layout_marginRight="200dp" android:orientation="vertical" > <AbsoluteLayout android:layout_width="fill_parent" android:layout_height="50dp" android:background="@color/grey21" android:padding="10dp" > <TextView android:layout_width="50dp" android:layout_height="wrap_content" android:text="设置" android:textColor="@android:color/background_light" android:textSize="20sp" /> </AbsoluteLayout> <com.example.slidemenutest.MyLinearLayout android:id="@+id/mylaout" android:layout_width="fill_parent" android:layout_height="fill_parent" android:layout_weight="1" > <ListView android:id="@+id/lv_set" android:layout_width="fill_parent" android:layout_height="fill_parent" > </ListView> </com.example.slidemenutest.MyLinearLayout> </LinearLayout> <LinearLayout android:id="@+id/layout_right" android:layout_width="fill_parent" android:layout_height="fill_parent" android:layout_alignParentRight="true" 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="50dp" android:layout_alignParentLeft="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:layout_width="fill_parent" android:layout_height="fill_parent" android:scaleType="fitXY" android:src="@drawable/bg_guide_5" /> </LinearLayout> </RelativeLayout>主界面包含左右两个Layout,左边的初始时将会隐藏,只有向右滑动或点击设置按钮才会显现,当其显现时向左滑动或点击设置按钮,它将会再次隐藏。
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <TextView android:id="@+id/tv_item" android:layout_width="fill_parent" android:layout_height="wrap_content" android:padding="10dp" android:textColor="@color/black" android:textSize="20sp" /> </LinearLayout>
package com.example.slidemenutest; import android.content.Context; import android.util.AttributeSet; import android.view.GestureDetector; import android.view.MotionEvent; import android.view.View; import android.view.GestureDetector.SimpleOnGestureListener; import android.widget.LinearLayout; /*** * 自定义布局文件 */ public class MyLinearLayout extends LinearLayout { private GestureDetector mGestureDetector; View.OnTouchListener mGestureListener; private boolean isLock = false;// 左右移动锁. public OnScrollListener onScrollListener;// 自定义滑动接口 private boolean b;// 拦截touch标识 public MyLinearLayout(Context context) { super(context); } public void setOnScrollListener(OnScrollListener onScrollListener) { this.onScrollListener = onScrollListener; } public MyLinearLayout(Context context, AttributeSet attrs) { super(context, attrs); mGestureDetector = new GestureDetector(new MySimpleGesture()); } /*** * 事件分发 */ @Override public boolean dispatchTouchEvent(MotionEvent ev) { b = mGestureDetector.onTouchEvent(ev);// 获取手势返回值. /*** * 松开时记得处理缩回... */ if (ev.getAction() == MotionEvent.ACTION_UP) { onScrollListener.doLoosen(); } return super.dispatchTouchEvent(ev); } /*** * 事件拦截处理 * 要明白机制,如果返回ture的话,那就是进行拦截,处理自己的ontouch. 返回false的话,那么就会向下传递... */ @Override public boolean onInterceptTouchEvent(MotionEvent ev) { super.onInterceptTouchEvent(ev); return b; } /*** * 事件处理 */ @Override public boolean onTouchEvent(MotionEvent event) { isLock = false; return super.onTouchEvent(event); } /*** * 自定义手势执行 */ class MySimpleGesture extends SimpleOnGestureListener { @Override public boolean onDown(MotionEvent e) { isLock = true; return super.onDown(e); } @Override public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { if (!isLock) onScrollListener.doScroll(distanceX); // 垂直大于水平 if (Math.abs(distanceY) > Math.abs(distanceX)) { return false; } else { return true; } } } /*** * 自定义接口 实现滑动 * */ public interface OnScrollListener { void doScroll(float distanceX);// 滑动... void doLoosen();// 手指松开后执行... } }
package com.example.slidemenutest; import com.example.slidemenutest.MyLinearLayout.OnScrollListener; import android.os.Bundle; import android.app.Activity; import android.os.AsyncTask; import android.util.Log; import android.view.GestureDetector; import android.view.KeyEvent; import android.view.MotionEvent; import android.view.View; import android.view.ViewGroup; 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; public class MainActivity extends Activity implements OnTouchListener, GestureDetector.OnGestureListener, OnItemClickListener { 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 = 0; /** 每次自动展开/收缩的速度 */ private final static int SPEED = 30; private final static int sleep_time = 5; private GestureDetector mGestureDetector;// 手势 private boolean isScrolling = false; private float mScrollX = 0; // 滑块滑动距离 private int window_width;// 屏幕的宽度 private String TAG = "CJL"; private View view = null;// 点击的view private String title[] = { "用户", "同步", "标准", "帮助", "关于"}; private MyLinearLayout mylayout; /*** * 初始化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); mylayout = (MyLinearLayout) findViewById(R.id.mylaout); lv_set.setAdapter(new ArrayAdapter<String>(this, R.layout.item, R.id.tv_item, title)); /*** * 实现该接口 */ mylayout.setOnScrollListener(new OnScrollListener() { @Override public void doScroll(float distanceX) { doScrolling(distanceX); } @Override public void doLoosen() { RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) layout_right .getLayoutParams(); Log.e(TAG, "layoutParams.rightMargin=" + layoutParams.rightMargin); // 缩回去 if (layoutParams.rightMargin < -window_width / 2) { new AsynMove().execute(-SPEED); } else { new AsynMove().execute(SPEED); } } }); // 点击监听 lv_set.setOnItemClickListener(this); layout_right.setOnTouchListener(this); layout_left.setOnTouchListener(this); iv_set.setOnTouchListener(this); mGestureDetector = new GestureDetector(this); // 禁用长按监听 mGestureDetector.setIsLongpressEnabled(false); getMAX_WIDTH(); } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.activity_main); InitView(); } /*** * listview 正在滑动时执行. */ void doScrolling(float distanceX) { isScrolling = true; mScrollX += distanceX;// distanceX:向左为正,右为负 RelativeLayout.LayoutParams layoutParams_left = (RelativeLayout.LayoutParams) layout_left .getLayoutParams(); RelativeLayout.LayoutParams layoutParams_right = (RelativeLayout.LayoutParams) layout_right .getLayoutParams(); layoutParams_right.rightMargin += mScrollX; layoutParams_left.rightMargin = window_width + layoutParams_right.rightMargin; if (layoutParams_right.rightMargin >= 0) { isScrolling = false;// 拖过头了不需要再执行AsynMove了 layoutParams_right.rightMargin = 0; layoutParams_left.rightMargin = window_width; } else if (layoutParams_right.rightMargin <= -MAX_WIDTH) { // 拖过头了不需要再执行AsynMove了 isScrolling = false; layoutParams_left.rightMargin = window_width - MAX_WIDTH; layoutParams_right.rightMargin = -MAX_WIDTH; } layout_left.setLayoutParams(layoutParams_left); layout_right.setLayoutParams(layoutParams_right); } /*** * 获取移动距离 */ void getMAX_WIDTH() { ViewTreeObserver viewTreeObserver = layout_right.getViewTreeObserver(); // 获取控件宽度 viewTreeObserver.addOnPreDrawListener(new OnPreDrawListener() { @Override public boolean onPreDraw() { if (!hasMeasured) { window_width = getWindowManager().getDefaultDisplay() .getWidth(); MAX_WIDTH = layout_left.getWidth(); RelativeLayout.LayoutParams layoutParams_left = (RelativeLayout.LayoutParams) layout_left .getLayoutParams(); RelativeLayout.LayoutParams layoutParams_right = (RelativeLayout.LayoutParams) layout_right .getLayoutParams(); ViewGroup.LayoutParams layoutParams_mylayout = mylayout .getLayoutParams(); // 设置layout_left的初始位置. layoutParams_left.rightMargin = window_width; layout_left.setLayoutParams(layoutParams_left); // 注意:设置lv_set的宽度防止被在移动的时候控件被挤压 layoutParams_mylayout.width = MAX_WIDTH; mylayout.setLayoutParams(layoutParams_mylayout); // 注意: 设置layout_right的宽度。防止被在移动的时候控件被挤压 layoutParams_right.width = window_width; layout_right.setLayoutParams(layoutParams_right); Log.v(TAG, "MAX_WIDTH=" + MAX_WIDTH + "width=" + window_width); hasMeasured = true; } return true; } }); } @Override public boolean onKeyDown(int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0) { RelativeLayout.LayoutParams layoutParams_right = (RelativeLayout.LayoutParams) layout_right .getLayoutParams(); if (layoutParams_right.rightMargin < 0) { new AsynMove().execute(-SPEED); return false; } } return super.onKeyDown(keyCode, event); } @Override public boolean onTouch(View v, MotionEvent event) { view = v;// 记录点击的控件 // 松开的时候要判断,如果不到半屏幕位子则缩回去, if (MotionEvent.ACTION_UP == event.getAction() && isScrolling == true) { RelativeLayout.LayoutParams layoutParams_left = (RelativeLayout.LayoutParams) layout_left .getLayoutParams(); // 缩回去 if (layoutParams_left.rightMargin > window_width - MAX_WIDTH / 2) { new AsynMove().execute(-SPEED); } else { new AsynMove().execute(SPEED); } } return mGestureDetector.onTouchEvent(event); } @Override public boolean onDown(MotionEvent e) { int position = lv_set.pointToPosition((int) e.getX(), (int) e.getY()); if (position != ListView.INVALID_POSITION) { View child = lv_set.getChildAt(position - lv_set.getFirstVisiblePosition()); if (child != null) child.setPressed(true); } mScrollX = 0; isScrolling = false; // 将之改为true,才会传递给onSingleTapUp,不然事件不会向下传递. return true; } @Override public void onShowPress(MotionEvent e) { } /*** * 点击松开执行 */ @Override public boolean onSingleTapUp(MotionEvent e) { // 点击的不是layout_left if (view != null && view == iv_set) { RelativeLayout.LayoutParams layoutParams_right = (RelativeLayout.LayoutParams) layout_right .getLayoutParams(); // 右移动 if (layoutParams_right.rightMargin >= 0) { new AsynMove().execute(SPEED); lv_set.setSelection(0);// 设置为首位. } else { // 左移动 new AsynMove().execute(-SPEED); } } else if (view != null && view == layout_right) { RelativeLayout.LayoutParams layoutParams_right = (android.widget.RelativeLayout.LayoutParams) layout_right .getLayoutParams(); if (layoutParams_right.rightMargin < 0) { // 说明layout_left处于移动最左端状态,这个时候如果点击layout_left应该直接所以原有状态.(更人性化) // 左移动 new AsynMove().execute(-SPEED); } } return true; } /*** * 滑动监听 就是一个点移动到另外一个点. distanceX=后面点x-前面点x,如果大于0,说明后面点在前面点的右边及向右滑动 */ @Override public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { // 执行滑动. doScrolling(distanceX); 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 = 0; if (MAX_WIDTH % Math.abs(params[0]) == 0)// 整除 times = MAX_WIDTH / Math.abs(params[0]); else times = MAX_WIDTH / Math.abs(params[0]) + 1;// 有余数 for (int i = 0; i < times; i++) { publishProgress(params[0]); try { Thread.sleep(sleep_time); } catch (InterruptedException e) { e.printStackTrace(); } } return null; } /** * update UI */ @Override protected void onProgressUpdate(Integer... values) { RelativeLayout.LayoutParams layoutParams_left = (RelativeLayout.LayoutParams) layout_left .getLayoutParams(); RelativeLayout.LayoutParams layoutParams_right = (RelativeLayout.LayoutParams) layout_right .getLayoutParams(); if (values[0] < 0) { // 左移动 layoutParams_left.rightMargin = Math .min(layoutParams_left.rightMargin - values[0], window_width); layoutParams_right.rightMargin = Math.min( layoutParams_right.rightMargin - values[0], 0); } else { // 右移动 layoutParams_left.rightMargin = Math.max( layoutParams_left.rightMargin - values[0], window_width - MAX_WIDTH); layoutParams_right.rightMargin = Math.max( layoutParams_right.rightMargin - values[0], -MAX_WIDTH); System.out.println("left=" + layoutParams_left.rightMargin + ",right=" + layoutParams_right.rightMargin); } layout_left.setLayoutParams(layoutParams_left); layout_right.setLayoutParams(layoutParams_right); } } @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { RelativeLayout.LayoutParams layoutParams_right = (RelativeLayout.LayoutParams) layout_right .getLayoutParams(); // 只要没有滑动则都属于点击 if (layoutParams_right.rightMargin == -MAX_WIDTH) Toast.makeText(MainActivity.this, title[position], 1).show(); } }在研究具体的代码中,其实从向左滑动变为向右滑动还是需要费点事的,你需要理解其具体的内容。