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

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

AndroidScroller及下拉刷新組件原理解析

Android事件攔截機制

創(chuàng)新互聯(lián)專注于順平企業(yè)網(wǎng)站建設,響應式網(wǎng)站設計,商城網(wǎng)站建設。順平網(wǎng)站建設公司,為順平等地區(qū)提供建站服務。全流程定制制作,專業(yè)設計,全程項目跟蹤,創(chuàng)新互聯(lián)專業(yè)和態(tài)度為您提供的服務

Android中事件的傳遞和攔截和View樹結(jié)構(gòu)是相關聯(lián)的,在View樹中,分為葉子節(jié)點和普通節(jié)點,普通節(jié)點有子節(jié)點只能是ViewGroup,葉子節(jié)點可以是View或者ViewGroup。Android和事件分發(fā)攔截相關的方法有
dispatchTouchEvent(MotionEvent ev)
事件分發(fā)相關的方法,沿著View樹將一個用戶的觸摸事件向下分發(fā)。
onInterceptTouchEvent(MotionEvent ev)
在dispatchTouchEvent中被調(diào)用,用來判斷某一層級是否攔截一個事件,返回true即攔截,事件不會再向下分發(fā),注意View樹中葉子節(jié)點(View和ViewGroup)直接攔截事件。
onTouchEvent(MotionEvent ev)
一個某一個層級攔截了事件,那么所有事件序列都會交由它處理,后面onInterceptTouchEvent不會再被調(diào)用,轉(zhuǎn)而onTouchEvent被調(diào)用。OnTouchEvent返回true則消耗掉這個事件序列,如果沒有消耗ACTION_DOWN事件則事件序列將沿著View樹向上傳遞,去找能處理這個事件的父View。如果消耗了ACTION_DOWN而沒有消耗其它事件,那么這個事件序列將消失。

整體過程描述:事件產(chǎn)生傳遞到某一個ViewGroup時,首先其onInterceptTouchEvent會被調(diào)用,如果當前ViewGroup選擇攔截這個事件則返回true,于是它的onTouchEvent會被調(diào)用。否則將繼續(xù)調(diào)用子View的dispatchTouchEvent進行方法的攔截判斷和相應的處理。
當一個View處理事件時,首先會調(diào)用它的OnTouchListener,如果OnTouchListener返回false則會繼續(xù)調(diào)用onTouchEvent,在onTouchEvent中才會檢查onClickListener,由此可見三種處理事件方法的優(yōu)先級是:OnTouchListener > onTouchEvent > onClickListener。

ScrollTo,ScrollBy,Scroller

在實現(xiàn)滑動效果的時候,最常用的三個方法就是ScrollTo,ScrollBy和Scroller
首先介紹ScrollTo和ScrollBy,兩個方法一個是滑動到某個位置,一個是滑動多少位置。關鍵在于,ScrollTo和ScrollBy對于普通的View組件比如TextView、ImageView的效果是移動View的內(nèi)容,也就是相應的字體、照片,僅對于ViewGroup才是移動所有的子View。也就是說,ScrollTo和ScrollBy通常用在自定義的ViewGroup實現(xiàn)滑動效果時。
其次要理解ViewGroup滑動的坐標系,如下圖左邊是滑動前的布局,一個ViewGroup下面有兩個子View,在ViewGroup中調(diào)用ScrollTo(0,300)就是將ViewGroup向下滑動,可以將ViewGroup看做一個透明窗口,向下滑動后第一個子View消失不見,第二個子View相對效果即是向上滑動。所以這里要注意ScrollTo和ScrollBy的正負值,同時記住滑動的是ViewGroup,子View只是間接滑動的。
最后,Scroller很簡單,Scroller更類似于動畫中的插值器,處理計算和存儲坐標值,什么也沒有做。當我們調(diào)用
mScroller.startScroll(getScrollX(),getScrollY(),0,mHeaderHeight+getPaddingTop(),3000);

后,實際上是在其中根據(jù)時間和要移動的像素計算出每一時刻所應該在的像素位置,然后不停的調(diào)用scrollBy移動到這個位置并重繪。同時由于View在重繪時繪調(diào)用computeScroll方法,所以我們要在其中進行判斷并繼續(xù)scroll,形成有條件遞歸,形成動畫。

Android Scroller及下拉刷新組件原理解析

下拉刷新組件的簡單原理

基本介紹

Android Scroller及下拉刷新組件原理解析

一個典型的下拉刷新界面如上,對于下拉刷新功能而言,界面主要包含兩個部分,一個是展示Refresh界面的部分,一個是展示如ListView之類列表的部分。為了實現(xiàn)下拉刷新功能,我們所需要的就是自定義一個ViewGroup。我們的RefreshLayout中包含兩個子View,header和content。header界面如下:

Android Scroller及下拉刷新組件原理解析

content可以是ListView,同樣也是一個ViewGroup。界面初始時由于header和content都可以看到,所以我們在RefreshLayout的onLayout方法結(jié)束前,調(diào)用scrollTo(0,headerHeight)可以將header滑動出界面。然后,總的思路就是分析RefreshLayout和ListView對于一個觸摸事件,誰來攔截誰來處理的問題。

RefreshLayout實現(xiàn):

RefreshLayout繪制過程:

首先通過 LayoutInflater.from(context).inflate以及addView方法,在RefreshLayout構(gòu)造函數(shù)中向布局添加header和content。對于一個ViewGroup而言,繪制過程中最重要的是onMeasure和onLayout方法。
onMeasure

@Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  int width = MeasureSpec.getSize(widthMeasureSpec);
  int height = 0;
  for(int i=0;i

onMeasure方法中,一定要對全部子View進行measure,在這里調(diào)用的是measureChild方法,因為measureChild內(nèi)部還會根據(jù)子View的LayoutParams進一步封裝出MeasureSpec進行測量。

@Override
 protected void onLayout(boolean changed, int l, int t, int r, int b) {
  int count = getChildCount();
  int left =getPaddingLeft();
  Log.d("TAG", l + " " + t + " " + r + " " + b);
  int top = getPaddingTop();
  for(int i=0;i

onLayout方法中進行我們想要的布局,注意由于重新繪制時,onMeasure和onLayout會多次被調(diào)用,所以要注意一些初始化方法的執(zhí)行。

RefreshLayout事件攔截及處理

@Override
 public boolean onInterceptTouchEvent(MotionEvent ev) {
  switch (ev.getAction()) {
   case MotionEvent.ACTION_DOWN:
    prevY = (int) ev.getRawY();
    break;
   case MotionEvent.ACTION_MOVE:
    int delY = (int) (ev.getRawY() - prevY);
    Log.d("TAG", "delY " + delY);
    if(delY>0) {
     return true;
    }
    break;
  }
  return false;
 }

在攔截事件中,只做了一個簡單的判斷,一旦滑動的縱向距離大于0,表明手指再從上向下滑,同時這里應該判斷一下ListView中顯示的第一條是不是全部數(shù)據(jù)中的第一條。然后攔截事件后交由onTouchEvent處理。

@Override
 public boolean onTouchEvent(MotionEvent event) {
  switch (event.getAction()) {
   case MotionEvent.ACTION_MOVE:
    int dy = (int) (event.getRawY() - prevY);
    int sy = mHeaderHeight-dy;
    scrollTo(0,sy>0?sy:0);
    Log.d("TAG", "dy " + dy);
    break;
   case MotionEvent.ACTION_UP:
    refresh();
    break;
  }
  return true;
 }

之前將ViewGroup向下滑動了headerHeight的距離,為了讓header顯示出來,其實應該讓ViewGroup向上滑動也即y軸變小,同時為了避免過分滑動還要進行一下判斷。當手指抬起時,還要根據(jù)移動的y軸增量判斷一下是否是有效的滑動,然后處理響應的業(yè)務邏輯。注意的是,由于當前是主線程,所以要使用

  new Thread(new Runnable() {
   @Override
   public void run() {
    mission();
    post(new Runnable() {
     @Override
     public void run() {
      mScroller.startScroll(getScrollX(),getScrollY(),0,mHeaderHeight+getPaddingTop(),3000);
      mArrowView.setVisibility(VISIBLE);
      mProgress.setVisibility(GONE);
     }
    });
   }
  }).start();

新起一個線程完成mission,同時通過當前ViewGroup的消息隊列,在任務完成后修改UI。

涉及到的原理大致就是這些,完整的代碼可以查看何洪洋老師的博客:
https://github.com/hehonghui/android_my_pull_refresh_view

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。


文章標題:AndroidScroller及下拉刷新組件原理解析
轉(zhuǎn)載來于:http://weahome.cn/article/jhesod.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部