当新的应用实例启动时,如何将数据传递给已经运行的应用实例

时间:2017-10-17 10:12:01

标签: java android firebase android-activity

我的应用架构包含启动画面作为启动器活动。应用程序启动后会破坏启动画面。因此,为了防止重新启动应用程序(因为已经破坏了启动器活动),我在onCreate方法中为启动屏幕添加了这个:

if (!isTaskRoot())
{
  finish();
  return;
}

因此,如果已经运行实例并且显示正在运行的实例,则新实例将自行完成。目前很好。 但是,新的应用实例并不总是从启动器开始。我已将Firebase Messaging设置为从我的服务器接收推送消息。推送消息包含一些额外的数据,可以通过这种方式加入:

String messageId = getIntent().getStringExtra("id_message");

点击推送消息通知后启动新的应用实例时。如果应用程序已经运行,我需要在推送通知单击(它工作)后显示它,然后处理message_id已经运行的应用程序实例(新实例应该完成自己)。但我不知道如何将message_id传递给已经运行的实例,以便能够处理它并显示消息详细信息,而无需重新启动整个应用程序。

你有什么想法吗?

4 个答案:

答案 0 :(得分:1)

您可以使用中间活动来选择要启动的活动。

首先,创建LaunchActivity

public class LaunchActivity extends Activity {
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.empty);

     BaseApp application = (BaseApp) this.getApplication();

    // determine where to go
    if(application.isHasGoToMain()) {
      Intent intent = new Intent(this, MainActivity.class);
      intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
      finish();
      startActivity(intent);
    } else {
      Intent intent = new Intent(this, SplashScreenActivity.class);
      intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
      finish();
      startActivity(intent);
    }
  }
}

empty布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:visibility="gone">

</LinearLayout>

第二,如果已启动主要活动,则创建要保存的应用程序:

public class BaseApp extends Application {
  private boolean hasGoToMain = false;

  public boolean isHasGoToMain() {
    return hasGoToMain;
  }

  public void setHasGoToMain(boolean hasGoToMain) {
    this.hasGoToMain = hasGoToMain;
  }
}

您需要在启动画面活动中设置标记。

第三次,将LaunchActivity设置为AndroidManifest.xml中的启动器并将其设置为隐藏且singleTask

<activity
    android:name=".LaunchActivity"
    android:screenOrientation="portrait"
    android:configChanges="orientation"
    android:launchMode="singleTask"
    android:theme="@android:style/Theme.NoDisplay"
    android:windowSoftInputMode="stateHidden">
  <intent-filter>
    <action android:name="android.intent.action.MAIN"/>
    <category android:name="android.intent.category.LAUNCHER"/>
  </intent-filter>
</activity>

第四,您还需要以singleTask创建初始屏幕和主要活动:

<activity
    android:name=".SplashScreenActivity"
    android:launchMode="singleTask"
    android:configChanges="screenSize|orientation|screenLayout">

</activity>

<activity
    android:name=".MainActivity"
    android:launchMode="singleTask"
    android:label="@string/app_name"
    android:theme="@style/AppThemeMain.NoActionBar">

上次,您需要在启动启动活动后设置标志:

public class SplashScreenActivity extends AppCompatActivity {
  ...
  @Override 
  protected void onCreate(Bundle savedInstanceState) {
    ...

    BaseApp baseApp = (BaseApp) this.getApplication();
    baseApp.setHasGoToMain(true);
  }
}

现在您可以发送意图并在主要活动的onNewIntent()内处理它。

答案 1 :(得分:0)

首先在清单文件中的活动标记中添加启动模式,如

<activity
android:name="your_activity_name"
android:launchMode="singleTop" /> 

当您点击通知并开始任何活动时

如果您的活动不存在,则会开始新的活动。 如果该活动已经打开,那么它的onNewIntent(Intent intent)会被调用,你将收到你在这个intent参数中发送的数据。

答案 2 :(得分:0)

当应用程序从firebase收到推送时,它将由 MyFirebaseMessagingService 处理。在那里你需要创建一个待定的意图,即当用户点击通知时需要打开的活动。

Intent resultIntent = new Intent(context, ActivityToOpen.class);

    TaskStackBuilder stackBuilder = TaskStackBuilder.create(context);
    stackBuilder.addNextIntentWithParentStack(resultIntent);
    PendingIntent resultPendingIntent = stackBuilder.getPendingIntent(0, 
  PendingIntent.FLAG_UPDATE_CURRENT);

用户点击通知后,他们将转到活动 ActivityToOpen 。在该活动中,您可以输入以下代码

 Intent upIntent = 
 NavUtils.getParentActivityIntent(ActivityToOpenOnBackPress.this);
        if (NavUtils.shouldUpRecreateTask(CurrentActivityName.this, 
  upIntent)) 
       {

            TaskStackBuilder.create(CurrentActivityName.this)
                    .addNextIntentWithParentStack(upIntent)
                    .startActivities();
        } else {
            NavUtils.navigateUpTo(CurrentActivityName.this, upIntent);
        }

将此代码放在onBackPressed函数中或工具栏导航内部单击listener.So当用户按下后退按钮时,会创建一个人工堆栈,android会处理其余的。

在androidmanifest.xml中,通过添加以下代码为每个活动分配父活动     android:parentActivityName=".ParentActivityName"

答案 3 :(得分:0)

我遇到了几乎相同的问题。我的应用程序从启动器或意图开始。添加启动画面活动后,如果应用程序处于后台,则意图会从应用程序启动第二个实例。

我在清单中的两个活动中添加了android:launchMode =“singleTask”,现在它像以前一样工作。一次只有一个应用实例。

对我没有任何意义,但只要它有效我就不在乎......