android 有彈性的ScrollView 簡單實現,與處理ScrollView和ListView,GridView之間的衝突

來源:互聯網
上載者:User

標籤:

處理ScrollView和ListView,GridView之間的衝突,

最好的辦法就是繼承這兩個類,重寫他們的onMeasure方法即可:

ListView:

import android.widget.ListView;/** * ScrollView中嵌入ListView,讓ListView全顯示出來 * @author John * */public class MyListView extends ListView{    public MyListView(android.content.Context context,android.util.AttributeSet attrs){        super(context, attrs);    }    /**     * 設定不滾動     */    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec)    {        int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,                MeasureSpec.AT_MOST);        super.onMeasure(widthMeasureSpec, expandSpec);    }    }

GridView:

import android.widget.GridView;public class MyGridView extends GridView {    public MyGridView(android.content.Context context,            android.util.AttributeSet attrs) {        super(context, attrs);    }    /**     * 設定不滾動     */    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {        int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,                MeasureSpec.AT_MOST);        super.onMeasure(widthMeasureSpec, expandSpec);    }}

如果是ListView中嵌套GridView也可以用這個辦法。簡單實用的處理了這個衝突。

下面是重寫ScrollView,實現有彈性的ScrollView

 /*** 有彈性的ScrollView* 實現下拉彈回和上拉彈回* @author John*/public class OwnScroview extends ScrollView {       private static final String TAG = "ElasticScrollView";       //移動因子, 是一個百分比, 比如手指移動了100px, 那麼View就只移動20px   //目的是達到一個延遲的效果   private static final float MOVE_FACTOR = 0.4f;       //鬆開手指後, 介面回到正常位置需要的動畫時間   private static final int ANIM_TIME = 300;       //ScrollView的子View, 也是ScrollView的唯一一個子View   private View contentView;        //手指按下時的Y值, 用於在移動時計算移動距離   //如果按下時不能上拉和下拉, 會在手指移動時更新為當前手指的Y值   private float startY;       //用於記錄正常的布局位置   private Rect originalRect = new Rect();       //手指按下時記錄是否可以繼續下拉   private boolean canPullDown = false;       //手指按下時記錄是否可以繼續上拉   private boolean canPullUp = false;       //在手指滑動的過程中記錄是否移動了布局   private boolean isMoved = false;   public OwnScroview(Context context) {       super(context);   }       public OwnScroview(Context context, AttributeSet attrs) {       super(context, attrs);   }   @Override   protected void onFinishInflate() {       if (getChildCount() > 0) {           contentView = getChildAt(0);       }   }       @Override   protected void onLayout(boolean changed, int l, int t, int r, int b) {       super.onLayout(changed, l, t, r, b);               if(contentView == null) return;       //ScrollView中的唯一子控制項的位置資訊, 這個位置資訊在整個控制項的生命週期中保持不變       originalRect.set(contentView.getLeft(), contentView.getTop(), contentView               .getRight(), contentView.getBottom());   }   /**    * 在觸摸事件中, 處理上拉和下拉的邏輯    */   @Override   public boolean dispatchTouchEvent(MotionEvent ev) {               if (contentView == null) {           return super.dispatchTouchEvent(ev);       }       int action = ev.getAction();               switch (action) {       case MotionEvent.ACTION_DOWN:                       //判斷是否可以上拉和下拉           canPullDown = isCanPullDown();           canPullUp = isCanPullUp();                       //記錄按下時的Y值           startY = ev.getY();           break;                   case MotionEvent.ACTION_UP:                       if(!isMoved) break;  //如果沒有移動布局, 則跳過執行                       // 開啟動畫           TranslateAnimation anim = new TranslateAnimation(0, 0, contentView.getTop(),                   originalRect.top);           anim.setDuration(ANIM_TIME);                       contentView.startAnimation(anim);                       // 設定回到正常的布局位置           contentView.layout(originalRect.left, originalRect.top,                    originalRect.right, originalRect.bottom);                       //將標誌位設回false           canPullDown = false;           canPullUp = false;           isMoved = false;                       break;       case MotionEvent.ACTION_MOVE:                       //在移動的過程中, 既沒有滾動到可以上拉的程度, 也沒有滾動到可以下拉的程度           if(!canPullDown && !canPullUp) {               startY = ev.getY();               canPullDown = isCanPullDown();               canPullUp = isCanPullUp();                               break;           }                       //計算手指移動的距離           float nowY = ev.getY();           int deltaY = (int) (nowY - startY);                       //是否應該移動布局           boolean shouldMove =                    (canPullDown && deltaY > 0)    //可以下拉, 並且手指向下移動                   || (canPullUp && deltaY< 0)    //可以上拉, 並且手指向上移動                   || (canPullUp && canPullDown); //既可以上拉也可以下拉(這種情況出現在ScrollView包裹的控制項比ScrollView還小)                       if(shouldMove){               //計算位移量               int offset = (int)(deltaY * MOVE_FACTOR);                               //隨著手指的移動而移動布局               contentView.layout(originalRect.left, originalRect.top + offset,                       originalRect.right, originalRect.bottom + offset);                               isMoved = true;  //記錄移動了布局           }                       break;       default:           break;       }       return super.dispatchTouchEvent(ev);   }       /**    * 判斷是否滾動到頂部    */   private boolean isCanPullDown() {       return getScrollY() == 0 ||                contentView.getHeight() < getHeight() + getScrollY();   }       /**    * 判斷是否滾動到底部    */   private boolean isCanPullUp() {       return  contentView.getHeight() <= getHeight() + getScrollY();   }    }

然後再xml中用上就行了。

android 有彈性的ScrollView 簡單實現,與處理ScrollView和ListView,GridView之間的衝突

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在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.