当应用未运行时,无法通过通知调用活动方法

时间:2018-08-13 12:57:22

标签: java android android-intent android-notifications android-pendingintent

我有一个警报提醒,显示带有额外数据的通知。我要做的是通过意图从通知中将多余的数据发送到我的主要活动,然后打开应用并在该活动中调用特定方法(位置信息)。该方法将根据我通过的locationId打开一个新活动。

当我的应用程序在后台 中运行时,它很好用,但是当它被终止时,即当应用程序完全关闭时,它将打开该应用程序的主要活动,但不会调用该方法。

我的通知代码过去曾故意发送数据:

Intent intent = new Intent(this, MainActivity.class);
intent.putExtra("fromNotification", true);
intent.putExtra("locationId", locationId);
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP 
              | Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_NEW_TASK);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 111, intent, 
              PendingIntent.FLAG_UPDATE_CURRENT);

这就是我称呼我的方法:

@Override
protected void onNewIntent(Intent intent) {
    super.onNewIntent(intent);
    setIntent(intent);
    if (getIntent().hasExtra("fromNotification")) {
        String locationId = getIntent().getStringExtra("locationId");
        CallFetchLocationInfoFromMapsFragment(locationId);
    }
}

我还尝试了在主要活动的onCreate末尾添加this,这本来可以解决,但它什么也没做:

if (getIntent().hasExtra("fromNotification")) {
    String locationId = getIntent().getStringExtra("locationId");
    CallFetchLocationInfoFromMapsFragment(locationId);
}

我尝试了更多解决方案(包括thisthisthisthisthisthis,{{3 }}等),但无法使其正常工作。

清单(只是主要活动部分):

<application
    android:name="com.weekendcoder.kemo.hotels.app.AppController"
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">
    <activity
        android:name=".MainActivity"
        android:label="@string/app_name"
        android:screenOrientation="portrait"
        android:theme="@style/AppTheme.NoActionBar"
        android:launchMode="singleTask">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    ...
</application>

这是我理想的工作流程的描述-如果单击通知,它将打开我的MainActivity,其中包含地图片段。将所有内容加载到地图上后,我想从主活动的地图开始一个新活动-我想使用从通知中收到的ID打开locationDetails活动。由于调试会在我关闭应用程序后立即停止,因此无法查明问题所在,但是我尝试添加以下内容:

if (getIntent().hasExtra("fromNotification")) 
{
    Toast.makeText(MainActivity.this, "Notification!", Toast.LENGTH_LONG).show();
}

在我的onCreate开头,它确实显示了吐司。由于无法加载和设置地图,因此无法在onCreate的开头使用此条件,因此我在onCreate的末尾添加了此条件。在这种情况下,它永远不会被调用。

我可以通过直接从通知中打开该新活动来使其工作,但这不是我的理想解决方案。我希望能够在应用程序关闭时从通知中调用活动中的任何方法。

我也尝试添加此内容:

if (getIntent().hasExtra("fromNotification")) {
    String locationId = getIntent().getStringExtra("locationId ");
    CallFetchLocationInfoFromMapsFragment(locationId);
    Toast.makeText(MainActivity.this, "Notification id: " locationId, Toast.LENGTH_LONG).show();
}else{
    Toast.makeText(MainActivity.this, "No notification", Toast.LENGTH_LONG).show();
}

在onCreate的最后,只是看是否可以显示此值,但我一直在收到“无通知”吐司。

当我在onCreate开头放置相同的if条件时,它会崩溃:

08-14 18:06:03.319 18591-18591/com.weekendcoder.kemo.hotels D/SQLiteHandler: Fetching notification settings from sqlite db
08-14 18:06:03.329 18591-18591/com.weekendcoder.kemo.hotels W/Settings: Setting airplane_mode_on has moved from android.provider.Settings.System to android.provider.Settings.Global, returning read-only value.
08-14 18:06:03.384 18591-18591/com.weekendcoder.kemo.hotels D/AndroidRuntime: Shutting down VM
08-14 18:06:03.384 18591-18591/com.weekendcoder.kemo.hotels W/dalvikvm: threadid=1: thread exiting with uncaught exception (group=0x41cdc700)
08-14 18:06:03.389 18591-18591/com.weekendcoder.kemo.hotels E/AndroidRuntime: FATAL EXCEPTION: main
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.weekendcoder.kemo.hotels/com.weekendcoder.kemo.hotels.MainActivity}: java.lang.NullPointerException
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2295)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2349)
    at android.app.ActivityThread.access$700(ActivityThread.java:159)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1316)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:176)
    at android.app.ActivityThread.main(ActivityThread.java:5419)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:525)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1046)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:862)
    at dalvik.system.NativeStart.main(Native Method)
 Caused by: java.lang.NullPointerException
    at com.weekendcoder.kemo.hotels.sliderfragments.MapsFragment.fetchlocationDetails(MapsFragment.java:545)
    at com.weekendcoder.kemo.hotels.MainActivity.CallFetchlocationDetailsFromMapsFragment(MainActivity.java:661)
    at com.weekendcoder.kemo.hotels.MainActivity.onCreate(MainActivity.java:324)
    at android.app.Activity.performCreate(Activity.java:5372)
    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1104)
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2257)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2349) 
    at android.app.ActivityThread.access$700(ActivityThread.java:159) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1316) 
    at android.os.Handler.dispatchMessage(Handler.java:99) 
    at android.os.Looper.loop(Looper.java:176) 
    at android.app.ActivityThread.main(ActivityThread.java:5419) 
    at java.lang.reflect.Method.invokeNative(Native Method) 
    at java.lang.reflect.Method.invoke(Method.java:525) 
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1046) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:862) 
    at dalvik.system.NativeStart.main(Native Method) 

编辑:

有可能可行的解决方案,但我对此不太满意,希望找到更智能/更安全的解决方案。解决方案是先获取ID,然后添加以下代码:

if (locationId != null) {
    final Handler handler = new Handler();
    handler.postDelayed(new Runnable() {
        @Override
        public void run() {
            //Do something after 100ms
            CallFetchLocationInfoFromMapsFragment(locationId);
        }
    }, 5000);
}

这将使我的应用程序等待5秒钟,然后在执行所有其他方法之后调用该方法。 此方法有效,但是我不确定这是否是进行此操作的最佳方法,某些设备可能会比较慢。最好找到一种方法来检查onCreated是否完全完成执行,然后调用此方法。

2 个答案:

答案 0 :(得分:1)

我认为您需要这个

if (getIntent().hasExtra("fromNotification")) {
    String locationId = getIntent().getStringExtra("locationId");
    CallFetchLocationInfoFromMapsFragment(locationId);
}

同时为清单中的活动设置singleTask启动模式

<activity android:name=".YourActivity"
        android:launchMode="singleTask">
    ...
</activity>

答案 1 :(得分:1)

在上面的评论讨论中,您表示您正在使用launchMode="singleTask"。这是一种特殊的启动模式,对于您的应用程序不是必需的。它通常会带来更多无法解决的问题。请删除特殊的启动模式,看看是否有帮助。完成此操作后,在应用未运行时点按通知将启动应用并创建MainActivity的新实例。

由于您的onCreate()尚未设置,您很有可能无法在Fragment中做您想做的事情。您可能只需要记住自己在onCreate()中想要做的事情(开始位置的详细信息),但是实际上要等到onResume()或完成Fragment或其他操作之后再进行。