使用MVP打造项目框架

前言

在目前的项目框架中大多是用Viewpager+Fragment实现,而通常情况下一个Fragment中包含以下功能,但是如果将这些功能全部集成在一个fragment中会造成,逻辑不清楚,而且我们编写的时候本身也不容易理清顺序,比如在刷新界面的时候要分多种情况,如果是加载第一页且没有缓存数据的时候显示进度动画,否则显示listview自带的下拉刷新动画,当发生错误的时候也要根据有无显示的内容做相应的判断,如果有内容则显示Toast提示用户,否则切换到重试页面。如果将这些逻辑处理与fragment捆绑的话,如果需要更换banner控件,或则下拉刷新控件,则这些逻辑不能复用。所以我尝试使用MVP的方式实现项目框架。

这里写图片描述

功能

先看一下我目前实现的功能有哪些。

1.下拉刷新,上拉加载更多

这里写图片描述

2.懒加载,加载动画,数据缓存

这里写图片描述

3.错误提示(此处有数据,无法显示重试界面,不过如果没有数据的话是有重试界面的)

这里写图片描述

4.下一页重试,(因为已经有了一页数据,所以不能显示重试界面,否则会把用户的数据覆盖,此处添加了一个FooterView用于重试)

这里写图片描述

实现

1.为什么选择MVP

MVP(mode view presenter)与传统的MVC(mode view controller)更注重了view的独立性,在mvc中更注重model的独立性,即model是不变的而有多个view对应于同一个mode,这样就照成了view依赖mode,而view中不可避免的含义逻辑处理的部分,这样将照成view的重用性下降,系统的耦合性提高,而在mvp中mode 与 view 都是独立的。而主要的逻辑都在presenter中,而在presenter中持有的也是model与view的接口,所以这样就使得mode与view的重用性提高,系统的灵活性增加。

我的项目结构如下

这里写图片描述

2.接口抽取

这里面最关键的是对view接口的抽取,以前也看过许多人的mvpdemo但是等到自己写的时候却不知道要怎么办,我总结了一下对view接口抽取主要是三点

  1. 需要从view中获取的数据
  2. 需要让view显示的状态
  3. 需要让view执行的操作

IListView借口

package com.zgh.mvpdemo.view.news;

import android.content.DialogInterface;

import com.zgh.mvpdemo.bean.BannerItem;
import com.zgh.mvpdemo.bean.NewsItem;

import java.util.List;

/**
 * Created by zhuguohui on 2016/6/29.
 */
public interface IListView {
    /*******加载首页相关*******/

    //显示加载首页的加载效果
    void showLoadingFirstPage();

    //隐藏加载首页动画
    void hideLoadingFirstPage();

    //首页加载失败的时候调用
    void showRetryFirstPage();

    //首页为空的时候调用
    void showEmpty();

    //获取到首页数据的时候调用
    void showFirstPageData(List<NewsItem> listData);


    /***********加载下一页相关*************/

    //显示加载下一页
    void showLoadingNextPage();

    //隐藏加载下一页
    void hideLoadingNextPage();

    //获取到下一页数据是调用
    void showNextPageData(List<NewsItem> listData);

    //显示还有下一页
    void showHaveNextPage();

    //显示重试加载下一页
    void showRetryNextPage();

    //没有更多了
    void showNoMore();

    /**************Banner相关*********************/

    //隐藏banner在没有banner数据的时候调用
    void hideBanner();

    //显示banner数据
    void showBanner(List<BannerItem> bannerData);


    /*****************获取数据***************************/

    //判断是否还有下一页
    boolean haveNextPage(int dataSize);

    //是否已经有显示的内容了,用于判断在没有网络的时候是否显示重试界面,如果有内容则显示内容,否则显示重试。
    boolean haveContent();

    //获取首页的url地址
    String getFirstPageUrl();

    //下一页的url地址
    String getNextPageUrl();

    /**************点击事件****************************/

    interface OnBannerClickListener {
        void onBannnerClick(BannerItem item);
    }

    interface OnNewsItemClickListener {
        void onNewsItemClick(NewsItem item);
    }

    void setOnBannerItemClickListener(OnBannerClickListener listener);

    void setOnListItemClickListener(OnNewsItemClickListener listener);

    void toItemDetail(NewsItem item);

    void toBannerDetail(BannerItem item);


    /****************通知*******************************/

    void showToast(String info);

}

IListMode 接口

package com.zgh.mvpdemo.model.news;

/**
 * Created by zhuguohui on 2016/6/29.
 */
public interface IListMode {
    /**
     * 
     * @param useCache 是否使用缓存
     * @param url 数据url
     * @param listener 回调接口
     */
    void LoadData(boolean useCache,String url,DataResultListener listener);
}

对Presenter就没有抽取接口了

package com.zgh.mvpdemo.presenter.news;

import com.zgh.mvpdemo.bean.BannerItem;
import com.zgh.mvpdemo.bean.NewsItem;
import com.zgh.mvpdemo.model.news.DataResultListener;
import com.zgh.mvpdemo.model.news.IListMode;
import com.zgh.mvpdemo.model.news.ListMode;
import com.zgh.mvpdemo.view.news.IListView;

import java.util.List;

/**
 * Created by zhuguohui on 2016/6/29.
 */
public class ListPresenter implements IListView.OnBannerClickListener, IListView.OnNewsItemClickListener {
    IListView listView;
    IListMode listMode;

    public ListPresenter(IListView listView) {
        this.listView = listView;

        listMode = new ListMode();
        //设置点击事件
        listView.setOnBannerItemClickListener(this);
        listView.setOnListItemClickListener(this);
    }

    public void LoadFirstPage(boolean useCache) {
        //在没有内容的时候才显示进度条,在下拉刷新的时候不需要
        if(!listView.haveContent()) {
            listView.showLoadingFirstPage();
        }
        listMode.LoadData(useCache,listView.getFirstPageUrl(), new DataResultListener() {
            @Override
            public void onSuccess(List<NewsItem> newsData, List<BannerItem> bannerData) {
                listView.hideLoadingFirstPage();
                //如果banner数据不为空才显示banner,否则隐藏
                if (bannerData != null && bannerData.size() > 0) {
                    listView.showBanner(bannerData);
                } else {
                    listView.hideBanner();
                }
                //根据list是否有数据设置显示样式
                if (newsData != null && newsData.size() > 0) {
                    listView.showFirstPageData(newsData);
                    //根据item的数量判断是否有下一页
                    if (listView.haveNextPage(newsData.size())) {
                        listView.showHaveNextPage();
                    } else {
                        listView.showNoMore();
                    }
                } else {
                    listView.showEmpty();
                }

            }

            @Override
            public void onError(String error) {
                listView.hideLoadingFirstPage();
                //在没有内容的时候才显示重试,否则只提示,还是显示原来的缓存内容
                if (!listView.haveContent()) {
                    listView.showRetryFirstPage();
                }
                listView.showToast(error);
            }

        });
    }

    public void LoadNextPage() {
        //显示加载下一页
        listView.showLoadingNextPage();
        listMode.LoadData(false,listView.getNextPageUrl(), new DataResultListener() {
            @Override
            public void onSuccess(List<NewsItem> newsData, List<BannerItem> bannerData) {
                //在加载下一页的时候不需要判断banner的数据

                if (newsData != null && newsData.size() > 0) {
                    listView.showNextPageData(newsData);
                    //如果没有下一页,则显示没有更多了
                    if (!listView.haveNextPage(newsData.size())) {
                        listView.showNoMore();
                    }else{
                        listView.showHaveNextPage();
                    }
                } else {
                    listView.showToast("没有更多了");
                    listView.showNoMore();
                }
                listView.hideLoadingFirstPage();
            }

            @Override
            public void onError(String error) {
                listView.hideLoadingNextPage();
                //在没有内容的时候才显示重试,否则只提示,还是显示原来的缓存内容
                if (!listView.haveContent()) {
                    listView.showRetryFirstPage();
                }else{
                    listView.showRetryNextPage();
                }
                listView.showToast(error);
            }

        });
    }

    @Override
    public void onBannnerClick(BannerItem item) {
        listView.toBannerDetail(item);
    }

    @Override
    public void onNewsItemClick(NewsItem item) {
        listView.toItemDetail(item);
    }
}

3.状态切换

状态切换我使用的是张鸿洋的LoadingAndRetryManager 关于具体的用法大家自己去看吧。需要说明的是,LoadingAndRetryManager在ViewPager中的fragment中使用的时候要这么用,要将LoadingAndRetryLayout作为fragment的view返回,而第一个mBaseView是你自己创建的view。

    mLoadingAndRetryManager = new LoadingAndRetryManager(mBaseView, new OnLoadingAndRetryListener() {
            @Override
            public void setRetryEvent(View retryView) {
                retryView.findViewById(R.id.id_btn_retry).setOnClickListener(new View.OnClickListener() {
                    @Override
                    public void onClick(View v) {
                        mPresenter.LoadFirstPage(true);
                    }
                });
            }
        });
        mBaseView = mLoadingAndRetryManager.mLoadingAndRetryLayout;

4.Fragment与ViewPager使用时的注意事项

1.懒加载

由于viewpager会预先缓存几页Fragment,所以Fragment的生命周期在ViewPager中其实是没有多少意义的,因此为了实现用户滑动到这个界面才显示这个界面的数据的功能,我们必须在setUserVisibleHint中加载数据,就像这样

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        if (isVisibleToUser && !haveInint) {
            haveInint = true;
            mPresenter.LoadFirstPage(true);
        }
    }

View的缓存

在viewpager的滑动过程中,某个fragment的onCreateView()方法和onDestroyView()方法会多次调用,也就是会先销毁view而保留Fragment中的成员变量,等内存不足是再销毁Fragment。由于View的创建也是很耗时的操作,所以我选择缓存view,而且为了在数据请求返回的时候view都存在,防止每次都有判断view是否为空,因此我将view的创建放在了onCreate()方法中,且先创建view再请求数据。

   @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        createView();
        bindData();
        setListener();
        mPresenter = new ListPresenter(this);
        mLoadingAndRetryManager.showLoading();
    }


    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        ViewParent parent = mBaseView.getParent();
        if (parent != null && parent instanceof ViewGroup) {
            ViewGroup group = (ViewGroup) parent;
            group.removeView(mBaseView);
        }
        return mBaseView;
    }

5.ListView的FooterView与HeardView的显示与隐藏

我觉得这个技巧还是蛮有用的,于是专门写一下。在很多时候我们将view添加到ListView中作为FootView或者HeardView,想隐藏的时候就调用view.setVisibility(View.GONE),然而结果是view的确不显示了,但是它所占用的空间还在,与调用view.setVisibility(View.INVISIBLE)效果类似,后来我发现在添加footview或headview之前用一层layout进行包裹就可以实现隐藏footerview的效果了,代码如下:

//添加footerview
        tv_bottom_info = (TextView) View.inflate(getActivity(), R.layout.view_bottom_retry, null);
        tv_bottom_info.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, DensityUtil.dip2px(getActivity(), 45)));
        //要想footerview实现隐藏效果,必须在其外部包裹一层layout,heardview 同理
        LinearLayout footerviewParent = new LinearLayout(getActivity());
        footerviewParent.addView(tv_bottom_info);
        tv_bottom_info.setVisibility(View.GONE);
        mListView.addFooterView(footerviewParent);

Demo

更多细节请看我的demo,欢迎star哈。

MVPDemo

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

推荐阅读更多精彩内容