android - 使用持久通知允许用户返回运行 Android 应用程序

标签 android notifications history android-notifications

我正在开发一个包含大量 Activity 的应用程序。我想创建一个持久性通知(或多或少)说“AppName - Return to AppName”,只要我的后台服务运行,它就会出现。 创建和处理通知没有问题。

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

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

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

对 Java/Android 新手有什么意见或建议吗?仅供引用,我的主要历史是使用基于脚本的语言。

最佳答案

我更喜欢你创建“returnFromNotify” Activity 的最初想法,而不是你提出的解决方法,因为它可以检测 ResumeActivity 是否在堆栈底部(因此唯一的 Activity 在堆栈中)。

这里是你如何做到的:

将您的 ResumeActivity 添加到 list 并指定 noHistory属性:

<activity android:name=".ResumeActivity" android:noHistory="true" />

指定 noHistory 将确保此 Activity 不会在完成后立即留在堆栈中。这样您就知道只有当前正在运行的 ResumeActivity 实例会显示在堆栈中。

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

<uses-permission android:name="android.permission.GET_TASKS" />

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

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;
    }

}

我知道你在 2 年前问过这个问题,但我提供这个答案是为了防止其他人遇到这种特殊情况(就像我一样)。我认为您最初致力于解决的问题是正确的。

关于android - 使用持久通知允许用户返回运行 Android 应用程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3568250/

相关文章:

file-io - TortoiseHg 中重命名文件的文件历史记录

java - git 检测到我们 Java 项目中的重命名/移动很差 - 怎么办?

android - 带有自定义 NumberPickers 的 ListView - 不显示递增数字

java - Android中的调用方法

android - Lollipop 通知背景色

android - 如何在 android 状态栏中创建通知?

android - 升级手机上安装的 Kivy 版本?

android - 如何在 Jetpack Compose 1.2.0 中访问 "androidx.savedstate.ViewTreeSavedStateRegistryOwner"?

ios - 从 xcassets 中的图像创建 URL 总是返回 nil

bash - 将最后一个 bash 命令复制到剪贴板