我正在尝试开发一个应用程序使用莫斯比和EventBus。我想要的第一个事件是在用户登录后,创建一个粘性事件,以便每个屏幕都可以随时访问登录信息。
基于Mosby邮件示例,我有一个类似下面的BasePresenter:
public abstract class EventBusPresenter<V extends MvpView> extends MvpBasePresenter<V> {
private static final java.lang.String TAG = tag(EventBusPresenter.class);
@Inject
protected EventBus mEventBus;
LoginSuccessfulEvent userInfo;
@Subscribe(sticky = true, threadMode = ThreadMode.MAIN)
public void onLoginSuccessful(LoginSuccessfulEvent event) {
MCLog.i(TAG, "Received a login event!");
userInfo = event;
onLoginReceived(event);
}
protected abstract void onLoginReceived(LoginSuccessfulEvent e);
public LoginSuccessfulEvent getUserInfo(){
return userInfo;
}
@Override
public void attachView(V view) {
super.attachView(view);
mEventBus.register(this);
}
@Override
public void detachView(boolean retainInstance) {
super.detachView(retainInstance);
mEventBus.unregister(this);
}
}当用户登录时,我使用以下代码:
public void doLogin(String username, String password) {
if (isViewAttached()) {
getView().showLoading();
}
cancelSubscription();
mSubscriber = new Subscriber<AuthenticationResponse>() {
@Override
public void onCompleted() {
if (isViewAttached()) {
getView().loginSuccessful();
}
}
@Override
public void onError(Throwable e) {
if (isViewAttached()) {
getView().showLoginError();
}
}
@Override
public void onNext(AuthenticationResponse authenticationResponse) {
User user = authenticationResponse.getUser();
MCLog.w(TAG, String.format("Login was successful with user: %s", user) );
mEventBus.postSticky(new LoginSuccessfulEvent(user, authenticationResponse.getToken()));
String token = authenticationResponse.getToken();
MCLog.i(TAG, String.format("Token obtained = %s", token));
mSharedPreferences.edit().putString(PreferenceKeys.TOKEN_KEY, token).apply();
}
};我的想法是,对于每个屏幕,一旦加载,它就可以通过EventBus订阅检索UserInfo。
问题是--这个事件发生得太快了。根据mosby自己的BaseFragment类,我这样做:
public abstract class BaseFragment<V extends MvpView, P extends MvpPresenter<V>> extends MvpFragment<V, P> {
private Unbinder mUnbinder;
@LayoutRes
protected abstract int getLayoutRes();
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
return inflater.inflate(getLayoutRes(), container, false);
}
@Override public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
inject();
super.onViewCreated(view, savedInstanceState);
mUnbinder = ButterKnife.bind(this, view);
}
@Override public void onDestroyView() {
super.onDestroyView();
mUnbinder.unbind();
}
/**
* Inject dependencies
*/
protected void inject() {
}
}这意味着注入是在视图创建之前到达的,所以每当我尝试响应接收到的LoginEvent时,需要更新的UI元素就不再存在了。
我如何才能做到这一点?这是正确的做法吗?
发布于 2016-11-22 21:12:43
在onCreateView()中执行mUnbinder = ButterKnife.bind(this, view);,或者如果您更喜欢在调用super.onViewCreated()之前在onViewCreated()调用mUnbinder = ButterKnife.bind(this, view);中执行此操作。
顺便说一句。使用EventBus来实现这一点听起来很痛苦。为什么不简单地创建一个像UserMananger这样的类,在这个类中您可以获得当前经过身份验证的用户,或者如果用户没有经过身份验证,则使用null。此外,由于您使用的是RxJava,因此将UserManager包含在每个屏幕上的RxJava链/流中也是有意义的。
https://stackoverflow.com/questions/40635789
复制相似问题