13

我正在使用这个解决方案:如何使通知意图恢复而不是新的意图?

当我正常运行我的应用程序时,我工作正常。但是我的应用程序具有共享功能。

当我从图库应用程序中选择要共享的图像并在我的活动中打开它时,我会在活动中创建一个通知。我希望当用户单击通知时,它会打开现有活动(由图库应用程序打开)

问题是,当我单击通知时,它不会恢复该活动,而是打开该活动的新实例或之前已经打开的另一个实例

编辑:更多解释

我的应用程序称为 ShareApp,活动称为 ShareActivity 问题是,当我通过图库应用程序打开 ShareActivity 时,会在图库应用程序任务堆栈的顶部创建一个 ShareActivity 实例。现在,当我创建一个指向我的 ShareActivity 的通知时,我使用了意图:

Intent intent = new Intent(this, ShareActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);

问题是,当我单击通知时,它指向 ShareApp 任务堆栈中的 ShareActivity 实例,而不是图库应用程序任务的堆栈。

知道如何指向正确的任务堆栈吗?

编辑2:我的代码

        int defaults = Notification.FLAG_NO_CLEAR;
        NotificationCompat.Builder mBuilder =
                    new NotificationCompat.Builder(this)
                    .setSmallIcon(R.drawable.ic_launcher)
                    .setContentTitle(getString(R.string.app_name))
                        .setContentText(text)
                        .setDefaults(defaults);
        Intent intent = new Intent(this, this.getClass());
        intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
        intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);

        PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT);

        mBuilder.setContentIntent(pendingIntent);
        NotificationManager mNotificationManager =
            (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
        mNotificationManager.notify(_ID, mBuilder.build());

编辑 3: adb shell dumpsys 活动(在应用 David https://stackoverflow.com/a/16901603/1334268的代码之后)

我的应用程序称为 shareApp,我的活动是 ShareActivity

在点击通知之前:

Main stack:
    TaskRecord{41965708 #6 A com.android.gallery3d}
    Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.android.gallery3d/.app.Gallery bnds=[364,50][108,108]}
      Hist #2: ActivityRecord{417ccc28 com.yeahman.shareapp/.ShareActivity}
        Intent { act=android.intent.action.SEND_MULTIPLE typ=image/* cmp=com.yeahman.shareapp/.ShareActivity (has extras) }
        ProcessRecord{41c868d0 6088:com.yeahman.shareapp/10116}
      Hist #1: ActivityRecord{4135e540 com.android.gallery3d/.app.Gallery}
        Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.android.gallery3d/.app.Gallery bnds=[364,50][108,108] }

点击通知后:

 Main stack:
    TaskRecord{41965708 #6 A com.android.gallery3d}
    Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.android.gallery3d/.app.Gallery bnds=[364,50][108,108]}
      Hist #3: ActivityRecord{4169f358 com.android.gallery3d/.app.Gallery}
        Intent { flg=0x20000000 cmp=com.android.gallery3d/.app.Gallery bnds=[0,205][480,301] }
        ProcessRecord{4175dd28 5808:com.android.gallery3d/10036}
      Hist #2: ActivityRecord{417ccc28 com.yeahman.shareapp/.ShareActivity}
        Intent { act=android.intent.action.SEND_MULTIPLE typ=image/* cmp=com.yeahman.shareapp/.ShareActivity (has extras) }
        ProcessRecord{41c868d0 6088:com.yeahman.shareapp/10116}
      Hist #1: ActivityRecord{4135e540 com.android.gallery3d/.app.Gallery}
        Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.android.gallery3d/.app.Gallery bnds=[364,50][108,108] }
4

5 回答 5

31
    Notification.Builder mBuilder =
            new Notification.Builder(this)
            .setSmallIcon(R.drawable.cmplayer)
            .setContentTitle("CoderoMusicPlayer")
            .setContentText("PLayer0!");

    Intent resultIntent = new Intent(this, AndroidBuildingMusicPlayerActivity.class);
    resultIntent.setAction(Intent.ACTION_MAIN);
    resultIntent.addCategory(Intent.CATEGORY_LAUNCHER);

    PendingIntent pendingIntent = PendingIntent.getActivity(this, 0,
            resultIntent, 0);

    mBuilder.setContentIntent(pendingIntent);
    NotificationManager mNotificationManager =
        (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
    mNotificationManager.notify(1, mBuilder.build());

只需复制代码并将其粘贴到您的主启动器活动中。

于 2014-01-29T07:43:47.407 回答
7

如果您在不同的任务中有多个ShareActivityactive 实例,则您无法通过启动 Activity 来告诉 Android 要重新激活哪个任务。在我看来,在这种情况下,您想要做的不是启动您的活动,而是启动 Gallery 应用程序。如果 Gallery 应用程序已经在运行并且您的 Activity 位于任务堆栈的顶部,那么您需要做的就是将 Gallery 应用程序置于前台。为此,请为 Gallery 应用程序创建一个“启动 Intent”,并确保Intent.FLAG_ACTIVITY_NEW_TASK在 Intent 上进行设置。您应该能够PackageManager通过使用获得“启动意图”getLaunchIntentForPackage()

编辑:使用 ActivityManager 的示例代码

你可以试试这个:

// Get the root activity of the task that your activity is running in
ActivityManager am = (ActivityManager)getSystemService(ACTIVITY_SERVICE);
List<ActivityManager.RunningTaskInfo> tasks = am.getRunningTasks(1);
ActivityManager.RunningTaskInfo task = tasks.get(0); // Should be my task
ComponentName rootActivity = task.baseActivity;

// Now build an Intent that will bring this task to the front
Intent intent = new Intent();
intent.setComponent(rootActivity);
// Set the action and category so it appears that the app is being launched
intent.setAction(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);

注意:您需要GET_TASKS为此获得许可。我没有尝试过,不能保证它会起作用。文档不鼓励使用getRunningTasks(),但这并不意味着它不适合您的目的。

编辑将动作/类别添加到意图

注意:我实际上构建了一个小应用程序来测试它。我需要将 ACTION=MAIN 和 CATEGORY=LAUNCHER 添加到 Intent 中(如果您使用过,就会出现PackageManager.getLaunchIntentForPackage()。我有一个 HTC One S,所以在我的设备上实际调用了“Gallery”应用程序com.htc.album/.AlbumMain.ActivityMainDropList,但这应该仍然为你工作。

于 2013-06-03T16:38:30.683 回答
5

android:launchMode="singleTop"

在您manifest.xml的主要活动中,看看是否有帮助。

于 2013-06-02T18:20:38.037 回答
3

这是我多年来一直使用的解决方案。

你的问题是你的 PendingIntent 从一个无应用的内容开始它的目标。

您的第一步是将 Intent 的结果带入您的应用程序上下文中。为此,请让您的通知向“ForwardingReceiver”广播

Intent newIntent = new Intent(context, ForwardingReceiver.class);
...
mBuilder.setContentIntent(PendingIntent.getBroadcast(context, category, newIntent, PendingIntent.FLAG_UPDATE_CURRENT));

ForwardingReceiverextendsBroadcastReceiver和它的方法onReceive()你现在有你的应用程序当前上下文并且可以sendOrderedBroadcast()像这样使用:

Intent forwardIntent = new Intent();
forwardIntent.putExtras(intent.getExtras());
forwardIntent.setAction(GenericReceiver.class.getCanonicalName());
context.sendOrderedBroadcast(forwardIntent, null);

现在,有趣的部分。您需要从一个共同的 Activity 父级扩展您的所有应用程序活动。您可能已经为日志记录或分析执行此操作。在这个CommonActivityParentonResume()中需要注册接收优先级在0以上(比如10)的广播

mReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        performActionUsingIntent(intent);
    }
};

IntentFilter filter = new IntentFilter();
filter.addAction(GenericReceiver.class.getCanonicalName());
filter.setPriority(10);
registerReceiver(mReceiver, filter);

(记得在 onPause() 中取消注册)

这意味着如果您的应用程序在前台,它将(间接)收到通知点击并允许您执行操作。

但是,如果您的 Activity 在后台,则不会注册此接收器。没问题,你的“GenericReceiver”应该是这样的

@Override
public void onReceive(Context context, Intent intent) {
    PackageManager pm = context.getPackageManager();
    Intent newIntent = pm.getLaunchIntentForPackage(context.getPackageName());
    newIntent.putExtras(intent.getExtras());
    context.startActivity(newIntent);
}

这将开始您的启动活动。这个启动活动应该在它的 onCreate() 中有代码来检测它是否是任务根。如果不是,它应该立即关闭(因此用户永远不可见)。这有效地将最后一个打开的任务置于前台:

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

    if (!isTaskRoot()
            && getIntent().hasCategory(Intent.CATEGORY_LAUNCHER)
            && getIntent().getAction() != null
            && getIntent().getAction().equals(Intent.ACTION_MAIN)) {

        // Use this space to add the intent to a static singleton. This will be used in CommonActivityParent to perform an action.
        CommonActivityParent.sIntent = getIntent();
        finish();
        return;
    }
}

要完成,请确保在您在registerReceiver()上面调用 CommonActivityParent 之后,执行以下检查:

if(sIntent != null) {
     Intent tempIntent = sIntent;
     sIntent = null;
     performActionUsingIntent(tempIntent);
}

嘿 presto - 您的应用程序现在的行为就像通知点击实际上是来自您的应用程序内的点击一样。所需要的只是大量的工作!

希望有人有一种不那么复杂的方式来做到这一点。

于 2016-05-27T14:39:58.780 回答
0

有关其他信息,请不要忘记添加待处理的意图标志以使 onNewIntent 工作!

真的:

PendingIntent.getActivity(context, 0, resultIntent, PendingIntent.FLAG_UPDATE_CURRENT)

错误的:

PendingIntent.getActivity(context, 0, resultIntent, 0)
于 2020-05-12T00:43:01.957 回答