DataBinding & LiveData: две реализации (Kotlin и Java) и не могут заставить работать Java - PullRequest
0 голосов
/ 22 сентября 2019

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

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

KOTLIN

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)}».Это работает правильно.

JAVA

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:текст = "@ {noteViewModel.test}".Но текст никогда не отображается и остается пустым.

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

EDIT

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

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);

на

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

, что не одно и то же!

Рад получить эту субтилиту сейчас!

1 Ответ

1 голос
/ 22 сентября 2019

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

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

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

И теперь вы можете получить представление с помощью макета inflater

binding = ActivityMainBinding.inflate(getLayoutInflater());
...