国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 系統(tǒng) > Android > 正文

Android UI使用HorizontalListView實(shí)現(xiàn)水平滑動(dòng)

2019-10-21 21:27:05
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

今天就介紹一個(gè)大神級(jí)人物自定義的ListView實(shí)現(xiàn)水平滑動(dòng),我知道要實(shí)現(xiàn)一個(gè)可以水平滑動(dòng)的方法有很多,但是這個(gè)HorizontalListView用起來(lái)是真的很不錯(cuò)!!! 

先看一下效果圖:

Android,UI,HorizontalListView,水平滑動(dòng)

界面做的不怎么看得上眼,但是基本的動(dòng)能還是在的,下面給出HorizontalListView的代碼:

/*  * HorizontalListView.java v1.5  *  *  * The MIT License  * Copyright (c) 2011 Paul Soucy (paul@dev-smart.com)  *  * Permission is hereby granted, free of charge, to any person obtaining a copy  * of this software and associated documentation files (the "Software"), to deal  * in the Software without restriction, including without limitation the rights  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell  * copies of the Software, and to permit persons to whom the Software is  * furnished to do so, subject to the following conditions:  *  * The above copyright notice and this permission notice shall be included in  * all copies or substantial portions of the Software.  *  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN  * THE SOFTWARE.  *  */import java.util.LinkedList;import java.util.Queue;import android.content.Context;import android.database.DataSetObserver;import android.graphics.Rect;import android.util.AttributeSet;import android.view.GestureDetector;import android.view.GestureDetector.OnGestureListener;import android.view.MotionEvent;import android.view.View;import android.widget.AdapterView;import android.widget.ListAdapter;import android.widget.Scroller;public class HorizontalListView extends AdapterView<ListAdapter> { public boolean mAlwaysOverrideTouch = true; protected ListAdapter mAdapter; private int mLeftViewIndex = -1; private int mRightViewIndex = 0; protected int mCurrentX; protected int mNextX; private int mMaxX = Integer.MAX_VALUE; private int mDisplayOffset = 0; protected Scroller mScroller; private GestureDetector mGesture; private Queue<View> mRemovedViewQueue = new LinkedList<View>(); private OnItemSelectedListener mOnItemSelected; private OnItemClickListener mOnItemClicked; private OnItemLongClickListener mOnItemLongClicked; private boolean mDataChanged = false; public HorizontalListView(Context context, AttributeSet attrs) {  super(context, attrs);  initView(); } private synchronized void initView() {  mLeftViewIndex = -1;  mRightViewIndex = 0;  mDisplayOffset = 0;  mCurrentX = 0;  mNextX = 0;  mMaxX = Integer.MAX_VALUE;  mScroller = new Scroller(getContext());  mGesture = new GestureDetector(getContext(), mOnGesture); } @Override public void setOnItemSelectedListener(AdapterView.OnItemSelectedListener listener) {  mOnItemSelected = listener; } @Override public void setOnItemClickListener(AdapterView.OnItemClickListener listener){  mOnItemClicked = listener; } @Override public void setOnItemLongClickListener(AdapterView.OnItemLongClickListener listener) {  mOnItemLongClicked = listener; } private DataSetObserver mDataObserver = new DataSetObserver() {  @Override  public void onChanged() {   synchronized(HorizontalListView.this){    mDataChanged = true;   }   invalidate();   requestLayout();  }  @Override  public void onInvalidated() {   reset();   invalidate();   requestLayout();  } }; @Override public ListAdapter getAdapter() {  return mAdapter; } @Override public View getSelectedView() {  //TODO: implement  return null; } @Override public void setAdapter(ListAdapter adapter) {  if(mAdapter != null) {   mAdapter.unregisterDataSetObserver(mDataObserver);  }  mAdapter = adapter;  mAdapter.registerDataSetObserver(mDataObserver);  reset(); } private synchronized void reset(){  initView();  removeAllViewsInLayout();  requestLayout(); } @Override public void setSelection(int position) {  //TODO: implement } private void addAndMeasureChild(final View child, int viewPos) {  LayoutParams params = child.getLayoutParams();  if(params == null) {   params = new LayoutParams(LayoutParams.FILL_PARENT, LayoutParams.FILL_PARENT);  }  addViewInLayout(child, viewPos, params, true);  child.measure(MeasureSpec.makeMeasureSpec(getWidth(), MeasureSpec.AT_MOST),    MeasureSpec.makeMeasureSpec(getHeight(), MeasureSpec.AT_MOST)); } @Override protected synchronized void onLayout(boolean changed, int left, int top, int right, int bottom) {  super.onLayout(changed, left, top, right, bottom);  if(mAdapter == null){   return;  }  if(mDataChanged){   int oldCurrentX = mCurrentX;   initView();   removeAllViewsInLayout();   mNextX = oldCurrentX;   mDataChanged = false;  }  if(mScroller.computeScrollOffset()){   int scrollx = mScroller.getCurrX();   mNextX = scrollx;  }  if(mNextX <= 0){   mNextX = 0;   mScroller.forceFinished(true);  }  if(mNextX >= mMaxX) {   mNextX = mMaxX;   mScroller.forceFinished(true);  }  int dx = mCurrentX - mNextX;  removeNonVisibleItems(dx);  fillList(dx);  positionItems(dx);  mCurrentX = mNextX;  if(!mScroller.isFinished()){   post(new Runnable(){    @Override    public void run() {     requestLayout();    }   });  } } private void fillList(final int dx) {  int edge = 0;  View child = getChildAt(getChildCount()-1);  if(child != null) {   edge = child.getRight();  }  fillListRight(edge, dx);  edge = 0;  child = getChildAt(0);  if(child != null) {   edge = child.getLeft();  }  fillListLeft(edge, dx); } private void fillListRight(int rightEdge, final int dx) {  while(rightEdge + dx < getWidth() && mRightViewIndex < mAdapter.getCount()) {   View child = mAdapter.getView(mRightViewIndex, mRemovedViewQueue.poll(), this);   addAndMeasureChild(child, -1);   rightEdge += child.getMeasuredWidth();   if(mRightViewIndex == mAdapter.getCount()-1) {    mMaxX = mCurrentX + rightEdge - getWidth();   }   if (mMaxX < 0) {    mMaxX = 0;   }   mRightViewIndex++;  } } private void fillListLeft(int leftEdge, final int dx) {  while(leftEdge + dx > 0 && mLeftViewIndex >= 0) {   View child = mAdapter.getView(mLeftViewIndex, mRemovedViewQueue.poll(), this);   addAndMeasureChild(child, 0);   leftEdge -= child.getMeasuredWidth();   mLeftViewIndex--;   mDisplayOffset -= child.getMeasuredWidth();  } } private void removeNonVisibleItems(final int dx) {  View child = getChildAt(0);  while(child != null && child.getRight() + dx <= 0) {   mDisplayOffset += child.getMeasuredWidth();   mRemovedViewQueue.offer(child);   removeViewInLayout(child);   mLeftViewIndex++;   child = getChildAt(0);  }  child = getChildAt(getChildCount()-1);  while(child != null && child.getLeft() + dx >= getWidth()) {   mRemovedViewQueue.offer(child);   removeViewInLayout(child);   mRightViewIndex--;   child = getChildAt(getChildCount()-1);  } } private void positionItems(final int dx) {  if(getChildCount() > 0){   mDisplayOffset += dx;   int left = mDisplayOffset;   for(int i=0;i<getChildCount();i++){    View child = getChildAt(i);    int childWidth = child.getMeasuredWidth();    child.layout(left, 0, left + childWidth, child.getMeasuredHeight());    left += childWidth + child.getPaddingRight();   }  } } public synchronized void scrollTo(int x) {  mScroller.startScroll(mNextX, 0, x - mNextX, 0);  requestLayout(); } @Override public boolean dispatchTouchEvent(MotionEvent ev) {  boolean handled = super.dispatchTouchEvent(ev);  handled |= mGesture.onTouchEvent(ev);  return handled; } protected boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,        float velocityY) {  synchronized(HorizontalListView.this){   mScroller.fling(mNextX, 0, (int)-velocityX, 0, 0, mMaxX, 0, 0);  }  requestLayout();  return true; } protected boolean onDown(MotionEvent e) {  mScroller.forceFinished(true);  return true; } private OnGestureListener mOnGesture = new GestureDetector.SimpleOnGestureListener() {  @Override  public boolean onDown(MotionEvent e) {   return HorizontalListView.this.onDown(e);  }  @Override  public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,        float velocityY) {   return HorizontalListView.this.onFling(e1, e2, velocityX, velocityY);  }  @Override  public boolean onScroll(MotionEvent e1, MotionEvent e2,        float distanceX, float distanceY) {   synchronized(HorizontalListView.this){    mNextX += (int)distanceX;   }   requestLayout();   return true;  }  @Override  public boolean onSingleTapConfirmed(MotionEvent e) {   for(int i=0;i<getChildCount();i++){    View child = getChildAt(i);    if (isEventWithinView(e, child)) {     if(mOnItemClicked != null){      mOnItemClicked.onItemClick(HorizontalListView.this, child, mLeftViewIndex + 1 + i, mAdapter.getItemId( mLeftViewIndex + 1 + i ));     }     if(mOnItemSelected != null){      mOnItemSelected.onItemSelected(HorizontalListView.this, child, mLeftViewIndex + 1 + i, mAdapter.getItemId( mLeftViewIndex + 1 + i ));     }     break;    }   }   return true;  }  @Override  public void onLongPress(MotionEvent e) {   int childCount = getChildCount();   for (int i = 0; i < childCount; i++) {    View child = getChildAt(i);    if (isEventWithinView(e, child)) {     if (mOnItemLongClicked != null) {      mOnItemLongClicked.onItemLongClick(HorizontalListView.this, child, mLeftViewIndex + 1 + i, mAdapter.getItemId(mLeftViewIndex + 1 + i));     }     break;    }   }  }  private boolean isEventWithinView(MotionEvent e, View child) {   Rect viewRect = new Rect();   int[] childPosition = new int[2];   child.getLocationOnScreen(childPosition);   int left = childPosition[0];   int right = left + child.getWidth();   int top = childPosition[1];   int bottom = top + child.getHeight();   viewRect.set(left, top, right, bottom);   return viewRect.contains((int) e.getRawX(), (int) e.getRawY());  } };}

在使用的時(shí)候直接當(dāng)做普通的ListView使用就可以了!!!(有一點(diǎn)需要注意,也算是這個(gè)自定義ListView的一點(diǎn)小瑕疵吧,在直接在xml使用該View的時(shí)候,如果view的高度設(shè)置為wrap_content,實(shí)際上回匹配其父布局的高度,所以在使用的時(shí)候可以更多情況下需要我們指定list的確切高度)

好了,關(guān)于這個(gè)MIT的horizontalListView就簡(jiǎn)單說(shuō)到這里。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持VEVB武林網(wǎng)。


注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到Android開(kāi)發(fā)頻道。
發(fā)表評(píng)論 共有條評(píng)論
用戶(hù)名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 灵台县| 星座| 金乡县| 苍梧县| 阳西县| 万年县| 于都县| 裕民县| 九江市| 定南县| 吴川市| 桓仁| 日照市| 大港区| 海盐县| 孟州市| 周口市| 安达市| 黔东| 通州区| 嘉禾县| 比如县| 彝良县| 化隆| 青冈县| 泰州市| 图木舒克市| 弥渡县| 安陆市| 苏尼特左旗| 咸宁市| 苗栗市| 临邑县| 逊克县| 思南县| 吉木乃县| 衢州市| 永吉县| 古蔺县| 义乌市| 云阳县|