Androidのアイテムのカスタムリストビュークリックの問題


110

したがって、カスタムのListViewオブジェクトがあります。リストアイテムには、2つのテキストビューが互いに積み重ねられており、さらに、実際に何かを行うまで非表示にしたい水平プログレスバーがあります。右端にあるのは、ユーザーがデータベースに更新をダウンロードする必要がある場合にのみ表示するチェックボックスです。可視性をVisibility.GONEに設定してチェックボックスを無効にすると、リスト項目をクリックできます。チェックボックスが表示されている場合、チェックボックス以外はリスト内でクリックできません。いくつか検索を行いましたが、現在の状況に関連するものは見つかりませんでした。この質問を見つけましたただし、データベースのリストを内部的に格納するためにArrayListsを使用しているため、オーバーライドされたArrayAdapterを使用しています。Tomのように、LinearLayoutビューを取得してonClickListenerを追加するだけでよいですか?よく分かりません。

リストビューの行レイアウトXMLは次のとおりです。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="?android:attr/listPreferredItemHeight"
    android:padding="6dip">
    <LinearLayout
        android:orientation="vertical"
        android:layout_width="0dip"
        android:layout_weight="1"
        android:layout_height="fill_parent">
        <TextView
            android:id="@+id/UpdateNameText"
            android:layout_width="wrap_content"
            android:layout_height="0dip"
            android:layout_weight="1"
            android:textSize="18sp"
            android:gravity="center_vertical"
            />
        <TextView
            android:layout_width="fill_parent"
            android:layout_height="0dip"
            android:layout_weight="1"
            android:id="@+id/UpdateStatusText"
            android:singleLine="true"
            android:ellipsize="marquee"
            />
        <ProgressBar android:id="@+id/UpdateProgress" 
                     android:layout_width="fill_parent" 
                     android:layout_height="wrap_content"
                     android:indeterminateOnly="false" 
                     android:progressDrawable="@android:drawable/progress_horizontal" 
                     android:indeterminateDrawable="@android:drawable/progress_indeterminate_horizontal" 
                     android:minHeight="10dip" 
                     android:maxHeight="10dip"                    
                     />
    </LinearLayout>
    <CheckBox android:text="" 
              android:id="@+id/UpdateCheckBox" 
              android:layout_width="wrap_content" 
              android:layout_height="wrap_content" 
              />
</LinearLayout>

そして、ここにListActivityを拡張するクラスがあります。明らかにそれはまだ開発中であるので、足りないものや置き去りにされているかもしれないものを許してください:

public class UpdateActivity extends ListActivity {

    AccountManager lookupDb;
    boolean allSelected;
    UpdateListAdapter list;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        lookupDb = new AccountManager(this);
        lookupDb.loadUpdates();

        setContentView(R.layout.update);
        allSelected = false;

        list = new UpdateListAdapter(this, R.layout.update_row, lookupDb.getUpdateItems());
        setListAdapter(list);

        Button btnEnterRegCode = (Button)findViewById(R.id.btnUpdateRegister);
        btnEnterRegCode.setVisibility(View.GONE);

        Button btnSelectAll = (Button)findViewById(R.id.btnSelectAll);
        btnSelectAll.setOnClickListener(new Button.OnClickListener() {
            @Override
            public void onClick(View v) {
                allSelected = !allSelected;

                for(int i=0; i < lookupDb.getUpdateItems().size(); i++) {
                    lookupDb.getUpdateItem(i).setSelected(!lookupDb.getUpdateItem(i).isSelected());
                }

                list.notifyDataSetChanged();
                // loop through each UpdateItem and set the selected attribute to the inverse 

            } // end onClick
        }); // end setOnClickListener

        Button btnUpdate = (Button)findViewById(R.id.btnUpdate);
        btnUpdate.setOnClickListener(new Button.OnClickListener() {
            @Override
            public void onClick(View v) {
            } // end onClick
        }); // end setOnClickListener

        lookupDb.close();
    } // end onCreate


    @Override
    protected void onDestroy() {
        super.onDestroy();

        for (UpdateItem item : lookupDb.getUpdateItems()) {
            item.getDatabase().close();        
        }
    }

    @Override
    protected void onListItemClick(ListView l, View v, int position, long id) {
        super.onListItemClick(l, v, position, id);

        UpdateItem item = lookupDb.getUpdateItem(position);

        if (item != null) {
            item.setSelected(!item.isSelected());
            list.notifyDataSetChanged();
        }
    }

    private class UpdateListAdapter extends ArrayAdapter<UpdateItem> {
        private List<UpdateItem> items;

        public UpdateListAdapter(Context context, int textViewResourceId, List<UpdateItem> items) {
            super(context, textViewResourceId, items);
            this.items = items;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            View row = null;

            if (convertView == null) {
                LayoutInflater li = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                row = li.inflate(R.layout.update_row, null);
            } else {
                row = convertView;
            }

            UpdateItem item = items.get(position);

            if (item != null) {
                TextView upper = (TextView)row.findViewById(R.id.UpdateNameText);
                TextView lower = (TextView)row.findViewById(R.id.UpdateStatusText);
                CheckBox cb = (CheckBox)row.findViewById(R.id.UpdateCheckBox);

                upper.setText(item.getName());
                lower.setText(item.getStatusText());

                if (item.getStatusCode() == UpdateItem.UP_TO_DATE) {
                    cb.setVisibility(View.GONE);
                } else {
                    cb.setVisibility(View.VISIBLE);
                    cb.setChecked(item.isSelected());
                }

                ProgressBar pb = (ProgressBar)row.findViewById(R.id.UpdateProgress);
                pb.setVisibility(View.GONE);
            }
            return row;
        }

    } // end inner class UpdateListAdapter
}

編集:私はまだこの問題を抱えています。テキストビューにonClickハンドラーをだまして追加していますが、チェックボックスをクリックしていないときにonListItemClick()関数がまったく呼び出されないのは非常に愚かです。

回答:


243

問題は、Androidでは、フォーカス可能な要素が含まれているリストアイテムを選択できないことです。リストアイテムのチェックボックスを次のような属性に変更しました。

android:focusable="false"

これで、チェックボックスを含む(ボタンでも機能する)リストアイテムが従来の意味で「選択可能」になります(ライトが点灯し、リストアイテムの任意の場所をクリックすると、「onListItemClick」ハンドラーが起動します)。

編集:更新として、コメンターは「ボタンの可視性を変更した後、プログラムで再びフォーカスを無効にする必要があったことに注意してください」と述べました。


24
ImageButtonがリストアイテムで使用されている場合、このソリューションは機能しないようです。:(
ジュリアンA.

1
わかりましたが、チェックボックスとリストビューのクリックを相互に排他的にするにはどうすればよいですか?ListViewアイテムを選択した場合、必ずしもチェックボックスをオンにする必要はありません。これはこのソリューションで可能ですか?
Mike6679 2011年

@マイクはい、このソリューションでは、対話するためにチェックボックスを明示的にクリックする必要があります。リストアイテムのレイアウトにフォーカス可能な要素がある場合、不可解なことにバイパスされるListViewの機能を再び有効にするだけです。
MattC、

2
すばらしい回答のおかげで、ImageButtonsでもうまく機能します。ただ注意してください。ボタンの表示を変更した後、プログラムでフォーカスを再度無効にする必要がありました。
Ljdawson、2012年

1
また、android:clickable = "false"を設定する必要がある場合もあります
Mina Samy

43

リストアイテム内にImageButtonがある場合descendantFocusabilityは、ルートリストアイテム要素で値を「blocksDescendants」に設定する必要があります。

android:descendantFocusability="blocksDescendants"

そして、ビュー内のfocusableInTouchModeフラグ。trueImageButton

android:focusableInTouchMode="true"

1
これは、imagebuttonsを持つlistitemに対して完全に機能します。しかし、これはImageButtonを除いて機能する可能性があります
raksja

12

同様の問題が発生していて、CheckBoxがListViewでかなり厄介であることがわかりました。何が起こるかは、それがListItem全体に意志を課すことであり、一種のonListItemClickをオーバーライドします。TextViewを使用する代わりに、そのためのクリックハンドラーを実装し、CheckBoxのtextプロパティを設定することもできます。

このViewオブジェクトも調べてみます。CheckBoxよりもうまく機能する可能性があります。

チェックテキストビュー


6
リストのフォーカス可能なアイテムの「focusable」設定をfalseに設定すると、onListItemClick関数を再び使用できることがわかりました。
MattC、2009

2
ただし、チェックボックスはリストアイテムのようにオレンジ色に変わります。誰かが解決策を知っていますか?
erdomester 2012

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