之前 APP 一直都是通过桌面的 icon 点击进入应用,所以都会跳转到登录页进行验证,再跳转到首页,最近有新的需求,第三方直接打开应用的某个 activity ,如果没有 token 会报 500 或者 token 无效,所以给应用加个登录拦截,如果没有登录,就跳转到登录页登录之后再返回要跳转的 activity 。
先是看了一遍网上的实现方式,觉得都挺麻烦的,就自己试试通过 ActivityLifecycleCallbacks 来做一个登录拦截。
首先写个 MyApplicationLifecycleCallback 实现 ActivityLifecycleCallbacks 接口,该 class 如下:
public class MyApplicationLifecycleCallback implements Application.ActivityLifecycleCallbacks {
@Override
public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
}
@Override
public void onActivityStarted(final Activity activity) {
}
@Override
public void onActivityResumed(Activity activity) {
}
@Override
public void onActivityPaused(Activity activity) {
}
@Override
public void onActivityStopped(Activity activity) {
}
@Override
public void onActivitySaveInstanceState(Activity activity, Bundle outState) {
}
@Override
public void onActivityDestroyed(Activity activity) {
}
}
然后在 onActivityStarted 里面写拦截的代码:
// 当前activity的名字
String className = activity.getLocalClassName();
// 不需要拦截的activity名字
String permissionClassName = "com.permission.PermissionActivity";
// 是否是需要拦截的activity
boolean isInterceptActivity = !loginClassName.equals(className) && !permissionClassName.equals(className);
// 是否有token 这里取application的全局token
boolean haveToken = !(GlobalValue.TOKEN== null || "".equals(GlobalValue.TOKEN));
if (isInterceptActivity && !haveToken) {
Intent loginIntent = new Intent();
loginIntent.setClass(activity, LoginActivity.class);
activity.startActivity(loginIntent);
}
这样子就可以拦截没有登录的时候,其他应用打开需要登录的界面打开的就是登录页了。
紧接着发现一个问题。在 ActivityLifecycleCallbacks 拦截的话,被拦截的 activity 已经 onResumed() 显示到前台,再跳转到 LoginActivity ,所以所有操作已经做完了,包括网络请求,所以跳转登录之后,finish 掉 LoginActivity 后,打开的是一个空白页面,不是我们想要的效果。
所以我就想到保存当前跳转的 intent ,结束掉当前被拦截的 activity ,等到登录结束后再跳回去。
// 当前activity的名字
String className = activity.getLocalClassName();
// 不需要拦截的activity名字
String permissionClassName = "com.permission.PermissionActivity";
// 是否是需要拦截的activity
boolean isInterceptActivity = !loginClassName.equals(className) && !permissionClassName.equals(className);
// 是否有token 这里取application的全局token
boolean haveToken = !(GlobalValue.TOKEN== null || "".equals(GlobalValue.TOKEN));
if (isInterceptActivity && !haveToken) {
Intent loginIntent = new Intent();
loginIntent.setClass(activity, LoginActivity.class);
activity.startActivity(loginIntent);
// 被拦截的activity的intent
beforeIntent = activity.getIntent();
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
// 在这里finish掉,然后一会loginActivity结束了再打开
activity.finish();
}
}, 500);
}
然后在 onActivityStopped 里面增加判断
String className = activity.getLocalClassName();
if (loginClassName.equals(className)) {
// 当loginActivity结束的时候,判断beforeIntent为不为空,不为空就说明是第三方打开的,跳转回去
if (beforeIntent != null) {
activity.startActivity(beforeIntent);
beforeIntent = null;
}
}
所以这个 MyApplicationLifecycleCallback.java 文件的全部代码是:
public class MyApplicationLifecycleCallback implements Application.ActivityLifecycleCallbacks {
private Intent beforeIntent;
String loginClassName = "com.LoginActivity";
@Override
public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
}
@Override
public void onActivityStarted(final Activity activity) {
// 登录拦截 登录过后再跳回去
// 当前activity的名字
String className = activity.getLocalClassName();
// 不需要拦截的activity名字
String permissionClassName = "com.permission.PermissionActivity";
// 是否是需要拦截的activity
boolean isInterceptActivity = !loginClassName.equals(className) && !permissionClassName.equals(className);
// 是否有token 这里取application的全局token
boolean haveToken = !(GlobalValue.TOKEN== null || "".equals(GlobalValue.TOKEN));
if (isInterceptActivity && !haveToken) {
Intent loginIntent = new Intent();
loginIntent.setClass(activity, LoginActivity.class);
activity.startActivity(loginIntent);
// 被拦截的activity的intent
beforeIntent = activity.getIntent();
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
// 在这里finish掉,然后一会loginActivity结束了再打开
activity.finish();
}
}, 500);
}
}
@Override
public void onActivityResumed(Activity activity) {
}
@Override
public void onActivityPaused(Activity activity) {
}
@Override
public void onActivityStopped(Activity activity) {
String className = activity.getLocalClassName();
if (loginClassName.equals(className)) {
// 当loginActivity结束的时候,判断beforeIntent为不为空,不为空就说明是第三方打开的,跳转回去
if (beforeIntent != null) {
activity.startActivity(beforeIntent);
beforeIntent = null;
}
}
}
@Override
public void onActivitySaveInstanceState(Activity activity, Bundle outState) {
}
@Override
public void onActivityDestroyed(Activity activity) {
}
}
网友评论