Item-Selection with Checkbox in Android-GridView

情到浓时终转凉″ 提交于 2020-01-25 09:35:27

问题


I've come to a problem with the Android GridView. I already tried various solutions from Stackoverflow but none of them actually worked.

I'm populating a GridView with the help of an ArrayAdapter. The Adapter inflates a layout containing only an image and a checkbox. I'm using that "convertView"-thing in order to prevent flickering when the images get added (nearly 500 images are getting added to the GridView).

The Problem: When I check a checkbox and scroll, the checked-state disappears. Also, when I for instance check multiple GridView item's checkboxes, the pattern of selection endlessly continues for all the following elements.


My GridView is the child-element of a RelativeLayout. The RelativeLayout is used as the root element of a fragment being used in a ViewPager:

<GridView
    android:id="@+id/MainGrid"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/appBackgroundColor"
    android:drawSelectorOnTop="false"
    android:listSelector="#00000000"
    android:numColumns="2"
    android:paddingLeft="6dp"
    android:paddingTop="5dp"
    android:paddingRight="6dp"
    android:translationZ="2dp" />

I'm populating the GridView with the help of a thread, which is downloading image-data from a website. I think there shouldn't be a problem with this, but for completeness, here is the code:

//Declaring the Array
ArrayList<GridImage> ImageArray = new ArrayList<>();

//And the adapter
Adapter_GridAdapter GridViewAdapter = new Adapter_GridAdapter (getActivity().getApplicationContext(), ImageArray );

//The following is executed 500 times in a thread
ImageArray.add(new GridImage(URL));
getActivity().runOnUiThread(new Runnable() {
    @Override
    public void run() {
        GridViewAdapter.notifyDataSetChanged();
    }
});

The code of the GridView adapter:

public class Adapter_GridAdapter extends ArrayAdapter<GridImage> {

Context context;

public Adapter_GridAdapter (Context context, ArrayList<GridImage> SearchResults) {
    super(context, 0, SearchResults);
    this.context = context;
}

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

    LayoutInflater inflater = LayoutInflater.from(getContext());
    View view = convertView;
    if (convertView == null) {
        view = inflater.inflate(R.layout.griditem_ig_post, null);

        final ImageView ThisViewIMG = view.findViewById(R.id.GridImageView);

        //Calculating Image Height (Square Image)
        ThisViewIMG.addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
            @Override
            public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
                ThisViewIMG.removeOnLayoutChangeListener(this);
                ThisViewIMG.getLayoutParams().height = ThisViewIMG.getWidth();
                ThisViewIMG.requestLayout();
            }
        });

    }

    final GridImage ThisImg = getItem(position);
    final ImageView ThumbnailImage = view.findViewById(R.id.GridImageView);

    //THIS IS THE CUSTOM CHECKBOX!
    final CustomCheckBox scb = (CustomCheckBox) view.findViewById(R.id.MyCheckBox);

    //I AM SAVING THE VALUE IN THE OBJECT (TRY #38181 THAT DIDN'T WORK)
    scb.setChecked(ThisImg.IsChecked);
    scb.setOnCheckedChangeListener(new CustomCheckBox.OnCheckedChangeListener() {
        @Override
        public void onCheckedChanged(CustomCheckBox checkBox, boolean isChecked) {
            Log.d("CustomCheckBox", String.valueOf(isChecked));
            ThisImg.setChecked(isChecked);
        }
    });
Glide.with(context).load(Post.IMG_ThumbnailURL).fitCenter().into(ThumbnailImage);


    return view;
}
}

The Object itself is quite simple, the problem (i think) also shouldn't hide in here:

public class GridImage{
    public String PostURL;
    public Boolean IsChecked = false;

    GridImage(String URL){
        PostURL = URL;
    }

    public void setChecked(Boolean checked) {
        IsChecked = checked;
    }
}

What I already tried:

  • https://stackoverflow.com/a/24016687/7850133 - Which is my solution just with tags instead of saving straight into the object, am I right?
  • https://stackoverflow.com/a/17188179/7850133 - SparseBooleanArray, but I'm not a 100% sure I did this the right way

回答1:


When doing with ListView/GridView adapters, I always avoid to have final and onCheckedChangeListener because both may have problem sometimes even the logic seems right, so try the code below:

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

    LayoutInflater inflater = LayoutInflater.from(getContext());
    View view = convertView;
    if (convertView == null) {
        view = inflater.inflate(R.layout.griditem_ig_post, null);

        ImageView ThisViewIMG = view.findViewById(R.id.GridImageView);

        //Calculating Image Height (Square Image)
        ThisViewIMG.addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
            @Override
            public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
                v.removeOnLayoutChangeListener(this);
                v.getLayoutParams().height = v.getWidth();
                v.requestLayout();
            }
        });

    }

    GridImage ThisImg = getItem(position);
    ImageView ThumbnailImage = view.findViewById(R.id.GridImageView);

    //THIS IS THE CUSTOM CHECKBOX!
    CustomCheckBox scb = (CustomCheckBox) view.findViewById(R.id.MyCheckBox);

    scb.setChecked(ThisImg.IsChecked);
    scb.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            CustomCheckBox cb = (CustomCheckBox)view;
            int position = (int)cb.getTag();
            GridImage ThisImg = getItem(position);
            Log.d("CustomCheckBox", String.valueOf(!ThisImg.IsChecked));
            ThisImg.setChecked(!ThisImg.IsChecked);
            cb.setChecked(ThisImg.IsChecked);
        }
    });
    Glide.with(context).load(Post.IMG_ThumbnailURL).fitCenter().into(ThumbnailImage);

    scb.setTag(position);
    return view;
}

Hope that helps!



来源:https://stackoverflow.com/questions/56875819/item-selection-with-checkbox-in-android-gridview

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!