Android - ListView не получает OnItemClick для текстового просмотра с интерактивными ссылками

У меня есть ListView, который содержит TextView в каждой строке, помимо ряда других представлений. TextView отображает содержимое html, которое может содержать ссылки.

Следующий код отображается в списке.           m_textview.setMovementMethod(LinkMovementMethod.getInstance());           m_textview.setText(Html.fromHtml(myhtmlcontent));

Это приводит к тому, что listview больше не получает события щелчка. Я решил поместить список onclick-кода в представление, которое возвращается адаптером. Это не работает по желанию. Теперь я могу запустить другое действие, когда я нажимаю в любом месте строки, кроме textview. Я хочу, чтобы пользователи могли нажимать на ссылку, не связанную с текстом, и запускать другое действие.

Если я перемещаю onclick в текстовое представление вместо его родительского представления, он работает, но теперь щелчок по ссылке вызывает два события: одно для щелчка по ссылке и другое для текстового представления (что нежелательно).

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

Ответ 1

На самом деле это BUG. Чтобы решить эту проблему, вы можете добавить android:descendantFocusability="blocksDescendants" ваш макет xml. Например,

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:descendantFocusability="blocksDescendants"
android:orientation="vertical" >

 <TextView
    android:id="@+id/lblStatusMessage"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:autoLink="web"
    android:focusable="false"
    android:textSize="15sp" />
</LinearLayout>

Источник: this и this

Удачи:)

Ответ 2

Фокусные представления внутри элемента ListView будут отключать возможность выбора элементов ListView. Применение android:focusable="false" к TextView позволит OnItemClick снова работать. Вам также может потребоваться применить android:focusableInTouchMode="false", чтобы заставить трекбол игнорировать ссылки, потому что нажатие трекбола над фокусируемым элементом в ListView может как щелкнуть ссылку, так и элемент ListView.

Ответ 3

Вы можете прикрепить в списке список setOnItemClickListener.

Ответ 4

У меня была такая же проблема, и ни один из этих ответов не работал у меня. В конце концов, мне удалось устранить проблему, удалив атрибут android:inputType="textMultiLine".

Ответ 5

TextView, который реагирует только на события касания ссылок. Основано на                      fooobar.com/info/32897/...

        public class AutoLinkTextView extends TextView {

            public AutoLinkTextView(Context context, AttributeSet attrs, int defStyle) {
                super(context, attrs, defStyle);
                init();
            }

            public AutoLinkTextView(Context context, AttributeSet attrs) {
                super(context, attrs);
                init();
            }

            public AutoLinkTextView(Context context) {
                super(context);
                init();
            }

            private void init() {
                this.setAutoLinkMask(Linkify.ALL);
            }

            /**
             * @Linkify applies to a movementMethod to the textView @LinkMovementMethod.
             *          That movement method thought it implements a scrolling
             *          vertically method it overrides any other scrolling method the
             *          parent has.
             * 
             *          Although touchEvent can be dispached to the parent, the specific
             *          parent ScrollView needed the whole sequence ACTION_DOWN ,
             *          ACTION_MOVE, ACTION_UP to perform (sweep detection). So the
             *          solution to this problem is after applying @Linkify we need to
             *          remove the textView scrolling method and handle the @LinkMovementMethod
             *          link detection action in onTouchEvent of the textView.
             */
            @Override
            public boolean onTouchEvent(MotionEvent event) {
                final TextView widget = (TextView) this;
                final Object text = widget.getText();
                if (text instanceof Spannable) {
                    final Spannable buffer = (Spannable) text;
                    final int action = event.getAction();

                    if (action == MotionEvent.ACTION_UP
                            || action == MotionEvent.ACTION_DOWN) {
                        int x = (int) event.getX();
                        int y = (int) event.getY();

                        x -= widget.getTotalPaddingLeft();
                        y -= widget.getTotalPaddingTop();

                        x += widget.getScrollX();
                        y += widget.getScrollY();

                        final Layout layout = widget.getLayout();
                        final int line = layout.getLineForVertical(y);
                        final int off = layout.getOffsetForHorizontal(line, x);

                        final ClickableSpan[] link = buffer.getSpans(off, off,
                                ClickableSpan.class);

                        if (link.length != 0) {
                            if (action == MotionEvent.ACTION_UP) {
                                link[0].onClick(widget);
                            } else if (action == MotionEvent.ACTION_DOWN) {
                                Selection.setSelection(buffer,
                                        buffer.getSpanStart(link[0]),
                                        buffer.getSpanEnd(link[0]));
                            }
                            return true;
                        }
                    }

                }
                return false;
            }

            @Override
            public void setText(CharSequence text, BufferType type) {
                super.setText(text, type);
                this.setMovementMethod(null);
            }
        }