Jak korzystać z powiązania danych z Fragment

182

Próbuję podążać za przykładem wiązania danych z oficjalnego dokumentu Google https://developer.android.com/tools/data-binding/guide.html

poza tym, że próbuję zastosować przetwarzanie danych do fragmentu, a nie działania.

błąd, który obecnie pojawia się podczas kompilacji

Error:(37, 27) No resource type specified (at 'text' with value '@{marsdata.martianSols}.

onCreate fragment wygląda następująco:

@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    MartianDataBinding binding = MartianDataBinding.inflate(getActivity().getLayoutInflater());
    binding.setMarsdata(this);
}

onCreateView fragment wygląda następująco:

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    return inflater.inflate(R.layout.martian_data, container, false);
}

a części mojego pliku układu dla fragmentu wyglądają tak:

<?xml version="1.0" encoding="utf-8"?>

<layout xmlns:android="http://schemas.android.com/apk/res/android">
    <data>
        <variable
            name="marsdata"
            type="uk.co.darkruby.app.myapp.MarsDataProvider" />
    </data>
...

        <TextView
            android:layout_height="wrap_content"
            android:layout_width="wrap_content"
            android:text="@{marsdata.martianSols}"
        />

    </RelativeLayout>
</layout>

podejrzewam, że MartianDataBindingnie wie, z którym plikiem układu ma być związany - stąd błąd. Jakieś sugestie?

dark_ruby
źródło

Odpowiedzi:

354

Implementacja powiązania danych musi być w onCreateViewmetodzie fragmentu, usuń wszelkie powiązanie danych, które istnieją w Twojej OnCreatemetodzie, onCreateViewpowinieneś wyglądać tak:

public View onCreateView(LayoutInflater inflater, 
                         @Nullable ViewGroup container, 
                         @Nullable Bundle savedInstanceState) {
    MartianDataBinding binding = DataBindingUtil.inflate(
            inflater, R.layout.martian_data, container, false);
    View view = binding.getRoot();
    //here data must be an instance of the class MarsDataProvider
    binding.setMarsdata(data);
    return view;
}
hdioui abdeljalil
źródło
Musiałem dodać wezwanie do super, aby moja klasa Binding generowała.
joey_g216,
3
Walczę z tym problemem przez wiele godzin. Problem polegał na tym, że zwróciłem zły widok. +1
TharakaNirmana
1
View view = binding.getRoot(); Utknąłem w tym tak długo, że jestem uzasadniony dość zdenerwowany, że nie mogłem znaleźć dokumentacji na developer.android.com ... Rozwiązałem problem. Dziękuję Ci!
Victor Ude,
1
Jeśli używasz LiveData i ViewModel, koniecznie przeczytaj tę odpowiedź .
Big McLargeHuge
1
co to jest setMarsdata ()? myślę, że tutaj używamy setViewModel () ??
suv
59

W rzeczywistości zachęca się do korzystania z inflatemetody wygenerowanego powiązania, a nie DataBindingUtil:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    MainFragmentBinding binding = MainFragmentBinding.inflate(inflater, container, false);
    //set variables in Binding
    return binding.getRoot();
}

Dokumenty dla DataBindingUtil.inflate () :

Używaj tej wersji tylko wtedy, gdy layoutId nie jest wcześniej znany. W przeciwnym razie zastosuj metodę generowania wiązania Binding, aby zapewnić bezpieczne napełnianie typu.

Do
źródło
Niestety to mnie zabija z powodu cannot be resolved to a typebłędu kompilacji. Moim zdaniem nie jest to wiarygodne. Jeśli najpierw przejdę do, DataBindingUtil.inflate(inflater, R.layout.fragment_camera, container, false);a następnie zmienię na FragmentCameraBinding.inflate(inflater, container, false);, działa, ale po przebudowaniu ponownie wyświetla błąd.
Alex Burdusel
Działa świetnie. Właściwie nie ma potrzeby określania identyfikatora rozdzielczości układu (co wcześniej się zastanawiałem), ponieważ automatycznie wybiera on z wygenerowanego pliku powiązań.
eC Droid
2
gdzie ustawiasz identyfikator układu fragmentu (np. R.layout.fragment_) w tym przykładzie?
Lenin Raj Rajasekaran
to powinna być zaakceptowana odpowiedź. zachęca się do korzystania z powiązania wygenerowanego przez układ, zamiastDataBindingUtil.inflate
mochadwi,
@LeninRajRajasekaran Identyfikator układu jest implikowany przez użycie MainFragmentBindingklasy. Ta klasa jest generowana z pliku układu, więc żądany układ jest automatycznie stosowany.
Emil S.
19

Nawet inne odpowiedzi mogą działać dobrze, ale chcę powiedzieć najlepsze podejście.

Używaj Binding class's inflatezgodnie z zaleceniami w Dokumentacji Androida .

Jedną z opcji jest nadmuchanie, DataBindingUtil ale gdy tylko nie wiesz, wygenerowałeś klasę wiążącą .

- Masz wygenerowane automatycznie binding class, użyj tej klasy zamiast używać DataBindingUtil.

W Javie

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    HomeFragmentBinding binding = HomeFragmentBinding.inflate(inflater, container, false);
    //set binding variables here
    return binding.getRoot();
}

W Kotlinie

lateinit var binding: HomeFragmentBinding 
override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
    binding = HomeFragmentBinding.inflate(inflater, container, false)
    return binding.root
}

W dokumentacji klasy DataBindingUtil można zobaczyć.

nadmuchać

T inflate (LayoutInflater inflater, 
                int layoutId, 
                ViewGroup parent, 
                boolean attachToParent)

Używaj tej wersji tylko wtedy, gdy layoutId nie jest wcześniej znany. W przeciwnym razie zastosuj metodę generowania wiązania Binding, aby zapewnić bezpieczne napełnianie typu.

Jeśli twoja klasa binowania układu nie jest generowana @ Zobacz tę odpowiedź .

Khemraj
źródło
dlaczego nie skorzystać z inflatemetody, która bierze LayoutInflaterza swój jedyny argument?
Florian Walther
@FlorianWalther Czy to działa bez ViewGroup container?
Khemraj
Cóż, nie wiedziałem, kiedy napisałem ten komentarz. Ale mam tutaj dobrą odpowiedź: stackoverflow.com/questions/61571381/...
Florian Walther
1
@FlorianWalther ok, przeczytałem odpowiedź, która containerjest potrzebna, kiedy attachToRootjest true.
Khemraj
16

Jeśli korzystasz z ViewModel i LiveData Jest to wystarczająca składnia

Składnia Kotlina:

override fun onCreateView(
    inflater: LayoutInflater,
    container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    return MartianDataBinding.inflate(
        inflater,
        container,
        false
    ).apply {
        lifecycleOwner = viewLifecycleOwner
        vm = viewModel    // Attach your view model here
    }.root
}
Saman Sattari
źródło
10

Wypróbuj to w Android DataBinding

FragmentMainBinding binding;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        binding = DataBindingUtil.inflate(inflater, R.layout.fragment_main, container, false);
        View rootView = binding.getRoot();
        initInstances(savedInstanceState);
        return rootView;
}
Jirawat Harnsiriwatanakit
źródło
7

Można po prostu pobrać obiekt widoku, jak wspomniano poniżej

public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

View view = DataBindingUtil.inflate(inflater, R.layout.layout_file, container, false).getRoot();

return view;

}
Imran Solanki - GSLab
źródło
7

Składnia Kotlina:

lateinit var binding: MartianDataBinding
override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
    binding = DataBindingUtil.inflate(inflater, R.layout.martian_data, container, false)
    return binding.root
}
muneikh
źródło
7

Tak jak większość powiedziała, ale nie zapomnij ustawić LifeCycleOwner
Sample w Javie tj

public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {
    super.onCreateView(inflater, container, savedInstanceState);
    BindingClass binding = DataBindingUtil.inflate(inflater, R.layout.fragment_layout, container, false);
    ModelClass model = ViewModelProviders.of(getActivity()).get(ViewModelClass.class);
    binding.setLifecycleOwner(getActivity());
    binding.setViewmodelclass(model);

    //Your codes here

    return binding.getRoot();
}
Leworęczny
źródło
5

pracuje w moim kodzie.

private FragmentSampleBinding dataBiding;
private SampleListAdapter mAdapter;

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    super.onCreateView(inflater, container, savedInstanceState);
    dataBiding = DataBindingUtil.inflate(inflater, R.layout.fragment_sample, null, false);
    return mView = dataBiding.getRoot();
}
UJWAL GHONGADE
źródło
5

Kompletny przykład fragmentów powiązania danych

FragmentMyProgramsBinding to klasa wiązania generowana dla res / layout / fragment_my_programów

public class MyPrograms extends Fragment {
    FragmentMyProgramsBinding fragmentMyProgramsBinding;

    public MyPrograms() {
        // Required empty public constructor
    }


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
    FragmentMyProgramsBinding    fragmentMyProgramsBinding = DataBindingUtil.inflate(inflater, R
                .layout.fragment_my_programs, container, false);
        return fragmentMyProgramsBinding.getRoot();
    }

    @Override
    public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);

    }
}
vivek yadav
źródło
2

Bardzo pomocny blog o wiązaniu danych: https://link.medium.com/HQY2VizKO1

class FragmentBinding<out T : ViewDataBinding>(
    @LayoutRes private val resId: Int
) : ReadOnlyProperty<Fragment, T> {

    private var binding: T? = null

    override operator fun getValue(
        thisRef: Fragment,
        property: KProperty<*>
    ): T = binding ?: createBinding(thisRef).also { binding = it }

    private fun createBinding(
        activity: Fragment
    ): T = DataBindingUtil.inflate(LayoutInflater.from(activity.context),resId,null,true)
}

Zadeklaruj wartość wiązania w następujący sposób we fragmencie:

private val binding by FragmentBinding<FragmentLoginBinding>(R.layout.fragment_login)

Nie zapomnij napisać tego we fragmentach

override fun onCreateView(
    inflater: LayoutInflater,
    container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    return binding.root
}
Dev Soni
źródło
1

Kolejny przykład w Kotlinie:

override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
    val binding = DataBindingUtil
            .inflate< MartianDataBinding >(
                    inflater,
                    R.layout.bla,
                    container,
                    false
            )

    binding.modelName = // ..

    return binding.root
}

Pamiętaj, że nazwa „MartianDataBinding” zależy od nazwy pliku układu. Jeśli plik ma nazwę „martian_data”, poprawna nazwa to MartianDataBinding.

akohout
źródło
0

Wszyscy mówią o inflate(), ale co, jeśli chcemy to wykorzystać onViewCreated()?

Można użyć bind(view)metody konkretnej klasy wiązania, aby uzyskać ViewDataBindinginstancję dla klasy view.


Zwykle piszemy BaseFragment coś takiego (uproszczonego):

// BaseFragment.kt
abstract fun layoutId(): Int

override fun onCreateView(inflater, container, savedInstanceState) = 
    inflater.inflate(layoutId(), container, false)

I użyj go we fragmencie potomnym.

// ConcreteFragment.kt
override fun layoutId() = R.layout.fragment_concrete

override fun onViewCreated(view, savedInstanceState) {
    val binding = FragmentConcreteBinding.bind(view)
    // or
    val binding = DataBindingUtil.bind<FragmentConcreteBinding>(view)
}


Jeśli wszystkie fragmenty używają powiązania danych, możesz nawet uprościć to, używając parametru type.

abstract class BaseFragment<B: ViewDataBinding> : Fragment() {
    abstract fun onViewCreated(binding: B, savedInstanceState: Bundle?)

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        onViewCreated(DataBindingUtil.bind<B>(view)!!, savedInstanceState)
    }
}

Nie wiem, czy można twierdzić, że nie ma tu wartości zerowej, ale ... masz pomysł. Jeśli chcesz mieć wartość zerową, możesz to zrobić.

Tura
źródło