热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android模仿知乎的回答详情页的动画效果

这篇文章主要介绍了Android模仿“知乎”的回答详情页的动画效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

 废话不多说,咱们第一篇文章就是模仿“知乎”的回答详情页的动画效果,先上个原版的效果图,咱们就是要做出这个效果


    在实现之前,我们先根据上面的动画效果,研究下需求,因为gif帧数有限,所以不是很连贯,推荐你直接下载一个知乎,找到这个界面自己玩玩
☞当文章往上移动到一定位置之后,最上面的标题栏Bar和问题布局Title是会隐藏的,回答者Author布局不会隐藏
☞当文章往下移动移动到一定位置之后,原先隐藏的标题栏Bar和问题布局Title会下降显示
☞当文章往上移动的时候,下部隐藏的Tools布局会上升显示
☞当文章往下移动的时候,如果Tools布局是显示的,则隐藏
☞当标题栏Bar和问题布局Title下降显示的时候,Title是从Bar的下面出来的,有个遮挡的效果
☞当快速滑动内容到达底部的时候,隐藏的Tools会显示出来
☞当快速滑动内容到顶部的时候,隐藏的Bar和Title也会显示出来

    不分析不知道,这样一个简单地效果,经过分析需要完成不少东西呢,那么下面根据要实现的需求,咱们分析一下解决方案。
    在做这种仿界面之前,我们可以使用ADT带的View Hierarchy工具看一下“知乎”原生是怎么实现的


    从右边的分析图可以看出,知乎的这个界面,内容用的WebView,这很正常,因为用户的回答里面格式比较复杂,用WebView是最好的解决方案,而标题栏是一个VIew,是ActionBar还是自定义View呢,不得而知,下面是就是一个LinearLayout包了4个ToggleButton,布局很简单,我们没有WebView,所以使用ScrollView代替,上面的布局直接ImageView了,设置个src,模拟一个布局。
    其实布局很简单,咱们一个效果一个效果的来实现。
    首先是下面的Tools如何显示和隐藏呢?当然是用动画了!什么动画呢?能实现的有属性动画和帧动画,属性动画能够真实的改变View的属性,帧动画只是视觉上移动了,View的实际属性并不改变,这两种都可以,我们这里使用属性动画

/** 
  * 显示工具栏 
  */ 
 private void showTools() { 
 
  ObjectAnimator anim = ObjectAnimator.ofFloat(img_tools, "y", img_tools.getY(), 
    img_tools.getY() - img_tools.getHeight()); 
  anim.setDuration(TIME_ANIMATION); 
  anim.start(); 
 
  isToolsHide = false; 
 } 
 
 /** 
  * 隐藏工具栏 
  */ 
 private void hideTools() { 
 
  ObjectAnimator anim = ObjectAnimator.ofFloat(img_tools, "y", img_tools.getY(), 
    img_tools.getY() + img_tools.getHeight()); 
  anim.setDuration(TIME_ANIMATION); 
  anim.start(); 
 
  isToolsHide = true; 
 
 } 

    那么什么时候调用呢?从上面的需求分析中,我们知道,用户手指下拉的时候,Tools显示,反之隐藏,那么我们就可以监听ScrollView的onTouch,判断手指方向,实现动画效果的调用

mScroller.setOnTouchListener(new View.OnTouchListener() { 
   @Override 
   public boolean onTouch(View v, MotionEvent event) { 
 
 
    switch (event.getAction()) { 
 
     case MotionEvent.ACTION_DOWN: 
      lastY = event.getY(); 
      break; 
     case MotionEvent.ACTION_MOVE: 
 
      float disY = event.getY() - lastY; 
 
      //垂直方向滑动 
      if (Math.abs(disY) > viewSlop) { 
       //是否向上滑动 
       isUpSlide = disY <0; 
 
       //实现底部tools的显示与隐藏 
       if (isUpSlide) { 
        if (!isToolsHide) 
         hideTools(); 
       } else { 
        if (isToolsHide) 
         showTools(); 
       } 
      } 
 
      break; 
    } 
 
    return false; 
   } 
  }); 

     用变量isToolsHide放置代码重复调用。

    下面的Tools的问题解决了,我们再看一下上面的布局动画如何来实现。上面的思路和下面一样,也是通过属性动画,完成位置的移动,移动的布局有Bar、Title和根布局。为什么答题人布局Author不移动呢?因为根布局必须移动,否则就会挡住下面的文字内容,根布局的移动会让子布局都跟着移动,所以只移动根布局即可。
    对了,为了更大范围的现实文本,“知乎”的WebView是占据整个布局的,其他布局都是在根布局FrameLayout里面,所以,在首次加载的时候,下面的文本在开头需要留出一定的间隔,防止被遮挡,当上面的布局隐藏之后,就没有问题了。
    在简单分析之后,我再给出实现的布局的代码

 
 
  
 
 
 

    效果图如下,文本留了一些空行,保证不被遮挡。

    有的同学看了上面的效果图可能会疑惑,这里为什么没有答题人的布局呢?
    其实是这样的,为了模拟上部的布局显示时,Title从Bar下面出现的效果,所以特意这样设计的。我试过用linearLayout实现,效果也是可以实现的,但是当Title往下移动显示的时候,会覆盖在Bar上面,这也很好理解,LinearLayout没有层次顺序,所以会遮挡。我试过View.bringToFront(),试图把Bar的布局提高层次,但是这样会导致布局的紊乱,在首次加载的时候,Bar会显示在最下面,是因为提高层次之后,Bar的布局重新计算,所以不按照LinearLayout的布局规则来了。无奈之下,换成了Framelayout,但是又出现了问题,Bar的高度可以设置,但是Title的高度会随着文本的增加而改变,这样一来,最下面Author的布局的位置就不能设置了,因为不知道距离上面多远,所以我们只能在代码里面动态的计算Bar和Title的高度,然后在界面加载的时候,动态的给Author的布局设置MargenTop,保证位置的正确。
    因为在onCreate里面,还没有开始View的绘制,所以得不到控件的真实高度,我们可以用下面的方法,获取这个时期的高度

//获取Bar和Title的高度,完成auther布局的margenTop设置 
  ViewTreeObserver viewTreeObserver = fl_top.getViewTreeObserver(); 
  viewTreeObserver.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() { 
   @Override 
   public boolean onPreDraw() { 
 
    if (!hasMeasured) { 
     FrameLayout.LayoutParams layoutParams = new FrameLayout.LayoutParams(FrameLayout 
       .LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.WRAP_CONTENT); 
     layoutParams.setMargins(0, img_bar.getHeight() + tv_title.getHeight(), 0, 0); 
     img_author.setLayoutParams(layoutParams); 
     hasMeasured = true; 
    } 
    return true; 
   } 
  }); 

     获取了高度之后,我们就可以正确地设置位置了。但是,如果保证上面的布局随着我们的内容的移动,而改变现实状态呢?
    经过我手动直观测试,知乎的这个界面是根据一个固定的值,来改变显示状态的,因此,我们可以监听ScrollView的滑动距离,来判断。但是ScrollView并没有给我们这样一个监听器,咋办?重写!

/** 
 * Created by zhaokaiqiang on 15/2/26. 
 */ 
public class MyScrollView extends ScrollView { 
 
 private BottomListener bottomListener; 
 
 private onScrollListener scrollListener; 
 
 
 public MyScrollView(Context context) { 
  this(context, null); 
 } 
 
 public MyScrollView(Context context, AttributeSet attrs) { 
  super(context, attrs); 
 } 
 
 protected void onScrollChanged(int l, int t, int oldl, int oldt) { 
  super.onScrollChanged(l, t, oldl, oldt); 
  if (getScrollY() + getHeight() >= computeVerticalScrollRange()) { 
 
   if (null != bottomListener) { 
    bottomListener.onBottom(); 
   } 
 
  } 
 
  if (null != scrollListener) { 
   scrollListener.onScrollChanged(l, t, oldl, oldt); 
  } 
 
 
 } 
 
 public void setBottomListener(BottomListener bottomListener) { 
  this.bottomListener = bottomListener; 
 } 
 
 public void setScrollListener(onScrollListener scrollListener) { 
 
  this.scrollListener = scrollListener; 
 
 } 
 
 
 public interface onScrollListener { 
 
  public void onScrollChanged(int l, int t, int oldl, int oldt); 
 
 } 
 
 
 public interface BottomListener { 
 
  public void onBottom(); 
 
 } 
 
 
} 

    我们只需要重写onScrollChange()方法即可,在里面不光可以时时的得到位置的变化,还添加了一个BottomListener接口来监听滑动到底部的事件,写好之后就很简单了

mScroller.setBottomListener(this); 
mScroller.setScrollListener(this); 

/** 
  * 显示上部的布局 
  */ 
 private void showTop() { 
 
  ObjectAnimator anim1 = ObjectAnimator.ofFloat(img_bar, "y", img_bar.getY(), 
    0); 
  anim1.setDuration(TIME_ANIMATION); 
  anim1.start(); 
 
  ObjectAnimator anim2 = ObjectAnimator.ofFloat(tv_title, "y", tv_title.getY(), 
    img_bar.getHeight()); 
  anim2.setInterpolator(new DecelerateInterpolator()); 
  anim2.setDuration(TIME_ANIMATION + 200); 
  anim2.start(); 
 
  ObjectAnimator anim4 = ObjectAnimator.ofFloat(fl_top, "y", fl_top.getY(), 
    0); 
  anim4.setDuration(TIME_ANIMATION); 
  anim4.start(); 
 
  isTopHide = false; 
 } 
 
 
 /** 
  * 隐藏上部的布局 
  */ 
 private void hideTop() { 
 
  ObjectAnimator anim1 = ObjectAnimator.ofFloat(img_bar, "y", 0, 
    -img_bar.getHeight()); 
  anim1.setDuration(TIME_ANIMATION); 
  anim1.start(); 
 
  ObjectAnimator anim2 = ObjectAnimator.ofFloat(tv_title, "y", tv_title.getY(), 
    -tv_title.getHeight()); 
  anim2.setDuration(TIME_ANIMATION); 
  anim2.start(); 
 
  ObjectAnimator anim4 = ObjectAnimator.ofFloat(fl_top, "y", 0, 
    -(img_bar.getHeight() + tv_title.getHeight())); 
  anim4.setDuration(TIME_ANIMATION); 
  anim4.start(); 
 
  isTopHide = true; 
 } 
 
 @Override 
 public void onBottom() { 
  if (isToolsHide) { 
   showTools(); 
  } 
 } 
 
 @Override 
 public void onScrollChanged(int l, int t, int oldl, int oldt) { 
 
  if (t <= dp2px(TOP_DISTANCE_Y) && isTopHide && isAnimationFinish) { 
   showTop(); 
   Log.d(TAG, "显示"); 
  } else if (t > dp2px(TOP_DISTANCE_Y) && !isTopHide && isAnimationFinish) { 
   hideTop(); 
   Log.d(TAG, "隐藏"); 
  } 
 } 

    我们只需要根据当前的位置,来实现布局的显示和隐藏就可以啦!

    OK,这篇文章就到这里,希望对大家的学习有所帮助。


推荐阅读
  • 作为一名在大型手机游戏公司工作的程序员,尽管主要负责游戏逻辑和内容的开发,但对iOS底层开发接触较少。现在有了iPhone和可以虚拟MAC环境的电脑,希望能找到有效的iOS开发学习路径。 ... [详细]
  • Android 九宫格布局详解及实现:人人网应用示例
    本文深入探讨了人人网Android应用中独特的九宫格布局设计,解析其背后的GridView实现原理,并提供详细的代码示例。这种布局方式不仅美观大方,而且在现代Android应用中较为少见,值得开发者借鉴。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 本文探讨了如何在不重新加载URL的情况下,触发WebView的PictureListener.onNewPicture()方法,以实现页面的重新绘制或渲染。 ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 本文介绍如何使用布局文件在Android应用中排列多行TextView和Button,使其占据屏幕的特定比例,并提供示例代码以帮助理解和实现。 ... [详细]
  • 本文介绍了Android开发中Intent的基本概念及其在不同Activity之间的数据传递方式,详细展示了如何通过Intent实现Activity间的跳转和数据传输。 ... [详细]
  • 优化 Android 按钮状态下的背景和文本颜色变化
    本文介绍如何通过 Android 的 Selector 实现按钮在不同状态下(如按压)的背景和文本颜色动态变化。我们将详细讲解实现步骤,并提供完整的代码示例。 ... [详细]
  • 本文详细介绍超文本标记语言(HTML)的基本概念与语法结构。HTML是构建网页的核心语言,通过标记标签描述页面内容,帮助开发者创建结构化、语义化的Web页面。 ... [详细]
  • 本文回顾了2017年的转型和2018年的收获,分享了几家知名互联网公司提供的工作机会及面试体验。 ... [详细]
  • 本文详细介绍了如何在Android 4.4及以上版本中配置WebView以实现内容的自动高度调整和屏幕适配,确保中文显示正常,并提供代码示例。 ... [详细]
  • Web App vs Native App:未来的移动应用趋势
    随着移动互联网的发展,Web App和Native App之间的竞争日益激烈。对于开发者而言,选择哪一种技术路径更为明智?本文将深入探讨两种应用模式的特点及未来趋势。 ... [详细]
  • 本文详细介绍了如何在Android应用中处理和使用Cookie,包括登录时获取Cookie并将其存储,以及在WebView中同步这些Cookie以保持用户会话。 ... [详细]
author-avatar
几杯茶轶事
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有