Почему imageButton в ListView не переключается до прокрутки?

У меня есть простой listView через пользовательский адаптер. Каждая строка в представлении списка состоит из ImageButton и TextView.

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

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

Вот APK для примера тестового приложения: https://www.dropbox.com/s/pzlahtqkgj010m8/app-ListViewExample.apk?dl=0 Вот соответствующие части моего кода:

avtivity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
    android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity">


    <ListView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/listView"

        />
</RelativeLayout>

и для визуализации единственной строки в ListView используется следующий xml:

single_row.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="horizontal" android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_marginTop="20dp"
    android:layout_marginBottom="20dp"
    >


    <ImageButton
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/imageButton"
        android:src="@drawable/b"
        android:layout_weight="2"
        style="@style/ListView.first"
        android:layout_gravity="center"
        />

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textAppearance="?android:attr/textAppearanceLarge"
        android:text="Large Text"
        android:id="@+id/textView"
        android:layout_weight="10"
        android:layout_gravity="center"
        />
</LinearLayout>

переход к Java-коду,

MainActivity.java является

public class MainActivity extends AppCompatActivity {

    String[] titles;
    ListView list;
    private listViewAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Resources res = getResources();

        titles = res.getStringArray(R.array.titles);
        list = (ListView) findViewById(R.id.listView);
        adapter = new listViewAdapter(this, titles);
        list.setAdapter(adapter);
    }
}

и пользовательский адаптер listViewAdapter.java выглядит следующим образом

class sListItem {
    public ImageButton button;
    public TextView text;
    public boolean active;
}

public class listViewAdapter extends ArrayAdapter<String> {

    private final String[] titles;
    Context context;

    private int size = 15;
    public sListItem mp[] = new sListItem[size];


    public listViewAdapter(Context context, String[] titles) {
        super(context, R.layout.single_row, R.id.imageButton, titles);
        this.context = context;
        this.titles = titles;

    }

    @Override
    public View getView(final int position, View convertView, ViewGroup parent) {
        LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        View row = inflater.inflate(R.layout.single_row, parent, false);

        if(mp[position] == null)
            mp[position] = new sListItem();

        mp[position].button = (ImageButton) row.findViewById(R.id.imageButton);
        mp[position].button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mp[position].active = !mp[position].active;

                setIcon(position, mp[position].active);
            }
        });

        mp[position].text = (TextView) row.findViewById(R.id.textView);
        mp[position].text.setText(titles[position]);
        setIcon(position, mp[position].active);

        return row;
    }


    private void setIcon(int position, boolean active) {
        Drawable drawable;

        if(active){
            drawable = getContext().getResources().getDrawable(R.drawable.a);
        } else {
            drawable = getContext().getResources().getDrawable(R.drawable.b);
        }
        mp[position].button.setImageDrawable(drawable);
    }
}

представление списка в примере

РЕДАКТИРОВАТЬ: Добавлена ​​ссылка на образец тестового приложения.

1 ответ

Решение

Вы должны сказать адаптеру обновить, позвонив notifydatasetchanged в слушателе щелчка

 mp[position].button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mp[position].active = !mp[position].active;
                setIcon(position, mp[position].active);
                notifydatasetchanged();
            }
        });
Другие вопросы по тегам