一个超高自定义度又简单使用的页面状态管理库

StatusLayout : 一个超高自定义度又简单的页面状态管理库

业务场景需求:

在日常开发App的过程中,我们少不了对Activity/Fragment 等做一些不同状态不同UI的状态管理逻辑,比如空页面 错误重试页面 等等,网上也有很多作者写了开源库来处理这些问题 但是我看了一下这些库,个人认为有以下几个小问题

  • 大部分都是只定义了错误 空数据 loading等3-5个左右的状态,并且切换的时候也是已经规定好调用哪些方法,这样就会大大限制了拓展性以及对一些复杂的业务场景的适配
  • 对于一些点击事件或者view的处理也不是很到位,比如错误重试等大多数都是传入一个id
  • 多个地方需要同样的设置的时候,需要不断copy代码过去
以上几个小问题相信有些开发者也有发现,在用的时候也会觉得还有改进的空间
我在公司的项目中也发现了这些问题,所以在空闲时间写了一个管理库用来管理页面,接下来就给大家介绍一下,相信能给大家在日常开发中带来更多便利,更少的代码,更多的可操作性

StatusLayout有以下几个优点

  • 自由定制需要的状态以及对应布局,只需要一行代码
  • 可以定制动画效果
  • 可以用在旧项目上,不需要修改原有xml文件
  • 可设置全局属性避免重复劳动

Github地址: https://github.com/Colaman0/StatusLayout


效果图:

image

效果图来看比起普通的库多了一个淡入淡出的动画效果,这部分可以自定义,这里只给大家展现一个最基本的效果

依赖 :

    allprojects {
        repositories {
            ...
            maven { url 'https://jitpack.io' }
        }
    }
    dependencies {
            implementation 'com.github.Colaman0:StatusLayout:1.0.8'
    }

具体使用步骤如下:

1.第一步先把StatusLayout作为根布局,这里有以下两种写法

  • StatusLayout作为根布局在activity/fragment/view 中使用

    在xml内直接把StatusLayout作为根布局,注意StatusLayout内部子view数量不能超过1个,
    所以如果UI上需求需要排列多个View的时候,需要多套一层布局,比如:

    <StatusLayout>
        <LinearLayout>    
            <View/>        
            <View/>
            <View/>
        </LinearLayout>
    </StatusLayout>
    
    
  • 通过 StatusLayout.init()方法传入Context以及你要显示到的layout资源文件,这个方法会返回一个StatusLayout对象,所以大家可以在封装BaseActivity的时候这样写:
    // 后续可以通过mStatusLayout添加不同状态对应的UI  
    StatusLayout mStatusLayout = StatusLayout.init(this, R.layout.activity_main);   
    setContentView(mStatusLayout);
    

2. 添加不同状态对应的UI以及响应点击事件

通过add(statusconfig : StatusConfig)方法来添加一种状态布局,传入StatusConfig参数类,给大家讲一下每个参数的作用
  • status : 作为一个状态布局的status标记,比如空页面 错误页面 等等你想要添加进去的页面,设置一下自己想要添加的status

  • layoutRes : 对应上面的status, 一个status对应一个view,一个布局,比如上面status传入了一个empty,那我们这里对应可以添加一个空页面的layout资源文件id

  • view : 跟layoutRes相似,考虑到有时候业务需求,某个状态下的页面可能按钮或者一些需要写的逻辑比较多比较复杂, 这个时候可以让开发者自己写一个view传进来,对应的一些逻辑判断则让view内部去处理 ,StatusLayout只负责切换

  • clickRes :每一个布局,可以传递一个id进来,比如错误重试页面 可以传一个button的id进来,这样在button被点击的时候,可以通过回调来接收到点击事件

      data class StatusConfig(        
              var status: String?,        
              @field:LayoutRes        
              var layoutRes: Int = -1,        
              var view: View? = null,       
              @field:IdRes       
              var clickRes: Int = -1)
    
    
    

3. 切换布局

通过switchLayout()/showDefaultContent()两种方法来切换布局
  • switchLayout(status : String)方法是用于切换你add进去的布局,只要传入你前面add布局的时候传入的status就可以了
  • showDefaultContent()用于切换回你默认的UI,比如在切到error状态的UI时,你点击了重试按钮请求成功之后,通过showDefaultContent()方法切换正常的布局,可能是你在xml里默认的一个布局,也可以是通过add方法添加进去的布局,通过add添加进去的布局需要statusSTATUS_NORMAL才会被StatusLayout认为是默认的布局。总体来说切换回正常状态布局调用这个方法就可以了,具体可以参考下面关于回调的代码

4. 不同布局点击的回调

上面在add方法中讲到了StatusConfig中一个clickRes变量,相当于告诉StatusLayout我要监听这个id的view的点击事件,当它被点击的时候告诉我,可以通过setLayoutClickListener()方法来设置监听

            setLayoutClickListener(new StatusLayout.OnLayoutClickListener() {
                @Override
                public void OnLayoutClick(View view, String status) {
                     // View: 对应status的rootView  
                     // status:当前status,可以判断当前页面处于哪个status
                    switch (status) {
                        case LOADING:
                            Toast.makeText(MainActivity.this, LOADING, Toast.LENGTH_SHORT).show();
                            break;
                        case EMPTY:
                            Toast.makeText(MainActivity.this, EMPTY, Toast.LENGTH_SHORT).show();
                            break;
                        case ERROR:  
                             // 这里通过showDefaultContent()方法展示默认的布局
                            mStatusLayout.showDefaultContent();
                            break;
                    }
                }
            });
            

讲解一下上面的代码,设置一个layout点击的回调监听,当layout/clickRes 对应的view被点击的时候,会回调当前是哪一个status的页面,以及对应的布局view,当我们的clickRes不传的时候,默认是整个页面响应点击事件,所以在add的时候比较灵活的处理了关于点击事件的处理,比较复杂的页面建议就在add的时候传入一个view然后在内部做处理比较合适了,避免拓展出一大堆方法。

5. 设置显示/隐藏的动画

通过setAnimation() 来设置页面显示/隐藏的的动画, 也可以通过setGlobalAnim()来设置一个全局的动画效果,setAnimation()的优先级比setGlobalAnim()更高

6.设置全局属性

考虑到APP里常见的空页面 loading 之类的页面都是比较统一的,这个时候可以通过StatusLayout.setGlobalData()方法来设置全局的属性,这个时候可以设置全局属性来避免重复添加的代码,后续可以通过add()方法来覆盖全局属性。
setGlobalData方法传入的参数和通过add()方法传入的参数值是一样的,可以参考一下代码,并且这里考虑到有些地方没有机会用到这些布局或者说不需要这些布局,所以StatusLayout只有在切换布局的时候才会去加载这些全局属性布局。

StatusLayout.setGlobalData(        
    StatusConfig(status = StatusLayout.STATUS_EMPTY, layoutRes = R.layout.include_empty),        
    StatusConfig(status = StatusLayout.STATUS_ERROR, layoutRes = R.layout.include_error, clickRes = R.id.btn_retry))

总结

以上六点就讲解完了StatusLayout的一个使用,在我一开始写的时候,是想着能尽量适应多种场景以及尽可能少的代码逻辑

  • 比如电商APP中一个订单支付成功失败loading等场景,可以通过StatusLayout去对应添加布局,并且对于一些通用的状态,可以设置全局属性避免不同地方出现一样的代码,并且以后需要更换布局的时候只要在设置全局属性的地方修改一下layout就可以了。
  • 点击回调通过setLayoutClickListener 去处理不同状态下的点击事件回调,不需要写不同的回调事件,也可以更好的对于多种多样的布局来做一个适应

整个库的核心想法就是通过status来管理页面,StatusLayout只负责管理你添加进来的布局,以及对应切换某个status的布局。并不会限制得很死要调用某个方法,所以你可以尽情得自定义你的页面,添加各种各样的布局进去,然后通过switchLayout()来切换布局就可以了。


Github地址: https://github.com/Colaman0/StatusLayout

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351