Android Studio自定义模板实现一键创建MVP结构

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

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

<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 为默认名。

<parameter

id=“generateLayout”

name=“Generate Layout File”

type=“boolean”

default=“true”

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

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

<parameter

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 目录一直进入,直到看到下面两个文件

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

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

package ${packageName};

import ${superClassFqcn};

import android.os.Bundle;

<#if (includeCppSupport!false) && generateLayout>

import android.widget.TextView;

</#if>

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

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

<#if generateLayout>

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

<#include “…/…/…/…/common/jni_code_usage.java.ftl”>

<#elseif includeCppSupport!false>

// Example of a call to a native method

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

</#if>

}

<#include “…/…/…/…/common/jni_code_snippet.java.ftl”>

}

  • ${packageName}:表示当前包名

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

  • {superClass}:表示继承的Activity,同时为了让这个父类生效,需要在import中加入{superClassFqcn}

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

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

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

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

样例代码

====

接口部分: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

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

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

<?xml version="1.0"?>

<#import “root://activities/common/kotlin_macros.ftl” as kt>

<#include “…/common/recipe_manifest.xml.ftl” />

<@kt.addAllKotlinDependencies />

<#if generateLayout>

<#include “…/common/recipe_simple.xml.ftl” />

</#if>

<instantiate from=“root/src/app_package/MvpActivity.java.ftl”

to=“ e s c a p e X m l A t t r i b u t e ( s r c O u t ) / a c t i v i t y / {escapeXmlAttribute(srcOut)}/activity/ escapeXmlAttribute(srcOut)/activity/{activityClass}.java” />

<instantiate from=“root/src/app_package/MvpModel.java.ftl”

to=“ e s c a p e X m l A t t r i b u t e ( s r c O u t ) / m o d e l / {escapeXmlAttribute(srcOut)}/model/ escapeXmlAttribute(srcOut)/model/{activityClass}Model.java” />

<instantiate from=“root/src/app_package/MvpContact.java.ftl”

to=“ e s c a p e X m l A t t r i b u t e ( s r c O u t ) / c o n t a c t / {escapeXmlAttribute(srcOut)}/contact/ escapeXmlAttribute(srcOut)/contact/{activityClass}Contact.java” />

<instantiate from=“root/src/app_package/MvpPresenter.java.ftl”

to=“ e s c a p e X m l A t t r i b u t e ( s r c O u t ) / p r e s e n t e r / {escapeXmlAttribute(srcOut)}/presenter/ escapeXmlAttribute(srcOut)/presenter/{activityClass}Presenter.java” />

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

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

感谢您的阅读,在文末给大家准备一个福利。本人从事Android开发已经有十余年,算是一名资深的移动开发架构师了吧。根据我的观察发现,对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

所以在此将我十年载,从萌新小白一步步成长为Android移动开发架构师的学习笔记,从Android四大组件到手写实现一个架构设计,我都有一一的对应笔记为你讲解。

当然我也为你们整理好了百度、阿里、腾讯、字节跳动等等互联网超级大厂的历年面试真题集锦。这也是我这些年来养成的习惯,一定要学会把好的东西,归纳整理,然后系统的消化吸收,这样才能极大的提高学习效率和成长进阶。碎片、零散化的东西,我觉得最没有价值的。就好比你给我一张扑克牌,我只会觉得它是一张废纸,但如果你给我一副扑克牌,它便有了它的价值。这和我们收集资料就要收集那些系统化的,是一个道理。

最后,赠与大家一句诗,共勉!

不驰于空想,不骛于虚声。不忘初心,方得始终。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

/>

最后

感谢您的阅读,在文末给大家准备一个福利。本人从事Android开发已经有十余年,算是一名资深的移动开发架构师了吧。根据我的观察发现,对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

所以在此将我十年载,从萌新小白一步步成长为Android移动开发架构师的学习笔记,从Android四大组件到手写实现一个架构设计,我都有一一的对应笔记为你讲解。

当然我也为你们整理好了百度、阿里、腾讯、字节跳动等等互联网超级大厂的历年面试真题集锦。这也是我这些年来养成的习惯,一定要学会把好的东西,归纳整理,然后系统的消化吸收,这样才能极大的提高学习效率和成长进阶。碎片、零散化的东西,我觉得最没有价值的。就好比你给我一张扑克牌,我只会觉得它是一张废纸,但如果你给我一副扑克牌,它便有了它的价值。这和我们收集资料就要收集那些系统化的,是一个道理。

[外链图片转存中…(img-O0AnDiQQ-1712618686113)]

最后,赠与大家一句诗,共勉!

不驰于空想,不骛于虚声。不忘初心,方得始终。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值