각 행마다 다른 레이아웃의 Android ListView


348

각 행에 대해 다른 레이아웃을 포함하는 단일 ListView를 갖는 가장 좋은 방법을 결정하려고합니다. 전체 목록보기에 대해 사용자 정의 행을 지원하기 위해 사용자 정의 행 + 사용자 정의 배열 어댑터를 만드는 방법을 알고 있지만 ListView에서 다양한 행 스타일을 어떻게 구현할 수 있습니까?


1
업데이트 : 안드로이드의 RecyclerView를 사용하여 여러 행 레이아웃에 대한 데모 Viewcode2concept.blogspot.in/2015/10/…
nitesh

답변:


413

레이아웃 유형이 몇 개인 지 알고 있으므로 이러한 방법을 사용할 수 있습니다.

getViewTypeCount() -이 메소드는 목록에 몇 개의 행 유형이 있는지에 대한 정보를 리턴합니다.

getItemViewType(int position) -위치에 따라 사용해야하는 레이아웃 유형 정보를 반환합니다.

그런 다음 null 인 경우에만 레이아웃을 팽창시키고을 사용하여 유형을 결정하십시오 getItemViewType.

자세한 내용 은 이 자습서 를 참조하십시오.

주석에서 설명 한 구조에서 일부 최적화를 달성하려면 다음을 제안합니다.

  • 라는 객체에 뷰를 저장합니다 ViewHolder. findViewById()매번 getView메서드 를 호출 할 필요가 없으므로 속도가 증가 합니다. API 데모의 List14를 참조하십시오 .
  • 모든 속성 조합을 준수하는 하나의 일반 레이아웃을 만들고 현재 위치에없는 경우 일부 요소를 숨 깁니다.

도움이 되길 바랍니다. 데이터 구조와 XML 스텁을 정확히 행에 매핑하려는 방법을 제공 할 수 있다면보다 정확한 조언을 줄 수있을 것입니다. 픽셀 단위.


블로그가 매우 좋았지 만 확인란을 추가했습니다. 첫 번째 항목을 확인하고 목록을 스크롤하는 데 문제가있었습니다. 확인 된 이상하게 익명 항목. 이를위한 솔루션을 제공 할 수 있습니까? 감사합니다
Lalit Poptani

2
이것을 다시 파고해서 죄송하지만 실제로 getItemViewType을 사용하여 각각 팽창되는 별도의 레이아웃 파일 대신 하나의 큰 레이아웃 파일을 가지고 그 부분의 가시성을 제어하는 ​​것이 좋습니다.
Makibo

2
당신도 그렇게 할 수 있습니다. 나는 여전히 여기에 노출되는 방식을 선호하지만. 달성하고자하는 것을보다 명확하게합니다.
Cristian

그러나 다중 레이아웃 전략에서는 setTag에 하나의 뷰 홀더 만 포함 할 수 있고 행 레이아웃이 다시 전환 될 때마다 findViewById ()를 호출해야하므로 사용자 뷰 홀더를 올바르게 사용할 수 없습니다. 이것은 목록보기를 매우 낮은 성능으로 만듭니다. 나는 개인적으로 그것을 경험 했습니까?
pyus13

@ pyus13은 단일 뷰 홀더에서 원하는만큼의 뷰를 선언 할 수 있으며 뷰 홀더에서 선언 된 모든 뷰를 사용할 필요는 없습니다. 샘플 코드가 필요한 경우 알려 주시면 게시하겠습니다.
Ahmad Ali Nasir

62

전체 목록보기에 대해 사용자 정의 행을 지원하기 위해 사용자 정의 행 + 사용자 정의 배열 어댑터를 만드는 방법을 알고 있습니다. 그러나 하나의 목록보기가 다양한 행 스타일을 어떻게 지원할 수 있습니까?

당신은 이미 기본을 알고 있습니다. 제공되는 행 / 커서 정보에 따라 다른 레이아웃 / 뷰를 반환하도록 사용자 정의 어댑터를 가져와야합니다.

A ListViewAdapterView 에서 파생되므로 여러 행 스타일을 지원할 수 있습니다 .

AdapterView는 하위 항목이 어댑터에 의해 결정되는 보기 입니다.

Adapter 를 보면 행별보기 사용을 설명하는 메소드가 표시됩니다.

abstract int getViewTypeCount()
// Returns the number of types of Views that will be created ...

abstract int getItemViewType(int position)
// Get the type of View that will be created ...

abstract View getView(int position, View convertView, ViewGroup parent)
// Get a View that displays the data ...

후자의 두 가지 방법 은 위치를 제공 하므로 해당 행에 사용해야하는 뷰 유형결정할 수 있습니다 .


물론, 일반적으로 AdapterView 및 Adapter를 직접 사용하지 않고 서브 클래스 중 하나에서 사용하거나 파생합니다. Adapter의 서브 클래스는 다른 행에 대한 사용자 정의 레이아웃을 얻는 방법을 변경하는 추가 기능을 추가 할 수 있습니다. 주어진 행에 사용 된보기는 어댑터에 의해 구동되므로, 어댑터가 주어진 행에 대해 원하는보기를 리턴하도록하는 것이 요령입니다. 이를 수행하는 방법은 특정 어댑터에 따라 다릅니다.

예를 들어, 사용하는 ArrayAdapter와를 ,

  • getView()지정된 위치에 대해 원하는 뷰를 팽창시키고 채우고 반환하려면 재정의 합니다. 이 getView()방법에는 convertView매개 변수 를 통한 기회 재사용보기가 포함됩니다 .

그러나 파생 상품 사용 CursorAdapter를 ,

  • 우선 newView(), 팽창 채우기, 현재 커서 상태에 대해 원하는보기 (즉, 현재 "행")을 반환하는 [당신은 또한 재정의해야 bindView하므로 위젯 뷰를 재사용 할 수 있음]

그러나 사용하는 SimpleCursorAdapter를 ,

  • 주어진 행 (현재 커서 상태) 및 데이터 "열"에 대해 원하는 뷰를 팽창, 채우기 및 반환 SimpleCursorAdapter.ViewBinder하는 setViewValue()메소드를 사용하여 를 정의합니다 . 이 메서드는 "특별한"뷰만 정의하고 "일반"바인딩에 대한 SimpleCursorAdapter의 표준 동작을 지연시킬 수 있습니다.

사용하는 어댑터 종류에 대한 특정 예 / 자습서를 찾아보십시오.


이러한 어댑터 유형 중 유연한 어댑터 구현에 가장 적합한 생각은 무엇입니까? 이것을 위해 보드에 다른 질문을 추가하고 있습니다.
안드로이드

1
@Androider- "유연성에 가장 적합"은 매우 개방적입니다. 모든 요구를 충족시킬 수있는 모든 클래스가 없습니다. 그것은 풍부한 계층 구조입니다-그것은 당신의 목적에 유용한 서브 클래스의 기능이 있는지에 달려 있습니다. 그렇다면 해당 서브 클래스로 시작하십시오. 그렇지 않은 경우로 이동하십시오 BaseAdapter. BaseAdapter에서 파생하는 것이 가장 "유연한"것이지만 이미 다른 어댑터에 포함 된 지식과 성숙도를 이용하지 않기 때문에 코드 재사용 및 성숙도에서 최악이 될 것입니다. BaseAdapter다른 어댑터가 맞지 않는 비표준 컨텍스트가 있습니까?
Bert F

3
사이의 미세한 구별 +1 CursorAdapter하고 SimpleCursorAdapter.
Giulio Piancastelli

1
또한 오버라이드 (override)하는 경우주의 ArrayAdapter, 당신이 한, 생성자를 제공 레이아웃 중요하지 않습니다 getView()팽창 반환 레이아웃의 오른쪽 종류
woojoo666

1
주목해야한다 getViewTypeCount()전화 할 때마다 한 번만 트리거 ListView.setAdapter()하지 모든 들어 Adapter.notifyDataSetChanged().
hidro

43

아래 코드를 살펴보십시오.

먼저 커스텀 레이아웃을 만듭니다. 이 경우 네 가지 유형이 있습니다.

even.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:background="#ff500000"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/white"
        android:layout_width="match_parent"
        android:layout_gravity="center"
        android:textSize="24sp"
        android:layout_height="wrap_content" />

 </LinearLayout>

odd.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:background="#ff001f50"
    android:gravity="right"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/white"
        android:layout_width="wrap_content"
        android:layout_gravity="center"
        android:textSize="28sp"
        android:layout_height="wrap_content"  />

 </LinearLayout>

white.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:background="#ffffffff"
    android:gravity="right"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/black"
        android:layout_width="wrap_content"
        android:layout_gravity="center"
        android:textSize="28sp"
        android:layout_height="wrap_content"   />

 </LinearLayout>

black.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:background="#ff000000"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/text"
        android:textColor="@android:color/white"
        android:layout_width="wrap_content"
        android:layout_gravity="center"
        android:textSize="33sp"
        android:layout_height="wrap_content"   />

 </LinearLayout>

그런 다음 목록보기 항목을 만듭니다. 우리의 경우, 문자열과 타입.

public class ListViewItem {
        private String text;
        private int type;

        public ListViewItem(String text, int type) {
            this.text = text;
            this.type = type;
        }

        public String getText() {
            return text;
        }

        public void setText(String text) {
            this.text = text;
        }

        public int getType() {
            return type;
        }

        public void setType(int type) {
            this.type = type;
        }

    }

그런 다음 뷰 홀더를 만듭니다. Android OS는 레이아웃 참조를 유지하여 항목이 사라지고 화면에 다시 나타날 때 항목을 재사용 할 것을 권장합니다. 이 방법을 사용하지 않으면 항목이 화면에 나타날 때마다 Android OS에서 새로운 항목을 생성하여 앱에서 메모리 누수가 발생합니다.

public class ViewHolder {
        TextView text;

        public ViewHolder(TextView text) {
            this.text = text;
        }

        public TextView getText() {
            return text;
        }

        public void setText(TextView text) {
            this.text = text;
        }

    }

마지막으로 getViewTypeCount () 및 getItemViewType (int position)을 재정의하는 사용자 지정 어댑터를 만듭니다.

public class CustomAdapter extends ArrayAdapter {

        public static final int TYPE_ODD = 0;
        public static final int TYPE_EVEN = 1;
        public static final int TYPE_WHITE = 2;
        public static final int TYPE_BLACK = 3;

        private ListViewItem[] objects;

        @Override
        public int getViewTypeCount() {
            return 4;
        }

        @Override
        public int getItemViewType(int position) {
            return objects[position].getType();
        }

        public CustomAdapter(Context context, int resource, ListViewItem[] objects) {
            super(context, resource, objects);
            this.objects = objects;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {

            ViewHolder viewHolder = null;
            ListViewItem listViewItem = objects[position];
            int listViewItemType = getItemViewType(position);


            if (convertView == null) {

                if (listViewItemType == TYPE_EVEN) {
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_even, null);
                } else if (listViewItemType == TYPE_ODD) {
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_odd, null);
                } else if (listViewItemType == TYPE_WHITE) {
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_white, null);
                } else {
                    convertView = LayoutInflater.from(getContext()).inflate(R.layout.type_black, null);
                }

                TextView textView = (TextView) convertView.findViewById(R.id.text);
                viewHolder = new ViewHolder(textView);

                convertView.setTag(viewHolder);

            } else {
                viewHolder = (ViewHolder) convertView.getTag();
            }

            viewHolder.getText().setText(listViewItem.getText());

            return convertView;
        }

    }

우리의 활동은 다음과 같습니다.

private ListView listView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main); // here, you can create a single layout with a listview

        listView = (ListView) findViewById(R.id.listview);

        final ListViewItem[] items = new ListViewItem[40];

        for (int i = 0; i < items.length; i++) {
            if (i == 4) {
                items[i] = new ListViewItem("White " + i, CustomAdapter.TYPE_WHITE);
            } else if (i == 9) {
                items[i] = new ListViewItem("Black " + i, CustomAdapter.TYPE_BLACK);
            } else if (i % 2 == 0) {
                items[i] = new ListViewItem("EVEN " + i, CustomAdapter.TYPE_EVEN);
            } else {
                items[i] = new ListViewItem("ODD " + i, CustomAdapter.TYPE_ODD);
            }
        }

        CustomAdapter customAdapter = new CustomAdapter(this, R.id.text, items);
        listView.setAdapter(customAdapter);
        listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView adapterView, View view, int i, long l) {
                Toast.makeText(getBaseContext(), items[i].getText(), Toast.LENGTH_SHORT).show();
            }
        });

    }
}

이제 mainactivity.xml 안에 다음과 같이 목록보기를 만듭니다.

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout 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"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    tools:context="com.example.shivnandan.gygy.MainActivity">

    <android.support.design.widget.AppBarLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:theme="@style/AppTheme.AppBarOverlay">

        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:background="?attr/colorPrimary"
            app:popupTheme="@style/AppTheme.PopupOverlay" />

    </android.support.design.widget.AppBarLayout>

    <include layout="@layout/content_main" />

    <ListView
        android:layout_width="match_parent"

        android:layout_height="match_parent"

        android:id="@+id/listView"
        android:layout_alignParentRight="true"
        android:layout_alignParentEnd="true"


        android:layout_marginTop="100dp" />

</android.support.design.widget.CoordinatorLayout>

<include layout = "@ layout / content_main"/> 어디에서 오는가
nyxee

(convertView == null) 절만 필요했습니다. '조회자'가 필요하지 않음
Jomme

14

사용자 정의 배열 어댑터에서는 아마도 익숙한 getView () 메소드를 대체합니다. 그런 다음 getView 메소드에 전달 된 위치 인수에 따라 switch 문 또는 if 문을 사용하여 특정 사용자 정의보기를 리턴하면됩니다. 안드로이드는 위치 / 행에 적합한 유형의 convertView 만 제공한다는 점에서 영리합니다. 올바른 유형인지 확인하지 않아도됩니다. getItemViewType () 및 getViewTypeCount () 메소드를 적절하게 대체하여 Android에 도움을 줄 수 있습니다.


4

목록보기에서 다른 유형의보기를 표시 해야하는 경우보기를 전환하지 않고 어댑터에서 getViewTypeCount () 및 getItemViewType ()을 사용하는 것이 좋습니다 .VIEW.GONE 및 VIEW.VISIBLE은 getView () 내부에서 매우 비싼 작업 일 수 있습니다. 목록 스크롤에 영향을줍니다.

어댑터에서 getViewTypeCount () 및 getItemViewType ()을 사용하려면이 항목을 확인하십시오.

링크 : get-use-of-getviewtypecount


1

ListView는 모든 행 항목에 대해 동일한 정적보기와 같은 간단한 사용 사례를위한 것입니다.
ViewHolders를 작성하고을 많이 사용 getItemViewType()하고 다른 행 항목 레이아웃 xml을 동적으로 표시해야하므로 Android API 22에서 사용 가능한 RecyclerView 를 사용하여이를 수행해야 합니다. 여러보기 유형에 대해 더 나은 지원 및 구조를 제공합니다.

RecyclerView를 사용하여 원하는 것을 수행하는 방법에 대한 이 자습서 를 확인하십시오 .

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.