Android - динамически добавлять представления в представление - PullRequest
139 голосов
/ 02 июня 2011

У меня есть макет для представления -

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:padding="0px"
    android:orientation="vertical">

    <TextView
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:id="@+id/items_header"
        style="@style/Home.ListHeader" />

    <TextView 
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:id="@+id/items_none"
        android:visibility="gone"
        style="@style/TextBlock"
        android:paddingLeft="6px" />

    <ListView 
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:id="@+id/items_list" />


</LinearLayout>

То, что я хочу сделать, это в моей основной деятельности с макетом, подобным этому

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:padding="0px"
    android:id="@+id/item_wrapper">
</LinearLayout>

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

Ответы [ 5 ]

215 голосов
/ 02 июня 2011

Используйте LayoutInflater для создания представления на основе шаблона макета, а затем вставьте его в представление, где оно вам необходимо.

LayoutInflater vi = (LayoutInflater) getApplicationContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View v = vi.inflate(R.layout.your_layout, null);

// fill in any details dynamically here
TextView textView = (TextView) v.findViewById(R.id.a_text_view);
textView.setText("your text");

// insert into main view
ViewGroup insertPoint = (ViewGroup) findViewById(R.id.insert_point);
insertPoint.addView(v, 0, new ViewGroup.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT, ViewGroup.LayoutParams.FILL_PARENT));

Возможно, вам придется настроить индекс, в который вы хотите вставить представление.

Кроме того, установите LayoutParams в соответствии с тем, как вы хотите, чтобы он вписывался в родительский вид. например с FILL_PARENT или MATCH_PARENT и т. д.

34 голосов
/ 02 июня 2011

См. LayoutInflater класс.

LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
ViewGroup parent = (ViewGroup)findViewById(R.id.where_you_want_to_insert);
inflater.inflate(R.layout.the_child_view, parent);
19 голосов
/ 02 июня 2011

Похоже, что вы действительно хотите, чтобы ListView с пользовательским адаптером раздувал указанную разметку.Используя ArrayAdapter и метод notifyDataSetChanged(), вы получаете полный контроль над генерацией и рендерингом представлений.

Посмотрите эти учебные пособия

9 голосов
/ 08 августа 2017

Чтобы сделать ответ @Mark Fisher более понятным, вставляемый вид должен быть раздувным xml-файлом в папке макета, но без макета (ViewGroup), такого как LinearLayout и т.д. внутри. Мой пример:

res / layout / my_view.xml

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/i_am_id"
    android:text="my name"
    android:textSize="17sp"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_weight="1"/>

Тогда точка вставки должна быть такой, как LinearLayout:

res / layout / activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/aaa"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <LinearLayout
        android:id="@+id/insert_point"
        android:layout_width="match_parent"
        android:layout_height="match_parent">

    </LinearLayout>

</RelativeLayout>

Тогда код должен быть

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

    LayoutInflater inflater = getLayoutInflater();
    View view = inflater.inflate(R.layout.my_view, null);
    ViewGroup main = (ViewGroup) findViewById(R.id.insert_point);
    main.addView(view, 0);
}

Причина, по которой я публикую этот очень похожий ответ, заключается в том, что когда я пытался реализовать решение Марка, я застрял на том, какой XML-файл я должен использовать для insert_point и дочернего представления. Сначала я использовал макет в детском представлении, и он полностью не работал, что потребовало несколько часов, чтобы понять. Поэтому надеюсь, что мое исследование может сэкономить время других.

1 голос
/ 24 октября 2017
// Parent layout
LinearLayout parentLayout = (LinearLayout)findViewById(R.id.layout);

// Layout inflater
LayoutInflater layoutInflater = getLayoutInflater();
View view;

for (int i = 1; i < 101; i++){
    // Add the text layout to the parent layout
    view = layoutInflater.inflate(R.layout.text_layout, parentLayout, false);

    // In order to get the view we have to use the new view with text_layout in it
    TextView textView = (TextView)view.findViewById(R.id.text);
    textView.setText("Row " + i);

    // Add the text view to the parent layout
    parentLayout.addView(textView);
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...