上一篇写过自定义展开收起的textview,不过最近发现有个问题。
那就是在列表页中,如果点击了全部,会触发view的点击事件,导致展开后接着进入了详情。
这显然不是想要的结果。
这里可以通过自定义LinkMovementMethod来解决。
第一步:
首先先屏蔽掉view的点击事件,全部用逻辑控制。
这样的话,就只会响应ClickableSpan事件了。
第二步:
接下来就需要通过LinkMovementMethod来控制了。
点进去看LinkMovementMethod的代码,可以看到是在onTouchEvent中处理的。
参数中的textview就是我们的自定义展开收起view。
@Override public boolean onTouchEvent(TextView widget, Spannable buffer, MotionEvent event) { if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_DOWN) { ... ClickableSpan[] links = buffer.getSpans(off, off, ClickableSpan.class); if (links.length != 0) { ClickableSpan link = links[0]; if (action == MotionEvent.ACTION_UP) { if (link instanceof TextLinkSpan) { ((TextLinkSpan) link).onClick( widget, TextLinkSpan.INVOCATION_METHOD_TOUCH); } else { link.onClick(widget); } } else if (action == MotionEvent.ACTION_DOWN) { Selection.setSelection(buffer, buffer.getSpanStart(link), buffer.getSpanEnd(link)); } return true; } else { Selection.removeSelection(buffer); } } return super.onTouchEvent(widget, buffer, event); }
现在就是要改变这里的逻辑,我这里直接把这个类复制出来,然后其它不变,直接处理这块逻辑。
if (links.length != 0) { if (action == MotionEvent.ACTION_UP) { links[0].onClick(widget); } else if (action == MotionEvent.ACTION_DOWN) { Selection.setSelection(buffer, buffer.getSpanStart(links[0]), buffer.getSpanEnd(links[0])); } //Log.e("onTouchEvent","点击了tip"); return true; } else { if (action == MotionEvent.ACTION_UP) {//防止滑动触发 //Log.e("onTouchEvent", "点击了其它"); Selection.removeSelection(buffer); widget.performClick(); return super.onTouchEvent(widget, buffer, event); } } } //Log.e("onTouchEvent","onTouchEvent"); return super.onTouchEvent(widget, buffer, event);
上面的 widget.performClick() 就是触发了view的点击事件,这里就可以随意控制点击事件了。
在设置span的时候添加进去就行了。
看运行效果图,非常丝滑,正常滑动,点击,展开。