ボタンを追加してもListView setOnItemClickListenerが機能しない


89

テキストとボタンが各行にあるリストビューがあります。リストビューsetOnItemClickListener()が機能していません。アイテムのクリックイベントとボタンのクリックイベントを別々に処理することは可能ですか(アイテムのクリックはActivityAを呼び出し、ボタンのクリックはActivityBを呼び出す必要があります)。誰かが解決策を持っていますか

    private ArrayList<String> userIDArr = null;
    private ArrayList<String> userNameArr = null;
    private DatabaseHelper dbHelper = null;
    private ListView userListView=null; 


    public void onCreate(Bundle savedInstanceState) 
        {
          super.onCreate(savedInstanceState);         
          setContentView(R.layout.list_view);         
          dbHelper = new DatabaseHelper(this.getApplicationContext());        
          Map<String,ArrayList<String>> displayMap = dbHelper.getUserListToDisplay();
          userIDArr = displayMap.get("UserID");
          userNameArr = displayMap.get("FirstName1");           


          userListView = (ListView) findViewById(R.id.listView2);
          userListView.setAdapter(new UserListAdapter(this,userIDArr));


          userListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
              @Override
              public void onItemClick(AdapterView<?> arg0, View arg1, int position, long arg3) {

                  Toast.makeText(usersListActivity.this,
                            "Item in position " + position + " clicked", Toast.LENGTH_LONG).show();
              }
            });
     }


    public class UserListAdapter extends ArrayAdapter<String>
    {
        Activity context;
        public UserListAdapter(Activity context, ArrayList<String> names) {
            super(context, R.layout.list_item, names);
            this.context = context;
        }
        private class ViewHolder {
            public TextView UserNameAndID;
            public TextView Description;
            public Button  UploadBtn;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            ViewHolder holder;
            View rowView = convertView;
            if (rowView == null) {
                LayoutInflater inflater = context.getLayoutInflater();
                rowView = inflater.inflate(R.layout.list_item, null, true);
                holder = new ViewHolder();
                holder.UserNameAndID = (TextView) rowView.findViewById(R.id.User_detailsTxt);
                holder.Description = (TextView) rowView.findViewById(R.id.User_status);
                holder.UploadBtn = (Button) rowView.findViewById(R.id.uploadbutton);
                holder.UploadBtn.setOnClickListener(new View.OnClickListener() {  

                        public void onClick(View v) {  
                        Toast.makeText(usersListActivity.this," Button clicked",Toast.LENGTH_SHORT).show();
                        }   
                    }); 
                    rowView.setTag(holder);
            } else {
                holder = (ViewHolder) rowView.getTag();
            }
            String s = userNameArr.get(position)+","+userIDArr.get(position);
            holder.UserNameAndID.setText(s);
            holder.Description.setText("U r in middle");
            return rowView;
        }
    }
}`

回答:


260

次のようにボタン(またはリストアイテム内のクリックを処理する他のビュー)を設定してみてください。

android:focusable="false"
android:focusableInTouchMode="false"

@ベン・リー:これらをチェックボックスに追加しました!まだ動作していません。何が悪いのでしょうか?
TDaver

1
android:focusableInTouchMode="false"私を解決します。カスタムのLinearLayout行で2つのTextViewを使用しています。Yuk android ...
Youngjae 2013

12
ImageButtonは機能しません。リストアイテムビューでImageButtonを含み、android:descendantFocusability = "blocksDescendants"を設定します。
ChangUZ 2014年

8
android:clickable = "false"の追加も必要になる場合があります(レイアウトデザイナでオンとオフを切り替えて、明示的にXMLに追加します)
jrg

11
警告:ImageButtonを使用している場合は、setFocusable(false)を使用する必要があります。これは、ImageButtonのコンストラクターがxmlファイルからの拡張後にこの属性を有効にするためです。しかし良い答え
Mateu

118

時々、リストはまだクリックリスナーを通過させることができません。この場合、属性をもう1つ追加する必要があります。

android:descendantFocusability="blocksDescendants" 

そして、この属性は、ListView要素を提供したXMLの最上位レイアウトに追加する必要があります。


EditTextのように行にフォーカスが必要なものがある場合、これは望ましくありません。今はバカな気分です。
キタルダ2015年

私の場合、このアドバイスは役に立ちました。TextViewとRadioButtonがあります。1つ目はクリックされておらず、2つ目はTextViewを選択していませんでした。RadioButtonに「android:clickable = "false"」と書きました。
CoolMind 2016

これは、ボタンがImageButtonである場合のソリューションです
Mahonster

15

リストビューにアクティブビュー/フォーカス可能なビューがある場合は無効になりonItemClickListenerますandroid:focusable="false"。通常はフォーカス可能なビューに次を追加することで、フォーカスを解除 できます。


8

バッター事は両方を追加することでListener全体にrowView及びにButton内部Adapter。このようなもの。

public class MyAdapter extends BaseAdapter implements View.OnClickListener{

    @Override
    public View getView(int position, View convertView, ViewGroup parent)
    {
            View rowView = convertView;
            if(rowView == null)
            {
                    //intialize rowView, and set onclick listener only once.
                    rowView = someIntilizationMehhodOrInflatorMethod();
                    //add listener to the button also and also to the row view
                    rowView.setOnClickListener(this);
            }
            //all your inflation and setting values goes here and at the end,
            //set position as tag to get the correct position, rather buggy one.
            rowView.setTag(String.valueOf(position));


            return rowView;
    }
    public void onClick(View v)
    {
            //now get the tag of View v and convert it to integer.
            int pos = Integer.parseInt(v.getTag().toString());
            Toast.makeText(context,"Item in position " + pos + " clicked",Toast.LENGTH_LONG).show();
    }
}

3
これはどのようにリモートでさらに優れていますか?すべてのデータ要素に対して1つのリスナーに対してのみonClickListenerを作成しています。あなたは、どのビューをリサイクルすることすらありません。
JRomero 2013年

@ J.Romeroこれは単なる例ですが、コードを最適化する余地はたくさんあります。ただし、コードを更新しました。
Adil Soomro 2013年

5

セットを試してみてくださいsetClickable(false)ごとにButtonImageButtonなど、とViewこのように:

view.setClickable(false);
button.setClickable(false);
imagebutton.setClickable(false);

また、追加する必要があります

android:descendantFocusability="blocksDescendants"

メイン(第1レベル)のレイアウトに


2
私は、set blocksDescendantsで十分であることに気づきました。フォーカス可能などをキャンセルする必要はありません
。Android4.1

2

ボタンを追加する代わりに、正常に動作ImageViewするsetOnItemClcikリスナーを追加して提供します。

public View getView(final int position, View convertView, ViewGroup parent) {
     MyViewHolder mViewHolder;
     if(convertView == null) {
         convertView = inflater.inflate(R.layout.youtubesearchrow, null);
         mViewHolder = new MyViewHolder();
         mViewHolder.alarm = (ImageView)convertView.findViewById(R.id.alarm);
         convertView.setTag(mViewHolder);
     } else {
         mViewHolder = (MyViewHolder) convertView.getTag();
     }

     mViewHolder.tvTitle = detail(convertView, R.id.tvTitle,    
     // mViewHolder.tvDesc  = detail(convertView, R.id.tvDesc,  
     mViewHolder.ivIcon  = detail_image(convertView, R.id.ivIcon,  

     mViewHolder.alarm.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View v) {
        // TODO Auto-generated method stub
        }
    });

        return convertView;
}

1

同じエラーが発生します。リストにあなたがそれを置き換えるために、いずれかのボタンを持ってレイアウトする場合TextView、すなわちImageButtonではImageView。参考までに、それらは私のコードです:

private Activity activity;
private LayoutInflater inflater;

private List<ItemList> itemListsItems;

private Bookmark_SharedPref pref;

ImageLoader imageLoader = AppController.getInstance().getImageLoader();

public CustomListAdapter_Item(Activity activity,List<ItemList> itemListsItems){
    this.activity=activity;
    this.itemListsItems=itemListsItems;
    pref = new Bookmark_SharedPref(activity);
}


@Override
public int getCount() {
    return itemListsItems.size();
}

@Override
public Object getItem(int position) {
    return itemListsItems.get(position);
}

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

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

    View v = convertView;
    ViewHolder holder;

    if(inflater == null){
        inflater = (LayoutInflater)activity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }
    if(convertView == null){
        convertView = inflater.inflate(R.layout.item_layout,null);

        holder = new ViewHolder();
        convertView.setTag(holder);
    }
    else{
        holder = (ViewHolder)convertView.getTag();
    }


    if(imageLoader == null){
        imageLoader = AppController.getInstance().getImageLoader();
    }




     holder.img = (NetworkImageView)convertView.findViewById(R.id.item_image);
    holder.Title = (TextView)convertView.findViewById(R.id.item_title);
    holder.Cat = (TextView)convertView.findViewById(R.id.item_category);

    holder.id = (TextView)convertView.findViewById(R.id.itemid);

    holder.Desc = (TextView)convertView.findViewById(R.id.item_description);

    holder.book = (ImageView)convertView.findViewById(R.id.bookmark_star);



    // getting blog data or the row
    ItemList il = itemListsItems.get(position);

    //setting image
    holder.img.setImageUrl(il.getImageUrl(), imageLoader);

    // setting title
    holder.Title.setText(il.getTitle());

    //setting category
    holder.Cat.setText(il.getCategory());


    //setting blog id
    holder.id.setText(il.getId());

    //setting description
    holder.Desc.setText(il.getDescription());

    //bookmarking the post
    holder.book.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {

            itemListsItems.get(position).isFav = !itemListsItems.get(position).isFav;
            ((ImageView)v.findViewById(R.id.bookmark_star)).setImageResource(itemListsItems.get(position).isFav ? R.mipmap.active_star : R.mipmap.inactive_star);

            if(itemListsItems.get(position).isFav){
                String data = Integer.toString(itemListsItems.get(position).id);
                Toast.makeText(v.getContext(),"fav hai "+data,Toast.LENGTH_SHORT).show();
                pref.addPref(data);
            } else {
                String data = Integer.toString(itemListsItems.get(position).id);
                Toast.makeText(v.getContext(),"not fav"+ data,Toast.LENGTH_SHORT).show();
                pref.removePref(data);
            }
            //notifyDataSetChanged();
        }
    });

    ((ImageView)convertView.findViewById(R.id.bookmark_star)).setImageResource(itemListsItems.get(position).isFav ? R.mipmap.active_star : R.mipmap.inactive_star);

    /*
    if(itemListsItems.get(position).isFav){
        String data = Integer.toString(itemListsItems.get(position).id);
        Toast.makeText(convertView.getContext(),"fav hai "+data,Toast.LENGTH_SHORT).show();
        pref.addPref(data);
    } else {
        String data = Integer.toString(itemListsItems.get(position).id);
        //Toast.makeText(convertView.getContext(),"not fav"+ data,Toast.LENGTH_SHORT).show();
        pref.removePref(data);
    }
    */
    return convertView;
}

private static class ViewHolder {
   public TextView Title, Desc,Cat,id;
    ImageView book;
    NetworkImageView img;
}

またandroid:focussable="true"、ボタン/テキストパラメータを使用する場合と同様に、リストレイアウトを作成することも覚えておいてくださいandroid:focussable="false"


修正され、それがあるべきアンドロイド:フォーカス可能=「false」ではないfocussable
ケニーDabiri

1

android:focusable="false"ボタンに属性を追加し、親ビューにandroid:descendantFocusability="blocksDescendants"このような属性を追加します。

<RelativeLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:descendantFocusability="blocksDescendants">

    <ImageButton
        android:id="@+id/sell_button"
        android:layout_width="wrap_content"
        android:focusable="false"
        android:layout_height="wrap_content"
        android:src="@drawable/ic_sell" />
</RelativeLayout>

0

XMLがどのように変更されたかによって、Javaでsetonitemclickedが機能するようになったのは驚くべきことでした。

android:descendantFocusability="blocksDescendants"
android:focusable="false"
android:focusableInTouchMode="false"

0

このように、親のクリック可能属性をfalseにしandroid:clickable="false"ます。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:clickable="false"
    android:orientation="vertical">

再び問題に直面し、すべての子ビューに対して、このクリック可能なfalseを追加します。


0

正確な理由はわかりませんが、アダプターを設定するとリスナーがクリアされることがあります。その後、またはのsetOnClickListener後に呼び出す必要があります。setAdaptersetListAdapter


弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.