20

我正在尝试将 android 应用程序与新的 facebook 3.0 api 集成,但我得到了这个例外:

java.lang.RuntimeException:无法恢复活动 {dk.imu.konnekt/com.facebook.LoginActivity}:com.facebook.FacebookException:无法使用 null 调用包调用 LoginActivity。如果调用者的launchMode 是singleInstance,就会发生这种情况。

我已经搜索过这个错误,但似乎没有其他人对此有任何问题。我想这是因为我为每个选项卡使用了 TabHost 和 TabsGroupActivities。但我不知道如何解决它。

我在这里添加了相关代码:

public class MainTabActivity extends TabActivity {    
    public void onCreate(Bundle savedInstanteState){
        super.onCreate(savedInstanteState);
        setContentView(R.layout.tab_layout);
        TabHost tabHost = getTabHost();

        View shareTab = getLayoutInflater().inflate(R.layout.share_tab, null);
        tabHost.addTab(tabHost.newTabSpec("Share").setIndicator(shareTab)
        .setContent(new Intent(MainTabActivity.this, ShareGroupActivity.class)));

        ...
    }
}

-

public class ShareGroupActivity extends TabsGroupActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        startChildActivity("ShareActivity", new Intent(this, ShareActivity.class));
    }
}

-

public class ShareActivity extends BaseActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.share);

        testFacebookConnection();
    }

    public void testFacebookConnection(){
        Session session = new Session(this);
        Session.setActiveSession(session);
        SessionState state = session.getState();

        Settings.addLoggingBehavior(LoggingBehavior.INCLUDE_ACCESS_TOKENS);

        Session.StatusCallback statusCallback = 
            new Session.StatusCallback() {
            @Override
            public void call(Session session, SessionState state, Exception exception) {
                Toast.makeText(ShareActivity.this, "Facebook session status changed", Toast.LENGTH_SHORT).show(); 
            }
        };

        if (!session.isOpened() && !session.isClosed() && session.getState() != SessionState.OPENING) {
            OpenRequest open = new OpenRequest(this).setCallback(statusCallback);
            List<String> permission = new ArrayList<String>();
            permission.add("publish_actions");
            open.setPermissions(permission);
            session.openForPublish(open);
        } else {
            Session.openActiveSession(this, true, statusCallback);
        }
    }   

    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        Session.getActiveSession().onActivityResult(this, requestCode, resultCode, data);
    }
}

关于如何解决它的任何线索?

更新 1 堆栈跟踪:

致命异常:主要 java.lang.RuntimeException:无法恢复活动 {dk.imu.konnekt/com.facebook.LoginActivity}:com.facebook.FacebookException:无法使用空调用包调用 LoginActivity。如果调用者的launchMode 是singleInstance,就会发生这种情况。在 android.app.ActivityThread.performResumeActivity(ActivityThread.java:2812) 在 android.app.ActivityThread.handleResumeActivity(ActivityThread.java:2851) 在 android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2234) 在 android.app。 ActivityThread.access$600(ActivityThread.java:139) at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1261) at android.os.Handler.dispatchMessage(Handler.java:99) android.os.Looper.loop (Looper.java:154) 在 android.app.ActivityThread.main(ActivityThread.java:4945) 在 java.lang.reflect.Method。在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784) 在 com.android.internal 的 java.lang.reflect.Method.invoke(Method.java:511) 的 invokeNative(Native Method)。 dalvik.system.NativeStart.main(Native Method) 处的 os.ZygoteInit.main(ZygoteInit.java:551) 原因:com.facebook.FacebookException:无法使用空调用包调用 LoginActivity。如果调用者的launchMode 是singleInstance,就会发生这种情况。在 com.facebook.LoginActivity.onResume(LoginActivity.java:110) 在 android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1236) 在 android.app.Activity.performResume(Activity.java:4613) 在 android.app。 ActivityThread.performResumeActivity(ActivityThread.java:2796) ... 12 更多 os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551) at dalvik.system.NativeStart.main(Native Method) 原因:com. facebook.FacebookException:无法使用空调用包调用 LoginActivity。如果调用者的launchMode 是singleInstance,就会发生这种情况。在 com.facebook.LoginActivity.onResume(LoginActivity.java:110) 在 android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1236) 在 android.app.Activity.performResume(Activity.java:4613) 在 android.app。 ActivityThread.performResumeActivity(ActivityThread.java:2796) ... 12 更多 os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551) at dalvik.system.NativeStart.main(Native Method) 原因:com. facebook.FacebookException:无法使用空调用包调用 LoginActivity。如果调用者的launchMode 是singleInstance,就会发生这种情况。在 com.facebook.LoginActivity.onResume(LoginActivity.java:110) 在 android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1236) 在 android.app.Activity.performResume(Activity.java:4613) 在 android.app。 ActivityThread.performResumeActivity(ActivityThread.java:2796) ... 12 更多 FacebookException:无法使用空调用包调用 LoginActivity。如果调用者的launchMode 是singleInstance,就会发生这种情况。在 com.facebook.LoginActivity.onResume(LoginActivity.java:110) 在 android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1236) 在 android.app.Activity.performResume(Activity.java:4613) 在 android.app。 ActivityThread.performResumeActivity(ActivityThread.java:2796) ... 12 更多 FacebookException:无法使用空调用包调用 LoginActivity。如果调用者的launchMode 是singleInstance,就会发生这种情况。在 com.facebook.LoginActivity.onResume(LoginActivity.java:110) 在 android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1236) 在 android.app.Activity.performResume(Activity.java:4613) 在 android.app。 ActivityThread.performResumeActivity(ActivityThread.java:2796) ... 12 更多

更新2: 我查看了代码,找到了startChildActivity的实现:

public void startChildActivity(String Id, Intent intent) {     
   Window window = getLocalActivityManager().startActivity(Id,intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP));
   if (window != null) {
       mIdList.add(Id);
       setContentView(window.getDecorView()); 
   }    
}

它使用标志 FLAG_ACTIVITY_CLEAR_TOP。我试图删除它,但结果没有改变。

更新 3:

https://github.com/facebook/facebook-android-sdk/blob/master/facebook/src/com/facebook/LoginActivity.java

Facebook 代码使用

callingPackage = getCallingPackage();

if (callingPackage == null) {
       throw new FacebookException(NULL_CALLING_PKG_ERROR_MSG);
}

http://developer.android.com/reference/android/app/Activity.html#getCallingPackage()

这个方法有个说明:

如果调用活动不期望结果(即它没有使用包含请求代码的 startActivityForResult(Intent, int) 形式),那么调用包将为空。

在 startChildActivity 方法中,我在扩展 ActivityGroup 的 TabsGroupActivity 中使用 getLocalActivityManager().startActivity 来处理选项卡活动。 http://developer.android.com/reference/android/app/LocalActivityManager.html#startActivity(java.lang.String, android.content.Intent)

此方法与注释中所说的不同。它不期望结果并且不使用 startActivityForResult 方法。该方法还确保类似于单实例启动模式。我应该如何更改此方法的实现,以便它可以与 facebook 一起使用?

4

4 回答 4

7

我设法找到了我的问题。虽然我没有设置

android:launchMode="singleTask"

我的 LoginActivity 有

机器人:noHistory =“真”

这导致了该异常。我将 noHistory 设为 true 是因为我不希望用户在登录后的第一个活动上按下返回按钮并返回登录屏幕。现在我需要找到另一个解决方案。

于 2013-03-28T15:55:33.473 回答
5

经过大量搜索后,我发现似乎没有办法使用选项卡中使用的 LocalActivityManager 来 startActivityForResult 。

所以我最终接受了它需要一个充满整个屏幕的活动。该活动仅在网络连接良好的情况下显示一秒钟左右 - 我还通过重新发布错误选项来实现它..

开始发布活动:

Intent intent = new Intent(this, FacebookShareActivity.class);
intent.putExtra(Constants.FACEBOOK_MESSAGE, shareMessage.getMessage());
startActivityForResult(intent, 1);

Facebook 分享活动代码 - 发布到用户墙:

public class FacebookShareActivity extends Activity {
    String message;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.facebook_publishing);

        message = getIntent().getExtras().getString(Constants.FACEBOOK_MESSAGE);
        createFacebookConnection();
    }

    public void republishButton_Click(View view){
        setVisibilityForRepublishButton(false);
        createFacebookConnection();
    }

    public void createFacebookConnection() {
        Session session = new Session(this);
        Session.setActiveSession(session);

        Settings.addLoggingBehavior(LoggingBehavior.INCLUDE_ACCESS_TOKENS);

        Session.StatusCallback statusCallback = new Session.StatusCallback() {
            @Override
            public void call(Session session, SessionState state, Exception exception) {
                String message = "Facebook session status changed - " + session.getState() + " - Exception: " + exception;
                //Toast.makeText(FacebookShareActivity.this, message, Toast.LENGTH_SHORT).show();
                Log.w("Facebook test", message);

                if (session.isOpened() || session.getPermissions().contains("publish_actions")) {
                    publishToWall();
                } else if (session.isOpened()) {
                    OpenRequest open = new OpenRequest(FacebookShareActivity.this).setCallback(this);
                    List<String> permission = new ArrayList<String>();
                    permission.add("publish_actions");
                    open.setPermissions(permission);
                    Log.w("Facebook test", "Open for publish");
                    session.openForPublish(open);
                }
            }
        };

        if (!session.isOpened() && !session.isClosed() && session.getState() != SessionState.OPENING) {
            session.openForRead(new Session.OpenRequest(this).setCallback(statusCallback));
        } else {
            Log.w("Facebook test", "Open active session");
            Session.openActiveSession(this, true, statusCallback);
        }
    }

    private void setVisibilityForRepublishButton(Boolean visible) {
        ((Button) findViewById(R.id.republishButton)).setVisibility(visible ? View.VISIBLE : View.GONE);
    }

    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        Session.getActiveSession().onActivityResult(this, requestCode, resultCode, data);
        //Toast.makeText(FacebookShareActivity.this, "onActivityResult", Toast.LENGTH_SHORT).show();
    }

    void publishToWall() {
        Session session = Session.getActiveSession();

        Bundle postParams = new Bundle();
        postParams.putString("message", message);

        final Context context = this;
        Request.Callback callback = new Request.Callback() {
            public void onCompleted(Response response) {
                FacebookRequestError error = response.getError();
                if (error != null) {
                    setVisibilityForRepublishButton(true);
                    Toast.makeText(context, error.getErrorMessage(), Toast.LENGTH_SHORT).show();
                } else {
                    JSONObject graphResponse = response.getGraphObject().getInnerJSONObject();
                    String postId = null;
                    try {
                        postId = graphResponse.getString("id");
                    } catch (JSONException e) {
                        setVisibilityForRepublishButton(true);
                        Log.i("Facebook error", "JSON error " + e.getMessage());
                    }
                    //Toast.makeText(context, postId, Toast.LENGTH_LONG).show();
                    finish();
                }
            }
        };

        Request request = new Request(Session.getActiveSession(), "me/feed", postParams, HttpMethod.POST, callback);

        RequestAsyncTask task = new RequestAsyncTask(request);
        task.execute();
    }
}
于 2013-01-12T08:48:17.690 回答
2

我遇到了同样的问题:尝试使用 SDK 内部提供的对话框登录 facebook,但来自一个选项卡组内的活动;就像上面的 ShareActivity。

我所做的基本上是在 ShareActivity 的父活动(即 ShareGroupActivity)上调用 startActivityForResult,而不是在 ShareActivity 上调用它。

所以 1,在 ShareGroupActivity 中添加这个:

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    System.out.println("facebook status called");
    super.onActivityResult(requestCode, resultCode, data);
    Session.getActiveSession().onActivityResult(this, requestCode, resultCode, data);
}

2 你需要修改 Session 类,在 FacebookSDK 项目中,在 src com.facebook 下

2.1 添加一个布尔成员

public boolean insideTabGroup;

2.2 修改StartActivityDelegate,即session用来打开登录;添加布尔值作为参数

interface StartActivityDelegate {
    public void startActivityForResult(Intent intent, int requestCode, boolean insideTabGroup);

    public Activity getActivityContext();
}

2.3 在内部类AuthorizationRequest里面,修改这个delegate的实现:

    AuthorizationRequest(final Activity activity) {
        startActivityDelegate = new StartActivityDelegate() {
            @Override
            public void startActivityForResult(Intent intent, int requestCode, boolean insideTabGroup) {
                if(insideTabGroup) {
                    ActivityGroup parentActivity = (ActivityGroup) activity.getParent();
                    parentActivity.startActivityForResult(intent,requestCode);

                } else {
                    activity.startActivityForResult(intent, requestCode);
                }
            }

            @Override
            public Activity getActivityContext() {
                return activity;
            }
        };
    }

2.4 此外,修改 AuthorizationRequest 的其他构造函数,只需添加布尔参数即可。因为我不使用活动以外的其他地方登录 facebook,所以没关系。

2.5 修改Session类的tryLoginActivity方法,使用布尔成员作为参数:

private boolean tryLoginActivity(AuthorizationRequest request) {
    Intent intent = getLoginActivityIntent(request);

    if (!resolveIntent(intent)) {
        return false;
    }

    try {
        request.getStartActivityDelegate().startActivityForResult(intent, request.getRequestCode(),this.insideTabGroup);

    } catch (ActivityNotFoundException e) {
        return false;
    }

    return true;
}

3 在会话中设置布尔成员:

Session session = Session.getActiveSession();
session.insideTabGroup = true;

这应该够了吧。

于 2013-07-14T14:08:22.870 回答
0

我在 Parse.com Facebook SDK 集成中遇到了同样的错误,我错过了文档ParseFacebookUtils.finishAuthentication中提到的调用。

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  super.onActivityResult(requestCode, resultCode, data);
  ParseFacebookUtils.finishAuthentication(requestCode, resultCode, data);
}
于 2014-07-22T17:00:01.037 回答