Android側邊欄的自訂滑動實現(附源碼),android源碼

來源:互聯網
上載者:User

Android側邊欄的自訂滑動實現(附源碼),android源碼

林炳文Evankaka原創作品。轉載請註明出處http://blog.csdn.net/evankaka

    本文要實現手指在手機上向左或向右移動時,能相應的移動左右兩個視圖。通過自訂來實現,不藉助第三方外掛程式。實現的思路很簡單,通過判斷手指滑動的距離和速度來決定是否要滾動顯示功能表項目。(源碼免費下載)

先來看看效果:


一、實現思路1.思路

菜單在左,內容在右,然後菜單顯示時和手機右邊框有一定的間隔,內容顯示一小部分。內容全部顯示時,菜單全部不可見。如下面兩個圖

                                 顯示內容

                                       

                            顯示菜單2.判斷邏輯

這是判斷手指按著螢幕和手指抬起時要不要顯示還是隱藏菜單

二、代碼清單

首先來看下布局:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:id="@+id/layout"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:orientation="horizontal"    tools:context=".MainActivity" >    <LinearLayout        android:id="@+id/menu"        android:layout_width="match_parent"        android:layout_height="match_parent"        android:orientation="horizontal"        android:background="@drawable/pn" >    </LinearLayout>    <LinearLayout        android:id="@+id/content"        android:layout_width="match_parent"        android:layout_height="match_parent"        android:orientation="horizontal"        android:background="@drawable/sn">        </LinearLayout> </LinearLayout>
接下來看看代碼

package com.example.learningjava;import com.example.learningjava.R.string;import android.R.integer;import android.R.menu;import android.os.AsyncTask;import android.os.Build;import android.os.Bundle;import android.annotation.SuppressLint;import android.annotation.TargetApi;import android.widget.LinearLayout.LayoutParams;import android.app.Activity;import android.util.DisplayMetrics;import android.util.Log;import android.view.Menu;import android.view.MotionEvent;import android.view.VelocityTracker;import android.view.View;import android.view.View.OnTouchListener;import android.view.Window;import android.widget.LinearLayout;public class MainActivity extends Activity implements OnTouchListener{private LinearLayout menuLayout;//功能表項目private LinearLayout contentLayout;//內容項private LayoutParams  menuParams;//功能表項目的參數private LayoutParams contentParams;//內容項目的參數contentLayout的寬度值 private int disPlayWidth;//手機螢幕解析度    private float xDown;//手指點下去的橫座標    private float xMove;//手指移動的橫座標    private float xUp;//記錄手指上抬後的橫座標        private VelocityTracker mVelocityTracker; // 用於計算手指滑動的速度。      float velocityX;//手指左右移動的速度    public static final int SNAP_VELOCITY = 400; //滾動顯示和隱藏menu時,手指滑動需要達到的速度。      private boolean menuIsShow = false;//初始化功能表項目不可翽private static final int menuPadding=80;//menu完成顯示,留給content的寬度  protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        requestWindowFeature(Window.FEATURE_NO_TITLE);        setContentView(R.layout.activity_main);        initLayoutParams();            }  /**   *初始化Layout並設定其相應的參數   */  private void initLayoutParams()  {//得到螢幕的大小       DisplayMetrics dm = new DisplayMetrics();      getWindowManager().getDefaultDisplay().getMetrics(dm);        disPlayWidth =dm.widthPixels;              //獲得控制項      menuLayout = (LinearLayout) findViewById(R.id.menu);      contentLayout = (LinearLayout) findViewById(R.id.content);      findViewById(R.id.layout).setOnTouchListener(this);            //獲得控制項參數      menuParams=(LinearLayout.LayoutParams)menuLayout.getLayoutParams();      contentParams = (LinearLayout.LayoutParams) contentLayout.getLayoutParams();             //初始化菜單和內容的寬和邊距      menuParams.width = disPlayWidth - menuPadding;      menuParams.leftMargin = 0 - menuParams.width;      contentParams.width = disPlayWidth;      contentParams.leftMargin=0;            //設定參數      menuLayout.setLayoutParams(menuParams);      contentLayout.setLayoutParams(contentParams);    }   @Override  public boolean onTouch(View v, MotionEvent event)  {  acquireVelocityTracker(event);  switch (event.getAction())      {      case MotionEvent.ACTION_DOWN:      xDown=event.getRawX();             break;                case MotionEvent.ACTION_MOVE:      xMove=event.getRawX();         isScrollToShowMenu();          break;                case MotionEvent.ACTION_UP:      xUp=event.getRawX();             isShowMenu();      releaseVelocityTracker();            break;                case MotionEvent.ACTION_CANCEL:            releaseVelocityTracker();            break;        }      return true;  }  /**   * 根據手指按下的距離,判斷是否滾動顯示菜單   */  private void isScrollToShowMenu()  {        int distanceX = (int) (xMove - xDown);              if (!menuIsShow) {      scrollToShowMenu(distanceX);        }else{      scrollToHideMenu(distanceX);        }  }  /**   * 手指抬起之後判斷是否要顯示菜單   */  private void isShowMenu()  {       velocityX =getScrollVelocity();       if(wantToShowMenu()){       if(shouldShowMenu()){       showMenu();       }else{       hideMenu();       }       }       else if(wantToHideMenu()){       if(shouldHideMenu()){       hideMenu();       }else{  showMenu();   }       }      }  /**   *想要顯示菜單,當向右移動距離大於0並且菜單不可見   */  private boolean wantToShowMenu(){  return !menuIsShow&&xUp-xDown>0;  }  /**   *想要隱藏菜單,當向左移動距離大於0並且菜單可見   */  private boolean wantToHideMenu(){  return menuIsShow&&xDown-xUp>0;  }  /**   *判斷應該顯示菜單,當向右移動的距離超過菜單的一半或者速度超過給定值   */  private boolean shouldShowMenu(){  return xUp-xDown>menuParams.width/2||velocityX>SNAP_VELOCITY;  }  /**   *判斷應該隱藏菜單,當向左移動的距離超過菜單的一半或者速度超過給定值   */  private boolean shouldHideMenu(){  return xDown-xUp>menuParams.width/2||velocityX>SNAP_VELOCITY;  }  /**   * 顯示功能表列   */  private void showMenu()  {      new showMenuAsyncTask().execute(50);      menuIsShow=true;  }  /**   * 隱藏功能表列   */  private void hideMenu()  { new showMenuAsyncTask().execute(-50);     menuIsShow=false;  }  /**   *指標按著時,滾動將菜單慢慢顯示出來   *@param scrollX 每次滾動移動的距離   */  private void scrollToShowMenu(int scrollX)  {  if(scrollX>0&&scrollX<= menuParams.width)  menuParams.leftMargin =-menuParams.width+scrollX;  menuLayout.setLayoutParams(menuParams);   }  /**   *指標按著時,滾動將菜單慢慢隱藏出來   *@param scrollX 每次滾動移動的距離   */  private void scrollToHideMenu(int scrollX)  {  if(scrollX>=-menuParams.width&&scrollX<0)  menuParams.leftMargin=scrollX;  menuLayout.setLayoutParams(menuParams);   }       /**     * 建立VelocityTracker對象,並將觸摸content介面的滑動事件加入到VelocityTracker當中。    * @param event 向VelocityTracker添加MotionEvent     */    private void acquireVelocityTracker(final MotionEvent event) {        if(null == mVelocityTracker) {            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 releaseVelocityTracker() {        if(null != mVelocityTracker) {            mVelocityTracker.clear();            mVelocityTracker.recycle();            mVelocityTracker = null;        }    }    /**  *  *:類比動畫過程,讓肉眼能看到滾動的效果  *  */  class showMenuAsyncTask extends AsyncTask<Integer, Integer, Integer>  {      @Override      protected Integer doInBackground(Integer... params)      {          int leftMargin = menuParams.leftMargin;          while (true)          {// 根據傳入的速度來滾動介面,當滾動到達左邊界或右邊界時,跳出迴圈。              leftMargin += params[0];              if (params[0] > 0 && leftMargin > 0)              {              leftMargin= 0;                  break;              } else if (params[0] < 0 && leftMargin <-menuParams.width)              {              leftMargin=-menuParams.width;                  break;              }              publishProgress(leftMargin);              try              {                  Thread.sleep(40);//休眠一下,肉眼才能看到滾動效果              } catch (InterruptedException e)              {                  e.printStackTrace();              }          }          return leftMargin;      }      @Override      protected void onProgressUpdate(Integer... value)      {          menuParams.leftMargin = value[0];          menuLayout.setLayoutParams(menuParams);      }      @Override      protected void onPostExecute(Integer result)      {          menuParams.leftMargin = result;          menuLayout.setLayoutParams(menuParams);      }  }}

三、效果與說明

       側滑菜單在很多應用中都會見到,其實實現起來原理非常簡單,只不過是要有一大堆的判斷,你要判斷手指移動的距離和方向,還要判斷手指移動的速度。所以代碼寫出來可能有點兒多,但是原理瞭解了就不難了。另外,為了滾動效果肉眼可以看到,加了個showMenuAsyncTask類,它在滾動視圖的過程中,每sleep(40)然後再滾動,當然,這裡時間還可以改到一此,效果會更加好。

源碼免費下載

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.