DataBinding и LiveData: две реализации (Kotlin и Java) и не могут заставить Java работать

У меня возникают проблемы при использовании DataBinding и LiveData в проекте Java. Я прошел предыдущий курс по Kotlin, и когда я пытаюсь реализовать то же поведение, я просто не могу заставить его работать. Мне явно чего-то не хватает в плане понимания, поэтому я хотел бы высказать свои мысли.

Я вставлю код из Kotlin (рабочий) пример, а затем из Java (не рабочий).

КОТЛИН

score_fragment.xml

...
<data>

    <variable
        name="scoreViewModel"
        type="com.example.android.guesstheword.screens.score.ScoreViewModel" />
</data>

<androidx.constraintlayout.widget.ConstraintLayout
    android:id="@+id/score_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".screens.score.ScoreFragment">

    <TextView
        android:id="@+id/score_text"
        ...
        android:text="@{String.valueOf(scoreViewModel.score)}"
        .../>
...

ScoreFragment.kt

class ScoreFragment : Fragment() {

private lateinit var viewModelFactory: ScoreViewModelFactory
private lateinit var viewModel: ScoreViewModel

override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
): View? {

        // Inflate view and obtain an instance of the binding class.
        val binding: ScoreFragmentBinding = DataBindingUtil.inflate(
                inflater,
                R.layout.score_fragment,
                container,
                false
        )

        // Get args using by navArgs property delegate
        val scoreFragmentArgs by navArgs<ScoreFragmentArgs>()

        viewModelFactory = ScoreViewModelFactory(scoreFragmentArgs.score)
        viewModel = ViewModelProviders.of(this, viewModelFactory)
                .get(ScoreViewModel::class.java)

        binding.scoreViewModel = viewModel
        binding.lifecycleOwner = this

        return binding.root
    }
}

ScoreViewModel.kt

class ScoreViewModel(finalScore: Int) : ViewModel() {

    private val _score = MutableLiveData<Int>()
    val score: LiveData<Int>
        get() = _score

    private val _eventPlayAgain = MutableLiveData<Boolean>()
    val eventPlayAgain: LiveData<Boolean>
        get() = _eventPlayAgain

    init {
        Timber.i("ScoreViewModel created")
        _score.value = finalScore
    }

    fun onPlayAgain() {
        _eventPlayAgain.value = true
    }

    fun onPlayAgainComplete() {
        _eventPlayAgain.value = false
    }

    override fun onCleared() {
        super.onCleared()
        Timber.i("ScoreViewModel cleared")
    }
}

Пояснения: давайте сосредоточимся только на значении оценки. В ScoreViewModel значение имеет тип LiveData. Когда фрагмент запущен, значение правильно отображается на экране через "@ {String.valueOf (scoreViewModel.score)}". Это работает правильно.

ДЖАВА

activity_main.xml

<data>
    <variable
        name="noteViewModel"
        type="com.example.architectureapp.viewModel.NoteViewModel" />
</data>

<androidx.constraintlayout.widget.ConstraintLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/textview"
        android:text="@{noteViewModel.test}" />

</androidx.constraintlayout.widget.ConstraintLayout>

MainActivity

public class MainActivity extends AppCompatActivity {
    private ActivityMainBinding binding;
    private NoteViewModel mNoteViewModel;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        binding = ActivityMainBinding.inflate(getLayoutInflater());
        mNoteViewModel = ViewModelProviders.of(this).get(NoteViewModel.class);
        binding.setNoteViewModel(mNoteViewModel);
        binding.setLifecycleOwner(this);
    }
}

NoteViewModel

public class NoteViewModel extends AndroidViewModel {
    public MutableLiveData<String> test = new MutableLiveData<>("TesT");

    public NoteViewModel(@NonNull Application application) {
        super(application);
    }
}

Пояснения: здесь я устанавливаю тест MutableLiveData со значением "TesT", а затем я намерен отобразить его с помощью android: text = "@ {noteViewModel.test}". Но текст никогда не отображается и остается пустым.

Очевидно, что-то не так, но, несмотря на синтаксические различия между двумя реализациями, я просто не могу понять, почему версия Java не отображает значение в Textview.

РЕДАКТИРОВАТЬ

Благодаря Раджнишу сурьяванши я не получал правильную привязку, мне пришлось использовать только:

binding = DataBindingUtil.setContentView (this, R.layout.activity_main);

Он устанавливает представление содержимого с предоставленным макетом И возвращает привязку. В качестве альтернативы вы можете сделать:

binding = ActivityMainBinding.inflate (getLayoutInflater ()); (возвращает привязку, но не устанавливает представление содержимого)

setContentView (binding.getRoot ()); (установить представление содержимого с корневым представлением привязки)

Я нашел это вводящим в заблуждение -> https://developer.android.com/topic/libraries/data-binding/expressions

В нем говорится, что мы можем заменить

Привязка ActivityMainBinding = DataBindingUtil.setContentView (this, R.layout.activity_main);

by

Привязка ActivityMainBinding = ActivityMainBinding.inflate (getLayoutInflater ());

что не то же самое!

Рад получить тонкость сейчас!


person Gilles    schedule 22.09.2019    source источник
comment
добавить метод получения для тестирования изменяемых данных в реальном времени.   -  person Rajnish suryavanshi    schedule 22.09.2019
comment
используйте android: text = @ {safeUnbox (viewModel. test)}   -  person Zafer Celaloglu    schedule 22.09.2019
comment
@Rajnishsuryavanshi, я пробовал, но безуспешно. Между прочим, и для целей примера test объявлен общедоступным, поэтому в геттере нет необходимости.   -  person Gilles    schedule 22.09.2019
comment
@ZaferCelaloglu, я не могу найти метод safeUnbox (java.lang.String) в классе androidx.databinding.ViewDataBinding. Я где-то читал, что это может произойти при использовании двусторонней привязки, но здесь это не так.   -  person Gilles    schedule 22.09.2019


Ответы (1)


Взгляните на эти две строки.

setContentView(R.layout.activity_main);
binding = ActivityMainBinding.inflate(getLayoutInflater());

Вы не создаете привязку при раздутии макета. Вместо setContentView используйте DataBindingUtil.setContentView(this, R.layout.activity_main);

И теперь вы можете получить представление, используя надуватель макета

binding = ActivityMainBinding.inflate(getLayoutInflater());
person Rajnish suryavanshi    schedule 22.09.2019
comment
Хм, на самом деле это не работает, но ты поставил меня на правильный путь. Я не понимаю, что на самом деле делает надувание. Выполняется: DataBindingUtil.setContentView (this, R.layout.activity_main); binding = ActivityMainBinding.inflate (getLayoutInflater ()); не работает, но это: binding = ActivityMainBinding.inflate (getLayoutInflater ()); setContentView (binding.getRoot ()); и это: binding = DataBindingUtil.setContentView (this, R.layout.activity_main); работа. Это вводило в заблуждение - ›developer.android.com/topic/libraries/data-binding / выражения - person Gilles; 22.09.2019