我有一个警报提醒,显示带有额外数据的通知。我要做的是通过意图从通知中将多余的数据发送到我的主要活动,然后打开应用并在该活动中调用特定方法(位置信息)。该方法将根据我通过的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);
}
我尝试了更多解决方案(包括this,this,this,this,this,this,{{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是否完全完成执行,然后调用此方法。
答案 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
或其他操作之后再进行。