Fragment之我的解决方案:Fragmentation

Fragment系列文章:
1、Fragment全解析系列(一):那些年踩过的坑
2、Fragment全解析系列(二):正确的使用姿势
3、Fragment之我的解决方案:Fragmentation

附:SwipeBackFragment的实现分析

如果你通读了本系列的前两篇,我相信你可以写出大部分场景都能正常运行的Fragment了。如果你想了解更多,那么你可以看看我封装的这个库:Fragmentation。
本篇主要介绍这个库,解决了一些BUG,使用简单,提供实时查看栈视图等实用功能。


源码地址:Github,欢迎Fork,提Issues 。

Demo网盘下载
Demo演示:单Activity+多Fragment


demo.gif

Fragmentation

为"单Activity + 多Fragment的架构","多模块Activity + 多Fragment的架构"而生,帮你简化使用过程,修复了官方Fragment库存在的一些BUG。

特性

1、为重度使用Fragment而生,提供了方便的管理Fragment的方法
2、有效解决Fragment重叠问题
3、完美的防抖动解决方案(防止用户点击速度过快,导致启动多个Fragment)
4、实时查看Fragment的(包括嵌套Fragment)栈视图的对话框和Log,方便调试
5、增加启动模式、startForResult等类似Activity方法
6、修复官方库里pop(tag/id)出栈多个Fragment时的一些BUG
7、完美解决进出栈动画的一些BUG,更自由的管理Fragment的动画
8、支持SwipeBack滑动边缘退出(需要使用Fragmentation_SwipeBack库,详情README)


通过logFragmentStackHierarchy(TAG)查看Log

如何使用

1、项目下app的build.gradle中依赖:

compile 'me.yokeyword:fragmentation:0.6.1'
// appcompat v7包也是必须的
// 如果想使用SwipeBack 滑动边缘退出Fragment/Activity功能,请再添加下面的库
// compile 'me.yokeyword:fragmentation-swipeback:0.3.0'


2、Activity继承SupportActivity:

public class MainActivity extends SupportActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(...);
        if (savedInstanceState == null) {
            start(HomeFragment.newInstance());
        }
    }

    /**
    *  设置Container的id,必须实现
    */
    @Override
    public int setContainerId() {
        return R.id.fl_container;
    }

    /**
     *  设置全局动画,在SupportFragment可以自由更改其动画
     */
    @Override
    protected FragmentAnimator onCreateFragmentAnimator() {
        // 默认竖向(和安卓5.0以上的动画相同)
        return super.onCreateFragmentAnimator();
        // 设置横向(和安卓4.x动画相同)
        // return new DefaultHorizontalAnimator();
        // 设置自定义动画
        // return new FragmentAnimator(enter,exit,popEnter,popExit);
    }

        // 这里注意
        // new FragmentAnimator(enter,exit,popEnter,popExit)的4个参数与官方setCustomAnimations(enter,exit,popEnter,popExit)
        // 并不一致,对于官方来说Fragment出栈时的动画对应的是popExit(即第4个参数)
        // 对于FragmentAnimator来说Fragment出栈时的动画对应的是exit参数(即第2个参数)
        // 这里和官方不一致的原因主要出于我们绝大多数人认为enter,exit才是对应进出栈动画的角度考虑的。


3、Fragment继承SupportFragment

重大更新日志

0.6.X:

1、完美的防抖动解决方案:使用Activity的dispatchTouchEvent方案替换了原来的通过时间判断方案(为0.7版本,管理Fragment嵌套等复杂场景的更新作铺垫)
2、对执行popTo()、start(F,SingleTask)等方法时的动画作了进一步优化兼容
3、进行了一些方法的重构
4、进行了debug相关类的包迁移

API

SupportActivity
打开 栈视图 的提示框,在复杂嵌套的时候,可以通过这个来清洗的理清不同阶级的栈视图。

// 弹出 栈视图 提示框
showFragmentStackHierarchyView();
// 打印 栈视图 Log
logFragmentStackHierarchy(TAG);


除此之外包含大部分SupportFragment的方法,请自行查看。

SupportFragment
1、启动相关:

// 启动新的Fragment
start(SupportFragment fragment)
// 以某种启动模式,启动新的Fragment
start(SupportFragment fragment, int launchMode)
// 启动新的Fragment,并能接收到新Fragment的数据返回
startForResult(SupportFragment fragment,int requestCode)
// 启动目标Fragment,并关闭当前Fragment
startWithPop(SupportFragment fragment)


2、关闭Fragment:

// 关闭当前Fragment(在当前Fragment所在栈内pop)
pop();

// 关闭某一个Fragment栈内之上的Fragments
popTo(Class fragmentClass, boolean includeSelf);

// 如果想出栈后,紧接着开始.beginTransaction()开始一个事务,请使用下面的方法:
// 在 第二篇 文章内的 Fragment事务部分的问题 有解释原因
popTo(Class fragmentClass, boolean includeSelf, Runnable afterTransaction)


3、在SupportFragment内,支持监听返回按钮事件:

@Override
public boolean onBackPressedSupport() {
   // 返回false,则继续传递返回事件; 返回true则不继续传递
    return false;
}


4、 定义当前Fragment的动画,复写onCreateFragmentAnimation方法:

@Override
protected FragmentAnimator onCreateFragmentAnimation() {
    // 获取在SupportActivity里设置的全局动画对象 或者通过 _mActivity.getFragmentAnimator()获取
    FragmentAnimator fragmentAnimator = super.onCreateFragmentAnimation();
    fragmentAnimator.setEnter(0);
    fragmentAnimator.setExit(0);
    return fragmentAnimator;

    // 也可以直接通过
    // return new FragmentAnimator(enter,exit,popEnter,popExit)设置一个全新的动画
}


5、获取Fragment

// 获取所在栈内的栈顶(子)Fragment:
getTopFragment();
getTopChildFragment();

// 获取所在栈内的某个(子)Fragment对象:
findFragment(Class fragmentClass);
findChildFragment(Class fragmentClass);

// 获取所在栈内的前一个Fragment对象:
getPreFragment();


其他
隐藏/显示 输入法:(因为Fragment被销毁时,不会自动隐藏软键盘,以及弹出软键盘有些麻烦,故提供下面2个方法)

// 隐藏软键盘 一般用在onHiden里
hideSoftInput();
// 显示软键盘,调用该方法后,会在onPause时自动隐藏软键盘
showSoftInput(View view);

下面是DetailFragment startForResult ModifyTitleFragment的代码:

DetailFragment.class里:

startForResult(ModifyDetailFragment.newInstance(mTitle), REQ_CODE);

@Override
public void onFragmentResult(int requestCode, int resultCode, Bundle data) {
    super.onFragmentResult(requestCode, resultCode, data);
    if (requestCode == REQ_CODE && resultCode == RESULT_OK ) {
    // 在此通过Bundle data 获取返回的数据
    }
}

ModifyTitleFragment.class里:

Bundle bundle = new Bundle();
bundle.putString("title", "xxxx");
setFramgentResult(RESULT_OK, bundle);


下面是以SingleTask模式重新启动一个已存在的Fragment的标准代码:
比如:HomeFragment->BFragment->CFragment CFragment以SingleTask模式重新启动HomeFragment

HomeFragment fragment = findFragment(HomeFragment.class);
Bundle newBundle = new Bundle();
// 传递的bundle数据,会调用目标Fragment的onNewBundle(Bundle newBundle)方法
fragment.putNewBundle(newBundle);
// 栈内的homeFragment以SingleTask模式启动
start(fragment, SupportFragment.SINGLETASK);

// 在HomeFragment.class中:
@Override
protected void onNewBundle(Bundle newBundle){
    // 在此可以接收到数据  类似Activity中的onNewIntent()
}


关于Fragmentation帮你恢复Fragment,你需要知道的

2个概念:

"同级"式:比如QQ的主界面,“消息”、“联系人”、“动态”,这3个Fragment属于同级关系
“流程”式:比如登录->注册/忘记密码->填写信息->跳转到主页Activity

对于Activity内的“流程”式Fragments(比如登录->注册/忘记密码->填写信息->跳转到主页Activity),Fragmentation帮助你处理了栈内的恢复,保证Fragment不会重叠,你不需要再自己处理了。

但是如果你的Activity内的Fragments是“同级”的,那么你需要自己去恢复处理,因为库并不能知道你想恢复哪个Fragment。

而如果你有Fragment嵌套,那么不管是“同级”式还是“流程”式,你都需要自己去恢复处理。

附:已经找到了恢复各种嵌套等环境的恢复解决方案,在进一步测试中...



文/YoKey(简书作者)
原文链接:http://www.jianshu.com/p/38f7994faa6b
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
A powerful library that manage Fragment for Android!为"单Activity + 多Fragment","多模块Activity 多Fragment"架构而生,简化开发,轻松解决动画、嵌套、事务相关等问题。为了更好的使用和了解该库,推荐阅读下面的文章:Fragment全解析系列(一):那些年踩过的坑Fragment全解析系列(二):正确的使用姿势Demo演示:均为单Activity 多Fragment,第一个为简单流式demo,第二个为仿微信交互的demo(全页面支持滑动退出),第三个为仿知乎交互的复杂嵌套demo下载APK   特性1、可以快速开发出各种嵌套设计的Fragment App2、悬浮球/摇一摇实时查看Fragment的栈视图Dialog,降低开发难度3、增加启动模式、startForResult等类似Activity方法4、类似Android事件分发机制的Fragment回退方法:onBackPressedSupport(),轻松为每个Fragment实现Back按键事件5、提供onSupportVisible()等生命周期方法,简化嵌套Fragment的开发过程; 提供统一的onLazyInitView()懒加载方法6、提供 Fragment转场动画 系列解决方案,动态更换动画7、提供Activity作用域的EventBus辅助类,Fragment通信更简单、独立(需要使用EventBusActivityScope库)8、支持SwipeBack滑动边缘退出(需要使用Fragmentation_SwipeBack库)      如何使用1. 项目下app的build.gradle中依赖:// appcompat-v7包是必须的,v1.1.9兼容v4-27.0.0 compile 'me.yokeyword:fragmentation:1.1.9' // 如果不想继承SupportActivity/Fragment,自己定制Support,可仅依赖: // compile 'me.yokeyword:fragmentation-core:1.1.9' // 如果想使用SwipeBack 滑动边缘退出Fragment/Activity功能,完整的添加规则如下: compile 'me.yokeyword:fragmentation:1.1.9' // swipeback基于fragmentation, 如果是自定制SupportActivity/Fragment,则参照SwipeBackActivity/Fragment实现即可 compile 'me.yokeyword:fragmentation-swipeback:1.1.9' // Activity作用域的EventBus,更安全,可有效避免after onSavenInstanceState()异常 compile 'me.yokeyword:eventbus-activity-scope:1.1.0' // Your EventBus's version compile 'org.greenrobot:eventbus:{version}'2. Activity继承SupportActivity:// v1.0.0开始,不强制继承SupportActivity,可使用接口+委托形式来实现自己的SupportActivity public class MainActivity extends SupportActivity {     @Override     protected void onCreate(Bundle savedInstanceState) {         super.onCreate(savedInstanceState);         setContentView(...);         // 建议在Application里初始化         Fragmentation.builder()              // 显示悬浮球 ; 其他Mode:SHAKE: 摇一摇唤出   NONE:隐藏              .stackViewMode(Fragmentation.BUBBLE)              .debug(BuildConfig.DEBUG)              ... // 更多查看wiki或demo              .install();         if (findFragment(HomeFragment.class) == null)

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值