mui android 列表,mui框架安卓离线打包高级应用

mui离线打包功能可以通过插件的方式让js和原生代码通信,以达到调用第三方原生sdk等目的,但有些第三方原生sdk需要在activity的生命周期里调用某些方法进行初始化、释放资源等操作,这就没法通过简单的插件来实现了。这两天研究了一下这方面的东西,在此记录。

需求

第三方sdk有初始化、暂停、启用、释放资源这几个生命周期相关方法,要求在页面onCreate中调用初始化,在页面onPause时调用暂停,在页面onResume时调用恢复,在销毁时调用释放资源。

分析

如果是原生应用,这完全没难度,重写生命周期方法并在里面调用对应的sdk方法即可,但mui的离线打包框架中的activity没法直接碰到,那么首先我们要找找有没有相关的生命周期回调可以设置,如果没有,尝试继承它看能否实现。

动手做

百度一下

这绝对应该是第一步,如果已经有现成的解决方案,将大大节约我们的时间。不过遗憾的是,没找到。。

研究源码流程

首先,这个离线打包框架不是开源的,我们只能通过反编译来研究它的流程(Androidstudio自带了反编译)。大致流程是:应用入口是PandoraEntry这个Activity直接子类,这里只重写了onCreate方法,在里面判断了本应用是否是steam_app(貌似是mui自己搞的一个类似Quick App和Instant App的尴尬东西,无视它),如果不是,则启动PandoraEntryActivity,注意,这个类名比前面那个多了个Activity,不得不吐槽mui神(la)奇(ji)的命名方式,这个PandoraEntryActivity就是整个app的承载者,因为它的父亲在启动它20毫秒后就绝望的自尽了(所有html页面(也就是webview)都要由它来养活,不绝望都难),而它本身也不会启动任何其他activity,除非你用原生的方式在插件里启动。

那么接下来我们要做的就是在这个PandoraEntryActivity的生命周期回调里加入调用第三方sdk方法的逻辑。怎么做?继承,我们自己写一个Activity继承它,然后写入需要的逻辑。代码如下:

public class MyPandoraEntryActivity extends PandoraEntryActivity {

//第三方sdk封装

private EidReader eidReader;

@Override

public void onCreate(Bundle bundle) {

super.onCreate(bundle);

//初始化

eidReader = EidReader.getInstance(this);

eidReader.init();

}

@Override

public void onDestroy() {

super.onDestroy();

//销毁、释放资源

eidReader.release();

}

@Override

public void onResume() {

super.onResume();

//恢复第三方sdk功能

eidReader.enableReadModeIfNeed();

}

@Override

public void onPause() {

super.onPause();

//启用第三方sdk功能

eidReader.disableReadModeIfNeed();

}

}

好,子类写完了,那么怎么让它取代原来的PandoraEntryActivity呢?还是通过继承,我们继承入口的PandoraEntry,然后把它设置为程序入口。但是具体实现上有两种方法:

1.直接继承Activity

前面我们通过反编译已经知道PandoraEntry只是重写了onCreate方法,在onCreate方法中它描述了打开PandoraEntryActivity的逻辑,那我们就把它拷贝出来,然后修改。代码如下:

@Override

protected void onCreate(Bundle bundle) {

super.onCreate(bundle);

Intent intent = this.getIntent();

boolean isSteamApp = false;

try {

isSteamApp = intent.getBooleanExtra("is_stream_app", isSteamApp);

} catch (Exception var5) {

var5.printStackTrace();

this.finish();

return;

}

if (isSteamApp) {

intent.setClass(this, WebAppActivity.class);

intent.putExtra("is_stream_app", true);

} else {

intent.putExtra("short_cut_class_name", PandoraEntry.class.getName());

//这里把原来的类替换成了我们继承的

intent.setClass(this, MyPandoraEntryActivity.class);

}

this.startActivity(intent);

(new Handler()).postDelayed(new Runnable() {

public void run() {

MyPandoraEntry.this.finish();

}

}, 20L);

}

2.继承PandoraEntry并重写startActivity,替换ComponentName

这个也比较好理解,直接上代码了:

@Override

public void startActivity(Intent intent) {

ComponentName componentName = intent.getComponent();

String packageName = componentName.getPackageName();

String className = componentName.getClassName();

if(className.equals("io.dcloud.PandoraEntryActivity")){

intent.setComponent(new ComponentName(packageName, "com.gopha.qxtandroidwrapper.MyPandoraEntryActivity"));

}

Log.e("mypandoraEntry", String.format("activity = %s, startActivity, activity name = %s", this, intent.getComponent().getClassName()));

super.startActivity(intent);

}

当然,这种方法只是理论上可行,如果真这么做了,相当于onCreate里做了点无用功,所以在能碰到onCreate中逻辑的前提下,不建议这么做。

经测试,以上两种方法都能满足需求。诶,等等,功能是能正常使用,但为什么app退出了之后再进就闪退了呢?

排查bug

首先看log,发现log并没有异常信息输出,然后发现,原来是PandoraEntry并没有启动我们的MyPandoraEntryActivity,但它的20毫秒后自杀的逻辑还是正常执行,所以看起来就像闪退了一样。这种情况首先想到的就是flag是否相同,因为aar中的manifest我看不到,而flag是多个值经按位或操作拼装成的一个复合值,分析它没有必要,所以这里直接用一种最简单粗暴的方法:调试原PandoraEntry的startActivity方法,直接把flag的int值取出来然后在我们子类的startActivity中设置回去,经测试,问题完美解决。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值