真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些

本篇文章給大家分享的是有關(guān)Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些,小編覺得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

站在用戶的角度思考問題,與客戶深入溝通,找到阿榮網(wǎng)站設(shè)計(jì)與阿榮網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個(gè)性化、用戶體驗(yàn)好的作品,建站類型包括:成都網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì)、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、域名注冊、虛擬空間、企業(yè)郵箱。業(yè)務(wù)覆蓋阿榮地區(qū)。

一、Android坐標(biāo)體系

在講解滑動之前,我們有必要簡單提一下Android的坐標(biāo)體系,因?yàn)榛瑒拥膶?shí)質(zhì)就是坐標(biāo)的不斷改變,所以我們先來了解一下Android坐標(biāo)系和視圖坐標(biāo)系兩個(gè)概念。直接放上兩張圖片吧,一目了然。

Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些

Android坐標(biāo)系

Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些

視圖坐標(biāo)系

從上面的兩張圖可以看出,Android坐標(biāo)系的坐標(biāo)原點(diǎn)位于屏幕的左上角,而視圖坐標(biāo)系的原點(diǎn)位于父視圖的左上角,既然提供了兩種不同的坐標(biāo)系,那么我們?nèi)绾蝸慝@取坐標(biāo)呢,Android已經(jīng)給我們提供了一些方法用于獲取這些坐標(biāo),看下面的圖便一目了然。

Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些

Android獲取坐標(biāo)的各種方法

二、layout方法

在View進(jìn)行繪制時(shí),是調(diào)用onLayout()方法來確定View的位置的,同樣我們也可以調(diào)用layout()方法來傳入我們滑動后的坐標(biāo)便可以實(shí)現(xiàn)View的滑動,當(dāng)然坐標(biāo)的獲取我們可以在觸控事件中進(jìn)行獲取,下面我們做一個(gè)View隨手指進(jìn)行滑動的小例子來進(jìn)行說明。

public class DragView extends View {
  private int mLastX;
  private int mLastY;
  public DragView(Context context) {
    this(context, null);
  }

  public DragView(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
  }

  public DragView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
  }

  @Override
  public boolean onTouchEvent(MotionEvent event) {
    int x = (int) event.getX();
    int y = (int) event.getY();
    int lastX = 0, lastY = 0;
    switch (event.getAction()){
      case MotionEvent.ACTION_DOWN:
        mLastX = x;
        mLastY = y;
        break;
      case MotionEvent.ACTION_MOVE:
        int offsetX = x - mLastX;
        int offsetY = y - mLastY;
        layout(getLeft() + offsetX, getTop() + offsetY,
            getRight() + offsetX, getBottom() + offsetY);
        break;
    }
    return true;
  }
}

上面我們在觸控事件中獲取到獲取到手指按下時(shí)的坐標(biāo)(lastX, lastY),然后在手指移動時(shí)不斷計(jì)算X和Y方向上的偏移量,然后再調(diào)用layout()方法來改變View的位置從而實(shí)現(xiàn)滑動。當(dāng)然上面我們是通過getX()和getY()來獲取視圖坐標(biāo)來進(jìn)行修改,我們也可以通過getRawX()和getRawY()來獲取絕對坐標(biāo)來實(shí)現(xiàn)上面的效果。代碼如下:

private int mLastX;
private int mLastY;
@Override
public boolean onTouchEvent(MotionEvent event) {
  int x = (int) event.getRawX();
  int y = (int) event.getRawY();
  switch (event.getAction()){
    case MotionEvent.ACTION_DOWN:
      mLastX = x;
      mLastY = y;
      break;
    case MotionEvent.ACTION_MOVE:
      int offsetX = x - mLastX;
      int offsetY = y - mLastY;
      layout(getLeft() + offsetX, getTop() + offsetY,
          getRight() + offsetX, getBottom() + offsetY);
      //重新設(shè)置初始坐標(biāo)
      mLastX = x;
      mLastY = y;
      break;
  }
  return true;
}

上面一定要注意,我們在改變完View的位置后必須調(diào)用設(shè)置初始坐標(biāo),這樣才能準(zhǔn)確獲取偏移量。

三、offsetLeftAndRight和offsetTopAndBottom

這一種方法和上一種方法大部分步驟都是相同的,只是在移動View上有所差別,代碼如下:

offsetLeftAndRight(offsetX);
offsetTopAndBottom(offsetY);

上面的這種方法只是多了一層封裝,可以實(shí)現(xiàn)比上面實(shí)現(xiàn)同樣的效果。

四、設(shè)置LayoutParams

LayoutParams可以通過改變的布局參數(shù),我們可以通過下面的代碼實(shí)現(xiàn)上面同樣的效果。

LinearLayout.LayoutParams layoutParams = (LinearLayout.LayoutParams) getLayoutParams();
layoutParams.leftMargin = getLeft() + offsetX;
layoutParams.topMargin = getTop() + offsetY;
setLayoutParams(layoutParams);

注意:我們的LayoutParams可以通過getLayoutParams()方法來獲取,但是要注意,如果View的父布局是LinearLayout,那么我們的LayoutParams就是LinearLayout.LayoutParams,如果View的父布局是RelativeLayout,則我們的LayoutParams就是RelativeLayout.LayoutParams。當(dāng)然我們還有一種簡單的方法,不用再管父布局的布局方式。代碼如下:

ViewGroup.MarginLayoutParams marginLayoutParams = (ViewGroup.MarginLayoutParams) getLayoutParams();
marginLayoutParams.leftMargin = getLeft() + offsetX;
marginLayoutParams.topMargin = getTop() + offsetY;
setLayoutParams(marginLayoutParams);

上面的這種方法不用管父布局的類型,使用起來更加方便。

五、scrollTo和scrollBy方法

關(guān)于這兩個(gè)方法我們需要仔細(xì)說一下其中的一些注意事項(xiàng)

1 . scrollTo的參數(shù)是具體的一個(gè)坐標(biāo)點(diǎn)(x, y), 而scrollBy的參數(shù)是在x, y方向上的坐標(biāo)偏移

2 . scrollTo和scrollBy移動的是View的內(nèi)容。這一點(diǎn)很重要!!!!

如果我們對ViewGroup使用scrollTo和scrollBy則移動的是內(nèi)部的所有子View, 如果對TextView使用scrollTo和scrollBy則移動的是其中額文本。

3 . 視圖移動還有一個(gè)不太好理解的地方在于坐標(biāo),我們下面結(jié)合圖片來說明一下:

Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些

視圖移動1

Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些

視圖移動2

我們可以這樣理解,我們的手機(jī)屏幕作為一個(gè)蓋板,在手機(jī)屏幕下面是一個(gè)巨大的畫布,我們的手機(jī)屏幕這個(gè)蓋板是透明的,導(dǎo)致只有和手機(jī)屏幕重合的畫布部分才會被我們看到,我們調(diào)用scrollToscrollBy也可以理解為是在移動手機(jī)上面的蓋板。如圖中所示,按鈕在ViewGroup中的坐標(biāo)是(20, 10),當(dāng)我們調(diào)用scrollBy(20, 10)之后,就相當(dāng)于移動了屏幕上的蓋板,然后我們看到的按鈕就到了ViewGroup的左上角。這樣如果我們想讓按鈕在水平和豎直方向上各移動2010個(gè)單位,我們就必須調(diào)用scrollBy(-20, -10)

經(jīng)過了上面的知識準(zhǔn)備,我們這里也使用scrollBy來實(shí)現(xiàn)前面實(shí)現(xiàn)的那個(gè)View隨手指移動的小例子:

((View)getParent()).scrollBy(-offsetX, -offsetY);

六、使用Scroller

Scroller也是滑動中很重要的一個(gè)角色,進(jìn)過前面的scrollTo和scrollBy大家也會發(fā)現(xiàn),它們的移動時(shí)瞬間完成的,滑動顯得十分突兀,Google為了改善用戶體驗(yàn),便給出了Scroller,它可以實(shí)現(xiàn)平滑的移動,從而使滑動過程更加真實(shí),用戶體驗(yàn)更好,下面我們先簡單說說Scroller的實(shí)現(xiàn)原理。

Scroller也是滑動中很重要的一個(gè)角色,進(jìn)過前面的scrollToscrollBy大家也會發(fā)現(xiàn),它們的移動時(shí)瞬間完成的,滑動顯得十分突兀,Google為了改善用戶體驗(yàn),便給出了Scroller,它可以實(shí)現(xiàn)平滑的移動,從而使滑動過程更加真實(shí),用戶體驗(yàn)更好,下面我們先簡單說說Scroller的實(shí)現(xiàn)原理。

Scroller的實(shí)現(xiàn)方式類似于scrollToscrollBy,scrollToscrollBy的移動都是從一個(gè)坐標(biāo)點(diǎn)瞬間移動到另一個(gè)左邊點(diǎn),而Scroller則是將移動的這段距離切分成好幾段的微小的位移,然后每一段調(diào)用scrollTo來不斷移動這些微小的位移,由于人眼的視覺暫留效果,就會給人平滑移動的視覺效果。

下面我們在上一步的基礎(chǔ)上增加一個(gè)小功能,第一部分還是View隨手指移動,但是當(dāng)我們松開手指時(shí),讓View自己平滑移動到最初始的位置(屏幕左上角),下面我們就來一步步介紹Scroller的用法

1 . 聲明Scroller變量,并在構(gòu)造方法中進(jìn)行初始化

2 . 在觸控事件的ACTION_UP(手指抬起)事件中傳入開始滑動的坐標(biāo)和需要滑動的距離并觸發(fā)Scroller的滑動事件

3 . 重寫computeScroll(),實(shí)現(xiàn)真正的滑動

下面是完整的代碼示例:

public class DragView extends View {
  private int mLastX;
  private int mLastY;
  //聲明Scroller變量
  private Scroller mScroller;
  public DragView(Context context) {
    this(context, null);
  }

  public DragView(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
  }

  public DragView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    //在構(gòu)造方法中初始化Scroller變量
    mScroller = new Scroller(context);
  }

  @Override
  public boolean onTouchEvent(MotionEvent event) {
    int x = (int) event.getRawX();
    int y = (int) event.getRawY();
    switch (event.getAction()){
      case MotionEvent.ACTION_DOWN:
        mLastX = x;
        mLastY = y;
        break;
      case MotionEvent.ACTION_MOVE:
        int offsetX = x - mLastX;
        int offsetY = y - mLastY;
        //實(shí)現(xiàn)View跟隨手指移動的效果
        ((View)getParent()).scrollBy(-offsetX, -offsetY);
        //重新設(shè)置初始坐標(biāo)
        mLastX = x;
        mLastY = y;
        break;
      case MotionEvent.ACTION_UP:
        //當(dāng)手指抬起時(shí)執(zhí)行滑動過程
        View view = (View) getParent();
        mScroller.startScroll(view.getScrollX(), view.getScrollY(),
            view.getScrollX(), view.getScrollY(), 5000);
        //調(diào)用重繪來間接調(diào)用computeScroll()方法
        invalidate();
        break;
    }
    return true;
  }

  @Override
  public void computeScroll() {
    super.computeScroll();
    //判斷滑動過程是否完成
    if (mScroller.computeScrollOffset()){
      ((View)getParent()).scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
      //通過重繪來不斷調(diào)用computeScroll()方法
      invalidate();
    }
  }
}

上面的代碼View隨手指移動的代碼部分是與前面相同的,我們只說說Scroller的部分以及一些注意事項(xiàng)

1 .  startScroll()方法各參數(shù)的意義,我們可以看看下面的源碼:

/**
 * Start scrolling by providing a starting point, the distance to travel,
 * and the duration of the scroll.
 *
 * @param startX Starting horizontal scroll offset in pixels. Positive
 *    numbers will scroll the content to the left.
 * @param startY Starting vertical scroll offset in pixels. Positive numbers
 *    will scroll the content up.
 * @param dx Horizontal distance to travel. Positive numbers will scroll the
 *    content to the left.
 * @param dy Vertical distance to travel. Positive numbers will scroll the
 *    content up.
 * @param duration Duration of the scroll in milliseconds.
 */
public void startScroll(int startX, int startY, int dx, int dy, int duration)

可以看出startXstartY參數(shù)就是開始滾動的(x, y)坐標(biāo),那么我們就可以通過ViewGroup(子View的父視圖)getScrollX()getScrollY()來獲取,這里一定要注意,我們在滑動時(shí)的content就是子View,所以我們通過子View的父視圖(ViewGroup)的getScrollX()getScrollY()獲取到的就是子View在X和Y方向上滑動的距離,即就是我們需要的當(dāng)我們手指抬起時(shí)子View的(x, y)坐標(biāo)。而如果我們對子View調(diào)用getScrollX()getScrollY()方法,則獲得的是子View內(nèi)部的視圖的滑動距離及坐標(biāo)。

dxdy分別是在X和Y方向上的偏移量,而且注釋中說了,如果我們傳入的dxdy的值是正值,那么將會向上向左移動這個(gè)content(其實(shí)就是我們這里的View),即我們就可以讓子View回到左上角,這里我們還是可以借助于上一小節(jié)中提到的視圖移動的概念,我們想讓子View向坐上方移動,其實(shí)就是想讓覆蓋在上面的蓋板向右下角移動,我們可以將dxdy理解為父視圖(覆蓋在上面的蓋板)的偏移量。

假設(shè)我們剛開始是讓子View隨手指向右下方移動,那么相當(dāng)于覆蓋在上面的蓋板是向左上方移動,所以我們通過getScrollX()getScrollY()獲得的值是負(fù)值,我們現(xiàn)在松開手指想讓子View向左上方移動(即回到屏幕左上角),那么就相當(dāng)于蓋板向右下角移動,所以我們的dxdy的值必須是-getScrollX()-getScrollY(),此時(shí)的兩個(gè)值都是正值。

2 . 由于我們的computeScroll()方法不會主動調(diào)用,但是我們又需要它不斷調(diào)用從而不斷進(jìn)行微小移動從而實(shí)現(xiàn)平滑的滑動,所以我們可以通過下面的方法。

這三個(gè)按照以下順序進(jìn)行調(diào)用 invalidate()--->onDraw()--->computeScroll(),所以我們可以可以在ACTION_UP中調(diào)用完startScroll()方法后調(diào)用invalidate()方法,然后在computeScroll()方法中判斷滑動是否結(jié)束,如果沒結(jié)束,則通過getCurrX()getCurrY()來獲得當(dāng)前需要移動的微小的位移的坐標(biāo)點(diǎn),然后傳入scrollTo()方法中,這時(shí)候子View還只是移動了一小段距離,然后我們再次調(diào)用invalidate()方法,然后接著調(diào)用onDraw()方法,然后再次進(jìn)入computeScroll()中再次讓子View移動一小段距離,直到滑動結(jié)束,computeScrollOffset()返回false,則這個(gè)循環(huán)調(diào)用的過程結(jié)束,從而完成平滑移動的過程。

以上就是Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些,小編相信有部分知識點(diǎn)可能是我們?nèi)粘9ぷ鲿姷交蛴玫降摹OM隳芡ㄟ^這篇文章學(xué)到更多知識。更多詳情敬請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


本文題目:Android項(xiàng)目中實(shí)現(xiàn)滑動功能的方法有哪些
瀏覽路徑:http://weahome.cn/article/isjgjg.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部