Поддержка MvpConstraintLayout
Я пытаюсь поддерживать MvpConstraintLayout самостоятельно, просто скопируйте код из MvpLinearLayout
public abstract class MvpConstraintLayout<V extends MvpView, P extends MvpPresenter<V>>
extends ConstraintLayout implements MvpView, ViewGroupDelegateCallback<V, P> {
protected P presenter;
protected ViewGroupMvpDelegate<V, P> mvpDelegate;
public MvpConstraintLayout(Context context) {
super(context);
}
public MvpConstraintLayout(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MvpConstraintLayout(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
/**
* Get the mvp delegate. This is internally used for creating presenter, attaching and detaching
* view from presenter etc.
*
* <p><b>Please note that only one instance of mvp delegate should be used per android.view.View
* instance</b>.
* </p>
*
* <p>
* Only override this method if you really know what you are doing.
* </p>
*
* @return {@link ViewGroupMvpDelegate}
*/
@NonNull
protected ViewGroupMvpDelegate<V, P> getMvpDelegate() {
if (mvpDelegate == null) {
mvpDelegate = new ViewGroupMvpDelegateImpl<>(this, true);
}
return mvpDelegate;
}
@Override protected void onAttachedToWindow() {
super.onAttachedToWindow();
getMvpDelegate().onAttachedToWindow();
}
@Override protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
getMvpDelegate().onDetachedFromWindow();
}
@SuppressLint("MissingSuperCall") @Override protected Parcelable onSaveInstanceState() {
return getMvpDelegate().onSaveInstanceState();
}
@SuppressLint("MissingSuperCall") @Override
protected void onRestoreInstanceState(Parcelable state) {
getMvpDelegate().onRestoreInstanceState(state);
}
/**
* Instantiate a presenter instance
*
* @return The {@link MvpPresenter} for this view
*/
public abstract P createPresenter();
@Override public P getPresenter() {
return presenter;
}
@Override public void setPresenter(P presenter) {
this.presenter = presenter;
}
@Override public V getMvpView() {
return (V) this;
}
@Override public final Parcelable superOnSaveInstanceState() {
return super.onSaveInstanceState();
}
@Override public final void superOnRestoreInstanceState(Parcelable state) {
super.onRestoreInstanceState(state);
}
}
Затем используйте его в коде
public class StatusBarLayout extends MvpConstraintLayout<StatusView, StatusPresenter>
implements StatusView {
public StatusBarLayout(Context context) {
super(context);
init();
}
public StatusBarLayout(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
public StatusBarLayout(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
@Override
protected void onFinishInflate() {
super.onFinishInflate();
ButterKnife.bind(this, this);
}
@Override
public StatusPresenter createPresenter() {
return DaggerStatusBarLayoutComponent.builder().build().presenter();
}
private void init() {
View.inflate(getContext(), R.layout.layout_status, this);
}
}
В предварительном просмотре IDE сказал
java.lang.IllegalStateException: Could not find the surrounding Activity
at com.hannesdorfmann.mosby3.PresenterManager.getActivity(PresenterManager.java:231)
at com.hannesdorfmann.mosby3.mvp.delegate.ViewGroupMvpDelegateImpl.<init>(ViewGroupMvpDelegateImpl.java:70)
at com.skybornecn.ifdive.widgets.MvpConstraintLayout.getMvpDelegate(MvpConstraintLayout.java:55)
at com.skybornecn.ifdive.widgets.MvpConstraintLayout.onAttachedToWindow(MvpConstraintLayout.java:63)
at android.view.View.dispatchAttachedToWindow(View.java:15395)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2953)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2960)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2960)
at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2960)
at android.view.AttachInfo_Accessor.setAttachInfo(AttachInfo_Accessor.java:42)
at com.android.layoutlib.bridge.impl.RenderSessionImpl.inflate(RenderSessionImpl.java:333)
at com.android.layoutlib.bridge.Bridge.createSession(Bridge.java:429)
at com.android.ide.common.rendering.LayoutLibrary.createSession(LayoutLibrary.java:368)
at com.android.tools.idea.rendering.RenderTask$2.compute(RenderTask.java:567)
at com.android.tools.idea.rendering.RenderTask$2.compute(RenderTask.java:549)
at com.intellij.openapi.application.impl.ApplicationImpl.runReadAction(ApplicationImpl.java:863)
at com.android.tools.idea.rendering.RenderTask.createRenderSession(RenderTask.java:549)
at com.android.tools.idea.rendering.RenderTask.lambda$inflate$1(RenderTask.java:680)
at java.util.concurrent.FutureTask.run(FutureTask.java:266)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
at java.lang.Thread.run(Thread.java:745)
Я прочитал Mosby PresenterManager, но до сих пор не знаю, почему. Кроме того, установить на телефон, раскладка отображается просто ок.
1 ответ
ОБНОВЛЕНИЕ: это исправлено в Mosby 3.0.1.
Это просто проблема андроид-инструментов с Мосби. Редактор макета Android на самом деле создает экземпляр MvpConstraintLayout
для предварительного просмотра в Android Studio. Тем не менее, Android Studio не делает этого, прикрепляя вид как MvpContraintLayout
к реальной деятельности, но внутренне использует какую-то насмешливую "контекстную" вещь. PresenterManager Мосби внутренне пытается найти активность для вашего MvpConstraintLayout
, но в Android Studio нет реальной активности, используемой (как уже объяснено), и, следовательно, редактор пользовательского интерфейса отображает это сообщение об ошибке, и при установке на телефоне оно работает должным образом (поскольку используется реальная активность).
Чтобы исправить это, вы должны использовать view.isInEditMode()
перед вызовом делегата вот так:
public abstract class MvpConstraintLayout<V extends MvpView, P extends MvpPresenter<V>>
extends ConstraintLayout implements MvpView, ViewGroupDelegateCallback<V, P> {
...
@Override protected void onAttachedToWindow() {
super.onAttachedToWindow();
if (!isInEditMode()) getMvpDelegate().onAttachedToWindow();
}
@Override protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
if (!isInEditMode()) getMvpDelegate().onDetachedFromWindow();
}
@SuppressLint("MissingSuperCall") @Override protected Parcelable onSaveInstanceState() {
if (!isInEditMode()) return getMvpDelegate().onSaveInstanceState();
return super.onSaveInstanceState();
}
@SuppressLint("MissingSuperCall") @Override
protected void onRestoreInstanceState(Parcelable state) {
if (!isInEditMode()) getMvpDelegate().onRestoreInstanceState(state);
}
/**
* Instantiate a presenter instance
*
* @return The {@link MvpPresenter} for this view
*/
public abstract P createPresenter();
@Override public P getPresenter() {
return presenter;
}
@Override public void setPresenter(P presenter) {
this.presenter = presenter;
}
@Override public V getMvpView() {
return (V) this;
}
@Override public final Parcelable superOnSaveInstanceState() {
return super.onSaveInstanceState();
}
@Override public final void superOnRestoreInstanceState(Parcelable state) {
super.onRestoreInstanceState(state);
}
}
Это на самом деле что-то ViewGroupMvpDelegate
следует позаботиться о себе. Это должно быть исправлено с помощью Mosby 3.0.1.