android studio mvp 代码模板,Android Studio自定义模板实现一键创建MVP结构

前言

之前有写过关于如何使用 DataBinding 的两篇文章,不仅仅是为了消灭掉一部分重复代码,更是为了提高开发效率。详情可以点击下方的传送门

DataBinding入门进阶指南(一)

DataBinding入门进阶指南(二)

而这篇文章主要介绍的就是如何通过 Android Studio 提供的模版功能去自定义模版结构,从而实现类似于一键创建整个MVP代码的功能。可以说在提高效率的道路上,又向前走了一大步

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

下面可以来看一看具体效果:

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

介绍

在 Android Studio 中,创建一个 Activity 可以直接通过 File -> New -> Activity 来进行选择创建

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

通过这种方式创建的 Activity 会自动在 AndroidManifest.xml 中完成注册,创建其他组件也可以通过这种方式。

不过,如果你正在使用某种开发模式,譬如 MVP、MVVM 等,你每创建一个 Activity 就意味着需要同时创建一系列其他相关的类。

为了避免这种毫无意义的重复性劳动,我们可以编写模板代码去实现一键创建重复代码。

开始

下面我们就来开始模版的编写吧。

首先,找到你的 Android Studio 的安装目录,然后根据这个目录找到 ...\templates 目录:

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

然后进入 activityes 目录,我们将要编写的各种模版就在这个目录内:

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

要说如何去编写模版代码,一开始我也是一无所知的,不过好在 Android Studio 已经为我们提供了这些例子,我们直接参考例子去写。

就拿最简单的 Empty Activity 来开始吧

进入到 EmptyActivity 目录

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

globals.xml.ftl

打开 globals.xml.ftl 文件,下面是它的内容:

根据文件名来看, globals.xml.ftl 的作用是用来控制一些全局变量,比如是否显示 ActionBar 等,暂且先不用管它

recipe.xml.ftl

recipe.xml.ftl 文件内容如下:

#if>

to="${escapeXmlAttribute(srcOut)}/${activityClass}.${ktOrJavaExt}" />

第一段

就是用于导入Kotlin的相关命令,同时它的别名为 kt

主要还是注意 instantiate 代码块中的相关信息, 其中 ${ktOrJavaExt} 表示当你创建模版的时候,创建的 .java 文件还是 .kt 文件,而相对应的,你需要在编写模版例子的时候分别写上对应的两份 Java 与 Kotlin 代码

open 代码块就是创建模版后,默认打开的文件

template.xml

template.xml 代码略长,这里只是贴出了大致代码

format="5"

revision="5"

name="Empty Activity"

minApi="9"

minBuildApi="14"

description="Creates a new empty activity">

id="activityClass"

name="Activity Name"

type="string"

constraints="class|unique|nonempty"

suggest="${layoutToActivity(layoutName)}"

default="MainActivity"

help="The name of the activity class to create" />

...

template_blank_activity.png

我们挑出其中的重点来说

表示当前的这个模版的分类,当前的 Value 是 Activity ,就表示它会出现在 File -> New -> Activity 中,这个是可以自定义的.

template_blank_activity.png

thumbs 用于指定创建模版时所展示出来的图片

而最重要的,还是 parameter 代码块的内容了,在这之中,我们只需要关注以下几个,其他的顾名思义即可。

id="activityClass"

name="Activity Name"

type="string"

constraints="class|unique|nonempty"

suggest="${layoutToActivity(layoutName)}"

default="MainActivity"

help="The name of the activity class to create" />

activityClass 表示所要创建的 Activity ,其中 default 为默认名。

id="generateLayout"

name="Generate Layout File"

type="boolean"

default="true"

help="If true, a layout file will be generated" />

上面的代码块表示是否同时自动创建一个Activity对应的布局

id="layoutName"

name="Layout Name"

type="string"

constraints="layout|unique|nonempty"

suggest="${activityToLayout(activityClass)}"

default="activity_main"

visibility="generateLayout"

help="The name of the layout to create for the activity" />

layoutName 则表示布局的名字,这里的 suggest 属性所填写的内容即为布局名,${activityToLayout(activityClass)}则为跟随Activity的名字,其中 activityClass 是Activity名字的引用

剩下的不用再作说明,基本上可以见名知意。

模版代码

接下来我们从 EmptyActivity 中的 root 目录一直进入,直到看到下面两个文件

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

可以看到,一个后缀是 java.ftl 另外一个后缀是 kt.ftl,他们分别用于创建 Java模版与Kotlin模版,如果你暂时不使用Kotlin的话,可以不用去关心 Kotlin模版,当你完成了Java模版的编写,也可以使用 Android Studio自带的转换功能,还是蛮方便的。

下面来看一下Java的模版代码:

package ${packageName};

import ${superClassFqcn};

import android.os.Bundle;

import android.widget.TextView;

#if>

public class ${activityClass} extends ${superClass} {

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.${layoutName});

// Example of a call to a native method

android.util.Log.d("${activityClass}", stringFromJNI());

#if>

}

}

${packageName}:表示当前包名

${activityClass}:表示当前的Activity名字

math?formula=%7BsuperClass%7D%EF%BC%9A%E8%A1%A8%E7%A4%BA%E7%BB%A7%E6%89%BF%E7%9A%84Activity%EF%BC%8C%E5%90%8C%E6%97%B6%E4%B8%BA%E4%BA%86%E8%AE%A9%E8%BF%99%E4%B8%AA%E7%88%B6%E7%B1%BB%E7%94%9F%E6%95%88%EF%BC%8C%E9%9C%80%E8%A6%81%E5%9C%A8import%E4%B8%AD%E5%8A%A0%E5%85%A5{superClassFqcn}

${layoutName}:当前Activity所对应的布局名

目前我们只需要关注上面这部分,接下来可以看一下我们实际想要创建的MVP结构:

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

编写模版代码前,最好的方式是先写一遍例子,然后对照例子去替换关键名部分,这样做是最轻松的。

下面就来看一看具体的实现吧:

样例代码

接口部分:TestActivityContact

package com.example.testcustomtemplates.contact;

public interface TestActivityContact {

interface Presenter {

void succeed(T t);

void failed(T t);

void error(Throwable e);

void subscribe();

void unSubscribe();

}

interface View {

void setPresenter(Presenter presenter);

void succeed(T t);

void failed(T t);

void error(Throwable e);

}

interface Model {

void setPresenter(Presenter presenter);

}

}

为了方便测试,这里并没有另外创建一些基类接口,可以看到上面代码中分别对应 MVP 结构中三个模块的接口,写的是最基本的需求方法,不过 MVP 也不都是完全一样的,这里你可以定义自己想写的方法。

Model层:TestActivityModel

package com.example.testcustomtemplates.model;

import android.content.Context;

import com.example.testcustomtemplates.contact.TestActivityContact;

public class TestActivityModel implements TestActivityContact.Model {

private Context context;

private TestActivityContact.Presenter mPresenter;

public TestActivityModel(Context context) {

this.context = context;

}

@Override

public void setPresenter(TestActivityContact.Presenter presenter) {

this.mPresenter = presenter;

}

}

Model层主要就是做一些网络请求,存储之类的数据相关操作,不可以持有对View的引用,他是通过Presenter去和View进行交互的。

Presenter层:TestActivityPresenter

package com.example.testcustomtemplates.presenter;

import android.content.Context;

import com.example.testcustomtemplates.contact.TestActivityContact;

import com.example.testcustomtemplates.model.TestActivityModel;

public class TestActivityPresenter implements TestActivityContact.Presenter {

private TestActivityContact.View mView;

private TestActivityModel mModel;

private Context context;

public TestActivityPresenter(TestActivityContact.View mView, Context context) {

this.mView = mView;

this.context = context;

mModel = new TestActivityModel(context);

}

@Override

public void succeed(T t) {

}

@Override

public void failed(T t) {

}

@Override

public void error(Throwable e) {

}

@Override

public void subscribe() {

}

@Override

public void unSubscribe() {

}

}

Presenter层自然不必多说,他最好是不要持有View控件的引用,大部分的逻辑操作需要他来完成,不过不可避免的,如果业务逻辑复杂了,Presenter层也会变得臃肿,这也是MVP结构的一个短处。

View层:TestActivity

package com.example.testcustomtemplates.activity;

import android.support.v7.app.AppCompatActivity;

import android.os.Bundle;

import com.example.testcustomtemplates.R;

import com.example.testcustomtemplates.contact.TestActivityContact;

import com.example.testcustomtemplates.presenter.TestActivityPresenter;

public class TestActivity extends AppCompatActivity implements TestActivityContact.View {

private TestActivityContact.Presenter mPresenter;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_test);

new TestActivityPresenter(this, this);

}

@Override

public void setPresenter(TestActivityContact.Presenter presenter) {

this.mPresenter = presenter;

}

@Override

public void succeed(T t) {

}

@Override

public void failed(T t) {

}

@Override

public void error(Throwable e) {

}

}

Activity或者Fragment都可以用作View层,这层主要是对一些视图控件的状态进行切换,不做复杂的逻辑操作。

看完上面的这些代码后,其实就可以开始直接编写我们的模版代码了。

模版编写

首先,可以Copy一份 EmptyActivity 整个模版的文件,然后改一下名字,随便什么都可以,这里我将其改成 MvpDemoActivity

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

然后我们首先对 template.xml 文件进行修改,主要修改下面这个部分:

然后是对 recipe.xml.ftl 文件进行修改,修改后如下:

#if>

to="${escapeXmlAttribute(srcOut)}/activity/${activityClass}.java" />

to="${escapeXmlAttribute(srcOut)}/model/${activityClass}Model.java" />

to="${escapeXmlAttribute(srcOut)}/contact/${activityClass}Contact.java" />

to="${escapeXmlAttribute(srcOut)}/presenter/${activityClass}Presenter.java" />

上面的代码表示只编写了Java版,当然你在修改这个文件之前还是需要创建相对应的几个类的模版代码的。这里出于篇幅考虑暂时就不贴出实际的模版代码了,下面会给出github地址,编写了Java版和Kotlin版的,大家可以拿去参考

当然,有好的模版也可以一起分享一下

efa312bf49a0?utm_campaign=maleskine&utm_content=note&utm_medium=seo_notes&utm_source=recommendation

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值