У меня возникают проблемы при использовании 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 ());
что не то же самое!
Рад получить тонкость сейчас!