it-swarm.com.ru

Android: элементы ListView с несколькими нажимаемыми кнопками

У меня есть ListView, где каждый элемент в списке содержит TextView и две разные кнопки. Что-то вроде этого:

ListView
--------------------
[Text]
[Button 1][Button 2]
--------------------
[Text]
[Button 1][Button 2]
--------------------
... (and so on) ...

С помощью этого кода я могу создать OnItemClickListener для всего элемента:

listView.setOnItemClickListener(new OnItemClickListener() {
    @Override
    public void onItemClick(AdapterView<?> list, View view, int position, long id) {
        Log.i(TAG, "onListItemClick: " + position);

        }

    }
});

Однако я не хочу, чтобы весь элемент был кликабельным, а только две кнопки каждого элемента списка.

Поэтому мой вопрос заключается в том, как реализовать onClickListener для этих двух кнопок со следующими параметрами:

  • int button (какая кнопка элемента была нажата)
  • int position (элемент в списке, по которому происходило нажатие кнопки)

Обновление: Я нашел решение, как описано в моем ответе ниже. Теперь я могу нажать/нажать кнопку через сенсорный экран. Тем не менее, я не могу вручную выбрать его с помощью трекбола. Он всегда выбирает весь элемент списка и оттуда сразу переходит к следующему элементу списка, игнорируя кнопки, хотя я установил .setFocusable(true) и setClickable(true) для кнопок в getView().

Я также добавил этот код в свой адаптер списка:

@Override
public boolean  areAllItemsEnabled() {
    return false;           
}

@Override
public boolean isEnabled(int position) {
        return false;
}

Это приводит к тому, что ни один элемент списка больше не может быть выбран вообще. Но это не помогло сделать вложенные кнопки выбираемыми.

Есть идеи?

181
znq

Решение этого на самом деле проще, чем я думал. Вы можете просто добавить в методе getView() своего пользовательского адаптера setOnClickListener () для используемых вами кнопок.

Любые данные, связанные с кнопкой, должны быть добавлены с помощью myButton.setTag() в getView() и доступны в onClickListener через view.getTag()

Я разместил подробное решение на мой блог в качестве учебного пособия.

150
znq

Это своего рода ответ придатка @ znq ...

Есть много случаев, когда вы хотите узнать положение строки для элемента, по которому щелкнули, И вы хотите знать, какой вид в строке был нажат. Это будет намного важнее в планшетных интерфейсах.

Вы можете сделать это с помощью следующего пользовательского адаптера:

private static class CustomCursorAdapter extends CursorAdapter {

    protected ListView mListView;

    protected static class RowViewHolder {
        public TextView mTitle;
        public TextView mText;
    }

    public CustomCursorAdapter(Activity activity) {
        super();
        mListView = activity.getListView();
    }

    @Override
    public void bindView(View view, Context context, Cursor cursor) {
        // do what you need to do
    }

    @Override
    public View newView(Context context, Cursor cursor, ViewGroup parent) {
        View view = View.inflate(context, R.layout.row_layout, null);

        RowViewHolder holder = new RowViewHolder();
        holder.mTitle = (TextView) view.findViewById(R.id.Title);
        holder.mText = (TextView) view.findViewById(R.id.Text);

        holder.mTitle.setOnClickListener(mOnTitleClickListener);
        holder.mText.setOnClickListener(mOnTextClickListener);

        view.setTag(holder);

        return view;
    }

    private OnClickListener mOnTitleClickListener = new OnClickListener() {
        @Override
        public void onClick(View v) {
            final int position = mListView.getPositionForView((View) v.getParent());
            Log.v(TAG, "Title clicked, row %d", position);
        }
    };

    private OnClickListener mOnTextClickListener = new OnClickListener() {
        @Override
        public void onClick(View v) {
            final int position = mListView.getPositionForView((View) v.getParent());
            Log.v(TAG, "Text clicked, row %d", position);
        }
    };
}
62
greg7gkb

Для будущих читателей:

Чтобы вручную выбрать кнопки с помощью трекбола:

myListView.setItemsCanFocus(true);

И чтобы отключить фокус на весь список элементов:

myListView.setFocusable(false);
myListView.setFocusableInTouchMode(false);
myListView.setClickable(false);

У меня это работает нормально, я могу нажимать на кнопки с сенсорным экраном, а также позволяет фокусировать клик с помощью клавиатуры

22
Fabricio PH

У меня не так много опыта, как у вышеупомянутых пользователей, но я столкнулся с той же проблемой и решил ее с помощью нижеприведенного решения.

<Button
        Android:id="@+id/btnRemove"
        Android:layout_width="0dp"
        Android:layout_height="wrap_content"
        Android:layout_toRightOf="@+id/btnEdit"
        Android:layout_weight="1"
        Android:background="@drawable/btn"
        Android:text="@string/remove" 
        Android:onClick="btnRemoveClick"
        />

btnRemoveClick Событие клика

public void btnRemoveClick(View v)
{
    final int position = listviewItem.getPositionForView((View) v.getParent()); 
    listItem.remove(position);
    ItemAdapter.notifyDataSetChanged();

}
12
Bhavin Chauhan

Возможно, вы нашли, как это сделать, но вы можете позвонить

ListView.setItemsCanFocus(true)

и теперь ваши кнопки поймают фокус

9
Rafal

Я не уверен, что будет лучшим способом, но работает нормально, и весь код остается в вашем ArrayAdapter.

package br.com.fontolan.pessoas.arrayadapter;

import Java.util.List;

import Android.content.Context;
import Android.text.Editable;
import Android.text.TextWatcher;
import Android.view.LayoutInflater;
import Android.view.View;
import Android.view.View.OnClickListener;
import Android.view.ViewGroup;
import Android.widget.ArrayAdapter;
import Android.widget.EditText;
import Android.widget.ImageView;
import br.com.fontolan.pessoas.R;
import br.com.fontolan.pessoas.model.Telefone;

public class TelefoneArrayAdapter extends ArrayAdapter<Telefone> {

private TelefoneArrayAdapter telefoneArrayAdapter = null;
private Context context;
private EditText tipoEditText = null;
private EditText telefoneEditText = null;
private ImageView deleteImageView = null;

public TelefoneArrayAdapter(Context context, List<Telefone> values) {
    super(context, R.layout.telefone_form, values);
    this.telefoneArrayAdapter = this;
    this.context = context;
}

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

    tipoEditText = (EditText) view.findViewById(R.id.telefone_form_tipo);
    telefoneEditText = (EditText) view.findViewById(R.id.telefone_form_telefone);
    deleteImageView = (ImageView) view.findViewById(R.id.telefone_form_delete_image);

    final int i = position;
    final Telefone telefone = this.getItem(position);
    tipoEditText.setText(telefone.getTipo());
    telefoneEditText.setText(telefone.getTelefone());

    TextWatcher tipoTextWatcher = new TextWatcher() {
        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
        }

        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {
        }

        @Override
        public void afterTextChanged(Editable s) {
            telefoneArrayAdapter.getItem(i).setTipo(s.toString());
            telefoneArrayAdapter.getItem(i).setIsDirty(true);
        }
    };

    TextWatcher telefoneTextWatcher = new TextWatcher() {
        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
        }

        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {
        }

        @Override
        public void afterTextChanged(Editable s) {
            telefoneArrayAdapter.getItem(i).setTelefone(s.toString());
            telefoneArrayAdapter.getItem(i).setIsDirty(true);
        }
    };

    tipoEditText.addTextChangedListener(tipoTextWatcher);
    telefoneEditText.addTextChangedListener(telefoneTextWatcher);

    deleteImageView.setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View v) {
            telefoneArrayAdapter.remove(telefone);
        }
    });

    return view;
}

}
5
mFontolan

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

 public class CustomAdapter extends BaseAdapter {

    TextView title;
  Button button1,button2;

    public long getItemId(int position) {
        return position;
    }

    public int getCount() {
        return mAlBasicItemsnav.size();  // size of your list array
    }

    public Object getItem(int position) {
        return position;
    }

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

        if (convertView == null) {
            convertView = getLayoutInflater().inflate(R.layout.listnavsub_layout, null, false); // use sublayout which you want to inflate in your each list item
        }

        title = (TextView) convertView.findViewById(R.id.textViewnav); // see you have to find id by using convertView.findViewById 
        title.setText(mAlBasicItemsnav.get(position));
      button1=(Button) convertView.findViewById(R.id.button1);
      button1.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            //your click action 

           // if you have different click action at different positions then
            if(position==0)
              {
                       //click action of 1st list item on button click
        }
           if(position==1)
              {
                       //click action of 2st list item on button click
        }
    });

 // similarly for button 2

   button2=(Button) convertView.findViewById(R.id.button2);
      button2.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            //your click action 

    });



        return convertView;
    }
}
3
Manohar Reddy