android view实现横向滑动选择

 更新时间:2018年07月25日 10:13:12   作者:s569646547  
这篇文章主要为大家详细介绍了android view实现横向滑动选择,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了android view实现横向滑动选择的具体代码,供大家参考,具体内容如下

做文字编辑,从网上找来的。

HorizontalScrollSelectView:

  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 OnScrollListener mScrollListener;
  /**
   * 选中item时图片
   */
  private Drawable mDrawable;
  private boolean mDataChanged = false;
  private Context context;
  private boolean scrollerFalg1 = false;
  private boolean scrollerFalg2 = false;
  private int position = 0x7f020000;
 
 
  public HorizontalScrollSelectView(Context context, AttributeSet attrs) {
    super(context, attrs);
    this.context = context;
    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);
  }
 
  public void setMScrollListener(OnScrollListener listener) {
    mScrollListener = listener;
  }
 
  @Override
  public void setOnItemSelectedListener(OnItemSelectedListener listener) {
    mOnItemSelected = listener;
  }
 
  @Override
  public void setOnItemClickListener(OnItemClickListener listener) {
    mOnItemClicked = listener;
  }
 
  @Override
  public void setOnItemLongClickListener(OnItemLongClickListener listener) {
    mOnItemLongClicked = listener;
  }
 
  /**
   * 设置选中状态时的图片
   *
   * @param mDrawable
   */
  public void setSelectBitmap(Drawable mDrawable) {
    this.mDrawable = mDrawable;
  }
 
  private DataSetObserver mDataObserver = new DataSetObserver() {
 
    @Override
    public void onChanged() {
      synchronized (HorizontalScrollSelectView.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
    int positionX = position * this.getWidth();
    int maxWidth = this.getChildCount() * this.getWidth();
    if (positionX <= 0) {
      positionX = 0;
    }
    if (positionX > maxWidth) {
      positionX = maxWidth;
    }
    scrollTo(positionX);
  }
 
  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));
  }
 
 
  @SuppressLint("NewApi")
  @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);
// Log.e("onlayout", "mLeftViewIndex"+(mLeftViewIndex+1));
    mCurrentX = mNextX;
 
    if (!mScroller.isFinished()) {
      post(new Runnable() {
        @Override
        public void run() {
          requestLayout();
        }
      });
 
    } else {
      if (scrollerFalg1 || (!scrollerFalg1 && scrollerFalg2)) {
        View chid = getChildAt(0);
        if (chid != null) {
          mDisplayOffset = 0;
          positionItems(0);
          if (mDrawable != null) {
            getChildAt(2).setBackground(mDrawable);
          }
          if (mScrollListener != null) {
            int position = (int) getChildAt(2).getTag(this.position);
            mScrollListener.onScrollSelectItem(this, position);
          }
        }
 
      }
    }
 
 
  }
 
  /**
   * 获取屏幕宽度
   *
   * @param context
   * @return
   */
  public static int getSecreenWidth(Context context) {
    DisplayMetrics dm = new DisplayMetrics();
    dm = context.getResources().getDisplayMetrics();
    int screenWidth = dm.widthPixels;
    return screenWidth / 5;
  }
 
  private void fillList(final int dx) {
    int edge = 0;
    View child = getChildAt(0);
    if (child != null) {
      edge = child.getLeft();
    }
    fillListLeft(edge, dx);
 
    edge = 0;
    child = getChildAt(getChildCount() - 1);
    if (child != null) {
      edge = child.getRight();
    }
    fillListRight(edge, dx);
 
  }
 
  private void fillListRight(int rightEdge, final int dx) {
    //&& mRightViewIndex < mAdapter.getCount()
    while (rightEdge + dx < getWidth()) {
      if (mRightViewIndex >= mAdapter.getCount()) {
        mRightViewIndex = 0;
      }
      View child = mAdapter.getView(mRightViewIndex, mRemovedViewQueue.poll(), this);
      child.setTag(position, mRightViewIndex);
      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) {
    //&& mLeftViewIndex >= 0
    while (leftEdge + dx > 0) {
      if (mLeftViewIndex <= -1) {
        mLeftViewIndex = mAdapter.getCount() - 1;
      }
      View child = mAdapter.getView(mLeftViewIndex, mRemovedViewQueue.poll(), this);
      child.setTag(position, mLeftViewIndex);
      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++;
      if (mLeftViewIndex >= mAdapter.getCount()) {
        mLeftViewIndex = 0;
      }
      child = getChildAt(0);
 
    }
 
    child = getChildAt(getChildCount() - 1);
    while (child != null && child.getLeft() + dx >= getWidth()) {
      mRemovedViewQueue.offer(child);
      removeViewInLayout(child);
      mRightViewIndex--;
      if (mRightViewIndex <= -1) {
        mRightViewIndex = mAdapter.getCount() - 1;
      }
      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);
        getChildAt(2).setBackground(null);
        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);
 
    switch (ev.getAction()) {
      case MotionEvent.ACTION_UP:
        scrollerFalg2 = true;
        requestLayout();
        break;
      default:
        break;
    }
    return handled;
  }
 
  protected boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
               float velocityY) {
    synchronized (HorizontalScrollSelectView.this) {
      //mNextX
      mScroller.fling(mNextX, 0, (int) -velocityX, 0, Integer.MIN_VALUE, mMaxX, 0, 0);
    }
    requestLayout();
    scrollerFalg1 = true;
    return true;
  }
 
  protected boolean onDown(MotionEvent e) {
    mScroller.forceFinished(true);
    return true;
  }
 
  public interface OnScrollListener {
    /**
     * 互动过程中选中的item
     *
     * @param position
     */
    public void onScrollSelectItem(ViewGroup viewGroup, int position);
 
  }
 
 
  private OnGestureListener mOnGesture = new GestureDetector.SimpleOnGestureListener() {
 
    @Override
    public boolean onDown(MotionEvent e) {
      return HorizontalScrollSelectView.this.onDown(e);
    }
 
    @Override
    public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
                float velocityY) {
      return HorizontalScrollSelectView.this.onFling(e1, e2, velocityX, velocityY);
    }
 
    @Override
    public boolean onScroll(MotionEvent e1, MotionEvent e2,
                float distanceX, float distanceY) {
 
      synchronized (HorizontalScrollSelectView.this) {
        mNextX += (int) distanceX;
      }
      requestLayout();
      scrollerFalg1 = false;
      scrollerFalg2 = false;
      return true;
    }
 
 
    @Override
    public void onShowPress(MotionEvent e) {
      super.onShowPress(e);
    }
 
    @Override
    public boolean onSingleTapUp(MotionEvent e) {
      Log.e("onSingleTapUp", "mLeftViewIndex" + (mLeftViewIndex + 1));
//  scrollerFalg=true;
      return super.onSingleTapUp(e);
    }
 
    @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(HorizontalScrollSelectView.this, child, mLeftViewIndex + 1 + i, mAdapter.getItemId(mLeftViewIndex + 1 + i));
          }
          if (mOnItemSelected != null) {
            mOnItemSelected.onItemSelected(HorizontalScrollSelectView.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(HorizontalScrollSelectView.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());
    }
  };

demo截图:

源码下载:android view实现横向滑动选择

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • 获取Activity栈,判断当前Activity位置的方法

    获取Activity栈,判断当前Activity位置的方法

    下面小编就为大家分享一篇获取Activity栈,判断当前Activity位置的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-03-03
  • Android数据存储之SQLite使用

    Android数据存储之SQLite使用

    SQLite是D.Richard Hipp用C语言编写的开源嵌入式数据库引擎。它支持大多数的SQL92标准,并且可以在所有主要的操作系统上运行
    2016-01-01
  • Flutter进阶之实现动画效果(六)

    Flutter进阶之实现动画效果(六)

    这篇文章主要为大家详细介绍了Flutter进阶之实现动画效果第六篇,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-08-08
  • Android解析JSON格式数据的两种方式(JSONObject和Gson)

    Android解析JSON格式数据的两种方式(JSONObject和Gson)

    json数据的解析相对而言,还是比较容易的,实现的代码也十分简单,下面这篇文章主要给大家介绍了关于Android解析JSON格式数据的两种方式,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-07-07
  • 浅谈Android Studio 4.1 更新内容

    浅谈Android Studio 4.1 更新内容

    这篇文章主要介绍了浅谈Android Studio 4.1 更新内容,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-10-10
  • Android性能优化getResources()与Binder导致界面卡顿优化

    Android性能优化getResources()与Binder导致界面卡顿优化

    这篇文章主要为大家介绍了Android性能优化getResources()与Binder导致界面卡顿优化示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-02-02
  • Handler与Android多线程详解

    Handler与Android多线程详解

    一开始,相信很多人都以为myThread中的run()方法会在一个新的线程中运行,但事实并非如此。以下代码中的handler并没有调用线程myThread的start()方法,而是直接调用了run()方法,这也就意味着实际上并没有创建一个新的线程,只是在当前线程中调用run()方法而已
    2013-10-10
  • Android实现3秒钟自动关闭界面

    Android实现3秒钟自动关闭界面

    这篇文章主要为大家详细介绍了Android实现3秒钟自动关闭界面,以支付成功为例,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-02-02
  • Android开发笔记之探秘WebView

    Android开发笔记之探秘WebView

    浏览器控件是每个开发环境都具备的,这为马甲神功提供了用武之地,windows的有webbrowser,android和ios都有webview。只是其引擎不同,相对于微软的webbrowser,android及ios的webview的引擎都是webkit,对Html5提供支持。本篇主要介绍android的webview。
    2014-08-08
  • Android实现动态添加标签及其点击事件

    Android实现动态添加标签及其点击事件

    这篇文章主要为大家详细介绍了Android实现动态添加标签及其点击事件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12

最新评论