一起撸个微信图片浏览的BaseActivity吧(上)——初步思考与基础结构

本项目git: https://github.com/razerdp/ZoomViewActivity

【下篇】一起撸个微信图片浏览的BaseActivity吧(下)——过渡动画的实现

项目预览图:

距离上次更新博客有两三个月了。。。。太懒了orz...

在微信的日常使用中,我们点击图片放大的时候都有一个动画效果,这个动画效果过渡看起来很自然,在5.0之后,拥有ShareElement之后做到这个还是比较好做的,然而目前在我们的日常开发中,大多数app兼容都是下限为4.0而不是5.0,所以要实现这个动画效果就需要我们花费一点心思了。

事实上,在朋友圈项目中,我们就实现过这样的一个图片浏览动画,详情点我→《一起撸个朋友圈吧 - 图片浏览(中)【图片浏览器】》

然而在这里的实现按照我目前的看法,是不太完美的,原因有二:

  • 图片浏览视图跟时间线(timeline)处于同一Activity,即便我将它移到一个代理类里面,但还是显得依赖性很大。
  • 基于第一点,不便于其他Activity使用

总的来说,就是一个定制性的类,不太符合我们的“通用性”思想。

于是,再稍微整理和封装之后,我们就有了今天的这个项目。

在说明之前,先声明一下目前仍然有的不足:

  • 对于图片的scaleType支持不好
  • 暂时没有针对多图浏览(ViewPager)做优化

暂且算是几个issue吧,有空再处理一下。

废话说完,那么就正式开始我们的项目吧。


【Step 1】思考

在朋友圈项目中,最难的那一部分——即如何做到图片放大缩小已经是解决了(感谢官方代码-V-),那么现在我们遇到的难题有两个:

  • 如何做到顺利的过渡到新的Activity中
  • 图片缩小的时候如何正确的回归到前一个Activity的小图中

在朋友圈项目里,我们知道做到这种图片的由小到大的过渡实际上是一个障眼法,就是大图一开始不可见并且以小图的大小开始显示,并做放大和位移动画达到一种视觉上看起来像是从小图放大的感觉。

而这两者的实际核心在于得到View的绘制区域,也就是 getGlobalVisibleRect() 方法,在朋友圈项目里,我把大图和时间线放到同一个Activity的原因就是因为即使View不可见,但只要执行到resume后,就可以拿到绘制区域。

但如果放到一个新的Activity里,我们就没法这么做了,因为在 onCreate() 里面,我们并无法拿到View的属性信息,也就拿不到绘制区域了。

然而当初做点击展开控件的时候(链接→)《一起撸个朋友圈吧(step5) - 控件篇【点击展开】》 我讲述过TextView的 onPreDraw() 方法,那时候我留意到TextView实现了OnPreDrawListener,便以为只有某些View实现了这个方法,其他View使用的话是无效的,直到最近查阅了View的资料之后,才发现其实无论是什么View,都会在ViewRootImpl的performTraversals()方法里检测onPreDraw(cancelDraw),而在执行这个方法之前,实际上已经是measure过了,所以这个方法对于任何View都是有效的。

有了这一点,我们就可以解决上面的问题了,在onPreDraw里面得到绘制区域,然后计算比率之后进行动画的展开就可以实现进入Activity的时候开展动画。

回到我们的第一个问题,解决了动画播放之后,我们还要解决的是如何打开窗口的时候不进行动画,关于这一点是在再简单不过了,我们只需要startActivity后执行overridePendingTransition(0, 0);就可以禁用窗口切换动画了。

至此,我们第一个问题解决的思路如下:

  • 目标ImageView实现onPreDrawListener,并在里面获取getGlobalVisibleRect。
  • startActivity禁用动画(特指位移动画,实际上Alpha动画还是可以接受的),使用户的焦点集中在图片中而不集中在Activity过场动画中。

然后第二个问题,在朋友圈项目中也讲解过,在view点击的时候就把view的rect传过来,最后执行退出动画时回归原来的位置即可。


【Step 2】封装

如题,我们的标题名字叫做BaseActivity,因此我们的目的很简单,就是让子类轻松实现这个效果,并且可以更好的拓展,而不要说只能是固定的一个Activity。

因此我们的BaseActivity需要实现以下几个功能:

  • 得到目标View,即最终放大的View
  • 播放进场动画/退场动画
  • 实现核心算法,并保证私有,对内保护
  • 判断是否进行动画

综上所述,我们暂时可以写出如下的代码结构:

public abstract class BaseScaleElementAnimaActivity<V extends ImageView> extends AppCompatActivity {

    @Override protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
		//数据初始化
        initData();
    }

    @Override public void setContentView(@LayoutRes int layoutResID) {
        super.setContentView(layoutResID);
		//针对目标View的初始化
        initImageView();
    }

    private void initData() {

    }

    private void initImageView() {
        
    }

	//进场过渡动画(放大)
    private void playEnterAnima() {
      
    }

	//退场过渡动画(缩小)
    private void playExitAnima() {
      
    }

    @Override public void finish() {
        super.finish();
        overridePendingTransition(0, android.R.anim.fade_out);
    }

	//子类限制
    protected abstract V getAnimaedImageView();
	//子类限制(此处用的Glide)
    protected abstract void onLoadingPicture(SimpleTarget targetImageView, String url);
	//放大/缩小比例计算
    private float[] calculateRatios(Rect startBounds, Rect finalBounds) {
      
    }
	//startActivity方法
    public static void startWithScaleElementActivity(Activity from,
                                                     @Nullable String picUrl,
                                                     @Nullable Rect fromRect,
                                                     Class<? extends BaseScaleElementAnimaActivity> clazz) {
        Intent intent = new Intent(from, clazz);
        intent.putExtra("url", picUrl);
        intent.putExtra("fromRect", fromRect);
        from.startActivity(intent);
        //禁用过渡动画
        from.overridePendingTransition(0, 0);
    }
}

复制代码

对于子类而言,它并不需要知道如何实现放大/缩小动画,它只需要提供最终展示的View和在什么时候载入图片的时机(本项目采用Glide,其他图片框架请自行替换设计)。

所以在父类的onCreate中,我们需要拿到前一个Activity点击的View的绘制区域以及图片url,在setContentView中,我们需要子类提供目标View,其余操作都放在父类执行。

在使用该功能的Activity时必须采用对应的静态方法,毕竟咱们有点特殊是吧。。。

【第一章节完,下一章开始实现动画】

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值