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);
}
}