我有一个简单的ListActivity,它使用自定义ListAdapter生成列表中的视图。通常,ListAdapter只会用TextViews填充视图,但是现在我也想在其中放置一个按钮。
根据我的理解和经验,当单击列表项时,将可聚焦视图放在列表项中会阻止在ListActivity中触发onListItemClick()。该按钮在列表项中仍然正常运行,但是当按下按钮以外的内容时,我希望触发onListItemClick。
我该如何工作?
我有一个简单的ListActivity,它使用自定义ListAdapter生成列表中的视图。通常,ListAdapter只会用TextViews填充视图,但是现在我也想在其中放置一个按钮。
根据我的理解和经验,当单击列表项时,将可聚焦视图放在列表项中会阻止在ListActivity中触发onListItemClick()。该按钮在列表项中仍然正常运行,但是当按下按钮以外的内容时,我希望触发onListItemClick。
我该如何工作?
Answers:
android:descendantFocusability="blocksDescendants"很完美。
                    我希望我能在这里提供帮助。我假设您具有listView项的自定义布局,并且此布局包含按钮和一些其他视图-如TextView,ImageView或其他。现在,您希望在单击按钮时触发不同的事件,而在单击其他所有东西时触发不同的事件。 
您可以在不使用ListActivity的onListItemClick()的情况下实现这一点。
这是您必须做的:
您正在使用自定义布局,因此可能是从自定义适配器覆盖getView()方法。诀窍是为按钮设置不同的侦听器,并为整个视图(您的行)设置不同的侦听器。看一下示例:
private class MyAdapter extends ArrayAdapter<String> implements OnClickListener {
    public MyAdapter(Context context, int resource, int textViewResourceId,
            List<String> objects) {
        super(context, resource, textViewResourceId, objects);
    }
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        String text = getItem(position);
        if (null == convertView) {
            convertView = mInflater.inflate(R.layout.custom_row, null);
        }
        //take the Button and set listener. It will be invoked when you click the button.
        Button btn = (Button) convertView.findViewById(R.id.button);
        btn.setOnClickListener(this);
        //set the text... not important     
        TextView tv = (TextView) convertView.findViewById(R.id.text);
        tv.setText(text);
        //!!! and this is the most important part: you are settin listener for the whole row
        convertView.setOnClickListener(new OnItemClickListener(position));
        return convertView;
    }
    @Override
    public void onClick(View v) {
        Log.v(TAG, "Row button clicked");
    }
}您的OnItemClickListener类可以这样声明:
private class OnItemClickListener implements OnClickListener{       
    private int mPosition;
    OnItemClickListener(int position){
        mPosition = position;
    }
    @Override
    public void onClick(View arg0) {
        Log.v(TAG, "onItemClick at position" + mPosition);          
    }       
}当然,您可能会向OnItemClickListener构造函数添加更多参数。
还有一件重要的事情-上面显示的getView的实现非常丑陋,通常您应该使用ViewHolder模式来避免findViewById调用。.但是您可能已经知道了。
我的custom_row.xml文件是带有ID为“ button”的Button,ID为“ text”的TextView和ID为“ image”的ImageView的RelativeLayout-只是为了使内容清晰。
 
问候!
当自定义ListView包含可聚焦元素时,onListItemClick将不起作用(我认为这是预期的行为)。只需从自定义视图中移出焦点,即可达到目的:
例如:
public class ExtendedCheckBoxListView extends LinearLayout {
    private TextView mText;
    private CheckBox mCheckBox;
    public ExtendedCheckBoxListView(Context context, ExtendedCheckBox aCheckBoxifiedText) {
         super(context);
         …
         mText.setFocusable(false);
         mText.setFocusableInTouchMode(false);
         mCheckBox.setFocusable(false);
         mCheckBox.setFocusableInTouchMode(false);
        …       
    }
}我有同样的问题:OnListItemClick未触发![解决]
 
这是在扩展ListActivity的类上发生的,其中ListActivity 
的布局将内容TextBox和ListView嵌套到LinearLayout中,
而另一种行布局(将CheckBox和TextBox嵌套到LineraLayout中)。  
那是代码:
res / layout / configpage.xml(主要用于ListActivity)
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"  
    android:orientation="vertical"  
    android:layout_width="fill_parent"  
    android:layout_height="fill_parent" > 
    <TextView  
        android:id="@+id/selection"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:text="pippo" /> 
    <ListView  
        android:id="@android:id/list"  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:drawSelectorOnTop="false"  
        android:background="#aaFFaa" > 
    </ListView> 
<LinearLayout> 
res/layout/row.xml (layout for single row)  
<LinearLayout  
  xmlns:android="http://schemas.android.com/apk/res/android"  
  android:layout_width="fill_parent"  
  android:layout_height="wrap_content"> 
  <CheckBox  
      android:id="@+id/img"  
      android:layout_width="wrap_content"  
      android:layout_height="wrap_content"  
      **android:focusable="false"**  
      **android:focusableInTouchMode="false"** /> 
  <TextView  
      android:id="@+id/testo"  
      android:layout_width="wrap_content"  
      android:layout_height="wrap_content"  
      **android:focusable="false"**  
      **android:focusableInTouchMode="false"** /> 
</LinearLayout> src /.../.../ ConfigPage.java
public class ConfigPage extends ListActivity
{
    TextView selection;
    public void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.configpage);  
        // loaded from res/value/strings  
        String[] azioni = getResources().getStringArray(R.array.ACTIONS);  
        setListAdapter(new ArrayAdapter<String>(this, R.layout.row, 
                R.id.testo,   azioni));  
        selection = (TextView) findViewById(R.id.selection);  
    }       
    public void onListItemClick(ListView parent, View view, int position, long id)
    {
        selection.setText(" " + position);
    }
}当我在row.xml上添加时,这开始工作
android:focusable="false"android:focusableInTouchMode="false"我使用Eclipse 3.5.2 
Android SDK 10.0.1 
min SDK版本:3  
我希望这会有所帮助
...对不起我的英语:(  
我在ToggleButton上遇到了同样的问题。经过半天的撞墙碰壁,我终于解决了。就像使用“ android:focusable”使可聚焦视图变得不可聚焦一样简单。您还应该避免使用列表行的可聚焦性和可单击性(我刚刚组成的单词),只需将它们保留为默认值即可。
当然,既然列表行中的可聚焦视图无法聚焦,那么使用键盘的用户可能会遇到问题,将它们聚焦。这不太可能出现问题,但是如果您要编写100%无缺陷的应用程序,则可以使用onItemSelected事件使选定行的元素可聚焦,而先前选定行的元素不可聚焦。
 ListView lv = getListView();
lv.setTextFilterEnabled(true);
lv.setOnItemClickListener(new OnItemClickListener() {
public void onItemClick(AdapterView<?> parent, View view,
    int position, long id) {
  // When clicked, show a toast with the TextView text
  Toast.makeText(getApplicationContext(), ((TextView) view).getText(),
      Toast.LENGTH_SHORT).show();
}
});