Android开源AndroidSideMenu实现抽屉和侧滑菜单

 更新时间:2021年11月14日 13:16:57   作者:ShouCeng  
这篇文章主要为大家详细介绍了Android开源AndroidSideMenu实现抽屉和侧滑菜单,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

AndroidSideMenu能够让你轻而易举地创建侧滑菜单。需要注意的是,该项目自身并不提供任何创建菜单的工具,因此,开发者可以自由创建内部菜单。

核心类如下:

/* 
 * Copyright dmitry.zaicew@gmail.com Dmitry Zaitsev 
 * 
 * Licensed under the Apache License, Version 2.0 (the "License"); 
 * you may not use this file except in compliance with the License. 
 * You may obtain a copy of the License at 
 * 
 * http://www.apache.org/licenses/LICENSE-2.0 
 * 
 * Unless required by applicable law or agreed to in writing, software 
 * distributed under the License is distributed on an "AS IS" BASIS, 
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 
 * See the License for the specific language governing permissions and 
 * limitations under the License. 
 */ 
 
package com.agimind.widget; 
 
import java.util.LinkedList; 
import java.util.Queue; 
 
import android.content.Context; 
import android.graphics.Bitmap; 
import android.graphics.Canvas; 
import android.graphics.Color; 
import android.graphics.Paint; 
import android.graphics.PorterDuff.Mode; 
import android.graphics.Rect; 
import android.graphics.Region.Op; 
import android.os.Build; 
import android.util.AttributeSet; 
import android.view.MotionEvent; 
import android.view.View; 
import android.view.animation.Animation; 
import android.view.animation.DecelerateInterpolator; 
import android.view.animation.Transformation; 
import android.widget.FrameLayout; 
 
public class SlideHolder extends FrameLayout { 
 
  public final static int DIRECTION_LEFT = 1; 
  public final static int DIRECTION_RIGHT = -1; 
   
  protected final static int MODE_READY = 0; 
  protected final static int MODE_SLIDE = 1; 
  protected final static int MODE_FINISHED = 2; 
   
  private Bitmap mCachedBitmap; 
  private Canvas mCachedCanvas; 
  private Paint mCachedPaint; 
  private View mMenuView; 
   
  private int mMode = MODE_READY; 
  private int mDirection = DIRECTION_LEFT; 
   
  private int mOffset = 0; 
  private int mStartOffset; 
  private int mEndOffset; 
   
  private boolean mEnabled = true; 
  private boolean mInterceptTouch = true; 
  private boolean mAlwaysOpened = false; 
  private boolean mDispatchWhenOpened = false; 
   
  private Queue<Runnable> mWhenReady = new LinkedList<Runnable>(); 
   
  private OnSlideListener mListener; 
   
  public SlideHolder(Context context) { 
    super(context); 
     
    initView(); 
  } 
   
  public SlideHolder(Context context, AttributeSet attrs) { 
    super(context, attrs); 
     
    initView(); 
  } 
   
  public SlideHolder(Context context, AttributeSet attrs, int defStyle) { 
    super(context, attrs, defStyle); 
     
    initView(); 
  } 
   
  private void initView() { 
    mCachedPaint = new Paint( 
          Paint.ANTI_ALIAS_FLAG 
          | Paint.FILTER_BITMAP_FLAG 
          | Paint.DITHER_FLAG 
        ); 
  } 
   
  @Override 
  public void setEnabled(boolean enabled) { 
    mEnabled = enabled; 
  } 
   
  @Override 
  public boolean isEnabled() { 
    return mEnabled; 
  } 
   
  /** 
   * 
   * @param direction - direction in which SlideHolder opens. Can be: DIRECTION_LEFT, DIRECTION_RIGHT 
   */ 
  public void setDirection(int direction) { 
    closeImmediately(); 
     
    mDirection = direction; 
  } 
   
  /** 
   * 
   * @param allow - if false, SlideHolder won't react to swiping gestures (but still will be able to work by manually invoking mathods) 
   */ 
  public void setAllowInterceptTouch(boolean allow) { 
    mInterceptTouch = allow; 
  } 
   
  public boolean isAllowedInterceptTouch() { 
    return mInterceptTouch; 
  } 
   
  /** 
   * 
   * @param dispatch - if true, in open state SlideHolder will dispatch touch events to main layout (in other words - it will be clickable) 
   */ 
  public void setDispatchTouchWhenOpened(boolean dispatch) { 
    mDispatchWhenOpened = dispatch; 
  } 
   
  public boolean isDispatchTouchWhenOpened() { 
    return mDispatchWhenOpened; 
  } 
   
  /** 
   * 
   * @param opened - if true, SlideHolder will always be in opened state (which means that swiping won't work) 
   */ 
  public void setAlwaysOpened(boolean opened) { 
    mAlwaysOpened = opened; 
     
    requestLayout(); 
  } 
   
  public int getMenuOffset() { 
    return mOffset; 
  } 
   
  public void setOnSlideListener(OnSlideListener lis) { 
    mListener = lis; 
  } 
   
  public boolean isOpened() { 
    return mAlwaysOpened || mMode == MODE_FINISHED; 
  } 
   
  public void toggle(boolean immediately) { 
    if(immediately) { 
      toggleImmediately(); 
    } else { 
      toggle(); 
    } 
  } 
   
  public void toggle() { 
    if(isOpened()) { 
      close(); 
    } else { 
      open(); 
    } 
  } 
   
  public void toggleImmediately() { 
    if(isOpened()) { 
      closeImmediately(); 
    } else { 
      openImmediately(); 
    } 
  } 
   
  public boolean open() { 
    if(isOpened() || mAlwaysOpened || mMode == MODE_SLIDE) { 
      return false; 
    } 
     
    if(!isReadyForSlide()) { 
      mWhenReady.add(new Runnable() { 
         
        @Override 
        public void run() { 
          open(); 
        } 
      }); 
       
      return true; 
    } 
     
    initSlideMode(); 
     
    Animation anim = new SlideAnimation(mOffset, mEndOffset); 
    anim.setAnimationListener(mOpenListener); 
    startAnimation(anim); 
     
    invalidate(); 
     
    return true; 
  } 
   
  public boolean openImmediately() { 
    if(isOpened() || mAlwaysOpened || mMode == MODE_SLIDE) { 
      return false; 
    } 
     
    if(!isReadyForSlide()) { 
      mWhenReady.add(new Runnable() { 
         
        @Override 
        public void run() { 
          openImmediately(); 
        } 
      }); 
       
      return true; 
    } 
     
    mMenuView.setVisibility(View.VISIBLE); 
    mMode = MODE_FINISHED; 
    requestLayout(); 
     
    if(mListener != null) { 
      mListener.onSlideCompleted(true); 
    } 
     
    return true; 
  } 
   
  public boolean close() { 
    if(!isOpened() || mAlwaysOpened || mMode == MODE_SLIDE) { 
      return false; 
    } 
     
    if(!isReadyForSlide()) { 
      mWhenReady.add(new Runnable() { 
         
        @Override 
        public void run() { 
          close(); 
        } 
      }); 
       
      return true; 
    } 
     
    initSlideMode(); 
     
    Animation anim = new SlideAnimation(mOffset, mEndOffset); 
    anim.setAnimationListener(mCloseListener); 
    startAnimation(anim); 
     
    invalidate(); 
     
    return true; 
  } 
   
  public boolean closeImmediately() { 
    if(!isOpened() || mAlwaysOpened || mMode == MODE_SLIDE) { 
      return false; 
    } 
     
    if(!isReadyForSlide()) { 
      mWhenReady.add(new Runnable() { 
         
        @Override 
        public void run() { 
          closeImmediately(); 
        } 
      }); 
       
      return true; 
    } 
     
    mMenuView.setVisibility(View.GONE); 
    mMode = MODE_READY; 
    requestLayout(); 
     
    if(mListener != null) { 
      mListener.onSlideCompleted(false); 
    } 
     
    return true; 
  } 
   
  @Override 
  protected void onLayout(boolean changed, int l, int t, int r, int b) { 
    final int parentLeft = 0; 
    final int parentTop = 0; 
    final int parentRight = r - l; 
    final int parentBottom = b - t; 
     
    View menu = getChildAt(0); 
    int menuWidth = menu.getMeasuredWidth(); 
     
    if(mDirection == DIRECTION_LEFT) { 
      menu.layout(parentLeft, parentTop, parentLeft+menuWidth, parentBottom); 
    } else { 
      menu.layout(parentRight-menuWidth, parentTop, parentRight, parentBottom); 
    } 
     
    if(mAlwaysOpened) { 
      if(mDirection == DIRECTION_LEFT) { 
        mOffset = menuWidth; 
      } else { 
        mOffset = 0; 
      } 
    } else if(mMode == MODE_FINISHED) { 
      mOffset = mDirection*menuWidth; 
    } else if(mMode == MODE_READY) { 
      mOffset = 0; 
    } 
     
    View main = getChildAt(1); 
    main.layout( 
          parentLeft + mOffset, 
          parentTop, 
          parentLeft + mOffset + main.getMeasuredWidth(), 
          parentBottom 
        ); 
     
    invalidate(); 
     
    Runnable rn; 
    while((rn = mWhenReady.poll()) != null) { 
      rn.run(); 
    } 
  } 
   
  private boolean isReadyForSlide() { 
    return (getWidth() > 0 && getHeight() > 0); 
  } 
   
  @Override 
  protected void onMeasure(int wSp, int hSp) { 
    mMenuView = getChildAt(0); 
     
    if(mAlwaysOpened) { 
      View main = getChildAt(1); 
       
      if(mMenuView != null && main != null) { 
        measureChild(mMenuView, wSp, hSp); 
        LayoutParams lp = (LayoutParams) main.getLayoutParams(); 
         
        if(mDirection == DIRECTION_LEFT) { 
          lp.leftMargin = mMenuView.getMeasuredWidth(); 
        } else { 
          lp.rightMargin = mMenuView.getMeasuredWidth(); 
        } 
      } 
    } 
     
    super.onMeasure(wSp, hSp); 
  } 
 
  private byte mFrame = 0; 
   
  @Override 
  protected void dispatchDraw(Canvas canvas) { 
    try { 
      if(mMode == MODE_SLIDE) { 
        View main = getChildAt(1); 
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) { 
          /* 
           * On new versions we redrawing main layout only 
           * if it's marked as dirty 
           */ 
          if(main.isDirty()) { 
            mCachedCanvas.drawColor(Color.TRANSPARENT, Mode.CLEAR); 
            main.draw(mCachedCanvas); 
        } 
        } else { 
          /* 
           * On older versions we just redrawing our cache 
           * every 5th frame 
           */ 
          if(++mFrame % 5 == 0) { 
            mCachedCanvas.drawColor(Color.TRANSPARENT, Mode.CLEAR); 
            main.draw(mCachedCanvas); 
          } 
        } 
 
        /* 
         * Draw only visible part of menu 
         */ 
         
        View menu = getChildAt(0); 
        final int scrollX = menu.getScrollX(); 
        final int scrollY = menu.getScrollY(); 
         
        canvas.save(); 
         
        if(mDirection == DIRECTION_LEFT) { 
          canvas.clipRect(0, 0, mOffset, menu.getHeight(), Op.REPLACE); 
        } else { 
          int menuWidth = menu.getWidth(); 
          int menuLeft = menu.getLeft(); 
           
          canvas.clipRect(menuLeft+menuWidth+mOffset, 0, menuLeft+menuWidth, menu.getHeight()); 
        } 
         
        canvas.translate(menu.getLeft(), menu.getTop()); 
        canvas.translate(-scrollX, -scrollY); 
         
        menu.draw(canvas); 
         
        canvas.restore(); 
         
        canvas.drawBitmap(mCachedBitmap, mOffset, 0, mCachedPaint); 
      } else { 
        if(!mAlwaysOpened && mMode == MODE_READY) { 
          mMenuView.setVisibility(View.GONE); 
        } 
         
        super.dispatchDraw(canvas); 
      } 
    } catch(IndexOutOfBoundsException e) { 
      /* 
       * Possibility of crashes on some devices (especially on Samsung). 
       * Usually, when ListView is empty. 
       */ 
    } 
  } 
   
  private int mHistoricalX = 0; 
  private boolean mCloseOnRelease = false; 
   
  @Override 
  public boolean dispatchTouchEvent(MotionEvent ev) { 
    if(((!mEnabled || !mInterceptTouch) && mMode == MODE_READY) || mAlwaysOpened) { 
      return super.dispatchTouchEvent(ev); 
    } 
     
    if(mMode != MODE_FINISHED) { 
      onTouchEvent(ev); 
       
      if(mMode != MODE_SLIDE) { 
        super.dispatchTouchEvent(ev); 
      } else { 
        MotionEvent cancelEvent = MotionEvent.obtain(ev); 
        cancelEvent.setAction(MotionEvent.ACTION_CANCEL); 
        super.dispatchTouchEvent(cancelEvent); 
        cancelEvent.recycle(); 
      } 
       
      return true; 
    } else { 
      final int action = ev.getAction(); 
       
      Rect rect = new Rect(); 
      View menu = getChildAt(0); 
      menu.getHitRect(rect); 
       
      if(!rect.contains((int) ev.getX(), (int) ev.getY())) { 
        if (action == MotionEvent.ACTION_UP && mCloseOnRelease && !mDispatchWhenOpened) { 
          close(); 
          mCloseOnRelease = false; 
        } else { 
          if(action == MotionEvent.ACTION_DOWN && !mDispatchWhenOpened) { 
            mCloseOnRelease = true; 
          } 
           
          onTouchEvent(ev); 
        } 
         
        if(mDispatchWhenOpened) { 
          super.dispatchTouchEvent(ev); 
        } 
         
        return true; 
      } else { 
        onTouchEvent(ev); 
         
        ev.offsetLocation(-menu.getLeft(), -menu.getTop()); 
        menu.dispatchTouchEvent(ev); 
         
        return true; 
      } 
    } 
  } 
   
  private boolean handleTouchEvent(MotionEvent ev) { 
    if(!mEnabled) { 
      return false; 
    } 
     
    float x = ev.getX(); 
     
    if(ev.getAction() == MotionEvent.ACTION_DOWN) { 
      mHistoricalX = (int) x; 
       
      return true; 
    } 
     
    if(ev.getAction() == MotionEvent.ACTION_MOVE) { 
 
      float diff = x - mHistoricalX; 
 
      if((mDirection*diff > 50 && mMode == MODE_READY) || (mDirection*diff < -50 && mMode == MODE_FINISHED)) { 
        mHistoricalX = (int) x; 
         
        initSlideMode(); 
      } else if(mMode == MODE_SLIDE) { 
        mOffset += diff; 
         
        mHistoricalX = (int) x; 
         
        if(!isSlideAllowed()) { 
          finishSlide(); 
        } 
      } else { 
        return false; 
      } 
    } 
     
    if(ev.getAction() == MotionEvent.ACTION_UP) { 
      if(mMode == MODE_SLIDE) { 
        finishSlide(); 
      } 
       
      mCloseOnRelease = false; 
       
      return false; 
    } 
     
    return mMode == MODE_SLIDE; 
  } 
   
  @Override 
  public boolean onTouchEvent(MotionEvent ev) { 
    boolean handled = handleTouchEvent(ev); 
     
    invalidate(); 
     
    return handled; 
  } 
   
  private void initSlideMode() { 
    mCloseOnRelease = false; 
     
    View v = getChildAt(1); 
     
    if(mMode == MODE_READY) { 
      mStartOffset = 0; 
      mEndOffset = mDirection*getChildAt(0).getWidth(); 
    } else { 
      mStartOffset = mDirection*getChildAt(0).getWidth(); 
      mEndOffset = 0; 
    } 
     
    mOffset = mStartOffset; 
     
    if(mCachedBitmap == null || mCachedBitmap.isRecycled() || mCachedBitmap.getWidth() != v.getWidth()) { 
      mCachedBitmap = Bitmap.createBitmap(v.getWidth(), v.getHeight(), Bitmap.Config.ARGB_8888); 
      mCachedCanvas = new Canvas(mCachedBitmap); 
    } else { 
      mCachedCanvas.drawColor(Color.TRANSPARENT, Mode.CLEAR); 
    } 
     
    v.setVisibility(View.VISIBLE); 
     
    mCachedCanvas.translate(-v.getScrollX(), -v.getScrollY()); 
    v.draw(mCachedCanvas); 
     
    mMode = MODE_SLIDE; 
     
    mMenuView.setVisibility(View.VISIBLE); 
  } 
   
  private boolean isSlideAllowed() { 
    return (mDirection*mEndOffset > 0 && mDirection*mOffset < mDirection*mEndOffset && mDirection*mOffset >= mDirection*mStartOffset) 
        || (mEndOffset == 0 && mDirection*mOffset > mDirection*mEndOffset && mDirection*mOffset <= mDirection*mStartOffset); 
  } 
   
  private void completeOpening() { 
    mOffset = mDirection*mMenuView.getWidth(); 
    requestLayout(); 
     
    post(new Runnable() { 
       
      @Override 
      public void run() { 
        mMode = MODE_FINISHED; 
        mMenuView.setVisibility(View.VISIBLE); 
      } 
    }); 
     
    if(mListener != null) { 
      mListener.onSlideCompleted(true); 
    } 
  } 
   
  private Animation.AnimationListener mOpenListener = new Animation.AnimationListener() { 
     
    @Override 
    public void onAnimationStart(Animation animation) {} 
     
    @Override 
    public void onAnimationRepeat(Animation animation) {} 
     
    @Override 
    public void onAnimationEnd(Animation animation) { 
      completeOpening(); 
    } 
  }; 
   
  private void completeClosing() { 
    mOffset = 0; 
    requestLayout(); 
     
    post(new Runnable() { 
       
      @Override 
      public void run() { 
        mMode = MODE_READY; 
        mMenuView.setVisibility(View.GONE); 
      } 
    }); 
     
    if(mListener != null) { 
      mListener.onSlideCompleted(false); 
    } 
  } 
   
  private Animation.AnimationListener mCloseListener = new Animation.AnimationListener() { 
     
    @Override 
    public void onAnimationStart(Animation animation) {} 
     
    @Override 
    public void onAnimationRepeat(Animation animation) {} 
     
    @Override 
    public void onAnimationEnd(Animation animation) { 
      completeClosing(); 
    } 
  }; 
   
  private void finishSlide() { 
    if(mDirection*mEndOffset > 0) { 
      if(mDirection*mOffset > mDirection*mEndOffset/2) { 
        if(mDirection*mOffset > mDirection*mEndOffset) mOffset = mEndOffset; 
         
        Animation anim = new SlideAnimation(mOffset, mEndOffset); 
        anim.setAnimationListener(mOpenListener); 
        startAnimation(anim); 
      } else { 
        if(mDirection*mOffset < mDirection*mStartOffset) mOffset = mStartOffset; 
         
        Animation anim = new SlideAnimation(mOffset, mStartOffset); 
        anim.setAnimationListener(mCloseListener); 
        startAnimation(anim); 
      } 
    } else { 
      if(mDirection*mOffset < mDirection*mStartOffset/2) { 
        if(mDirection*mOffset < mDirection*mEndOffset) mOffset = mEndOffset; 
         
        Animation anim = new SlideAnimation(mOffset, mEndOffset); 
        anim.setAnimationListener(mCloseListener); 
        startAnimation(anim); 
      } else { 
        if(mDirection*mOffset > mDirection*mStartOffset) mOffset = mStartOffset; 
         
        Animation anim = new SlideAnimation(mOffset, mStartOffset); 
        anim.setAnimationListener(mOpenListener); 
        startAnimation(anim); 
      } 
    } 
  } 
   
  private class SlideAnimation extends Animation { 
     
    private static final float SPEED = 0.6f; 
     
    private float mStart; 
    private float mEnd; 
     
    public SlideAnimation(float fromX, float toX) { 
      mStart = fromX; 
      mEnd = toX; 
       
      setInterpolator(new DecelerateInterpolator()); 
 
      float duration = Math.abs(mEnd - mStart) / SPEED; 
      setDuration((long) duration); 
    } 
     
    @Override 
    protected void applyTransformation(float interpolatedTime, Transformation t) { 
      super.applyTransformation(interpolatedTime, t); 
       
      float offset = (mEnd - mStart) * interpolatedTime + mStart; 
      mOffset = (int) offset; 
       
      postInvalidate(); 
    } 
     
  } 
   
  public static interface OnSlideListener { 
    public void onSlideCompleted(boolean opened); 
  } 
 
} 

使用:

package com.agimind.sidemenuexample; 
 
import com.agimind.widget.SlideHolder; 
 
import android.os.Bundle; 
import android.view.MenuItem; 
import android.view.View; 
import android.app.ActionBar; 
import android.app.Activity; 
 
public class MainActivity extends Activity { 
 
  private SlideHolder mSlideHolder; 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 
 
    mSlideHolder = (SlideHolder) findViewById(R.id.slideHolder); 
    // mSlideHolder.setAllowInterceptTouch(false); 
    // mSlideHolder.setAlwaysOpened(true); 
    /* 
     * toggleView can actually be any view you want. Here, for simplicity, 
     * we're using TextView, but you can easily replace it with button. 
     * 
     * Note, when menu opens our textView will become invisible, so it quite 
     * pointless to assign toggle-event to it. In real app consider using UP 
     * button instead. In our case toggle() can be replaced with open(). 
     */ 
 
    ActionBar actionBar = getActionBar(); 
    actionBar.setDisplayShowHomeEnabled(true); 
    actionBar.setHomeButtonEnabled(true); 
     
    View toggleView = findViewById(R.id.textView); 
    toggleView.setOnClickListener(new View.OnClickListener() { 
 
      @Override 
      public void onClick(View v) { 
        mSlideHolder.toggle(); 
      } 
    }); 
  } 
 
  @Override 
  public boolean onOptionsItemSelected(MenuItem item) { 
    switch (item.getItemId()) { 
    case android.R.id.home: 
      mSlideHolder.toggle(); 
      break; 
 
    default: 
      break; 
    } 
    return super.onOptionsItemSelected(item); 
  } 
} 

布局如下:

<com.agimind.widget.SlideHolder xmlns:android="http://schemas.android.com/apk/res/android" 
  xmlns:tools="http://schemas.android.com/tools" 
  android:id="@+id/slideHolder" 
  android:layout_width="fill_parent" 
  android:layout_height="fill_parent" 
  tools:context=".MainActivity" > 
 
  <ScrollView 
    android:layout_width="200dp" 
    android:layout_height="fill_parent" 
    android:background="@android:color/black" > 
 
    <LinearLayout 
      android:layout_width="200dp" 
      android:layout_height="wrap_content" 
      android:orientation="vertical" > 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
 
      <Button 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:text="@string/menu_settings" /> 
    </LinearLayout> 
  </ScrollView> 
 
  <RelativeLayout 
    android:layout_width="fill_parent" 
    android:layout_height="fill_parent" > 
 
    <TextView 
      android:id="@+id/textView" 
      android:layout_width="wrap_content" 
      android:layout_height="wrap_content" 
      android:layout_centerHorizontal="true" 
      android:layout_centerVertical="true" 
      android:text="@string/swipe" 
      android:textSize="25sp" /> 
 
  </RelativeLayout> 
 
</com.agimind.widget.SlideHolder> 

下载:AndroidSideMenu

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • android如何获取view在布局中的高度与宽度详解

    android如何获取view在布局中的高度与宽度详解

    这篇文章主要给大家介绍了关于android如何获取view在布局中的高度与宽度的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-10-10
  • Android实现自定义View控件的流程详解

    Android实现自定义View控件的流程详解

    这篇文章主要为大家详细介绍了Android中实现自定义View控件的流程,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2023-06-06
  • android控件Spinner(下拉列表)的使用例子

    android控件Spinner(下拉列表)的使用例子

    这篇文章主要给大家介绍了关于android控件Spinner(下拉列表)的使用例子,在Android开发中下拉框(Spinner)是常用的UI控件之一,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2023-11-11
  • Android实现IP地址输入框的方法示例代码

    Android实现IP地址输入框的方法示例代码

    输入框是我们日常开发中经常遇到的一个控件,如果更好的控制输入框是对用户体验很重要的一步,所以下面这篇文章主要给大家介绍了关于Android如何实现IP输入框的相关资料,需要的朋友可以参考下。
    2017-10-10
  • Android实现旋转,放大,缩小图片的方法

    Android实现旋转,放大,缩小图片的方法

    这篇文章主要介绍了Android实现旋转,放大,缩小图片的方法,结合实例形式分析了Android基于Drawable针对图片的缩放与旋转等处理技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2016-10-10
  • Android自定义View实现两种二维码的扫描效果

    Android自定义View实现两种二维码的扫描效果

    这篇文章主要为大家详细介绍了Android如何自定义View实现两种二维码的扫描效果,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-01-01
  • Android游戏开发学习①弹跳小球实现方法

    Android游戏开发学习①弹跳小球实现方法

    这篇文章主要介绍了Android游戏开发学习①弹跳小球实现方法,涉及Android通过物理引擎BallThread类模拟小球运动的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-10-10
  • 过滤Android工程中多余资源文件的解决方法

    过滤Android工程中多余资源文件的解决方法

    这篇文章主要介绍了过滤Android工程中多余资源文件的解决方法,非常实用!需要的朋友可以参考下
    2014-07-07
  • Android自定义ViewGroup实现流式布局

    Android自定义ViewGroup实现流式布局

    这篇文章主要为大家详细介绍了Android自定义ViewGroup实现流式布局,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-09-09
  • 解决Android自定义view获取attr中自定义颜色的问题

    解决Android自定义view获取attr中自定义颜色的问题

    这篇文章主要介绍了Android自定义view获取attr中自定义颜色的问题解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-12-12

最新评论