在Android数据绑定更新后,Edittext游标重置为左侧

时间:2015-08-31 23:33:05

标签: android data-binding 2-way-object-databinding

我正在尝试新的Android数据绑定库(1.0-rc1),我创建了一个包含三个字符串字段(名称,电子邮件和年龄)的User对象,并将它们链接到3个EditTexts在我的布局中。

在第一个字段(名称)上,我放置了 TextWatcher 。一切似乎都运作良好。我通过在允许它调用setName之前检查文本是否不同来阻止名称字段中的 notifyPropertyChanged 循环。

问题是,每次输入名称字段时,光标会在每个字符后重置为EditText的左侧。我搜索了一个解决方案,但大多数光标问题的修复建议都说要抓取对EditText的引用并手动调整光标位置。但是我想避免这样做,因为我需要在EditText中找到ViewByID,而Data Binding的目的是试图避免这样做。谢谢你的帮助。

我的布局如下:

<layout>

<data>
    <variable name="user" type="com.carlpoole.databindingstest.User"/>
</data>

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:bind="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
    android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity">

    <EditText
        android:layout_width="200dp"
        android:layout_height="wrap_content"
        android:id="@+id/name"
        android:text="@{user.name}"
        bind:addTextChangedListener="@{user.nameChanged}"
        />

    <EditText
        android:layout_width="200dp"
        android:layout_height="wrap_content"
        android:id="@+id/email"
        android:layout_below="@+id/name"
        android:text="@{user.email}"/>

    <EditText
        android:layout_width="200dp"
        android:layout_height="wrap_content"
        android:id="@+id/age"
        android:layout_below="@+id/email"
        android:text="@{user.age}"/>

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_below="@+id/age"
        android:text="@{user.name}"/>

</RelativeLayout>

我的用户对象如下所示:

import android.databinding.BaseObservable;
import android.databinding.Bindable;
import android.text.Editable;
import android.text.TextWatcher;

public class User extends BaseObservable {

    private String name;
    private String email;
    private String age;

    public User(String name, String email, String age) {
        this.name = name;
        this.email = email;
        this.age = age;
    }

    public User(){};

    @Bindable
    public String getName() {
        return name;
    }

    @Bindable
    public String getEmail() {
        return email;
    }

    @Bindable
    public String getAge() {
        return age;
    }

    public final TextWatcher nameChanged = new TextWatcher() {
        @Override
        public void afterTextChanged(Editable s) {
            if(!s.toString().equalsIgnoreCase(name))
                setName(s.toString());
        }

        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {}

        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {}
    };

    public void setName(String name) {
        this.name = name;
        notifyPropertyChanged(com.carlpoole.databindingstest.BR.name);
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public void setAge(String age) {
        this.age = age;
    }
}

我的活动看起来像这样

import android.databinding.DataBindingUtil;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import com.carlpoole.databindingstest.databinding.ActivityMainBinding;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
        User user = new User("Carl Poole", "mail@carlpoole.com", "26");
        binding.setUser(user);

    }
}

5 个答案:

答案 0 :(得分:11)

这里最好的选择是使用自定义@BindingAdapter,它已经引用了EditText。这样,如果EditText中的文字与您的模型匹配,则可以避免重新绑定,这将解决您的光标问题。

首先,将android:text="@{user.name}"更改为bind:binding="@{user.name}"。然后,在项目的任何位置添加此静态方法。我们将所有这些都保存在一个名为BindingAdapters.java的类中。顺便说一句,从RC2开始,你可以创建非静态绑定适配器方法,但这可能不是你现在最关心的问题。

@BindingAdapter("binding")
public static void bindEditText(EditText editText, CharSequence value) {
  if (!editText.getText().toString().equals(value.toString())) {
    editText.setText(value);
  }
}

答案 1 :(得分:7)

要修复将光标重置为EditText开头的奇怪数据绑定行为,可以添加以下InverseBindingAdapter:

  @InverseBindingAdapter(attribute = "android:text") 
   public static String getText(EditText view) {   
        String text = view.getText().toString();
        view.setSelection(text.length());
        return text;
    }

答案 2 :(得分:3)

问题在于setter混淆:根据DataBinding文档的建议,你的setter调用notifyPropertyChanged方法。但是,notifyPropertyChanged方法除其他外,重置光标,这是导致问题的原因。当它是调用setter的UI(TextWatcher)时,他们不需要setter更新UI。解决方案是让setter仅在某些后端计算/操作应该导致UI更新时调用notifyPropertyChanged方法。

答案 3 :(得分:1)

尝试一下:

@BindingAdapter("android:text")
fun setStringWIthSelection(view: EditText, str : String) {
    view.setText(str)
    view.setSelection(view.text.length)
}

答案 4 :(得分:0)

其他朋友的回答很好,但是当我按照这些方法进行操作时仍然会遇到一些错误。当我在 EditText 视图中间编辑文本时,光标将移动到末尾,而不是在编辑位置。我为该问题添加了一个修复程序:

        int selection = mEditText.getSelectionEnd();
        int updateTextLength = text == null ? 0 : text.length();
        mEditText.setText(text);
        mEditText.setSelection(Math.min(selection, updateTextLength));