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

news/2024/7/4 8:23:48

本项目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时必须采用对应的静态方法,毕竟咱们有点特殊是吧。。。

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


http://www.niftyadmin.cn/n/595141.html

相关文章

认识SOAR-安全事件编排自动化响应

SOAR是最近几年安全市场上最火热的词汇之一。SOAR究竟是什么&#xff0c;发展历程是什么&#xff0c;能够起什么作用&#xff0c;带着这些问题我们来认识一下SOAR。 一、SOAR是什么 SOAR 一词来自分析机构 Gartner&#xff0c;SOAR-Security Orchestration, Automation and R…

matlab 创建批量文件夹_PS批量处理图片技巧!

当我们在工作中&#xff0c;如果有上百张图片&#xff0c;一张一张处理估计一天时间也不够&#xff0c;那么怎么批量处理图片呢&#xff0c;比如批量压图、批量加个人水印等问题。于是写下我批量处理照片的方法与大家分享。现在&#xff0c;让我们搞起来&#xff01;一&#xf…

在power query中连接mysql_Power Query 建立数据连接的6种常用方式

Power Query有5大类40种的连接方式&#xff0c;可以连接文件、数据库、在线服务等等各种数据源。今天我们介绍6种常用的连接方式&#xff1a;自文件中的3种&#xff1a;从工作簿从文本/CSV从文件夹从其他源中的3种&#xff1a;自表格区域自网络空白查询从工作簿建立连接&#x…

树莓派python开发_树莓派 Raspberry Pi 编写Python的Hello world!

树莓派 Raspberry Pi 编写Python的Hello world&#xff01;版权声明&#xff1a;本文为博主原创文章&#xff0c;未经博主允许不得转载。2019年3月16日发布本文介绍在树莓派中如何编写python程序。树莓派通电开机&#xff0c;使用SSH登陆。查看Python的版本我们输入命令&#x…

mysql复选框_前端框架(二)DIV多选复选框框的封装和MySql数据库存取

图能够包括的寓意和含义是文字不能比拟的&#xff0c;先有一个效果图你也就知道这篇文章的主要内容是关于什么问题的。省去了一大堆文字的累述。看以下这张图&#xff1a;这个需求就是要实现某个人具有第二种特性。具有多对多关系。比方某个人既喜欢运动、有喜欢上网等等。这样…

[阅读笔记]fsnotify源码阅读

fsnotify的github地址是https://github.com/howeyc/fsnotifyfsnotify是一个文件夹监控应用。可以使用创建一个watcher来对某个文件夹进行监控文件目录很简单&#xff0c;实际就两个程序文件&#xff0c;fsnotify.go 和 各平台的fsnotify_XXX.go后一个文件是各个不同平台的实现e…

CentOS7下python3+Flask+uWSGI+Nginx+Supervisor环境搭建

在生产环境中通常用uwsgi作为Flask的web服务网关&#xff0c;通过nginx反向代理进行负载均衡&#xff0c;通过supervior进行服务进行的管理。这一套搭下来还是有一些坑要踩&#xff0c;本文通过一个简单的Flask web应用记录了CentOS7下python3FlaskuWSGINginxSupervisor环境搭建…

JS框架刷新,iframe父子窗口间js方法调用

Response.Write("<script languagejavascript>window.open(x.aspx,main);</script>"); Javascript刷新页面的几种方法&#xff1a;1 history.go(0) 2 location.reload() 3 locationlocation 4 location.assign(location) 5 document.ex…