ViewBinding - jak uzyskać powiązanie dla dołączonych układów?

11

Podczas pracy z ViewBinding natrafiam na kilka nieudokumentowanych przypadków.

Po pierwsze: jak uzyskać powiązanie dla dołączonych części układu widoku ogólnego, wiązanie główne widzi tylko elementy w układzie głównym?

Po drugie: jak uzyskać powiązanie dla zawartych części układu typu scalania, ponownie główne wiązanie widzi tylko elementy w głównym układzie?

Artur Kasprzak
źródło

Odpowiedzi:

14

W przypadku:

  1. Dołącz z ogólnym układem (nie scalaj węzła), musimy przypisać identyfikator do dołączonej części, w ten sposób w powiązaniu uzyskamy dostęp do dołączonej części
<include
    android:id="@+id/your_id"
    layout="@layout/some_layout" />

W ten sposób w kodzie aktywności:

private lateinit var exampleBinding: ActivityExampleBinding  //activity_example.xml layout

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    exampleBinding = ActivityExampleBinding.inflate(layoutInflater)
    setContentView(exampleBinding.root)
    //we will be able to access included layouts view like this
    val includedView: View = exampleBinding.yourId.idOfIncludedView
//[...]
}
  1. Uwzględnij z blokiem scalania w układzie zewnętrznym. Nie możemy dodać do niego identyfikatora, ponieważ blok scalania nie jest widokiem. Powiedzmy, że mamy taki wieczny układ scalania (merge_layout.xm):
<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    tools:showIn="@layout/activity_example">

    <TextView
        android:id="@+id/some_view"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World" />
</merge>

Aby poprawnie powiązać taki układ scalania, musimy:

W twoim kodzie aktywności:

private lateinit var exampleBinding: ActivityExampleBinding  //activity_example.xml layout
private lateinit var mergeBinding: MergeLayoutBinding  //merge_layout.xml layout

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    exampleBinding = ActivityExampleBinding.inflate(layoutInflater)
    //we need to bind the root layout with our binder for external layout
    mergeBinding = MergeLayoutBinding.bind(exampleBinding.root)
    setContentView(exampleBinding.root)
    //we will be able to access included in merge layout views like this
    val mergedView: View = mergeBinding.someView
//[...]
}
Artur Kasprzak
źródło
1
To wygląda na błąd. To powinno po prostu działać .
Miguel
7

Twoje pierwsze pytanie, które działa z dołączonym układem za pomocą ViewBinding, można tak łatwo rozwiązać.

Oto przykładowy plik main_fragment.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/recycler_view_main"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1" />

    <include
        android:id="@+id/toolbar"
        layout="@layout/toolbar" />

</LinearLayout>

A MainFragment.java może być tak

public class MeaningFragment extends Fragment {

    private MainFragmentBinding binding;
    private ToolbarBinding toolbarBinding;

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

        binding = MainFragmentBinding.inflate(inflater, container, false);
        toolbarBinding = binding.toolbar;

        return binding.getRoot();
    }

    @Override
    public void onDestroy() {
        super.onDestroy();

        toolbarBinding = null;
        binding = null;
    }
}

Teraz masz dwa wiązania. jeden z nich jest domyślny, a następny pochodzi z dołączonego układu.

Emi Raz
źródło
1
Bardzo prosta odpowiedź i wykorzystuje nową składnię - wszystko działa dla mnie w działaniu nie fragmentarycznym o podobnej składni onCreate(). Dzięki. (Mam tylko trochę problemów z używaniem dla a DrawerLayout)
Fat Monk
0

Innym prostym sposobem byłoby użycie biblioteki powiązania danych. Następnie zawiń układ XML tagiem, aby w przypadku korzystania z biblioteki automatycznie generowało klasy wymagane do powiązania widoków w układzie z obiektami danych. Szczerze mówiąc, myślę, że to dobra droga. Postępuj zgodnie z instrukcjami tutaj

Arun Gurung
źródło