5

我的库模块中有一个抽象的 BaseActivity:

 @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //Normal method to inflate the layout
        mBinding = DataBindingUtil.setContentView(this, R.layout.base_view_stub_layout);
        AppCompatDelegate.setCompatVectorFromResourcesEnabled(true);
        if (mBinding != null && mBinding.viewStub != null && mBinding.viewStub.getViewStub() != null) {
            mBinding.viewStub.getViewStub().setLayoutResource(getLayoutId());
        }

        if (!hasStubInflated) {
            View inflatedView = mBinding.viewStub.getViewStub().inflate();
            //Abstract method
            onViewStubInflated(inflatedView, savedInstanceState);
            //Normal method to hide progress bar
            onViewStubInflated();
        }
        initControllers();
        handleViews();
        setListeners();
        restoreValues(savedInstanceState);
    }

AppModule 中的 MainActivity 扩展了库模块的 BaseActivity,因此获得了一个抽象方法来覆盖:

MainActivity 扩展 BaseActivity

private ActivityMainBinding mBinding;

@Override
    public int getLayoutId() {
        return R.layout.activity_main;
    }

@Override
    public void onViewStubInflated(View inflatedView, Bundle savedInstanceState) {
        if (mBinding == null || mBinding.getRoot() != inflatedView) {
            mBinding = ActivityMainBinding.bind(inflatedView);
        }
    }

当我运行该应用程序时,我得到:

 Caused by: java.lang.RuntimeException: view must have a tag
        at com.example.android.emailapp.DataBinderMapperImpl.getDataBinder(DataBinderMapperImpl.java:67)
        at androidx.databinding.MergedDataBinderMapper.getDataBinder(MergedDataBinderMapper.java:74)
        at androidx.databinding.DataBindingUtil.bind(DataBindingUtil.java:199)
        at androidx.databinding.ViewDataBinding.bind(ViewDataBinding.java:693)
        at com.example.android.emailapp.databinding.ActivityMainBinding.bind(ActivityMainBinding.java:99)
        at com.example.android.emailapp.databinding.ActivityMainBinding.bind(ActivityMainBinding.java:87)
        at com.example.android.emailapp.login.MainActivity.onViewStubInflated(MainActivity.java:86)
        at com.library.android.common.ui.baseui.BaseActivity.onCreate(BaseActivity.java:52)

请注意,我在 BaseActivity 和 MainActivity 的 xml 中都有布局标记。

我试过这个没有成功。

mBinding.viewStub.getViewStub ().setLayoutResource(getLayoutId())是 BaseActivity 中的一个问题,因为它会尝试从 AppModule 获取布局?如果是这样,我该如何解决?

如果您知道答案,请尝试解释为什么会发生这种情况以及解决方案...

已经试过了

视图必须有标签错误

view must have a tag error是什么意思

如何为viewstub使用数据绑定

4

2 回答 2

0

您还必须包装 viewStub 的布局<layout></layout>,这是他们所期望的:要么在任何地方使用 DataBinding,要么在任何地方都没有。


如果您不想在 viewStub 上使用 DataBinding,则需要完成额外的工作。

例如,如果itemData是我的 listview 的 DataBindingImpl 对象,并且我要膨胀某种more options叠加 ViewStub,那么我需要自己存储膨胀的视图,并避免访问 DataBinding 的 ViewStubProxy 的 `mRoot 字段

ViewGroup inflated = (ViewGroup) itemData.someView.getTag();
if(inflated==null) {
    ViewStub stub = itemData.viewStubProxy.getViewStub();
    if(stub!=null) {
        stub.setOnInflateListener(null);
        inflated = (ViewGroup) stub.inflate();
        // what I need is simply to set onClickListener on all of it's children. (one for-loop inside)
        Utils.setOnClickListenersOneDepth(inflated, onClickListener, 3, null);
        itemData.someView.setTag(inflated);
    }
}
if(inflated!=null) {
    // what I need then is to animate the overlay. 
    View mRoot = inflated;
    int width = itemData.getRoot().getWidth();
    mRoot.setAlpha(0);
    mRoot.setTranslationX(width/3);
    mRoot.animate().alpha(1);
    mRoot.animate().translationX(0);
}

永远不要调用itemData.viewStubProxy.getViewStub().getRoot()、<code>.isInflated() 左右,因为我们现在存储的inflated是 DataBindingImpl 对象的外部。

于 2021-02-17T07:14:32.687 回答
0

如果您使用自定义类作为根视图,your_layout.xml那么您必须覆盖视图的所有构造函数

你的班

class MyCustomView : FrameLayout {
    constructor(context: Context) : super(context){
        ...
    }

    constructor(context: Context, attrs: AttributeSet?) : super(context, attrs){
        ...
    }
    constructor(context: Context, attrs: AttributeSet?, defStyleAttr: Int) : super(context, attrs, defStyleAttr){
        ...
    }
}

你的布局

<layout xmlns:android="http://schemas.android.com/apk/res/android">
    <data>
        <variable name="viewModel" type="......" />
    </data>

    <MyCustomView
        android:layout_width="match_parent"
        android:layout_height="match_parent">

    </MyCustomView>

</layout>
于 2020-04-02T10:02:51.723 回答