[英]Facebook Android SDK 3.0 always returns OPENING when logging in
我按照https://developers.facebook.com/docs/howtos/androidsdk/3.0/login-with-facebook/上的說明進行操作。 當我觸摸“登錄”按鈕時,會話狀態是打開 - 始終。 當我嘗試將故事發布到用戶的牆上時,我收到錯誤消息
“會話:嘗試為具有待處理請求的會話請求新權限。”
這是我的代碼:
public class MainFragment extends Fragment {
private static final String TAG = "MainFragment";
private UiLifecycleHelper uiHelper;
private Session.StatusCallback callback = new Session.StatusCallback() {
@Override
public void call(Session session, SessionState state,
Exception exception) {
onSessionStateChange(session, state, exception);
}
};
private void onSessionStateChange(Session session, SessionState state,
Exception exception) {
if (state.isOpened()) {
Log.d(TAG, "Logged in...");
} else if (state.isClosed()) {
Log.d(TAG, "Logged out...");
} else {
Log.d(TAG, "Unknown state: " + state);
}
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
uiHelper = new UiLifecycleHelper(getActivity(), callback);
uiHelper.onCreate(savedInstanceState);
}
@Override
public void onResume() {
super.onResume();
// For scenarios where the main activity is launched and user
// session is not null, the session state change notification
// may not be triggered. Trigger it if it's open/closed.
Session session = Session.getActiveSession();
if (session != null && (session.isOpened() || session.isClosed())) {
onSessionStateChange(session, session.getState(), null);
}
uiHelper.onResume();
}
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
uiHelper.onActivityResult(requestCode, resultCode, data);
}
@Override
public void onPause() {
super.onPause();
uiHelper.onPause();
}
@Override
public void onDestroy() {
super.onDestroy();
uiHelper.onDestroy();
}
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
uiHelper.onSaveInstanceState(outState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.facebook_login, container, false);
LoginButton authButton = (LoginButton) view
.findViewById(R.id.authButton);
authButton.setFragment(this);
return view;
}
private boolean isSubsetOf(Collection<String> subset,
Collection<String> superset) {
for (String string : subset) {
if (!superset.contains(string)) {
return false;
}
}
return true;
}
private static final List<String> PERMISSIONS = Arrays
.asList("publish_actions");
private static final String PENDING_PUBLISH_KEY = "pendingPublishReauthorization";
private boolean pendingPublishReauthorization = false;
public void publishStory() {
Session session = Session.getActiveSession();
if (session != null) {
// Check for publish permissions
List<String> permissions = session.getPermissions();
if (!isSubsetOf(PERMISSIONS, permissions)) {
pendingPublishReauthorization = true;
Session.NewPermissionsRequest newPermissionsRequest = new Session.NewPermissionsRequest(
this, PERMISSIONS);
session.requestNewPublishPermissions(newPermissionsRequest);
return;
}
Bundle postParams = new Bundle();
postParams.putString("name", "Facebook SDK for Android");
postParams.putString("caption",
"Build great social apps and get more installs.");
postParams
.putString(
"description",
"The Facebook SDK for Android makes it easier and faster to develop Facebook integrated Android apps.");
postParams.putString("link",
"https://developers.facebook.com/android");
postParams
.putString("picture",
"https://raw.github.com/fbsamples/ios-3.x-howtos/master/Images/iossdk_logo.png");
Request.Callback callback = new Request.Callback() {
public void onCompleted(Response response) {
JSONObject graphResponse = response.getGraphObject()
.getInnerJSONObject();
String postId = null;
try {
postId = graphResponse.getString("id");
} catch (JSONException e) {
Log.i(TAG, "JSON error " + e.getMessage());
}
FacebookRequestError error = response.getError();
if (error != null) {
Toast.makeText(getActivity().getApplicationContext(),
error.getErrorMessage(), Toast.LENGTH_SHORT)
.show();
} else {
Toast.makeText(getActivity().getApplicationContext(),
postId, Toast.LENGTH_LONG).show();
}
}
};
Request request = new Request(session, "me/feed", postParams,
HttpMethod.POST, callback);
RequestAsyncTask task = new RequestAsyncTask(request);
task.execute();
}
}
}
怎么了?
編輯:我還想提一下“登錄”-Button上的文字不會改變。 我在驗證時沒有收到錯誤,但據我所知,它應該更改為“Log Out”。
弄清楚這一點並不容易。 :-)
如果您實際檢查onSessionStateChange()
exception
,您會發現它並非真正停留在OPENING
。 該異常包含: java.lang.UnsupportedOperationException:Session:嘗試打開具有掛起請求的會話。
我最終發現的原因是沒有調用onActivityResult()
。 它有責任回電給uiHelper
並且該呼叫將調用登錄的完成部分,因此如果沒有被調用,狀態將永遠保持OPENING
狀態。 除上述建議的明顯的碎片,這是我發現了什么,他們都需要登錄到工作:
確保同時使用片段和活動,如FB文檔中所述。 即使文檔暗示某項活動足夠,但事實並非如此。 即使活動只有調用單個片段,你也需要兩者。
確保沒有為活動設置歷史記錄(清單和代碼中都沒有)。
確保根據文檔覆蓋片段的onActivityResult()
。 並確保在活動中覆蓋相同的內容:
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
}
是的,這看起來很奇怪但沒有它就行不通。
最后這對我有用。 關鍵點是補充
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
uiHelper.onActivityResult(requestCode, resultCode, data);
}
和
loginBtn.setFragment(this);
loginBtn.setUserInfoChangedCallback(new LoginButton.UserInfoChangedCallback() {
@Override
public void onUserInfoFetched(GraphUser user) {
graphUser = user;
}
});
片段內的代碼:
private GraphUser graphUser;
private UiLifecycleHelper uiHelper;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
uiHelper = new UiLifecycleHelper(getActivity(), callback);
uiHelper.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
return inflater.inflate(R.layout.test_frame, container, false);
}
@Override
public void onViewCreated(View view, Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
view.findViewById(R.id.getMeBtn).setOnClickListener(this);
LoginButton loginBtn = (LoginButton) view.findViewById(R.id.loginBtn);
loginBtn.setFragment(this);
loginBtn.setUserInfoChangedCallback(new LoginButton.UserInfoChangedCallback() {
@Override
public void onUserInfoFetched(GraphUser user) {
graphUser = user;
}
});
}
@Override
public void onResume() {
super.onResume();
uiHelper.onResume();
updateUI();
}
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
uiHelper.onSaveInstanceState(outState);
}
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
uiHelper.onActivityResult(requestCode, resultCode, data);
}
@Override
public void onPause() {
super.onPause();
uiHelper.onPause();
}
@Override
public void onDestroy() {
super.onDestroy();
uiHelper.onDestroy();
}
private Session.StatusCallback callback = new Session.StatusCallback() {
@Override
public void call(Session session, SessionState state, Exception exception) {
onSessionStateChange(session, state, exception);
}
};
private void onSessionStateChange(Session session, SessionState state, Exception exception) {
updateUI();
}
private void updateUI() {
if (graphUser != null) {
Log.d("TEST", graphUser.getFirstName());
}
}
@Override
public void onClick(View view) {
Session session = Session.getActiveSession();
Log.d("TEST", "session.getState() = " + session.getState());
if (session.getState().isOpened()) {
Request.executeMeRequestAsync(session, new Request.GraphUserCallback() {
@Override
public void onCompleted(GraphUser user, Response response) {
if (user != null) {
updateUI();
}
}
});
}
}
我遇到過同樣的問題。 國家總是OPENING
。 事實證明@Gabor是懷疑的。
當會話狀態為OPENING
一個方法onActivityResult
被調用, 這是肯定的。
如果您正在使用片段,則在包含片段的ACTIVITY中調用onActivityResult
,而不是在片段中調用。 那是由於傳遞的背景等。
所以我這樣做了:
檢查onActivityResults
被調用的內容(通過logcat) - 只是為了確保
在該方法中添加一行Session.getActiveSession().onActivityResult(this, requestCode, resultCode, data);
請注意,參數this
是活動上下文。
要么
使用UiLifecycleHelper
的onActivityResult
方法,如下所示: UiHelper.onActivityResult(requestCode, resultCode, data);
更多關於它的信息
如果其他人仍然遇到此問題,可能是由於重寫了Activity中的onActivityResult而沒有調用super方法。
活動
protected void onActivityResult(final int requestCode, final int resultCode, final Intent data) {
/* other code here */
super.onActivityResult(requestCode, resultCode, data);
}
分段
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
uiHelper.onActivityResult(requestCode, resultCode, data);
super.onActivityResult(requestCode, resultCode, data);
}
您應該在不在片段中的活動中實現onActivityResult
。 為我工作。
正在調用我的Activity的onActivityResult
...所以我在那里做了非常糟糕的黑客攻擊:
public void onActivityResult(int requestCode, int resultCode, Intent data) {
if(requestCode == 64206) //Bad hacking
if(df!=null)
df.onActivityResult(requestCode, resultCode, data);
df是我的片段。 這使得會議開放......這太瘋狂了。 非常有興趣為此找到更好的解決方案。
在我的例子中,我試圖將片段類從android.support.v4.app.Fragment
更改為android.app.Fragment
(我不支持API-11之前的Android版本)並省略了FBAuthButton.setFragment(this);
聲明。 這實際上是造成了這個問題。
我遇到了同樣的問題,我卸載了我的應用程序並重新安裝,它工作了!
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.