0

I am getting a very weird behaviour on scrolling the views. Fragment "A" uses the RecyclerView and Fragment "B" uses the listView.

For recycler view, I am using the following code:

list.setOnScrollListener(new RecyclerView.OnScrollListener() {
        int mLastFirstVisibleItem = 0;

        @Override
        public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
        }

        @Override
        public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
            super.onScrolled(recyclerView, dx, dy);
            final int currentFirstVisibleItem = linearLayoutManager.findFirstVisibleItemPosition();

            if (currentFirstVisibleItem > this.mLastFirstVisibleItem) {
                ((AppCompatActivity)getActivity()).getSupportActionBar().hide();
            } else if (currentFirstVisibleItem < this.mLastFirstVisibleItem) {
                ((AppCompatActivity)getActivity()).getSupportActionBar().show();
            }

            this.mLastFirstVisibleItem = currentFirstVisibleItem;
        }
    });

For ListView, I am using the following code:

list.setOnScrollListener(new AbsListView.OnScrollListener() {
        int mLastFirstVisibleItem = 0;

        @Override
        public void onScrollStateChanged(AbsListView view, int scrollState) {   }

        @Override
        public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
            if (view.getId() == list.getId()) {
                final int currentFirstVisibleItem = list.getFirstVisiblePosition();

                if (currentFirstVisibleItem > mLastFirstVisibleItem) {
                    // getSherlockActivity().getSupportActionBar().hide();
                    ((AppCompatActivity)getActivity()).getSupportActionBar().hide();
                } else if (currentFirstVisibleItem < mLastFirstVisibleItem) {
                    // getSherlockActivity().getSupportActionBar().show();
                    ((AppCompatActivity)getActivity()).getSupportActionBar().show();
                }

                mLastFirstVisibleItem = currentFirstVisibleItem;
            }
        }
    });

Following are the GIFs, So, basically, I scroll and hold it right there and don't release my finger and I get the following behaviour.

enter image description here

enter image description here

Karanveer Singh
  • 961
  • 12
  • 27

2 Answers2

1

If the appBar behavior you are looking for is similar to for example WhatsApp, I think a better approach is to control the appBar behavior from your layout instead of from the Java. Here is an example of how to do this for both a fragment containing a RecyclerView and one containing a ListView:

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <android.support.design.widget.AppBarLayout
        android:id="@+id/appbarlayout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            app:layout_collapseMode="pin"
            app:layout_scrollFlags="scroll|enterAlways|snap"/>

        <android.support.design.widget.TabLayout
            android:id="@+id/tablayout"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>

    </android.support.design.widget.AppBarLayout>

    <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior" />

</android.support.design.widget.CoordinatorLayout>

fragmenta.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout 
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    xmlns:android="http://schemas.android.com/apk/res/android">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/rec_fragmenta"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:splitMotionEvents="false"
        android:clipToPadding="false"/>

</android.support.design.widget.CoordinatorLayout>

fragmentb.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout 
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    xmlns:android="http://schemas.android.com/apk/res/android">

    <ListView
        android:id="@+id/list_fragmentb"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:nestedScrollingEnabled="true"/>

</android.support.design.widget.CoordinatorLayout>

MainActivity.java

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        TabLayout tabLayout = (TabLayout) findViewById(R.id.tablayout);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);
        SectionsPagerAdapter mSectionsPagerAdapter = new 
        SectionsPagerAdapter(getSupportFragmentManager());
        ViewPager mViewPager = (ViewPager) findViewById(R.id.viewpager);
        tabLayout.setupWithViewPager(mViewPager);
        mViewPager.setAdapter(mSectionsPagerAdapter);
    }


    private class SectionsPagerAdapter extends FragmentPagerAdapter {

        private SectionsPagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            switch (position) {
                case 0:
                    return new FragmentA();
                case 1:
                    return new FragmentB();
                case 2:
                    return new FragmentC();
            }
            return null;
        }

        @Override
        public int getCount() {
            return 3;
        }

        @Override
        public CharSequence getPageTitle(int position) {
            switch (position) {
                case 0:
                    return "A";
                case 1:
                    return "B";
                case 2:
                    return "C";
            }
            return null;
        }
    }
}

FragmentA

public class FragmentA extends Fragment {

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View rootView = inflater.inflate(R.layout.fragmenta, container, false);
        LinearLayoutManager layoutManager = new LinearLayoutManager(getActivity());
        RecyclerView recyclerView = rootView.findViewById(R.id.rec_fragmenta);
        recyclerView.setLayoutManager(layoutManager);
        List<String> list = getList();
        MyListAdapter adapter = new MyListAdapter(list);
        recyclerView.setAdapter(adapter);
        return rootView;
    }
}

FragmentB

public class FragmentB extends Fragment {

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View rootView = inflater.inflate(R.layout.fragmentb, container, false);
        ListView listView = rootView.findViewById(R.id.list_fragmentb);
        List<String> list = getList();
        ArrayAdapter<String> adapter = new ArrayAdapter<>(getActivity(),
                android.R.layout.simple_list_item_1, android.R.id.text1, list);
        listView.setAdapter(adapter);
        return rootView;
    }
}
BramH
  • 221
  • 5
  • 24
0

When the Toolbar gets hidden, the viewport of the ListView & RecyclerView gets extended to top, showing again the 1st item and therefore showing the Toolbar again.

You should check if variable dy is positive or negative; if dy is positive then user is scrolling up, else down. If we take this the code becomes like this:

@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
    super.onScrolled(recyclerView, dx, dy);
    final int currentFirstVisibleItem = linearLayoutManager.findFirstVisibleItemPosition();

    if (currentFirstVisibleItem > this.mLastFirstVisibleItem && dy > 0) {
        ((AppCompatActivity)getActivity()).getSupportActionBar().hide();
    } else if (currentFirstVisibleItem < this.mLastFirstVisibleItem && dy < 0) {
        ((AppCompatActivity)getActivity()).getSupportActionBar().show();
    }

    this.mLastFirstVisibleItem = currentFirstVisibleItem;
}

or you can just ignore properties like firstVisibleItem and just use dy as well (if you don't really need to check for the 1st item is shown):

@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
    super.onScrolled(recyclerView, dx, dy);

    if (dy > 0) {
        ((AppCompatActivity)getActivity()).getSupportActionBar().hide();
    } else if (dy < 0) {
        ((AppCompatActivity)getActivity()).getSupportActionBar().show();
    }
}
shiftpsh
  • 1,926
  • 17
  • 24