點擊圖片大圖預覽,RecyclerView + PhotoView 滑動衝突,RecyclerView不回調SCROLL_STATE_IDLE

轉載請標明出處:juejin.im/post/5c99b5… ,謝謝。bash

背景

最近項目須要集成一個點擊圖片大圖預覽的功能。網上不少都是ViewPager + PhotoView的形式;此次想換個姿式,使用了 RecyclerView + PhotoView的方式,其中主要是RecyclerView的PagerSnapHelper來實現模擬ViewPager的翻頁效果。ide

問題

開發過程當中,其餘的卻是一切正常,主要遇到如下兩個問題:post

  1. PhotoView雙指放大 和 RecyclerView左右滑動 衝突,原本是想雙指放大PhotoView,結果RecyclerView滑動了;
  2. 爲實現ViewPager的onPageSelected監聽效果,給RecyclerView增長onScrollStateChanged監聽來判斷當前顯示的item,使用中發現當快速滾動到第一個時,SCROLL_STATE_IDLE狀態不會被回調。

解決

1. 針對第一個問題,目前的思路是自定義RecyclerView

判斷是不是多點觸控,是的話就放行給子PhotoView處理,不是就RecyclerView處理。可是這樣會有一個問題,就是兩個或者三個或更多手指在屏幕上時就沒辦法左右滑動RecyclerView了。可是我也以爲無所謂,畢竟應該沒有人用多個手指去左右翻頁。。。
代碼以下:ui

/**
 * 圖片預覽 RecyclerView
 * Create By lishilin On 2019/3/25
 */
public class PreviewRecyclerView extends RecyclerView {

    private boolean isLock;// 是否鎖住 RecyclerView ,避免和 PhotoView 雙指放大縮小操做衝突

    public PreviewRecyclerView(@NonNull Context context) {
        super(context);
    }

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

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

    @Override
    public boolean onInterceptTouchEvent(MotionEvent event) {
        switch (event.getActionMasked()) {
            case MotionEvent.ACTION_POINTER_DOWN:// 非第一個觸點按下
                isLock = true;
                break;
            case MotionEvent.ACTION_UP:// 最後一個觸點擡起
                isLock = false;
                break;
        }
        if (isLock) {
            return false;// 不攔截,交給子View處理
        }
        return super.onInterceptTouchEvent(event);
    }

    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        switch (event.getActionMasked()) {
            case MotionEvent.ACTION_POINTER_DOWN:// 非第一個觸點按下
                isLock = true;
                break;
            case MotionEvent.ACTION_UP:// 最後一個觸點擡起
                isLock = false;
                break;
        }
        return super.dispatchTouchEvent(event);
    }

}
複製代碼

2. 針對第二個問題,暫時發現兩種解決辦法。

  1. 在onScrollStateChanged回調中不判斷state狀態,這樣就繞過了SCROLL_STATE_IDLE的問題,可是會一直執行代碼,須要注意效率問題,特別是耗時操做,避免卡頓。
    代碼以下:
rv_photo.addOnScrollListener(new RecyclerView.OnScrollListener() {
            @Override
            public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);
                View view = snapHelper.findSnapView(layoutManager);
                if (view == null) {
                    return;
                }
                int position = layoutManager.getPosition(view);
                if (lastPosition == position) {
                    return;
                }
                lastPosition = position;
                refreshCountTips(lastPosition);
                PhotoPreviewRecyclerAdapter.ViewHolder holder = (PhotoPreviewRecyclerAdapter.ViewHolder) rv_photo.getChildViewHolder(view);
                if (holder == null || holder.img_content == null) {
                    return;
                }
                if (holder.img_content.getScale() != 1f) {
                    holder.img_content.setScale(1f, true);
                }
            }
        });
複製代碼
  1. 在上面的回調中正常加入SCROLL_STATE_IDLE的判斷,可是須要手動調用recyclerView.stopScroll()方法。
if (newState != RecyclerView.SCROLL_STATE_IDLE) {
                    return;
                }
複製代碼
rv_photo.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                switch (event.getActionMasked()) {
                    case MotionEvent.ACTION_UP:
                        if (rv_photo.getScrollState() == RecyclerView.SCROLL_STATE_IDLE) {
                            break;
                        }
                        View view = snapHelper.findSnapView(layoutManager);
                        if (view == null) {
                            break;
                        }
                        int position = layoutManager.getPosition(view);
                        if (position != 0) {
                            break;
                        }
                        if (rv_photo.getChildAt(0).getX() == 0 && rv_photo.canScrollHorizontally(1)) {
                            rv_photo.stopScroll();
                        }
                        break;
                }
                return false;
            }
        });
複製代碼
相關文章
相關標籤/搜索