全局实现点击TitleBar滚动到顶部
来源:互联网 发布:谁与知同梅飘雪 编辑:程序博客网 时间:2024/04/26 01:18
前几天产品拿着Android App问我们为什么他点击通知栏或者TitleBar都不回滚动到顶部,这不是系统自带的么?这还真不是,苹果是自带功能,而有些安卓厂商也有自实现(例如锤子或魅族?),但毕竟不是Android系统自带,所以我们就考虑在我们应用实现此功能。
与系统功能不同,我们是在应用里的TitleBar里实现点击滚动到顶部的,单界面实现这个功能并不难,只要在TitleBar里设置点击/双击事件,然后把对应的可滚动的View传给它,让它去调用scrollToTop滚动方法,如ListView和GridView的smoothScrollToTop()方法就可以了。但如果要在每个界面都实现这功能,我们当然不能每个界面都这样去设置一遍,这样太蠢后期也不好维护,我们分析了一下,既然是所有有滚动的界面都要有此功能,那么只能在基类里面实现了,最好做到对所有具体界面无感,不改具体界面每一行代码。针对我们现有的界面架构(TitleBar有抽象出来,默认会动态添加到Activity里),我认为实现难度应该不大,但有几个问题需要解决:
如何获取当前界面可滚动的View,如果有多个可滚动的View那么该滚动哪个?
我们的Activity与Fragment的界面搭配较多,一个Activity里的不同Fragment来回切换时怎么把当前的Fragment里可滚动View传给Titlebar
Fragment里面如果再多次嵌套Fragment,如何实现上面遇到的问题
如果是滑动的item数较多时,怎么一下子就滑动到顶部,而不是按匀速缓慢的移动到顶部
带着上面这些问题,我扑哧扑哧的干了起来,一个个将它们击破:
问题1:
由于我们界面没有多块区域可滚动的情况,所以我们只要从界面的最下层遍历上来,以第一个可滚动的View为准就可以,实现代码如下:
/** * 遍历找到viewGroup下面的第一个可滑动的View * @param viewGroup * @return 第一个可滑动的子View */public static View findFirstCanScrollView(ViewGroup viewGroup) { if (viewGroup != null) { for (int i = 0, N = viewGroup.getChildCount(); i < N; i++) { View child = viewGroup.getChildAt(i); if (child instanceof AbsListView) { Log.i("findFirstCanScrollView", "get AbsListView"); return child; } else if (child instanceof ScrollView) { Log.i("findFirstCanScrollView", "get ScrollView"); return child; } else if (child instanceof WebView) { Log.i("findFirstCanScrollView", "get WebView"); return child; } else if (child instanceof ViewGroup) { View scrollView = findFirstCanScrollView((ViewGroup) child); if (scrollView != null) { return scrollView; } } } } return null;}
从上面的代码可以看到,只要把我们界面的最下层的ViewGroup传进来,只要找到任意一个可滑动的View直接就返回,我们现在用到的可滚动的View总共有几类,ListView/GridView/ScrollView/WebView,现在RecyclerView用得还比较少,等用到了再加上去。这种方式实现唯一的一个缺点就是可能引起性能问题,但由于我们界面还不算特别复杂,从我多次测试结果上看,遍历一次最多耗时不回超过5ms,大部分情况是0/1/2ms,且一般只会在初始化遍历一次,所以此方法是可行的。
问题2:
下图是我们界面最为复杂的一个界面框架草图:
先说一下Activity和Fragment搭配的界面,由于我们的TitleBar是依附于Activity,所以当前的Fragment就得去设置Activity里TitleBar的可滑动View,这只需要在BaseFragment里做就可以:
/** * 设置Activity的滑动View * @param canScrollView */ protected void setSmoothToTopView(View canScrollView) { if (getActivity() instanceof BaseActivity) { ((BaseActivity)getActivity()).setSmoothToTopView(canScrollView); } }
从上图可以看到,现在APP一般首页界面都由几个Tab,当点击底部不同Tab时,切换到不同的Fragment,这种情况下所以每次切换Tab时,都要重新设置Activity里的ScrollView。在Android系统里,不同Fragment的切换通常情况下有两种方式:
一种是用FragmentManager来hide和show不同的Fragment,这种情况每次show和hide都会回调Fragment里的onHiddenChanged(boolean hidden)方法,只要复写这个方法就可以。
一种是用ViewPager配合Fragment的切换,这种情况下会回调Fragment里面的setUserVisibleHint(boolean isVisibleToUser),在显示的Fragment传入true,在隐藏的Fragment传入false
问题3:
从上面的app草图里面可看到,在Titlebar的下面还有另外一排tab,这样的界面在安卓的app上也能经常看到,但这种界面会碰到新的问题:例如你在这个界面点击顶部的了TAB3,但是你又点击了底部其他的TAB,待会重新点回有头部tab的fragment,你会发现,这时并不会调用上面tab相应fragment的setUserVisibleHint(boolean isVisibleToUser)方法(一般上面的tab用ViewPager来实现),这也能理解,因为顶部tab是切换顶部才会调用,切换底部跟它没关系。
此时我们能做的办法就是,主动调用一次刚才顶部显示的fragment 的setUserVisibleHint(boolean isVisibleToUser)方法,代码如下:
/** * viewPager显示或隐藏Fragment时的回调 * @param isVisibleToUser */ @Override public void setUserVisibleHint(boolean isVisibleToUser) { super.setUserVisibleHint(isVisibleToUser); //去遍历找 processFirstVisibleToUser(isVisibleToUser); if (isVisibleToUser) { /** * 用于fragment嵌套fragment的情况,当点击父fragment时,父fragment也要回调子fragment的setUserVisibleHint方法 */ List<Fragment> fragments = getChildFragmentManager().getFragments(); if (fragments != null && fragments.size() > 0) { for (Fragment fragment : fragments) { if (fragment != null && fragment.getUserVisibleHint()) { fragment.setUserVisibleHint(true); } } } } }
关于Activity和Fragment的搭配还碰到一个问题,是当Fragment只是占Activity的一小块区域,而此时的Activity里的主要区域又是可滑动的,这时Fragment会把Activity的ScrollView冲掉,这种情况我们只能提供一个特殊处理办法,对于这种没有ScrollView或者那些不想滚动到顶部的Fragement,让它们去复写父类的方法:
/** * 有些界面不需要滑动到顶部,可复写此方法,返回false * @return */ protected boolean isScrollToTop() { return true; }
问题4:
对于滚动缓慢的问题比较好办,AbsListView这种类型,比如你已经滑动到100行了,想一下子滚动到顶部,可以先调用setSelection(10),然后再调用smoothScrollToPosition(0),从滑动效果上看没破绽。对于WebView,由于它没有smoothScrollToTop之类的方法,可以用ObjectAnimator来动态改变它的ScrollY轴,这种方式只要设置duration就好。TitleBar里的调用滚动的方法如下:
/** * 双击操作,目前只对WebView,ScrollView,AbsListView做操作 */ private void doubleClick() { if (mSmoothToTopView != null) { if (mSmoothToTopView instanceof AbsListView) { AbsListView smoothToTopView = (AbsListView) mSmoothToTopView; int firstVisiblePosition = smoothToTopView.getFirstVisiblePosition(); if (firstVisiblePosition < 11) { smoothToTopView.smoothScrollToPosition(0); } else { //太多item时加快速度 smoothToTopView.setSelection(10); smoothToTopView.smoothScrollToPosition(0); } smoothToTopView.clearFocus(); } else if (mSmoothToTopView instanceof ScrollView) { ScrollView scrollView = ((ScrollView) mSmoothToTopView); scrollView.smoothScrollTo(scrollView.getScrollX(), 0); } else if (mSmoothToTopView instanceof WebView) { ObjectAnimator anim = ObjectAnimator.ofInt(mSmoothToTopView, "scrollY", mSmoothToTopView.getScrollY(), 0); anim.setDuration(200).start(); } } }
看到这里你可能会发现,把细节封装的基类虽然很好,但要处理和兼容很多特殊情况,这也是一个框架鲁棒性的体现。
- 全局实现点击TitleBar滚动到顶部
- 点击滚动到顶部
- TableView,点击状态栏滚动到顶部
- js实现超过页面一屏后,点击图标滚动到页面顶部top
- 多个tableview 点击状态栏当前scrollView滚动到顶部
- jquery设置点击标签,标签滚动到页面顶部
- 滚动到页面顶部
- UITableView滚动到顶部
- UITableView滚动到顶部
- tablebview滚动到顶部
- scrollview滚动到顶部
- 滚动到顶部
- UITextView滚动到顶部
- 滚动到顶部 html
- jq滚动到顶部
- 点击按钮滚动至顶部
- jQuery 实现 Scroll to Top 滚动到页面顶部
- JS实现滚动监听以及滑动到顶部
- Unity UI(四):Text、Image/RawImage和Mask
- Java并发编程(Java Concurrency)(4) - 并发模型
- const限定符
- 字符串数组、预处理、多文件编程
- 关于蓝牙技术GATT属性介绍。
- 全局实现点击TitleBar滚动到顶部
- nodejs开发一个web小demo
- 51NOD 1024 矩阵中不重复的元素
- lr-svm
- 《Javascript DOM编程艺术》第2版 知识点汇总
- Android Studio 小技巧合集
- 浏览器input file 转换base 64
- PAT乙级(Basic Level)1010(C++)
- mybatis绑定错误