【问题标题】:ListView OnItemLongClickListener() not triggeredListView OnItemLongClickListener() 未触发
【发布时间】:2013-11-29 18:46:40
【问题描述】:

我有一个扩展 ListActivity 的类,其中列表项响应 OnClick 事件。添加 OnItemLongClickListener 不起作用。不调用 onItemLongClick() 函数(不显示日志输出或 Toast),而是处理正常的 OnClick() 事件。

我想在长按时显示一个上下文操作栏。在新项目中使用我的代码的最小示例可以正常工作。所以我的问题是:什么可能阻止 onItemLongClick() 触发器被触发?

我的最低 API 是 11。我还将 listView 设置为 longClickable="true"

活动代码(选定功能):

public class EventListActivity extends ListActivity {

    private ArrayList<Event> arrEvents = null;
    private ArrayAdapter<Event> adpEvents = null;

    private ActionMode mActionMode = null;

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

        // only create list adapter and set it
        arrEvents = new ArrayList<Event>();
        adpEvents = new ArrayAdapter<Event>(this, android.R.layout.simple_list_item_activated_2, android.R.id.text1, arrEvents) {
            @Override
            public View getView(int position, View convertView, ViewGroup parent) {
                View view = super.getView(position, convertView, parent);
                TextView text1 = (TextView) view.findViewById(android.R.id.text1);
                TextView text2 = (TextView) view.findViewById(android.R.id.text2);
                text1.setText(arrEvents.get(position).getTitle());
                text2.setText(arrEvents.get(position).getDateTimeFormatted());
                return view;
            }
        };

        setListAdapter(adpEvents);

        // add CAB to ListView
        setupCAB();
    }


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

        // populate list and refresh adapter
        createEventList();
        adpEvents.notifyDataSetChanged();

        // if list empty show emtpy msg, otherwise hide it
        setContentView(R.layout.activity_event_list);
        TextView empty = (TextView) findViewById(R.id.text_empty);
        if(arrEvents.isEmpty()) {
            empty.setVisibility(View.VISIBLE);
        } else {
            empty.setVisibility(View.GONE);
        }
    }


    private void setupCAB() {
        // Important: to select single mode
        getListView().setChoiceMode(ListView.CHOICE_MODE_SINGLE);

        getListView().setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() {
            // Called when the user long-clicks an item on the list
            @Override
            public boolean onItemLongClick(AdapterView<?> parent, View row, int position, long rowid) {
                Log.w("EventListActivity", "Long click detected!");
                Toast.makeText(EventListActivity.this, "Long click detected!", Toast.LENGTH_SHORT).show();
                if (mActionMode != null) {
                    return false;
                }

                // Important: to mark the editing row as activated
                getListView().setItemChecked(position, true);

                // Start the CAB using the ActionMode.Callback defined above
                mActionMode = EventListActivity.this.startActionMode(mActionModeCallback);
                return true;
            }
        });
    }

    private ActionMode.Callback mActionModeCallback = new ActionMode.Callback() {
        // Called when the action mode is created; startActionMode() was called
        @Override
        public boolean onCreateActionMode(ActionMode mode, Menu menu) {
            // Inflate a menu resource providing context menu items
            MenuInflater inflater = mode.getMenuInflater();
            inflater.inflate(R.menu.event_context, menu);
            return true;
        }

        // Called when the user enters the action mode
        @Override
        public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
            // Disable the list to avoid selecting other elements while editing one
            EventListActivity.this.getListView().setEnabled(false);
            return true; // Return false if nothing is done
        }

        // Called when the user selects a contextual menu item
        @Override
        public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
            switch (item.getItemId()) {
            case R.id.mnu_share_event:
                //TODO share event
                mode.finish();
                return true;
            default:
                return false;
            }
        }

        // Called when the user exits the action mode
        @Override
        public void onDestroyActionMode(ActionMode mode) {
            // Re-enable the list after edition
            EventListActivity.this.getListView().setEnabled(true);
            mActionMode = null;
        }
    };

}

activity_event_list.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:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context=".EventListActivity" >

    <TextView
        android:id="@+id/text_empty"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentTop="true"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="45dp"
        android:text="@string/empty"
        android:textAppearance="?android:attr/textAppearanceLarge"
        android:visibility="gone" />

    <ListView
        android:id="@android:id/list"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:longClickable="true" >
    </ListView>

</RelativeLayout>

【问题讨论】:

    标签: java android listview contextual-action-bar


    【解决方案1】:

    如果您的列表视图中有响应 onClick() 事件的按钮,则需要在包含这些按钮的容器中设置以下内容:

    android:descendantFocusability="blocksDescendants"
    

    如果您拥有的是文本视图,那么问题会稍微棘手一些。看到这个:Focusable EditText inside ListView

    【讨论】:

    • 谢谢,但我的 ListView 中没有按钮或 EditText 视图。就像我在问题中发布的一样,只是普通的 TextViews (android.R.layout.simple_list_item_activated_2)。
    • 您是否尝试过添加 blocksDescendants?
    • 是的,但它并没有改变任何东西,因为我的问题来自不同的来源(请参阅我对 Rick Falck 回答的评论)。
    【解决方案2】:

    此答案不能解决用户 1 的问题,但症状与我的问题相似(即 OnItemClickListener 被调用但 OnItemLongClickListener 未被调用)。我在这里发布我的答案,以防其他人像我在尝试解决我的问题时一样偶然发现这个问题。

    我在 Fragment 中使用 ListView 并实现了监听器方法:

    public class MyFragment extends Fragment implements OnClickListener,
            OnLongClickListener, OnItemClickListener, OnItemLongClickListener {
    

    这是运行良好的onItemClick 方法:

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position,
            long rowId) {
    
        Log.i("Chimee", "short click working");
    
    }
    

    这是没有触发的onItemLongClick 方法:

    @Override
    public boolean onItemLongClick(AdapterView<?> parent, View view, int position,
            long rowId) {
    
        Log.i("Chimee", "Long click working");
        return false;
    }
    

    当然,简单的答案是我忘了setOnItemLongClickListener。我在我一直拥有的setOnItemClickListener 之后添加了它,然后它运行良好。

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
    
        View v = inflater.inflate(R.layout.my_fragment, container, false);
    
        lvSuggestions = (ListView) v.findViewById(R.id.lvSuggestions);
        lvSuggestions.setChoiceMode(ListView.CHOICE_MODE_SINGLE);
        lvSuggestions.setOnItemClickListener(this);
        lvSuggestions.setOnItemLongClickListener(this); // Forgot this
    
        ...
    }
    

    【讨论】:

      【解决方案3】:

      当使用 ListActivity 或 ListFragment 时,没有可以覆盖长按的方法,并且在 onCreateView() 中无法访问 ListView,因为它是由父类控制的。

      所以,为了克服这个问题,我这样做了,因为 getListView() 命令在创建视图之前不起作用:

      @Override
      public void onViewCreated(View view, Bundle savedInstanceState) {
          super.onViewCreated(view, savedInstanceState);
          mRecipeListView = this.getListView();
          mRecipeListView.setOnItemLongClickListener(new ListView.OnItemLongClickListener() {
      
              @Override
              public boolean onItemLongClick(AdapterView<?> arg0, View view, int position, long row_id) {
                  // Process the long-click
              }
          });
      }
      

      【讨论】:

      • 谢谢,答案就在你的最后一句话中“因为 getListView() 命令在视图创建后才会起作用” ---> 我打电话给我的@调用setContentView(R.layout.activity_event_list); 之前的987654322@ 方法,因此列表视图尚不可用。我应该把你的代码放在哪里我还是不明白......
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2016-06-18
      • 1970-01-01
      • 1970-01-01
      • 2013-08-22
      • 2016-07-30
      • 2013-05-12
      相关资源
      最近更新 更多