我正在使用数据绑定来绑定我的Android应用中的布局。
我已经设置了我的布局(my_custom.xml)并生成了绑定类(MyCustomBinding),但是Android Studio似乎没有立即找到Binding类的.inflate(...)方法,标记它作为错误(红色文字!)。
代码似乎是正确的,因为它可以编译并构建到APK中。
如何让Android Studio正确更新?
这是我自定义的查看代码:
public class MyCustomView extends FrameLayout {
public MyCustomView(Context context) {
this(context, null, 0);
}
public MyCustomView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public MyCustomView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
MyCustomBinding binding = MyCustomBinding.inflate(inflater, this, true);
binding.aButton.setText("Whatever");
}
}
<?xml version="1.0" encoding="utf-8"?>
<layout
xmlns:android="http://schemas.android.com/apk/res/android">
<data>
</data>
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
>
<TextView
android:id="@+id/a_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Click me!"
android:padding="10dp"
android:background="#000"
android:textColor="#fff"
android:layout_gravity="center"
/>
</FrameLayout>
</layout>
答案 0 :(得分:9)
Android Studio中的某些内容尚未完成,因为您尚未实际实施数据绑定。
将变量添加到布局的data
元素后,可以按预期找到inflate
方法。也就是说,通过绑定直接设置文本字段的值,您实际上无法获得数据绑定的好处。您应该在绑定中设置视图模型,然后让绑定相应地更新视图。例如:
创建一个视图模型:
public class MyViewModel {
public final ObservableField<String> name;
public MyViewModel(String name) {
this.name = new ObservableField<>(name);
}
}
并在您的布局中使用它
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
<variable name="model" type="com.victoriaschocolates.conceirge.MyViewModel" />
</data>
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{model.name}"
android:padding="10dp"
android:background="#000"
android:textColor="#fff"
android:layout_gravity="center"
/>
</FrameLayout>
</layout>
(请注意variable
元素中声明的data
,以及TextView&#39; text
属性中的引用方式
然后在自定义视图中绑定两个:
public class MyCustomView extends FrameLayout {
public MyCustomView(Context context) {
this(context, null, 0);
}
public MyCustomView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public MyCustomView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
MyCustomBinding binding = MyCustomBinding.inflate(inflater, this, true);
MyViewModel model = new MyViewModel("Whatever");
binding.setModel(model);
}
}
当然,通过自定义视图类中的setter传入数据或者甚至从容器视图传入数据可能会更好(参见http://developer.android.com/tools/data-binding/guide.html#includes)
答案 1 :(得分:3)
您可以尝试为自定义视图设置自定义类名称,并在自定义视图中引用它,以明确您使用的布局:
<layout>
<data class="MyCustomBinding">
</data>
</layout>
如果这不起作用,请使用DataBindingUtil
而不是MyCustomBinding
,它返回基础DataBinding类并使用inflate()
方法:
MyCustomBinding binding = DataBindingUtil.inflate(inflater, this, true);
来自docs:
有时无法事先知道绑定。在这种情况下,可以使用DataBindingUtil类创建绑定:
ViewDataBinding binding = DataBindingUtil.inflate(LayoutInflater, layoutId,
parent, attachToParent);
ViewDataBinding binding = DataBindingUtil.bindTo(viewRoot, layoutId);
答案 2 :(得分:0)
你不需要那种类型的东西,DataBinding在这里包含了DataBindingUtils类。
public MyCustomView(Context context, AttributeSet attrs, int defStyle){
super(context, attrs, defStyle);
LayoutInflater inflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
MyCustomBinding binding = DataBindingUtil.inflate(inflater, this, true);
binding.aButton.setText("Whatever");
}
答案 3 :(得分:0)
这是基于最新数据绑定和androidx的更新答案,在观看了以上答案后,我来这里解决我自己的问题,我是用自己的工作代码段开发的。希望对您有帮助
public class CustomActionBar1 extends RelativeLayout {
public MutableLiveData<String> title = new MutableLiveData<>("Sample text");
CustomActionBar1Binding binding;
public CustomActionBar1(Context context) {
super(context);
init(context, this);
}
public CustomActionBar1(Context context, AttributeSet attrs) {
super(context, attrs);
init(context, this);
}
public CustomActionBar1(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init(context, this);
}
public CustomActionBar1(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
init(context, this);
}
private void init(Context context, ViewGroup viewGroup) {
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
binding = DataBindingUtil.inflate(inflater, R.layout.custom_action_bar_1, viewGroup, true);
}
// helper to change title
public void changeTitle(String title) {
if (title != null)
this.title.setValue(title);
}
public void setTitleVisibility(Boolean visibile){
binding.titleText.setVisibility(visibile ? View.VISIBLE : View.INVISIBLE);
}
}
Xml
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="customActionBar1"
type="com.actionBar.CustomActionBar1" />
</data>
<RelativeLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:background="@color/yellow"
android:layout_height="@dimen/action_bar_height"
android:id="@+id/main_header_relative">
<TextView
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:text="@{customActionBar1.title}"
android:layout_centerInParent="true"
android:gravity="center"
android:id="@+id/titleText"
android:visibility="visible"
android:textColor="@color/black" />
</RelativeLayout>
</layout>