Exception when opening Parse push notification [cl

2019-01-08 09:41发布

问题:

I setup Parse push Android SDK 1.7.0 and can receive pushes without any problem. However, when opening the push notification my app crashes with the following exception:

android.content.ActivityNotFoundException: No Activity found to handle Intent { act=android.intent.action.VIEW dat= flg=0x1000c000 (has extras) }

I also tried subclassing ParsePushBroadcastReceiver with getActivity() always returning the main activity class, same result. Below is the full stacktrace:

E/AndroidRuntime﹕ FATAL EXCEPTION: main
Process: com.ers.test, PID: 13513
java.lang.RuntimeException: Unable to start receiver com.ers.test.MyPushBroadcastReceiver: android.content.ActivityNotFoundException: No Activity found to handle Intent { act=android.intent.action.VIEW dat= flg=0x1000c000 (has extras) }
        at android.app.ActivityThread.handleReceiver(ActivityThread.java:2452)
        at android.app.ActivityThread.access$1700(ActivityThread.java:144)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1281)
        at android.os.Handler.dispatchMessage(Handler.java:102)
        at android.os.Looper.loop(Looper.java:136)
        at android.app.ActivityThread.main(ActivityThread.java:5146)
        at java.lang.reflect.Method.invokeNative(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:515)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:796)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:612)
        at dalvik.system.NativeStart.main(Native Method)
 Caused by: android.content.ActivityNotFoundException: No Activity found to handle Intent { act=android.intent.action.VIEW dat= flg=0x1000c000 (has extras) }
        at android.app.Instrumentation.checkStartActivityResult(Instrumentation.java:1632)
        at android.app.Instrumentation.execStartActivitiesAsUser(Instrumentation.java:1481)
        at android.app.ContextImpl.startActivitiesAsUser(ContextImpl.java:1136)
        at android.content.ContextWrapper.startActivitiesAsUser(ContextWrapper.java:344)
        at android.content.ContextWrapper.startActivitiesAsUser(ContextWrapper.java:344)
        at android.app.TaskStackBuilder.startActivities(TaskStackBuilder.java:221)
        at android.app.TaskStackBuilder.startActivities(TaskStackBuilder.java:232)
        at android.app.TaskStackBuilder.startActivities(TaskStackBuilder.java:208)
        at com.parse.TaskStackBuilderHelper.startActivities(TaskStackBuilderHelper.java:19)
        at com.parse.ParsePushBroadcastReceiver.onPushOpen(ParsePushBroadcastReceiver.java:202)
        at com.parse.ParsePushBroadcastReceiver.onReceive(ParsePushBroadcastReceiver.java:108)
        at android.app.ActivityThread.handleReceiver(ActivityThread.java:2445)
        at android.app.ActivityThread.access$1700(ActivityThread.java:144)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1281)
        at android.os.Handler.dispatchMessage(Handler.java:102)
        at android.os.Looper.loop(Looper.java:136)
        at android.app.ActivityThread.main(ActivityThread.java:5146)
        at java.lang.reflect.Method.invokeNative(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:515)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:796)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:612)
        at dalvik.system.NativeStart.main(Native Method)

回答1:

After spending few hours. Found a solution: Implement your receiver and extends ParsePushBroadcastReceiver class.

Receiver.java

public class Receiver extends ParsePushBroadcastReceiver {

    @Override
    public void onPushOpen(Context context, Intent intent) {
        Log.e("Push", "Clicked");
        Intent i = new Intent(context, HomeActivity.class);
        i.putExtras(intent.getExtras());
        i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(i);
    }
}

Use it in manifest, (Instead of using ParsePushBroadcastReceiver)

Code for project's manifest:

<receiver
            android:name="your.package.name.Receiver"
            android:exported="false" >
            <intent-filter>
                <action android:name="com.parse.push.intent.RECEIVE" />
                <action android:name="com.parse.push.intent.DELETE" />
                <action android:name="com.parse.push.intent.OPEN" />
            </intent-filter>
        </receiver>

Happy Coding!!



回答2:

The cause of this exception is when ParsePushBroadcastReceiver wants to open the empty uri in your push message in this code :

String uriString = null;
    try
    {
      JSONObject pushData = new JSONObject(intent.getStringExtra("com.parse.Data"));
      uriString = pushData.optString("uri");
    }
    catch (JSONException e)
    {
      Parse.logE("com.parse.ParsePushReceiver", "Unexpected JSONException when receiving push data: ", e);
    }
    if (uriString != null) {
          activityIntent = new Intent("android.intent.action.VIEW", Uri.parse(uriString));
        } else {
          activityIntent = new Intent(context, cls);
        }

If you just push a message, the uriString will be empty not null, So context.startActivity(activityIntent); will open an empty uri and exception occurs.

To solve this issue you can subclass ParsePushBroadcastReceiver (thanks to @Ahmad Raza) and override onPushopen like this :

public class Receiver extends ParsePushBroadcastReceiver {

    @Override
    protected void onPushOpen(Context context, Intent intent) {

        ParseAnalytics.trackAppOpenedInBackground(intent);

        String uriString = null;
        try {
            JSONObject pushData = new JSONObject(intent.getStringExtra("com.parse.Data"));
            uriString = pushData.optString("uri");
        } catch (JSONException e) {
            Log.v("com.parse.ParsePushReceiver", "Unexpected JSONException when receiving push data: ", e);
        }
        Class<? extends Activity> cls = getActivity(context, intent);
        Intent activityIntent;
        if (uriString != null && !uriString.isEmpty()) {
            activityIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(uriString));
        } else {
            activityIntent = new Intent(context, MainActivity.class);
        }
        activityIntent.putExtras(intent.getExtras());
        if (Build.VERSION.SDK_INT >= 16) {
            TaskStackBuilder stackBuilder = TaskStackBuilder.create(context);
            stackBuilder.addParentStack(cls);
            stackBuilder.addNextIntent(activityIntent);
            stackBuilder.startActivities();
        } else {
            activityIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            activityIntent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
            context.startActivity(activityIntent);
        }
    }
} 

and update Manifest like this :

<receiver
            android:name="your.package.name.Receiver"
            android:exported="false" >
            <intent-filter>
                <action android:name="com.parse.push.intent.RECEIVE" />
                <action android:name="com.parse.push.intent.DELETE" />
                <action android:name="com.parse.push.intent.OPEN" />
            </intent-filter>
        </receiver>

I hope they solve the problem in their next update.