1、新建DemoContract接口
public interface DemoContract {
@PresenterAnno(DemoPresenter.class)
interface IDemoPresenter extends BaseContract.IBasePresenter{
@RequestView(DemoView.class)
void doDemo(String param);
}
interface DemoView extends BaseContract.BaseView{
void onDemoResult(String result);
}
}
注:
DemoContract接口内部有两个内部接口IDemoPresenter和DemoView,
其中IDemoPresenter继承IBasePresenter,DemoView继承BaseView。
IDemoPresenter
在IDemoPresenter上有一个@PresenterAnno注解,该注解传入一个class类型的参数:DemoPresenter.class,表明这个IDemoPresenter接口的实现类是DemoPresenter.class(我们会在后面实现这个类)。在IDemoPresenter里面声明了一个方法doDemo(String param),该方法上面有一个@RequestView注解,该注解传入一个class类型的参数:DemoView.class,表明调用这个doDemo()方法的类需要实现DemoView接口
DemoView
将DemoPresenter处理的结果通过DemoView传递给视图层显示。
2、创建DemoPresenter类,继承BasePresenter并实现IDemoPresenter接口
public class DemoPresenter extends BasePresenter implements DemoContract.IDemoPresenter {
@Override
public void doDemo(String param) {
//do something...
//getView().showToast("toast");
getView(DemoContract.DemoView.class).onDemoResult("result");
}
}
注:
DemoPresenter类继承BasePresenter并实现了IDemoPresenter接口。
在doDemo方法里,可以调用对应View层的方法,其中getView()获取的是BaseView对象,而getView(DemoContract.DemoView.class)获取的是DemoView对象。
3、创建DemoActivity,继承BaseActivity并实现DemoView接口(Fragment同理)
public class DemoActivity extends BaseActivity implements DemoContract.DemoView {
DemoContract.IDemoPresenter demoPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//setContentView(R.layout.activity_demo);
demoPresenter.doDemo("params");
}
@Override
public void onDemoResult(String result) {
//do something
}
}
注:
可以看到我们在DemoActivity声明了一个变量demoPresenter,类型是DemoContract.IDemoPresenter,但我们并没有初始化这个变量。这是因为在之前创建IDemoPresenter接口的时候添加了一个@PresenterAnno(DemoPresenter.class)注解,BaseActivity会在Oncreate方法里找到有这个注解的变量并通过反射自动初始化,有兴趣的可以看看BaseActivity的源码
3、View接口的问题
如果我的Activity并没有实现DemoView接口,但是在DemoPresenter里又通过getView(DemoView.class)方法强行调用,这样不会报错崩溃吗?
这时候我们的lint依赖就起作用了。记得我们在创建DemoPresenter的时候,在doDemo方法上面加了一个@RequestView(DemoView.class)注解吗?这表明要调用doDemo()方法,则必须实现DemoView接口。
如果没有实现DemoView接口,则会在编译时就报错,如下图所示
[图片上传失败...(image-2ed65e-1628753849287)]
4、扩展,一个Presenter对应多个View
传统的MVP模式,一个Presenter只能对应一个View,但项目里已对这个优化了。简单来讲就是在Presenter里面,我们通过调用getView(Class<?>)方法获取到对应的View,然后调用对应的View方法传递数据给对应的视图层(如Activity、Fragment等)。
通过以上的使用可以发现,Presenter的方法里可以通过添加@RequestView注解表明,调用该方法时需要实现的View接口。所以我们就可以实现一个Presenter对应多个View的需求,进而大大减少MVP带来的接口数量。
5、一个Presneter对应多个View的例子
创建DemoContract
public interface DemoContract {
@PresenterAnno(DemoPresenter.class)
interface IDemoPresenter extends BaseContract.IBasePresenter{
@RequestView(DemoView.class)
void doDemo(String param);
@RequestView(OtherDemoView.class)
void doOtherDemo(String param); //这里添加了一个新的方法
}
interface DemoView extends BaseContract.BaseView{
void onDemoResult(String result);
}
interface OtherDemoView extends BaseContract.BaseView{ //这里添加了一个新的View
void onOtherResult(String result);
}
}
创建DemoPresenter
public class DemoPresenter extends BasePresenter implements DemoContract.IDemoPresenter {
@Override
public void doDemo(String param) {
//do something...
//getView().showToast("toast");
getView(DemoContract.DemoView.class).onDemoResult("result");
}
@Override
public void doOtherDemo(String para) {
getView(DemoContract.OtherDemoView.class).onOtherResult("result");
}
}
创建DemoActivity
public class DemoActivity extends BaseActivity implements DemoContract.DemoView, DemoContract.OtherDemoView {
DemoContract.IDemoPresenter demoPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//setContentView(R.layout.activity_demo);
demoPresenter.doDemo("params");
demoPresenter.doOtherDemo("params");
}
@Override
public void onDemoResult(String result) {
}
@Override
public void onOtherResult(String result) {
}
}