使用Spannable和LinkMovementMethod选择TextView

时间:2012-07-17 00:35:57

标签: android android-listview textview linkify spannable

到目前为止,我所做的是具有普通文本和可点击跨度的textviews的列表视图: Item untouched

单击我打开URL的范围,单击textView周围的项目查看会导致listView OnItemClickListener导航到项目详细信息,这很好: Item touched outside the textView

现在的问题是:

Item textView normal text touched Item textView span touched  触摸textView使正常文本有点突出显示(完全选择项目时具有相同的颜色),textView的OnTouchListener触摸事件触发而不是OnFocusChangeListener事件,并且项目的视图无法获得选择风格。尝试了列表视图的FOCUS_BLOCK_DESCENDANTS的所有变体,项目视图,启用或禁用了textView focusable,结果相同。

幸运的是,textView OnClickListener事件以这种方式触发,但这太丑了:由于所选文本颜色与项目颜色相同,因此文本是不可见的,而触摸未被释放,没有其他迹象表明除了那个丑陋的文字消失之外,用户还会看到项目细节。

我怀疑发生这种情况是因为textView的内容是Spannable,并且不是CliclableSpan-s的部分以这种奇怪的方式运行。

一旦触摸普通文本,我有机会选择项目吗?


listView项目布局:

<?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="fill_parent"
    android:descendantFocusability="blocksDescendants"
    android:orientation="horizontal" >

    <ImageView
        android:id="@+id/icon"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical"
        android:layout_marginBottom="5dp"
        android:layout_marginLeft="5dp"
        android:layout_marginRight="5dp"
        android:layout_marginTop="5dp"
        android:focusable="false" />

    <LinearLayout
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical" >

        <TextView
            android:id="@+id/title"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:layout_marginBottom="5dp"
            android:layout_marginTop="5dp"
            android:focusable="false"
            android:text=""
            android:textAppearance="?android:attr/textAppearanceLarge"
            android:textStyle="bold" />

        <TextView
            android:id="@+id/info"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:layout_marginBottom="4dp"
            android:layout_marginLeft="4dp"
            android:layout_marginRight="4dp"
            android:focusable="false"
            android:text=""
            android:textAppearance="?android:attr/textAppearanceSmall" />

        <TextView
            android:id="@+id/details"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:layout_marginBottom="4dp"
            android:layout_marginLeft="4dp"
            android:layout_marginRight="4dp"
            android:focusable="false"
            android:gravity="fill_horizontal|right"
            android:text=""
            android:textAppearance="?android:attr/textAppearanceMedium"/>

    </LinearLayout>

</LinearLayout>

使用文本视图setClickable(false)我能够以触摸文本视图区域时没有任何反应的方式禁用这种奇怪的选择样式,不好但可能对解决方案有用。

还尝试添加不可聚焦的&amp;每个项目都没有可点击的按钮,当它被触摸时,选择了完整的项目,当触摸被释放时,项目的点击事件被传递,这正是我对具有Spannable内容的textView所期望的。

4 个答案:

答案 0 :(得分:4)

您是否尝试将TextView的背景设置为Android的默认list_selector_background

    textView.setMovementMethod(LinkMovementMethod.getInstance());
    textView.setBackgroundResource(android.R.drawable.list_selector_background);

对我来说,它似乎给你想要的结果。

更新:看到该项目不仅仅是一个TextView

嗯,这不是一个完美的解决方案(因为它可能更好地修复TextView - ListView以某种方式突出显示交互),但它运行良好。

我发现不是在TextView上设置移动方法(触发问题),而是更简单地检查ListView的onItemClick()(确认点击后),看看我们是否应该在我们的ClickableSpans上启动onClick()

public class MyActivity extends Activity {

    private final Rect mLastTouch = new Rect();

    private boolean spanClicked(ListView list, View view, int textViewId) {
        final TextView widget = (TextView) view.findViewById(textViewId);
        list.offsetRectIntoDescendantCoords(widget, mLastTouch);
        int x = mLastTouch.right;
        int y = mLastTouch.bottom;

        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 Editable buffer = widget.getEditableText();
        final ClickableSpan[] link = buffer.getSpans(off, off, ClickableSpan.class);

        if (link.length == 0) return false;

        link[0].onClick(widget);
        return true;
    }

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

        // ...

        listView.setOnItemClickListener(new OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                if (spanClicked(listView, view, R.id.details)) return;

                // no span is clicked, normal onItemClick handling code here ..                    
            }
        });
        listView.setOnTouchListener(new OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                if (event.getAction() == MotionEvent.ACTION_UP) {
                    mLastTouch.right = (int) event.getX();
                    mLastTouch.bottom = (int) event.getY();
                }
                return false;
            }
        });

        // ...
    }

}

spanClicked()方法基本上是框架中LinkMovementMethod的onTouchEvent()方法的缩写版本。要捕获最后的MotionEvent坐标(检查click事件),我们只需在ListView上添加一个OnTouchListener。

答案 1 :(得分:0)

我认为解决此问题的最佳方法是使用选择器。你可以选择背景颜色,这样他们就会看到相同的

示例选择器代码:

<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="true"
       android:color="#000000" /> <!-- pressed -->
    <item android:state_focused="true"
       android:color="#000000" /> <!-- focused -->
    <item android:color="#FFFFFF" /> <!-- default -->
</selector>

将此类似的内容实施到您的链接。

让我知道结果

答案 2 :(得分:0)

引自setMovementMethod()

TextView的文档
  

设置要用于此的移动方法(箭头键处理程序)   TextView中。这可以为null,以禁止使用箭头键移动   光标或滚动视图。

     

如果您需要TextView,请注意   使用关键监听器或移动方法不可聚焦,或者如果你   想要一个没有键监听器或移动方法的TextView   可聚焦,你必须再次调用android.view.View.setFocusable(boolean)   在调用它之后,以你想要的方式恢复可聚焦性。

第二段解释了我们应该在设置LinkMovementMethod后明确调用setFocusable方法,以便以我们想要的方式工作。

在扩充列表项后,您需要在代码中执行此操作。 一旦能够获得焦点,就可以将选择器设置为具有不同的颜色文本。

以下是setMovementMethod的源代码。

答案 3 :(得分:0)

使用TextView的{​​{1}}