大家看到微信首页切换效果有没有觉得很炫,滑动切换,点击底部bar瞬间切换,滑动切换渐变效果,线上效果图:

之前也在博客上看到别人的实现,再次基础上,我做了些优化。首先说下实现原理,大神略过,o(╯□╰)o

页面上看到的三个页面是三个Fragment, 左右滑动使用viewpager,相信大家也都是这么再用,那么底部用的是什么技术呢,底部渐变其实就是重写了ImageView,以及在左右滑动时,改变了TextView的颜色值,是不是很简单...下面我们一步一步的来:

1.自定义ImageView:

  1. /**
  2. * 初始化资源图片bitmap及相关绘制对象
  3. * @param normal normals
  4. * @param selected focus
  5. */
  6. public final void init(int normal, int selected, int width, int height) {
  7. this.mNormalIcon = createBitmap(normal);
  8. this.mSelectedIcon = createBitmap(selected);
  9. this.mNormalRect = new Rect(0, 0, width, height);
  10. this.mSelectedRect = new Rect(0, 0, width, height);
  11. this.mPaint = new Paint(1);
  12. }

这里定义了两个Bitmap,分别对应获得焦点和失去焦点时显示的bitmap图像,两个矩阵,在绘制过程中使用到,定义了一个外部调用的方法,在左右滑动过程中,通过偏移值改变透明值,两张图片叠加就是对应的过度效果。

然后通通过滑动过程中不断刷新view完成重新绘制,由此有了重写onDraw方法:

  1. @Override
  2. protected void onDraw(Canvas canvas) {
  3. super.onDraw(canvas);
  4. if (this.mPaint == null) {
  5. return;
  6. }
  7. this.mPaint.setAlpha(255 - this.mSelectedAlpha);
  8. canvas.drawBitmap(this.mNormalIcon, null, this.mNormalRect, this.mPaint);
  9. this.mPaint.setAlpha(this.mSelectedAlpha);
  10. canvas.drawBitmap(this.mSelectedIcon, null, this.mSelectedRect, this.mPaint);
  11. }

这里可以看到同伙Paint改变传入的两个bitmap透明度,从而达到渐变效果,其中mSelectedAlpha为外部传入透明值

2.自定义实现底部bar容器,这里通过重写LinearLayout实现(姑且叫做container),  在container中我们要做这么几件事:

1).定义外表调用接口,接收底部显示资源信息:

a.首先是初始化参数:

  1. public void initContainer (String[] titles, int[][] iconsRes, int[] colors, boolean showTransitionColor) {
  2. this.mTitles = titles;
  3. this.mIconRes = iconsRes;
  4. this.mTextNormalColor = getResources().getColor(colors[0]);
  5. this.mTextSelectedColor = getResources().getColor(colors[1]);
  6. this.mShowTransitionColor = showTransitionColor;
  7. }

这里传入了tab显示的文字数组、显示的图片资源数组、默认颜色和获得焦点时颜色值数组(数组大小=2),以及切换时是否显示过渡效果

b.设置布局文件及布局文件里对应的控件ID、显示图片时图片宽高参数,提供了三种方式:

①图文tab:

  1. /**
  2. * 设置布局文件及相关控件id
  3. * @param layout layout布局文件 id
  4. * @param iconId ImageView 控件 id id <=0 时不显示
  5. * @param textId TextView 控件 id id <=0 时不显示
  6. * @param width icon 宽度
  7. * @param height icon 高度
  8. */
  9. public void setContainerLayout (int layout, int iconId, int textId, int width, int height) {
  10. mLayoutId = layout;
  11. mTextViewId = textId;
  12. mIconVIewId = iconId;
  13. mIconWidth = width;
  14. mIconHeight = height;
  15. }

这里的layout及tab的布局文件, iconId对应的是自定义ImageView的资源Id, textId对应的是TextView的Id, 宽高指的是图片显示的宽高

②只有文字tab: 只显示文字tab时传入iconId即可

③只有图片tab: 相应的,是在图文tab提供的方法上,传入文本textId=0即可

c.注入ViewPager:这里需要监听ViewPager的滑动来改变渐变色

2).添加tab到容易container中:

这里需要判断iconId以及TextId是否大于0,=0即不显示,同时为了居中平分底部container长度, 所有tab等分底部container

  1. /**
  2. * <p>添加tab view到当前容器</p>
  3. */
  4. private void addTabViewToContainer() {
  5. final PagerAdapter adapter = mViewPager.getAdapter();
  6. mTabView = new View[adapter.getCount()]; //这里根据adapter判断底部要显示的tab总数
  7.  
  8. for (int index = 0, len = adapter.getCount(); index < len; index++) {
  9.  
  10. final View tabView = LayoutInflater.from(getContext()).inflate(mLayoutId, this, false); //加载tab布局
  11. mTabView[index] = tabView;
  12.  
  13. /*tabIconView初始化*/
  14. TabIconView iconView = null;
  15. if (mIconVIewId > 0) { // 传入的图片资源文件ID不为0时,表示需要显示icon,然后初始化该View
  16. iconView = (TabIconView) tabView.findViewById(mIconVIewId);
  17. iconView.init(mIconRes[index][0], mIconRes[index][1], mIconWidth, mIconHeight); //这里调了自定义ImageView的init方法
  18. }
  19.  
  20. /*tabTextView初始化*/
  21. TextView textView = null;
  22. if (mTextViewId > 0) {
  23. textView = (TextView) tabView.findViewById(mTextViewId);
  24. textView.setText(mTitles[index]);
  25.  
  26. }
  27.  
  28. /*设置宽度,等分container*/
  29. LayoutParams lp = (LayoutParams) tabView.getLayoutParams();
  30. lp.width = 0;
  31. lp.weight = 1;
  32.  
  33. /*添加tab点击事件*/
  34. addTabOnClickListener(tabView, index);
  35.  
  36. /*设置当前状态*/
  37. if (index == mViewPager.getCurrentItem()) { //当先显示tab,设置初始状态为获得焦点状态
  38. if (iconView != null) {
  39. iconView.offsetChanged(0);
  40. }
  41. tabView.setSelected(true);
  42. if (textView != null) {
  43. textView.setTextColor(mTextSelectedColor);
  44. }
  45. }
  46.  
  47. addView(tabView);
  48. }
  49. }

3).监听viewPager的滑动事件,根据偏移值更新container,完成重绘操作

4).在container的onDraw中根据偏移量计算透明值,这里文本偏移值计算用了一个开源的代码

  1. @Override
  2. protected void onDraw(Canvas canvas) {
  3. super.onDraw(canvas);
  4. final int childCount = getChildCount();
  5. if (childCount > 0) {
  6. /*当发生偏移时,绘制渐变区域*/
  7. if (mSelectionOffset > 0f && mSelectedPosition < (getChildCount() - 1) && mShowTransitionColor) {
  8.  
  9. /*获取当前tab和下一tab view */
  10. View selectedTab = getChildAt(mSelectedPosition);
  11. View nextTab = getChildAt(mSelectedPosition + 1);
  12.  
  13. /*显示tab icon时,刷新各自view 透明度*/
  14. if (mIconVIewId > 0) {
  15. View selectedIconView = selectedTab.findViewById(mIconVIewId);
  16. View nextIconView = nextTab.findViewById(mIconVIewId);
  17.  
  18. //draw icon alpha
  19. if (selectedIconView instanceof TabIconView && nextIconView instanceof TabIconView) {
  20. ((TabIconView) selectedIconView).offsetChanged(mSelectionOffset);
  21. ((TabIconView) nextIconView).offsetChanged(1 - mSelectionOffset);
  22. }
  23. }
  24.  
  25. /*显示tab text,刷新各自view 透明度*/
  26. if (mTextViewId > 0) {
  27. View selectedTextView = selectedTab.findViewById(mTextViewId);
  28. View nextTextView = nextTab.findViewById(mTextViewId);
  29.  
  30. //draw text color
  31. Integer selectedColor = (Integer) evaluate(mSelectionOffset, mTextSelectedColor, mTextNormalColor);
  32. Integer nextColor = (Integer) evaluate(1 - mSelectionOffset, mTextSelectedColor, mTextNormalColor);
  33.  
  34. if (selectedTextView instanceof TextView && nextTextView instanceof TextView) {
  35. ((TextView) selectedTextView).setTextColor(selectedColor);
  36. ((TextView) nextTextView).setTextColor(nextColor);
  37. }
  38. }
  39.  
  40. }
  41. }
  42. }

3.定义个FragmentAdapter,这个就略过,比较简单了

4.做了以上准备工作,就可以写个测试例子试试效果了,当然这里为了看到效果,我们需要事先准备好几张图片,以及几个fragment

  1. private void initViews() {
    //得到apdater
  2. TabFragmentAdapter mAdapter = new TabFragmentAdapter(getSupportFragmentManager(), fragments);
  3. ViewPager mPager = (ViewPager) findViewById(R.id.tab_pager);
  4. mPager.setAdapter(mAdapter);

  5. //如果当前类需要对viewPager做监听
  6. TabContainerView mTabLayout = (TabContainerView) findViewById(R.id.ll_tab_container);
  7. mTabLayout.setOnPageChangeListener(this);
  8.  
  9. mTabLayout.initContainer(getResources().getStringArray(R.array.tab_main_title), ICONS_RES, TAB_COLORS, true);
  10.  
  11. int width = getResources().getDimensionPixelSize(R.dimen.tab_icon_width);
  12. int height = getResources().getDimensionPixelSize(R.dimen.tab_icon_height);
  13. mTabLayout.setContainerLayout(R.layout.tab_container_view, R.id.iv_tab_icon, R.id.tv_tab_text, width, height);
  14. // mTabLayout.setSingleTextLayout(R.layout.tab_container_view, R.id.tv_tab_text);
  15. // mTabLayout.setSingleIconLayout(R.layout.tab_container_view, R.id.iv_tab_icon);
  16.  
  17. mTabLayout.setViewPager(mPager);
  18. mPager.setCurrentItem(getIntent().getIntExtra("tab", 0));
  19. }

ManActivity对应的xml就比较简单了,可以参考源码,最后运行效果,就是上面的贴图了,到此防微信的滑动切换就完成了,源码请访问以下链接:

源码下载:https://github.com/JarekWang/wechathome.git

Android防微信首页左右滑动切换的更多相关文章

  1. [Android] 使用Include布局+Fragment滑动切换屏幕

        前面的文章已经讲述了"随手拍"项目图像处理的技术部分,该篇文章主要是主界面的布局及屏幕滑动切换,并结合鸿洋大神的视频和郭神的第一行代码(强推两人Android博客),完毕了 ...

  2. 【Android UI】案例03滑动切换效果的实现(ViewPager)

    本例使用ViewPager实现滑动切换的效果.本例涉及的ViewPager.为android.support.v4.view.ViewPager.所以须要在android项目中导入android-su ...

  3. Android仿微信图片上传,可以选择多张图片,缩放预览,拍照上传等

    仿照微信,朋友圈分享图片功能 .可以进行图片的多张选择,拍照添加图片,以及进行图片的预览,预览时可以进行缩放,并且可以删除选中状态的图片 .很不错的源码,大家有需要可以下载看看 . 微信 微信 微信 ...

  4. Android 深入ViewPager补间动画,实现类京东商城首页广告Banner切换效果

    如有转载,请声明出处: 时之沙: http://blog.csdn.net/t12x3456 某天看到京东商城首页的滑动广告的Banner,在流动切换的时候有立体的动画效果,感觉很有意思,然后研究了下 ...

  5. Android之fragment点击切换和滑动切换结合

    学了一小段时间的Android,主要接触的是UI设计,打交道最多莫过于fragment了吧.在Android3.0引入了fragment的概念后,几乎在所以的Android的应用中都可以看见其身影,已 ...

  6. Android自定义顶部栏及侧滑菜单和fragment+viewpag滑动切换的实现

    嘿嘿嘿,关于android滑动的操作,是不是经常都会用到呢. 我肯定也要学习一下啦. https://blog.csdn.net/u013184970/article/details/82882107 ...

  7. Android的Activity屏幕切换动画(一)-左右滑动切换

    (国内知名Android开发论坛eoe开发者社区推荐:http://www.eoeandroid.com/) Android的Activity屏幕切换动画(一)-左右滑动切换 在Android开发过程 ...

  8. Android:简单实现ViewPager+TabHost+TabWidget实现导航栏导航和滑动切换

    viewPager是v4包里的一个组件,可以实现滑动显示多个界面. android也为viewPager提供了一个adapter,此adapter最少要重写4个方法: public int getCo ...

  9. Android:使用ViewPager实现左右滑动切换图片(图上有点点)

    在以下实例的基础上加上点点 Android:使用ViewPager实现左右滑动切换图片 (简单版) 效果预览: 因为要把点点放图片上,所以修改布局为相对布局: <?xml version=&qu ...

随机推荐

  1. hdfs[命令] fsck

    Usage: DFSck <path> [-list-corruptfileblocks | [-move | -delete | -openforwrite] [-files [-blo ...

  2. VB IE 清除历史记录

    VB删除Cookie,仅适用于IE7版本 IE7版本为我们提供了命令行删除Cookie,清除临时文件缓存,清除历史记录表单的方法,下面是详细的命令运行方式. '注:以下代码仅支持IE7. '清除Int ...

  3. 第二百五十一天 how can I 坚持

    hadoop,namenote和datanode.namenode如果要是在启动时加载到内存,会不会对内存的要求比较高呢. edits-->fsimage. secondnamenode,那么n ...

  4. 现代程序设计——homework-09

    Lambda表达式 // homework-09.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <iostream ...

  5. POJ 2240 Arbitrage (求负环)

    Arbitrage 题目链接: http://acm.hust.edu.cn/vjudge/contest/122685#problem/I Description Arbitrage is the ...

  6. thymeleaf学习

    一.简单表达格式:   thymeleaf的官方参考文档 1.变量的表达式:${...} 2.选择变量表达式:*{...} 3.信息表达:#{...} 4.链接URL表达式:@{...} 二.字面值 ...

  7. hdu 1443 Joseph (约瑟夫环)

    Joseph Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

  8. codeforces 630B Moore's Law

    B. Moore's Law time limit per test 0.5 seconds memory limit per test 64 megabytes input standard inp ...

  9. 在 Web 层应用程序中使用Spring

    前面已经配置成功后,就可以在Web 层的Servlet或Jsp中调用访问Spring了,如果你 编制的是一个Servlet/Jsp 程序,那么在你的Servlet/Jsp 使用下面的代码通过Sprin ...

  10. ecshop后台限制IP登录

    ecshop是开源系统,所以难免会有漏洞   黑客攻击网站,往往是通过漏洞获取后台管理员权限,然后再做一些破坏 如果我们在后台文件里限制指定的IP才能登录后台,就相对安全多了 下面给出大家解决方案: ...