使用持久通知允许用户返回运行的Android应用

Slo*_*aum 9 notifications history android android-notifications

我正在开发一个包含众多活动的应用程序 我想创建一个持久性通知(或多或少)说"AppName - 返回AppName",每当我的后台服务运行时它都会出现. 创建和处理通知没有问题.

现在,用户可以在多个屏幕/活动中的任何一个,离开应用程序,然后想要通过通知重新进入应用程序. 问题是,通知必须具有意图,该意图启动预定的活动.我希望通知重新进入应用程序,无论Activity在历史堆栈的顶部.

我第一次尝试一个丑陋的解决方法是进行一项活动(让我们称之为"returnFromNotify"),其唯一的工作就是"完成"自己的"onCreate".通知将在应用程序历史记录的范围内打开"returnFromNotify",然后立即将其自身删除,将用户发送回应用程序堆栈中的先前历史记录状态.这似乎有效...除非用户使用"后退"完全退出应用程序.然后,当他们点击通知时,"returnFromNotify"加载,然后完成,将它们发送回主屏幕(因为应用程序的历史堆栈中没有活动).

我考虑过在"returnFromNotify"之前尝试检测历史堆栈中是否有任何内容,如果没有,则启动我的主Activity.我似乎无法找到办法做到这一点.

对Java/Android新手的任何输入或建议?仅供参考,我的主要历史是使用基于脚本的语言.

小智 6

我喜欢你原来创建"returnFromNotify"活动比你提出的解决办法更好的主意,因为它可以检测,如果ResumeActivity是在栈(因此在堆栈中唯一的活动)的底部.

这是你如何做到的:

将ResumeActivity添加到清单并指定noHistory属性:

<activity android:name=".ResumeActivity" android:noHistory="true" />
Run Code Online (Sandbox Code Playgroud)

指定noHistory将确保此Activity一旦完成就不会保留在堆栈中.这样您就知道只有当前运行的ResumeActivity实例才会显示在堆栈中.

为了检查应用程序堆栈,您还必须要求GET_TASKS权限:

<uses-permission android:name="android.permission.GET_TASKS" />
Run Code Online (Sandbox Code Playgroud)

现在,您可以使用ActivityManager :: getRunningTasks()来确定ResumeActivity是否是堆栈中唯一的活动:

public class ResumeActivity extends Activity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        if(isOnlyActivityInStack()) { //check the application stack
            //This activity is the only activity in the application stack, so we need to launch the main activity
            Intent main = new Intent(this, MainActivity.class);
            main.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
            startActivity(main);
        } else {
            //Return the user to the last activity they had open
            this.finish();
        }
    }

    /**
     * Checks the currently running tasks. If this activity is the base activity, we know it's the only activity in the stack
     *
     * @return  boolean  This activity is the only activity in the stack?
     **/
    private boolean isOnlyActivityInStack() {
        ActivityManager manager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
        boolean onlyActivityInStack = false;

        for(RunningTaskInfo tasks : manager.getRunningTasks(Integer.MAX_VALUE)) {
            if(tasks.baseActivity.getPackageName().equals(this.getPackageName())) { //find this package's application stack
                if(tasks.baseActivity.getClassName().equals(this.getClass().getName())) {
                    //If the ResumeActivity is the base activity, we know that it is the only activity in the stack
                    onlyActivityInStack = true;
                    break;
                }
            }
        }

        return onlyActivityInStack;
    }

}
Run Code Online (Sandbox Code Playgroud)

我知道你在2年前问过这个问题,但我提供了这个答案,以防其他人遇到这种特殊情况(正如我所做的那样).我认为您正在使用您最初努力的解决方案.