我最终对与Android GridView有关的问题感到困惑。很抱歉,我的描述有点长,但是我没有太多时间来优化我的问题。

这就是我想要得到的。
应该有一个带有(8x10)元素的GridView。
每个元素都有5种受支持颜色集的初始生成的随机颜色。 (我已经使用Android View类型作为元素,并且只设置了背景颜色)。

每次用户单击某个元素时,都应“删除”该元素,并将其上方的所有元素下移以填充空闲空间。 (我从上面的元素获取颜色ID,并将其设置为当前元素,直到到达最顶部的元素,然后为该列的顶部元素随机生成新颜色)。

在完成所有这些工作之后,我只需在我的适配器上调用用于GridView的函数notifyDataSetChanged()。

问题在于行为是不可预测的。在某些设备上,它可以在某些设备上运行,它会更新所有元素,并且所有元素都会更改。有时我发现没有颜色设置与优化有关,并且似乎找到了解决方法...

那么,有人遇到某种问题吗?您如何达到稳定(预期)的行为?
也许您可以提出一些其他(更轻松的)解决方案或一些相关的文档/示例/课程,我已经阅读了Google带来的有关GridLayouts的几乎所有内容。

谢谢您花费在我的问题上的时间!
谢谢,
阿森

更新:

这是我正在使用的部分代码:
很抱歉没有发布一些详细信息。

在layout.xml中

 <GridView
    class="<name>"
android:id="@+id/gridView1"
android:listSelector="@null"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="0dp"
android:layout_alignParentBottom="true"
android:horizontalSpacing="0dp"
android:numColumns="8"
android:padding="0dp"
android:stretchMode="columnWidth"
android:cacheColorHint="@android:color/transparent"
android:verticalSpacing="0dp" >
 </GridView>


在layout.simple_grid_view_element中:

<View xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/box"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:gravity="center_vertical"
/>


在扩展BaseAdapter的ImageAdapder中:

 ...
 @Override
 public View getView(int position, View convertView, ViewGroup parent) {
View v = convertView;
if (convertView == null) {  // for the first time initialize each element.
      LayoutInflater Inf = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
      convertView = Inf.inflate(R.layout.simple_grid_view_element, parent, false);
      v = (ImageView) convertView.findViewById(R.id.box);
      WindowManager wm = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
      Display display = wm.getDefaultDisplay();
      int width = display.getWidth();
      int height = display.getHeight() - 40;
      v.setLayoutParams(new GridView.LayoutParams(((int)(width)/(mEngine.ColumnNumber)), ((int)((height)/mEngine.RowNumber) - 5)));
          v.setBackgroundResource( mEngine.generate_color_based_on_box_color(mEngine.TABLE.get(position)));
    }
    return v;
 }


在MainActivity中:

....
GridView gridview = (GridView) findViewById(R.id.gridView1);
gridview.setAdapter(mEngine.getAdapter());
gridview.setVerticalScrollBarEnabled(false);
gridview.setOnItemClickListener(this);

....
@Override
public void onItemClick(AdapterView<?> parent, View view, int position,
        long id) {
mEngine.RespondOnClick(parent, position);
}


在Engine类中,我只是从onItemClick函数获取具有给定位置的项目的背景色:
该函数使用了一个辅助数组TABLE,该数组实际上是用于保留颜色索引的,因为View.getBackground()函数返回的巨大值由于JAVA没有未签名的类型而无法正常工作...(但此问题已解决)。我已经调试了该应用程序,并且TABLE数组中的所有更改都正确执行。

...
private ImageAdapter mAdapter;
...
Engine(Context mContext) {
    Columns = new ArrayList<ArrayList<Integer>>(ColumnNumber);
for (int i = 0; i < ColumnNumber; ++i) {
    Columns.add(new ArrayList<Integer>());
}

TABLE = new ArrayList<Integer>(ColumnNumber * RowNumber);
for (int i = 0; i < ColumnNumber * RowNumber; ++i) {
    TABLE.add(i, generateRandomColor());
}

    mAdapter = new ImageAdapter(mContext, this);
}

public void RespondOnClick(AdapterView<?> parent, final int position) {
CollectBoxesMarkedToRemoveByLine();
for (int i = 0; i < Columns.size(); ++i) {
    while (0 != Columns.get(i).size()) {
        int TopDeletedBoxPosition = Columns.get(i).get(0);
        int CurrentRowValue = (TopDeletedBoxPosition / ColumnNumber);
        while (0 < CurrentRowValue) {
            int PrevBoxColor = TABLE.get(TopDeletedBoxPosition - ColumnNumber);
            TABLE.set(TopDeletedBoxPosition, PrevBoxColor);
            this.getBox(TopDeletedBoxPosition).setBackgroundResource(generate_color_based_on_box_color(PrevBoxColor));
            --CurrentRowValue;
            TopDeletedBoxPosition -= ColumnNumber;
        }
        if (0 == CurrentRowValue) {
            int newColor = generateRandomColor();
            TABLE.set(TopDeletedBoxPosition, newColor);
            this.getBox(TopDeletedBoxPosition).setBackgroundResource(generate_color_based_on_box_color(TABLE.get(TopDeletedBoxPosition)));
        }
        Columns.get(i).remove(0);
    }
}
...
mAdapter.notifyDataSetChanged();  // called after RespondOnClick() has done all the replacements.

最佳答案

我刚刚解决了这个问题,并希望分享经验。

该问题的原因来自getView()函数。后台资源设置不正确。在两种情况下,函数第一次调用且convertview为null时,以及在gridview更新期间进行进一步调用时,都应如此。

因此,我唯一要做的就是从if-then语句移至该行:

之前:

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    View v = convertView;
    if (convertView == null) {  // for the first time initialize each element.
        ...
        v.setBackgroundResource( mEngine.generate_color_based_on_box_color(mEngine.TABLE.get(position)));
    }
    return v;
}


后:

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    View v = convertView;
    if (convertView == null) {  // for the first time initialize each element.
        ...
    }
    v.setBackgroundResource( mEngine.generate_color_based_on_box_color(mEngine.TABLE.get(position)));
    return v;
}


希望这对某人有帮助:)

干杯,
阿森

关于android - Android GridView不可预测的行为,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/24515809/

10-10 18:32
查看更多