Android股票列表联动

 

使用ScrollerView+RecycleView实现股票列表联动效果;一个可以拿来可以直接使用的解决方案,而且滑动流畅,带标题栏悬浮功能,支持整行的点击效果,支持长安事件;

效果如下图所示:

效果图1效果图2

实现原理:

原理图1

使用两个ScrollView:一个控制垂直方向上的滑动,一个控制水平方向上的滚动;

public class SVerticallScrollView extends ScrollView {

    private int downY;
    private int mTouchSlop;
    private static SVYOnScrollListener myOnScrollListener;

    public SVerticallScrollView(Context context) {
        super(context);
        mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
    }

    public SVerticallScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
        mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
    }

    public SVerticallScrollView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent e) {
        int action = e.getAction();
        switch (action) {
            case MotionEvent.ACTION_DOWN:
                downY = (int) e.getRawY();
                break;
            case MotionEvent.ACTION_MOVE:
                int moveY = (int) e.getRawY();
                if (Math.abs(moveY - downY) > mTouchSlop) {
                    return true;
                }
            case MotionEvent.ACTION_UP:
                break;
        }
        return super.onInterceptTouchEvent(e);
    }

    @Override
    protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        super.onScrollChanged(l, t, oldl, oldt);
        if(myOnScrollListener != null) {
            myOnScrollListener.onScroll(t);
        } 

    }

    public void setSVYOnScrollListener(SVYOnScrollListener listener) {
        myOnScrollListener = listener;
    }

    public  interface SVYOnScrollListener{
        void onScroll(int scrolly);
    }
}
public class SHorizontalScrollView extends HorizontalScrollView {

    private static SHorizontalScrollView.SHXOnScrollListener myOnScrollListener;

    public SHorizontalScrollView(Context context) {
        super(context);
    }

    public SHorizontalScrollView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    public SHorizontalScrollView(Context context, @Nullable AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    private float lastX, lastY;

    @Override
    public boolean onInterceptTouchEvent(MotionEvent e) {

        switch (e.getAction()) {
            case MotionEvent.ACTION_DOWN:
                lastX = e.getX();
                lastY = e.getY();
                break;
            case MotionEvent.ACTION_MOVE:
                // 只要横向大于竖向,就拦截掉事件。
                // 部分机型点击事件(slopx==slopy==0),会触发MOVE事件。
                // 所以要加判断(slopX > 0 || sloy > 0)
                float slopX = Math.abs(e.getX() - lastX);
                float slopY = Math.abs(e.getY() - lastY);
                if((slopX > 0 || slopY > 0) && slopX >= slopY){
                    requestDisallowInterceptTouchEvent(true);
                    return true;
                }
                break;

        }
        return super.onInterceptTouchEvent(e);
    }

    @Override
    protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        super.onScrollChanged(l, t, oldl, oldt);
        if(myOnScrollListener != null) {
            myOnScrollListener.onScrollX(l);
        } 

    }


    public void setSVHOnScrollListener(SHorizontalScrollView.SHXOnScrollListener listener) {
        myOnScrollListener = listener;
    }

    public  interface SHXOnScrollListener{
        void onScrollX(int scrollX);
    }

}

 

两个RecycleView:一个固定展示股票名称,一个可以水平滑动展示更多的股票数据信息;

使用手势来控制点击事件和长按事件:给RecycleView添加addOnItemTouchListener来获取点击和长按事件的回调

点击效果和长按效果的实现:通过手势获取到当前点击RecycleView的Item的index,获取到两个RecycleView对应位置上的ViewHold,设置其状态为pressed的状态(会显示出布局文件中设置的backgroud的selector效果),再定时恢复状态;

recyclerView.addOnItemTouchListener(new OnItemTouchListener(recyclerView, brecyclerView) {
            @Override
            public void onItemClick(RecyclerView.ViewHolder vh, int index) {
                Toast.makeText(getBaseContext(), " onItemClick index = " + index, Toast.LENGTH_SHORT).show();
            }

            @Override
            public void onItemLongClick(RecyclerView.ViewHolder vh, int index) {
                Toast.makeText(getBaseContext(), " onItemLongClick index = " + index, Toast.LENGTH_SHORT).show();
            }
        });
public static abstract class OnItemTouchListener implements RecyclerView.OnItemTouchListener {

        private GestureDetectorCompat mGestureDetectorCompat;
        private RecyclerView mRecyclerView;
        private RecyclerView mBindRecyclerView;

        public OnItemTouchListener(RecyclerView recyclerView) {
            mRecyclerView = recyclerView;
            mGestureDetectorCompat = new GestureDetectorCompat(mRecyclerView.getContext(),
                    new MyGestureListener());
        }
        public OnItemTouchListener(RecyclerView recyclerView, RecyclerView brecyclerView) {
            mRecyclerView = recyclerView;
            mBindRecyclerView = brecyclerView;
            mGestureDetectorCompat = new GestureDetectorCompat(mRecyclerView.getContext(),
                    new MyGestureListener());
        }

        @Override
        public void onTouchEvent(RecyclerView rv, MotionEvent e) {
            mGestureDetectorCompat.onTouchEvent(e);
        }

        @Override
        public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) {
            mGestureDetectorCompat.onTouchEvent(e);
            return false;
        }

        @Override
        public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {

        }

        public abstract void onItemClick(RecyclerView.ViewHolder vh, int index);
        public abstract void onItemLongClick(RecyclerView.ViewHolder vh, int index);


        Handler handler = new Handler();
        private class MyGestureListener extends GestureDetector.SimpleOnGestureListener {
            @Override
            public boolean onSingleTapUp(MotionEvent e) {
                View childe = mRecyclerView.findChildViewUnder(e.getX(), e.getY());
                if (childe != null) {
                    RecyclerView.ViewHolder VH = mRecyclerView.getChildViewHolder(childe);
                    int index = VH.getLayoutPosition();
                    setPressState(VH, index);
                    onItemClick(VH, index);

                }
                return true;
            }

            @Override
            public void onLongPress(MotionEvent e) {
                View childe = mRecyclerView.findChildViewUnder(e.getX(), e.getY());
                if (childe != null) {
                    RecyclerView.ViewHolder VH = mRecyclerView.getChildViewHolder(childe);
                    int index = VH.getLayoutPosition();
                    setPressState(VH, index);
                    onItemLongClick(VH, index);
                }
            }


            private void setPressState(RecyclerView.ViewHolder VH, int index) {
                setPressedState(VH);
                if(mBindRecyclerView != null) {
                    RecyclerView.ViewHolder BVH = mBindRecyclerView.findViewHolderForLayoutPosition(index);
                    setPressedState(BVH);
                }

            }

            private void setPressedState(RecyclerView.ViewHolder VH) {
                if(VH != null) {
                    if(VH instanceof NameItemAdapter.MyViewHolder) {
                        setNameItemPressed(VH);
                    } else if(VH instanceof DataItemAdapter.MyViewHolder) {
                        setDataItemPressed(VH);
                    }
                }
            }

            private void setNameItemPressed(RecyclerView.ViewHolder VH) {
                final NameItemAdapter.MyViewHolder viewHolder = (NameItemAdapter.MyViewHolder) VH;
                setItemViewPressed(viewHolder.ItemView);
            }
            private void setDataItemPressed(RecyclerView.ViewHolder VH) {
                final DataItemAdapter.MyViewHolder viewHolder = (DataItemAdapter.MyViewHolder) VH;
                setItemViewPressed(viewHolder.ItemView);
            }

            private void setItemViewPressed(final View view) {
                view.setPressed(true);
                handler.postDelayed(new Runnable() {
                    @Override
                    public void run() {
                        view.setPressed(false);
                    }
                }, 100);
            }

        }
    }

悬浮效果的实现:监听垂直方向上ScrollView的滑动事件,获取滑动的y值,控制悬浮view的显示和隐藏

@Override
    public void onScroll(int scrollY) {
        int height = mVScrollView.getTop();
        //判断滑动距离scrollY是否大于0,因为大于0的时候就是可以滑动了,此时mTabViewLayout.getTop()才能取到值。
        if (scrollY > 0 && scrollY >= height) {
            mTopLabView.setVisibility(View.VISIBLE);
        } else {
            mTopLabView.setVisibility(View.GONE);

        }
    }

悬浮栏左右滑动的联动效果实现:监听水平方向ScrollView的滑动事件,获取滑动的x值,实现标题栏和HScrollView的联动,达到一致的效果;

@Override
    public void onScrollX(int scrollX) {
        mTabDataView.setTranslationX(-scrollX);
    }

 

界面布局xml文件:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    >
    <com.sjczyy.mm.stocklist.view.SVerticallScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        >

        <LinearLayout
            android:focusableInTouchMode="true"
            android:orientation="vertical"
            android:layout_width="match_parent"
            android:layout_height="match_parent">

            <TextView
                android:layout_width="match_parent"
                android:layout_height="100dp"
                android:gravity="center"
                android:text="自定义展示内容"
                android:textSize="20sp"
                android:background="#00ff00"
                />

            <com.sjczyy.mm.stocklist.view.SVerticallScrollView
                android:id="@+id/scrollView_v"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:background="#f2f2f2"
                >
                <!-- android:descendantFocusability="blocksDescendants" 防止Scrollview自动定位到获得焦点的控件 -->
                <RelativeLayout
                    android:descendantFocusability="blocksDescendants"
                    android:layout_width="match_parent"
                    android:layout_height="match_parent">

                    <TextView
                        android:id="@+id/txt_groupname"
                        android:layout_width="80dp"
                        android:layout_height="50dp"
                        android:text="股票名称"
                        android:gravity="center"
                        />

                    <android.support.v7.widget.RecyclerView
                        android:id="@+id/rv_name"
                        android:layout_width="match_parent"
                        android:layout_height="wrap_content"
                        android:layout_below="@id/txt_groupname"
                        />

                    <com.sjczyy.mm.stocklist.view.SHorizontalScrollView
                        android:id="@+id/scrolllview_h"
                        android:layout_width="match_parent"
                        android:layout_height="wrap_content"
                        android:scrollbars="none"
                        android:layout_marginLeft="80dp"
                        >
                        <LinearLayout
                            android:orientation="vertical"
                            android:layout_width="match_parent"
                            android:layout_height="match_parent">
                            <include
                                layout="@layout/layout_stockdata_head"
                                />
                            <android.support.v7.widget.RecyclerView
                                android:id="@+id/rv_data"
                                android:layout_width="match_parent"
                                android:layout_height="wrap_content"
                                />

                        </LinearLayout>


                    </com.sjczyy.mm.stocklist.view.SHorizontalScrollView>

                </RelativeLayout>
            </com.sjczyy.mm.stocklist.view.SVerticallScrollView>
        </LinearLayout>
    </com.sjczyy.mm.stocklist.view.SVerticallScrollView>
    <RelativeLayout
        android:id="@+id/ll_tabtopview"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:orientation="horizontal"
        android:visibility="gone"
        android:background="#f2f2f2"
        android:clipChildren="false"
        android:gravity="center_vertical"
        >

        <FrameLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:clipChildren="false"
            android:layout_marginLeft="80dp">
            <include
                android:id="@+id/top_data_view"
                layout="@layout/layout_stockdata_head"
                android:layout_width="match_parent"
                android:layout_height="match_parent"

                />
        </FrameLayout>
        <TextView
            android:id="@+id/top_txt_groupname"
            android:layout_width="80dp"
            android:layout_height="50dp"
            android:text="股票名称"
            android:gravity="center"
            android:background="#f2f2f2"
            />


    </RelativeLayout>
</RelativeLayout>

 

整个项目的下载资源后续上传;

 

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Android 左右列表联动可以通过使用 RecyclerView联动滑动来实现。下面是一种可能的实现方式: 1. 在布局文件中添加两个 RecyclerView,一个放置于左侧,另一个放置于右侧。 2. 在 Activity 或 Fragment 中分别获取这两个 RecyclerView,并为它们设置 LinearLayoutManager。 3. 为左侧 RecyclerView 设置一个 OnScrollListener,在滑动过程中获取当前第一个可见的 item position,并将其传递给右侧 RecyclerView。 4. 右侧 RecyclerView 根据传递过来的 item position,滚动到对应的位置。 下面是一个简单的示例代码: ``` // 获取左侧 RecyclerView RecyclerView leftRecyclerView = (RecyclerView) findViewById(R.id.leftRecyclerView); LinearLayoutManager leftLayoutManager = new LinearLayoutManager(this); leftRecyclerView.setLayoutManager(leftLayoutManager); // 获取右侧 RecyclerView RecyclerView rightRecyclerView = (RecyclerView) findViewById(R.id.rightRecyclerView); LinearLayoutManager rightLayoutManager = new LinearLayoutManager(this); rightRecyclerView.setLayoutManager(rightLayoutManager); // 为左侧 RecyclerView 设置 OnScrollListener leftRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() { @Override public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { super.onScrolled(recyclerView, dx, dy); // 获取当前第一个可见的 item position int firstVisibleItemPosition = leftLayoutManager.findFirstVisibleItemPosition(); // 将其传递给右侧 RecyclerView rightLayoutManager.scrollToPosition(firstVisibleItemPosition); } }); ``` 当左侧 RecyclerView 滚动时,右侧 RecyclerView 会跟随滚动并展示对应位置的 item。需要注意的是,这里只是一个简单的示例代码,并没有考虑到复杂的布局和滚动效果,需要根据实际需求进行修改和完善。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值