当前位置: 首页 > news >正文

Android fragment 转场动画

fragment 动画

  • 使用 setCustomAnimations 绘制转场动画
  • 使用共享元素的动画
    • recyclerview in
  • 参考地址

使用 setCustomAnimations 绘制转场动画

概述

在 Android 中,可以使用 setCustomAnimations() 方法来绘制自定义的 Fragment 转场动画。该方法接受四个参数,分别对应四种类型的动画:

  • enter: 新 Fragment 进入时的动画
  • exit: 旧 Fragment 离开时的动画
  • popEnter: 从 Back Stack 中弹出 Fragment 时的动画
  • popExit: 将 Fragment 添加到 Back Stack 时的动画

步骤

  1. 创建动画资源文件。

    转场动画可以使用 XML 动画资源文件来定义。每个动画资源文件定义一种类型的动画,例如 enter.xmlexit.xmlpop_enter.xmlpop_exit.xml

    动画资源文件的格式与其他动画资源文件相同,可以使用各种动画属性来定义动画效果。例如,可以使用 alpha 属性来定义透明度变化,使用 translate 属性来定义位移变化,使用 scale 属性来定义缩放变化等等。

  2. 在 Fragment 中设置自定义动画。

    在 Fragment 中,可以使用 setCustomAnimations() 方法来设置自定义的转场动画。该方法接受四个参数,分别对应四个类型的动画资源文件的 ID。

    例如,以下代码设置了 Fragment 进入和离开时的自定义动画:

fragment.setCustomAnimations(R.anim.enter, R.anim.exit);

示例

以下是一个示例,演示如何使用 setCustomAnimations() 方法来绘制自定义的 Fragment 转场动画。

1. 创建动画资源文件

<set xmlns:android="http://schemas.android.com/apk/res/android"><alphaandroid:fromAlpha="0.0"android:toAlpha="1.0"android:duration="500" />
</set><set xmlns:android="http://schemas.android.com/apk/res/android"><alphaandroid:fromAlpha="1.0"android:toAlpha="0.0"android:duration="500" />
</set>

2. 在 Fragment 中设置自定义动画

public class MyFragment extends Fragment {@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {// Inflate the layout for this fragmentView view = inflater.inflate(R.layout.fragment_my, container, false);// Set custom animationssetCustomAnimations(R.anim.enter, R.anim.exit);return view;}
}

效果

运行该示例后,可以看到 Fragment 进入和离开时都会播放自定义的动画效果。

注意事项

  • 使用 setCustomAnimations() 方法设置自定义动画时,必须在 Fragment 添加到 Activity 之前调用该方法。
  • 如果不设置自定义动画,则 Fragment 会使用系统默认的转场动画。

进阶

除了使用 XML 动画资源文件来定义动画效果之外,还可以使用代码来定义动画效果。例如,可以使用 ObjectAnimator 类来创建更加复杂的动画效果。

使用共享元素的动画

FirstFragment 中:

public class FirstFragment extends Fragment {@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {return inflater.inflate(R.layout.fragment_first, container, false);}@Overridepublic void onViewCreated(View view, Bundle savedInstanceState) {super.onViewCreated(view, savedInstanceState);ImageView sharedImageView = view.findViewById(R.id.shared_image);sharedImageView.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {navigateToSecondFragment(sharedImageView);}});}private void navigateToSecondFragment(View sharedElement) {SecondFragment newFragment = new SecondFragment();FragmentTransaction transaction = getParentFragmentManager().beginTransaction();// 设置退出动画TransitionSet exitTransition = new TransitionSet();exitTransition.addTransition(new Fade());exitTransition.addTransition(new Slide(Gravity.START));newFragment.setExitTransition(exitTransition);// 设置共享元素退出动画TransitionSet sharedElementExitTransition = new TransitionSet();sharedElementExitTransition.addTransition(new ChangeBounds());sharedElementExitTransition.addTransition(new ChangeTransform());sharedElementExitTransition.addTransition(new ChangeImageTransform());newFragment.setSharedElementExitTransition(sharedElementExitTransition);// 设置进入动画TransitionSet enterTransition = new TransitionSet();enterTransition.addTransition(new Fade());enterTransition.addTransition(new Slide(Gravity.END));newFragment.setEnterTransition(enterTransition);// 设置共享元素进入动画TransitionSet sharedElementEnterTransition = new TransitionSet();sharedElementEnterTransition.addTransition(new ChangeBounds());sharedElementEnterTransition.addTransition(new ChangeTransform());sharedElementEnterTransition.addTransition(new ChangeImageTransform());newFragment.setSharedElementEnterTransition(sharedElementEnterTransition);// 添加共享元素 可以添加多个共享元素transaction.addSharedElement(sharedElement, sharedElement.getTransitionName());// 替换当前 Fragmenttransaction.replace(R.id.fragment_container_view, newFragment);transaction.addToBackStack(tag);transaction.commit();}
}

SecondFragment 中:

public class SecondFragment extends Fragment {@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);}@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {return inflater.inflate(R.layout.fragment_second, container, false);}
}

recyclerview in

对于在 RecyclerView 或 ListView 中使用共享元素的情况,你可以在适配器的 onBindViewHolder() 方法中为每个共享元素设置 transitionName 属性。

假设你有一个 RecyclerView,其中的每个项目包含一个 ImageView,并且你想要在这些 ImageView 之间应用共享元素动画。在 RecyclerView 的适配器中,你可以这样设置 transitionName

public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> {private List<MyData> dataList;public MyAdapter(List<MyData> dataList) {this.dataList = dataList;}@NonNull@Overridepublic ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_layout, parent, false);return new ViewHolder(view);}@Overridepublic void onBindViewHolder(@NonNull ViewHolder holder, int position) {MyData data = dataList.get(position);// 设置 transitionNameholder.imageView.setTransitionName("shared_image_" + position);// 加载图片或其他数据// Glide.with(holder.imageView.getContext()).load(data.getImageUrl()).into(holder.imageView);}@Overridepublic int getItemCount() {return dataList.size();}public static class ViewHolder extends RecyclerView.ViewHolder {ImageView imageView;public ViewHolder(@NonNull View itemView) {super(itemView);imageView = itemView.findViewById(R.id.imageView);}}
}

在这个示例中,我们在适配器的 onBindViewHolder() 方法中为每个 ImageView 设置了唯一的 transitionName 属性,例如 "shared_image_" + position。这样做可以确保 RecyclerView 中的每个共享元素都具有唯一的 transitionName,以便在共享元素动画中正确识别和匹配它们。

参考地址

chatgpt

相关文章:

  • android实现真随机数生成器(物理随机)
  • 设计模式学习笔记 - 面向对象 - 5.接口和抽象类的区别
  • 倒计时41天
  • chatgpt:还有哪些人工智能和科技值得关注?
  • AI浅谈:计算机视觉(CV)技术的优势和挑战
  • 策略模式:封装行为策略,灵活切换实现多态业务逻辑
  • uniapp的扩展组件uni-popup 弹出层自动打开
  • python学习笔记-内置异常
  • CMS垃圾回收器
  • Kotlin:协程基础
  • Leetcoder Day25| 回溯part05:子集+排列
  • 大概了解一下G1收集器
  • Redis 学习笔记 3:黑马点评
  • 服务器部署java 的docker项目,以及常用的一些命令
  • 可视化图文报表
  • [rust! #004] [译] Rust 的内置 Traits, 使用场景, 方式, 和原因
  • [笔记] php常见简单功能及函数
  • Docker入门(二) - Dockerfile
  • eclipse的离线汉化
  • ERLANG 网工修炼笔记 ---- UDP
  • extjs4学习之配置
  • Java教程_软件开发基础
  • JDK9: 集成 Jshell 和 Maven 项目.
  • JS创建对象模式及其对象原型链探究(一):Object模式
  • MySQL数据库运维之数据恢复
  • 基于 Ueditor 的现代化编辑器 Neditor 1.5.4 发布
  • 简单数学运算程序(不定期更新)
  • 山寨一个 Promise
  • 我的业余项目总结
  • 协程
  • 携程小程序初体验
  • 一个项目push到多个远程Git仓库
  • 用mpvue开发微信小程序
  • 自制字幕遮挡器
  • ​MySQL主从复制一致性检测
  • ​secrets --- 生成管理密码的安全随机数​
  • #define 用法
  • (13)Hive调优——动态分区导致的小文件问题
  • (arch)linux 转换文件编码格式
  • (ibm)Java 语言的 XPath API
  • (pojstep1.1.2)2654(直叙式模拟)
  • (仿QQ聊天消息列表加载)wp7 listbox 列表项逐一加载的一种实现方式,以及加入渐显动画...
  • (转)http协议
  • (最全解法)输入一个整数,输出该数二进制表示中1的个数。
  • ..回顾17,展望18
  • .[backups@airmail.cc].faust勒索病毒的最新威胁:如何恢复您的数据?
  • .form文件_SSM框架文件上传篇
  • .NET Core SkiaSharp 替代 System.Drawing.Common 的一些用法
  • .NET/C# 检测电脑上安装的 .NET Framework 的版本
  • .net6使用Sejil可视化日志
  • .NET的微型Web框架 Nancy
  • .net下简单快捷的数值高低位切换
  • ::前边啥也没有
  • @DataRedisTest测试redis从未如此丝滑
  • [AMQP Connection 127.0.0.1:5672] An unexpected connection driver error occured