Высота Gridview уменьшается

125

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

Настройка android:layout_height="300dp"заставляет это работать. wrap_ contentи, fill_parentвидимо, нет.

Моя сетка:

<GridView
    android:id="@+id/myId"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:gravity="center"
    android:horizontalSpacing="2dp"
    android:isScrollContainer="false"
    android:numColumns="4"
    android:stretchMode="columnWidth"
    android:verticalSpacing="20dp" />

Ресурс "Мои предметы":

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

    <ImageView
        android:id="@+id/appItemIcon"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:src="@android:drawable/ic_dialog_info"
        android:scaleType="center" />      

    <TextView
        android:id="@+id/appItemText"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="My long application name"
        android:gravity="center_horizontal"
        android:textAppearance="?android:attr/textAppearanceSmall" />

</LinearLayout>

Проблема, похоже, не связана с нехваткой вертикального пространства.

Что я могу сделать ?

tacone
источник

Ответы:

351

После (слишком долгого ) исследования я наткнулся на отличный ответ Нила Трафта .

Адаптировать его работу к GridViewигре было очень легко.

ExpandableHeightGridView.java:

package com.example;
public class ExpandableHeightGridView extends GridView
{

    boolean expanded = false;

    public ExpandableHeightGridView(Context context)
    {
        super(context);
    }

    public ExpandableHeightGridView(Context context, AttributeSet attrs)
    {
        super(context, attrs);
    }

    public ExpandableHeightGridView(Context context, AttributeSet attrs,
            int defStyle)
    {
        super(context, attrs, defStyle);
    }

    public boolean isExpanded()
    {
        return expanded;
    }

    @Override
    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
    {
        // HACK! TAKE THAT ANDROID!
        if (isExpanded())
        {
            // Calculate entire height by providing a very large height hint.
            // View.MEASURED_SIZE_MASK represents the largest height possible.
            int expandSpec = MeasureSpec.makeMeasureSpec(MEASURED_SIZE_MASK,
                    MeasureSpec.AT_MOST);
            super.onMeasure(widthMeasureSpec, expandSpec);

            ViewGroup.LayoutParams params = getLayoutParams();
            params.height = getMeasuredHeight();
        }
        else
        {
            super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        }
    }

    public void setExpanded(boolean expanded)
    {
        this.expanded = expanded;
    }
}

Включите его в свой макет следующим образом:

<com.example.ExpandableHeightGridView
    android:id="@+id/myId"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:gravity="center"
    android:horizontalSpacing="2dp"
    android:isScrollContainer="false"
    android:numColumns="4"
    android:stretchMode="columnWidth"
    android:verticalSpacing="20dp" />

Наконец, вам просто нужно попросить его расширить:

mAppsGrid = (ExpandableHeightGridView) findViewById(R.id.myId);
mAppsGrid.setExpanded(true);
tacone
источник
20
это решение неэффективно с точки зрения памяти, и приложение выйдет из строя, если ячейки являются изображениями. Это решение сообщает scrollview, какова высота полного представления сетки, чтобы он мог опускаться, но проблема в том, что для этого он отображает все без использования повторного использования. Не могло работать более 200 предметов.
Мариано Латорре
7
@adamp Думаю, для этого есть полезные кейсы. Если у вас ограниченное количество элементов для отображения в 2-мерном массиве, использование такого типа GridView кажется проще, чем попытка создать какой-то настраиваемый / динамический TableLayout, не так ли?
greg7gkb 07
5
У меня не работает, я поставил ExpandableHeightGridView в ScrollView, он вырезает последнее представление.
Chirag Nagariya
3
@tacone Существует ряд лучших решений для такого рода проблем, которые легко доступны во фреймворке, библиотеке поддержки и другом открытом исходном коде в сети, самым простым из которых может быть простой цикл for, извлекающий представления из адаптера или где-либо еще и добавление их в GridLayout (не GridView; GridLayout также доступен в библиотеке поддержки) TableLayout или аналогичный.
adamp
11
@adamp Если это не очень хорошо, пожалуйста, добавьте свой ответ с лучшим решением, которое вы можете придумать
алеб
34

После использования ответа от @tacone и проверки его работоспособности я решил попробовать сократить код. Это мой результат. PS: Это эквивалент того, что для логического ответа «расширенный» в tacones всегда установлено значение true.

public class StaticGridView extends GridView {

    public StaticGridView(Context context) {
        super(context);
    }

    public StaticGridView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public StaticGridView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    @Override
    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(MEASURED_SIZE_MASK, MeasureSpec.AT_MOST));
        getLayoutParams().height = getMeasuredHeight();
    }
}
Джейкоб Р
источник
Но подождите - это все еще страдает проблемой использования памяти; ты больше не перерабатываешь, верно?
Fattie
6

Другой аналогичный подход, который сработал для меня, - это вычислить высоту для одной строки, а затем со статическими данными (вы можете адаптировать их для разбивки на страницы) вы можете рассчитать, сколько строк у вас есть, и легко изменить размер высоты GridView.

    private void resizeGridView(GridView gridView, int items, int columns) {
    ViewGroup.LayoutParams params = gridView.getLayoutParams();
    int oneRowHeight = gridView.getHeight();
    int rows = (int) (items / columns);
    params.height = oneRowHeight * rows;
    gridView.setLayoutParams(params);
}

Используйте этот код после настройки адаптера и при отрисовке GridView, иначе вы получите height = 0.

gridView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                if (!gridViewResized) {
                    gridViewResized = true;
                    resizeGridView(gridView, numItems, numColumns);
                }
            }
        });
Pelanes
источник
1
У меня это сработало - я использую кучу GridView внутри ListView. Не уверен, что это плохая идея или нет - нужно исследовать производительность с большим набором данных. Но в любом случае спасибо за код. Я думаю, что есть ошибка по одному - мне пришлось использоватьint rows = items / columns + 1;
Эндрю
для java.lang.ClassCastException: android.widget.RelativeLayout$LayoutParams cannot be cast to android.widget.AbsListView$LayoutParams
ОС
ViewGroup.LayoutParams params = gridView.getLayoutParams (); выбрасывает исключение NullPointerException
Люк Эллисон
4

Найденные тиконы полезны ... поэтому я перенес его на C # (Xamarin)

public class ExpandableHeightGridView: GridView
{
    bool _isExpanded = false;

    public ExpandableHeightGridView(Context context) : base(context)
    {            
    }

    public ExpandableHeightGridView(Context context, IAttributeSet attrs) : base(context, attrs)
    {            
    }

    public ExpandableHeightGridView(Context context, IAttributeSet attrs, int defStyle) : base(context, attrs, defStyle)
    {            
    }

    public bool IsExpanded
    {
        get { return _isExpanded; }

        set { _isExpanded = value;  }
    }

    protected override void OnMeasure(int widthMeasureSpec, int heightMeasureSpec)
    {
        // HACK! TAKE THAT ANDROID!
        if (IsExpanded)
        {
            // Calculate entire height by providing a very large height hint.
            // View.MEASURED_SIZE_MASK represents the largest height possible.
            int expandSpec = MeasureSpec.MakeMeasureSpec( View.MeasuredSizeMask, MeasureSpecMode.AtMost);
            base.OnMeasure(widthMeasureSpec,expandSpec);                

            var layoutParameters = this.LayoutParameters;
            layoutParameters.Height = this.MeasuredHeight;
        }
        else
        {
            base.OnMeasure(widthMeasureSpec,heightMeasureSpec);    
        }
    }
}
spaceMonster
источник
1
Престижный чувак. Отлично работает в xamarin.android
Сучит
0

Просто рассчитайте высоту для AT_MOST и установите ее по мере измерения. Здесь GridView Scroll так работать не будет. Необходимо явно использовать вертикальную прокрутку.

 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
     int heightSpec;

     if (getLayoutParams().height == LayoutParams.WRAP_CONTENT) {

         heightSpec = MeasureSpec.makeMeasureSpec(
                        Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
     }
     else {
         // Any other height should be respected as is.
         heightSpec = heightMeasureSpec;
     }

     super.onMeasure(widthMeasureSpec, heightSpec);
 }
Удай Шраван К
источник
Сучит